The general output apparatus for writing a block of documentation.
- §6. Block buffer
- §8. Top-level renderer
- §10. Rendering text
- §10.2.1. Code mode
- §10.2.4. Tabular mode
- §10.2.7. Regular mode
- §10.2.2.3. Javascript paste icons
- §13. Rendering cross-references to other sections
- §15. Handling the formatted file
§1. The output mechanism produces documentation in chunks called "blocks", and it has a buffer which stores one block at a time. Each block contains a list of paragraphs, numbered from 0.
define MAX_PARAGRAPHS_PER_BLOCK 1000
typedef struct paragraph { int par_indentation; int par_suppression; struct text_stream *par_texts; struct text_stream *par_prefix; struct text_stream *par_styles; int par_shortened; } paragraph;
- The structure paragraph is accessed in 2/rr and here.
int no_paras_in_block_buffer = 0; paragraph paragraphs[MAX_PARAGRAPHS_PER_BLOCK];
§3. The blocks are written to a file of "formatted text":
filename *current_FTD_filename = NULL; with NULL meaning that no file is open text_stream current_FTD_stream; otherwise, this is where the text goes
§4. It's convenient to record all the formatted text filenames written to, and those are the keys of the following hash:
typedef struct formatted_file { struct filename *name; CLASS_DEFINITION } formatted_file;
- The structure formatted_file is accessed in 2/trs, 4/prp, 4/jsn, 5/mrk, 5/mv, 5/im, 2/utc and here.
int index_to_examples = FALSE; used to index examples to particular sections int unique_code_pos_counter = 0; used to uniquely ID code samples example *code_example = NULL;
§6. Block buffer. When the scanner starts a new block, it calls this:
void Renderer::clear_block_buffer(void) { no_paras_in_block_buffer = 0; index_to_examples = FALSE; }
§7. It then calls this to add paragraphs to the block:
void Renderer::add_para_to_block_buffer(text_stream *text, int indentation, int suppression, text_stream *prefix, text_stream *style, int shortened) { if (no_paras_in_block_buffer >= MAX_PARAGRAPHS_PER_BLOCK) Errors::fatal("too many paragraphs in block"); paragraph *P = ¶graphs[no_paras_in_block_buffer++]; P->par_indentation = indentation; P->par_texts = Str::duplicate(text); P->par_suppression = suppression; P->par_prefix = Str::duplicate(prefix); P->par_styles = Str::duplicate(style); P->par_shortened = shortened; }
§8. Top-level renderer. If the block consists of text from a section, then the scanner calls the following when the buffer is filled. If, on the other hand, the block comes from an example, it calls render_text_of_block instead, a simpler routine which doesn't surround the text with navigational gadgets and headings.
text_stream *Renderer::render_block(OUTPUT_STREAM, volume *V, section *S) { OUT = Renderer::formatted_file_must_be(OUT, V, S); Nav::render_navigation_top(OUT, V, S); Renderer::render_text_of_block(OUT, V, S); Nav::render_navigation_middle(OUT, V, S); Render the examples below the text of the block8.1; Nav::render_navigation_bottom(OUT, V, S); return OUT; }
§8.1. Render the examples below the text of the block8.1 =
TEMPORARY_TEXT(form_of_title_to_test) Adapt the block title to the form of the title to test8.1.1; int no_examples_rendered_here = 0; for (int n = 0; n < no_examples; n++) { example *E = V->examples_sequence[n]; if (E->example_displayed_at_section[V->allocation_id] == S) { no_examples_rendered_here++; if (no_examples_rendered_here == 1) Nav::render_navigation_example_top(OUT, V, S); Render the example here8.1.2; if (indoc_settings->examples_mode == EXMODE_open_internal) HTMLUtilities::ruled_line(OUT); } } if (no_examples_rendered_here > 0) Nav::render_navigation_example_bottom(OUT, V, S);
- This code is used in §8.
§8.1.1. Examples need to connect with particular sections of documentation, but they do so by title, not by block number, to protect them from renumbering as sections are added or removed. So if the current block is called
2.3. Sailing Ships
then we need to look for the text Sailing Ships to see if an example belongs here. (But, owing to a historical accident, in the Recipe Book section names are capitalised for this purpose.)
Adapt the block title to the form of the title to test8.1.1 =
Str::copy(form_of_title_to_test, S->unlabelled_title); match_results mr = Regexp::create_mr(); if (Regexp::match(&mr, form_of_title_to_test, U"(%d+.)* *(%c*)")) { Str::copy(form_of_title_to_test, mr.exp[1]); Regexp::dispose_of(&mr); } if (V->allocation_id == 1) LOOP_THROUGH_TEXT(pos, form_of_title_to_test) Str::put(pos, Characters::toupper(Str::get(pos)));
- This code is used in §8.1.
§8.1.2. Render the example here8.1.2 =
TEMPORARY_TEXT(index_term) WRITE_TO(index_term, "%S=___=!example", E->ex_public_name); Indexes::mark_index_term(index_term, V, NULL, NULL, E, NULL, NULL); DISCARD_TEXT(index_term) if (indoc_settings->format == HTML_FORMAT) { TEMPORARY_TEXT(comment) WRITE_TO(comment, "START EXAMPLE \"%d: %S\" \"e%d\"", E->example_position[0], E->ex_public_name, E->allocation_id); HTML::comment(OUT, comment); DISCARD_TEXT(comment) } Examples::render_example_cue(OUT, E, V, 0); code_example = E; OUT = Renderer::render_example_body(OUT, E, V, TRUE); code_example = NULL; if (indoc_settings->format == HTML_FORMAT) HTML::comment(OUT, I"END EXAMPLE");
- This code is used in §8.1.
text_stream *Renderer::render_example_body(OUTPUT_STREAM, example *E, volume *V, int empty) { int hide = FALSE; if (indoc_settings->examples_mode == EXMODE_openable_internal) hide = TRUE; if (indoc_settings->format == HTML_FORMAT) { TEMPORARY_TEXT(id) WRITE_TO(id, "example%d", E->allocation_id); HTML::begin_div_with_class_and_id_S(OUT, I"egpanel", id, hide, __FILE__, __LINE__); } OUT = Rawtext::process_example_rawtext_file(OUT, V, E); if (indoc_settings->format == HTML_FORMAT) { HTML::end_div(OUT); if (empty) { HTML_OPEN("p"); HTML_CLOSE("p"); } } return OUT; }
§10. Rendering text. The actual contents of the buffer are rendered here, then:
text_stream *Renderer::render_text_of_block(OUTPUT_STREAM, volume *V, section *S) { if (indoc_settings->format == PLAIN_FORMAT) Render the block buffer as plain text10.1 else if (indoc_settings->format == HTML_FORMAT) Render the block buffer as HTML10.2; return OUT; }
§10.1. Plain text is very plain indeed:
Render the block buffer as plain text10.1 =
for (int i=0; i<no_paras_in_block_buffer; i++) { Indent using tabs int ic = paragraphs[i].par_indentation; while (ic > 0) { ic--; WRITE("\t"); } Remove any paste markers entirely text_stream *raw = paragraphs[i].par_texts; match_results mr = Regexp::create_mr(); if ((indoc_settings->treat_code_as_verbatim == FALSE) && (Regexp::match(&mr, raw, U"{%*+} *(%c*)"))) WRITE("%S\n", mr.exp[0]); else WRITE("%S\n", raw); Regexp::dispose_of(&mr); }
- This code is used in §10.
Render the block buffer as HTML10.2 =
int code_mode = FALSE; int tabular_mode = FALSE; int last_xref_type = 0; 0 means "none"; 1 means "to section"; 2 means "to example" for (int i=0; i<no_paras_in_block_buffer; i++) { paragraph *P = &(paragraphs[i]); if (P->par_indentation == 0) { if (tabular_mode) Exit tabular mode10.2.6; if (code_mode) Exit code mode10.2.3; Look for cross-references and then render10.2.7; } else { if (tabular_mode == FALSE) { if (code_mode == FALSE) Enter code mode10.2.1 else HTML_TAG("br"); if (Regexp::match(NULL, P->par_texts, U"%c*%C\t+%C%c*")) Enter tabular mode10.2.4; } if (tabular_mode) Render the tab-divided line as an HTML table row10.2.5 else Render the line in code mode10.2.2; } } if (tabular_mode) Exit tabular mode10.2.6; if (code_mode) Exit code mode10.2.3;
- This code is used in §10.
§10.2.1. Code mode. Code mode is when the renderer is working on a displayed quotation of source code, broken off from the main narrative.
Enter code mode10.2.1 =
code_mode = 1; TEMPORARY_TEXT(id) WRITE_TO(id, "c%d", ++unique_code_pos_counter); if (code_example) WRITE_TO(id, "_%d", code_example->allocation_id); TEMPORARY_TEXT(comment) WRITE_TO(comment, "START CODE \"%S\"", id); HTML::comment(OUT, comment); DISCARD_TEXT(comment) HTML_OPEN_WITH("blockquote", "class=\"code\""); HTML_OPEN_WITH("p", "class=\"quoted\""); HTML::anchor(OUT, id);
- This code is used in §10.2.
§10.2.2. Quoted code is pretty well passed through in raw form, except for Javascript paste markers, which occupy a lot of code below but don't actually come up very much.
Render the line in code mode10.2.2 =
Render some indentation10.2.2.1; TEMPORARY_TEXT(raw) Str::copy(raw, P->par_texts); if (indoc_settings->treat_code_as_verbatim == FALSE) { Regexp::replace(raw, U"{%*%*}", NULL, REP_REPEATING); Remove any paste-continuation marker Take note of any named inline example10.2.2.2; match_results mr = Regexp::create_mr(); if (Regexp::match(&mr, raw, U"(%c*?){%*}(%c*)")) { Convert this paste marker to a Javascript paste mechanism10.2.2.3; Regexp::dispose_of(&mr); } } WRITE("%S\n", raw);
- This code is used in §10.2.
§10.2.2.1. A distinctly olde-worlde way to indent.
Render some indentation10.2.2.1 =
int ic = P->par_indentation; while (ic > 1) { ic--; WRITE("    "); }
- This code is used in §10.2.2.
§10.2.2.2. Take note of any named inline example10.2.2.2 =
match_results mr = Regexp::create_mr(); if ((index_to_examples) && (Regexp::match(&mr, raw, U"{%*}"(%c*)"%c*")) && (Str::ne_wide_string(mr.exp[0], U"Midsummer Day"))) { ExamplesIndex::add_to_alphabetic_examples_index(mr.exp[0], S, NULL, TRUE); Regexp::dispose_of(&mr); }
- This code is used in §10.2.2.
§10.2.3. Exit code mode10.2.3 =
code_mode = FALSE; HTML_CLOSE("p"); HTML_CLOSE("blockquote"); HTML::comment(OUT, I"END CODE");
- This code is used in §10.2 (twice).
§10.2.4. Tabular mode. Tabular mode is not an alternative to code mode: it's a deeper mode within it, and is used for a display of I7 source text (i.e., code) when it needs to show a Table.
Enter tabular mode10.2.4 =
tabular_mode = TRUE; HTML_CLOSE("p"); HTML_CLOSE("blockquote"); HTML_OPEN_WITH("table", "class=\"codetable\"");
- This code is used in §10.2.
§10.2.5. Within tabular mode, the following renders lines.
Note that any run of one or more tabs is treated as a single column division, and that leading or trailing tabs are ignored — so there is no way to code for an entirely empty cell. (This is fine for Inform documentation purposes since the only empty cells are trailing ones in the line anyway; people use the blank marker -- explicitly if they want blanks.)
Render the tab-divided line as an HTML table row10.2.5 =
HTML_OPEN("tr"); HTML_OPEN_WITH("td", "class=\"quotedtablecell\""); HTML_OPEN_WITH("p", "class=\"quoted\""); TEMPORARY_TEXT(row) match_results mr = Regexp::create_mr(); Regexp::match(&mr, P->par_texts, U" *(%c*?) *"); Str::copy(row, mr.exp[0]); Strip leading and trailing space while (Regexp::match(&mr, row, U"(%c*?)\t+(%c*)")) { WRITE("%S", mr.exp[0]); Place a cell division at any run of one or more tabs HTML_CLOSE("p"); HTML_CLOSE("td"); HTML_OPEN_WITH("td", "class=\"quotedtablecell\""); HTML_OPEN_WITH("p", "class=\"quoted\""); Str::copy(row, mr.exp[1]); } WRITE("%S", row); DISCARD_TEXT(row) Regexp::dispose_of(&mr); HTML_CLOSE("p"); HTML_CLOSE("td"); HTML_CLOSE("tr");
- This code is used in §10.2.
§10.2.6. Exit tabular mode10.2.6 =
tabular_mode = FALSE; HTML_CLOSE("table"); HTML_OPEN("blockquote"); HTML_OPEN("p");
- This code is used in §10.2 (twice).
§10.2.7. Regular mode. So this is what happens when we're not in either tabular or code mode.
The foot of a block of documentation sometimes contains cross-references to other blocks (resolved by name), and this is where we recognise and convert those to HTML links.
Look for cross-references and then render10.2.7 =
match_results mr = Regexp::create_mr(); if (Regexp::match(&mr, P->par_texts, U"%((-*)See {(%c*?)} for (%c*?).%) *")) { if (Str::len(mr.exp[0]) == 0) { Renderer::render_cross_reference(OUT, mr.exp[1], mr.exp[2], V, 0); } else { if (last_xref_type == 0) { HTML_TAG("hr"); } Renderer::render_cross_reference(OUT, mr.exp[1], mr.exp[2], V, 1); } last_xref_type = 1; } else if (Regexp::match(&mr, P->par_texts, U"%((-*)See (%c*?) for (%c*?).%) *")) { if (Str::len(mr.exp[0]) == 0) { Renderer::render_cross_reference(OUT, mr.exp[1], mr.exp[2], V, 0); } else { if (last_xref_type == 0) { HTML_TAG("hr"); } Renderer::render_cross_reference(OUT, mr.exp[1], mr.exp[2], V, 1); } last_xref_type = 1; } else if (Regexp::match(&mr, P->par_texts, U"%(See example "(%c*?)"%) *")) { if (last_xref_type == 1) { HTML_TAG("hr"); } Renderer::render_example_cross_reference(OUT, mr.exp[0], V); last_xref_type = 2; } else Render a non-quotation paragraph to HTML10.2.7.1; Regexp::dispose_of(&mr);
- This code is used in §10.2.
§10.2.7.1. Blank lines are simply ignored.
Render a non-quotation paragraph to HTML10.2.7.1 =
if (Regexp::match(NULL, P->par_texts, U" *") == FALSE) { WRITE("%S", P->par_prefix); if (P->par_suppression == TRUE) { WRITE("%S\n", P->par_texts); } else { if (Str::len(P->par_styles) > 0) { TEMPORARY_TEXT(details) WRITE_TO(details, "class=\"%S\"", P->par_styles); HTML::open(OUT, "p", details, __FILE__, __LINE__); DISCARD_TEXT(details) WRITE("%S", P->par_texts); HTML_CLOSE("p"); } else { HTML_OPEN("p"); WRITE("%S", P->par_texts); HTML_CLOSE("p"); } } }
- This code is used in §10.2.7.
§10.2.2.3. Javascript paste icons. That's the whole rendering routine, except for the handling of Javascript paste icons. In rawtext, these look like so:
{*}A useful sentence.
The {*} is replaced by a button which, when clicked on, performs a Javascript function call to paste "A useful sentence" into the Inform application's source text pane. (Note that the "A useful sentence" text is still also rendered on screen — it doesn't vanish into the button.) These pastes can occur only in code mode, and can extend for multiple lines, as we'll see below.
Convert this paste marker to a Javascript paste mechanism10.2.2.3 =
Str::copy(raw, mr.exp[0]); TEMPORARY_TEXT(right) Str::copy(right, mr.exp[1]); if (indoc_settings->javascript == FALSE) { WRITE_TO(raw, "%S", right); } else { WRITE_TO(raw, "<a href=\"javascript:pasteCode("); TEMPORARY_TEXT(J_text) Determine the quoted J-text10.2.2.3.1; TEMPORARY_TEXT(titling) if (code_example) { WRITE_TO(titling, "Example - "); LOOP_THROUGH_TEXT(pos, code_example->ex_public_name) { inchar32_t c = Str::get(pos); if (c == '\'') WRITE_TO(titling, "\\'"); else PUT_TO(titling, c); } } Renderer::apply_Inform_escape_characters(titling); WRITE_TO(raw, "'%S\\n'", J_text); WRITE_TO(raw, ")\">"); if (indoc_settings->retina_images) HTMLUtilities::image_element_scaled(raw, I"paste@2x.png", 13, 13); else HTMLUtilities::image_element(raw, I"paste.png"); WRITE_TO(raw, "</a> "); if ((indoc_settings->support_creation) && (Str::len(titling) > 0)) { WRITE_TO(raw, "<a href=\"javascript:createNewProject"); WRITE_TO(raw, "("); WRITE_TO(raw, "'%S\\n', '%S'", J_text, titling); WRITE_TO(raw, ")\">"); if (indoc_settings->retina_images) { HTMLUtilities::image_element_scaled(raw, I"create@2x.png", 26, 13); } else { HTMLUtilities::image_element(raw, I"create.png"); } WRITE_TO(raw, "</a>"); WRITE_TO(raw, " "); } WRITE_TO(raw, "%S", right); }
- This code is used in §10.2.2.
§10.2.2.3.1. The rawtext is doing something like this:
{*}A ball is in the bag. The bag is on the kitchen table. This single sentence doesn't make much of a simulation. Let's add: {**}The stitched seam is part of the ball.
The line count i points to the first line of this. The paste consists of the two lines about the bag and the table, but with the stitched seam line added in, because of the {**} continuation marker. The "range" is down to the last line which is included in the paste.
Determine the quoted J-text10.2.2.3.1 =
int up_to; one line beyond what will be pasted Find the range of rawtext lines which fall into this paste10.2.2.3.1.1; Collate the indented lines in that range into the J-text10.2.2.3.1.2; Renderer::apply_Inform_escape_characters(J_text);
- This code is used in §10.2.2.3.
§10.2.2.3.1.1. Find the range of rawtext lines which fall into this paste10.2.2.3.1.1 =
for (up_to = i; ((up_to<no_paras_in_block_buffer) && ((paragraphs[up_to].par_indentation > 0) || (Str::len(paragraphs[up_to].par_texts) == 0))); up_to++) { ; } int extended_range = TRUE; while (extended_range) { extended_range = FALSE; int l; for (l=up_to; ((l<no_paras_in_block_buffer) && (paragraphs[l].par_indentation == 0)); l++) { ; } if ((l<no_paras_in_block_buffer) && (Regexp::match(NULL, paragraphs[l].par_texts, U"%c*{%*%*}%c*"))) { for (l++; ((l<no_paras_in_block_buffer) && ((paragraphs[l].par_indentation > 0) || (Str::len(paragraphs[l].par_texts) == 0))); l++) { ; } up_to = l; extended_range = TRUE; } }
- This code is used in §10.2.2.3.1.
§10.2.2.3.1.2. Collate the indented lines in that range into the J-text10.2.2.3.1.2 =
for (int j=i; j<up_to; j++) { int ic = paragraphs[j].par_indentation; TEMPORARY_TEXT(joinbit) while (ic > 1) { ic--; PUT_TO(J_text, '\t'); } if (j == i) Str::copy(joinbit, right); else Str::copy(joinbit, paragraphs[j].par_texts); if ((paragraphs[j].par_indentation == 0) && (Str::len(joinbit) > 0)) { TEMPORARY_TEXT(br) WRITE_TO(br, "[%S]", joinbit); Str::copy(joinbit, br); DISCARD_TEXT(br) } inchar32_t prev_c = 0; for (int k=0, L = Str::len(joinbit); k<L; k++) { inchar32_t c = Str::get_at(joinbit, k); switch (c) { case '\\': WRITE_TO(J_text, "___backslash___"); break; case '\'': WRITE_TO(J_text, "\\'"); break; case '\t': if (prev_c != '\t') WRITE_TO(J_text, "\\t"); break; case '&': if (Str::includes_wide_string_at(joinbit, U"amp;", k+1)) k += 4; PUT_TO(J_text, c); break; case '{': if (Str::includes_wide_string_at(joinbit, U"**}", k+1)) k += 3; else PUT_TO(J_text, c); break; default: PUT_TO(J_text, c); break; } prev_c = c; } PUT_TO(J_text, '\n'); DISCARD_TEXT(joinbit) }
- This code is used in §10.2.2.3.1.
void Renderer::remove_paste_markers(text_stream *text) { for (int i=0, L=Str::len(text); i<L; i++) { if ((Str::get_at(text, i) == '{') && (Str::get_at(text, i+1) == '*') && (Str::get_at(text, i+2) == '*') && (Str::get_at(text, i+3) == '}')) { Renderer::remove_paste_markers_from(text, i); return; } } } void Renderer::remove_paste_markers_from(text_stream *text, int i) { TEMPORARY_TEXT(modified) for (int j=0; j<i; j++) PUT_TO(modified, Str::get_at(text, j)); for (int L=Str::len(text); i<L; i++) { inchar32_t c = Str::get_at(text, i); if ((c == '{') && (Str::get_at(text, i+1) == '*') && (Str::get_at(text, i+2) == '*') && (Str::get_at(text, i+3) == '}')) i+=3; else PUT_TO(modified, c); } Str::copy(text, modified); DISCARD_TEXT(modified) }
void Renderer::apply_Inform_escape_characters(text_stream *text) { TEMPORARY_TEXT(modified) for (int i=0, L=Str::len(text); i<L; i++) { inchar32_t c = Str::get_at(text, i); switch (c) { case '\\': c = Str::get_at(text, ++i); if (c == '\'') WRITE_TO(modified, "[=0x0027=]"); else if (c == 't') WRITE_TO(modified, "[=0x0009=]"); else WRITE_TO(modified, "\\%c", c); break; case '\"': WRITE_TO(modified, "[=0x0022=]"); break; case '\x0a': WRITE_TO(modified, "[=0x000A=]"); break; case '\x0d': WRITE_TO(modified, "[=0x000A=]"); break; case '\t': WRITE_TO(modified, "[=0x0009=]"); break; case '<': WRITE_TO(modified, "[=0x003C=]"); break; case '>': WRITE_TO(modified, "[=0x003E=]"); break; case '_': if (Str::includes_wide_string_at(text, U"__backslash___", i+1)) { WRITE_TO(modified, "[=0x005C=]"); i+=14; } else PUT_TO(modified, c); break; case '&': if (Str::includes_wide_string_at(text, U"quot;", i+1)) { WRITE_TO(modified, "[=0x0022=]"); i+=5; } else if (Str::includes_wide_string_at(text, U"lt;", i+1)) { WRITE_TO(modified, "[=0x003C=]"); i+=3; } else if (Str::includes_wide_string_at(text, U"gt;", i+1)) { WRITE_TO(modified, "[=0x003E=]"); i+=3; } else { WRITE_TO(modified, "[=0x0026=]"); } break; default: PUT_TO(modified, c); break; } } Str::copy(text, modified); DISCARD_TEXT(modified) }
§13. Rendering cross-references to other sections. These occur when the rawtext contains paragraphs with a very specific arrangement:
(-See Units for a more sophisticated capacity system.)
The idea is that, except for the brackets and dash, the text makes sense as it stands; but in HTML, we can use the section title ("Units" here) to find which block is meant, and encode this as a link.
void Renderer::render_cross_reference(OUTPUT_STREAM, text_stream *sname, text_stream *reason, volume *V, int quieter) { if (indoc_settings->format == PLAIN_FORMAT) WRITE("(See %S for %S.)\n", sname, reason); if (indoc_settings->format == HTML_FORMAT) { TEMPORARY_TEXT(dest) WRITE_TO(dest, "index.html"); Identify the reference destination and be sure it exists13.1; HTML_OPEN_WITH("p", "class=\"crossreference\""); HTML::begin_link_with_class(OUT, I"xreflink", dest); HTMLUtilities::asterisk_image(OUT, I"xref.png"); WRITE(" <i>See </i><b>%S</b>", sname); HTML::end_link(OUT); if (quieter == FALSE) HTML_OPEN("i"); WRITE(" for %S", reason); if (quieter == FALSE) HTML_CLOSE("i"); HTML_CLOSE("p"); DISCARD_TEXT(dest) } }
§13.1. Identify the reference destination and be sure it exists13.1 =
section *S = (section *) Dictionaries::read_value(V->sections_by_name, sname); if (S) Str::copy(dest, S->section_URL); else Errors::with_text("cross-reference to %S points to no section", sname);
- This code is used in §13.
§14. And similarly, for cross-referencing to examples by name:
(See example "Blink")
void Renderer::render_example_cross_reference(OUTPUT_STREAM, text_stream *ename, volume *V) { example *E = (example *) Dictionaries::read_value(examples_by_name, ename); if (E) Examples::render_example_cue(OUT, E, V, 1); else Errors::with_text("cross-reference to %S points to no section", ename); }
§15. Handling the formatted file. The idea is that several sections in a row may need to be written to the same file, or may not. So this routine is called to guarantee that the right file is open, rather than always to open one.
text_stream *Renderer::formatted_file_must_be(OUTPUT_STREAM, volume *V, section *S) { if (Filenames::eq(S->section_filename, current_FTD_filename) == FALSE) { if (current_FTD_filename) OUT = Renderer::close_formatted_file(OUT); current_FTD_filename = S->section_filename; OUT = ¤t_FTD_stream; if (Streams::open_to_file(OUT, S->section_filename, UTF8_ENC) == FALSE) Errors::fatal_with_file("can't write documentation", S->section_filename); if (indoc_settings->wrapper == WRAPPER_epub) { ebook_page *page = Epub::note_page(indoc_settings->ebook, S->section_filename, S->section_file_title, I""); if (S == V->sections[0]) { ebook_volume *ev = Epub::starts_volume(indoc_settings->ebook, page, V->vol_title); filename *F = Filenames::in(indoc_settings->destination, V->vol_CSS_leafname); Epub::use_CSS(ev, F); } if (S->begins_which_chapter) S->begins_which_chapter->ebook_ref = Epub::starts_chapter(indoc_settings->ebook, page, S->begins_which_chapter->chapter_full_title, S->begins_which_chapter->chapter_URL); } formatted_file *ftd = CREATE(formatted_file); ftd->name = current_FTD_filename; if (indoc_settings->format == HTML_FORMAT) Write the HTML header for the formatted file15.2; } return OUT; }
§15.1. When we certainly want to dispose of the current file:
text_stream *Renderer::close_formatted_file(OUTPUT_STREAM) { if (current_FTD_filename) { if (indoc_settings->format == HTML_FORMAT) Write the HTML footer for the formatted file15.1.1; Streams::close(¤t_FTD_stream); current_FTD_filename = NULL; } return NULL; }
§15.2. The HTML files are topped and tailed either using a template supplied, or with a <head> we make ourselves.
Write the HTML header for the formatted file15.2 =
TEMPORARY_TEXT(top) HTMLUtilities::get_tt_matter(top, 0, 1); if (Str::len(top) > 0) { match_results mr = Regexp::create_mr(); if (Regexp::match(&mr, top, U"(%c*?)<title>%c*?</title>(%c*)")) WRITE("%S<title>Inform 7 - %S</title>%S", mr.exp[0], S->section_file_title, mr.exp[1]); else WRITE("%S", top); Regexp::dispose_of(&mr); } else { HTMLUtilities::begin_file(OUT, V); HTMLUtilities::write_title(OUT, S->section_file_title); if (indoc_settings->javascript) { HTML::open_javascript(OUT, FALSE); HTMLUtilities::write_javascript_for_buttons(OUT); HTML::close_javascript(OUT); } HTML::end_head(OUT); HTML::begin_body(OUT, I"paper papertint"); } if (indoc_settings->javascript) { HTMLUtilities::paste_script(OUT, NULL, 0); HTMLUtilities::create_script(OUT, NULL, 0, NULL); } DISCARD_TEXT(top)
- This code is used in §15.
§15.1.1. Write the HTML footer for the formatted file15.1.1 =
TEMPORARY_TEXT(tail) HTMLUtilities::get_tt_matter(tail, 0, 0); if (Str::len(tail) > 0) { WRITE("%S", tail); } else { HTML::end_body(OUT); } DISCARD_TEXT(tail)
- This code is used in §15.1.