Formatting
Modonomicon texts support a subset of Markdown, with some quirks due to how minecraft text rendering works. Additionally there is some non-standard functionality to support minecraft-specific use cases.
General
Bold
Format: **bold**
Italics
Format: *italics*
or _italics_
Underline
Format: ++underline++
Strikethrough
Format: ~~stricken through~~
Newline
Newlines generally work like in markdown, meaning a linebreak only renders as newline if it is preceded by three spaces.
If using Java Text Blocks to datagen texts java actually trims spaces at the end of lines so you need to use forced line breaks!
Alternatively you can place \s
at the end of the line after the three spaces to prevent java from trimming the spaces.
In markdown linebreak can also be forced with a backslash \
. In modonomicon, due to the underlying techonology you need to escape the backslash in both Java and JSON by simply doing two backslashes: \\
.
Empty Line
Because in many cases a line of text does not perfectly and at the end of the maximum length a line can have in the book, you need to use a little "trick" to force a fully empty line.
You need two escaped backslashes (= 4 backslashes) like so \\\\
, which will tell the markdown parser to do two line breaks. The first linebreak ends the previous text line, the second line then is the empty line.
As seen in the code sample below, you can also spread the two escaped backslashes over two lines for better readability:
this.add(helper.pageText(),
"""
Chalk is used to draw pentacle runes and define the pentacle shape. Different types of chalk are used for different purposes, as outlined on the next pages.
\\
\\
The different runes are purely decorative.
""");
or
this.add(helper.pageText(),
"""
Chalk is used to draw pentacle runes and define the pentacle shape. Different types of chalk are used for different purposes, as outlined on the next pages.\\
\\
The different runes are purely decorative.
""");
This is only necessary in modonomicon (but not in "normal" markdown editors) because of the above-mentioned behaviour of java to strip trailing spaces. In normal markdown you would simply add three spaces at the end of your text line, and then insert the forced line break with a backslash.
Lists
Lists work like in markdown using -
or *
for unordered lists and 1.
, 2.
, ... for ordered lists.
A newline/empty line should be placed after a list, otherwise the parser will add content of the line to the last list item.
Nested lists only have limited support and may not correctly wrap to the next line. If there are issues reduce nesting level or force a line break with \\
.
Headings
Markdown headings are not supported. Page titles are a separate JSON element.
Images
Like headings, images are not supported. There are specific page types to display images.
Translatable Content nested in Markdown Texts
To included the translated contents from a DescriptionId contained in lang.json
simply use <t>my.description.id</t>
.
The translated contents will not be sent through the markdown renderer, however you can use this to e.g. have an item (or any vanilla/modded object type that uses the minecraft translation system) name automatically be translated.
Links
Markdown links use the following syntax: [text](url)
.
HTTP Links
HTTP URLs will be handled like in the minecraft chat, which means clicking them will show an approval dialogue window and will then open the URL in an external browser.
Book Links
Book links are special links between different pages of the same book, or even to other books. There are three types:
Book Link
Open another book on it's default page. Syntax:
[display text](book://<book-id>)
The <book-id>
must include the namespace, e.g. occultism:dictionary_of_spirits
Category Link
Open a category (in the same book, or in another book). Syntax:
[display text](category://<book-id>/<category-id>)
[display text](category://<category-id>)
If <book-id>
is ommitted the current book is assumed.
Entry Link
Opens an entry (in the same book, or in another book), optionally at either a given page number or page anchor. Syntax:
[display text](entry://<book-id>/<entry-id>[#page-number][@page-anchor])
.[display text](entry://<entry-id>[#page-number][@page-anchor])
.
If <book-id>
is ommitted the current book is assumed.
This is the recommended way to link to entries.
The entry id is the full path of the entry within the /entries/
folder, that often includes the category id, if it is used as a subdirectory, e.g. for /entries/my-category/my-entry
the entry id is my-category/my-entry
.
Command Link
Command links are links that can run Commands when clicked. See Commands on how to create commands. Syntax:
[display text](command://<book-id>/<command-id>
.[display text](command://<command-id>
.
If no book id is provided the link will look for the command in the currently open book.
Command links can be used to e.g. give rewards to players by adding them to a page.
Item / Block Link
Item links are links that cannot be clicked, but that will show the (translated) item or block name if hovered, and if not provided with a link text, will also automatically have the translated name as rendered text.
Syntax:
[optional display text](item://minecraft:apple)
[](item://minecraft:chest)
Patchouli Links
Patchouli links are special links that can be used to link to a particular patchouli entry and open it on click.
Syntax:
[display text](patchouli://<mod_id>:<patchouli_book_id>//<entry_id>#<page_number>)
.[display text](patchouli://<mod_id>:<patchouli_book_id>//<entry_id>)
.
Example:
[Link to a Patchouli Entry](patchouli://occultism:dictionary_of_spirits//misc/books_of_calling)
Note the double //
separating the book id from the entry id. This is required, because both book and entry ids may contain one or multile /
characters if the files are in subdirectories.
The entry id is the full path of the entry within the /entries/
folder in patchouli_books
, the path you would also use for links within patchouli with the $(l:<entry_id>)
syntax.
Translations
On hover the link will attempt to display the name of the patchouli page that will be opened on click.
Patchouli does not have a standard DescriptionId format for entry names (in fact, entry names can be provided without using the translation system at all), so you need to manually include the translation for the link text in your lang.json
file.
The DescriptionId used for hover texts is patchouli.<patchouli_book_id>.<entry_id>.name
, e.g. patchouli.occultism.dictionary_of_spirits.misc.books_of_calling.name
. Make sure to provide a translation for this DescriptionId in your lang.json
file (or better, in your language datagen), otherwise the hover text will show the DescriptionId itself.
The <patchouli_book_id>
will include the mod id, the <entry_id>
will not.
Non-Standard Markdown
In order to provide a bit more flexibility, the markdown parser supports a few non-standard instructions.
Text Color
Color instructions (ab)use the link syntax as follows: to start a colored region use [#](<hexcode>)
, to reset to the default color use [#]()
.
Example:
[#](ff0000)Red text [#](00ff00)from here on green [#](0000ff)now blue [#]()and finally back to default color.
Dynamic Text Macro
Dynamic macros are runtime text replacements. The text replacement happens once serverside after all books and all minecraft/modded data has been loaded and is then cached for rendering on the client.
Dynamic can be used to e.g. show config or registry values in the book.
The ModonomiconProviderBase datagen child classes, such as EntryProvider, also offers registerMacro(String, String), macros(), macro(String). These methods are for static datagen-time macros!
Macro instructions (ab)use the link syntax as follows: to start a macro use [{}](<macro-key>)
.
[{}](my.dynamic.key) could be replaced to show a config value!
To register that macro call the following code during server setup:
LoaderRegistry.registerDynamicTextMacroLoader(<my-book-id>, () -> {
return Map.of(
"my.dynamic.key", MyConfig.SOME_VALUE.get()
);
});
You can use any code to build the map, it does not have to be an immutable map as shown here.
You can register multiple macro loaders per book id, but in the end all macros from all loaders will be merged into one map per book.