Snippets to customise RakuDoc blocks
RakuDoc-Render (RR) uses plugins to customise blocks. A plugin will add custom data and templates to an instance of an RR processor.
The templates are added to an instance of the Template-directory class held within the processor.
The custom data is also added to the processor's Template-directory object.
Helper callables that can be used inside a template can also be added to the processor's Template-directory object.
Within a template, all the registered templates, the custom data of the processor, and the helper callables can be accessed. See helper callables for more information.
When a template is added to a Template-directory and the template name already exists, the old value is pushed onto a stack, and can be accessed.
A generic RakuDoc::Processor will populate the Template-directory with text templates that then served as the generic defaults.
When a template is added to the directory, the source attribute on the Template-directory is set, and copied into each template. In this way, the origin of a template can be traced.
When the debug attribute on the Template-directory is True, the name and origin of each Template is reported whenever a Template is called.
A Template-directory object is an extended Hash structure.
Templates are specified as a list of Pairs of the form
# psuedocode <key> => -> <Hash>, <Template object> <Block> # or as an example head => -> %prm, $tmpl { ... }
where
%prm is an ordinary Hash of the parameters that are accessed inside the block
In order to access 'contents', the code inside the block would be %prm<contents>
The name of the Hash parameter is arbitrary, but conventionally is called %prm.
$tmpl is a Template object, see below, and conventionally is called $tmpl.
The block shown above is the pointy form, so the object returned is the value of the last statement
If the Raku program in the block is more easily written using a return statement, then a sub form should be used, eg.
key => sub ( %prm, $tmpl ) { ... }
A Template-directory object also has a %.data structure, which is intended for use by plugins that need to make extra data available for templates.
For example, suppose a custom block is written to include data about all the documents in a website, and the information is collected into a structure called %files, which is to be available to the template listfiles, then we could have
my %temp-dir is Target-directory = listfile => -> %prm, $tmpl {
my %file-data = $tmpl.globals.data<listfiles>
# code to create the output string using the data
}
# somewhere later
%temp-dir.data<listfiles> = %files;
# use the template
my $rv = %temp-dir<listfile>;
The %temp-dir object, which is an instance of the Template-directory class, provides access to the data through the .globals attribute of the Template object.
The Template object contains a reference to the Template-directory object, so all the templates registered with the RakuDoc processor, and all the data attached to it, can be accessed.
Inside the Raku block of a new template, the following methods can be used on the $tmpl object.
This calls the previously defined block of the template with the same name, with the same parameters provided to current block. The use case is to allow some pre- and (limited) post- processing of the parameters while keeping the previous template.
Pre-processing Suppose a new template is required that merely adds the word 'Chapter' to the contents of a =Chapter block. So the parameter needs to be preprocessed and the previous template called. Assuming %prm<contents> is a Str.
chapter => -> %prm, $tmpl {
%prm<contents> = 'Chapter ' ~ %prm<contents>;
$tmpl.prev( %prm ); # pass the new value of contents
}
Post-processing For example, suppose a template 'table' has been defined, but a new template is needed that substitutes the HTML class, then some post-processing of the old template is sufficient, eg.,
table => -> %prm, $tmpl {
($tmpl.prev).subst( / '<table' \s+ 'class="pod-table' /, '<table class="table is-centered')
}
This assumes that the return object from both templates chapter and table is a Str, which it could be. But generically, it is best not to assume this. A fuller example is given below.
Suppose a template generates objects that need to be added to the Table of Contents (ToC). The ToC structure is collected by the Renderer leaf by leaf, and the order of the leaf data is constructed because the order of the ToC is important.
So the code in the template block can call
$tmpl.globals.helper<add-to-toc>(
:caption<...>, :target<...>, :level(1)
)
Each output (eg HTML or MarkDown) may have its own escape method. The escape method can be called within a template as follows: $tmpl.globals.escape.( $string-to-be-escaped )
Note the . after escape. This is to ensure the string is passed to the block.
The block registered with key aaa can be called inside another template block, with or without parameters.
Without parameters, eg $tmpl<aaa> or $tmpl('aaa') the other template block is called with the same parameters, eg
page => -> %prm, $tmpl {
$tmpl<header> ~ $tmpl<body> ~ $tmpl<footer>
}
where header, body, and footer are all registered in the RR processor.
With parameters, eg. $tmpl('aaa', %( :attr(1), :new-attr<some string> ) ), the block registered with the key aaa is called with the new set of parameters specified in the Hash. It can be used to provide a subset of parameters, or to rename the parameters for a different template.
Four attributes can be set on a Template-directory object to aid with debugging templates, especially when templates have content that is derived from other templates:
debug
verbose
test
pretty
When set to True, eg
information about the name of the template being used, and the source of the template is sent to STDOUT via say. This is used by the debug(Templates) command in a Rakudoc::Processor object.
The information generated is for the registered template, not to be confused with the .test or .pretty attributes.
The Template-directory class has a .verbose attribute. When set to a string corresponding to the name of a template, eg.
my Template-directory %template-dir = %(
one => -> %prm, $tml { 'Hello world' },
two => -> %prm, $tml { 'Not again' },
);
# later ...
%template-dir.verbose = 'one';
the verbose result of that template (eg. 'one') will be sent to STDOUT via say.
The output of only one template at a time is supported at the moment.
When this attribute is set for the Template-directory object, eg.
my Template-directory %template-dir;
# assign some templates to %template-dir
%template-dir = %(
aaa => -> %prm, $tmpl {
'orig: ' ~ %prm<contents>;
},
ggg =. -> %prm, $tmpl {
%prm<contents> = 'Chapter ' ~ %prm<contents>
}
);
%template-dir.test = True;
In order to test the template with a set of unit tests, it is important for the results of the template to be uniform and predictable.
Also the renderer will call the template with options defined at run time, eg., because of a =config directive.
The output from a template when test is True does not depend on the block registered with the template. Instead, all the options (which may include the output from other templates) are returned in alphabetical order.
For example, the result of calling the templates defined above, with test=True, where the output from one template is contained in another, will be:
say %template-dir<aaa>(%(
:contents<something>,
ggg => %template-dir<ggg>(%(:contents<more stiff>,)),
));
# output
<aaa>
contents: 「something」
ggg: 「<ggg>
contents: 「more stiff」
</ggg>
」
</aaa>
It does not matter how the templates were defined, but it does matter that they were defined. Calling a template that is not registered in the Templated-directory object will cause an error.
When this attribute is set for the Template-directory object, eg.
my Template-directory %template-dir;
# assign some templates to %template-dir
%template-dir = %(
aaa => -> %prm, $tmpl {
'orig: ' ~ %prm<contents>;
},
ggg =. -> %prm, $tmpl {
%prm<contents> = 'Chapter ' ~ %prm<contents>
}
);
%template-dir.pretty = True;
the output is similar to the .test attribute, which it overrides, but white space is added to make the content structure clearer.
For example,
say %template-dir<aaa>(%(
:contents<something>,
array => <one two three four>,
hash => ( <eight nine ten> Z=> 1..* ).hash,
));
# output produced
<aaa>
array: 「List=(
"one",
"two",
"three",
"four"
)」
contents: 「something」
hash: 「Hash={
:eight(1),
:nine(2),
:ten(3)
}」
</aaa>