Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Document how to create a new admonition type #9

Closed
paulvickers opened this issue Aug 16, 2014 · 41 comments
Closed

Document how to create a new admonition type #9

paulvickers opened this issue Aug 16, 2014 · 41 comments

Comments

@paulvickers
Copy link

I think it's an extension I need, but I'm not sure how to go about making one. Here's what I need:

I am converting a text book to various e-formats and the book has several in-text devices. Up till now I have been using the built-in admonitions and just created a new stylesheet to use different icons to match what I need. However, it seems to me that rather than subverting the existing admonitions it would be better to create new custom blocks for this project. For example, I need a block that looks and behaves exactly like an admonition except that it should be called thinkspot rather than tip, caution, warning, or note.

Can someone give me a step-by-step on what to do?

Thanks!

@mojavelinux
Copy link
Member

When you only need a change that can be controlled by the stylesheet, I think the best approach is to simply append a role in the AsciiDoc content and then append the necessary CSS to match. In this case, we should view the admonition types as broad categories or levels that can be "specialized" using a role. thinkspot (or just think) might fit either with tip or note. You'd type it as:

[NOTE.think]
====
A thinkspot admonition.
====

In HTML, that will output:

<div class="admonitionblock note think">
<table>
<tr>
<td class="icon">
<i class="fa icon-note" title="Note"></i>
</td>
<td class="content">
<div class="paragraph">
<p>A thinkspot admonition.</p>
</div>
</td>
</tr>
</table>
</div>

If you want the value of the title attribute to be “Thinkspot” instead of “Note”, you can set the caption too.

[NOTE.think,caption=Thinkspot]

You can then customize the icon using CSS, which can be appended using docinfo or a custom stylesheet.

.admonitionblock.think td.icon .icon-note:before {
  content:"\f0eb";
  color:#000;
}

@mojavelinux mojavelinux changed the title Tutorial Document how to create a new admonition type Jan 8, 2015
@nathany
Copy link

nathany commented Nov 17, 2015

How would I add additional admonitions, like [QUESTION] and customize the Font Awesome icon? I'd also like it to work in PDFs generated with asciidoctor-pdf.

@mojavelinux
Copy link
Member

If you add a new admonition type, then you'd technically be introducing a new type of block (named QUESTION, for instance), which would need to be handled by a custom block extension. You could then map the result back to an admonition node so that the built-in handler takes care of generating the output for it.

The icon is controlled by the "name" attribute (e.g., "icon-#{node.attr 'name'}"). It doesn't validate the name value, so it can be any value you want. You'd still need to add CSS to map that class to a Font Awesome icon.

The fact that admonition types are controlled by the block name is kind of strange. In UniDoc, I'd actually like to see the admonition type be a special attribute, something like:

[!QUESTION]
====
...
====

That way, we aren't introducing new block types for each type of admonition. But that's a separate discussion.

@mojavelinux
Copy link
Member

I thought of a way to mark this up for UniDoc (possibly supported in Asciidoctor before then). We overload the block title and allow a special prefix to indicate this is an admonition.

.QUESTION: optional title
====
...
====

That way, we aren't introducing a new block type for each admonition type. And it's much closer to the paragraph version (e.g., QUESTION: content...). You could technically implement this today using a Treeprocessor.

@nathany
Copy link

nathany commented Dec 1, 2015

Thanks. Curious, what is this UniDoc thing you speak of? :-)

@mojavelinux
Copy link
Member

UniDoc is an (eventual) goal of making a specification for AsciiDoc to make it easier to create tools and other implementations and to refine the syntax where necessary. For more info, see https://github.com/asciidoctor/asciidoctor/wiki/AsciiDoc-Specification-(aka-UniDoc)-Planning.

@nathany
Copy link

nathany commented Dec 6, 2015

Requiring a custom block type does seem like a bit much for adding a new admonition type. That I agree with.

I would basically just like to get rid of this warning:

asciidoctor: WARNING: ch01.adoc: line 93: invalid style for example block: QUESTION

If I do something like the README (which probably isn't the processing I want it to do) and require my file on the command line, that warning is still there.

-r ./lib/question-block.rb
RUBY_ENGINE == 'opal' ? (require 'question-block/extension') : (require_relative 'question-block/extension')

Extensions.register do
  block QuestionBlock
end
class QuestionBlock < Extensions::BlockProcessor
  use_dsl
  named :question
  on_context :open

  def process parent, reader, attrs
    create_paragraph parent, reader.lines, attrs
  end
end

@mojavelinux
Copy link
Member

mojavelinux commented Dec 6, 2015 via email

@nathany
Copy link

nathany commented Dec 6, 2015

Sure. Am I missing something here?

[QUESTION]
====
1. How can you save your work when using The Go Playground?
====

Also, my full command looks like this, in case it's just not loading for some reason:

asciidoctor -r asciidoctor-pdf -r ./lib/sectnumoffset-treeprocessor.rb -r ./lib/question-block.rb -b pdf -a source-highlighter=coderay -a pdf-fontsdir=resources/fonts -a pdf-style=resources/themes/manning-theme.yml manuscript/*.adoc -D pdf/

@mojavelinux
Copy link
Member

mojavelinux commented Dec 6, 2015 via email

@nathany
Copy link

nathany commented Dec 6, 2015

Made the change, I still get the warning when generating a pdf.

Maybe I shouldn't worry about it. As I understand it, my publisher is going to handle customizing the icons when they convert to DocBook. So I don't need it to look exactly right, which just means I get a warning when generating.

@mojavelinux
Copy link
Member

On a completely unrelated note, why not use the Q&A syntax for this?

[qanda]
How can you save your work when using The Go Playground?:: {empty}

@mojavelinux
Copy link
Member

...btw, I'm testing the extension...

@mojavelinux
Copy link
Member

Aha! The style is case sensitive. The following will work.

require 'asciidoctor/extensions'

class QuestionBlock < Asciidoctor::Extensions::BlockProcessor
  use_dsl
  named :QUESTION
  on_context :example 

  def process parent, reader, attrs
    create_paragraph parent, reader.lines, attrs
  end
end

Asciidoctor::Extensions.register do
  block QuestionBlock 
end

Btw, you probably want to create an admonition block instead of a paragraph in the process method. I'll try to fill in the details, then we can check this into the extension lab.

@mojavelinux
Copy link
Member

Voila!

require 'asciidoctor/extensions'

class QuestionBlock < Asciidoctor::Extensions::BlockProcessor
  use_dsl
  named :QUESTION
  on_context :example

  def process parent, reader, attrs
    attrs['name'] = 'question'
    attrs['caption'] = 'Question'
    admon = create_block parent, :admonition, nil, attrs, content_model: :compound
    parse_content admon, reader
    admon
  end
end

class QuestionBlockCss < Asciidoctor::Extensions::DocinfoProcessor
  use_dsl

  def process doc
    '<style>
.admonitionblock td.icon .icon-question:before{content:"\f128";color:#000}
</style>'
  end
end

Asciidoctor::Extensions.register do
  block QuestionBlock
  docinfo_processor QuestionBlockCss
end

@mojavelinux
Copy link
Member

I'll put this into the extensions lab so we can hack on it there.

@mojavelinux
Copy link
Member

It's there now.

@nathany
Copy link

nathany commented Dec 6, 2015

Thanks. It works. I haven't done any custom icons in the PDF (for TIP, etc.) but this works as well as those ones.

As for [qanda], I'm using it for the answers at the end of the chapter, but never thought to use them for questions by themselves, which are scattered across several pages.

@nathany
Copy link

nathany commented Dec 6, 2015

The combination of -r ./lib/custom-admonition-block.rb and -a icons=font result in a stack trace with asciidoctor-pdf

/usr/local/lib/ruby/gems/2.2.0/gems/asciidoctor-pdf-1.5.0.alpha.10/lib/asciidoctor-pdf/converter.rb:439:in
 `block (4 levels) in convert_admonition': undefined method `[]' for nil:NilClass (NoMethodError)

@mojavelinux
Copy link
Member

When using Asciidoctor PDF, you need to define an admonition type for question in the custom theme. See https://github.com/asciidoctor/asciidoctor-pdf/blob/master/docs/theming-guide.adoc#admonition. Otherwise, the icon resolves to nil.

@mojavelinux
Copy link
Member

The HTML5 backend just relies on the stylesheet to sort out the icon, but Asciidoctor PDF actually has to look it up.

@nathany
Copy link

nathany commented Dec 6, 2015

Gotcha. I'll look into it, thanks

@mojavelinux
Copy link
Member

That's also how you can customize all the icons in Asciidoctor PDF.

Technically, you don't even need a custom admonition type if you just hijack the icon :)

@nathany
Copy link

nathany commented Dec 6, 2015

Hah. Guess I could've just hijacked the warning icon or something. Doh.

I must be doing something wrong, as I don't quite understand the layout of the YAML. I either get that nil error or a /theme_loader.rb:90:inprocess_entry': undefined method map' for 0:Fixnum error

admonition:
  border_color: $base_border_color
  border_width: $base_border_width
  admonition_icon_note_stroke_color: 000000
  admonition_icon_question_name: fa-question
  admonition_icon_question_stroke_color: 000000
  admonition_icon_question_size: 24
  admonition_icon_experiment_name: fa-flask
  admonition_icon_experiment_stroke_color: 000000
  admonition_icon_experiment_size: 24

@nathany
Copy link

nathany commented Dec 6, 2015

no change with

admonition:
  admonition_icon_tip_name: fa-question

error asciidoctor-pdf/theme_loader.rb:90:in 'process_entry': undefined methodmap' for "fa-question":String (NoMethodError)` for either:

admonition_icon_tip_name: fa-question

or

admonition:
  icon_tip_name: fa-question

@mojavelinux
Copy link
Member

First, it has to be defined with the other admonition settings or else the theme is incomplete.

admonition:
  border_color: $base_border_color
  border_width: $base_border_width
  padding: [0, $horizontal_rhythm, 0, $horizontal_rhythm]
  icon:
    question:
      name: fa-question
      stroke_color: 000000
      size: 14

But that brings us to a separate problem. The converter attempts to merge the theme data with the built-in defaults, but the built-in default is nil. So a change is needed to Asciidoctor PDF at line https://github.com/asciidoctor/asciidoctor-pdf/blob/master/lib/asciidoctor-pdf/converter.rb#L1786.

@mojavelinux
Copy link
Member

With that change to Asciidoctor PDF, it works. I'll commit it. You'll need to use HEAD.

@nathany
Copy link

nathany commented Dec 6, 2015

Hm. I don't suppose putting this in my Gemfile and running bundle update is enough to get it working?

gem 'asciidoctor-pdf', github: 'asciidoctor/asciidoctor-pdf'

obviously not, it's still using alpha.10. hmmmm

/usr/local/lib/ruby/gems/2.2.0/gems/asciidoctor-pdf-1.5.0.alpha.10/lib/asciidoctor-pdf/converter.rb:1786:in `admonition_icon_data': undefined method `merge' for nil:NilClass (NoMethodError)
converter.rb:1786:in `admonition_icon_data': undefined method `merge' for nil:NilClass (NoMethodError)

UPDATE: -r asciidoctor-pdf changed to a specific path -r ~/src/github.com/asciidoctor/asciidoctor-pdf/lib/asciidoctor-pdf works

@nathany
Copy link

nathany commented Dec 7, 2015

Sweet. The PDF looks pretty great.

sample

There are some spots where it messes up, like when I have a quote or TIP inside the admonition. Those work fine when hijacking a different admonition, so it must be something in the CustomAdmonitionBlock code. Notice the byline is repeated below.

malacandra

[EXPERIMENT]
.Experiment: malacandra.go
====
[quote, C.S. Lewis, Out of the Silent Planet]
"Malacandra is much nearer than that: we shall make it in about twenty-eight days."

Malacandra is another name for Mars in a SciFi trilogy by C.S. Lewis. Write a program to determine how fast their ship would need to travel in order to reach Malacandra in 28 days. Assume a distance of 56,000,000 km.
====

Whether or not I should be doing that sorta thing is a whole other question.

@mojavelinux
Copy link
Member

The implementation is a bit naive at the moment. Now that we have the
basics down, we can fill in the gaps so it handles all the cases.
Le 6 déc. 2015 5:20 PM, "Nathan Youngman" notifications@github.com a
écrit :

Sweet. The PDF looks pretty great.

[image: sample]
https://cloud.githubusercontent.com/assets/4566/11616600/581c4644-9c3c-11e5-8949-80e8ac067395.png

There are some spots where it messes up, like when I have a quote or TIP
inside the admonition. Those work fine when hijacking a different
admonition, so it must be something in the CustomAdmonitionBlock code.
Notice the byline is repeated below.

[image: malacandra]
https://cloud.githubusercontent.com/assets/4566/11616640/84afed0e-9c3d-11e5-86a8-fe7c360ef54e.png

[EXPERIMENT].Experiment: malacandra.go====[quote, C.S. Lewis, Out of the Silent Planet]"Malacandra is much nearer than that: we shall make it in about twenty-eight days."Malacandra is another name for Mars in a SciFi trilogy by C.S. Lewis. Write a program to determine how fast their ship would need to travel in order to reach Malacandra in 28 days. Assume a distance of 56,000,000 km.====

Whether or not I should be doing that sorta thing is a whole other
question.


Reply to this email directly or view it on GitHub
#9 (comment)
.

@mojavelinux
Copy link
Member

mojavelinux commented Dec 7, 2015 via email

@mojavelinux
Copy link
Member

Fixed. Turns out I couldn't remember how to create a block extension right myself :)

@mojavelinux
Copy link
Member

@nathany When you use a GitHub reference in your Gemfile, you have to run everything using bundle exec. Otherwise, Ruby doesn't see the clone.

@nathany
Copy link

nathany commented Dec 7, 2015

Hah. I forgot about bundle exec. It's been a while. Everything looks to be working. Thanks for helping me out on your Sunday.

@mojavelinux
Copy link
Member

mojavelinux commented Dec 7, 2015 via email

@paulvickers
Copy link
Author

Thank you both for this. I have successfully added the four new blocks I need into a separate .rb file. Works a treat. The reason I wanted new blocks is to save having to remember which admonition I had mapped to thinkspot, etc. Now all I need to do is

[THINKSPOT]
====
1. How can you save your work when using The Go Playground?
====

@paulvickers
Copy link
Author

paulvickers commented Jul 28, 2016

One more question. One of my new blocks uses a custom icon which I had got to work before by hijacking the icon in the master css file thus:

.admonitionblock td.icon .icon-warning:before {content: url(''); } /* Brian QandA */

Unfortunately, when I try this in the new block definition file:

class BrianFAQBlockCss < Asciidoctor::Extensions::DocinfoProcessor
  use_dsl

  def process doc
    '<style>
.admonitionblock td.icon .icon-brianfaq:before {content: url(''); }
</style>'
  end

It throws errors when I try to run it through asciidoctor. Can I not use base64 images this way?

Also, how do I get the new blocks to work as paragraphs too? E.g.

THINKSPOT: This is a think spot

Rather than always having to use block mode:

[THINKSPOT]
====
This is a think spot
====

@paulvickers
Copy link
Author

Any suggestions on the base64 question?

@kwoot
Copy link

kwoot commented Jun 20, 2021

As an add-on question to the lovely discussion above. How hard is it to make a custom admonition block optional? So, if attribute "Teacher" is set, then all "Teacher" admonition blocks are included, otherwise they're not. In usecase lingo: An an author of school books I would like to have one source files that an be processed to a students- and to a teachers-edition (with extra included blocks to guide/hint the teaching process). I know I can create conditional text and that would work, but I' m just spoiled by all the fancy stuff Asciidoctor can do for me. :-)

@mojavelinux
Copy link
Member

@kwoot I encourage you to direct your question to https://asciidoctor.zulipchat.com, where we can discuss usage scenarios as a community (without risk of going off topic).

@mojavelinux
Copy link
Member

There is a custom admonition block extension in this lab. That's as far as we are going to take it here.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants