# HTML.pm: output tree as HTML. # # Copyright 2011-2026 Free Software Foundation, Inc. # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 3 of the License, # or (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # # # The documentation of the HTML customization API which is both # used and implemented in the current file is in the customization_api # Texinfo manual. # # Formatting and conversion functions that can be replaced by user-defined # functions should only use documented functions to pass information # and formatted content, such that users can overrides them independently # without risking unwanted results. Also in formatting functions, the state of # the converter should only be accessed through functions, such as in_math, # in_preformatted_context, preformatted_classes_stack and similar functions. # # Original author: Patrice Dumas # ALTIMP XSTexinfo/convert/ConvertXS.xs # ALTIMP C/convert/*.[ch] package Texinfo::Convert::HTML; # charnames::vianame is not documented in 5.6.0. use 5.008; # See 'The "Unicode Bug"' under 'perlunicode' man page. This means # that regular expressions will treat characters 128-255 in a Perl string # the same regardless of whether the string is using a UTF-8 encoding. # For older Perls, you can use utf8::upgrade on the strings, where the # difference matters. # Also follows unicode rules for uc() and lc (). use if $] >= 5.012, feature => 'unicode_strings'; use if $] >= 5.014, re => '/a'; # ASCII-only character classes in regexes use strict; # To check if there is no erroneous autovivification #no autovivification qw(fetch delete exists store strict); use Carp qw(cluck confess); # for abort #use POSIX; use File::Copy qw(copy); use File::Spec; my $updir = File::Spec->updir(); use Storable; use Encode qw(find_encoding decode encode); use charnames (); use Texinfo::Convert::ConvertXS; use Texinfo::XSLoader; use Texinfo::Commands; use Texinfo::Options; use Texinfo::CommandsValues; use Texinfo::UnicodeData; use Texinfo::HTMLData; use Texinfo::HTMLDataCSS; use Texinfo::TreeElement; use Texinfo::Common; use Texinfo::Config; use Texinfo::Convert::Unicode; use Texinfo::Convert::Texinfo; use Texinfo::Convert::Utils; use Texinfo::Convert::Text; use Texinfo::Convert::NodeNameNormalization; use Texinfo::ManipulateTree; use Texinfo::Structuring; use Texinfo::OutputUnits; # for index_entry_first_letter_text_or_command use Texinfo::Indices; use Texinfo::Convert::Converter; # used to convert Texinfo to LaTeX math in @math and @displaymath # for further conversion by softwares that only convert LaTeX. # NOTE mathjax does not implement some constructs output by the # Texinfo::Convert::LaTeX converter. Examples in 2022: # \mathord{\text{}} \textsl{} \copyright{} \mathsterling{} use Texinfo::Convert::LaTeX; require Exporter; our @ISA = qw(Texinfo::Convert::Converter); our $VERSION = '7.3'; my $XS_convert = Texinfo::XSLoader::XS_convert_enabled(); my %XS_overrides = ( "Texinfo::Convert::HTML::_default_format_protect_text" => "Texinfo::MiscXS::default_format_protect_text", "Texinfo::Convert::HTML::_entity_text" => "Texinfo::MiscXS::entity_text", ); my %XS_conversion_overrides = ( "Texinfo::Convert::HTML::_XS_format_setup" => "Texinfo::Convert::ConvertXS::html_format_setup", "Texinfo::Convert::HTML::converter_defaults" => "Texinfo::Convert::ConvertXS::converter_defaults", "Texinfo::Convert::HTML::_XS_html_converter_initialize_beginning" => "Texinfo::Convert::ConvertXS::html_converter_initialize_beginning", "Texinfo::Convert::HTML::_XS_html_converter_get_customization" => "Texinfo::Convert::ConvertXS::html_converter_get_customization_sv", "Texinfo::Convert::HTML::output" => "Texinfo::Convert::ConvertXS::html_output", "Texinfo::Convert::HTML::convert" => "Texinfo::Convert::ConvertXS::html_convert", # Not used by output or convert. May be called on a converter when # output nor convert are used. Happens in tests. "Texinfo::Convert::HTML::conversion_initialization" => "Texinfo::Convert::ConvertXS::html_conversion_initialization", "Texinfo::Convert::HTML::output_internal_links" => "Texinfo::Convert::ConvertXS::html_output_internal_links", "Texinfo::Convert::HTML::command_id" => "Texinfo::Convert::ConvertXS::html_command_id", "Texinfo::Convert::HTML::command_contents_target" => "Texinfo::Convert::ConvertXS::html_command_contents_target", "Texinfo::Convert::HTML::footnote_location_target" => "Texinfo::Convert::ConvertXS::html_footnote_location_target", "Texinfo::Convert::HTML::footnote_location_href" => "Texinfo::Convert::ConvertXS::html_footnote_location_href", "Texinfo::Convert::HTML::command_filename" => "Texinfo::Convert::ConvertXS::html_command_filename", "Texinfo::Convert::HTML::command_root_element_command" => "Texinfo::Convert::ConvertXS::html_command_root_element_command", "Texinfo::Convert::HTML::command_node" => "Texinfo::Convert::ConvertXS::html_command_node", "Texinfo::Convert::HTML::_internal_command_href" => "Texinfo::Convert::ConvertXS::html_internal_command_href", "Texinfo::Convert::HTML::command_contents_href" => "Texinfo::Convert::ConvertXS::html_command_contents_href", "Texinfo::Convert::HTML::_internal_command_text" => "Texinfo::Convert::ConvertXS::html_internal_command_text", "Texinfo::Convert::HTML::_internal_command_name" => "Texinfo::Convert::ConvertXS::html_internal_command_name", "Texinfo::Convert::HTML::command_description" => "Texinfo::Convert::ConvertXS::html_command_description", "Texinfo::Convert::HTML::special_unit_info_text", => "Texinfo::Convert::ConvertXS::html_special_unit_info_text", "Texinfo::Convert::HTML::global_direction_unit" => "Texinfo::Convert::ConvertXS::html_global_direction_unit", "Texinfo::Convert::HTML::global_direction_text" => "Texinfo::Convert::ConvertXS::html_global_direction_text", "Texinfo::Convert::HTML::_XS_set_shared_conversion_state" => "Texinfo::Convert::ConvertXS::html_set_shared_conversion_state", "Texinfo::Convert::HTML::_XS_get_shared_conversion_state" => "Texinfo::Convert::ConvertXS::html_get_shared_conversion_state", "Texinfo::Convert::HTML::get_info" => "Texinfo::Convert::ConvertXS::html_get_info", # Following functions are sometimes called from converting functions, # but mainly from within calls of convert_tree called from # converting functions. The make sure that the formatting contexts # accessed and modified are only the formatting contexts in C data. "Texinfo::Convert::HTML::_open_command_update_context" => "Texinfo::Convert::ConvertXS::html_open_command_update_context", "Texinfo::Convert::HTML::_convert_command_update_context", => "Texinfo::Convert::ConvertXS::html_convert_command_update_context", "Texinfo::Convert::HTML::_open_type_update_context", => "Texinfo::Convert::ConvertXS::html_open_type_update_context", "Texinfo::Convert::HTML::_convert_type_update_context" => "Texinfo::Convert::ConvertXS::html_convert_type_update_context", "Texinfo::Convert::HTML::_new_document_context" => "Texinfo::Convert::ConvertXS::html_new_document_context", "Texinfo::Convert::HTML::_pop_document_context" => "Texinfo::Convert::ConvertXS::html_pop_document_context", "Texinfo::Convert::HTML::_set_code_context" => "Texinfo::Convert::ConvertXS::html_set_code_context", "Texinfo::Convert::HTML::_pop_code_context" => "Texinfo::Convert::ConvertXS::html_pop_code_context", "Texinfo::Convert::HTML::_set_string_context" => "Texinfo::Convert::ConvertXS::html_set_string_context", "Texinfo::Convert::HTML::_unset_string_context" => "Texinfo::Convert::ConvertXS::html_unset_string_context", "Texinfo::Convert::HTML::_set_raw_context" => "Texinfo::Convert::ConvertXS::html_set_raw_context", "Texinfo::Convert::HTML::_unset_raw_context" => "Texinfo::Convert::ConvertXS::html_unset_raw_context", "Texinfo::Convert::HTML::_set_multiple_conversions" => "Texinfo::Convert::ConvertXS::html_set_multiple_conversions", "Texinfo::Convert::HTML::_unset_multiple_conversions" => "Texinfo::Convert::ConvertXS::html_unset_multiple_conversions", "Texinfo::Convert::HTML::_debug_print_html_contexts" => "Texinfo::Convert::ConvertXS::html_debug_print_html_contexts", "Texinfo::Convert::HTML::in_math" => "Texinfo::Convert::ConvertXS::html_in_math", "Texinfo::Convert::HTML::in_preformatted_context" => "Texinfo::Convert::ConvertXS::html_in_preformatted_context", "Texinfo::Convert::HTML::inside_preformatted" => "Texinfo::Convert::ConvertXS::html_inside_preformatted", "Texinfo::Convert::HTML::in_upper_case" => "Texinfo::Convert::ConvertXS::html_in_upper_case", "Texinfo::Convert::HTML::in_non_breakable_space" => "Texinfo::Convert::ConvertXS::html_in_non_breakable_space", "Texinfo::Convert::HTML::in_space_protected" => "Texinfo::Convert::ConvertXS::html_in_space_protected", "Texinfo::Convert::HTML::in_code" => "Texinfo::Convert::ConvertXS::html_in_code", "Texinfo::Convert::HTML::in_string" => "Texinfo::Convert::ConvertXS::html_in_string", "Texinfo::Convert::HTML::in_verbatim" => "Texinfo::Convert::ConvertXS::html_in_verbatim", "Texinfo::Convert::HTML::in_raw" => "Texinfo::Convert::ConvertXS::html_in_raw", "Texinfo::Convert::HTML::in_multiple_conversions" => "Texinfo::Convert::ConvertXS::html_in_multiple_conversions", "Texinfo::Convert::HTML::paragraph_number" => "Texinfo::Convert::ConvertXS::html_paragraph_number", "Texinfo::Convert::HTML::preformatted_number" => "Texinfo::Convert::ConvertXS::html_preformatted_number", "Texinfo::Convert::HTML::top_block_command" => "Texinfo::Convert::ConvertXS::html_top_block_command", "Texinfo::Convert::HTML::preformatted_classes_stack" => "Texinfo::Convert::ConvertXS::html_preformatted_classes_stack", "Texinfo::Convert::HTML::in_align" => "Texinfo::Convert::ConvertXS::html_in_align", "Texinfo::Convert::HTML::multi_expanded_region" => "Texinfo::Convert::ConvertXS::html_multi_expanded_region", "Texinfo::Convert::HTML::current_filename" => "Texinfo::Convert::ConvertXS::html_current_filename", "Texinfo::Convert::HTML::current_output_unit" => "Texinfo::Convert::ConvertXS::html_current_output_unit", "Texinfo::Convert::HTML::count_elements_in_filename" => "Texinfo::Convert::ConvertXS::html_count_elements_in_filename", "Texinfo::Convert::HTML::is_format_expanded", => "Texinfo::Convert::ConvertXS::html_is_format_expanded", "Texinfo::Convert::HTML::register_file_information" => "Texinfo::Convert::ConvertXS::html_register_file_information", "Texinfo::Convert::HTML::get_file_information", => "Texinfo::Convert::ConvertXS::html_get_file_information", "Texinfo::Convert::HTML::register_opened_section_level" => "Texinfo::Convert::ConvertXS::html_register_opened_section_level", "Texinfo::Convert::HTML::close_registered_sections_level" => "Texinfo::Convert::ConvertXS::html_close_registered_sections_level", "Texinfo::Convert::HTML::set_global_direction" => "Texinfo::Convert::ConvertXS::html_set_global_direction", "Texinfo::Convert::HTML::html_attribute_class" => "Texinfo::Convert::ConvertXS::html_attribute_class", "Texinfo::Convert::HTML::html_get_css_elements_classes" => "Texinfo::Convert::ConvertXS::html_get_css_elements_classes", "Texinfo::Convert::HTML::css_add_info" => "Texinfo::Convert::ConvertXS::html_css_add_info", "Texinfo::Convert::HTML::css_set_selector_style" => "Texinfo::Convert::ConvertXS::html_css_set_selector_style", "Texinfo::Convert::HTML::css_get_info" => "Texinfo::Convert::ConvertXS::html_css_get_info", "Texinfo::Convert::HTML::css_get_selector_style", => "Texinfo::Convert::ConvertXS::html_css_get_selector_style", "Texinfo::Convert::HTML::register_footnote", => "Texinfo::Convert::ConvertXS::html_register_footnote", "Texinfo::Convert::HTML::get_pending_footnotes", => "Texinfo::Convert::ConvertXS::html_get_pending_footnotes", "Texinfo::Convert::HTML::register_pending_formatted_inline_content" => "Texinfo::Convert::ConvertXS::html_register_pending_formatted_inline_content", "Texinfo::Convert::HTML::cancel_pending_formatted_inline_content", => "Texinfo::Convert::ConvertXS::html_cancel_pending_formatted_inline_content", "Texinfo::Convert::HTML::get_pending_formatted_inline_content", => "Texinfo::Convert::ConvertXS::html_get_pending_formatted_inline_content", "Texinfo::Convert::HTML::associate_pending_formatted_inline_content" => "Texinfo::Convert::ConvertXS::html_associate_pending_formatted_inline_content", "Texinfo::Convert::HTML::get_associated_formatted_inline_content", => "Texinfo::Convert::ConvertXS::html_get_associated_formatted_inline_content", "Texinfo::Convert::HTML::_push_referred_command_stack_command" => "Texinfo::Convert::ConvertXS::html_push_referred_command_stack_command", "Texinfo::Convert::HTML::_pop_referred_command_stack" => "Texinfo::Convert::ConvertXS::html_pop_referred_command_stack", "Texinfo::Convert::HTML::_command_is_in_referred_command_stack" => "Texinfo::Convert::ConvertXS::html_command_is_in_referred_command_stack", "Texinfo::Convert::HTML::_check_htmlxref_already_warned" => "Texinfo::Convert::ConvertXS::html_check_htmlxref_already_warned", "Texinfo::Convert::HTML::_translate_names" => "Texinfo::Convert::ConvertXS::html_translate_names", # Cannot be overriden, in general the trees are not registered in Perl #"Texinfo::Convert::HTML::_XS_html_convert_tree" # => "Texinfo::Convert::ConvertXS::html_convert_tree", ); # HTML C data initialization independent of customization and of Perl # default variables. sub _XS_format_setup() { } our $module_loaded = 0; sub import { if (!$module_loaded) { foreach my $sub (keys %XS_overrides) { Texinfo::XSLoader::override ($sub, $XS_overrides{$sub}); } if ($XS_convert) { foreach my $sub (keys %XS_conversion_overrides) { Texinfo::XSLoader::override ($sub, $XS_conversion_overrides{$sub}); } # initialize HTML C data _XS_format_setup(); } $module_loaded = 1; } # The usual import method goto &Exporter::import; } my %nobrace_commands = %Texinfo::Commands::nobrace_commands; my %line_commands = %Texinfo::Commands::line_commands; my %nobrace_symbol_text = %Texinfo::CommandsValues::nobrace_symbol_text; my %accent_commands = %Texinfo::Commands::accent_commands; my %sectioning_heading_commands = %Texinfo::Commands::sectioning_heading_commands; my %def_commands = %Texinfo::Commands::def_commands; my %ref_commands = %Texinfo::Commands::ref_commands; my %brace_commands = %Texinfo::Commands::brace_commands; my %block_commands = %Texinfo::Commands::block_commands; my %root_commands = %Texinfo::Commands::root_commands; my %preformatted_commands = %Texinfo::Commands::preformatted_commands; my %math_commands = %Texinfo::Commands::math_commands; my %preformatted_code_commands = %Texinfo::Commands::preformatted_code_commands; my %letter_no_arg_commands = %Texinfo::Commands::letter_no_arg_commands; my %formatted_line_commands = %Texinfo::Commands::formatted_line_commands; my %formatted_nobrace_commands = %Texinfo::Commands::formatted_nobrace_commands; my %formattable_line_commands = %Texinfo::Commands::formattable_line_commands; my %explained_commands = %Texinfo::Commands::explained_commands; my %inline_format_commands = %Texinfo::Commands::inline_format_commands; my %brace_code_commands = %Texinfo::Commands::brace_code_commands; my %default_index_commands = %Texinfo::Commands::default_index_commands; my %small_block_associated_command = %Texinfo::Common::small_block_associated_command; foreach my $def_command (keys(%def_commands)) { $formatted_line_commands{$def_command} = 1 if (exists($line_commands{$def_command})); } my %HTML_align_commands; foreach my $align_command('raggedright', 'flushleft', 'flushright', 'center') { $HTML_align_commands{$align_command} = 1; } my %composition_context_commands = (%preformatted_commands, %root_commands, %HTML_align_commands); $composition_context_commands{'float'} = 1; my %format_context_commands = (%block_commands, %root_commands); my %format_raw_commands; foreach my $block_command (keys(%block_commands)) { $composition_context_commands{$block_command} = 1 if ($block_commands{$block_command} eq 'menu'); if ($block_commands{$block_command} eq 'format_raw') { $format_raw_commands{$block_command} = 1; delete $format_context_commands{$block_command}; } } foreach my $misc_context_command('tab', 'item', 'itemx', 'headitem') { $format_context_commands{$misc_context_command} = 1; } # API for html formatting # similar to texinfo_register_global_direction in Texinfo::Config, to be # used to modify global directions after the converter initialization, # but before association of global directions with output units sub set_global_direction($$;$) { my ($self, $direction, $node_texi_name) = @_; if (!$self->{'all_directions'}->{$direction}) { $self->converter_document_warn( sprintf(__("not setting an unknown direction: %s"), $direction)); return; } $self->{'customized_global_directions'} = {} if (!exists($self->{'customized_global_directions'})); $self->{'customized_global_directions'}->{$direction} = $node_texi_name; return; } sub _collect_css_element_class($$) { my ($self, $element_class) = @_; #if (not $self->{'document_global_context_counter'} # and not exists($self->{'current_filename'})) { # cluck "BUG: $element_class: CSS no current file"; #} if (exists($self->{'css_element_class_styles'}->{$element_class})) { if ($self->{'document_global_context_counter'}) { $self->{'document_global_context_css'}->{$element_class} = 1; } elsif (exists($self->{'current_filename'})) { $self->{'page_css'}->{$self->{'current_filename'}} = {} if (!exists($self->{'page_css'}->{$self->{'current_filename'}})); $self->{'page_css'}->{$self->{'current_filename'}}->{$element_class} = 1; } } } # $classes should be an array reference or undef sub html_attribute_class($$;$) { my ($self, $element, $classes) = @_; if (defined($classes) and ref($classes) ne 'ARRAY') { confess("html_attribute_class: $classes not an array ref (for $element)"); } if (!defined($classes) or scalar(@$classes) == 0 or $self->get_conf('NO_CSS')) { if ($element eq 'span') { return ''; } else { return "<$element"; } } my $style = ''; if ($self->get_conf('INLINE_CSS_STYLE')) { my @styles = (); foreach my $style_class (@$classes) { if (not defined($style_class)) { confess("class not defined (for $element)"); } if (exists($self->{'css_element_class_styles'} ->{"$element.$style_class"})) { push @styles, $self->{'css_element_class_styles'}->{"$element.$style_class"}; } } if (scalar(@styles) > 0) { $style = ' style="'.join(';', @styles).'"'; } } else { foreach my $style_class (@$classes) { if (not defined($style_class)) { confess ("class not defined (for $element)"); } _collect_css_element_class($self, "$element.$style_class"); } } my $class_str = join(' ', map {_protect_class_name($self, $_)} @$classes); return "<$element class=\"$class_str\"$style"; } # returns an array of CSS element.class seen in the $FILENAME sub html_get_css_elements_classes($;$) { my ($self, $filename) = @_; my %css_elements_classes; if (exists($self->{'document_global_context_css'})) { %css_elements_classes = ( %{$self->{'document_global_context_css'}} ); } if (defined($filename) and exists($self->{'page_css'}) and exists($self->{'page_css'}->{$filename})) { %css_elements_classes = ( %css_elements_classes, %{$self->{'page_css'}->{$filename}} ); } if ($css_elements_classes{'a.copiable-link'}) { $css_elements_classes{'span:hover a.copiable-link'} = 1; } my @result = sort(keys(%css_elements_classes)); return \@result; } sub close_html_lone_element($$) { my ($self, $html_element) = @_; if ($self->get_conf('USE_XML_SYNTAX')) { return $html_element . '/>'; } return $html_element .'>'; } my $xml_named_entity_nbsp = ' '; my $html_default_entity_nbsp = $xml_named_entity_nbsp; sub substitute_html_non_breaking_space($$) { my ($self, $text) = @_; my $non_breaking_space = $self->get_info('non_breaking_space'); # using \Q \E on the substitution leads to spurious \ $text =~ s/\Q$html_default_entity_nbsp\E/$non_breaking_space/g; return $text; } my @image_files_extensions = ('.png', '.jpg', '.jpeg', '.gif'); # this can be used in init files to get the path of the image # files. In general the result of image formatting cannot # be used to get an image file name path, as the path is not # used in the output. sub html_image_file_location_name($$$$$) { my ($self, $cmdname, $command, $image_basefile, $args) = @_; my @extensions = @image_files_extensions; my $image_file; my $image_extension; # this variable is bytes encoded in the filesystem encoding my ($image_path, $image_path_encoding); my $extension; # NOTE should be consistent with $image_basefile formatting if (defined($args->[4]) and defined($args->[4]->{'filenametext'})) { $extension = $args->[4]->{'filenametext'}; unshift @extensions, ("$extension", ".$extension"); } foreach my $tried_extension (@extensions) { my ($file_name, $file_name_encoding) = $self->encoded_input_file_name($image_basefile.$tried_extension); my $located_image_path = Texinfo::Common::locate_include_file($file_name, $self->get_conf('INCLUDE_DIRECTORIES')); if (defined($located_image_path) and $located_image_path ne '') { $image_path = $located_image_path; $image_path_encoding = $file_name_encoding; # use the @-command argument and not the file found using the # include paths. It is considered that the files in include paths # will be moved by the caller anyway. # If the file path found was to be used it should be decoded to perl # codepoints too. $image_file = $image_basefile.$tried_extension; $image_extension = $tried_extension; last; } } if (!defined($image_file) or $image_file eq '') { if (defined($extension) and $extension ne '') { $image_file = $image_basefile.$extension; $image_extension = $extension; } else { $image_file = "$image_basefile.jpg"; $image_extension = '.jpg'; } } return ($image_file, $image_extension, $image_path, $image_path_encoding); } sub css_add_info($$$) { my ($self, $spec, $css_info) = @_; if ($spec eq 'rules') { push @{$self->{'css_rule_lines'}}, $css_info; } elsif ($spec eq 'imports') { push @{$self->{'css_import_lines'}}, $css_info; } } sub css_set_selector_style($$$) { my ($self, $css_info, $css_style) = @_; if (!defined($css_style)) { delete($self->{'css_element_class_styles'}->{$css_info}); } else { $self->{'css_element_class_styles'}->{$css_info} = $css_style; } } sub css_get_info($$) { my ($self, $spec) = @_; my @empty_array; if ($spec eq 'rules') { if (defined($self->{'css_rule_lines'})) { return $self->{'css_rule_lines'}; } else { return \@empty_array; } } elsif ($spec eq 'imports') { if (defined($self->{'css_import_lines'})) { return $self->{'css_import_lines'}; } else { return \@empty_array; } } else { my @result = sort(keys(%{$self->{'css_element_class_styles'}})); return \@result; } } sub css_get_selector_style($$) { my ($self, $css_info) = @_; if (exists($self->{'css_element_class_styles'}->{$css_info})) { return $self->{'css_element_class_styles'}->{$css_info}; } else { return undef; } } # flags used to specify the conversion contexts. Use flags to avoid # passing multiple arguments or hashes, and also could be easier for # passing to C. our $CTXF_string = 0x0001; our $CTXF_code = 0x0002; my %default_css_string_commands_conversion; my %default_css_string_types_conversion; my %default_css_string_formatting_references; sub html_convert_css_string($$$) { my ($self, $element, $context_str) = @_; my $saved_commands = {}; my $saved_types = {}; my $saved_formatting_references = {}; foreach my $cmdname (keys(%default_css_string_commands_conversion)) { $saved_commands->{$cmdname} = $self->{'commands_conversion'}->{$cmdname}; $self->{'commands_conversion'}->{$cmdname} = $default_css_string_commands_conversion{$cmdname}; } foreach my $type (keys(%default_css_string_types_conversion)) { $saved_types->{$type} = $self->{'types_conversion'}->{$type}; $self->{'types_conversion'}->{$type} = $default_css_string_types_conversion{$type}; } foreach my $formatting_reference (keys(%default_css_string_formatting_references)) { $saved_formatting_references->{$formatting_reference} = $self->{'formatting_function'}->{$formatting_reference}; $self->{'formatting_function'}->{$formatting_reference} = $default_css_string_formatting_references{$formatting_reference}; } my $css_string_context_str = 'CSS string '.$context_str; _new_document_context($self, $css_string_context_str, $CTXF_string); my $result = $self->convert_tree($element, "new_fmt_ctx C($css_string_context_str)"); _pop_document_context($self); foreach my $cmdname (keys (%default_css_string_commands_conversion)) { $self->{'commands_conversion'}->{$cmdname} = $saved_commands->{$cmdname}; } foreach my $type (keys(%default_css_string_types_conversion)) { $self->{'types_conversion'}->{$type} = $saved_types->{$type}; } foreach my $formatting_reference (keys(%default_css_string_formatting_references)) { $self->{'formatting_function'}->{$formatting_reference} = $saved_formatting_references->{$formatting_reference}; } return $result; } my %special_list_mark_css_string_no_arg_command = ( # tried to use HYPHEN BULLET \2043 for use as in a bullet list, but, at least # with my test of firefox the result is very different from a bullet. # hyphen minus or hyphen \2010 are even smaller than hyphen bullet. # Use the Unicode codepoint used normally for a mathematical minus \2212 # even though it is too large, since the others are too short... # (which is actually the default, but this could change). #'minus' => '-', #'minus' => '\2010 ', 'minus' => '\2212 ', ); sub html_convert_css_string_for_list_mark($$;$) { my ($self, $element, $explanation) = @_; my $saved_css_string_no_arg_command = {}; foreach my $cmdname (keys(%special_list_mark_css_string_no_arg_command)) { $saved_css_string_no_arg_command->{$cmdname} = $self->{'no_arg_commands_formatting'}->{$cmdname}->{'css_string'}; $self->{'no_arg_commands_formatting'}->{$cmdname}->{'css_string'} = $special_list_mark_css_string_no_arg_command{$cmdname}; } my $result = $self->html_convert_css_string($element, $explanation); foreach my $cmdname (keys(%special_list_mark_css_string_no_arg_command)) { $self->{'no_arg_commands_formatting'}->{$cmdname}->{'css_string'} = $saved_css_string_no_arg_command->{$cmdname}; } return $result; } # API to access converter state for conversion sub in_math($) { my $self = shift; return $self->{'document_context'}->[-1]->{'math'}; } # set if in menu or preformatted command sub in_preformatted_context($) { my $self = shift; return $self->{'document_context'}->[-1]->{'preformatted_context'}->[-1]; } sub inside_preformatted($) { my $self = shift; return $self->{'document_context'}->[-1]->{'inside_preformatted'}; } sub in_upper_case($) { my $self = shift; return $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'upper_case'}; } sub in_non_breakable_space($) { my $self = shift; return $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'no_break'}; } sub in_space_protected($) { my $self = shift; return $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'space_protected'}; } sub in_code($) { my $self = shift; return $self->{'document_context'}->[-1]->{'monospace'}->[-1]; } sub in_string($) { my $self = shift; return $self->{'document_context'}->[-1]->{'string'}; } sub in_verbatim($) { my $self = shift; return $self->{'document_context'}->[-1]->{'verbatim'}; } sub in_raw($) { my $self = shift; return $self->{'document_context'}->[-1]->{'raw'}; } sub in_multiple_conversions($) { my $self = shift; return scalar(@{$self->{'multiple_pass'}}); } sub paragraph_number($) { my $self = shift; return $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'paragraph_number'}; } sub preformatted_number($) { my $self = shift; return $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'preformatted_number'}; } sub top_block_command($) { my $self = shift; return $self->{'document_context'}->[-1]->{'block_commands'}->[-1]; } sub preformatted_classes_stack($) { my $self = shift; return $self->{'document_context'}->[-1]->{'preformatted_classes'}; } sub in_align($) { my $self = shift; my $context = $self->{'document_context'}->[-1]->{'composition_context'}->[-1]; if (exists($HTML_align_commands{$context})) { return $context; } else { return undef; } } sub multi_expanded_region($) { my $self = shift; if (scalar(@{$self->{'multiple_pass'}})) { return $self->{'multiple_pass'}->[-1]; } return undef; } sub count_elements_in_filename($$$) { my ($self, $spec, $filename) = @_; if (!defined($filename)) { confess("count_elements_in_filename: filename undef"); } if ($spec eq 'total') { if (defined($self->{'elements_in_file_count'}->{$filename})) { return $self->{'elements_in_file_count'}->{$filename}; } } elsif ($spec eq 'remaining') { if (defined($self->{'file_counters'}->{$filename})) { return $self->{'file_counters'}->{$filename}; } } elsif ($spec eq 'current') { if (defined($self->{'file_counters'}->{$filename})) { return $self->{'elements_in_file_count'}->{$filename} - $self->{'file_counters'}->{$filename} +1; } } return undef; } sub is_format_expanded($$) { my ($self, $format) = @_; return $self->{'expanded_formats'}->{$format}; } sub _set_code_context($$) { my ($self, $code) = @_; push @{$self->{'document_context'}->[-1]->{'monospace'}}, $code; } sub _pop_code_context($) { my $self = shift; pop @{$self->{'document_context'}->[-1]->{'monospace'}}; } # the main data structure of the element target API is a hash reference, called # the target information. # The 'target' and 'filename' keys should be set for every type of element, # but the other keys will only be set on some elements. # # The following keys can be set: # # Strings # # 'target': A unique string representing the target. Used as argument to # 'id' attribute. # 'contents_target': A unique string representing the target to the location # of the element in the table of content. # 'shortcontents_target': A unique string representing the target to the # location of the element in the short table of contents # 'node_filename': the file name deriving from the element node name # 'section_filename': the file name deriving from the element section name # 'special_unit_filename': the file name of special elements # (separate contents, about...) # 'filename': the file name the element content is output to # 'text', 'text_nonumber': a textual representation of the element where # there is no restriction on the text formatting (ie HTML elements # can be used). # With _nonumber, no section number. # 'string', 'string_nonumber': a textual representation of the element with # restrictions on the available formatting, in practice no # HTML elements, only entities to be able to use in attributes. # With _nonumber, no section number. # # Other types # # 'tree', 'tree_nonumber: a Texinfo tree element which conversion should # correspond to the element name. # With _nonumber, no section number. # 'node_command': the node element associated with the target element. # 'root_element_command': the command associated to the top level element # associated with the target element. # # Some functions cache their results in these hashes. # $COMMAND should be a tree element which is a possible target of a link. # return the target information. sub _get_target($$) { my ($self, $command) = @_; if (!defined($command)) { cluck("_get_target command argument not defined"); } if (exists($self->{'targets'}->{$command})) { return $self->{'targets'}->{$command}; } return undef; } # API for links and elements directions formatting # This returns the id specific of the $COMMAND tree element sub command_id($$) { my ($self, $command) = @_; my $target = _get_target($self, $command); if (defined($target)) { return $target->{'target'}; } else { return undef; } } sub command_contents_target($$$) { my ($self, $command, $contents_or_shortcontents) = @_; $contents_or_shortcontents = 'shortcontents' if ($contents_or_shortcontents eq 'summarycontents'); my $target = _get_target($self, $command); if (defined($target)) { return $target->{$contents_or_shortcontents .'_target'}; } else { return undef; } } sub _get_footnote_location_target($$) { my ($self, $command) = @_; if (exists($self->{'special_targets'}) and exists($self->{'special_targets'}->{'footnote_location'}) and exists($self->{'special_targets'}->{'footnote_location'}->{$command})) { return $self->{'special_targets'}->{'footnote_location'}->{$command}; } return undef; } sub footnote_location_target($$) { my ($self, $command) = @_; my $footnote_location_special_target_info = _get_footnote_location_target($self, $command); if (defined($footnote_location_special_target_info)) { return $footnote_location_special_target_info->{'target'}; } return undef; } sub command_filename($$) { my ($self, $command) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (exists($target->{'filename'})) { return $target->{'filename'}; } # this finds a special element for footnote command if such an element # exists. This is best, the special element filename is the footnote # filename. my ($root_element, $root_command) = _html_get_tree_root_element($self, $command, 1); if (defined($root_element) and exists($root_element->{'unit_filename'})) { $target->{'filename'} = $root_element->{'unit_filename'}; return $root_element->{'unit_filename'}; } else { $target->{'filename'} = undef; } } return undef; } sub command_root_element_command($$) { my ($self, $command) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (not exists($target->{'root_element_command'})) { # in contrast with command_filename() we find the root element through # the location holding the @footnote command. It is better, as the # footnote special element is not associated with a root command, # it is better to stay in the document to find a root element. my ($root_element, $root_command) = _html_get_tree_root_element($self, $command); if (defined($root_element) and $root_element->{'unit_type'} eq 'unit') { $target->{'root_element_command'} = $root_element->{'unit_command'}; } else { $target->{'root_element_command'} = undef; } } return $target->{'root_element_command'}; } return undef; } sub command_node($$) { my ($self, $command) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (not exists($target->{'node_command'})) { # this finds a special element for footnote command if # such an element exists my ($root_element, $root_command) = _html_get_tree_root_element($self, $command, 1); if (defined($root_command) and exists($root_command->{'cmdname'})) { if ($root_command->{'cmdname'} eq 'node') { $target->{'node_command'} = $root_command; } elsif (exists($self->{'document'})) { my $sections_list = $self->{'document'}->sections_list(); my $section_relations = $sections_list->[$root_command->{'extra'}->{'section_number'} -1]; if (exists($section_relations->{'associated_node'})) { $target->{'node_command'} = $section_relations->{'associated_node'}->{'element'}; } } } else { $target->{'node_command'} = undef; } } return $target->{'node_command'}; } return undef; } # $SPECIFIED_TARGET can be used to specify explicitly the target sub _internal_command_href($$;$$) { my ($self, $command, $source_filename, $specified_target) = @_; $source_filename = $self->{'current_filename'} if (!defined($source_filename)); my $target; if (defined($specified_target)) { $target = $specified_target; } else { my $target_command = $command; # for sectioning command prefer the associated node. If there is no # associated node, use the associated_anchor_command. This order # is important for sectioning commands, it means that in the following # case the @chapter href will be given by the @node, even if there is # an @xrefname in-between. # # @node my node # @xrefname name for my node # # @chapter Chapter without directly associated node if (exists($self->{'document'}) and exists($command->{'extra'})) { if ($command->{'extra'}->{'section_number'}) { my $sections_list = $self->{'document'}->sections_list(); my $section_relations = $sections_list->[$command->{'extra'}->{'section_number'} -1]; if (exists($section_relations->{'associated_node'})) { $target_command = $section_relations->{'associated_node'}->{'element'}; } elsif (exists($section_relations->{'associated_anchor_command'})) { $target_command = $section_relations->{'associated_anchor_command'}->{'element'}; } } elsif ($command->{'extra'}->{'heading_number'}) { my $headings_list = $self->{'document'}->headings_list(); my $heading_relations = $headings_list->[$command->{'extra'}->{'heading_number'} -1]; if (exists($heading_relations->{'associated_anchor_command'})) { $target_command = $heading_relations->{'associated_anchor_command'}->{'element'}; } } } my $target_information = _get_target($self, $target_command); $target = $target_information->{'target'} if (defined($target_information)); } return undef if (!defined($target)); my $href = ''; my $target_filename = $self->command_filename($command); if (!defined($target_filename)) { # Happens if there are no pages, for example if OUTPUT is set to '' # as in the test cases. Also for things in @titlepage when # titlepage is not output. if (exists($self->{'document_units'}->[0]->{'unit_filename'})) { # In that case use the first page. $target_filename = $self->{'document_units'}->[0]->{'unit_filename'}; } } if (defined($target_filename)) { if (!defined($source_filename) or $source_filename ne $target_filename) { $href .= $self->url_protect_file_text($target_filename); # omit target if the command is an element command, there is only # one element in file and there is a file in the href my $command_root_element_command = $self->command_root_element_command($command); if (defined($source_filename) and defined($command_root_element_command)) { my $possible_empty_target = 0; if ($command_root_element_command eq $command) { $possible_empty_target = 1; } elsif (exists($command_root_element_command->{'cmdname'}) and $command_root_element_command->{'cmdname'} eq 'node' and exists($command_root_element_command->{'extra'}) and $command_root_element_command->{'extra'}->{'node_number'} and exists($self->{'document'})) { my $nodes_list = $self->{'document'}->nodes_list(); my $node_relations = $nodes_list->[$command_root_element_command ->{'extra'}->{'node_number'} -1]; if (exists($node_relations->{'associated_section'}) and $node_relations->{'associated_section'}->{'element'} eq $command) { $possible_empty_target = 1; } } if ($possible_empty_target) { my $count_elements_in_file = $self->count_elements_in_filename('total', $target_filename); if (defined($count_elements_in_file) and $count_elements_in_file == 1) { $target = ''; } } } } } $href .= '#' . $target if ($target ne ''); if ($href eq '') { return undef; } return $href; } # Return string for linking to $COMMAND with # $SOURCE_COMMAND is only used for messages # $SPECIFIED_TARGET can be set to specify explicitly the target sub command_href($$;$$$) { my ($self, $command, $source_filename, $source_command, $specified_target) = @_; if (exists($command->{'extra'}) and exists($command->{'extra'}->{'manual_content'})) { return _external_node_href($self, $command, $source_command); } return _internal_command_href($self, $command, $source_filename, $specified_target); } my %contents_command_special_unit_variety = ( 'contents' => 'contents', 'shortcontents' => 'shortcontents', 'summarycontents' => 'shortcontents', ); # Return string for linking to $CONTENTS_OR_SHORTCONTENTS associated # element from $COMMAND with sub command_contents_href($$$;$) { my ($self, $command, $contents_or_shortcontents, $source_filename) = @_; $source_filename = $self->{'current_filename'} if (not defined($source_filename)); my ($special_unit_variety, $special_unit, $class_base, $special_unit_direction) = $self->command_name_special_unit_information($contents_or_shortcontents); my $target = $self->command_contents_target($command, $contents_or_shortcontents); my $target_filename; # !defined happens when called as convert() and not output() if (defined($special_unit)) { my $command = $special_unit->{'unit_command'}; $target_filename = $self->command_filename($command); } my $href = ''; if (defined($target_filename) and (!defined($source_filename) or $source_filename ne $target_filename)) { $href .= $target_filename; } $href .= '#' . $target if ($target ne ''); if ($href eq '') { return undef; } return $href; } sub footnote_location_href($$;$$$) { my ($self, $command, $source_filename, $specified_target, $target_filename) = @_; $source_filename = $self->{'current_filename'} if (not defined($source_filename)); my $footnote_location_target_info = _get_footnote_location_target($self, $command); my $target = ''; if (defined($specified_target)) { $target = $specified_target; } elsif (defined($footnote_location_target_info)) { $target = $footnote_location_target_info->{'target'}; } # In the default footnote formatting functions, which calls # footnote_location_href, the target file is always known as the # footnote in the document appears before the footnote text formatting. # $target_filename is therefore always defined. It is a good thing # for the case of @footnote being formatted more than once (in multiple # @insertcopying for instance) as the file found just below may not be the # correct one in such a case. if (not defined($target_filename)) { if (defined($footnote_location_target_info) and defined($footnote_location_target_info->{'filename'})) { $target_filename = $footnote_location_target_info->{'filename'}; } else { # in contrast with command_filename() we find the location holding # the @footnote command, not the footnote element with footnotes my ($root_element, $root_command) = _html_get_tree_root_element($self, $command); if (defined($root_element)) { if (not defined($footnote_location_target_info)) { $self->{'special_targets'}->{'footnote_location'}->{$command} = {}; $footnote_location_target_info = $self->{'special_targets'}->{'footnote_location'}->{$command}; } $footnote_location_target_info->{'filename'} = $root_element->{'unit_filename'}; $target_filename = $footnote_location_target_info->{'filename'}; } } } my $href = ''; if (defined($target_filename) and (!defined($source_filename) or $source_filename ne $target_filename)) { $href .= $target_filename; } $href .= '#' . $target if ($target ne ''); return $href; } sub _internal_command_tree($$$) { my ($self, $command, $no_number) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (!exists($target->{'tree'})) { my $tree; my $in_code; if (exists($command->{'type'}) and $command->{'type'} eq 'special_unit_element') { my $special_unit_variety = $command->{'associated_unit'}->{'special_unit_variety'}; $tree = $self->_special_unit_info_tree('heading', $special_unit_variety); } elsif (exists($command->{'cmdname'}) and ($command->{'cmdname'} eq 'node' or $command->{'cmdname'} eq 'anchor' or $command->{'cmdname'} eq 'namedanchor')) { # to be a target, the node or anchor cannot be empty (nor expand to # spaces only), so argument is necessarily set. my $label_element; if ($command->{'cmdname'} eq 'anchor' or $command->{'cmdname'} eq 'namedanchor') { $label_element = $command->{'contents'}->[0]; } else { # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; $label_element = $arguments_line->{'contents'}->[0]; } $tree = $label_element; $in_code = 1; } elsif (exists($command->{'cmdname'}) and ($command->{'cmdname'} eq 'float')) { $tree = $self->float_type_number($command); } else { my $line_arg; if (exists($root_commands{$command->{'cmdname'}})) { # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; $line_arg = $arguments_line->{'contents'}->[0]; } else { # @heading* commands $line_arg = $command->{'contents'}->[0]; } if (exists($line_arg->{'contents'})) { my $section_number; $section_number = $command->{'extra'}->{'section_heading_number'} if (exists($command->{'extra'}) and defined($command->{'extra'}->{'section_heading_number'})); if ($section_number and ($self->get_conf('NUMBER_SECTIONS') or !defined($self->get_conf('NUMBER_SECTIONS')))) { # NOTE since there is a copy, the elements that are found using their # hash won't be found during conversion. This is the case for # target elements such as @anchor. my $substituted_strings = {'number' => Texinfo::TreeElement::new({'text' => $section_number}), 'section_title' => Texinfo::ManipulateTree::copy_treeNonXS($line_arg)}; if ($command->{'cmdname'} eq 'appendix' and $command->{'extra'}->{'section_level'} == 1) { $tree = $self->cdt('Appendix {number} {section_title}', $substituted_strings); } else { # TRANSLATORS: numbered section title $tree = $self->cdt('{number} {section_title}', $substituted_strings); } } else { $tree = $line_arg; } } $target->{'tree_nonumber'} = $line_arg; } $target->{'tree'} = $tree; if ($in_code) { $target->{'in_code'} = $in_code; } } return $target->{'tree_nonumber'}, $target->{'in_code'} if ($no_number and exists($target->{'tree_nonumber'})); return $target->{'tree'}, $target->{'in_code'}; } return undef, undef; } sub _push_referred_command_stack_command($$) { my ($self, $command) = @_; push @{$self->{'referred_command_stack'}}, $command; } sub _pop_referred_command_stack($) { my $self = shift; pop @{$self->{'referred_command_stack'}}; } sub _command_is_in_referred_command_stack($$) { my ($self, $command) = @_; return grep {$_ eq $command} @{$self->{'referred_command_stack'}}; } sub _convert_command_tree($$$$$$) { my ($self, $command, $type, $selected_tree, $in_code, $command_info) = @_; my $explanation; my $context_name; if (exists($command->{'cmdname'})) { my $cmdname = $command->{'cmdname'}; $context_name = $cmdname; $explanation = "$command_info:$type \@$cmdname"; } else { $context_name = $command->{'type'}; if ($command->{'type'} eq 'special_unit_element') { my $special_unit_variety = $command->{'associated_unit'}->{'special_unit_variety'}; $explanation = "$command_info $special_unit_variety"; } } my $context_type = 0; if ($type eq 'string' or $type eq 'string_nonumber') { $context_type |= $CTXF_string; } if ($in_code) { $context_type |= $CTXF_code; } _new_document_context($self, $context_name, $context_type, $explanation); _set_multiple_conversions($self, undef); _push_referred_command_stack_command($self, $command); my $result = _convert($self, $selected_tree, $explanation); _pop_referred_command_stack($self); _unset_multiple_conversions($self); _pop_document_context($self); return $result; } sub _internal_command_text($$$) { my ($self, $command, $type) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (exists($target->{$type})) { return $target->{$type}; } my ($command_tree, $in_code) = _internal_command_tree($self, $command, 0); return '' if (!defined($command_tree)); my $selected_tree; if ($type =~ /^(.*)_nonumber$/ and defined($target->{'tree_nonumber'})) { $selected_tree = $target->{'tree_nonumber'}; } else { $selected_tree = $command_tree; } $target->{$type} = _convert_command_tree($self, $command, $type, $selected_tree, $in_code, 'command_text'); return $target->{$type}; } # Can happen # * if USE_NODES is 0 and there are no sectioning commands. # * if a special element target was set to undef in user defined code. # * for @*ref with missing targets (maybe @novalidate needed in that case). # * for @node header if the node consist only in spaces (example in sectioning # in_menu_only_special_ascii_spaces_node). # * for multiple targets with the same name, eg both @node and @anchor # * with @inforef with node argument only, without manual argument. return undef; } # Return text to be used for $COMMAND. # $TYPE refers to the type of value returned from this function: # 'text' - return text # 'text_nonumber' - return text, without the section/chapter number # 'string' - return simpler text that can be used in element attributes # 'string_nonumber' - same as string, without the section/chapter number sub command_text($$;$) { my ($self, $command, $type) = @_; if (!defined($type)) { $type = 'text'; } if (!defined($command)) { cluck "in command_text($type) command not defined"; } if (exists($command->{'extra'}) and exists($command->{'extra'}->{'manual_content'})) { my $tree = Texinfo::TreeElement::new( {'contents' => [Texinfo::TreeElement::new({'text' => '('}), $command->{'extra'}->{'manual_content'}, Texinfo::TreeElement::new({'text' => ')'})]}); if (exists($command->{'extra'}->{'node_content'})) { push @{$tree->{'contents'}}, $command->{'extra'}->{'node_content'}; } my $context_str = "command_text $type "; if (exists($command->{'cmdname'})) { # this never happens, as the external node label tree # element is never directly an @-command. It can be an @-command # argument, in a menu, or a reconstituted tree. $context_str .= '@'.$command->{'cmdname'}; } elsif (exists($command->{'type'})) { $context_str .= $command->{'type'}; } my $context_type = $CTXF_code; if ($type eq 'string' or $type eq 'string_nonumber') { $context_type |= $CTXF_string; } # NOTE the multiple pass argument is not unicized, and no global # context argument is given because this external node manual label # should in general be converted only once. # In addition, regarding multiple pass, it is unlikely for # @-commands which should better be converted only once to be present. my $result = $self->convert_tree_new_formatting_context($tree, $context_str, $context_type, 'command_text-manual_content'); return $result; } return _internal_command_text($self, $command, $type); } # Return undef if there is no tree for the name sub _internal_command_name_tree($$$) { my ($self, $command, $no_number) = @_; my $target = _get_target($self, $command); if (defined($target)) { if (!exists($target->{'name_tree'})) { my $tree; if (exists($command->{'cmdname'}) and $command->{'cmdname'} eq 'namedanchor' and scalar(@{$command->{'contents'}}) > 1 and exists($command->{'contents'}->[1]->{'contents'})) { $tree = $command->{'contents'}->[1]; } $target->{'name_tree'} = $tree; } # currently not possible #return $target->{'name_tree_nonumber'} if ($no_number # and $target->{'name_tree_nonumber'}); return $target->{'name_tree'}, $target->{'in_code'}; } return undef, undef; } sub _internal_command_name($$$) { my ($self, $command, $type) = @_; my $name_type = "name_$type"; my $target = _get_target($self, $command); if (defined($target)) { if (exists($target->{$name_type})) { return $target->{$name_type}; } my ($command_name_tree, $in_code) = _internal_command_name_tree($self, $command, 0); if (!defined($command_name_tree)) { ($command_name_tree, $in_code) = _internal_command_tree($self, $command, 0); } return '' if (!defined($command_name_tree)); my $selected_tree; if ($type =~ /^(.*)_nonumber$/ and exists($target->{'name_tree_nonumber'})) { $selected_tree = $target->{'name_tree_nonumber'}; } else { $selected_tree = $command_name_tree; } $target->{$name_type} = _convert_command_tree($self, $command, $type, $selected_tree, $in_code, 'command_name'); return $target->{$name_type}; } return undef; } # Return text to be used for $COMMAND using the name rather than the # label/identifier, when the distinction exists. # $TYPE refers to the type of value returned from this function: # 'text' - return text # 'text_nonumber' - return text, without the section/chapter number # 'string' - return simpler text that can be used in element attributes sub command_name($$;$) { my ($self, $command, $type) = @_; if (!defined($type)) { $type = 'text'; } if (!defined($command)) { cluck "in command_name($type) command not defined"; } if (exists($command->{'extra'}) and exists($command->{'extra'}->{'manual_content'})) { return command_text($self, $command, $type); } return _internal_command_name($self, $command, $type); } # Return text to be used for $COMMAND description. # $TYPE refers to the type of value returned from this function: # 'text' - return text # 'string' - return simpler text that can be used in element attributes sub command_description($$;$) { my ($self, $command, $type) = @_; if (!defined($type)) { $type = 'text'; } if (!defined($command)) { cluck "in command_description($type) command not defined"; } if (exists($command->{'extra'}) and exists($command->{'extra'}->{'manual_content'})) { return undef; } my $target = _get_target($self, $command); if (defined($target)) { my $cached_type = 'description_'.${type}; if (exists($target->{$cached_type})) { return $target->{$cached_type}; } if ((exists($command->{'type'}) and $command->{'type'} eq 'special_unit_element') or (exists($command->{'cmdname'}) and ($command->{'cmdname'} eq 'anchor' or $command->{'cmdname'} eq 'namedanchor' or $command->{'cmdname'} eq 'float'))) { $target->{$cached_type} = undef; return undef; } my $node; if (exists($command->{'cmdname'})) { if ($command->{'cmdname'} eq 'node') { $node = $command; } elsif (exists($self->{'document'})) { my $sections_list = $self->{'document'}->sections_list(); my $section_relations = $sections_list->[$command->{'extra'}->{'section_number'} -1]; if (exists($section_relations->{'associated_node'})) { $node = $section_relations->{'associated_node'}->{'element'}; } } } if (!defined($node) or !exists($node->{'extra'}) or !$node->{'extra'}->{'node_number'} or !exists($self->{'document'})) { return undef; } my $nodes_list = $self->{'document'}->nodes_list(); my $node_relations = $nodes_list->[$node->{'extra'}->{'node_number'} -1]; my $node_description; my $long_description = 0; if (exists($node_relations->{'node_description'})) { $node_description = $node_relations->{'node_description'}; } elsif (exists($node_relations->{'node_long_description'})) { $node_description = $node_relations->{'node_long_description'}; $long_description = 1; } else { return undef; } my $formatted_nodedescription_nr = _formatted_nodedescription_nr($self, $node); my $cmdname = $command->{'cmdname'}; my $context_name = "$cmdname description"; my $explanation = "command_description:$type \@$cmdname"; my $description_element; if (!$long_description) { $description_element = $node_description->{'contents'}->[0]; } else { # nodedescriptionblock $description_element = Texinfo::TreeElement::new( {'contents' => $node_description->{'contents'}}); } my $multiple_formatted; if ($formatted_nodedescription_nr > 1) { $multiple_formatted = 'node-description-'.$formatted_nodedescription_nr; } my $context_type; if ($type eq 'string') { $context_type = $CTXF_string; } $target->{$cached_type} = $self->convert_tree_new_formatting_context($description_element, $context_name, $context_type, $multiple_formatted, $explanation); return $target->{$cached_type}; } return undef; } # Return the element in the tree that $LABEL refers to. sub label_command($$) { my ($self, $label) = @_; if (!defined($label)) { cluck; } my $identifiers_target; if (exists($self->{'document'})) { $identifiers_target = $self->{'document'}->labels_information(); if (defined($identifiers_target)) { return $identifiers_target->{$label}; } } return undef; } # Currently the only possibility for $TYPE is heading sub _special_unit_info_tree($$$) { my ($self, $type, $special_unit_variety) = @_; if (exists($self->{'translated_special_unit_info_texinfo'}->{$type})) { if (not exists($self->{'translated_special_unit_info_tree'}->{$type} ->{$special_unit_variety})) { my $special_unit_info_texinfo_string = $self->{'translated_special_unit_info_texinfo'}->{$type} ->{$special_unit_variety}; my $translated_tree; if (defined($special_unit_info_texinfo_string)) { # NOTE to be kept in sync with generated context in # generate_code_convert_data.pl my $translation_context = "$special_unit_variety section $type"; $translated_tree = $self->pcdt($translation_context, $special_unit_info_texinfo_string); } $self->{'translated_special_unit_info_tree'}->{$type} ->{$special_unit_variety} = $translated_tree; return $translated_tree; } else { return $self->{'translated_special_unit_info_tree'}->{$type} ->{$special_unit_variety}; } } return undef; } # Currently the only possibility for $TYPE is heading sub special_unit_info_text($$$;$) { my ($self, $type, $special_unit_variety, $context) = @_; my $tree = $self->_special_unit_info_tree($type, $special_unit_variety); return '' if (!defined($tree)); $context = 'normal' if (!defined($context)); my $explanation = "convert $special_unit_variety $type/$context"; if ($context eq 'string') { return $self->convert_tree_new_formatting_context($tree, $explanation, $CTXF_string); } else { return $self->convert_tree($tree, $explanation); } } sub command_name_special_unit_information($$) { my ($self, $cmdname) = @_; my $special_unit_variety; if (exists($contents_command_special_unit_variety{$cmdname})) { $special_unit_variety = $contents_command_special_unit_variety{$cmdname}; } elsif ($cmdname eq 'footnote') { $special_unit_variety = 'footnotes'; } else { return (undef, undef, undef, undef); } my $special_unit_direction = $self->special_unit_info('direction', $special_unit_variety); my $special_unit = $self->global_direction_unit($special_unit_direction); my $class_base = $self->special_unit_info('class', $special_unit_variety); return ($special_unit_variety, $special_unit, $class_base, $special_unit_direction); } sub global_direction_unit($$) { my ($self, $direction) = @_; return $self->{'global_units_directions'}->{$direction}; } sub global_direction_text($$) { my ($self, $direction) = @_; return $self->{'global_texts_directions'}->{$direction}; } sub get_element_root_command_element($$) { my ($self, $element) = @_; my ($output_unit, $root_command) = _html_get_tree_root_element($self, $element); if (defined($root_command)) { if ($self->get_conf('USE_NODES')) { if (exists($root_command->{'cmdname'})) { if ($root_command->{'cmdname'} eq 'node') { return ($output_unit, $root_command); } elsif (exists($self->{'document'})) { my $sections_list = $self->{'document'}->sections_list(); my $section_relations = $sections_list->[$root_command->{'extra'}->{'section_number'} -1]; if (exists($section_relations->{'associated_node'})) { return ($output_unit, $section_relations->{'associated_node'}->{'element'}); } } } } elsif (exists($root_command->{'cmdname'}) and $root_command->{'cmdname'} eq 'node') { if (exists($self->{'document'})) { my $nodes_list = $self->{'document'}->nodes_list(); my $node_relations = $nodes_list->[$root_command->{'extra'}->{'node_number'} -1]; if (exists($node_relations->{'associated_section'})) { return ($output_unit, $node_relations->{'associated_section'}->{'element'}); } } } } return ($output_unit, $root_command); } my %valid_direction_return_type = ( # a string that can be used in a href linking to the direction 'href' => 1, # a string representing the direction that can be used in # context where only entities are available (ie with HTML attributes) 'string' => 1, # a string representing the direction to be used in contexts # not restricted in term of available formatting (ie with HTML elements) 'text' => 1, # same as 'text', but select node in priority 'node' => 1, # same as 'text_nonumber' but select section in priority 'section' => 1 ); foreach my $no_number_type ('text', 'string', 'section') { # without section number $valid_direction_return_type{$no_number_type .'_nonumber'} = 1; } # sub from_element_direction($SELF, $DIRECTION, $TYPE, $SOURCE_UNIT, # $SOURCE_FILENAME, $SOURCE_FOR_MESSAGES) # # Return text used for linking from $SOURCE_UNIT in direction $DIRECTION. # The text returned depends on $TYPE. # # This is used both for output units and external nodes # # If $SOURCE_UNIT is undef, $self->current_output_unit() is used. # # $SOURCE_FOR_MESSAGES is an element used for messages formatting, to get a # location in input file. It is better to choose the node and not the # sectioning command associated with the element, as the error messages # are about external nodes not found. # # $self->current_output_unit() undef happens when there is no # output file. In the test suite, that call results only from # from_element_direction being called from _get_links, itself # called from 'format_begin_file' ultimately called from output # without output file. There could probably be other cases # with crafted/test code, but it should never happen when output is # called from the main program as there is always an output file. sub from_element_direction($$$;$$$) { my ($self, $direction, $type, $source_unit, $source_filename, # for messages only $source_command) = @_; my $target_unit; my $command; $source_unit = $self->current_output_unit() if (!defined($source_unit)); # NOTE $source_filename is only used for a command_href call. If with XS, # if source_filename remains undef, the command_href XS code will set the # source_filename to the current filename in XS. Therefore undef # current_filename in that case leads to the same output as set # current_filename. # We still set it correctly in case it becomes used in other codes. $source_filename = $self->current_filename() if (!defined($source_filename)); if (!exists($valid_direction_return_type{$type})) { print STDERR "Incorrect type $type in from_element_direction call\n"; return undef; } my $global_target_unit = $self->global_direction_unit($direction); if (defined($global_target_unit)) { $target_unit = $global_target_unit; # output TOP_NODE_UP related info even if $source_unit is not defined, # which should correspond to cases when there is no output file, mainly in # tests. } elsif ((not defined($source_unit) or ($source_unit and $self->unit_is_top_output_unit($source_unit))) and defined($self->get_conf('TOP_NODE_UP_URL')) and ($direction eq 'Up' or $direction eq 'NodeUp')) { if ($type eq 'href') { return $self->get_conf('TOP_NODE_UP_URL'); } elsif ($type eq 'text' or $type eq 'node' or $type eq 'string' or $type eq 'section' or $type eq 'section_nonumber' or $type eq 'string_nonumber') { return $self->get_conf('TOP_NODE_UP'); } else { cluck("BUG: type $type not available for TOP_NODE_UP\n"); return ''; } } elsif (not defined($target_unit) and defined($source_unit) and exists($source_unit->{'directions'}) and exists($source_unit->{'directions'}->{$direction})) { $target_unit = $source_unit->{'directions'}->{$direction}; } if (defined($target_unit)) { ######## debug if (!exists($target_unit->{'unit_type'})) { die "No unit type for element_target $direction $target_unit: " . Texinfo::Common::debug_print_output_unit($target_unit) . "directions :" . Texinfo::OutputUnits::print_output_unit_directions($source_unit); } ######## if ($target_unit->{'unit_type'} eq 'external_node_unit') { my $external_node_element = $target_unit->{'unit_command'}; #print STDERR "FROM_ELEMENT_DIRECTION ext node $type $direction\n" # if ($self->get_conf('DEBUG')); if ($type eq 'href') { return _external_node_href($self, $external_node_element, $source_command); } elsif ($type eq 'text' or $type eq 'node') { return $self->command_text($external_node_element); } elsif ($type eq 'string') { return $self->command_text($external_node_element, $type); } } elsif ($type eq 'node') { if (exists($target_unit->{'unit_node'})) { $command = $target_unit->{'unit_node'}->{'element'}; } $type = 'text'; } elsif ($type eq 'section' or $type eq 'section_nonumber') { if (exists($target_unit->{'unit_section'})) { $command = $target_unit->{'unit_section'}->{'element'}; } if ($type eq 'section_nonumber') { $type = 'text_nonumber'; } else { $type = 'text'; } } else { $command = $target_unit->{'unit_command'}; if ($type eq 'href') { if (defined($command)) { return $self->command_href($command, $source_filename); } else { return undef; } } } } else { return undef; } if (defined($command)) { #print STDERR "FROM_ELEMENT_DIRECTION $type $direction\n" # if ($self->get_conf('DEBUG')); return $self->command_text($command, $type); } # We end up here if there is a target element, but not of the expected # type. For example, if type is section but there is no section associated # to the target element node. return undef; } my %valid_direction_string_type = ( # accesskey associated to the direction 'accesskey' => 1, # direction button name 'button' => 1, # description of the direction 'description' => 1, # section number corresponding to the example in About text 'example' => 1, # rel/ref string associated to the direction 'rel' => 1, # few words text associated to the direction 'text' => 1, ); my %valid_direction_string_context = ( 'normal' => 1, 'string' => 1, ); my %direction_type_translation_context = ( 'button' => 'button label', 'description' => 'description', 'text' => 'string', ); sub direction_string($$$;$) { my ($self, $direction, $string_type, $context) = @_; if (!exists($valid_direction_string_type{$string_type})) { print STDERR "Incorrect type $string_type in direction_string call\n"; return undef; } $context = 'normal' if (!defined($context)); if (!exists($valid_direction_string_context{$context})) { print STDERR "Incorrect context $context in direction_string call\n"; return undef; } $direction =~ s/^FirstInFile//; my $translated_directions_strings = $self->{'translated_direction_strings'}; if (!defined($translated_directions_strings)) { cluck(); } if (not exists($self->{'directions_strings'}->{$string_type}->{$direction}) or not exists($self->{'directions_strings'}->{$string_type} ->{$direction}->{$context})) { $self->{'directions_strings'}->{$string_type}->{$direction} = {} if (not exists($self->{'directions_strings'} ->{$string_type}->{$direction})); if (exists($translated_directions_strings->{$string_type}) # can exist and be undef if user-defined and also maybe for # some default directions, but maybe only for unlikely type. and defined($translated_directions_strings->{$string_type} ->{$direction}) and defined($translated_directions_strings->{$string_type} ->{$direction}->{'converted'})) { # translate already converted direction strings my $converted_directions = $translated_directions_strings->{$string_type} ->{$direction}->{'converted'}; my $context_converted_string; if (exists($converted_directions->{$context})) { $context_converted_string = $converted_directions->{$context}; } elsif ($context eq 'string' and defined($converted_directions->{'normal'})) { $context_converted_string = $converted_directions->{'normal'}; } if (defined($context_converted_string)) { my $result_string = $self->cdt_string($context_converted_string); $self->{'directions_strings'}->{$string_type}->{$direction}->{$context} = $self->substitute_html_non_breaking_space($result_string); } else { $self->{'directions_strings'}->{$string_type}->{$direction}->{$context} = undef; } } elsif (exists($translated_directions_strings->{$string_type}) # can exist and be undef if user-defined and also maybe for # some default directions, but maybe only for unlikely type. and defined($translated_directions_strings->{$string_type} ->{$direction}) and defined($translated_directions_strings->{$string_type} ->{$direction}->{'to_convert'})) { # translate direction strings that need to be translated and converted my $translation_context = $direction; $translation_context .= ' (current section)' if ($direction eq 'This'); $translation_context .= ' direction ' .$direction_type_translation_context{$string_type}; my $translated_tree = $self->pcdt($translation_context, $translated_directions_strings->{$string_type} ->{$direction}->{'to_convert'}); my $context_str = "DIRECTION $direction ($string_type/$context)"; my $context_type; if ($context eq 'string') { $context_type = $CTXF_string; } my $result_string = $self->convert_tree_new_formatting_context($translated_tree, $context_str, $context_type, undef, $context_str); # NOTE direction strings should be simple Texinfo code, but it is # possible to set to anything through customization. Since # anything except simple code is incorrect, there is no guarantee # on the output, but it is good if there is no crash. # If there is a @documentlanguage in $converted_tree, translate_names # would be called and # $self->{'directions_strings'}->{$string_type}->{$direction} would be # reset. So, for this very special case (tested in the test suite), # there may be a need to set again even though it was already done # just above. $self->{'directions_strings'}->{$string_type}->{$direction} = {} if (not $self->{'directions_strings'}->{$string_type}->{$direction}); $self->{'directions_strings'}->{$string_type}->{$direction}->{$context} = $result_string; } else { $self->{'directions_strings'}->{$string_type}->{$direction}->{$context} = undef; } } return $self->{'directions_strings'}->{$string_type} ->{$direction}->{$context}; } sub get_special_unit_info_varieties($$) { my ($self, $type) = @_; return sort(keys(%{$self->{'special_unit_info'}->{$type}})); } sub special_unit_info($$$) { my ($self, $type, $special_unit_variety) = @_; return $self->{'special_unit_info'}->{$type}->{$special_unit_variety}; } # API for misc conversion and formatting functions # if $OUTPUT_UNITS is defined, the first output unit is used if a proper # top output unit is not found. sub _get_top_unit($;$) { my ($self, $output_units) = @_; my $identifiers_target; if (exists($self->{'document'})) { $identifiers_target = $self->{'document'}->labels_information(); } my $node_top; $node_top = $identifiers_target->{'Top'} if (defined($identifiers_target)); my $section_top; my $global_commands; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); } $section_top = $global_commands->{'top'} if (defined($global_commands)); if (defined($section_top)) { return $section_top->{'associated_unit'}; } elsif (defined($node_top)) { if (!exists($node_top->{'associated_unit'})) { die "No associated unit for node_top: " .Texinfo::Common::debug_print_element($node_top, 1); } return $node_top->{'associated_unit'}; } elsif (defined($output_units)) { return $output_units->[0]; } return undef; } # it is considered 'top' only if element corresponds to @top or # element is a node sub unit_is_top_output_unit($$) { my ($self, $output_unit) = @_; my $top_output_unit = _get_top_unit($self); if (defined($top_output_unit) and $top_output_unit eq $output_unit) { return 1; } else { return 0; } } my %default_formatting_references; sub default_formatting_function($$) { my ($self, $format) = @_; return $default_formatting_references{$format}; } sub formatting_function($$) { my ($self, $format) = @_; return $self->{'formatting_function'}->{$format}; } my %defaults_format_special_unit_body_contents; sub defaults_special_unit_body_formatting($$) { my ($self, $special_unit_variety) = @_; return $defaults_format_special_unit_body_contents{$special_unit_variety}; } sub special_unit_body_formatting($$) { my ($self, $special_unit_variety) = @_; return $self->{'special_unit_body'}->{$special_unit_variety}; } # Return the default for the function references used for # the formatting of commands, in case a user still wants to call # default @-commands formatting functions when replacing functions, # using code along # &{$self->default_command_conversion($cmdname)}($self, $cmdname, $command, args, $content) my %default_commands_conversion; sub default_command_conversion($$) { my ($self, $cmdname) = @_; return $default_commands_conversion{$cmdname}; } sub command_conversion($$) { my ($self, $cmdname) = @_; return $self->{'commands_conversion'}->{$cmdname}; } my %default_commands_open; sub default_command_open($$) { my ($self, $cmdname) = @_; return $default_commands_open{$cmdname}; } # used for customization only (in t2h_singular.init) sub get_value($$) { my ($self, $value) = @_; if (exists($self->{'document'}) and exists($self->{'document'}->{'values'}) and exists($self->{'document'}->{'values'}->{$value})) { return $self->{'document'}->{'values'}->{$value}; } else { return undef; } } my %default_shared_conversion_states = ( 'top' => {'in_skipped_node_top' => ['integer'],}, 'abbr' => {'explained_commands' => ['string', 'string']}, 'acronym' => {'explained_commands' => ['string', 'string']}, 'footnote' => {'footnote_number' => ['integer'], 'footnote_id_numbers' => ['string', 'integer']}, 'listoffloats' => {'formatted_listoffloats' => ['string', 'integer']}, 'menu' => {'html_menu_entry_index' => ['integer']}, 'printindex' => {'formatted_index_entries' => ['index_entry', 'integer']}, 'nodedescription' => {'formatted_nodedescriptions' => ['element', 'integer']}, # also used for titlepage 'quotation' => {'quotation_titlepage_stack' => ['integer'], 'elements_authors' => ['integer', 'integer', 'element'], 'element_authors_number' => ['integer', 'integer']}, ); sub define_shared_conversion_state($$$$) { my ($self, $cmdname, $state_name, $specification) = @_; if (not exists($self->{'shared_conversion_state'}->{$cmdname})) { $self->{'shared_conversion_state'}->{$cmdname} = {}; } if (not exists($self->{'shared_conversion_state'} ->{$cmdname}->{$state_name})) { $self->{'shared_conversion_state'}->{$cmdname}->{$state_name} = {}; } my $state = $self->{'shared_conversion_state'}->{$cmdname}->{$state_name}; if (exists($state->{'spec'})) { warn("BUG: redefining shared_conversion_state: $cmdname: $state_name"); } $state->{'spec'} = $specification; } sub _get_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; my $state = $self->{'shared_conversion_state'}->{$cmdname}->{$state_name}; if (!defined($state)) { #print STDERR "DEBUG: [". # join('|',keys(%{$self->{'shared_conversion_state'}->{$cmdname}}))."]\n"; confess("BUG: $self: undef shared_conversion_state: $cmdname: $state_name\n"); } my $spec_nr = scalar(@{$state->{'spec'}}); if ($spec_nr == 1) { return $state->{'values'}; } if (!defined($state->{'values'})) { $state->{'values'} = {}; } my $spec_idx = 1; my $current = $state->{'values'}; foreach my $arg (@args) { if (!defined($arg)) { return $current; } if ($spec_idx == $spec_nr - 1) { return $current->{$arg}; } if (!$current->{$arg}) { $current->{$arg} = {}; } $current = $current->{$arg}; $spec_idx++; } return $current; } sub _XS_get_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; return _get_shared_conversion_state($self, $cmdname, $state_name, @args); } sub get_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; if (exists($default_shared_conversion_states{$cmdname}) and exists($default_shared_conversion_states{$cmdname}->{$state_name})) { my $result = _XS_get_shared_conversion_state($self, $cmdname, $state_name, @args); return $result; } return _get_shared_conversion_state($self, $cmdname, $state_name, @args); } sub _set_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; my $state = $self->{'shared_conversion_state'}->{$cmdname}->{$state_name}; my $spec_nr = scalar(@{$state->{'spec'}}); if (scalar(@args) != $spec_nr) { return undef; } if ($spec_nr == 1) { if (!defined($args[0])) { return undef; } $state->{'values'} = $args[0]; return $args[0]; } if (!exists($state->{'values'})) { $state->{'values'} = {}; } my $spec_idx = 1; my $current = $state->{'values'}; foreach my $arg (@args) { if (!defined($arg)) { return undef; } if ($spec_idx == $spec_nr - 1) { $current->{$arg} = $args[$spec_idx]; return $current->{$arg}; } if (!exists($current->{$arg})) { $current->{$arg} = {}; } $current = $current->{$arg}; $spec_idx++; } } sub _XS_set_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; _set_shared_conversion_state($self, $cmdname, $state_name, @args); } # XS is only used for default conversion states. sub set_shared_conversion_state($$$;@) { my $self = shift; my $cmdname = shift; my $state_name = shift; my @args = @_; if (exists($default_shared_conversion_states{$cmdname}) and exists($default_shared_conversion_states{$cmdname}->{$state_name})) { _XS_set_shared_conversion_state($self, $cmdname, $state_name, @args); return; } _set_shared_conversion_state($self, $cmdname, $state_name, @args); } sub register_footnote($$$$$$$) { my ($self, $command, $footid, $docid, $number_in_doc, $footnote_location_filename, $multi_expanded_region) = @_; my $in_skipped_node_top = $self->get_shared_conversion_state('top', 'in_skipped_node_top'); if (!defined($in_skipped_node_top) or $in_skipped_node_top != 1) { push @{$self->{'pending_footnotes'}}, [$command, $footid, $docid, $number_in_doc, $footnote_location_filename, $multi_expanded_region]; } } # No equivalent function in C, the pending_footnotes are accessed directly sub get_pending_footnotes($) { my $self = shift; my @result = @{$self->{'pending_footnotes'}}; @{$self->{'pending_footnotes'}} = (); return \@result; } # API to register, cancel and get inline content that should be output # when in an inline situation, mostly in a paragraph or preformatted sub register_pending_formatted_inline_content($$$) { my ($self, $category, $inline_content) = @_; if (!defined($inline_content)) { return; } if (not exists($self->{'pending_inline_content'})) { $self->{'pending_inline_content'} = []; } push @{$self->{'pending_inline_content'}}, [$category, $inline_content]; } # cancel only the first pending content for the category sub cancel_pending_formatted_inline_content($$) { my ($self, $category) = @_; if (exists($self->{'pending_inline_content'})) { my $pending_inline = $self->{'pending_inline_content'}; my $current_idx = scalar(@$pending_inline) - 1; if ($current_idx >= 0) { while ($current_idx >= 0) { if ($pending_inline->[$current_idx]->[0] eq $category) { my $removed = splice(@$pending_inline, $current_idx, 1); return $removed->[1]; } $current_idx--; } } } return undef; } sub get_pending_formatted_inline_content($) { my $self = shift; if (not exists($self->{'pending_inline_content'})) { return ''; } else { my $result = ''; foreach my $category_inline_content (@{$self->{'pending_inline_content'}}) { if (defined($category_inline_content->[1])) { $result .= $category_inline_content->[1]; } } delete $self->{'pending_inline_content'}; return $result; } } # API to associate inline content to an element, typically # paragraph or preformatted. Allows to associate the pending # content to the first inline element. sub associate_pending_formatted_inline_content($$$) { my ($self, $element, $inline_content) = @_; $self->{'associated_inline_content'}->{$element} .= $inline_content; } sub get_associated_formatted_inline_content($$) { my ($self, $element) = @_; if (exists($self->{'associated_inline_content'}->{$element})) { my $result = $self->{'associated_inline_content'}->{$element}; delete $self->{'associated_inline_content'}->{$element}; return $result; } return ''; } # API to register an information to a file and get it. To be able to # set an integer information during conversion and get it back during headers # and footers conversion sub register_file_information($$$) { my ($self, $key, $value) = @_; if (!exists($self->{'current_filename'})) { cluck(); } $self->{'html_files_information'}->{$self->{'current_filename'}} = {} if (!exists( $self->{'html_files_information'}->{$self->{'current_filename'}})); $self->{'html_files_information'}->{$self->{'current_filename'}}->{$key} = $value; } sub get_file_information($$;$) { my ($self, $key, $filename) = @_; if (not defined($filename)) { $filename = $self->{'current_filename'}; } if (not defined($filename) or not exists($self->{'html_files_information'}) or not exists($self->{'html_files_information'}->{$filename}) or not exists($self->{'html_files_information'}->{$filename}->{$key})) { return (0, undef); } return (1, $self->{'html_files_information'}->{$filename}->{$key}) } sub current_filename($) { my $self = shift; return $self->{'current_filename'}; } sub current_output_unit($) { my $self = shift; return $self->{'current_output_unit'}; } # information from converter available 'read-only', set up before # really starting the formatting. my %available_converter_info; foreach my $converter_info ('copying_comment', 'destination_directory', 'document', 'document_name', 'documentdescription_string', 'expanded_formats', 'jslicenses', 'line_break_element', 'non_breaking_space', 'paragraph_symbol', 'simpletitle_command_name', 'simpletitle_tree', 'title_string', 'title_tree', 'title_titlepage') { $available_converter_info{$converter_info} = 1; } sub get_info($$) { my ($self, $converter_info) = @_; if (not exists($available_converter_info{$converter_info})) { confess("BUG: $converter_info not an available converter info"); } if (defined($self->{'converter_info'}->{$converter_info})) { return $self->{'converter_info'}->{$converter_info}; } return undef; } sub convert_tree_in_code_context($$;$) { my ($self, $tree, $explanation) = @_; _set_code_context($self, 1); my $result = $self->convert_tree($tree, $explanation); _pop_code_context($self); return $result; } # Call convert_tree out of the main conversion flow. sub convert_tree_new_formatting_context($$$;$$$$) { my ($self, $tree, $context_string, $context_type, $multiple_pass, $document_global_context, $block_command) = @_; _new_document_context($self, $context_string, $context_type, $document_global_context, $block_command); my $context_string_str = "C($context_string)"; my $multiple_pass_str = ''; if (defined($multiple_pass)) { _set_multiple_conversions($self, $multiple_pass); $multiple_pass_str = '|M'; } print STDERR "new_fmt_ctx ${context_string_str}${multiple_pass_str}\n" if ($self->get_conf('DEBUG')); my $result = $self->convert_tree($tree, "new_fmt_ctx ${context_string_str}"); if (defined($multiple_pass)) { _unset_multiple_conversions($self); } _pop_document_context($self); return $result; } # values for integer and string options in code generated from # Texinfo/Convert/converters_defaults.txt my $regular_defaults = Texinfo::Options::get_regular_options('html_converter'); my %defaults = ( # Customization option variables %{$regular_defaults}, # Non-string customization variables # _default_panel_button_dynamic_direction use nodes direction based on USE_NODE_DIRECTIONS # or USE_NODES if USE_NODE_DIRECTIONS is undefined 'SECTION_BUTTONS' => [[ 'Next', \&_default_panel_button_dynamic_direction ], [ 'Prev', \&_default_panel_button_dynamic_direction ], [ 'Up', \&_default_panel_button_dynamic_direction ], 'Space', 'Contents', 'Index', 'About'], 'SECTION_FOOTER_BUTTONS' => [[ 'Next', \&_default_panel_button_dynamic_direction_section_footer ], [ 'Prev', \&_default_panel_button_dynamic_direction_section_footer ], [ 'Up', \&_default_panel_button_dynamic_direction_section_footer ], 'Space', 'Contents', 'Index'], 'NODE_FOOTER_BUTTONS' => [[ 'Next', \&_default_panel_button_dynamic_direction_node_footer ], [ 'Prev', \&_default_panel_button_dynamic_direction_node_footer ], [ 'Up', \&_default_panel_button_dynamic_direction_node_footer ], 'Space', 'Contents', 'Index'], 'LINKS_DIRECTIONS' => ['Top', 'Index', 'Contents', 'About', 'NodeUp', 'NodeNext', 'NodePrev'], 'ACTIVE_ICONS' => undef, 'PASSIVE_ICONS' => undef, ); foreach my $buttons ('CHAPTER_BUTTONS', 'TOP_BUTTONS') { $defaults{$buttons} = [@{$defaults{'SECTION_BUTTONS'}}]; } $defaults{'MISC_BUTTONS'} = ['Top', 'Contents', 'Index', 'About']; foreach my $buttons ('CHAPTER_FOOTER_BUTTONS', 'TOP_FOOTER_BUTTONS') { $defaults{$buttons} = [@{$defaults{'SECTION_FOOTER_BUTTONS'}}]; } # class, direction, order, file_string, target my %default_special_unit_info = %{ Texinfo::HTMLData::get_default_special_unit_info() }; # heading my %default_translated_special_unit_info = %{ Texinfo::HTMLData::get_default_translated_special_unit_info() }; my $direction_orders = Texinfo::HTMLData::get_directions_order(); # 'global', 'relative', 'file' # include space direction my @global_directions_order = @{$direction_orders->[0]}; my @all_directions_except_special_units; foreach my $direction_order (@$direction_orders) { push @all_directions_except_special_units, @$direction_order; } #print STDERR join('|', @all_directions_except_special_units)."\n"; # for rel, see http://www.w3.org/TR/REC-html40/types.html#type-links my %default_converted_directions_strings = %{ Texinfo::HTMLData::get_default_converted_directions_strings() }; # translation contexts should be consistent with # %direction_type_translation_context. If the direction is not used # as is, it should also be taken into account in direction_string(). # For now 'This' becomes 'This (current section)'. my %default_translated_directions_strings = %{ Texinfo::HTMLData::get_default_translated_directions_strings() }; my @style_commands_contexts = ('normal', 'preformatted'); my @no_args_commands_contexts = ('normal', 'preformatted', 'string', 'css_string'); sub _translate_names($) { my $self = shift; Texinfo::Convert::Text::set_language($self->{'convert_text_options'}, $self->get_conf('documentlanguage')); Texinfo::Convert::Utils::switch_lang_translations($self, $self->get_conf('documentlanguage'), $self->get_conf('COMMAND_LINE_ENCODING')); if ($self->get_conf('DEBUG')) { my $output_encoding_name = $self->get_conf('OUTPUT_ENCODING_NAME'); $output_encoding_name = 'UNDEF' if (!defined($output_encoding_name)); my $documentlanguage = $self->get_conf('documentlanguage'); $documentlanguage = 'UNDEF' if (!defined($documentlanguage)); print STDERR "\nTRANSLATE_NAMES encoding_name: $output_encoding_name" ." documentlanguage: $documentlanguage\n"; } # reset strings such that they are translated when needed. # could also use the keys of $self->{'translated_direction_strings'} foreach my $string_type (keys(%default_translated_directions_strings)) { $self->{'directions_strings'}->{$string_type} = {}; } # could also use keys of $self->{'translated_special_unit_info_texinfo'} foreach my $type (keys(%default_translated_special_unit_info)) { $self->{'translated_special_unit_info_tree'}->{$type} = {}; } # delete the tree and formatted results for special elements # such that they are redone with the new tree when needed. foreach my $special_unit_variety ($self->get_special_unit_info_varieties('direction')) { my $special_unit_direction = $self->special_unit_info('direction', $special_unit_variety); my $special_unit = $self->global_direction_unit($special_unit_direction); if (defined($special_unit)) { my $command = $special_unit->{'unit_command'}; if (defined($command) and exists($self->{'targets'}->{$command})) { my $target = $self->{'targets'}->{$command}; foreach my $key ('text', 'string', 'tree', 'description_text', 'description_string') { delete $target->{$key}; } } } } my %translated_commands; foreach my $cmdname (keys(%{$self->{'no_arg_commands_formatting'}})) { my $conversion_contexts = $self->{'no_arg_commands_formatting'}->{$cmdname}; my $translated_tree; if (exists($conversion_contexts->{'translated_to_convert'})) { $translated_tree = $self->cdt($conversion_contexts->{'translated_to_convert'}); } else { # default translated commands $translated_tree = $self->translated_command_tree($cmdname); } if (defined($translated_tree)) { $conversion_contexts->{'translated_tree'} = $translated_tree; $translated_commands{$cmdname} = 1; } foreach my $context (@no_args_commands_contexts) { if (exists($conversion_contexts->{$context}->{'translated_converted'}) and not exists($conversion_contexts->{$context}->{'unset'})) { $translated_commands{$cmdname} = 1; $conversion_contexts->{$context}->{'text'} = $self->cdt_string($conversion_contexts->{$context}->{'translated_converted'}); } } } foreach my $cmdname (keys(%translated_commands)) { _complete_no_arg_commands_formatting($self, $cmdname, 1); } print STDERR "END TRANSLATE_NAMES\n\n" if ($self->get_conf('DEBUG')); } # redefined functions # # Texinfo::Translations::cache_translate_string redefined to call user defined function. sub html_cache_translate_string($$$;$) { my ($self, $string, $lang_translations, $translation_context) = @_; if (defined($self->{'formatting_function'}->{'format_translate_message'})) { my $lang = $lang_translations->[0]; my $translated_string = &{$self->{'formatting_function'}->{'format_translate_message'}}($self, $string, $lang, $translation_context); if (defined($translated_string)) { my $translations; $lang = '' if (!defined($lang)); if (!exists($self->{'translation_cache'}->{$lang})) { $self->{'translation_cache'}->{$lang} = {}; } $translations = $self->{'translation_cache'}->{$lang}; # reuse the tree if the translation matches the cached translation # otherwise setup a new translation (without tree). my $translation_context_str; if (defined($translation_context)) { $translation_context_str = $translation_context; } else { $translation_context_str = ''; } my $strings_cache = $translations->{$translation_context_str}; if ($strings_cache) { my $translated_string_tree = $strings_cache->{$string}; if (defined($translated_string_tree)) { if ($translated_string_tree->[0] eq $translated_string) { return $translated_string_tree; } # if the string has changed, the cache is invalidated by # resetting the cached string array reference just below. } } else { $strings_cache = {}; $translations->{$translation_context_str} = $strings_cache; } my $result = [$translated_string]; $strings_cache->{$string} = $result; return $result; } } return Texinfo::Translations::cache_translate_string($string, $lang_translations, $translation_context); } # redefine generic Converter functions to pass a customized # cache_translate_string function sub cdt($$;$$) { my ($self, $string, $replaced_substrings, $translation_context) = @_; return Texinfo::Translations::gdt($string, $self->{'current_lang_translations'}, $replaced_substrings, $self->get_conf('DEBUG'), $translation_context, $self, \&html_cache_translate_string); } sub cdt_string($$;$$) { my ($self, $string, $replaced_substrings, $translation_context) = @_; return Texinfo::Translations::gdt_string($string, $self->{'current_lang_translations'}, $replaced_substrings, $translation_context, $self, \&html_cache_translate_string); } sub converter_defaults($;$) { my ($self, $conf) = @_; if (defined($conf) and $conf->{'TEXI2HTML'}) { my $default_ref = { %defaults }; my $texi2html_defaults = { %$default_ref }; _set_variables_texi2html($texi2html_defaults); return $texi2html_defaults; } return \%defaults; } my %default_css_element_class_styles = %{ Texinfo::HTMLDataCSS::get_base_default_css_info() }; $default_css_element_class_styles{'pre.format-preformatted'} = $default_css_element_class_styles{'pre.display-preformatted'}; my %preformatted_commands_context = %preformatted_commands; $preformatted_commands_context{'verbatim'} = 1; my %pre_class_commands; foreach my $preformatted_command (keys(%preformatted_commands_context)) { # no class for the @small* variants if ($small_block_associated_command{$preformatted_command}) { $pre_class_commands{$preformatted_command} = $small_block_associated_command{$preformatted_command}; } else { $pre_class_commands{$preformatted_command} = $preformatted_command; } } $pre_class_commands{'menu'} = 'menu'; my %default_pre_class_types; $default_pre_class_types{'menu_comment'} = 'menu-comment'; my %indented_preformatted_commands; foreach my $indented_format ('example', 'display', 'lisp') { $indented_preformatted_commands{$indented_format} = 1; $indented_preformatted_commands{"small$indented_format"} = 1; # div.lisp is output as div.example if ($indented_format ne 'lisp') { $default_css_element_class_styles{"div.$indented_format"} = 'margin-left: 3.2em'; } } # types that are in code style in the default case. my %default_code_types = ( ); # specification of arguments formatting # to obtain the same order of conversion as in C, order for one argument # should be: normal, monospace, string, monospacestring, monospacetext, # filenametext, url, raw # Also used to be converted automatically to Texinfo code for documentation. our %html_default_commands_args = ( 'anchor' => [['monospacestring']], 'namedanchor' => [['monospacestring'], ['normal']], 'email' => [['url', 'monospacestring'], ['normal']], 'footnote' => [[]], 'printindex' => [[]], 'uref' => [['url', 'monospacestring'], ['normal'], ['normal']], 'url' => [['url', 'monospacestring'], ['normal'], ['normal']], 'sp' => [[]], 'inforef' => [['monospace'],['normal'],['filenametext']], 'xref' => [['monospace'],['normal'],['normal'],['filenametext'],['normal']], 'pxref' => [['monospace'],['normal'],['normal'],['filenametext'],['normal']], 'ref' => [['monospace'],['normal'],['normal'],['filenametext'],['normal']], 'link' => [['monospace'],['normal'],['filenametext']], 'image' => [['monospacestring', 'filenametext', 'url'],['filenametext'],['filenametext'],['normal','string'],['filenametext']], # FIXME shouldn't it better not to convert if later ignored? 'inlinefmt' => [['monospacetext'],['normal']], 'inlinefmtifelse' => [['monospacetext'],['normal'],['normal']], 'inlineraw' => [['monospacetext'],['raw']], 'inlineifclear' => [['monospacetext'],['normal']], 'inlineifset' => [['monospacetext'],['normal']], 'item' => [[]], 'itemx' => [[]], 'value' => [['monospacestring']], ); foreach my $explained_command (keys(%explained_commands)) { $html_default_commands_args{$explained_command} = [['normal'], ['normal', 'string']]; } foreach my $accent_command (keys(%accent_commands)) { $html_default_commands_args{$accent_command} = [[]]; } my %kept_line_commands; my @informative_global_commands = ('documentlanguage', 'footnotestyle', 'xrefautomaticsectiontitle', 'deftypefnnewline'); my @contents_commands = ('contents', 'shortcontents', 'summarycontents'); foreach my $line_command (@informative_global_commands, @contents_commands, keys(%formattable_line_commands), keys(%formatted_line_commands), keys(%default_index_commands)) { $kept_line_commands{$line_command} = 1; } foreach my $line_command (keys(%line_commands)) { $default_commands_conversion{$line_command} = undef unless (exists($kept_line_commands{$line_command})); } foreach my $nobrace_command (keys(%nobrace_commands)) { $default_commands_conversion{$nobrace_command} = undef unless (exists($formatted_nobrace_commands{$nobrace_command})); } # formatted/formattable @-commands that are not converted in # HTML in the default case. $default_commands_conversion{'page'} = undef; $default_commands_conversion{'need'} = undef; $default_commands_conversion{'vskip'} = undef; foreach my $ignored_brace_commands ('caption', 'errormsg', 'hyphenation', 'shortcaption', 'seealso', 'seeentry', 'sortas') { $default_commands_conversion{$ignored_brace_commands} = undef; } foreach my $ignored_block_commands ('ignore', 'macro', 'rmacro', 'linemacro', 'copying', 'documentdescription', 'documentinfo', 'titlepage', 'publication', 'direntry', 'nodedescriptionblock') { $default_commands_conversion{$ignored_block_commands} = undef; }; # Formatting of commands without args # The hash holding the defaults for the formatting of # most commands without args. It has three contexts as keys, # 'normal' in normal text, 'preformatted' in @example and similar # commands, and 'string' for contexts where HTML elements should not # be used. my %default_no_arg_commands_formatting = ( 'normal' => {}, 'preformatted' => {}, 'string' => {}, # more internal 'css_string' => {}, ); foreach my $cmdname (keys(%Texinfo::Convert::Converter::xml_text_entity_no_arg_commands_formatting)) { $default_no_arg_commands_formatting{'normal'}->{$cmdname} = {'text' => $Texinfo::Convert::Converter::xml_text_entity_no_arg_commands_formatting{ $cmdname}}; } $default_no_arg_commands_formatting{'normal'}->{' '} = {'text' => ' '}; $default_no_arg_commands_formatting{'normal'}->{"\t"} = {'text' => ' '}; $default_no_arg_commands_formatting{'normal'}->{"\n"} = {'text' => ' '}; # possible example of use, right now not used, as # the generic Converter customization is directly used through # the call to translated_command_tree(). #$default_no_arg_commands_formatting{'normal'}->{'error'}->{'translated_converted'} = 'error-->'; ## This is used to have gettext pick up the chain to be translated #if (0) { # my $not_existing; # $not_existing->cdt('error-->'); #} $default_no_arg_commands_formatting{'normal'}->{'enddots'} = {'element' => 'small', 'text' => '...'}; $default_no_arg_commands_formatting{'preformatted'}->{'dots'} = {'text' => '...'}; $default_no_arg_commands_formatting{'preformatted'}->{'enddots'} = {'text' => '...'}; $default_no_arg_commands_formatting{'normal'}->{'*'} = {'text' => '
'}; # this is used in math too, not sure that it is the best # in that context, '
' could be better. $default_no_arg_commands_formatting{'preformatted'}->{'*'} = {'text' => "\n"}; # escaped code points in CSS # https://www.w3.org/TR/css-syntax/#consume-escaped-code-point # Consume as many hex digits as possible, but no more than 5. Note that this means 1-6 hex digits have been consumed in total. If the next input code point is whitespace, consume it as well. Interpret the hex digits as a hexadecimal number. # Note that in style= HTML attributes entities are used to # protect CSS strings. For example, the CSS string a'b" # is protected as CSS as a\'b", and " is escaped in an HTML style # attribute: style="list-style-type: 'a\'b"'" # for the commands without a good representation in the other maps my %css_no_arg_commands = ( # we want to set explicitly '*' => '\A ', # do not set to force using only translations (as the command # is in the default converter translated commands) 'error' => undef, ); foreach my $cmdname (keys(%{$default_no_arg_commands_formatting{'normal'}})) { if (exists($css_no_arg_commands{$cmdname})) { $default_no_arg_commands_formatting{'css_string'}->{$cmdname} = {'text' => $css_no_arg_commands{$cmdname}} if (defined($css_no_arg_commands{$cmdname})); } elsif (exists($Texinfo::Convert::Unicode::unicode_map{$cmdname})) { my $char_nr = hex($Texinfo::Convert::Unicode::unicode_map{$cmdname}); my $css_string; if ($char_nr < 128) { # 7bit ascii $css_string = chr($char_nr); } else { $css_string = "\\$Texinfo::Convert::Unicode::unicode_map{$cmdname} "; } $default_no_arg_commands_formatting{'css_string'}->{$cmdname} = {'text' => $css_string}; } elsif (exists($nobrace_symbol_text{$cmdname})) { $default_no_arg_commands_formatting{'css_string'}->{$cmdname} = {'text' => $nobrace_symbol_text{$cmdname}}; } elsif (exists($Texinfo::CommandsValues::text_brace_no_arg_commands{$cmdname})) { # complete the commands not in unicode maps: TeX, enddots, LaTeX, tie $default_no_arg_commands_formatting{'css_string'}->{$cmdname} = {'text' => $Texinfo::CommandsValues::text_brace_no_arg_commands{$cmdname}}; } else { warn "BUG: $cmdname: no css_string\n"; } } # w not in css_string, set the corresponding default_css_element_class_styles # especially, which also has none and not w in the class $default_css_element_class_styles{'ul.mark-none'} = 'list-style-type: none'; # setup default_css_element_class_styles for mark commands based on css strings foreach my $mark_command (keys(%{$default_no_arg_commands_formatting{'css_string'}})) { if (exists($brace_commands{$mark_command})) { my $css_string; if ($mark_command eq 'bullet') { $css_string = 'disc'; } elsif (exists($default_no_arg_commands_formatting{'css_string'} ->{$mark_command}) and $default_no_arg_commands_formatting{'css_string'} ->{$mark_command}->{'text'}) { if (exists($special_list_mark_css_string_no_arg_command{$mark_command})) { $css_string = $special_list_mark_css_string_no_arg_command{$mark_command}; } else { $css_string = $default_no_arg_commands_formatting{'css_string'} ->{$mark_command}->{'text'}; } $css_string =~ s/^(\\[A-Z0-9]+) $/$1/; $css_string = '"'.$css_string.'"'; } if (defined($css_string)) { $default_css_element_class_styles{"ul.mark-$mark_command"} = "list-style-type: $css_string"; } } } # used to show the built-in CSS rules sub builtin_default_css_text() { my $css_text = ''; foreach my $css_rule (sort(keys(%default_css_element_class_styles))) { if ($default_css_element_class_styles{$css_rule} ne '') { $css_text .= "$css_rule {$default_css_element_class_styles{$css_rule}}\n"; } } return $css_text; } sub _text_element_conversion($$$) { my ($self, $specification, $cmdname) = @_; my $text = ''; # note that there could be HTML elements in text if (exists($specification->{'text'})) { $text = $specification->{'text'}; } if (exists($specification->{'element'})) { return $self->html_attribute_class($specification->{'element'}, [$cmdname]) .'>'. $text . '{'element'}.'>'; } else { return $text; } } sub _convert_no_arg_command($$$) { my ($self, $cmdname, $command) = @_; if (in_upper_case($self) and exists($letter_no_arg_commands{$cmdname}) and exists($letter_no_arg_commands{uc($cmdname)})) { $cmdname = uc($cmdname); } my $result; if (in_preformatted_context($self) or in_math($self)) { $result = _text_element_conversion($self, $self->{'no_arg_commands_formatting'}->{$cmdname}->{'preformatted'}, $cmdname); } elsif (in_string($self)) { $result = _text_element_conversion($self, $self->{'no_arg_commands_formatting'}->{$cmdname}->{'string'}, $cmdname); } else { $result = _text_element_conversion($self, $self->{'no_arg_commands_formatting'}->{$cmdname}->{'normal'}, $cmdname); } return $result; } foreach my $cmdname(keys(%{$default_no_arg_commands_formatting{'normal'}})) { $default_commands_conversion{$cmdname} = \&_convert_no_arg_command; } sub _css_string_convert_no_arg_command($$$) { my ($self, $cmdname, $command) = @_; if (in_upper_case($self) and exists($letter_no_arg_commands{$cmdname}) and exists($self->{'no_arg_commands_formatting'}->{uc($cmdname)})) { $cmdname = uc($cmdname); } #if (not defined($self->{'no_arg_commands_formatting'}->{$cmdname}->{'css_string'}->{$cmdname})) { # cluck ("BUG: CSS $cmdname no text"); #} return $self->{'no_arg_commands_formatting'}->{$cmdname}->{'css_string'} ->{'text'}; } foreach my $cmdname(keys(%{$default_no_arg_commands_formatting{'normal'}})) { $default_css_string_commands_conversion{$cmdname} = \&_css_string_convert_no_arg_command; } sub _convert_today_command($$$) { my ($self, $cmdname, $command) = @_; my $tree = $self->expand_today(); return $self->convert_tree($tree, 'convert today'); } $default_commands_conversion{'today'} = \&_convert_today_command; # style commands my %quoted_style_commands; foreach my $quoted_command ('samp') { $quoted_style_commands{$quoted_command} = 1; } my %default_upper_case_commands = ( 'sc' => 1 ); my %style_commands_element = %{ Texinfo::HTMLData::get_html_style_commands_element() }; my %default_style_commands_formatting; my %style_brace_types = map {$_ => 1} ('style_other', 'style_code', 'style_no_code'); # @all_style_commands is the union of style brace commands and commands # in %style_commands_element, a few not being style brace commands, and # commands in %quoted_style_commands. # Using keys of a map generated hash does like uniq, it avoids duplicates. # The first grep selects style brace commands, ie commands with %brace_commands # type in %style_brace_types. my @all_style_commands = keys %{{ map { $_ => 1 } ((grep {$style_brace_types{$brace_commands{$_}}} keys(%brace_commands)), keys(%style_commands_element), keys(%quoted_style_commands)) }}; # NOTE only normal and preformatted contexts are used. css strings # are formatted in string context, and in string context the argument # is returned as is. foreach my $cmdname (@all_style_commands) { # indicateurl is formatted with a specific function next if ($cmdname eq 'indicateurl'); $default_style_commands_formatting{$cmdname} = {}; # default is no element. foreach my $context (@style_commands_contexts) { $default_style_commands_formatting{$cmdname}->{$context} = {} } if (exists($style_commands_element{$cmdname})) { my $html_element = $style_commands_element{$cmdname}; foreach my $context (@style_commands_contexts) { $default_style_commands_formatting{$cmdname}->{$context} = {'element' => $html_element}; } } if (exists($quoted_style_commands{$cmdname})) { foreach my $context (@style_commands_contexts) { $default_style_commands_formatting{$cmdname}->{$context}->{'quote'} = 1; } } $default_commands_conversion{$cmdname} = \&_convert_style_command; } $default_style_commands_formatting{'sc'}->{'preformatted'}->{'element'} = 'span'; # currently unused, could be re-used if there is a need to have attributes # specified in %style_commands_element sub _parse_attribute($) { my $element = shift; return ('', '', '') if (!defined($element)); my ($class, $attributes) = ('', ''); if ($element =~ /^(\w+)(\s+.*)/) { $element = $1; $attributes = $2; if ($attributes =~ s/^\s+class=\"([^\"]+)\"//) { $class = $1; } } return ($element, $class, $attributes); } sub _convert_style_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $text; if (defined($args) and defined($args->[0])) { $text = $args->[0]->{'normal'}; } else { # happens with bogus @-commands without argument, like @strong something return ''; } if (in_string($self)) { return $text; } my $style_cmdname; # effect of kbdinputstyle if ($cmdname eq 'kbd' and exists($command->{'extra'}) and $command->{'extra'}->{'code'}) { $style_cmdname = 'code'; } else { $style_cmdname = $cmdname; } if (exists($self->{'style_commands_formatting'}->{$style_cmdname})) { my $style_formatting = $self->{'style_commands_formatting'}->{$style_cmdname}; my $formatting_spec; if (in_preformatted_context($self)) { $formatting_spec = $style_formatting->{'preformatted'}; } else { $formatting_spec = $style_formatting->{'normal'}; } if (defined($formatting_spec)) { if (exists($formatting_spec->{'element'})) { my @classes = ($style_cmdname); if ($style_cmdname ne $cmdname) { push @classes, "as-${style_cmdname}-${cmdname}"; } my $style = $formatting_spec->{'element'}; my $open = $self->html_attribute_class($style, \@classes); if ($open ne '') { $text = $open . '>' . $text . ""; } } if (exists($formatting_spec->{'quote'})) { $text = $self->get_conf('OPEN_QUOTE_SYMBOL') . $text . $self->get_conf('CLOSE_QUOTE_SYMBOL'); } } } return $text; } sub _convert_w_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $text; if (defined($args) and defined($args->[0])) { $text = $args->[0]->{'normal'}; } else { $text = ''; } if (in_string($self)) { return $text; } else { return $text . ''; } } $default_commands_conversion{'w'} = \&_convert_w_command; sub _convert_value_command($$$$) { my ($self, $cmdname, $command, $args) = @_; return $self->convert_tree($self->cdt('@{No value for `{value}\'@}', {'value' => Texinfo::TreeElement::new( {'text' => $args->[0]->{'monospacestring'}}) }), 'Tr missing value'); } $default_commands_conversion{'value'} = \&_convert_value_command; sub _convert_email_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $args_nr = 0; if (defined($args)) { $args_nr = scalar(@$args); } my $mail = ''; my $mail_string = ''; if ($args_nr > 0 and defined($args->[0])) { my $mail_arg = $args->[0]; $mail = $mail_arg->{'url'}; $mail_string = $mail_arg->{'monospacestring'}; } my $text = ''; if ($args_nr > 1 and defined($args->[1]) and defined($args->[1]->{'normal'})) { my $text_arg = $args->[1]; $text = $text_arg->{'normal'}; } $text = $mail_string unless ($text ne ''); # match a non-space character. Both ascii and non-ascii spaces are # considered as spaces. When perl 5.18 is the oldest version # supported, it could become [^\s] return $text unless ($mail =~ /[^\v\h\s]/); if (in_string($self)) { return "$mail_string ($text)"; } else { return $self->html_attribute_class('a', [$cmdname]) .' href="'.$self->url_protect_url_text("mailto:$mail")."\">$text
"; } } $default_commands_conversion{'email'} = \&_convert_email_command; sub _convert_explained_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $explanation_result; my $explanation_string; my $normalized_type = ''; if (exists($command->{'contents'}) and exists($command->{'contents'}->[0]->{'contents'})) { $normalized_type = Texinfo::Convert::NodeNameNormalization::convert_to_identifier( $command->{'contents'}->[0]); } if (defined($args) and defined($args->[1]) and defined($args->[1]->{'string'}) and $args->[1]->{'string'} =~ /\S/) { $explanation_string = $args->[1]->{'string'}; $self->set_shared_conversion_state($cmdname, 'explained_commands', $normalized_type, $explanation_string); } else { $explanation_string = $self->get_shared_conversion_state($cmdname, 'explained_commands', $normalized_type); } my $result = ''; if (defined($args) and defined($args->[0])) { $result = $args->[0]->{'normal'}; } if (!in_string($self)) { my $explanation = ''; $explanation = " title=\"$explanation_string\"" if (defined($explanation_string)); my $html_element = 'abbr'; $result = $self->html_attribute_class($html_element, [$cmdname]) ."${explanation}>".$result.""; } if (defined($args) and defined($args->[1]) and defined($args->[1]->{'normal'})) { my $explanation_result = $args->[1]->{'normal'}; # TRANSLATORS: abbreviation or acronym explanation $result = $self->convert_tree($self->cdt('{explained_string} ({explanation})', {'explained_string' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $result}), 'explanation' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $explanation_result})}), "convert explained $cmdname"); } return $result; } foreach my $explained_command (keys(%explained_commands)) { $default_commands_conversion{$explained_command} = \&_convert_explained_command; } sub _convert_anchor_command($$$$) { my ($self, $cmdname, $command, $args) = @_; # in_multiple_conversions returning true while multi_expanded_region returns # undef happens when expanding target and heading commands text. Therefore # if in_multiple_conversions was used, the output could be different. # Note, however, that an anchor on a sectioning command line is never output # because the sectioning command line is copied to be combined with # the sectioning number through a translation and the copy is not registered # as a target element and command_id will return undef. # An anchor on a @node line can be output (it is invalid Texinfo) and it # would be different for that case if in_multiple_conversions was used. if (!multi_expanded_region($self) and !in_string($self)) { my $id = $self->command_id($command); if (defined($id) and $id ne '') { return &{$self->formatting_function('format_separate_anchor')}($self, $id, $cmdname); } } return ''; } $default_commands_conversion{'anchor'} = \&_convert_anchor_command; $default_commands_conversion{'namedanchor'} = \&_convert_anchor_command; sub _convert_footnote_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $foot_num = $self->get_shared_conversion_state('footnote', 'footnote_number'); if (!defined($foot_num)) { $foot_num = 0; } $foot_num++; $self->set_shared_conversion_state('footnote', 'footnote_number', $foot_num); my $number_in_doc = $foot_num; my $footnote_mark; if ($self->get_conf('NUMBER_FOOTNOTES')) { $footnote_mark = $number_in_doc; } else { $footnote_mark = $self->get_conf('NO_NUMBER_FOOTNOTE_SYMBOL'); $footnote_mark = '' if (!defined($footnote_mark)); } return "($footnote_mark)" if (in_string($self)); #print STDERR "FOOTNOTE $command\n"; my $footnote_id = $self->command_id($command); # happens for bogus footnotes if (!defined($footnote_id)) { return ''; } # ID for linking back to the main text from the footnote. my $footnote_docid = $self->footnote_location_target($command); # id used in output my $footid; my $docid; my $multiple_expanded_footnote = 0; my $multi_expanded_region = multi_expanded_region($self); if (defined($multi_expanded_region)) { # to avoid duplicate names, use a prefix that cannot happen in anchors my $target_prefix = "t_f"; $footid = $target_prefix.$multi_expanded_region.'_' .$footnote_id.'_'.$foot_num; $docid = $target_prefix.$multi_expanded_region.'_' .$footnote_docid.'_'.$foot_num; } else { my $footnote_id_number = $self->get_shared_conversion_state('footnote', 'footnote_id_numbers', $footnote_id); if (!defined($footnote_id_number)) { $self->set_shared_conversion_state('footnote', 'footnote_id_numbers', $footnote_id, $foot_num); $footid = $footnote_id; $docid = $footnote_docid; } else { # This should rarely happen, except for @footnote in @copying and # multiple @insertcopying... # Here it is not checked that there is no clash with another anchor. # However, unless there are more than 1000 footnotes this should not # happen at all, and even in that case it is very unlikely. $footid = $footnote_id.'_'.$foot_num; $docid = $footnote_docid.'_'.$foot_num; $multiple_expanded_footnote = 1; } } my $footnote_href; my $footnotestyle = $self->get_conf('footnotestyle'); if ((!defined($footnotestyle) or $footnotestyle ne 'separate') and (defined($multi_expanded_region) or $multiple_expanded_footnote)) { # if the footnote appears multiple times, command_href() will select # one, but it may not be the one expanded at the location currently # formatted (in general the first one, but it depends if it is in a # tree element or not, for instance in @titlepage). # With footnotestyle end, considering that the footnote is in the same file # has a better chance of being correct. $footnote_href = "#$footid"; } else { $footnote_href = $self->command_href($command, undef, undef, $footid); } $self->register_footnote($command, $footid, $docid, $number_in_doc, $self->current_filename(), $multi_expanded_region); my $footnote_number_text; if (in_preformatted_context($self)) { $footnote_number_text = "($footnote_mark)"; } else { $footnote_number_text = "$footnote_mark"; } return $self->html_attribute_class('a', [$cmdname]) ." id=\"$docid\" href=\"$footnote_href\">$footnote_number_text"; } $default_commands_conversion{'footnote'} = \&_convert_footnote_command; sub _convert_uref_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $args_nr = 0; if (defined($args)) { $args_nr = scalar(@$args); } my $text = ''; my $url = ''; my $url_string = ''; my $replacement = ''; if ($args_nr > 0 and defined($args->[0])) { my $url_arg = $args->[0]; $url = $url_arg->{'url'}; $url_string = $url_arg->{'monospacestring'}; } if ($args_nr > 1 and defined($args->[1])) { my $text_arg = $args->[1]; $text = $text_arg->{'normal'}; } if ($args_nr > 2 and defined($args->[2])) { my $replacement_arg = $args->[2]; $replacement = $replacement_arg->{'normal'}; } $text = $replacement if ($replacement ne ''); $text = $url_string if ($text eq ''); return $text if ($url eq ''); return "$text ($url_string)" if (in_string($self)); return $self->html_attribute_class('a', [$cmdname]) .' href="'.$self->url_protect_url_text($url)."\">$text"; } $default_commands_conversion{'uref'} = \&_convert_uref_command; $default_commands_conversion{'url'} = \&_convert_uref_command; sub _convert_image_command($$$$) { my ($self, $cmdname, $command, $args) = @_; # NOTE the choice of filenametext or url is somewhat arbitrary here. # url is formatted considering that it would be output as UTF-8 to fit # with percent encoding, filenametext is formatted according to the the # output encoding. It matter mostly for accent @-commands, @U and symbols # no args @-commands not in the ASCII range. # As a file name, filenametext could make sense, although a path # with all the characters encoded, which happens if UTF-8 is considered # as the output encoding may also make sense. Note that it is # also used as the path part of a percent encoded url. # In practice, the user should check that the output encoding # and the commands used in file names match, so url or # filenametext should lead to the same path. if (defined($args) and defined($args->[0]) and defined($args->[0]->{'filenametext'}) and $args->[0]->{'filenametext'} ne '') { my $image_basefile = $args->[0]->{'filenametext'}; my $basefile_string = ''; $basefile_string = $args->[0]->{'monospacestring'} if (defined($args->[0]->{'monospacestring'})); return $basefile_string if (in_string($self)); my ($image_file, $image_extension, $image_path) = $self->html_image_file_location_name($cmdname, $command, $image_basefile, $args); if (not defined($image_path)) { # it would have been relevant to output the message only if # if not ($self->in_multiple_conversions()) # However, @image formatted multiple times should be # rare out of test suites (and probably always incorrect), so we avoid # complexity and slowdown. We still check that source_info is set, if # not it should be a copy, therefore there is no need for error # output, especially without line information. if (exists($command->{'source_info'})) { $self->converter_line_warn(sprintf( __("\@image file `%s' (for HTML) not found, using `%s'"), $image_basefile, $image_file), $command->{'source_info'}); } } if (defined($self->get_conf('IMAGE_LINK_PREFIX'))) { $image_file = $self->get_conf('IMAGE_LINK_PREFIX') . $image_file; } my $alt_string; if (defined($args->[3]) and defined($args->[3]->{'string'}) and $args->[3]->{'string'} ne '') { $alt_string = $args->[3]->{'string'}; } else { $alt_string = $basefile_string; } return $self->close_html_lone_element( $self->html_attribute_class('img', [$cmdname]) . ' src="'.$self->url_protect_file_text($image_file) ."\" alt=\"$alt_string\""); } return ''; } $default_commands_conversion{'image'} = \&_convert_image_command; sub _convert_math_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $arg; if (defined($args) and defined($args->[0])) { $arg = $args->[0]->{'normal'}; } else { return ''; } my $math_type = $self->get_conf('HTML_MATH'); if (defined($math_type) and $math_type eq 'mathjax') { $self->register_file_information('mathjax', 1); return $self->html_attribute_class('em', [$cmdname, 'tex2jax_process']) .">\\($arg\\)"; } return $self->html_attribute_class('em', [$cmdname]).">$arg"; } $default_commands_conversion{'math'} = \&_convert_math_command; sub _accent_entities_html_accent($$$;$$$$) { my ($self, $text, $command, $index_in_stack, $accents_stack, $in_upper_case, $use_numeric_entities) = @_; my $accent = $command->{'cmdname'}; if ($in_upper_case and $text =~ /^\w$/) { $text = uc($text); } # do not return a dotless i or j as such if it is further composed # with an accented letter, return the letter as is if ($accent eq 'dotless') { if (exists($Texinfo::UnicodeData::unicode_accented_letters{$accent}) and exists($Texinfo::UnicodeData::unicode_accented_letters{ $accent}->{$text}) and ($index_in_stack > 0 and $Texinfo::UnicodeData::unicode_accented_letters{ $accents_stack->[$index_in_stack-1]->{'cmdname'} })) { return $text; } } if ($use_numeric_entities) { my $formatted_accent = Texinfo::Convert::Converter::xml_numeric_entity_accent($accent, $text); if (defined($formatted_accent)) { return $formatted_accent; } } else { my ($accent_command_entity, $accent_command_text_with_entities); if ($self->{'accent_entities'}->{$accent}) { ($accent_command_entity, $accent_command_text_with_entities) = @{$self->{'accent_entities'}->{$accent}}; } return "&${text}$accent_command_entity;" if ($accent_command_entity and defined($accent_command_text_with_entities) # \z ensures that a \n at the end prevents matching, we do not # want an end of line in the middle of the entity and ($text =~ /^[$accent_command_text_with_entities]\z/)); my $formatted_accent = Texinfo::Convert::Converter::xml_numeric_entity_accent($accent, $text); if (defined($formatted_accent)) { return $formatted_accent; } } # should only be the case of @dotless, as other commands have a diacritic # associated, and only if the argument is not i nor j. return $text; } sub _accent_entities_numeric_entities_accent($$$;$$$) { my ($self, $text, $command, $index_in_stack, $accents_stack, $in_upper_case) = @_; return _accent_entities_html_accent($self, $text, $command, $index_in_stack, $accents_stack, $in_upper_case, 1); } sub _convert_accent_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $format_accents; if ($self->get_conf('USE_NUMERIC_ENTITY')) { $format_accents = \&_accent_entities_numeric_entities_accent; } else { $format_accents = \&_accent_entities_html_accent; } return $self->convert_accents($command, $format_accents, $self->get_conf('OUTPUT_CHARACTERS'), in_upper_case($self)); } foreach my $cmdname (keys(%accent_commands)) { $default_commands_conversion{$cmdname} = \&_convert_accent_command; } sub _css_string_accent($$$;$$$) { my ($self, $text, $command, $index_in_stack, $accents_stack, $in_upper_case) = @_; my $accent = $command->{'cmdname'}; if ($in_upper_case and $text =~ /^\p{Word}$/) { $text = uc($text); } if (exists($Texinfo::UnicodeData::unicode_accented_letters{$accent}) and exists($Texinfo::UnicodeData::unicode_accented_letters{ $accent}->{$text})) { return '\\' . $Texinfo::UnicodeData::unicode_accented_letters{$accent}->{$text}. ' '; } if (exists($Texinfo::CommandsValues::unicode_diacritics{$accent})) { my $diacritic = '\\' .$Texinfo::CommandsValues::unicode_diacritics{$accent}. ' '; if ($accent ne 'tieaccent') { return $text . $diacritic; } else { # tieaccent diacritic is naturally and correctly composed # between two characters my $remaining_text = $text; # we consider that letters are either characters or escaped characters if ($remaining_text =~ s/^([\p{L}\d]|\\[a-zA-Z0-9]+ )([\p{L}\d]|\\[a-zA-Z0-9]+ )(.*)$/$3/) { return $1.$diacritic.$2 . $remaining_text; } else { return $text . $diacritic; } } } # There are diacritics for every accent command except for dotless. # We should only get there with dotless if the argument is not recognized. return $text; } sub _css_string_convert_accent_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $format_accents = \&_css_string_accent; return $self->convert_accents($command, $format_accents, $self->get_conf('OUTPUT_CHARACTERS'), in_upper_case($self)); } foreach my $cmdname (keys(%accent_commands)) { $default_css_string_commands_conversion{$cmdname} = \&_css_string_convert_accent_command; } # argument is formatted as code since indicateurl is in brace_code_commands sub _convert_indicateurl_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $text; if (defined($args) and defined($args->[0])) { $text = $args->[0]->{'normal'}; } else { return ''; } if (!defined($text)) { # happens with bogus @-commands without argument, like @strong something return ''; } if (!in_string($self)) { return $self->get_conf('OPEN_QUOTE_SYMBOL'). $self->html_attribute_class('code', [$cmdname]).'>'.$text .''.$self->get_conf('CLOSE_QUOTE_SYMBOL'); } else { return $self->get_conf('OPEN_QUOTE_SYMBOL').$text. $self->get_conf('CLOSE_QUOTE_SYMBOL'); } } $default_commands_conversion{'indicateurl'} = \&_convert_indicateurl_command; sub _convert_titlefont_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $text; if (defined($args) and defined($args->[0])) { $text = $args->[0]->{'normal'}; } else { # happens with empty command return ''; } return &{$self->formatting_function('format_heading_text')}($self, $cmdname, [$cmdname], $text, 0); } $default_commands_conversion{'titlefont'} = \&_convert_titlefont_command; sub _convert_U_command($$$$) { my ($self, $cmdname, $command, $args) = @_; if (defined($args) and defined($args->[0])) { my $arg_text = $args->[0]->{'normal'}; if ($arg_text ne '') { # checks on the value already done in Parser, just output it here. return "&#x$arg_text;"; } } return ''; } $default_commands_conversion{'U'} = \&_convert_U_command; sub _default_format_comment($$) { my ($self, $text) = @_; return $self->xml_comment(' '.$text); } # Note: has an XS override sub _default_format_protect_text { my ($self, $text) = @_; my $result = $self->xml_protect_text($text); $result =~ s/\f/ /g; return $result; } sub _default_css_string_format_protect_text($$) { my ($self, $text) = @_; $text =~ s/\\/\\\\/g; $text =~ s/\'/\\'/g; return $text; } # can be called on root commands, tree units, special elements # and title elements. $cmdname can be undef for special elements. sub _default_format_heading_text($$$$$;$$$) { my ($self, $cmdname, $classes, $text, $level, $id, $element, $target) = @_; return '' if ($text !~ /\S/ and not defined($id)); # This happens with titlefont in title for instance if (in_string($self)) { $text .= "\n" unless (defined($cmdname) and $cmdname eq 'titlefont'); return $text; } if ($level < 1) { $level = 1; } elsif ($level > $self->get_conf('MAX_HEADER_LEVEL')) { $level = $self->get_conf('MAX_HEADER_LEVEL'); } my $result = $self->html_attribute_class("h$level", $classes); if (defined($id)) { $result .= " id=\"$id\""; # The ID of this heading is likely the point the user would prefer being # linked to over the $target, since that's where they would be seeing a # copiable anchor. $target = $id; } $result .= '>'; my $anchor = _get_copiable_anchor($self, $target); if (defined($anchor)) { $result .= ''; } $result .= $text; if (defined($anchor)) { $result .= "$anchor"; } $result .= ""; # titlefont appears inline in text, so no end of line is # added. The end of line should be added by the user if needed. $result .= "\n" unless (defined($cmdname) and $cmdname eq 'titlefont'); $result .= $self->get_conf('DEFAULT_RULE') . "\n" if (defined($cmdname) and $cmdname eq 'part' and defined($self->get_conf('DEFAULT_RULE')) and $self->get_conf('DEFAULT_RULE') ne ''); return $result; } sub _default_format_separate_anchor($$;$) { my ($self, $id, $class) = @_; # html_attribute_class would not work with span, so if span is # used, html_attribute_class should not be used return $self->html_attribute_class('a', [$class])." id=\"$id\">"; } # Associated to a button. Return text to use for a link in button bar. # Depending on USE_NODE_DIRECTIONS and xrefautomaticsectiontitle # use section or node for link direction and string. sub _default_panel_button_dynamic_direction($$;$$$) { my ($self, $direction, $source_command, $omit_rel, $use_first_element_in_file_directions) = @_; my $result; if ((defined($self->get_conf('USE_NODE_DIRECTIONS')) and $self->get_conf('USE_NODE_DIRECTIONS')) or (not defined($self->get_conf('USE_NODE_DIRECTIONS')) and $self->get_conf('USE_NODES'))) { $direction = 'Node'.$direction; } if ($use_first_element_in_file_directions) { $direction = 'FirstInFile'.$direction; } my $href = $self->from_element_direction($direction, 'href', undef, undef, $source_command); my $node; my $xrefautomaticsectiontitle = $self->get_conf('xrefautomaticsectiontitle'); if (defined($xrefautomaticsectiontitle) and $xrefautomaticsectiontitle eq 'on') { $node = $self->from_element_direction($direction, 'section_nonumber'); } if (!defined($node)) { $node = $self->from_element_direction($direction, 'node'); } if (defined($node) and $node =~ /\S/) { my $hyperlink; if (defined($href) and $href ne '') { my $hyperlink_attributes = $omit_rel ? '' : _direction_href_attributes($self, $direction); $hyperlink = "$node"; } else { $hyperlink = $node; } # i18n my $direction_text = $self->direction_string($direction, 'text'); $direction_text = '' if (!defined($direction_text)); $result = ''; my $open = $self->html_attribute_class('span', ['nav-label']); if ($open ne '') { $result .= $open.'>'; } $result .= $direction_text.': '; if ($open ne '') { $result .= ''; } $open = $self->html_attribute_class('span', ['nav-link']); if ($open ne '') { $result .= $open.'>'; } $result .= $hyperlink; if ($open ne '') { $result .= ''; } } # 1 to communicate that a delimiter is needed for that button return ($result, 1); } # Used for button bar at the foot of a node, with "rel" and "accesskey" # attributes omitted. sub _default_panel_button_dynamic_direction_node_footer($$$) { my ($self, $direction, $source_command) = @_; return _default_panel_button_dynamic_direction($self, $direction, $source_command, 1); } # used for button bar at the foot of a section or chapter with # directions of first element in file used instead of the last # element directions. sub _default_panel_button_dynamic_direction_section_footer($$$) { my ($self, $direction, $source_command) = @_; return _default_panel_button_dynamic_direction($self, $direction, $source_command, undef, 1); } # Only used if ICONS is set and the button is active. sub _default_format_button_icon_img($$$;$) { my ($self, $button, $icon, $name) = @_; return '' if (!defined($icon)); $button = '' if (!defined ($button)); $name = '' if (!defined($name)); my $alt = ''; if ($name ne '') { if ($button ne '') { $alt = "$button: $name"; } else { $alt = $name; } } else { $alt = $button; } my $img = $self->html_attribute_class('img', ['nav-icon']); return $self->close_html_lone_element( "$img src=\"".$self->url_protect_url_text($icon)."\" alt=\"$alt\""); } sub _direction_href_attributes($$) { my ($self, $direction) = @_; my $href_attributes = ''; if ($self->get_conf('USE_ACCESSKEY')) { my $accesskey = $self->direction_string($direction, 'accesskey', 'string'); if (defined($accesskey) and ($accesskey ne '')) { $href_attributes = " accesskey=\"$accesskey\""; } } my $button_rel = $self->direction_string($direction, 'rel', 'string'); if (defined($button_rel) and ($button_rel ne '')) { $href_attributes .= " rel=\"$button_rel\""; } return $href_attributes; } my %html_default_node_directions; foreach my $node_directions ('NodeNext', 'NodePrev', 'NodeUp') { $html_default_node_directions{$node_directions} = 1; } sub _default_format_button($$;$) { my ($self, $button, $source_command) = @_; my ($active, $passive, $need_delimiter); if (ref($button) eq 'CODE') { ($active, $need_delimiter) = &$button($self); } elsif (ref($button) eq 'ARRAY' and scalar(@$button == 2)) { my $text = $button->[1]; my $direction = $button->[0]; # $direction is simple text and $text is a reference on code if (defined($direction) and ref($direction) eq '' and defined($text) and (ref($text) eq 'CODE')) { ($active, $need_delimiter) = &$text($self, $direction, $source_command); # $direction is simple text and $text is also a simple text } elsif (defined($direction) and ref($direction) eq '' and defined($text) and ref($text) eq '') { if ($text =~ s/^->\s*//) { # this case is mostly for tests, to test the direction type $text # with the direction $direction $active = $self->from_element_direction($direction, $text, undef, undef, $source_command); } $need_delimiter = 1; } } elsif (defined($self->global_direction_text($button))) { # handle "direction" text button without output unit (Space) if ($self->get_conf('ICONS')) { my $direction_icon; my $active_icons = $self->get_conf('ACTIVE_ICONS'); if (defined($active_icons)) { $direction_icon = $active_icons->{$button}; } if (defined($direction_icon) and $direction_icon ne '') { my $button_name_string = $self->direction_string($button, 'button', 'string'); $active = &{$self->formatting_function('format_button_icon_img')}($self, $button_name_string, $direction_icon); } else { $active = $self->direction_string($button, 'text'); } } else { $active = $self->direction_string($button, 'text'); } $need_delimiter = 0; } else { my $href = $self->from_element_direction($button, 'href', undef, undef, $source_command); if (defined($href)) { # button is active my $btitle = ''; my $description = $self->direction_string($button, 'description', 'string'); if (defined($description)) { $btitle = ' title="' . $description . '"'; } if ($self->get_conf('USE_ACCESSKEY')) { my $accesskey = $self->direction_string($button, 'accesskey', 'string'); if (defined($accesskey) and $accesskey ne '') { $btitle .= " accesskey=\"$accesskey\""; } } my $button_rel = $self->direction_string($button, 'rel', 'string'); if (defined($button_rel) and $button_rel ne '') { $btitle .= " rel=\"$button_rel\""; } if ($self->get_conf('ICONS')) { my $active_icon; my $active_icons = $self->get_conf('ACTIVE_ICONS'); if (defined($active_icons)) { $active_icon = $active_icons->{$button}; } if (defined($active_icon) and $active_icon ne '') { my $button_name_string = $self->direction_string($button, 'button', 'string'); $active = "". &{$self->formatting_function('format_button_icon_img')}($self, $button_name_string, $active_icon, $self->from_element_direction($button, 'string')) .""; } else { # use text my $button_text = $self->direction_string($button, 'text'); $button_text = '' if (!defined($button_text)); $active = '['."".$button_text."".']'; } } else { # use text my $button_text = $self->direction_string($button, 'text'); $button_text = '' if (!defined($button_text)); $active = '['."".$button_text."".']'; } } else { # button is passive if ($self->get_conf('ICONS')) { my $passive_icon; my $passive_icons = $self->get_conf('PASSIVE_ICONS'); if (defined($passive_icons)) { $passive_icon = $passive_icons->{$button}; } if (defined($passive_icon) and $passive_icon ne '') { my $button_name_string = $self->direction_string($button, 'button', 'string'); $passive = &{$self->formatting_function('format_button_icon_img')}( $self, $button_name_string, $passive_icon, $self->from_element_direction($button, 'string')); } else { my $button_text = $self->direction_string($button, 'text'); $button_text = '' if (!defined($button_text)); $passive = '['.$button_text. ']'; } } else { my $button_text = $self->direction_string($button, 'text'); $button_text = '' if (!defined($button_text)); $passive = '['.$button_text. ']'; } } $need_delimiter = 0; } if (not defined($need_delimiter)) { # NOTE other options could have been chosen in that case: # option 1: be forgiving if $need_delimiter is not set # if ($html_default_node_directions{$button}) { # $need_delimiter = 1; # } else { # $need_delimiter = 0; # } # option 2: be somewhat forgiving but show a backtrace #cluck ("need_delimiter not defined"); # $need_delimiter = 0; # option3: no pity confess ("need_delimiter not defined"); } return ($active, $passive, $need_delimiter); } # called for special elements and tree units sub _default_format_navigation_panel($$$$;$$) { my ($self, $buttons, $cmdname, $source_command, $vertical, $in_header) = @_; # a string may be passed, for instance through command line, therefore # it is useful to test that $buttons is an array reference to avoid # a Perl error message if (ref($buttons) ne 'ARRAY') { return ''; } # do the buttons first in case they are formatted as an empty string my $nr_of_buttons_shown = 0; my $result_buttons = ''; foreach my $button (@$buttons) { my $direction; if (ref($button) eq 'ARRAY' and defined($button->[0]) and ref($button->[0]) eq '') { $direction = $button->[0]; } elsif (defined($button) and ref($button) eq '') { $direction = $button; } # if the first button is an empty button, pass if (defined($direction) and $direction eq 'Space' and $nr_of_buttons_shown == 0) { next; } my ($active, $passive, $need_delimiter) # API info: using the API to allow for customization would be: # = &{$self->formatting_function('format_button')}($self, $button, # $source_command); = &{$self->{'formatting_function'}->{'format_button'}}($self, $button, $source_command); if ($self->get_conf('HEADER_IN_TABLE')) { $result_buttons .= ''."\n" if $vertical; $result_buttons .= $self->html_attribute_class('td', ['nav-button']).'>'; if (defined($active)) { $result_buttons .= $active; } elsif (defined($passive)) { $result_buttons .= $passive; } $result_buttons .= "\n"; $result_buttons .= "\n" if $vertical; $nr_of_buttons_shown++; } elsif (defined($active)) { # only active buttons are print out when not in table if ($need_delimiter and $nr_of_buttons_shown > 0) { $result_buttons .= ', '; } my $open = $self->html_attribute_class('span', ['nav-button']); if ($open ne '') { $result_buttons .= $open.'>'; } $result_buttons .= $active; if ($open ne '') { $result_buttons .= ''; } $nr_of_buttons_shown++; } } if ($result_buttons eq '') { return ''; } my $result = ''; # if $vertical/VERTICAL_HEAD_NAVIGATION, the buttons are in a vertical # table which is itself in the first column of a table opened in # header_navigation if ($self->get_conf('HEADER_IN_TABLE')) { $result .= $self->html_attribute_class('table', ['nav-panel']).'>'."\n"; $result .= "" unless $vertical; } else { $result .= $self->html_attribute_class('p', ['nav-panel']).">\n"; } $result .= $result_buttons; if ($self->get_conf('HEADER_IN_TABLE')) { $result .= "" unless $vertical; $result .= "\n"; } else { $result .= "

\n"; } return $result; } sub _default_format_navigation_header($$$$) { my ($self, $buttons, $cmdname, $element) = @_; my $result = ''; if ($self->get_conf('VERTICAL_HEAD_NAVIGATION')) { $result .= $self->html_attribute_class('table', ['vertical-navigation']).'>'."\n"; $result .= "\n"; $result .= $self->html_attribute_class('td', ['vertical-navigation']).'>'."\n"; } $result .= &{$self->formatting_function('format_navigation_panel')}($self, $buttons, $cmdname, $element, $self->get_conf('VERTICAL_HEAD_NAVIGATION'), 1); if ($self->get_conf('VERTICAL_HEAD_NAVIGATION')) { $result .= ' '; } elsif ($self->get_conf('SPLIT') and $self->get_conf('SPLIT') eq 'node' and $result ne '' and defined($self->get_conf('DEFAULT_RULE'))) { $result .= $self->get_conf('DEFAULT_RULE')."\n"; } return $result; } # this can only be called on root commands and associated tree units sub _default_format_element_header($$$$) { my ($self, $cmdname, $command, $output_unit) = @_; my $result = ''; print STDERR "FORMAT elt header " # uncomment to get perl object names #."$output_unit (@{$output_unit->{'unit_contents'}}) ". . "(".join('|', map{Texinfo::Common::debug_print_element($_)} @{$output_unit->{'unit_contents'}}) . ") ". Texinfo::OutputUnits::output_unit_texi($output_unit) ."\n" if ($self->get_conf('DEBUG')); # Do the heading if the command is the first command in the element if (($output_unit->{'unit_contents'}->[0] eq $command or (!exists($output_unit->{'unit_contents'}->[0]->{'cmdname'}) and $output_unit->{'unit_contents'}->[1] eq $command)) # and there is more than one element and (exists($output_unit->{'tree_unit_directions'}))) { my $is_top = $self->unit_is_top_output_unit($output_unit); my $first_in_page = 0; if (exists($output_unit->{'unit_filename'}) and $self->count_elements_in_filename('current', $output_unit->{'unit_filename'}) == 1) { $first_in_page = 1; } my $previous_is_top = 0; $previous_is_top = 1 if (exists($output_unit->{'tree_unit_directions'}->{'prev'}) and $self->unit_is_top_output_unit($output_unit->{'tree_unit_directions'} ->{'prev'})); print STDERR "Header ($previous_is_top, $is_top, $first_in_page): " .Texinfo::Convert::Texinfo::root_heading_command_to_texinfo($command)."\n" if ($self->get_conf('DEBUG')); if ($is_top) { # use TOP_BUTTONS for top. $result .= &{$self->formatting_function('format_navigation_header')}($self, $self->get_conf('TOP_BUTTONS'), $cmdname, $command) if ($self->get_conf('SPLIT') or $self->get_conf('HEADERS')); } else { my $split = $self->get_conf('SPLIT'); if ($first_in_page and !$self->get_conf('HEADERS')) { if (defined($split) and $split eq 'chapter') { $result .= &{$self->formatting_function('format_navigation_header')}($self, $self->get_conf('CHAPTER_BUTTONS'), $cmdname, $command); $result .= $self->get_conf('DEFAULT_RULE') ."\n" if (defined($self->get_conf('DEFAULT_RULE')) and !$self->get_conf('VERTICAL_HEAD_NAVIGATION')); } elsif (defined($split) and $split eq 'section') { $result .= &{$self->formatting_function('format_navigation_header')}($self, $self->get_conf('SECTION_BUTTONS'), $cmdname, $command); } } if (($first_in_page or $previous_is_top) and $self->get_conf('HEADERS')) { $result .= &{$self->formatting_function('format_navigation_header')}($self, $self->get_conf('SECTION_BUTTONS'), $cmdname, $command); } elsif ($self->get_conf('HEADERS') or (defined($split) and $split eq 'node')) { # got to do this here, as it isn't done otherwise since # navigation_header is not called $result .= &{$self->formatting_function('format_navigation_panel')}($self, $self->get_conf('SECTION_BUTTONS'), $cmdname, $command, undef, 1); } } } return $result; } sub register_opened_section_level($$$$) { my ($self, $filename, $level, $close_string) = @_; if (!exists($self->{'pending_closes'}->{$filename})) { $self->{'pending_closes'}->{$filename} = []; } my $pending_closes = $self->{'pending_closes'}->{$filename}; while (@$pending_closes < $level) { push(@$pending_closes, ""); } push(@$pending_closes, $close_string); } sub close_registered_sections_level($$$) { my ($self, $filename, $level) = @_; if (not defined($level)) { cluck 'close_registered_sections_level $level not defined'; } my @closed_elements; if (!exists($self->{'pending_closes'}->{$filename})) { return \@closed_elements; } my $pending_closes = $self->{'pending_closes'}->{$filename}; while (@$pending_closes > $level) { my $close_string = pop @$pending_closes; push(@closed_elements, $close_string) if ($close_string ne ""); } return \@closed_elements; } sub _contents_inline_element($$$) { my ($self, $cmdname, # undef unless called from @-command formatting function $element) = @_; print STDERR "CONTENTS_INLINE $cmdname\n" if ($self->get_conf('DEBUG')); my $table_of_contents = &{$self->formatting_function('format_contents')}($self, $cmdname, $element); if (defined($table_of_contents) and $table_of_contents ne '') { my ($special_unit_variety, $special_unit, $class_base, $special_unit_direction) = $self->command_name_special_unit_information($cmdname); my $result = $self->html_attribute_class('div', ["region-${class_base}"]); my $unit_command = $special_unit->{'unit_command'}; my $id = $self->command_id($unit_command); if (defined($id) and $id ne '') { $result .= " id=\"$id\""; } $result .= ">\n"; my $heading = $self->command_text($unit_command); $heading = '' if (!defined($heading)); $result .= &{$self->formatting_function('format_heading_text')}($self, $cmdname, [$class_base.'-heading'], $heading, $self->get_conf('CHAPTER_HEADER_LEVEL'))."\n"; $result .= $table_of_contents . "\n"; return $result; } return ''; } sub _convert_heading_command($$$$$) { my ($self, $cmdname, $element, $args, $content) = @_; my $result = ''; # No situation where this could happen if (in_string($self)) { $result .= $self->command_text($element, 'string') ."\n" if ($cmdname ne 'node'); $result .= $content if (defined($content)); return $result; } my $element_id = $self->command_id($element); print STDERR "CONVERT elt heading " # uncomment next line for the perl object name #."$element " .Texinfo::Convert::Texinfo::root_heading_command_to_texinfo($element)."\n" if ($self->get_conf('DEBUG')); my $document = $self->get_info('document'); my $sections_list; my $nodes_list; if (defined($document)) { $sections_list = $document->sections_list(); $nodes_list = $document->nodes_list(); } my $output_unit; my $section_relations; my $node_relations; if (exists($Texinfo::Commands::root_commands{$cmdname})) { if ($cmdname eq 'node') { if (defined($nodes_list) and exists($element->{'extra'}) and $element->{'extra'}->{'node_number'}) { $node_relations = $nodes_list->[$element->{'extra'}->{'node_number'} -1]; } } elsif (defined($sections_list)) { $section_relations = $sections_list->[$element->{'extra'}->{'section_number'} -1]; } # All the root commands are associated to an output unit, the condition # on associated_unit is always true. if (exists($element->{'associated_unit'})) { $output_unit = $element->{'associated_unit'}; } } my $element_header = ''; if ($output_unit) { $element_header = &{$self->formatting_function('format_element_header')}( $self, $cmdname, $element, $output_unit); } my $toc_or_mini_toc_or_auto_menu = ''; if ($self->get_conf('CONTENTS_OUTPUT_LOCATION') eq 'after_top' and $cmdname eq 'top' and defined($sections_list) and scalar(@{$sections_list}) > 1) { foreach my $content_command_name ('shortcontents', 'contents') { if ($self->get_conf($content_command_name)) { my $contents_text = _contents_inline_element($self, $content_command_name, undef); if ($contents_text ne '') { $toc_or_mini_toc_or_auto_menu .= $contents_text; } } } } my $format_menu = $self->get_conf('FORMAT_MENU'); if ($toc_or_mini_toc_or_auto_menu eq '' and defined($section_relations)) { if ($format_menu eq 'sectiontoc') { $toc_or_mini_toc_or_auto_menu = _mini_toc($self, $section_relations); } elsif (($format_menu eq 'menu' or $format_menu eq 'menu_no_detailmenu') and exists($section_relations->{'associated_node'})) { my $associated_node_relations = $section_relations->{'associated_node'}; # arguments_line type element my $arguments_line = $associated_node_relations->{'element'}->{'contents'}->[0]; my $automatic_directions = 1; if (scalar(@{$arguments_line->{'contents'}}) > 1) { $automatic_directions = 0; } if ($automatic_directions and !exists($associated_node_relations->{'menus'})) { my $identifiers_target = $document->labels_information(); my $menu_node; if ($format_menu eq 'menu') { $menu_node = Texinfo::Structuring::new_complete_menu_master_menu($self, $identifiers_target, $nodes_list, $associated_node_relations); } else { # $format_menu eq 'menu_no_detailmenu' $menu_node = Texinfo::Structuring::new_complete_node_menu( $associated_node_relations, $self->{'current_lang_translations'}, $self->get_conf('DEBUG')); } if (defined($menu_node)) { $toc_or_mini_toc_or_auto_menu = $self->convert_tree($menu_node, 'master menu'); } } } } if ($self->get_conf('NO_TOP_NODE_OUTPUT') and exists($Texinfo::Commands::root_commands{$cmdname})) { my $in_skipped_node_top = $self->get_shared_conversion_state('top', 'in_skipped_node_top'); $in_skipped_node_top = 0 if (!defined($in_skipped_node_top)); if ($in_skipped_node_top == 1) { my $id_class = $cmdname; $result .= &{$self->formatting_function('format_separate_anchor')}($self, $element_id, $id_class); $result .= $element_header; $result .= $toc_or_mini_toc_or_auto_menu; return $result; } } my $level_corrected_cmdname = $cmdname; my $level_set_class; if (exists($element->{'extra'}) and exists($element->{'extra'}->{'section_level'})) { # if the level was changed, use a consistent command name $level_corrected_cmdname = Texinfo::Structuring::section_level_adjusted_command_name($element); if ($level_corrected_cmdname ne $cmdname) { $level_set_class = "${cmdname}-level-set-${level_corrected_cmdname}"; } } # find the section starting here, can be through the associated node # preceding the section, or the section itself my $opening_section; my $level_corrected_opening_section_cmdname; if (defined($node_relations) and exists($node_relations->{'associated_section'})) { $opening_section = $node_relations->{'associated_section'}->{'element'}; $level_corrected_opening_section_cmdname = Texinfo::Structuring::section_level_adjusted_command_name( $opening_section); # if there is an associated node, it is not a section opening # the section was opened before when the node was encountered } elsif (defined($section_relations) and !exists($section_relations->{'associated_node'})) { $opening_section = $element; $level_corrected_opening_section_cmdname = $level_corrected_cmdname; } # could use empty args information also, to avoid calling command_text #my $empty_heading = (!scalar(@$args) or !defined($args->[0])); # $heading not defined may happen if the command is a @node, for example # if there is an error in the node. my $heading = $self->command_text($element); my $heading_level; # node is used as heading if there is nothing else. if (defined($node_relations)) { if (defined($output_unit) and exists($output_unit->{'unit_node'}) and $output_unit->{'unit_node'} eq $node_relations and !exists($node_relations->{'associated_title_command'})) { if ($element->{'extra'}->{'normalized'} eq 'Top') { $heading_level = 0; } else { # use node $heading_level = 3; } } } elsif (exists($element->{'extra'}) and exists($element->{'extra'}->{'section_level'})) { $heading_level = $element->{'extra'}->{'section_level'}; } else { # for *heading* @-commands which do not have a level # in the document as they are not associated with the # sectioning tree, but still have a $heading_level $heading_level = Texinfo::Common::section_level($element); } my $do_heading = (defined($heading) and $heading ne '' and defined($heading_level)); # if set, the id is associated to the heading text my $heading_id; if ($opening_section) { my $level; if (exists($opening_section->{'extra'}) and exists($opening_section->{'extra'}->{'section_level'})) { $level = $opening_section->{'extra'}->{'section_level'}; } else { # if Structuring sectioning_structure was not called on the # document (cannot happen in main program or test_utils.pl tests) $level = Texinfo::Common::section_level($opening_section); } my $closed_strings = $self->close_registered_sections_level( $self->current_filename(), $level); $result .= join('', @{$closed_strings}); $self->register_opened_section_level($self->current_filename(), $level, "\n"); # use a specific class name to mark that this is the start of # the section extent. It is not necessary where the section is. $result .= $self->html_attribute_class('div', ["${level_corrected_opening_section_cmdname}-level-extent"]); $result .= " id=\"$element_id\"" if (defined($element_id) and $element_id ne ''); $result .= ">\n"; } elsif (defined($element_id) and $element_id ne '') { if ($element_header ne '') { # case of a @node without sectioning command and with a header. # put the node element anchor before the header. # Set the class name to the command name if there is no heading, # else the class will be with the heading element. my $id_class = $cmdname; if ($do_heading) { $id_class = "${cmdname}-id"; } $result .= &{$self->formatting_function('format_separate_anchor')}($self, $element_id, $id_class); } else { $heading_id = $element_id; } } $result .= $element_header; if ($do_heading) { if ($self->get_conf('TOC_LINKS') and exists($Texinfo::Commands::root_commands{$cmdname}) and exists($sectioning_heading_commands{$cmdname})) { my $content_href = $self->command_contents_href($element, 'contents'); if (defined($content_href)) { $heading = "$heading"; } } my @heading_classes; push @heading_classes, $level_corrected_cmdname; if (defined($level_set_class)) { push @heading_classes, $level_set_class; } if (in_preformatted_context($self)) { my $id_str = ''; if (defined($heading_id)) { $id_str = " id=\"$heading_id\""; } $result .= $self->html_attribute_class('strong', \@heading_classes) ."${id_str}>".$heading.''."\n"; } else { $result .= &{$self->formatting_function('format_heading_text')}($self, $level_corrected_cmdname, \@heading_classes, $heading, $heading_level +$self->get_conf('CHAPTER_HEADER_LEVEL') -1, $heading_id, $element, $element_id); } } elsif (defined($heading_id)) { # case of a lone node and no header, and case of an empty @top $result .= &{$self->formatting_function('format_separate_anchor')}($self, $heading_id, $cmdname); } $result .= $content if (defined($content)); $result .= $toc_or_mini_toc_or_auto_menu; return $result; } foreach my $cmdname (keys(%sectioning_heading_commands), 'node') { $default_commands_conversion{$cmdname} = \&_convert_heading_command; } sub _convert_raw_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); if ($cmdname eq 'html') { return $content; } # In multiple conversions should only happen rarely, as in general, format # commands do not happen in inline context where most of the multiple # conversions are. A possibility is in float caption. if (!$self->in_multiple_conversions()) { $self->converter_line_warn(sprintf(__("raw format %s is not converted"), $cmdname), $command->{'source_info'}); } return &{$self->formatting_function('format_protect_text')}($self, $content); } foreach my $cmdname (keys(%format_raw_commands)) { $default_commands_conversion{$cmdname} = \&_convert_raw_command; } sub _convert_inline_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $format; if (defined($args) and defined($args->[0]) and defined($args->[0]->{'monospacetext'}) and $args->[0]->{'monospacetext'} ne '') { $format = $args->[0]->{'monospacetext'}; } else { return ''; } my $arg_index = undef; if (exists($inline_format_commands{$cmdname})) { if ($cmdname eq 'inlinefmtifelse' and !$self->is_format_expanded($format)) { $arg_index = 2; } elsif ($self->is_format_expanded($format)) { $arg_index = 1; } } elsif (exists($command->{'extra'}) and $command->{'extra'}->{'expand_index'}) { $arg_index = 1; } if (defined($arg_index) and $arg_index < scalar(@$args)) { my $text_arg = $args->[$arg_index]; if (defined($text_arg)) { if (defined($text_arg->{'normal'})) { return $text_arg->{'normal'}; } elsif (defined($text_arg->{'raw'})) { return $text_arg->{'raw'}; } } } return ''; } foreach my $cmdname (grep {$brace_commands{$_} eq 'inline'} keys(%brace_commands)) { $default_commands_conversion{$cmdname} = \&_convert_inline_command; } sub _indent_with_table($$$;$) { my ($self, $cmdname, $content, $extra_classes) = @_; my @classes = ($cmdname); push (@classes, @$extra_classes) if (defined($extra_classes)); return $self->html_attribute_class('table', \@classes) .'>'.$self->get_info('non_breaking_space').''.$content ."\n"; } sub _convert_preformatted_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; if (!defined($content) or $content eq '') { return ''; } if (in_string($self)) { return $content; } my @classes; # this is mainly for classes as there are purprosely no classes # for small* my $main_cmdname; if (exists($small_block_associated_command{$cmdname})) { $main_cmdname = $small_block_associated_command{$cmdname}; push @classes, $cmdname; } else { $main_cmdname = $cmdname; } if ($cmdname eq 'example') { # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; foreach my $example_arg (@{$arguments_line->{'contents'}}) { # convert or remove all @-commands, using simple ascii and unicode # characters my $converted_arg = Texinfo::Convert::NodeNameNormalization::convert_to_normalized( $example_arg); if ($converted_arg ne '') { push @classes, 'user-' . $converted_arg; } } } elsif ($main_cmdname eq 'lisp') { push @classes, $main_cmdname; $main_cmdname = 'example'; } if ($self->get_conf('INDENTED_BLOCK_COMMANDS_IN_TABLE') and exists($indented_preformatted_commands{$cmdname})) { return _indent_with_table($self, $cmdname, $content, \@classes); } else { unshift @classes, $main_cmdname; return $self->html_attribute_class('div', \@classes) .">\n".$content.''."\n"; } } foreach my $preformatted_command (keys(%preformatted_commands)) { $default_commands_conversion{$preformatted_command} = \&_convert_preformatted_command; } sub _convert_indented_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; if (!defined($content) or $content eq '') { return ''; } if (in_string($self)) { return $content; } my @classes; my $main_cmdname; if (exists($small_block_associated_command{$cmdname})) { push @classes, $cmdname; $main_cmdname = $small_block_associated_command{$cmdname}; } else { $main_cmdname = $cmdname; } if ($self->get_conf('INDENTED_BLOCK_COMMANDS_IN_TABLE')) { return _indent_with_table($self, $main_cmdname, $content, \@classes); } else { unshift @classes, $main_cmdname; return $self->html_attribute_class('blockquote', \@classes).">\n" . $content . ''."\n"; } } $default_commands_conversion{'indentedblock'} = \&_convert_indented_command; sub _convert_verbatim_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); if (!in_string($self)) { return $self->html_attribute_class('pre', [$cmdname]).'>' .$content . ''; } else { return $content; } } $default_commands_conversion{'verbatim'} = \&_convert_verbatim_command; sub _convert_displaymath_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); if (in_string($self)) { return $content; } my $result = ''; my $pre_classes = [$cmdname]; my $use_mathjax = ($self->get_conf('HTML_MATH') and $self->get_conf('HTML_MATH') eq 'mathjax'); if ($use_mathjax) { $self->register_file_information('mathjax', 1); push @$pre_classes, 'tex2jax_process'; } $result .= $self->html_attribute_class('pre', $pre_classes).'>'; if ($self->get_conf('HTML_MATH') and $self->get_conf('HTML_MATH') eq 'mathjax') { $result .= "\\[$content\\]"; } else { $result .= $content; } $result .= ''; return $result; } $default_commands_conversion{'displaymath'} = \&_convert_displaymath_command; sub _convert_verbatiminclude_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $verbatim_include_verbatim = $self->expand_verbatiminclude($command); if (defined($verbatim_include_verbatim)) { return $self->convert_tree($verbatim_include_verbatim, 'convert verbatiminclude'); } else { return ''; } } $default_commands_conversion{'verbatiminclude'} = \&_convert_verbatiminclude_command; sub _convert_command_simple_block($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); return $self->html_attribute_class('div', [$cmdname]).'>' .$content.''; } $default_commands_conversion{'raggedright'} = \&_convert_command_simple_block; $default_commands_conversion{'flushleft'} = \&_convert_command_simple_block; $default_commands_conversion{'flushright'} = \&_convert_command_simple_block; $default_commands_conversion{'group'} = \&_convert_command_simple_block; sub _convert_sp_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $sp_nr = 1; if (exists($command->{'extra'}) and exists($command->{'extra'}->{'misc_args'})) { $sp_nr = $command->{'extra'}->{'misc_args'}->[0]; } if ($sp_nr > 0) { if (in_preformatted_context($self) or in_string($self)) { return "\n" x $sp_nr; } else { return ($self->get_info('line_break_element')."\n") x $sp_nr; } } else { return ''; } } $default_commands_conversion{'sp'} = \&_convert_sp_command; sub _convert_exdent_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $arg = $self->get_pending_formatted_inline_content(); if (defined($args) and defined($args->[0])) { $arg .= $args->[0]->{'normal'}; } if (in_string($self)) { return $arg ."\n"; } # FIXME do something with CSS? Currently nothing is defined for exdent if (in_preformatted_context($self)) { return $self->html_attribute_class('pre', [$cmdname]).'>'.$arg ."\n"; } else { return $self->html_attribute_class('p', [$cmdname]).'>'.$arg ."\n

"; } } $default_commands_conversion{'exdent'} = \&_convert_exdent_command; sub _convert_center_command($$$$) { my ($self, $cmdname, $command, $args) = @_; if (!defined($args) or !defined($args->[0])) { return ''; } if (in_string($self)) { return $args->[0]->{'normal'}."\n"; } else { return $self->html_attribute_class('div', [$cmdname]).">" .$args->[0]->{'normal'}."\n"; } } $default_commands_conversion{'center'} = \&_convert_center_command; sub _convert_author_command($$$$) { my ($self, $cmdname, $command, $args) = @_; my $quotation_titlepage_nr = $self->get_shared_conversion_state('quotation', 'quotation_titlepage_stack'); if (defined($quotation_titlepage_nr) and $quotation_titlepage_nr > 0) { my $authors_nr = $self->get_shared_conversion_state('quotation', 'element_authors_number', $quotation_titlepage_nr); if ($authors_nr < 0) { # in titlepage if (!in_string($self)) { return $self->html_attribute_class('strong', [$cmdname]) .">$args->[0]->{'normal'}" .$self->get_info('line_break_element')."\n"; } else { return $args->[0]->{'normal'} . "\n"; } } else { # in quotation $self->set_shared_conversion_state('quotation', 'elements_authors', $quotation_titlepage_nr, $authors_nr, $command); $authors_nr++; $self->set_shared_conversion_state('quotation', 'element_authors_number', $quotation_titlepage_nr, $authors_nr); } } return ''; } $default_commands_conversion{'author'} = \&_convert_author_command; sub _convert_title_command($$$$) { my ($self, $cmdname, $command, $args) = @_; return '' if (!defined($args) or !defined($args->[0])); if (!in_string($self)) { return $self->html_attribute_class('h1', [$cmdname]) .">$args->[0]->{'normal'}\n"; } else { return $args->[0]->{'normal'}; } } $default_commands_conversion{'title'} = \&_convert_title_command; sub _convert_subtitle_command($$$$) { my ($self, $cmdname, $command, $args) = @_; return '' if (!defined($args) or !defined($args->[0])); if (!in_string($self)) { return $self->html_attribute_class('h3', [$cmdname]) .">$args->[0]->{'normal'}\n"; } else { return $args->[0]->{'normal'}; } } $default_commands_conversion{'subtitle'} = \&_convert_subtitle_command; sub _convert_insertcopying_command($$$) { my ($self, $cmdname, $command) = @_; my $global_commands; my $document = $self->get_info('document'); if (defined($document)) { $global_commands = $document->global_commands_information(); } if (defined($global_commands) and exists($global_commands->{'copying'})) { return $self->convert_tree( Texinfo::TreeElement::new( {'contents' => $global_commands->{'copying'}->{'contents'}}), 'convert insertcopying'); } return ''; } $default_commands_conversion{'insertcopying'} = \&_convert_insertcopying_command; sub _convert_maketitle_command($$$) { my ($self, $cmdname, $command) = @_; return $self->get_info('title_titlepage'); } $default_commands_conversion{'maketitle'} = \&_convert_maketitle_command; sub _convert_listoffloats_command($$$$) { my ($self, $cmdname, $command, $args) = @_; # should probably never happen return '' if (in_string($self)); my $floats; my $document = $self->get_info('document'); if (defined($document)) { $floats = $document->floats_information(); } my $listoffloats_name = $command->{'extra'}->{'float_type'}; my $formatted_listoffloats_nr = $self->get_shared_conversion_state('listoffloats', 'formatted_listoffloats', $listoffloats_name); $formatted_listoffloats_nr = 0 if (!defined($formatted_listoffloats_nr)); $formatted_listoffloats_nr++; $self->set_shared_conversion_state('listoffloats', 'formatted_listoffloats', $listoffloats_name, $formatted_listoffloats_nr); if (defined($floats) and exists($floats->{$listoffloats_name}) and scalar(@{$floats->{$listoffloats_name}})) { my $result = $self->html_attribute_class('dl', [$cmdname]).">\n" ; foreach my $float_and_section (@{$floats->{$listoffloats_name}}) { my ($float, $float_section) = @$float_and_section; my $float_href = $self->command_href($float); next if (!defined($float_href)); $result .= '
'; my $float_text = $self->command_text($float); if (defined($float_text) and $float_text ne '') { if ($float_href ne '') { $result .= "$float_text"; } else { $result .= $float_text; } } $result .= '
'; my $caption_element; my $caption_cmdname; my ($caption, $shortcaption) = Texinfo::Common::find_float_caption_shortcaption($float); if (defined($shortcaption)) { $caption_element = $shortcaption; $caption_cmdname = 'shortcaption'; } elsif (defined($caption)) { $caption_element = $caption; $caption_cmdname = 'caption'; } my $caption_text; my @caption_classes; if (defined($caption_element)) { my $multiple_formatted = 'listoffloats'; if ($formatted_listoffloats_nr > 1) { $multiple_formatted .= '-'.($formatted_listoffloats_nr - 1); } $caption_text = $self->convert_tree_new_formatting_context( $caption_element->{'contents'}->[0], $cmdname, undef, $multiple_formatted); push @caption_classes, "${caption_cmdname}-in-${cmdname}"; } else { $caption_text = ''; } $result .= $self->html_attribute_class('dd', \@caption_classes).'>' .$caption_text.''."\n"; } return $result . "\n"; } else { return ''; } } $default_commands_conversion{'listoffloats'} = \&_convert_listoffloats_command; sub _convert_menu_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); return $content if ($cmdname eq 'detailmenu'); $self->set_shared_conversion_state('menu', 'html_menu_entry_index', 0); if ($content !~ /\S/) { return ''; } # This can probably only happen with incorrect input. It happens with # menu in documentdescription. It does not seem that it could happen # in other situation with a Texinfo tree parsed from Texinfo code. if (in_string($self)) { return $content; } my $begin_row = ''; my $end_row = ''; if (inside_preformatted($self)) { $begin_row = ''; $end_row = ''; } return $self->html_attribute_class('table', [$cmdname]) .">${begin_row}\n" . $content . "${end_row}\n"; } $default_commands_conversion{'menu'} = \&_convert_menu_command; $default_commands_conversion{'detailmenu'} = \&_convert_menu_command; sub _convert_float_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); my ($caption_element, $prepended) = Texinfo::Convert::Converter::float_name_caption($self, $command); if (in_string($self)) { my $prepended_text; if (defined($prepended)) { $prepended_text = $self->convert_tree_new_formatting_context($prepended, 'float prepended'); } else { $prepended_text = ''; } my $caption_text = ''; if (defined($caption_element) and exists($caption_element->{'contents'}) and exists($caption_element->{'contents'}->[0]->{'contents'})) { $caption_text = $self->convert_tree_new_formatting_context( $caption_element->{'contents'}->[0], 'float caption'); } return $prepended_text.$content.$caption_text; } my $caption_command_name; if (defined($caption_element)) { $caption_command_name = $caption_element->{'cmdname'}; } my $result = $self->html_attribute_class('div', [$cmdname]); my $id = $self->command_id($command); if (defined($id) and $id ne '') { $result .= " id=\"$id\""; } $result .= ">\n" . $content; my $prepended_text; my $caption_text; if (defined($prepended)) { # TODO add a span with a class name for the prependend information # if not empty? $prepended_text = $self->convert_tree_new_formatting_context( Texinfo::TreeElement::new({'cmdname' => 'strong', 'contents' => [ Texinfo::TreeElement::new({'type' => 'brace_container', 'contents' => [$prepended]})]}), 'float number type'); if (defined($caption_element)) { # register the converted prepended tree to be prepended to # the first paragraph in caption formatting $self->register_pending_formatted_inline_content($caption_command_name, $prepended_text); $caption_text = $self->convert_tree_new_formatting_context( $caption_element->{'contents'}->[0], 'float caption'); my $cancelled_prepended = $self->cancel_pending_formatted_inline_content($caption_command_name); # unset if prepended text is in caption, i.e. is not cancelled $prepended_text = '' if (not defined($cancelled_prepended)); } if ($prepended_text ne '') { # prepended text is not empty and did not find its way in caption $prepended_text = '

'.$prepended_text.'

'; } } elsif (defined($caption_element)) { $caption_text = $self->convert_tree_new_formatting_context( $caption_element->{'contents'}->[0], 'float caption'); } if (defined($caption_text) and $caption_text ne '') { $result .= $self->html_attribute_class('div', [$caption_command_name]). '>' .$caption_text.''; } elsif (defined($prepended) and $prepended_text ne '') { $result .= $self->html_attribute_class('div', ['type-number-float']). '>' . $prepended_text .''; } return $result . ''; } $default_commands_conversion{'float'} = \&_convert_float_command; sub _convert_quotation_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); $self->cancel_pending_formatted_inline_content($cmdname); my $result; if (!in_string($self)) { my @classes; if (exists($small_block_associated_command{$cmdname})) { push @classes, $small_block_associated_command{$cmdname}; } push @classes, $cmdname; $result = $self->html_attribute_class('blockquote', \@classes).">\n" . $content . "\n"; } else { $result = $content; } my $quotation_titlepage_nr = $self->get_shared_conversion_state('quotation', 'quotation_titlepage_stack'); my $quotation_authors = []; if (defined($quotation_titlepage_nr) and $quotation_titlepage_nr > 0) { my $authors_nr = $self->get_shared_conversion_state('quotation', 'element_authors_number', $quotation_titlepage_nr); if ($authors_nr < 0) { print STDERR "BUG: unexpected negative element_authors_number" ." $authors_nr in convert_quotation_command\n"; $authors_nr = 0; } for (my $i = 0; $i < $authors_nr; $i++) { my $author = $self->get_shared_conversion_state('quotation', 'elements_authors', $quotation_titlepage_nr, $i); push @$quotation_authors, $author; } $quotation_titlepage_nr--; $self->set_shared_conversion_state('quotation', 'quotation_titlepage_stack', $quotation_titlepage_nr); } else { print STDERR "BUG: unexpected unset quotation_titlepage_stack" ."in convert_quotation_command\n"; } # TODO there is no easy way to mark with a class the @author # @-command. Add a span or a div (@center is in a div)? foreach my $author (@$quotation_authors) { if (exists($author->{'contents'}->[0]->{'contents'})) { # TRANSLATORS: quotation author my $centered_author = $self->cdt("\@center --- \@emph{{author}}", {'author' => $author->{'contents'}->[0]}); $result .= $self->convert_tree($centered_author, 'convert quotation author'); } } return $result; } $default_commands_conversion{'quotation'} = \&_convert_quotation_command; sub _convert_cartouche_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); my $title_content = ''; if (defined($args) and defined($args->[0]) and $args->[0]->{'normal'} ne '') { $title_content = "\n". $args->[0]->{'normal'} .""; } my $cartouche_content = ''; if ($content =~ /\S/) { $cartouche_content = "\n". $content .""; } if ($cartouche_content ne '' or $title_content ne '') { return $self->html_attribute_class('table', [$cmdname]) . ">${title_content}${cartouche_content}" . "\n"; } return $content; } $default_commands_conversion{'cartouche'} = \&_convert_cartouche_command; sub _convert_itemize_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); if (in_string($self)) { return $content; } # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; my $block_line_arg = $arguments_line->{'contents'}->[0]; my $command_as_argument_name; my $prepended_element = Texinfo::Common::itemize_line_prepended_element($block_line_arg); if (defined($prepended_element)) { $command_as_argument_name = $prepended_element->{'cmdname'}; } my $mark_class_name; if (defined($command_as_argument_name)) { if ($command_as_argument_name eq 'w') { $mark_class_name = 'none'; } else { $mark_class_name = $command_as_argument_name; } } if (defined($mark_class_name) and defined($self->css_get_selector_style('ul.mark-'.$mark_class_name))) { return $self->html_attribute_class('ul', [$cmdname, 'mark-'.$mark_class_name]) .">\n" . $content. "\n"; } elsif ($self->get_conf('NO_CSS')) { return $self->html_attribute_class('ul', [$cmdname]) .">\n" . $content. "\n"; } else { my $css_string = $self->html_convert_css_string_for_list_mark($block_line_arg, 'itemize arg'); if ($css_string ne '') { return $self->html_attribute_class('ul', [$cmdname]) ." style=\"list-style-type: '". &{$self->formatting_function('format_protect_text')}($self, $css_string) . "'\">\n" . $content. "\n"; } else { return $self->html_attribute_class('ul', [$cmdname]) .">\n" . $content. "\n"; } } } $default_commands_conversion{'itemize'} = \&_convert_itemize_command; sub _convert_enumerate_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; if (!defined($content) or $content eq '') { return ''; } elsif (in_string($self)) { return $content; } my $type_attribute = ''; my $start_attribute = ''; my ($start, $type); # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; my $block_line_arg = $arguments_line->{'contents'}->[0]; if (exists($block_line_arg->{'contents'}) and exists($block_line_arg->{'contents'}->[0]->{'text'})) { my $specification = $block_line_arg->{'contents'}->[0]->{'text'}; if ($specification =~ /^\d+$/ and $specification ne '1') { $start = $specification; } elsif ($specification =~ /^[A-Z]$/) { $start = 1 + ord($specification) - ord('A'); $type = 'A'; } elsif ($specification =~ /^[a-z]$/) { $start = 1 + ord($specification) - ord('a'); $type = 'a'; } $type_attribute = " type=\"$type\"" if (defined($type)); $start_attribute = " start=\"$start\"" if (defined($start)); } return $self->html_attribute_class('ol', [$cmdname]).$type_attribute .$start_attribute.">\n" . $content . "\n"; } $default_commands_conversion{'enumerate'} = \&_convert_enumerate_command; sub _convert_multitable_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; if (!defined($content)) { return ''; } if (in_string($self)) { return $content; } if ($content ne '') { return $self->html_attribute_class('table', [$cmdname]).">\n" . $content . "\n"; } else { return ''; } } $default_commands_conversion{'multitable'} = \&_convert_multitable_command; sub _convert_xtable_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; if (!defined($content)) { return ''; } if (in_string($self)) { return $content; } if ($content ne '') { return $self->html_attribute_class('dl', [$cmdname]).">\n" . $content . "\n"; } else { return ''; } } $default_commands_conversion{'table'} = \&_convert_xtable_command; $default_commands_conversion{'ftable'} = \&_convert_xtable_command; $default_commands_conversion{'vtable'} = \&_convert_xtable_command; sub _convert_item_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); if (in_string($self)) { return $content; } if (exists($command->{'parent'}->{'cmdname'}) and $command->{'parent'}->{'cmdname'} eq 'itemize') { if ($content =~ /\S/) { return '
  • ' . $content . '
  • '; } else { return ''; } } elsif (exists($command->{'parent'}->{'cmdname'}) and $command->{'parent'}->{'cmdname'} eq 'enumerate') { if ($content =~ /\S/) { return '
  • ' . ' ' . $content . '
  • '; } else { return ''; } } elsif (exists($command->{'contents'}) and exists($command->{'contents'}->[0]->{'type'}) and $command->{'contents'}->[0]->{'type'} eq 'line_arg') { if (exists($command->{'contents'}->[0]->{'contents'})) { my $result = ($cmdname eq 'item') ? '' : '
    '; my $index_entry_id = $self->command_id($command); my $anchor; if (defined($index_entry_id)) { $result .= ""; $anchor = _get_copiable_anchor($self, $index_entry_id); if (defined($anchor)) { $result .= ''; } } my $pre_class_close; if (in_preformatted_context($self)) { my $pre_classes = $self->preformatted_classes_stack(); foreach my $pre_class (@$pre_classes) { if (exists($preformatted_code_commands{$pre_class})) { $result .= $self->html_attribute_class('code', ['table-term-preformatted-code']).'>'; $pre_class_close = ''; last; } } } my $table_item_tree = $self->table_item_content_tree_noxs($command); $table_item_tree = $command->{'contents'}->[0] if (!defined($table_item_tree)); my $converted_item = $self->convert_tree($table_item_tree, 'convert table_item_tree'); $result .= $converted_item; if (defined($pre_class_close)) { $result .= $pre_class_close; } if (defined($anchor)) { $result .= $anchor . ''; } return $result . "
    \n"; } else { return ''; } } elsif ($command->{'parent'}->{'type'} and $command->{'parent'}->{'type'} eq 'row') { return &{$self->command_conversion('tab')}($self, $cmdname, $command, $args, $content); } return ''; } $default_commands_conversion{'item'} = \&_convert_item_command; $default_commands_conversion{'headitem'} = \&_convert_item_command; $default_commands_conversion{'itemx'} = \&_convert_item_command; sub _convert_tab_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); $content =~ s/^\s*//; $content =~ s/\s*$//; if (in_string($self)) { return $content; } my $cell_nr = $command->{'extra'}->{'cell_number'}; my $row = $command->{'parent'}; my $row_cmdname = $row->{'contents'}->[0]->{'cmdname'}; my $multitable = $row->{'parent'}->{'parent'}; my $columnfractions = Texinfo::Common::multitable_columnfractions($multitable); my $fractions = ''; if (defined($columnfractions)) { if (exists($columnfractions->{'extra'}->{'misc_args'}->[$cell_nr-1])) { my $percent = sprintf('%.0f', 100. * $columnfractions->{'extra'}->{'misc_args'}->[$cell_nr-1]); my $width = "$percent%"; if ($self->get_conf('_INLINE_STYLE_WIDTH')) { $fractions = " style=\"width: $width\""; } else { $fractions = " width=\"$width\""; } } } if ($row_cmdname eq 'headitem') { return "" . $content . ''; } else { return "" . $content . ''; } } $default_commands_conversion{'tab'} = \&_convert_tab_command; sub _convert_xref_commands($$$$) { my ($self, $cmdname, $command, $args) = @_; # may happen with bogus @-commands without argument, maybe only # at the end of a document if (!defined($args)) { return ''; } my $tree; my $name; if ($cmdname ne 'link' and $cmdname ne 'inforef' and defined($args->[2]) and defined($args->[2]->{'normal'}) and $args->[2]->{'normal'} ne '') { $name = $args->[2]->{'normal'}; } elsif (defined($args->[1]) and defined($args->[1]->{'normal'}) and $args->[1]->{'normal'} ne '') { $name = $args->[1]->{'normal'} } my $file_arg; if ($cmdname eq 'link' or $cmdname eq 'inforef') { if (defined($args->[2])) { $file_arg = $args->[2]; } } elsif (defined($args->[3])) { $file_arg = $args->[3]; } my $file; if (defined($file_arg) and defined($file_arg->{'filenametext'}) and $file_arg->{'filenametext'} ne '') { $file = $file_arg->{'filenametext'}; } my $book; $book = $args->[4]->{'normal'} if (defined($args->[4]) and exists($args->[4]->{'normal'}) and $args->[4]->{'normal'} ne ''); my $arg_node = $command->{'contents'}->[0]; # internal reference if ($cmdname ne 'inforef' and !defined($book) and !defined($file) and defined($arg_node) and exists($arg_node->{'extra'}) and exists($arg_node->{'extra'}->{'normalized'}) and !exists($arg_node->{'extra'}->{'manual_content'}) and $self->label_command($arg_node->{'extra'}->{'normalized'})) { my $target_node = $self->label_command($arg_node->{'extra'}->{'normalized'}); # This is the node if USE_NODES, otherwise this may be the sectioning # command (if the sectioning command is really associated to the node) my $target_root = $self->command_root_element_command($target_node); my $document = $self->get_info('document'); my $associated_section_relations; my $associated_title_command; if (defined($document) and $target_node->{'cmdname'} eq 'node') { my $nodes_list = $document->nodes_list(); my $node_relations = $nodes_list->[$target_node->{'extra'}->{'node_number'} -1]; $associated_section_relations = $node_relations->{'associated_section'}; $associated_title_command = $node_relations->{'associated_title_command'}; } if (!defined($associated_section_relations) or $associated_section_relations->{'element'} ne $target_root) { $target_root = $target_node; } my $href; if (!in_string($self)) { $href = $self->command_href($target_root, undef, $command); } if (!defined($name)) { if ($self->get_conf('xrefautomaticsectiontitle') eq 'on' and defined($associated_title_command) # this condition avoids infinite recursions, indeed in that case # the node will be used and not the section. There should not be # @*ref in nodes, and even if there are, it does not seems to be # possible to construct an infinite recursion with nodes only # as the node must both be a reference target and refer to a specific # target at the same time, which is not possible. and not _command_is_in_referred_command_stack($self, $associated_title_command)) { if (in_string($self)) { $name = $self->command_text($associated_title_command, 'string'); } else { $name = $self->command_text($associated_title_command, 'text_nonumber'); } } elsif ($target_node->{'cmdname'} eq 'float') { if (!$self->get_conf('XREF_USE_FLOAT_LABEL')) { if (in_string($self)) { # not tested $name = $self->command_text($target_root, 'string'); } else { $name = $self->command_text($target_root); } } if (!defined($name) or $name eq '') { if (defined($args->[0]->{'monospace'})) { $name = $args->[0]->{'monospace'}; } else { $name = ''; } } } elsif (!$self->get_conf('XREF_USE_NODE_NAME_ARG') and (defined($self->get_conf('XREF_USE_NODE_NAME_ARG')) or !in_preformatted_context($self)) # this condition avoids infinite recursions, example with # USE_NODES=0 and node referring to the section and section referring # to the node and not _command_is_in_referred_command_stack($self, $target_root)) { if ($self->get_conf('xrefautomaticsectiontitle') eq 'on') { if (in_string($self)) { $name = $self->command_name($target_root, 'string'); } else { $name = $self->command_name($target_root, 'text_nonumber'); } } elsif (in_string($self)) { $name = $self->command_text($target_root, 'string'); } else { $name = $self->command_text($target_root, 'text_nonumber'); } #die "$target_root $target_root->{'normalized'}" if (!defined($name)); } elsif (defined($args->[0]->{'monospace'})) { $name = $args->[0]->{'monospace'}; } else { $name = ''; } } my $reference = $name; if (defined($href)) { $reference = $self->html_attribute_class('a', [$cmdname]) ." href=\"$href\">$name"; } my $substrings = { 'reference_name' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $reference}) }; if ($cmdname eq 'pxref') { $tree = $self->cdt('see {reference_name}', $substrings); } elsif ($cmdname eq 'xref') { $tree = $self->cdt('See {reference_name}', $substrings); } elsif ($cmdname eq 'ref' or $cmdname eq 'link') { $tree = $self->cdt('{reference_name}', $substrings); } } else { # external reference, including unknown node without file nor book # We setup a label_element based on the node argument and not directly the # node argument to be able to use the $file argument my $label_element; my $node_content; if (defined($arg_node) and exists($arg_node->{'extra'}) and exists($arg_node->{'extra'}->{'node_content'})) { $node_content = $arg_node->{'extra'}->{'node_content'}; $label_element = Texinfo::TreeElement::new( {'extra' => {'node_content' => $node_content}}); if (exists($arg_node->{'extra'}->{'normalized'})) { $label_element->{'extra'}->{'normalized'} = $arg_node->{'extra'}->{'normalized'}; } } # file argument takes precedence over the file in the node (file)node entry if (defined($file)) { if (!$label_element) { $label_element = Texinfo::TreeElement::new({'extra' => {}}); } elsif (!exists($label_element->{'extra'})) { $label_element->{'extra'} = {}; } $label_element->{'extra'}->{'manual_content'} = $file_arg->{'arg_tree'}; } elsif (defined($arg_node) and exists($arg_node->{'extra'}) and exists($arg_node->{'extra'}->{'manual_content'})) { my $manual_content = $arg_node->{'extra'}->{'manual_content'}; if (!defined($label_element)) { $label_element = Texinfo::TreeElement::new({'extra' => {}}); } elsif (!exists($label_element->{'extra'})) { $label_element->{'extra'} = {}; } $label_element->{'extra'}->{'manual_content'} = $manual_content; $file = $self->convert_tree_in_code_context($manual_content, 'node file in ref'); } if (!defined($name)) { if (defined($book)) { if (defined($node_content)) { my $node_name = $self->convert_tree_in_code_context($node_content, 'node in ref'); if (defined($node_name) and $node_name ne 'Top') { $name = $node_name; } } } else { if (defined($label_element)) { $name = $self->command_text($label_element); } if (!defined($name) and defined($args->[0]) and defined($args->[0]->{'monospace'}) and $args->[0]->{'monospace'} ne '' and $args->[0]->{'monospace'} ne 'Top') { # unknown node (and no book nor file) or @inforef without file $name = $args->[0]->{'monospace'}; } } } my $href; if (defined($label_element) and !in_string($self)) { $href = $self->command_href($label_element, undef, $command); } my $reference; my $book_reference; if (defined($href)) { # attribute to distiguish links to Texinfo manuals from other links # and to provide manual name of target my $manual_name_attribute = ''; if (defined($file) and not $self->get_conf('NO_CUSTOM_HTML_ATTRIBUTE')) { $manual_name_attribute = "data-manual=\"". &{$self->formatting_function('format_protect_text')}($self, $file)."\" "; } if (defined($name)) { $reference = "$name"; } elsif (defined($book)) { $book_reference = "$book"; } } my $substrings; if (defined($book) and defined($reference)) { $substrings = {'reference' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $reference}), 'book' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $book })}; if ($cmdname eq 'pxref') { $tree = $self->cdt('see {reference} in @cite{{book}}', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See {reference} in @cite{{book}}', $substrings); } else { # @ref $tree = $self->cdt('{reference} in @cite{{book}}', $substrings); } } elsif (defined($book_reference)) { $substrings = { 'book_reference' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $book_reference })}; if ($cmdname eq 'pxref') { $tree = $self->cdt('see @cite{{book_reference}}', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See @cite{{book_reference}}', $substrings); } else { # @ref $tree = $self->cdt('@cite{{book_reference}}', $substrings); } } elsif (defined($book) and defined($name)) { $substrings = { 'section' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $name}), 'book' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $book })}; if ($cmdname eq 'pxref') { $tree = $self->cdt('see `{section}\' in @cite{{book}}', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See `{section}\' in @cite{{book}}', $substrings); } else { # @ref $tree = $self->cdt('`{section}\' in @cite{{book}}', $substrings); } } elsif (defined($book)) { # should seldom or even never happen $substrings = {'book' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $book })}; if ($cmdname eq 'pxref') { $tree = $self->cdt('see @cite{{book}}', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See @cite{{book}}', $substrings); } else { # @ref $tree = $self->cdt('@cite{{book}}', $substrings); } } elsif (defined($reference)) { $substrings = { 'reference' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $reference}) }; if ($cmdname eq 'pxref') { $tree = $self->cdt('see {reference}', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See {reference}', $substrings); } else { # @ref $tree = $self->cdt('{reference}', $substrings); } } elsif (defined($name)) { $substrings = { 'section' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $name}) }; if ($cmdname eq 'pxref') { $tree = $self->cdt('see `{section}\'', $substrings); } elsif ($cmdname eq 'xref' or $cmdname eq 'inforef') { $tree = $self->cdt('See `{section}\'', $substrings); } else { # @ref $tree = $self->cdt('`{section}\'', $substrings); } } if (!defined($tree)) { # May happen if there is no argument #die "external: $cmdname, ($args), '$name' '$file' '$book' '$href' '$reference'. tree undef"; return ''; } } return $self->convert_tree($tree, "convert xref $cmdname"); } foreach my $cmdname(keys(%ref_commands)) { $default_commands_conversion{$cmdname} = \&_convert_xref_commands; } sub _convert_printindex_command($$$$) { my ($self, $cmdname, $command, $args, $content) = @_; return '' if (in_string($self)); my $index_name; if (exists($command->{'extra'}) and exists($command->{'extra'}->{'misc_args'}) and defined($command->{'extra'}->{'misc_args'}->[0])) { $index_name = $command->{'extra'}->{'misc_args'}->[0]; } else { return ''; } my $index_entries_by_letter = $self->get_converter_indices_sorted_by_letter(); if (!defined($index_entries_by_letter) or !exists($index_entries_by_letter->{$index_name}) or !scalar(@{$index_entries_by_letter->{$index_name}})) { return ''; } my $document = $self->get_info('document'); my $indices_information; my $identifiers_target; if (defined($document)) { $indices_information = $document->indices_information(); $identifiers_target = $document->labels_information(); } #foreach my $letter_entry (@{$index_entries_by_letter->{$index_name}}) { # print STDERR "IDXLETTER $letter_entry->{'letter'}\n"; # foreach my $index_entry (@{$letter_entry->{'entries'}}) { # print STDERR " ".join('|', keys(%$index_entry))."||| $index_entry->{'key'}\n"; # } #} my $index_element_id; my $current_output_unit = $self->current_output_unit(); if (defined($current_output_unit) and exists($current_output_unit->{'unit_command'})) { $index_element_id = $self->command_id($current_output_unit->{'unit_command'}); } if (!defined($index_element_id)) { my ($output_unit, $root_command) = $self->get_element_root_command_element($command); if (defined($root_command)) { $index_element_id = $self->command_id($root_command); } if (not defined($index_element_id)) { # to avoid duplicate names, use a prefix that cannot happen in anchors my $target_prefix = 't_i'; $index_element_id = $target_prefix; } } my %letter_id; my %letter_is_symbol; # First collect the links that are used in entries and in letter summaries my $symbol_idx = 0; my $normalized_letter_idx = 0; my $no_unidecode; $no_unidecode = 1 if (defined($self->get_conf('USE_UNIDECODE')) and !$self->get_conf('USE_UNIDECODE')); my $in_test; $in_test = 1 if ($self->get_conf('TEST')); foreach my $letter_entry (@{$index_entries_by_letter->{$index_name}}) { my $letter = $letter_entry->{'letter'}; my $is_symbol = $letter !~ /^\p{Alpha}/; $letter_is_symbol{$letter} = $is_symbol; my $identifier; if ($is_symbol) { $symbol_idx++; $identifier = $index_element_id . "_${index_name}_symbol-$symbol_idx"; } else { my $normalized_letter = Texinfo::Convert::NodeNameNormalization::normalize_transliterate_texinfo( Texinfo::TreeElement::new({'text' => $letter}), $in_test, $no_unidecode); my $letter_identifier = $normalized_letter; if ($normalized_letter ne $letter) { # disambiguate, as it could be another letter, case of @l, for example $normalized_letter_idx++; $letter_identifier = "${normalized_letter}-${normalized_letter_idx}"; } $identifier = $index_element_id . "_${index_name}_letter-${letter_identifier}"; } $letter_id{$letter} = $identifier; } # FIXME not part of the API _new_document_context($self, $cmdname); my $rule = $self->get_conf('DEFAULT_RULE'); $rule = '' if (!defined($rule)); my %formatted_letters; # Next do the entries to determine the letters that are not empty my @letter_entries; my $result_index_entries = ''; foreach my $letter_entry (@{$index_entries_by_letter->{$index_name}}) { my $letter = $letter_entry->{'letter'}; my $entries_text = ''; my $entry_nr = -1; my $first_entry; # since we normalize, a different formatting will not trigger a new # formatting of the main entry or a subentry level. This is the # same for Texinfo TeX my @prev_normalized_entry_levels; foreach my $index_entry_ref (@{$letter_entry->{'entries'}}) { $entry_nr++; my $main_entry_element = $index_entry_ref->{'entry_element'}; next if ($self->get_conf('NO_TOP_NODE_OUTPUT') and exists($main_entry_element->{'extra'}->{'element_node'}) and $main_entry_element->{'extra'}->{'element_node'} eq 'Top'); # to avoid double error messages, call # convert_tree_new_formatting_context below with a multiple_pass # argument if an entry was already formatted once, for example if # there are multiple printindex. my $formatted_index_entry_nr = $self->get_shared_conversion_state('printindex', 'formatted_index_entries', $index_entry_ref); $formatted_index_entry_nr = 0 if (!defined($formatted_index_entry_nr)); $formatted_index_entry_nr++; $self->set_shared_conversion_state('printindex', 'formatted_index_entries', $index_entry_ref, $formatted_index_entry_nr); my $entry_content_element = Texinfo::Common::index_content_element($main_entry_element); my $in_code = 0; $in_code = 1 if ($indices_information->{$index_entry_ref->{'index_name'}}->{'in_code'}); my $entry_ref_tree = Texinfo::TreeElement::new({'contents' => [$entry_content_element]}); # determine the trees and normalized main entry and subentries, to be # compared with the previous line normalized entries to determine # what is already formatted as part of the previous lines and # what levels should be added. The last level is always formatted. my @new_normalized_entry_levels; my @entry_trees; # NOTE it seems that subentry is not followed in convert_to_normalized $new_normalized_entry_levels[0] = uc(Texinfo::Convert::NodeNameNormalization::convert_to_normalized( $entry_ref_tree)); $entry_trees[0] = $entry_ref_tree; my $subentry_level = 1; my $subentries_max_level = 2; my @subentries_list; Texinfo::Common::collect_subentries($main_entry_element, \@subentries_list); if (scalar(@subentries_list)) { foreach my $subentry (@subentries_list) { my $subentry_tree; my $line_arg = $subentry->{'contents'}->[0]; if (exists($line_arg->{'contents'}) and scalar(@{$line_arg->{'contents'}})) { my @contents; foreach my $content (@{$line_arg->{'contents'}}) { push @contents, $content unless (exists($content->{'cmdname'}) and $content->{'cmdname'} eq 'subentry'); } $subentry_tree = Texinfo::TreeElement::new({'contents' => \@contents}); } if ($subentry_level >= $subentries_max_level) { # at the max, concatenate the remaining subentries my $other_subentries_tree = Texinfo::Convert::Utils::comma_index_subentries_tree($subentry); if (defined($other_subentries_tree)) { if (defined($subentry_tree)) { push @{$subentry_tree->{'contents'}}, @{$other_subentries_tree->{'contents'}}; } else { $subentry_tree = Texinfo::TreeElement::new( {'contents' => [@{$other_subentries_tree->{'contents'}}]}); } } } elsif (defined($subentry_tree)) { push @new_normalized_entry_levels, uc(Texinfo::Convert::NodeNameNormalization::convert_to_normalized( $subentry_tree)); } push @entry_trees, $subentry_tree; $subentry_level++; last if ($subentry_level > $subentries_max_level); } } my $formatting_context; if ($in_code) { $formatting_context = $CTXF_code; } #print STDERR join('|', @new_normalized_entry_levels)."\n"; # level/index of the last entry my $last_entry_level = $subentry_level -1; my $with_new_formatted_entry = 0; # format the leading entries when there are subentries (all entries # except the last one), and when there is not such a subentry already # formatted on the previous lines. # Each on a line with increasing indentation, no hyperlink. for (my $level = 0; $level < $last_entry_level; $level++) { # skip levels already formatted as part of the previous lines if (!$with_new_formatted_entry and scalar(@prev_normalized_entry_levels) > $level and $prev_normalized_entry_levels[$level] eq $new_normalized_entry_levels[$level]) { next; } $with_new_formatted_entry = 1; my $convert_info = "index $index_name l $letter index entry $entry_nr subentry $level"; my $entry; if ($formatted_index_entry_nr > 1) { # call with multiple_pass argument $entry = $self->convert_tree_new_formatting_context($entry_trees[$level], $convert_info, $formatting_context, "index-formatted-$formatted_index_entry_nr"); } else { if ($in_code) { $entry = $self->convert_tree_in_code_context($entry_trees[$level], $convert_info); } else { $entry = $self->convert_tree($entry_trees[$level], $convert_info); } } $entry = '' .$entry .'' if ($in_code); my @td_entry_classes = (); if ($level == 0) { push @td_entry_classes, "$cmdname-index-entry"; } elsif ($level > 0) { # indent push @td_entry_classes, "$cmdname-index-subentry-level-$level"; } $entries_text .= '' # TODO same class used for leading entry rows here and # last element of the entry with the href below. Could be different. .$self->html_attribute_class('td', \@td_entry_classes).'>' . $entry . '' # empty cell, no section for this line . "\n"; } # last entry, always converted, associated to chapter/node and # with an hyperlink or to seeentry/seealso my $entry_tree = $entry_trees[$last_entry_level]; my $referred_entry; my $seeentry = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seeentry'); if (defined($seeentry)) { $referred_entry = $seeentry; } else { $referred_entry = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seealso'); } # index entry with @seeentry or @seealso if (defined($referred_entry)) { my $entry; # for @seealso, to appear where chapter/node ususally appear my $reference = ''; my $delimiter = ''; my $section_class; if (defined($seeentry)) { my $result_tree; if ($in_code) { $result_tree # TRANSLATORS: redirect to another index entry # TRANSLATORS: @: is discardable and is used to avoid a msgfmt error = $self->cdt('@code{{main_index_entry}}, @emph{See@:} @code{{seeentry}}', {'main_index_entry' => $entry_tree, 'seeentry' => $referred_entry}); } else { $result_tree # TRANSLATORS: redirect to another index entry # TRANSLATORS: @: is discardable and used to avoid a msgfmt error = $self->cdt('{main_index_entry}, @emph{See@:} {seeentry}', {'main_index_entry' => $entry_tree, 'seeentry' => $referred_entry}); } my $convert_info = "index $index_name l $letter index entry $entry_nr seeentry"; if ($formatted_index_entry_nr > 1) { # call with multiple_pass argument $entry = $self->convert_tree_new_formatting_context($result_tree, $convert_info, undef, "index-formatted-$formatted_index_entry_nr"); } else { $entry = $self->convert_tree($result_tree, $convert_info); } $section_class = "$cmdname-index-see-entry-section"; } else { my $reference_tree; if ($in_code) { # TRANSLATORS: refer to another index entry $reference_tree = $self->cdt('@emph{See also} @code{{see_also_entry}}', {'see_also_entry' => $referred_entry}); } else { # TRANSLATORS: refer to another index entry $reference_tree = $self->cdt('@emph{See also} {see_also_entry}', {'see_also_entry' => $referred_entry}); } my $conv_str_entry = "index $index_name l $letter index entry $entry_nr (with seealso)"; my $conv_str_reference = "index $index_name l $letter index entry $entry_nr seealso"; if ($formatted_index_entry_nr > 1) { # call with multiple_pass argument $entry = $self->convert_tree_new_formatting_context($entry_tree, $conv_str_entry, $formatting_context, "index-formatted-$formatted_index_entry_nr"); $reference = $self->convert_tree_new_formatting_context($reference_tree, $conv_str_reference, undef, "index-formatted-$formatted_index_entry_nr"); } else { if ($in_code) { $entry = $self->convert_tree_in_code_context($entry_tree, $conv_str_entry); } else { $entry = $self->convert_tree($entry_tree, $conv_str_entry); } $reference = $self->convert_tree($reference_tree, $conv_str_reference); } $entry = '' .$entry .'' if ($in_code); $delimiter = $self->get_conf('INDEX_ENTRY_COLON'); $section_class = "$cmdname-index-see-also"; } my @td_entry_classes = (); if (defined($seeentry)) { push @td_entry_classes, "$cmdname-index-see-entry"; } if ($last_entry_level == 0) { push @td_entry_classes, "$cmdname-index-entry"; } elsif ($last_entry_level > 0) { push @td_entry_classes, "$cmdname-index-subentry-level-$last_entry_level"; } $entries_text .= '' .$self->html_attribute_class('td', \@td_entry_classes).'>' . $entry . $delimiter . '' .$self->html_attribute_class('td', [$section_class]).'>'; $entries_text .= $reference; $entries_text .= "\n"; @prev_normalized_entry_levels = @new_normalized_entry_levels; } else { my $entry; if (!defined($entry_tree)) { # can happen at least with an empty subentry $entry = ''; } else { my $convert_info = "index $index_name l $letter index entry $entry_nr"; if ($formatted_index_entry_nr > 1) { # call with multiple_pass argument $entry = $self->convert_tree_new_formatting_context($entry_tree, $convert_info, $formatting_context, "index-formatted-$formatted_index_entry_nr"); } else { if ($in_code) { $entry = $self->convert_tree_in_code_context($entry_tree, $convert_info); } else { $entry = $self->convert_tree($entry_tree, $convert_info); } } } next if ($entry !~ /\S/ and $last_entry_level == 0); if (!defined($first_entry)) { $first_entry = $index_entry_ref; } @prev_normalized_entry_levels = @new_normalized_entry_levels; $entry = '' .$entry .'' if ($in_code); my $target_element; if (exists($index_entry_ref->{'entry_associated_element'})) { $target_element = $index_entry_ref->{'entry_associated_element'}; } else { $target_element = $main_entry_element; } my $entry_href = $self->command_href($target_element); my $formatted_entry = "$entry"; my @td_entry_classes = (); if ($last_entry_level == 0) { push @td_entry_classes, "$cmdname-index-entry"; } elsif ($last_entry_level > 0) { # subentry push @td_entry_classes, "$cmdname-index-subentry-level-$last_entry_level"; } $entries_text .= '' .$self->html_attribute_class('td', \@td_entry_classes).'>' . $formatted_entry . $self->get_conf('INDEX_ENTRY_COLON') . ''; my $associated_command; if ($self->get_conf('NODE_NAME_IN_INDEX')) { my $associated_command_id = $main_entry_element->{'extra'}->{'element_node'}; if (defined($associated_command_id) and defined($identifiers_target)) { $associated_command = $identifiers_target->{$associated_command_id}; } if (!defined($associated_command)) { $associated_command = $self->command_node($target_element); } if (!defined($associated_command) # do not warn if the entry is in a special region, like titlepage and not $main_entry_element->{'extra'}->{'element_region'} and $formatted_index_entry_nr == 1) { # NOTE $self->in_multiple_conversions() is not checked as printindex # should not happen in multiple tree conversion, but the error message # is printed for the first entry formatting only. $self->converter_line_warn( sprintf( __("entry for index `%s' for \@printindex %s outside of any node"), $index_entry_ref->{'index_name'}, $index_name), $main_entry_element->{'source_info'}); } } if (!defined($associated_command)) { $associated_command = $self->command_root_element_command($target_element); if (!defined($associated_command)) { # Use Top if not associated command found $associated_command = $self->global_direction_unit('Top')->{'unit_command'}; # NOTE the warning here catches the most relevant cases of # index entry that is not associated to the right command, which # are very few in the test suite. There is also a warning in the # parser with a much broader scope with possible overlap, but the # overlap is not a problem. # NODE_NAME_IN_INDEX may be undef even with USE_NODES set if the # converter is called as convert() as in the test suite if (defined($self->get_conf('NODE_NAME_IN_INDEX')) and not $self->get_conf('NODE_NAME_IN_INDEX') # do not warn if the entry is in a special region, like titlepage and not $main_entry_element->{'extra'}->{'element_region'} and $formatted_index_entry_nr == 1) { # NOTE $self->in_multiple_conversions() is not checked as printindex # should not happen in multiple tree conversion, but the error message # is printed for the first entry formatting only. # NOTE the index entry may be associated to a node in that case. $self->converter_line_warn( sprintf( __("entry for index `%s' for \@printindex %s outside of any section"), $index_entry_ref->{'index_name'}, $index_name), $main_entry_element->{'source_info'}); } } } $entries_text .= $self->html_attribute_class('td', ["$cmdname-index-section"]).'>'; if (defined($associated_command)) { my $associated_command_href = $self->command_href($associated_command); my $associated_command_text = $self->command_text($associated_command); if (defined($associated_command_href)) { $entries_text .= "" ."$associated_command_text"; } elsif (defined($associated_command_text)) { $entries_text .= $associated_command_text; } } $entries_text .= "\n"; } } # a letter and associated indice entries if ($entries_text ne '') { my $formatted_letter; my $letter_command; # may not be defined if there are only seeentry/seealso if (defined($first_entry)) { my $letter_text; ($letter_text, $letter_command) = Texinfo::Indices::index_entry_first_letter_text_or_command( $first_entry); } if (defined($letter_command) and !exists($accent_commands{$letter_command->{'cmdname'}}) and $letter_command->{'cmdname'} ne 'U' # special case, the uppercasing of that command is not done # if as a command, while it is done correctly in $letter and $letter_command->{'cmdname'} ne 'ss') { my $cmdname = $letter_command->{'cmdname'}; if (exists($letter_no_arg_commands{$cmdname}) and exists($letter_no_arg_commands{uc($cmdname)})) { $letter_command = Texinfo::TreeElement::new({'cmdname' => uc($cmdname)}); } $formatted_letter = $self->convert_tree($letter_command, "index letter $letter command"); } else { $formatted_letter = &{$self->formatting_function('format_protect_text')}($self, $letter); } $formatted_letters{$letter} = $formatted_letter; $result_index_entries .= '' . $self->html_attribute_class('th', ["index-letter-header-$cmdname", "$index_name-letter-header-$cmdname"]) ." colspan=\"2\" id=\"$letter_id{$letter}\">".$formatted_letter . "\n" . $entries_text . "${rule}\n"; push @letter_entries, $letter_entry; } } # Do the summary letters linking to the letters done above my @non_alpha = (); my @alpha = (); foreach my $letter_entry (@letter_entries) { my $letter = $letter_entry->{'letter'}; my $summary_letter_link = $self->html_attribute_class('a',["summary-letter-$cmdname"]) ." href=\"#$letter_id{$letter}\">".$formatted_letters{$letter} .''; if ($letter_is_symbol{$letter}) { push @non_alpha, $summary_letter_link; } else { push @alpha, $summary_letter_link; } } if (scalar(@non_alpha) + scalar(@alpha) == 0) { _pop_document_context($self); return ''; } my $non_breaking_space = $self->get_info('non_breaking_space'); # Format the summary letters my $join = ''; my $non_alpha_text = ''; my $alpha_text = ''; if (scalar(@non_alpha) + scalar(@alpha) > 1) { $join = " $non_breaking_space \n".$self->get_info('line_break_element')."\n" if (scalar(@non_alpha) and scalar(@alpha)); if (scalar(@non_alpha)) { $non_alpha_text = join("\n $non_breaking_space \n", @non_alpha) . "\n"; } if (scalar(@alpha)) { $alpha_text = join("\n $non_breaking_space \n", @alpha) . "\n $non_breaking_space \n"; } } my $result = $self->html_attribute_class('div', [$cmdname, "$index_name-$cmdname"]).">\n"; # format the summary if (scalar(@non_alpha) + scalar(@alpha) > 1) { my $summary_header = $self->html_attribute_class('table', ["index-letters-header-$cmdname", "$index_name-letters-header-$cmdname"]).'>' # TRANSLATORS: before list of letters and symbols grouping index entries . $self->convert_tree($self->cdt('Jump to'), 'Tr letters header text') . ": $non_breaking_space " . $non_alpha_text . $join . $alpha_text . "\n"; $result .= $summary_header; } # now format the index entries $result .= $self->html_attribute_class('table', ["index-entries-$cmdname", "$index_name-entries-$cmdname"]).">\n"; $result .= "${rule}\n"; $result .= $result_index_entries; $result .= "\n"; _pop_document_context($self); if (scalar(@non_alpha) + scalar(@alpha) > 1) { my $summary_footer = $self->html_attribute_class('table', ["index-letters-footer-$cmdname", "$index_name-letters-footer-$cmdname"]).'>' # TRANSLATORS: before list of letters and symbols grouping index entries . $self->convert_tree($self->cdt('Jump to'), 'Tr letters footer text') . ": $non_breaking_space " . $non_alpha_text . $join . $alpha_text . "\n"; $result .= $summary_footer } return $result . "\n"; } $default_commands_conversion{'printindex'} = \&_convert_printindex_command; sub _convert_informative_command($$$) { my ($self, $cmdname, $command) = @_; return '' if (in_string($self)); Texinfo::Common::set_informative_command_value($self, $command); return ''; } foreach my $informative_command (@informative_global_commands) { $default_commands_conversion{$informative_command} = \&_convert_informative_command; } sub _convert_contents_command($$$) { my ($self, $cmdname, $command) = @_; return '' if (in_string($self)); $cmdname = 'shortcontents' if ($cmdname eq 'summarycontents'); Texinfo::Common::set_informative_command_value($self, $command); my $document = $self->get_info('document'); my $sections_list; if (defined($document)) { $sections_list = $document->sections_list(); } if ($self->get_conf('CONTENTS_OUTPUT_LOCATION') eq 'inline' and ($cmdname eq 'contents' or $cmdname eq 'shortcontents') and $self->get_conf($cmdname) and defined($sections_list) and scalar(@{$sections_list}) > 1) { return _contents_inline_element($self, $cmdname, $command); } return ''; } foreach my $contents_command (@contents_commands) { $default_commands_conversion{$contents_command} = \&_convert_contents_command; } sub _convert_def_command($$$$$) { my ($self, $cmdname, $command, $args, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); my @classes; if ($cmdname ne 'defblock') { # The def* class is used for the def line, the first-def* class is # used for the whole block. my $command_name; if (exists($Texinfo::Common::def_aliases{$cmdname})) { $command_name = $Texinfo::Common::def_aliases{$cmdname}; push @classes, "first-$cmdname-alias-first-$command_name"; } else { $command_name = $cmdname; } unshift @classes, "first-$command_name"; } else { push @classes, $cmdname; } push @classes, 'def-block'; if (!$self->get_conf('DEF_TABLE')) { return $self->html_attribute_class('dl', \@classes).">\n" . $content ."\n"; } else { return $self->html_attribute_class('table', \@classes).">\n" . $content . "\n"; } } # Keys are tree element types, values are function references to convert # elements of that type. Can be overridden accessing # Texinfo::Config::GNUT_get_types_conversion, setup by # Texinfo::Config::texinfo_register_type_formatting() my %default_types_conversion; foreach my $cmdname (keys(%def_commands), 'defblock') { if (exists($line_commands{$cmdname})) { $default_commands_conversion{$cmdname} = \&_convert_def_line_type; } else { $default_commands_conversion{$cmdname} = \&_convert_def_command; } } # associate same formatting function for @small* command # as for the associated @-command foreach my $small_command (keys(%small_block_associated_command)) { $default_commands_conversion{$small_command} = $default_commands_conversion{$small_block_associated_command{$small_command}}; } # Can be used to check that all the relevant commands are converted if (0) { foreach my $cmdname (keys(%Texinfo::Common::all_commands)) { if (!exists($default_commands_conversion{$cmdname})) { # should be @if* @*index and item_LINE if ($cmdname =~ /^if/ or $cmdname =~ /index$/ or $cmdname eq 'item_LINE') {} else { warn "MISSING $cmdname\n"; } } } } sub _open_node_part_command($$$) { my ($self, $cmdname, $element) = @_; if ($self->get_conf('NO_TOP_NODE_OUTPUT')) { my $in_skipped_node_top = $self->get_shared_conversion_state('top', 'in_skipped_node_top'); $in_skipped_node_top = 0 if (!defined($in_skipped_node_top)); my $node_element; if ($cmdname eq 'node') { $node_element = $element; } elsif ($cmdname eq 'part') { my $document = $self->get_info('document'); if (defined($document)) { my $sections_list = $document->sections_list(); my $part_relations = $sections_list->[$element->{'extra'}->{'section_number'} -1]; if (exists($part_relations->{'part_following_node'})) { $node_element = $part_relations->{'part_following_node'}->{'element'}; } } } if (defined($node_element) or $cmdname eq 'part') { if (defined($node_element) and exists($node_element->{'extra'}) and exists($node_element->{'extra'}->{'normalized'}) and $node_element->{'extra'}->{'normalized'} eq 'Top') { $in_skipped_node_top = 1; $self->set_shared_conversion_state('top', 'in_skipped_node_top', $in_skipped_node_top); } elsif ($in_skipped_node_top == 1) { $in_skipped_node_top = -1; $self->set_shared_conversion_state('top', 'in_skipped_node_top', $in_skipped_node_top); } } } return ''; } $default_commands_open{'node'} = \&_open_node_part_command; $default_commands_open{'part'} = \&_open_node_part_command; sub _open_quotation_titlepage_stack($$) { my ($self, $element_authors_number) = @_; my $quotation_titlepage_nr = $self->get_shared_conversion_state('quotation', 'quotation_titlepage_stack'); $quotation_titlepage_nr = 0 if (!defined($quotation_titlepage_nr)); $quotation_titlepage_nr++; $self->set_shared_conversion_state('quotation', 'quotation_titlepage_stack', $quotation_titlepage_nr); $self->set_shared_conversion_state('quotation', 'element_authors_number', $quotation_titlepage_nr, $element_authors_number); } sub _open_quotation_command($$$) { my ($self, $cmdname, $command) = @_; my $formatted_quotation_arg_to_prepend; # arguments_line type element my $arguments_line = $command->{'contents'}->[0]; my $block_line_args = $arguments_line->{'contents'}->[0]; if (exists($block_line_args->{'contents'}) and scalar(@{$block_line_args->{'contents'}})) { $formatted_quotation_arg_to_prepend = $self->convert_tree($self->cdt('@b{{quotation_arg}:} ', {'quotation_arg' => $block_line_args}), "open $cmdname prepended arg"); } $self->register_pending_formatted_inline_content($cmdname, $formatted_quotation_arg_to_prepend); _open_quotation_titlepage_stack($self, 0); return ''; } $default_commands_open{'quotation'} = \&_open_quotation_command; # associate same opening function for @small* command # as for the associated @-command foreach my $small_command (keys(%small_block_associated_command)) { if (exists($default_commands_open{$small_block_associated_command{$small_command}})) { $default_commands_open{$small_command} = $default_commands_open{$small_block_associated_command{$small_command}}; } } # Keys are output units types, values are function references to convert # output units of that type. Can be overridden accessing # Texinfo::Config::GNUT_get_output_units_conversion, setup by # Texinfo::Config::texinfo_register_output_unit_formatting() my %default_output_units_conversion; sub default_output_unit_conversion($$) { my ($self, $type) = @_; return $default_output_units_conversion{$type}; } sub output_unit_conversion($$) { my ($self, $type) = @_; return $self->{'output_units_conversion'}->{$type}; } sub default_type_conversion($$) { my ($self, $type) = @_; return $default_types_conversion{$type}; } sub type_conversion($$) { my ($self, $type) = @_; return $self->{'types_conversion'}->{$type}; } my %default_types_open; sub default_type_open($$) { my ($self, $type) = @_; return $default_types_open{$type}; } # Ignored commands foreach my $type ( 'ignorable_spaces_after_command', 'ignorable_spaces_before_command', 'spaces_at_end', 'spaces_before_paragraph', # may be better not to ignore spaces when a : is postpended # and the user really wants a space #'space_at_end_menu_node', 'spaces_after_close_brace') { $default_types_conversion{$type} = undef; } foreach my $type ( 'postamble_after_end', 'preamble_before_beginning', 'preamble_before_setfilename', 'arguments_line') { $default_types_conversion{$type} = undef; } sub _convert_paragraph_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); $content = $self->get_associated_formatted_inline_content($element).$content; if (paragraph_number($self) == 1) { my $in_format = top_block_command($self); if ($in_format) { # no first paragraph in those environment to avoid extra spacing if ($in_format eq 'itemize' or $in_format eq 'enumerate' or $in_format eq 'multitable' # this should only happen if in @nodedescriptionblock, otherwise # there are no paragraphs, but preformatted or $in_format eq 'menu') { return $content; } } } return $content if (in_string($self)); if ($content =~ /\S/) { my $align = $self->in_align(); if ($align and exists($HTML_align_commands{$align})) { return $self->html_attribute_class('p', [$align.'-paragraph']).">" .$content."

    "; } else { return "

    ".$content."

    "; } } else { return ''; } } $default_types_conversion{'paragraph'} = \&_convert_paragraph_type; sub _open_inline_container_type($$$) { my ($self, $type, $element) = @_; my $pending_formatted = $self->get_pending_formatted_inline_content(); if (defined($pending_formatted)) { $self->associate_pending_formatted_inline_content($element, $pending_formatted); } return ''; } $default_types_open{'paragraph'} = \&_open_inline_container_type; $default_types_open{'preformatted'} = \&_open_inline_container_type; sub _preformatted_class($) { my $self = shift; my $pre_class; my $pre_classes = $self->preformatted_classes_stack(); foreach my $class (@$pre_classes) { $pre_class = $class unless (defined($pre_class) and exists($preformatted_code_commands{$pre_class}) and !(exists($preformatted_code_commands{$class}) or $class eq 'menu')); } return $pre_class.'-preformatted'; } sub _convert_preformatted_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); $content = $self->get_associated_formatted_inline_content($element).$content; return '' if ($content eq ''); if (top_block_command($self) eq 'multitable') { $content =~ s/^\s*//; $content =~ s/\s*$//; } if (in_string($self)) { return $content; } my $pre_class; # menu_entry_description is always in a preformatted container # in the tree, as the whole menu is meant to be an # environment where spaces and newlines are preserved. if (exists($element->{'parent'}->{'type'}) and $element->{'parent'}->{'type'} eq 'menu_entry_description') { if (!inside_preformatted($self)) { # If not in preformatted block command, # we don't preserve spaces and newlines in menu_entry_description, # instead the whole menu_entry is in a table, so no
     in that situation
          return $content;
        } else {
          # if directly in description, we want to avoid the linebreak that
          # comes with pre being a block level element, so set a special class
          $pre_class = 'menu-entry-description-preformatted';
        }
      }
    
      $content =~ s/^\n/\n\n/; # a newline immediately after a 
     is ignored.
    
      $pre_class = _preformatted_class($self) if (!defined($pre_class));
      my $result = $self->html_attribute_class('pre', [$pre_class]).'>'
                                                       . $content . '
    '; return $result; } $default_types_conversion{'preformatted'} = \&_convert_preformatted_type; sub _convert_balanced_braces_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return $content; } $default_types_conversion{'balanced_braces'} = \&_convert_balanced_braces_type; # use the type and not the index commands names, as they are diverse and # can be dynamically added, so it is difficult to use as selector for output # formatting. The command name can be obtained here as $element->{'cmdname'}. sub _convert_index_entry_command_type($$$$) { my ($self, $type, $element, $content) = @_; my $index_id = $self->command_id($element); # since in_multiple_conversions returning true while multi_expanded_region # returns undef can only happen when expanding target and heading commands # text, in which there should not be index commands, using one function # or the other in the conddition should not give a different result. if (defined($index_id) and $index_id ne '' and !in_multiple_conversions($self) and !in_string($self)) { my $result = &{$self->formatting_function('format_separate_anchor')}($self, $index_id, 'index-entry-id'); $result .= "\n" unless (in_preformatted_context($self)); return $result; } return ''; } $default_types_conversion{'index_entry_command'} = \&_convert_index_entry_command_type; sub _convert_definfoenclose_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); # TODO add a span to mark the original command as a class? # Not to be done as long as definfoenclose is deprecated. return &{$self->formatting_function('format_protect_text')}($self, $element->{'extra'}->{'begin'}) . $content . &{$self->formatting_function('format_protect_text')}($self, $element->{'extra'}->{'end'}); } $default_types_conversion{'definfoenclose_command'} = \&_convert_definfoenclose_type; # Note: has an XS override sub _entity_text { my $text = shift; $text =~ s/---/\&mdash\;/g; $text =~ s/--/\&ndash\;/g; $text =~ s/``/\&ldquo\;/g; $text =~ s/''/\&rdquo\;/g; $text =~ s/'/\&rsquo\;/g; $text =~ s/`/\&lsquo\;/g; return $text; } sub _convert_text($$$$) { my ($self, $type, $element, $text) = @_; if (in_verbatim($self)) { # API info: using the API to allow for customization would be: #return &{$self->formatting_function('format_protect_text')}($self, $text); return _default_format_protect_text($self, $text); } return $text if (in_raw($self)); $text = uc($text) if (in_upper_case($self)); # API info: using the API to allow for customization would be: #$text = &{$self->formatting_function('format_protect_text')}($self, $text); $text = _default_format_protect_text($self, $text); # API info: for efficiency, we cache the result of the calls to configuration # in $self->{'use_unicode_text'}. # API code conforming would be: #if ($self->get_conf('OUTPUT_CHARACTERS') # and $self->get_conf('OUTPUT_ENCODING_NAME') # and $self->get_conf('OUTPUT_ENCODING_NAME') eq 'utf-8') { if ($self->{'use_unicode_text'}) { $text = Texinfo::Convert::Unicode::unicode_text($text, (in_code($self) or in_math($self))); } elsif (!in_code($self) and !in_math($self)) { if ($self->get_conf('USE_NUMERIC_ENTITY')) { $text = $self->xml_format_text_with_numeric_entities($text); } elsif ($self->get_conf('USE_ISO')) { $text = _entity_text($text); } else { $text =~ s/``/"/g; $text =~ s/''/"/g; $text =~ s/---/\x{1F}/g; $text =~ s/--/-/g; $text =~ s/\x{1F}/--/g; } } return $text if (in_preformatted_context($self)); if (in_non_breakable_space($self)) { my $non_breaking_space = $self->get_info('non_breaking_space'); $text =~ s/\n/ /g; $text =~ s/ +/$non_breaking_space/g; } elsif (in_space_protected($self)) { if (chomp($text)) { my $line_break_element = $self->get_info('line_break_element'); # protect spaces in line_break_element formatting. # Note that this case is theoretical right now, as it is not possible # to redefine the line_break_element and there are no spaces # in the possible values. However this could be a deficiency of the API, # it could be better to be able to redefine line_break_element $line_break_element =~ s/ /\x{1F}/g; $text .= $line_break_element; } # Protect spaces within text my $non_breaking_space = $self->get_info('non_breaking_space'); $text =~ s/ /$non_breaking_space/g; # Revert protected spaces in leading html attribute $text =~ s/\x{1F}/ /g; } return $text; } $default_types_conversion{'text'} = \&_convert_text; sub _css_string_convert_text($$$$) { my ($self, $type, $element, $text) = @_; $text = uc($text) if (in_upper_case($self)); # need to hide \ otherwise it is protected in protect_text if (!in_code($self) and !in_math($self)) { $text =~ s/---/\x{1F}2014 /g; $text =~ s/--/\x{1F}2013 /g; $text =~ s/``/\x{1F}201C /g; $text =~ s/''/\x{1F}201D /g; $text =~ s/'/\x{1F}2019 /g; $text =~ s/`/\x{1F}2018 /g; } $text = &{$self->formatting_function('format_protect_text')}($self, $text); $text =~ s/\x{1F}/\\/g; return $text; } $default_css_string_types_conversion{'text'} = \&_css_string_convert_text; sub _simplify_text_for_comparison($) { my $text = shift; $text =~ s/[^\p{Word}]//g; return $text; } sub _convert_untranslated_def_line_arg_type($$$$) { my ($self, $type, $element, $content) = @_; my $translated; my $category_text = $element->{'contents'}->[0]->{'text'}; if (exists($element->{'extra'}) and exists($element->{'extra'}->{'translation_context'})) { $translated = $self->pcdt($element->{'extra'}->{'translation_context'}, $category_text); } else { $translated = $self->cdt($category_text); } my $result = $self->convert_tree($translated, 'translated TEXT'); return $result; } $default_types_conversion{'untranslated_def_line_arg'} = \&_convert_untranslated_def_line_arg_type; sub _convert_row_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); if ($content =~ /\S/) { my $result = '' . $content . ''; if (exists($element->{'contents'}) and $element->{'contents'}->[0]->{'cmdname'} ne 'headitem') { # if headitem, end of line added in _convert_multitable_head_type $result .= "\n"; } return $result; } else { return ''; } } $default_types_conversion{'row'} = \&_convert_row_type; sub _convert_multitable_head_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); if ($content =~ /\S/) { return '' . $content . '' . "\n"; } else { return ''; } } $default_types_conversion{'multitable_head'} = \&_convert_multitable_head_type; sub _convert_multitable_body_type($$$$) { my ($self, $type, $element, $content) = @_; return $content if (in_string($self)); if ($content =~ /\S/) { return '' . $content . '' . "\n"; } else { return ''; } } $default_types_conversion{'multitable_body'} = \&_convert_multitable_body_type; # The node is used, not the nodedescription because it is easier to # find the node in XS sub _formatted_nodedescription_nr($$) { my ($self, $node) = @_; # update the number of time the node description was formatted my $formatted_nodedescription_nr = $self->get_shared_conversion_state('nodedescription', 'formatted_nodedescriptions', $node); $formatted_nodedescription_nr = 0 if (!defined($formatted_nodedescription_nr)); $formatted_nodedescription_nr++; $self->set_shared_conversion_state('nodedescription', 'formatted_nodedescriptions', $node, $formatted_nodedescription_nr); return $formatted_nodedescription_nr; } sub _convert_menu_entry_type($$$) { my ($self, $type, $element) = @_; my $name_entry; my $menu_description; my $menu_entry_node; my $menu_entry_leading_text; my @menu_entry_separators; foreach my $arg (@{$element->{'contents'}}) { if ($arg->{'type'} eq 'menu_entry_leading_text') { $menu_entry_leading_text = $arg; } elsif ($arg->{'type'} eq 'menu_entry_name') { $name_entry = $arg; } elsif ($arg->{'type'} eq 'menu_entry_description') { $menu_description = $arg; } elsif ($arg->{'type'} eq 'menu_entry_separator') { push @menu_entry_separators, $arg; } elsif ($arg->{'type'} eq 'menu_entry_node') { $menu_entry_node = $arg; } } my $href; my $rel = ''; my $associated_title_command; my $node_description; my $long_description = 0; my $formatted_nodedescription_nr; # external node my $external_node; if (exists($menu_entry_node->{'extra'}) and exists($menu_entry_node->{'extra'}->{'manual_content'})) { $href = $self->command_href($menu_entry_node, undef, $element); $external_node = 1; # may not exist in case of menu entry node consisting only of spaces } elsif (exists($menu_entry_node->{'extra'}) and exists($menu_entry_node->{'extra'}->{'normalized'})) { my $node = $self->label_command($menu_entry_node->{'extra'}->{'normalized'}); if ($node) { my $node_relations; if ($node->{'cmdname'} eq 'node') { my $document = $self->get_info('document'); if (defined($document)) { my $nodes_list = $document->nodes_list(); $node_relations = $nodes_list->[$node->{'extra'}->{'node_number'} -1]; if (exists($node_relations->{'node_description'})) { $node_description = $node_relations->{'node_description'}; } elsif (exists($node_relations->{'node_long_description'})) { $node_description = $node_relations->{'node_long_description'}; $long_description = 1; } } } # if !NODE_NAME_IN_MENU, we pick the associated title command element if (!$self->get_conf('NODE_NAME_IN_MENU') and defined($node_relations)) { $associated_title_command = $node_relations->{'associated_title_command'}; } if (defined($associated_title_command)) { $href = $self->command_href($associated_title_command, undef, $element); } else { $href = $self->command_href($node, undef, $element); } if (exists($node->{'extra'}) and $node->{'extra'}->{'isindex'}) { # Mark the target as an index. See # http://microformats.org/wiki/existing-rel-values#HTML5_link_type_extensions $rel = ' rel="index"'; } if (defined($node_description) # not menu_description probably cannot happen and (not defined($menu_description) # empty description or (not exists($menu_description->{'contents'}) or (scalar(@{$menu_description->{'contents'}}) == 1 # preformatted inside menu_entry_description and (not (exists($menu_description->{'contents'}->[0] ->{'contents'})) or (scalar(@{$menu_description->{'contents'}->[0] ->{'contents'}}) == 1) and exists($menu_description->{'contents'}->[0] ->{'contents'}->[0]->{'text'}) and $menu_description->{'contents'}->[0] ->{'contents'}->[0]->{'text'} !~ /\S/))))) { $formatted_nodedescription_nr = _formatted_nodedescription_nr($self, $node); } } } my $html_menu_entry_index = $self->get_shared_conversion_state('menu', 'html_menu_entry_index'); $html_menu_entry_index = 0 if (!defined($html_menu_entry_index)); $html_menu_entry_index++; $self->set_shared_conversion_state('menu', 'html_menu_entry_index', $html_menu_entry_index); my $accesskey = ''; $accesskey = " accesskey=\"$html_menu_entry_index\"" if ($self->get_conf('USE_ACCESSKEY') and $html_menu_entry_index < 10); my $MENU_SYMBOL = $self->get_conf('MENU_SYMBOL'); my $MENU_ENTRY_COLON = $self->get_conf('MENU_ENTRY_COLON'); my $in_string = in_string($self); if (inside_preformatted($self) or $in_string) { my $leading_text = $menu_entry_leading_text->{'text'}; $leading_text =~ s/\*/$MENU_SYMBOL/; my $result_name_node = $leading_text; if (defined($name_entry)) { $result_name_node .= $self->convert_tree($name_entry, "menu_arg menu_entry_name preformatted"); my $name_separator = shift @menu_entry_separators; $result_name_node .= $self->convert_tree($name_separator, "menu_arg name separator preformatted"); } if (defined($menu_entry_node)) { my $name = $self->convert_tree_in_code_context($menu_entry_node, "menu_arg menu_entry_node preformatted"); if (defined($href) and !$in_string) { $result_name_node .= "$name"; } else { $result_name_node .= $name; } } if (scalar(@menu_entry_separators)) { my $node_separator = shift @menu_entry_separators; $result_name_node .= $self->convert_tree($node_separator, "menu_arg node separator preformatted"); } if (not $in_string) { my $pre_class = _preformatted_class($self); $result_name_node = $self->html_attribute_class('pre', [$pre_class]).'>' . $result_name_node . '
    '; } my $description = ''; if ($formatted_nodedescription_nr) { my $description_element; if (!$long_description) { $description_element = $node_description->{'contents'}->[0]; } else { # nodedescriptionblock $description_element = Texinfo::TreeElement::new( {'contents' => $node_description->{'contents'}}); } my $multiple_formatted; if ($formatted_nodedescription_nr > 1) { $multiple_formatted = 'preformatted-node-description-'.$formatted_nodedescription_nr; } $description .= $self->convert_tree_new_formatting_context($description_element, 'menu_arg node description preformatted', undef, $multiple_formatted, undef, 'menu'); } elsif ($menu_description) { $description .= $self->convert_tree($menu_description, 'menu_arg description preformatted'); } return $result_name_node . $description; } my $name; my $name_no_number; if (defined($associated_title_command) and defined($href)) { $name = $self->command_text($associated_title_command); if ($name ne '') { $name = "$name"; $name_no_number = $self->command_text($associated_title_command, 'text_nonumber'); } } # A leading menu symbol is only inserted if the section name is not # used since the section name usually comes with a section number (unless # NUMBER_SECTIONS is 0, or the section is unnumbered/heading/xrefname) if (!defined($name) or $name eq '') { if (defined($name_entry)) { $name = $self->convert_tree($name_entry, 'convert menu_entry_name'); } if (!defined($name) or $name eq '') { if (exists($menu_entry_node->{'extra'}) and exists($menu_entry_node->{'extra'}->{'manual_content'})) { $name = $self->command_text($menu_entry_node); } elsif (exists($menu_entry_node->{'extra'}) and exists($menu_entry_node->{'extra'}->{'node_content'})) { $name = $self->convert_tree_in_code_context( Texinfo::TreeElement::new({ 'contents' => [$menu_entry_node->{'extra'}->{'node_content'}]}), 'menu_arg name'); } else { $name = ''; } } $name =~ s/^\s*//; $name_no_number = $name; if (defined($href)) { $name = "$name"; } $name = "$MENU_SYMBOL ".$name; } my $description = ''; if ($formatted_nodedescription_nr) { my $description_element; if (!$long_description) { $description_element = $node_description->{'contents'}->[0]; } else { # nodedescriptionblock $description_element = Texinfo::TreeElement::new( {'contents' => $node_description->{'contents'}}); } my $multiple_formatted; if ($formatted_nodedescription_nr > 1) { $multiple_formatted = 'node-description-'.$formatted_nodedescription_nr; } $description = $self->convert_tree_new_formatting_context($description_element, 'menu_arg node description', undef, $multiple_formatted, undef, 'menu'); } elsif (defined($menu_description)) { $description = $self->convert_tree($menu_description, 'menu_arg description'); } my $non_breaking_space = $self->get_info('non_breaking_space'); return '' .$self->html_attribute_class('td', ['menu-entry-destination']).'>' ."$name$MENU_ENTRY_COLON" .$self->html_attribute_class('td', ['menu-entry-description']).'>' ."$description\n"; } $default_types_conversion{'menu_entry'} = \&_convert_menu_entry_type; sub _convert_menu_comment_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); if (inside_preformatted($self) or in_string($self)) { return $content; } else { return ''.$self->html_attribute_class('th', ['menu-comment']) . ' colspan="2">'.$content .''; } } $default_types_conversion{'menu_comment'} = \&_convert_menu_comment_type; sub _convert_before_item_type($$$$) { my ($self, $type, $element, $content) = @_; return '' if (!defined ($content) or $content !~ /\S/); return $content if (in_string($self)); my $top_block_command = top_block_command($self); if ($top_block_command eq 'itemize' or $top_block_command eq 'enumerate') { return '
  • '. $content .'
  • '; } elsif ($top_block_command eq 'table' or $top_block_command eq 'vtable' or $top_block_command eq 'ftable') { return '
    '. $content .'
    '."\n"; } elsif ($top_block_command eq 'multitable') { $content =~ s/^\s*//; $content =~ s/\s*$//; return ''.$content.''."\n"; } } $default_types_conversion{'before_item'} = \&_convert_before_item_type; sub _convert_table_term_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return '
    '.$content; } $default_types_conversion{'table_term'} = \&_convert_table_term_type; sub _convert_def_line_type($$$$) { my ($self, $type, $element, $content) = @_; if (in_string($self)) { # should probably never happen return &{$self->formatting_function('format_protect_text')}($self, Texinfo::Convert::Text::convert_to_text( $element, $self->{'convert_text_options'})); } my $index_label = ''; my $index_id = $self->command_id($element); # since in_multiple_conversions returning true while multi_expanded_region # returns undef can only happen when expanding target and heading commands # text, in which there should not be @def*, using one function # or the other in the conddition should not give a different result. if (defined($index_id) and $index_id ne '' and !in_multiple_conversions($self)) { $index_label = " id=\"$index_id\""; } my ($category_element, $class_element, $type_element, $name_element, $arguments) = Texinfo::Convert::Utils::definition_arguments_content($element); my $original_def_cmdname = $element->{'extra'}->{'original_def_cmdname'}; my $original_command_name; my $alias_class; if (exists($Texinfo::Common::def_aliases{$original_def_cmdname})) { $original_command_name = $Texinfo::Common::def_aliases{$original_def_cmdname}; $alias_class = "$original_def_cmdname-alias-$original_command_name"; } else { $original_command_name = $original_def_cmdname; } my $def_command = $element->{'extra'}->{'def_command'}; my $base_command_name; if (exists($Texinfo::Common::def_aliases{$def_command})) { $base_command_name = $Texinfo::Common::def_aliases{$def_command}; } else { $base_command_name = $def_command; } my @classes = (); push @classes, $original_command_name; if (defined($alias_class)) { push @classes, $alias_class; } if ($base_command_name ne $original_command_name) { push @classes, "def-cmd-$base_command_name"; } push @classes, 'def-line'; my $def_call = ''; if (defined($type_element)) { my $explanation = "DEF_TYPE $def_command"; my $type_text = $self->convert_tree_in_code_context($type_element, $explanation); if ($type_text ne '') { $def_call .= $self->html_attribute_class('code', ['def-type']).'>'. $type_text .''; } if (($base_command_name eq 'deftypefn' or $base_command_name eq 'deftypeop') and $self->get_conf('deftypefnnewline') and $self->get_conf('deftypefnnewline') eq 'on') { $def_call .= $self->get_info('line_break_element') . ' '; } elsif ($type_text ne '') { $def_call .= ' '; } } if (defined($name_element)) { my $def_name_text = $self->convert_tree_in_code_context($name_element, "DEF_NAME $def_command"); $def_call .= $self->html_attribute_class('strong', ['def-name']).'>'. $def_name_text .''; } if (defined($arguments)) { my $explanation = "DEF_ARGS $def_command"; # arguments not only metasyntactic variables # (deftypefn, deftypevr, deftypeop, deftypecv) if (exists($Texinfo::Common::def_no_var_arg_commands{$base_command_name})) { my $arguments_formatted = $self->convert_tree_in_code_context($arguments, $explanation); if ($arguments_formatted =~ /\S/) { $def_call .= ' ' unless($element->{'extra'}->{'omit_def_name_space'}); $def_call .= $self->html_attribute_class('code', ['def-code-arguments']).'>' . $arguments_formatted.''; } } else { # only metasyntactic variable arguments (deffn, defvr, deftp, defop, defcv) # TODO not in API # Has an effect only for those @def* in @example and similar block # commands that sets code expansion. _set_code_context($self, 0); my $arguments_formatted = $self->convert_tree($arguments, $explanation); _pop_code_context($self); if ($arguments_formatted =~ /\S/) { $def_call .= ' ' unless($element->{'extra'}->{'omit_def_name_space'}); $def_call .= $self->html_attribute_class('var', ['def-var-arguments']).'>' . $arguments_formatted .''; } } } if ($self->get_conf('DEF_TABLE')) { my $category_result = ''; my $def_category_tree = Texinfo::Convert::Utils::definition_category_tree($element, $self->{'current_lang_translations'}, $self->get_conf('COMMAND_LINE_ENCODING'), $self->get_conf('DEBUG'), $self); $category_result = $self->convert_tree($def_category_tree) if (defined($def_category_tree)); return $self->html_attribute_class('tr', \@classes) . "$index_label>".$self->html_attribute_class('td', ['call-def']).'>' . $def_call . ''.$self->html_attribute_class('td', ['category-def']) . '>' . '[' . $category_result . ']' . "\n"; } my $result = $self->html_attribute_class('dt', \@classes) . "$index_label>"; if (defined($category_element)) { my $e_category_tree; if (defined($class_element)) { my $substrings = {'category' => $category_element, 'class' => $class_element}; if ($base_command_name eq 'deftypeop' and defined($type_element) and $self->get_conf('deftypefnnewline') and $self->get_conf('deftypefnnewline') eq 'on') { $e_category_tree = $self->cdt('{category} on @code{{class}}:@* ', $substrings); } elsif ($base_command_name eq 'defop' or $base_command_name eq 'deftypeop') { $e_category_tree = $self->cdt('{category} on @code{{class}}: ', $substrings); } elsif ($base_command_name eq 'defcv' or $base_command_name eq 'deftypecv') { $e_category_tree = $self->cdt('{category} of @code{{class}}: ', $substrings); } } else { my $substrings = {'category' => $category_element}; if (defined($type_element) and ($base_command_name eq 'deftypefn' or $base_command_name eq 'deftypeop') and $self->get_conf('deftypefnnewline') and $self->get_conf('deftypefnnewline') eq 'on') { # TODO if in @def* in @example and with @deftypefnnewline # on there is no effect of @deftypefnnewline on, as @* in # preformatted environment becomes an end of line, but the def* # line is not in a preformatted environment. There should be # an explicit
    in that case. Probably requires changing # the conversion of @* in a @def* line in preformatted, nothing # really specific of @deftypefnnewline on. $e_category_tree = $self->cdt('{category}:@* ', $substrings); } else { $e_category_tree = $self->cdt('{category}: ', $substrings); } } if (defined($e_category_tree)) { my $open = $self->html_attribute_class('span', ['category-def']); if ($open ne '') { $result .= $open.'>'; } my $explanation = "DEF_CATEGORY $def_command"; $result .= $self->convert_tree($e_category_tree, $explanation); if ($open ne '') { $result .= ''; } } } my $anchor = _get_copiable_anchor($self, $index_id); if (defined($anchor)) { $result .= ''; } $result .= $def_call; if (defined($anchor)) { $result .= $anchor . ''; } $result .= "
    \n"; return $result; } sub _get_copiable_anchor($$) { my ($self, $id) = @_; if (defined($id) and $id ne '' and $self->get_conf('COPIABLE_LINKS')) { my $paragraph_symbol = $self->get_info('paragraph_symbol'); return $self->html_attribute_class('a', ['copiable-link']) ." href=\"#$id\"> $paragraph_symbol"; } return undef; } $default_types_conversion{'def_line'} = \&_convert_def_line_type; sub _convert_def_item_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); if ($content =~ /\S/) { if (! $self->get_conf('DEF_TABLE')) { return '
    ' . $content . '
    '; } else { return '' . $content . ''; } } } $default_types_conversion{'def_item'} = \&_convert_def_item_type; $default_types_conversion{'inter_def_item'} = \&_convert_def_item_type; $default_types_conversion{'before_defline'} = \&_convert_def_item_type; sub _convert_table_definition_type($$$$) { my ($self, $type, $element, $content) = @_; $content = '' if (!defined($content)); return $content if (in_string($self)); if ($content =~ /\S/) { return '
    ' . $content . '
    '."\n"; } } $default_types_conversion{'table_definition'} = \&_convert_table_definition_type; $default_types_conversion{'inter_item'} = \&_convert_table_definition_type; # Function for converting special output units sub _convert_special_unit_type($$$$) { my ($self, $type, $output_unit, $content) = @_; $content = '' if (!defined($content)); if (in_string($self)) { return ''; } my $result = ''; my $special_unit_variety = $output_unit->{'special_unit_variety'}; my $closed_strings = $self->close_registered_sections_level( $self->current_filename(), 0); $result .= join('', @{$closed_strings}); my $special_unit_body .= &{$self->special_unit_body_formatting($special_unit_variety)}($self, $special_unit_variety, $output_unit); # This may happen with footnotes in regions that are not expanded, # like @copying or @titlepage if ($special_unit_body eq '') { return ''; } my $unit_command = $output_unit->{'unit_command'}; my $id = $self->command_id($unit_command); my $class_base = $self->special_unit_info('class', $special_unit_variety); $result .= $self->html_attribute_class('div', ["element-${class_base}"]); if ($id ne '') { $result .= " id=\"$id\""; } $result .= ">\n"; if ($self->get_conf('HEADERS') # first in page or (exists($output_unit->{'unit_filename'}) and $self->count_elements_in_filename('current', $output_unit->{'unit_filename'}) == 1)) { $result .= &{$self->formatting_function('format_navigation_header')}($self, $self->get_conf('MISC_BUTTONS'), undef, $unit_command); } my $heading = $self->command_text($unit_command); my $level = $self->get_conf('CHAPTER_HEADER_LEVEL'); if ($special_unit_variety eq 'footnotes') { $level = $self->get_conf('FOOTNOTE_SEPARATE_HEADER_LEVEL'); } $result .= &{$self->formatting_function('format_heading_text')}($self, undef, [$class_base.'-heading'], $heading, $level)."\n"; $result .= $special_unit_body . ''; $result .= &{$self->formatting_function('format_element_footer')}($self, $type, $output_unit, $content, $unit_command); return $result; } $default_output_units_conversion{'special_unit'} = \&_convert_special_unit_type; # Function for converting the output units. The node and associated section # appear together in the output unit. $OUTPUT_UNIT was created in this # module (in _prepare_conversion_units), it's not a tree element (created # by the parser). # $CONTENT is the contents of the output unit, already converted. sub _convert_unit_type($$$$) { my ($self, $type, $output_unit, $content) = @_; $content = '' if (!defined($content)); if (in_string($self)) { return $content; } my $result = ''; if (not exists($output_unit->{'tree_unit_directions'}) or not exists($output_unit->{'tree_unit_directions'}->{'prev'})) { my $global_commands; my $document = $self->get_info('document'); if (defined($document)) { $global_commands = $document->global_commands_information(); } if (!(defined($global_commands) and exists($global_commands->{'maketitle'}))) { $result .= $self->get_info('title_titlepage'); } if (not exists($output_unit->{'tree_unit_directions'}) or not exists($output_unit->{'tree_unit_directions'}->{'next'})) { # only one unit, use simplified formatting $result .= $content; # if there is one unit it also means that there is no formatting # of footnotes in a separate unit. And if footnotestyle is end # the footnotes won't be done in format_element_footer either. $result .= &{$self->formatting_function('format_footnotes_segment')}($self); $result .= $self->get_conf('DEFAULT_RULE') ."\n" if ($self->get_conf('PROGRAM_NAME_IN_FOOTER') and defined($self->get_conf('DEFAULT_RULE'))); # do it here, as it is won't be done at end of page in # format_element_footer my $closed_strings = $self->close_registered_sections_level( $self->current_filename(), 0); $result .= join('', @{$closed_strings}); return $result; } } $result .= $content; my $unit_command; if (exists($output_unit->{'unit_command'})) { $unit_command = $output_unit->{'unit_command'}; } $result .= &{$self->formatting_function('format_element_footer')}($self, $type, $output_unit, $content, $unit_command); return $result; } $default_output_units_conversion{'unit'} = \&_convert_unit_type; sub _contents_shortcontents_in_title($) { my $self = shift; my $result = ''; my $document = $self->get_info('document'); my $sections_list; if (defined($document)) { $sections_list = $document->sections_list(); } if (defined($sections_list) and scalar(@{$sections_list}) > 1 and $self->get_conf('CONTENTS_OUTPUT_LOCATION') eq 'after_title') { foreach my $cmdname ('shortcontents', 'contents') { if ($self->get_conf($cmdname)) { my $contents_text = _contents_inline_element($self, $cmdname, undef); if ($contents_text ne '') { $result .= $contents_text; my $rule = $self->get_conf('DEFAULT_RULE'); if (defined($rule)) { $result .= $rule ."\n"; } } } } } return $result; } sub _format_maketile($$) { my ($self, $document) = @_; my $document_info = Texinfo::Convert::Utils::get_document_documentinfo($document); if (defined($document_info)) { my @contents; my $titlepage_text = $self->html_attribute_class('div', ['maketitle-titlepage']).">\n"; foreach my $cmdname ('title', 'subtitle', 'author') { if (exists($document_info->{$cmdname})) { push @contents, @{$document_info->{$cmdname}}; } } my $element = Texinfo::TreeElement::new({'contents' => \@contents}); # we do not need to collect the author commands in documentinfo, so # we use a little trick to initialize the authors number to -1 # to mean that we are in documentinfo _open_quotation_titlepage_stack($self, -1); my $quotation_titlepage_nr = $self->get_shared_conversion_state('quotation', 'quotation_titlepage_stack'); $titlepage_text .= $self->convert_tree($element, 'format maketitle'); $quotation_titlepage_nr--; $self->set_shared_conversion_state('quotation', 'quotation_titlepage_stack', $quotation_titlepage_nr); $titlepage_text .= "\n"; return $titlepage_text; } return undef; } # Convert @titlepage. Falls back to simpletitle. sub _default_format_titlepage($) { my $self = shift; my $titlepage_text; my $global_commands; my $document = $self->get_info('document'); if (defined($document)) { $global_commands = $document->global_commands_information(); } if (defined($global_commands) and exists($global_commands->{'titlepage'})) { # we do not need to collect the author commands in titlepage, so # we use a little trick to initialize the authors number to -1 # to mean that we are in titlepage _open_quotation_titlepage_stack($self, -1); $titlepage_text = $self->convert_tree( Texinfo::TreeElement::new( {'contents' => $global_commands->{'titlepage'}->{'contents'}}), 'convert titlepage'); my $quotation_titlepage_nr = $self->get_shared_conversion_state('quotation', 'quotation_titlepage_stack'); $quotation_titlepage_nr--; $self->set_shared_conversion_state('quotation', 'quotation_titlepage_stack', $quotation_titlepage_nr); } elsif (defined($global_commands) and exists($global_commands->{'maketitle'}) and defined($document)) { $titlepage_text = _format_maketile($self, $document); } else { my $simpletitle_tree = $self->get_info('simpletitle_tree'); if (defined($simpletitle_tree)) { my $simpletitle_command_name = $self->get_info('simpletitle_command_name'); my $title_text = $self->convert_tree_new_formatting_context($simpletitle_tree, "$simpletitle_command_name simpletitle"); $titlepage_text = &{$self->formatting_function('format_heading_text')}($self, $simpletitle_command_name, [$simpletitle_command_name], $title_text, 0); } } my $result = ''; if (defined($titlepage_text)) { $result .= $titlepage_text; my $rule = $self->get_conf('DEFAULT_RULE'); if (defined($rule)) { $result .= $rule."\n"; } } $result .= _contents_shortcontents_in_title($self); return $result; } sub _default_format_title_titlepage($) { my $self = shift; if ($self->get_conf('SHOW_TITLE')) { if ($self->get_conf('USE_TITLEPAGE_FOR_TITLE')) { return &{$self->formatting_function('format_titlepage')}($self); } else { my $result = ''; my $simpletitle_tree = $self->get_info('simpletitle_tree'); if (defined($simpletitle_tree)) { my $simpletitle_command_name = $self->get_info('simpletitle_command_name'); my $title_text = $self->convert_tree_new_formatting_context($simpletitle_tree, "$simpletitle_command_name simpletitle"); $result .= &{$self->formatting_function('format_heading_text')}($self, $simpletitle_command_name, [$simpletitle_command_name], $title_text, 0); } $result .= _contents_shortcontents_in_title($self); return $result; } } return ''; } # for output units, both normal and special sub _default_format_element_footer($$$$;$) { my ($self, $type, $output_unit, $content, $command) = @_; my $result = ''; my $is_top = $self->unit_is_top_output_unit($output_unit); my $next_is_top = (exists($output_unit->{'tree_unit_directions'}->{'next'}) and $self->unit_is_top_output_unit( $output_unit->{'tree_unit_directions'}->{'next'})); my $next_is_special = (exists($output_unit->{'tree_unit_directions'}->{'next'}) and exists($output_unit->{'tree_unit_directions'}->{'next'} ->{'unit_type'}) and $output_unit->{'tree_unit_directions'}->{'next'} ->{'unit_type'} eq 'special_unit'); my $is_end_page = (!exists($output_unit->{'tree_unit_directions'}->{'next'}) or (exists($output_unit->{'unit_filename'}) and $output_unit->{'unit_filename'} ne $output_unit->{'tree_unit_directions'}->{'next'} ->{'unit_filename'} and $self->count_elements_in_filename('remaining', $output_unit->{'unit_filename'}) == 1)); my $is_special = (defined($output_unit->{'unit_type'}) and $output_unit->{'unit_type'} eq 'special_unit'); my $split = $self->get_conf('SPLIT'); if (($is_end_page or $next_is_top or $next_is_special or $is_top) and $self->get_conf('VERTICAL_HEAD_NAVIGATION') and (!defined($split) or $split ne 'node' or $self->get_conf('HEADERS') or $is_special or $is_top)) { $result .= " "."\n"; } my $buttons; if ($is_end_page) { my $closed_strings = $self->close_registered_sections_level( $self->current_filename(), 0); $result .= join('', @{$closed_strings}); my $split = $self->get_conf('SPLIT'); # setup buttons for navigation footer if (($is_top or $is_special) and ($split or !$self->get_conf('MONOLITHIC')) and (($self->get_conf('HEADERS') or (defined($split) and $split ne 'node')))) { if ($is_top) { $buttons = $self->get_conf('TOP_FOOTER_BUTTONS'); } else { $buttons = $self->get_conf('MISC_BUTTONS'); } } elsif (defined($split) and $split eq 'section') { $buttons = $self->get_conf('SECTION_FOOTER_BUTTONS'); } elsif (defined($split) and $split eq 'chapter') { $buttons = $self->get_conf('CHAPTER_FOOTER_BUTTONS'); } elsif (defined($split) and $split eq 'node') { if ($self->get_conf('HEADERS')) { my $no_footer_word_count; if ($self->get_conf('WORDS_IN_PAGE')) { $content = '' if (!defined($content)); # NOTE it would have been better to skip a leading space, but # it cannot happen as the content should start with an HTML element. # splitting at [\h\v] may have been relevant, but then the result # would be different from XS code result and could give different # results in perl in some cases. # NOTE it seems that NO-BREAK SPACE and NEXT LINE (NEL) may # not be in \h and \v in some case, but not sure when. # It is supposed to be explained but it is not very clear # https://perldoc.perl.org/perlrecharclass#Whitespace # [\h\v]+ does not match on solaris 11 with perl 5.10.1, not sure # why. #my @cnt = split(/[\h\v]+/, $content); # Use an explicit list to match the same in all versions of perl. # TODO starting in Perl v5.14 could be replaced by \s\cK (with /a) # TODO starting in Perl v5.18 could be replaced by \s (with /a) my @cnt = split(/[\t\n\f\r \cK]+/, $content); if (scalar(@cnt) < $self->get_conf('WORDS_IN_PAGE')) { $no_footer_word_count = 1; } } $buttons = $self->get_conf('NODE_FOOTER_BUTTONS') unless ($no_footer_word_count); } } } # NOTE the following condition is almost a duplication of the # condition appearing in end_page except that the file counter # needs not to be 1 if (!exists($output_unit->{'tree_unit_directions'}->{'next'}) or (exists($output_unit->{'unit_filename'}) and $output_unit->{'unit_filename'} ne $output_unit->{'tree_unit_directions'}->{'next'} ->{'unit_filename'})) { my $footnotestyle = $self->get_conf('footnotestyle'); if (!defined($footnotestyle) or $footnotestyle ne 'separate') { $result .= &{$self->formatting_function('format_footnotes_segment')}($self); } } if ($buttons or !$is_end_page or $self->get_conf('PROGRAM_NAME_IN_FOOTER')) { my $rule; my $split = $self->get_conf('SPLIT'); if (!$is_end_page and ($is_top or $next_is_top or ($next_is_special and !$is_special))) { $rule = $self->get_conf('BIG_RULE'); } elsif (!$buttons or $is_top or $is_special or ($is_end_page and defined($split) and ($split eq 'chapter' or $split eq 'section')) or (defined($split) and $split eq 'node' and $self->get_conf('HEADERS'))) { $rule = $self->get_conf('DEFAULT_RULE'); } $result .= "$rule\n" if (defined($rule) and $rule ne ''); } if ($buttons) { my $cmdname; $cmdname = $command->{'cmdname'} if (defined($command) and exists($command->{'cmdname'})); $result .= &{$self->formatting_function('format_navigation_panel')}($self, $buttons, $cmdname, $command); } return $result; } # if $document_global_context is set, it means that the formatting # is not done within the document formatting flow, but the formatted # output may still end up in the document. In particular for # command_text() which caches its computations. sub _new_document_context($$;$$$) { my ($self, $context, $context_type, $document_global_context, $block_command) = @_; my $doc_context = {'context' => $context, 'formatting_context' => [{'context_name' => '_format'}], 'composition_context' => [''], 'preformatted_context' => [0], 'inside_preformatted' => 0, 'document_global_context' => $document_global_context, 'block_commands' => [], }; if (defined($document_global_context)) { $self->{'document_global_context_counter'}++; } if (defined($context_type) and ($context_type & $CTXF_code)) { $doc_context->{'monospace'} = [1]; } else { $doc_context->{'monospace'} = [0]; } if (defined($block_command)) { push @{$doc_context->{'block_commands'}}, $block_command; } if (defined($context_type) and ($context_type & $CTXF_string)) { $doc_context->{'string'}++; } push @{$self->{'document_context'}}, $doc_context; } sub _pop_document_context($) { my $self = shift; my $context = pop @{$self->{'document_context'}}; if (defined($context->{'document_global_context'})) { $self->{'document_global_context_counter'}--; } } sub _set_string_context($) { my $self = shift; $self->{'document_context'}->[-1]->{'string'}++; } sub _unset_string_context($) { my $self = shift; $self->{'document_context'}->[-1]->{'string'}--; } sub _set_raw_context($) { my $self = shift; $self->{'document_context'}->[-1]->{'raw'}++; } sub _unset_raw_context($) { my $self = shift; $self->{'document_context'}->[-1]->{'raw'}--; } sub _set_multiple_conversions($$) { my ($self, $multiple_pass) = @_; push @{$self->{'multiple_pass'}}, $multiple_pass; } sub _unset_multiple_conversions($) { my $self = shift; pop @{$self->{'multiple_pass'}}; } # can be set through Texinfo::Config::texinfo_register_file_id_setting_function my %customizable_file_id_setting_references; foreach my $customized_reference ('external_target_split_name', 'external_target_non_split_name', 'label_target_name', 'node_file_name', 'redirection_file_names', 'sectioning_command_target_name', 'special_unit_target_file_name', 'unit_file_name') { $customizable_file_id_setting_references{$customized_reference} = 1; } # Functions accessed with e.g. 'format_heading_text'. # used in Texinfo::Config %default_formatting_references = ( 'format_begin_file' => \&_default_format_begin_file, 'format_button' => \&_default_format_button, 'format_button_icon_img' => \&_default_format_button_icon_img, 'format_css_lines' => \&_default_format_css_lines, 'format_comment' => \&_default_format_comment, 'format_contents' => \&_default_format_contents, 'format_element_header' => \&_default_format_element_header, 'format_element_footer' => \&_default_format_element_footer, 'format_end_file' => \&_default_format_end_file, 'format_footnotes_segment' => \&_default_format_footnotes_segment, 'format_footnotes_sequence' => \&_default_format_footnotes_sequence, 'format_single_footnote' => \&_default_format_single_footnote, 'format_heading_text' => \&_default_format_heading_text, 'format_navigation_header' => \&_default_format_navigation_header, 'format_navigation_panel' => \&_default_format_navigation_panel, 'format_node_redirection_page' => \&_default_format_node_redirection_page, 'format_program_string' => \&_default_format_program_string, 'format_protect_text' => \&_default_format_protect_text, 'format_separate_anchor' => \&_default_format_separate_anchor, 'format_titlepage' => \&_default_format_titlepage, 'format_title_titlepage' => \&_default_format_title_titlepage, 'format_translate_message' => undef, ); # not up for customization %default_css_string_formatting_references = ( 'format_protect_text' => \&_default_css_string_format_protect_text, ); %defaults_format_special_unit_body_contents = ( 'contents' => \&_default_format_special_body_contents, 'about' => \&_default_format_special_body_about, 'footnotes' => \&_default_format_special_body_footnotes, 'shortcontents' => \&_default_format_special_body_shortcontents, ); # $RESET_CONTEXT is the context being set or reset. # $REF_CONTEXT is the context that should be used for defaults, if # defined and there is no information for $CMDNAME $RESET_CONTEXT. # In that case, 'unset' should exist for $RESET_CONTEXT. sub _reset_unset_no_arg_commands_formatting_context($$$$;$) { my ($self, $cmdname, $reset_context, $ref_context, $translate) = @_; my $conversion_contexts = $self->{'no_arg_commands_formatting'}->{$cmdname}; # should never happen as unset is set at initialization in that case #if (!exists($conversion_contexts->{$reset_context})) { # die "Non-existing no_arg_commands_formatting $cmdname $reset_context"; #} my $no_arg_command_context = $conversion_contexts->{$reset_context}; if (defined($ref_context) and exists($no_arg_command_context->{'unset'})) { foreach my $key (keys(%{$conversion_contexts->{$ref_context}})) { # If present, both 'translated_converted' and (possibly translated) # 'text' are referred to. # In case of 'text', if 'translated_tree' is referred to and # $translate is set, the 'text' will be replaced just below. $no_arg_command_context->{$key} = $conversion_contexts->{$ref_context}->{$key}; } } if ($translate and exists($conversion_contexts->{'translated_tree'}) and not exists($no_arg_command_context->{'translated_converted'})) { my $translated_tree = $conversion_contexts->{'translated_tree'}; my $translation_result; my $explanation = "Translated NO ARG \@$cmdname ctx $reset_context"; my $context_str = "Tr $cmdname ctx $reset_context"; if ($reset_context eq 'normal') { $translation_result = $self->convert_tree($translated_tree, $explanation); } elsif ($reset_context eq 'preformatted') { # there does not seems to be anything simpler... my $preformatted_cmdname = 'example'; _new_document_context($self, $context_str); _open_command_update_context($self, $preformatted_cmdname); $translation_result = $self->convert_tree($translated_tree, $explanation); _convert_command_update_context($self, $preformatted_cmdname); _pop_document_context($self); } elsif ($reset_context eq 'string') { _new_document_context($self, $context_str, $CTXF_string); $translation_result = $self->convert_tree($translated_tree, $explanation); _pop_document_context($self); } elsif ($reset_context eq 'css_string') { $translation_result = $self->html_convert_css_string($translated_tree, $context_str); } $no_arg_command_context->{'text'} = $translation_result; } } sub _complete_no_arg_commands_formatting($$;$) { my ($self, $cmdname, $translate) = @_; _reset_unset_no_arg_commands_formatting_context($self, $cmdname, 'normal', undef, $translate); _reset_unset_no_arg_commands_formatting_context($self, $cmdname, 'preformatted', 'normal', $translate); _reset_unset_no_arg_commands_formatting_context($self, $cmdname, 'string', 'preformatted', $translate); _reset_unset_no_arg_commands_formatting_context($self, $cmdname, 'css_string', 'string', $translate); } # transform
    to
    sub _xhtml_re_close_lone_element($) { my $element = shift; if ($element =~ /\/\s*>$/) { # already a closed lone element return $element; } $element =~ s/^(<[a-zA-Z][^<>]*)>$/$1\/>/; return $element; } my %htmlxref_entries = ( 'node' => [ 'node', 'section', 'chapter', 'mono' ], 'section' => [ 'section', 'chapter','node', 'mono' ], 'chapter' => [ 'chapter', 'section', 'node', 'mono' ], 'mono' => [ 'mono', 'chapter', 'section', 'node' ], ); # $FILES is an array reference of file names binary strings. sub _parse_htmlxref_files($$) { my ($self, $files) = @_; my $htmlxref = {}; foreach my $file (@$files) { my $fname = $file; if ($self->get_conf('TEST')) { my ($volume, $directories); # strip directories for out-of-source builds reproducible file names ($volume, $directories, $fname) = File::Spec->splitpath($file); } print STDERR "html refs config file: $file\n" if ($self->get_conf('DEBUG')); unless (open(HTMLXREF, $file)) { my $htmlxref_file_name = $file; my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); if (defined($encoding)) { $htmlxref_file_name = decode($encoding, $htmlxref_file_name); } $self->converter_document_warn( sprintf(__("could not open html refs config file %s: %s"), $htmlxref_file_name, $!)); next; } my $line_nr = 0; my %variables; while (1) { my $hline = ; last if (!defined($hline)); #my $line = $hline; $line_nr++; $hline =~ s/^\s*//; next if $hline =~ /^#/; next if $hline =~ /^$/; chomp ($hline); if ($hline =~ s/^(\w+)\s*=\s*//) { # handle variables my $var = $1; my $re = join '|', map { quotemeta $_ } keys %variables; $hline =~ s/\$\{($re)\}/defined $variables{$1} ? $variables{$1} : "\${$1}"/ge; $variables{$var} = $hline; next; } my @htmlxref = split /\s+/, $hline; my $manual = shift @htmlxref; my $split_or_mono = shift @htmlxref; #print STDERR "$fname: $line_nr: $manual $split_or_mono\n"; if (!defined($split_or_mono)) { $self->converter_line_warn(__("missing type"), {'file_name' => $fname, 'line_nr' => $line_nr}); next; } elsif (!exists($htmlxref_entries{$split_or_mono})) { $self->converter_line_warn(sprintf(__("unrecognized type: %s"), $split_or_mono), {'file_name' => $fname, 'line_nr' => $line_nr}); next; } my $href = shift @htmlxref; # No warning for an empty URL prefix as it is the only way to # override an entry appearing in a file processed later on #if (!defined($href)) { # $self->converter_line_warn(sprintf( # __("missing %s URL prefix for `%s'"), $split_or_mono, $manual), # {'file_name' => $fname, 'line_nr' => $line_nr}); #} # keep previously set value next if (exists($htmlxref->{$manual}) and exists($htmlxref->{$manual}->{$split_or_mono})); if (defined($href)) { # substitute 'variables' my $re = join '|', map { quotemeta $_ } keys %variables; $href =~ s/\$\{($re)\}/defined $variables{$1} ? $variables{$1} : "\${$1}"/ge; $href =~ s/\/*$// if ($split_or_mono ne 'mono'); } else { # Store empty text instead of undef, such that exists can safely # be used. $href = ''; } $htmlxref->{$manual} = {} if (!exists($htmlxref->{$manual})); $htmlxref->{$manual}->{$split_or_mono} = $href; } if (!close (HTMLXREF)) { my $htmlxref_file_name = $file; my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); if (defined($encoding)) { $htmlxref_file_name = decode($encoding, $htmlxref_file_name); } $self->converter_document_warn(sprintf(__( "error on closing html refs config file %s: %s"), $htmlxref_file_name, $!)); } } return $htmlxref; } sub _load_htmlxref_files($) { my $self = shift; my $deprecated_dirs = $self->{'deprecated_config_directories'}; my @htmlxref_files; my $htmlxref_mode = $self->get_conf('HTMLXREF_MODE'); return if (defined($htmlxref_mode) and $htmlxref_mode eq 'none'); my $htmlxref_file_name = 'htmlxref.cnf'; if (defined($htmlxref_mode) and $htmlxref_mode eq 'file') { if (defined($self->get_conf('HTMLXREF_FILE'))) { $htmlxref_file_name = $self->get_conf('HTMLXREF_FILE'); } my ($encoded_htmlxref_file_name, $htmlxref_file_encoding) = $self->encoded_output_file_name($htmlxref_file_name); if (-e $encoded_htmlxref_file_name and -r $encoded_htmlxref_file_name) { @htmlxref_files = ($encoded_htmlxref_file_name); } else { $self->converter_document_warn( sprintf(__("could not find html refs config file %s"), $htmlxref_file_name)); } } else { my @htmlxref_dirs; if ($self->get_conf('TEST')) { my $curdir = File::Spec->curdir(); # to have reproducible tests, do not use system or user # directories if TEST is set. @htmlxref_dirs = join('/', ($curdir, '.texinfo')); if ($Texinfo::ModulePath::texinfo_uninstalled) { unshift @htmlxref_dirs, join('/', ( $Texinfo::ModulePath::t2a_srcdir, 'perl', 't', 'input_files')); } } elsif ($self->get_conf('TEXINFO_LANGUAGE_DIRECTORIES') and scalar(@{$self->get_conf('TEXINFO_LANGUAGE_DIRECTORIES')}) > 0) { @htmlxref_dirs = @{$self->get_conf('TEXINFO_LANGUAGE_DIRECTORIES')}; } my $cnf_directory_name; # no htmlxref for tests, unless explicitly specified if ($self->get_conf('TEST')) { if (defined($self->get_conf('HTMLXREF_FILE'))) { $htmlxref_file_name = $self->get_conf('HTMLXREF_FILE'); } else { $htmlxref_file_name = undef; } } else { $cnf_directory_name = 'htmlxref.d'; if (defined($self->get_conf('HTMLXREF_FILE'))) { $htmlxref_file_name = $self->get_conf('HTMLXREF_FILE'); } } my ($encoded_htmlxref_file_name, $htmlxref_file_encoding); # encode file name and handle specific cases for the main htmlxref file # without search in directories. if (defined($htmlxref_file_name)) { ($encoded_htmlxref_file_name, $htmlxref_file_encoding) = $self->encoded_output_file_name($htmlxref_file_name); if (File::Spec->file_name_is_absolute($encoded_htmlxref_file_name)) { if (-e $encoded_htmlxref_file_name and -r $encoded_htmlxref_file_name) { push @htmlxref_files, $encoded_htmlxref_file_name; } $htmlxref_file_name = undef; } else { my ($volume, $path_directories, $file) = File::Spec->splitpath($htmlxref_file_name); my @path_directories = File::Spec->splitdir($path_directories); # do not search in directories if the file name already contains # directories. if (scalar(@path_directories) > 0) { if (-e $encoded_htmlxref_file_name and -r $encoded_htmlxref_file_name) { push @htmlxref_files, $encoded_htmlxref_file_name; } $htmlxref_file_name = undef; } } } # now search in directories if (defined($htmlxref_file_name) or defined($cnf_directory_name)) { my ($encoded_cnf_directory_name, $cnf_directory_encoding); if (defined($cnf_directory_name)) { ($encoded_cnf_directory_name, $cnf_directory_encoding) = $self->encoded_output_file_name($cnf_directory_name); } my $deprecated_dirs_used; foreach my $dir (@htmlxref_dirs) { next unless (-d $dir); my $deprecated_dir_set = 0; if (defined($htmlxref_file_name)) { my $possible_file = "$dir/$encoded_htmlxref_file_name"; if (-e $possible_file and -r $possible_file) { if (defined($deprecated_dirs) and $deprecated_dirs->{$dir}) { $deprecated_dirs_used = [] if (!defined($deprecated_dirs_used)); push @$deprecated_dirs_used, $dir; $deprecated_dir_set = 1; } push (@htmlxref_files, $possible_file); } } if (defined($cnf_directory_name)) { my $cnf_dir = "$dir/$encoded_cnf_directory_name"; if (-d $cnf_dir) { my $file_found = 0; # the internal simple quotes are for the case of spaces in $cnf_dir. my @possible_files = glob("'$cnf_dir/*.cnf'"); foreach my $possible_file (sort(@possible_files)) { if (-e $possible_file and -r $possible_file) { push (@htmlxref_files, $possible_file); $file_found = 1; } } if (!$deprecated_dir_set and $file_found and $deprecated_dirs and $deprecated_dirs->{$dir}) { $deprecated_dirs_used = [] if (!defined($deprecated_dirs_used)); push @$deprecated_dirs_used, $dir; $deprecated_dir_set = 1; } } } } if (defined($deprecated_dirs_used)) { foreach my $dir (@$deprecated_dirs_used) { my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); my ($dir_name, $replacement_dir); if (defined($encoding)) { $dir_name = decode($encoding, $dir); $replacement_dir = decode($encoding, $deprecated_dirs->{$dir}) } else { $dir_name = $dir; $replacement_dir = $deprecated_dirs->{$dir}; } $self->converter_document_warn(sprintf(__( "%s directory is deprecated. Use %s instead"), $dir_name, $replacement_dir)); } } } } $self->{'htmlxref'} = {}; if (scalar(@htmlxref_files)) { $self->{'htmlxref'} = _parse_htmlxref_files($self, \@htmlxref_files); } } # converter state # # No API # all_directions # determined parallelly in C # deprecated_config_directories # # API exists # # Get through converter set_global_document_commands with 'before'. No # specific API to set, but can use get_conf or force_conf in setup handler # commands_init_conf # # shared_conversion_state # Set through the shared_conversion_state API (among others): # explained_commands # used only in an @-command conversion function # # API converter_info get_info # document_name # destination_directory # paragraph_symbol # line_break_element # non_breaking_space # simpletitle_tree # simpletitle_command_name # title_string # title_tree # documentdescription_string # copying_comment # jslicenses # # API exists # current_filename # current_output_unit # index_entries # index_entries_by_letter # # API exists in Texinfo::Config for setting, not for getting # stage_handlers # # No API, but set by command-line and can be overriden by CSS # change API functions # files_css_import_lines # files_css_rule_lines # # API exists # css_element_class_styles # css_import_lines # css_rule_lines # # API exists # file_id_setting # commands_conversion # commands_open # types_conversion # types_open # # API exists for setting, no need to access directly, accessed through # no_arg_commands_formatting # customized_no_arg_commands_formatting # # API exists for setting (through customized_no_arg_commands_formatting # for no_arg_commands_formatting), not for getting and used in # commands_conversion # no_arg_commands_formatting # style_commands_formatting # # API exists # code_types # pre_class_types # # API exists # document_context # # API exists # pending_closes # # API exists # pending_footnotes # # API exists # pending_inline_content # associated_inline_content # # API exists # multiple_pass # # API exists # targets for directions. Keys are elements references, values are # target information hash references described above before # the API functions used to access this information. # special_targets # global_units_directions # # API exists for setting, not getting # customized_direction_strings # directions_strings # translated_direction_strings # # API exists # special_unit_info # translated_special_unit_info # # API exists # elements_in_file_count # the number of output units in file # file_counters # begin at elements_in_file_count decrease # # each time the unit is closed # # API exists # document_global_context_css # page_css # # API exists # files_information # # No API, converter internals # document_units # out_filepaths (partially common with Texinfo::Converter) # seen_ids # options_latex_math # htmlxref # check_htmlxref_already_warned # referred_command_stack # # from Converter # labels my %special_characters = ( 'paragraph_symbol' => ['¶', '00B6'], 'left_quote' => ['‘', '2018'], 'right_quote' => ['’', '2019'], 'bullet' => ['•', '2022'], 'non_breaking_space' => [$xml_named_entity_nbsp, '00A0'], ); sub _XS_html_converter_initialize_beginning($) { } sub _XS_html_converter_get_customization($$$$$$$$$$$$$$$$$$$) { } # this allows to get some debugging output for the file without setting # the customization variable. my $debug; # whether to print debugging output sub converter_initialize($) { my $self = shift; # beginning of initialization done either in Perl or XS if ($self->{'converter_descriptor'} and $XS_convert) { _XS_html_converter_initialize_beginning($self); } else { # used in initialization. Set if undef if (!defined($self->get_conf('FORMAT_MENU'))) { $self->force_conf('FORMAT_MENU', ''); } # NOTE we reset silently if the split specification is not one known. # The main program warns if the specific command line option value is # not known. We could add a warning here to catch mistakes in init # files. Wait for user reports. my $split = $self->get_conf('SPLIT'); if ($split and $split ne 'chapter' and $split ne 'section' and $split ne 'node') { $self->force_conf('SPLIT', 'node'); } my $max_header_level = $self->get_conf('MAX_HEADER_LEVEL'); if (!defined($max_header_level)) { $self->force_conf('MAX_HEADER_LEVEL', $defaults{'MAX_HEADER_LEVEL'}); } elsif ($max_header_level < 1) { $self->force_conf('MAX_HEADER_LEVEL', 1); } # For CONTENTS_OUTPUT_LOCATION # should lead to contents not output, but if not, it is not an issue, # the way to set contents to be output or not should be through the # contents and shortcontents @-commands and customization options. foreach my $conf ('CONTENTS_OUTPUT_LOCATION', 'INDEX_ENTRY_COLON', 'MENU_ENTRY_COLON') { if (!defined($self->get_conf($conf))) { $self->force_conf($conf, ''); } } _load_htmlxref_files($self); _prepare_css($self); } $self->{'output_units_conversion'} = {}; my $customized_output_units_conversion = Texinfo::Config::GNUT_get_output_units_conversion(); $customized_output_units_conversion = {} if (!defined($customized_output_units_conversion)); foreach my $type (keys(%default_output_units_conversion)) { if (exists($customized_output_units_conversion->{$type})) { $self->{'output_units_conversion'}->{$type} = $customized_output_units_conversion->{$type}; } else { $self->{'output_units_conversion'}->{$type} = $default_output_units_conversion{$type}; } } $self->{'types_conversion'} = {}; my $customized_types_conversion = Texinfo::Config::GNUT_get_types_conversion(); $customized_types_conversion = {} if (!defined($customized_types_conversion)); foreach my $type (keys(%default_types_conversion)) { if (exists($customized_types_conversion->{$type})) { $self->{'types_conversion'}->{$type} = $customized_types_conversion->{$type}; } else { $self->{'types_conversion'}->{$type} = $default_types_conversion{$type}; } } $self->{'types_open'} = {}; my $customized_types_open = Texinfo::Config::GNUT_get_types_open(); $customized_types_open = {} if (!defined($customized_types_open)); foreach my $type (keys(%default_types_conversion)) { if (exists($customized_types_open->{$type})) { $self->{'types_open'}->{$type} = $customized_types_open->{$type}; } elsif (exists($default_types_open{$type})) { $self->{'types_open'}->{$type} = $default_types_open{$type}; } } $self->{'code_types'} = {}; foreach my $type (keys(%default_code_types)) { $self->{'code_types'}->{$type} = $default_code_types{$type}; } $self->{'pre_class_types'} = {}; foreach my $type (keys(%default_pre_class_types)) { $self->{'pre_class_types'}->{$type} = $default_pre_class_types{$type}; } my $customized_code_types = Texinfo::Config::GNUT_get_types_code_info(); if (defined($customized_code_types)) { foreach my $type (keys(%$customized_code_types)) { $self->{'code_types'}->{$type} = $customized_code_types->{$type}; } } my $customized_pre_class_types = Texinfo::Config::GNUT_get_types_pre_class(); if (defined($customized_pre_class_types)) { foreach my $type (keys(%$customized_pre_class_types)) { $self->{'pre_class_types'}->{$type} = $customized_pre_class_types->{$type}; } } $self->{'upper_case_commands'} = {}; foreach my $cmdname (keys(%default_upper_case_commands)) { $self->{'upper_case_commands'}->{$cmdname} = $default_upper_case_commands{$cmdname}; } my $customized_upper_case_commands = Texinfo::Config::GNUT_get_upper_case_commands_info(); if (defined($customized_upper_case_commands)) { foreach my $cmdname (keys(%$customized_upper_case_commands)) { $self->{'upper_case_commands'}->{$cmdname} = $customized_upper_case_commands->{$cmdname}; } } $self->{'commands_conversion'} = {}; my $customized_commands_conversion = Texinfo::Config::GNUT_get_commands_conversion(); $customized_commands_conversion = {} if (!defined($customized_commands_conversion)); foreach my $cmdname (keys(%line_commands), keys(%brace_commands), keys (%block_commands), keys(%nobrace_commands)) { if (exists($customized_commands_conversion->{$cmdname})) { $self->{'commands_conversion'}->{$cmdname} = $customized_commands_conversion->{$cmdname}; } else { my $format_menu = $self->get_conf('FORMAT_MENU'); if ($format_menu ne 'menu' and $format_menu ne 'menu_no_detailmenu' and ($cmdname eq 'menu' or $cmdname eq 'detailmenu')) { $self->{'commands_conversion'}->{$cmdname} = undef; } elsif (exists($format_raw_commands{$cmdname}) and !$self->{'expanded_formats'}->{$cmdname}) { $self->{'commands_conversion'}->{$cmdname} = undef; } elsif (exists($default_commands_conversion{$cmdname})) { $self->{'commands_conversion'}->{$cmdname} = $default_commands_conversion{$cmdname}; } } } $self->{'commands_open'} = {}; my $customized_commands_open = Texinfo::Config::GNUT_get_commands_open(); $customized_commands_open = {} if (!defined($customized_commands_open)); foreach my $cmdname (keys(%line_commands), keys(%brace_commands), keys (%block_commands), keys(%nobrace_commands)) { if (exists($customized_commands_open->{$cmdname})) { $self->{'commands_open'}->{$cmdname} = $customized_commands_open->{$cmdname}; } elsif (exists($default_commands_open{$cmdname})) { $self->{'commands_open'}->{$cmdname} = $default_commands_open{$cmdname}; } } # get all the customization my %style_commands_customized_formatting_info; foreach my $cmdname (keys(%default_style_commands_formatting)) { foreach my $context (@style_commands_contexts) { my $style_commands_formatting_info = Texinfo::Config::GNUT_get_style_command_formatting($cmdname, $context); if (defined($style_commands_formatting_info)) { if (!exists($style_commands_customized_formatting_info{$cmdname})) { $style_commands_customized_formatting_info{$cmdname} = {}; } $style_commands_customized_formatting_info{$cmdname}->{$context} = $style_commands_formatting_info; } } } $self->{'style_commands_formatting'} = {}; foreach my $cmdname (keys(%default_style_commands_formatting)) { $self->{'style_commands_formatting'}->{$cmdname} = {}; foreach my $context (@style_commands_contexts) { if (exists($style_commands_customized_formatting_info{$cmdname}) and $style_commands_customized_formatting_info{$cmdname}->{$context}) { $self->{'style_commands_formatting'}->{$cmdname}->{$context} = $style_commands_customized_formatting_info{$cmdname}->{$context}; } elsif (exists($default_style_commands_formatting{$cmdname}->{$context})) { $self->{'style_commands_formatting'}->{$cmdname}->{$context} = $default_style_commands_formatting{$cmdname}->{$context}; } } } my %customized_accent_entities; foreach my $accent_command (keys(%Texinfo::Convert::Converter::xml_accent_entities)) { my ($accent_command_entity, $accent_command_text_with_entities) = Texinfo::Config::GNUT_get_accent_command_formatting($accent_command); if (defined($accent_command_entity) or defined($accent_command_text_with_entities)) { $customized_accent_entities{$accent_command} = [$accent_command_entity, $accent_command_text_with_entities]; } } $self->{'accent_entities'} = {}; foreach my $accent_command (keys(%Texinfo::Convert::Converter::xml_accent_entities)) { $self->{'accent_entities'}->{$accent_command} = []; my ($accent_command_entity, $accent_command_text_with_entities); if (exists($customized_accent_entities{$accent_command})) { ($accent_command_entity, $accent_command_text_with_entities) = @{$customized_accent_entities{$accent_command}}; } if (not defined($accent_command_entity) and defined($Texinfo::Convert::Converter::xml_accent_text_with_entities{ $accent_command})) { $accent_command_entity = $Texinfo::Convert::Converter::xml_accent_entities{$accent_command}; } if (not defined($accent_command_text_with_entities) and defined($Texinfo::Convert::Converter::xml_accent_text_with_entities{ $accent_command})) { $accent_command_text_with_entities = $Texinfo::Convert::Converter::xml_accent_text_with_entities{$accent_command}; } # an empty string means no formatting if (defined($accent_command_entity)) { $self->{'accent_entities'}->{$accent_command} = [$accent_command_entity, $accent_command_text_with_entities]; } } #print STDERR Data::Dumper->Dump([$self->{'accent_entities'}]); # get customization only at that point, as the defaults may be changed # with the encoding my $customized_no_arg_commands_formatting = {}; foreach my $cmdname (keys(%{$default_no_arg_commands_formatting{'normal'}})) { $customized_no_arg_commands_formatting->{$cmdname} = {}; foreach my $context (@no_args_commands_contexts, 'translated_to_convert') { my $no_arg_command_customized_formatting = Texinfo::Config::GNUT_get_no_arg_command_formatting($cmdname, $context); if (defined($no_arg_command_customized_formatting)) { $customized_no_arg_commands_formatting->{$cmdname}->{$context} = $no_arg_command_customized_formatting; } } } $self->{'customized_no_arg_commands_formatting'} = $customized_no_arg_commands_formatting; $self->{'file_id_setting'} = {}; my $customized_file_id_setting_references = Texinfo::Config::GNUT_get_file_id_setting_references(); if (defined($customized_file_id_setting_references)) { # first check the validity of the names foreach my $custom_file_id_setting (sort(keys(%{$customized_file_id_setting_references}))) { if (!exists($customizable_file_id_setting_references{ $custom_file_id_setting})) { $self->converter_document_warn( sprintf(__("Unknown file and id setting function: %s"), $custom_file_id_setting)); } else { $self->{'file_id_setting'}->{$custom_file_id_setting} = $customized_file_id_setting_references->{$custom_file_id_setting}; } } } my $customized_formatting_references = Texinfo::Config::GNUT_get_formatting_references(); # first check that all the customized_formatting_references # are in default_formatting_references if (defined($customized_formatting_references)) { foreach my $custom_formatting_ref (sort(keys(%{$customized_formatting_references}))) { if (!exists($default_formatting_references{$custom_formatting_ref})) { $self->converter_document_warn( sprintf(__("Unknown formatting function: %s"), $custom_formatting_ref)); } } } else { $customized_formatting_references = {}; } $self->{'formatting_function'} = {}; foreach my $formatting_reference (keys(%default_formatting_references)) { if (defined($customized_formatting_references->{$formatting_reference})) { $self->{'formatting_function'}->{$formatting_reference} = $customized_formatting_references->{$formatting_reference}; } else { $self->{'formatting_function'}->{$formatting_reference} = $default_formatting_references{$formatting_reference}; } } my $customized_special_unit_info = Texinfo::Config::GNUT_get_special_unit_info(); $customized_special_unit_info = {} if (!defined($customized_special_unit_info)); $self->{'special_unit_info'} = {}; foreach my $type (keys(%default_special_unit_info)) { $self->{'special_unit_info'}->{$type} = {}; foreach my $special_unit_variety (keys(%{$default_special_unit_info{$type}})) { if (exists($customized_special_unit_info->{$type}) and exists($customized_special_unit_info ->{$type}->{$special_unit_variety})) { $self->{'special_unit_info'}->{$type}->{$special_unit_variety} = $customized_special_unit_info->{$type}->{$special_unit_variety}; } else { $self->{'special_unit_info'}->{$type}->{$special_unit_variety} = $default_special_unit_info{$type}->{$special_unit_variety}; } } } # Note that with XS this information is not actually used, as # it is only accessed through functions that are overriden $self->{'translated_special_unit_info_texinfo'} = {}; $self->{'translated_special_unit_info_tree'} = {}; foreach my $type (keys(%default_translated_special_unit_info)) { $self->{'translated_special_unit_info_texinfo'}->{$type} = {}; $self->{'translated_special_unit_info_tree'}->{$type} = {}; foreach my $special_unit_variety (keys(%{$default_translated_special_unit_info{$type}})) { if (exists($customized_special_unit_info->{$type}) and exists($customized_special_unit_info ->{$type}->{$special_unit_variety})) { $self->{'translated_special_unit_info_texinfo'}->{$type} ->{$special_unit_variety} = $customized_special_unit_info->{$type}->{$special_unit_variety}; } else { $self->{'translated_special_unit_info_texinfo'}->{$type} ->{$special_unit_variety} = $default_translated_special_unit_info{$type} ->{$special_unit_variety}; } } } my $customized_special_unit_body = Texinfo::Config::GNUT_get_formatting_special_unit_body_references(); $self->{'special_unit_body'} = {}; foreach my $special_unit_variety (keys(%defaults_format_special_unit_body_contents)) { $self->{'special_unit_body'}->{$special_unit_variety} = $defaults_format_special_unit_body_contents{$special_unit_variety}; } foreach my $special_unit_variety (keys(%$customized_special_unit_body)) { $self->{'special_unit_body'}->{$special_unit_variety} = $customized_special_unit_body->{$special_unit_variety}; } # "directions" not associated to output units, but associated to text. $self->{'global_texts_directions'} = {}; $self->{'global_texts_directions'}->{'Space'} = 1; $self->{'all_directions'} = {}; foreach my $direction (@all_directions_except_special_units) { $self->{'all_directions'}->{$direction} = 1; } $self->{'customized_text_directions'} = Texinfo::Config::GNUT_get_text_directions(); if (defined($self->{'customized_text_directions'})) { foreach my $direction (keys(%{$self->{'customized_text_directions'}})) { if (!exists($self->{'all_directions'}->{$direction})) { $self->{'global_texts_directions'}->{$direction} = 1; $self->{'all_directions'}->{$direction} = 1; } } } $self->{'customized_global_directions'} = Texinfo::Config::GNUT_get_global_directions(); if (defined($self->{'customized_global_directions'})) { foreach my $direction (keys(%{$self->{'customized_global_directions'}})) { $self->{'all_directions'}->{$direction} = 1; } } # customized_global_directions are not used further here, as the output # unit need to be found with the document foreach my $variety (keys(%{$self->{'special_unit_info'}->{'direction'}})) { my $direction = $self->{'special_unit_info'}->{'direction'}->{$variety}; if (defined($direction)) { $self->{'all_directions'}->{$direction} = 1; } } #print STDERR join('|', sort(keys(%all_directions)))."\n"; my $customized_direction_strings = Texinfo::Config::GNUT_get_direction_string_info(); $customized_direction_strings = {} if (!defined($customized_direction_strings)); # Fill the translated direction strings information, corresponding to: # - strings already converted # - strings not already converted # Each of those types of translated strings are translated later on # and the translated values are put in $self->{'direction_strings'}. $self->{'translated_direction_strings'} = {}; foreach my $string_type (keys(%default_translated_directions_strings)) { $self->{'translated_direction_strings'}->{$string_type} = {}; foreach my $direction (keys(%{$self->{'all_directions'}})) { if (exists($customized_direction_strings->{$string_type}) and exists($customized_direction_strings->{$string_type}->{$direction})) { $self->{'translated_direction_strings'}->{$string_type}->{$direction} = $customized_direction_strings->{$string_type}->{$direction}; } else { if (exists($default_translated_directions_strings{$string_type} ->{$direction}) and exists($default_translated_directions_strings{$string_type} ->{$direction}->{'converted'})) { $self->{'translated_direction_strings'}->{$string_type} ->{$direction} = {'converted' => {}}; foreach my $context ('normal', 'string') { $self->{'translated_direction_strings'}->{$string_type} ->{$direction}->{'converted'}->{$context} = $default_translated_directions_strings{$string_type} ->{$direction}->{'converted'}; } } else { $self->{'translated_direction_strings'}->{$string_type}->{$direction} = $default_translated_directions_strings{$string_type}->{$direction}; } } } } # the customization information are not used further here, as # substitute_html_non_breaking_space is used and it depends on the document $self->{'customized_direction_strings'} = $customized_direction_strings; $self->{'stage_handlers'} = Texinfo::Config::GNUT_get_stage_handlers(); # XS parser initialization if ($self->{'converter_descriptor'} and $XS_convert) { _XS_html_converter_get_customization($self, \%default_formatting_references, \%default_css_string_formatting_references, \%default_commands_open, \%default_commands_conversion, \%default_css_string_commands_conversion, \%default_types_open, \%default_types_conversion, \%default_css_string_types_conversion, \%default_output_units_conversion, \%defaults_format_special_unit_body_contents, $customized_upper_case_commands, $customized_code_types, $customized_pre_class_types, \%customized_accent_entities, \%style_commands_customized_formatting_info, $customized_no_arg_commands_formatting, $customized_special_unit_info, $customized_direction_strings ); } return $self; } # remove data that leads to cycles related to output units and references # to output units. sub converter_release_output_units($) { my $self = shift; # remove references to output units if (exists($self->{'global_units_directions'})) { %{$self->{'global_units_directions'}} = (); } # Cannot do that, the content is still needed by the Converter #@{$self->{'document_units'}} = (); $self->{'document_units'} = []; } # remove data that leads to cycles and references to elements. sub converter_destroy($) { my $self = shift; if (exists($self->{'converter_info'})) { foreach my $key ('document', 'simpletitle_tree', 'title_tree') { delete $self->{'converter_info'}->{$key}; } } delete $self->{'current_node'}; delete $self->{'current_root_command'}; # a separate cache used if the user defines the translate_message function. delete $self->{'translation_cache'}; # remove shared conversion states pointing to elements if (exists($self->{'shared_conversion_state'})) { if (exists($self->{'shared_conversion_state'}->{'nodedescription'}) and exists($self->{'shared_conversion_state'}->{'nodedescription'} ->{'formatted_nodedescriptions'})) { delete $self->{'shared_conversion_state'}->{'nodedescription'} ->{'formatted_nodedescriptions'}; } if (exists($self->{'shared_conversion_state'}->{'quotation'}) and exists($self->{'shared_conversion_state'}->{'quotation'} ->{'elements_authors'})) { delete $self->{'shared_conversion_state'}->{'quotation'} ->{'elements_authors'}; } } if (exists($self->{'no_arg_commands_formatting'})) { foreach my $cmdname (keys(%{$self->{'no_arg_commands_formatting'}})) { my $no_arg_command_ctx = $self->{'no_arg_commands_formatting'}->{$cmdname}; if (defined($no_arg_command_ctx->{'translated_tree'})) { my $tree = $no_arg_command_ctx->{'translated_tree'}; # always a copy Texinfo::ManipulateTree::tree_remove_parents($tree); } } } delete $self->{'translated_special_unit_info_tree'}; if (exists($self->{'targets'})) { foreach my $command (keys(%{$self->{'targets'}})) { my $target = $self->{'targets'}->{$command}; # can be tree elements or results of translations through cdt delete $target->{'tree'}; delete $target->{'tree_nonumber'}; # tree elements delete $target->{'name_tree'}; delete $target->{'name_tree_nonumber'}; delete $target->{'root_element_command'}; delete $target->{'node_command'}; } } } sub _XS_html_convert_tree($$;$) { return undef; } # the entry point for _convert sub convert_tree($$;$) { my ($self, $tree, $explanation) = @_; # No XS, convert_tree is not called on trees registered in XS #my $XS_result = _XS_html_convert_tree($self, $tree, $explanation); #return $XS_result if (defined($XS_result)); # when formatting accents, goes through xml_accent without # explanation, as explanation is not in the standard API, but # otherwise the coverage of explanations should be pretty good #cluck if (! defined($explanation)); #print STDERR "CONVERT_TREE".(defined($explanation) ? " ".$explanation : '')."\n" # if ($self->get_conf('DEBUG')); return _convert($self, $tree, $explanation); } # Protect an url, in which characters with specific meaning in url are # considered to have their specific meaning. sub url_protect_url_text($$) { my ($self, $input_string) = @_; # turn end of lines to spaces, as it is most likely what is expected # rather than a percent encoded end of line. $input_string =~ s/[\n\r]+/ /g; # percent encode character string. It is better use UTF-8 irrespective # of the actual charset of the HTML output file, according to the tests done. my $href = encode("UTF-8", $input_string); # protect 'ligntly', do not protect unreserved and reserved characters + the % itself $href =~ s/([^^A-Za-z0-9\-_.!~*'()\$&+,\/:;=\?@\[\]\#%])/ sprintf "%%%02x", ord $1 /eg; return &{$self->formatting_function('format_protect_text')}($self, $href); } # Protect a file path used in an url. Characters appearing in file paths # are not protected. All the other characters that can be percent # protected are protected, including characters with specific meaning in url. sub url_protect_file_text($$) { my ($self, $input_string) = @_; # turn end of lines to spaces, as it is most likely what is expected. $input_string =~ s/[\n\r]+/ /g; # percent encode character string. It is better use UTF-8 irrespective # of the actual charset of the HTML output file, according to the tests done. my $href = encode("UTF-8", $input_string); # protect everything that can be special in url except ~, / and : that could # appear in file names and does not have much risk in being incorrectly # interpreted (for :, the interpretation as a scheme delimiter may be possible). $href =~ s/([^^A-Za-z0-9\-_.~\/:])/ sprintf "%%%02x", ord $1 /eg; return &{$self->formatting_function('format_protect_text')}($self, $href); } sub _normalized_to_id($) { my $id = shift; if (!defined($id)) { cluck "_normalized_to_id id not defined"; return ''; } $id =~ s/^([0-9_])/g_t$1/; return $id; } sub _default_format_css_lines($;$) { my ($self, $filename) = @_; return '' if ($self->get_conf('NO_CSS')); my $css_refs = $self->get_conf('CSS_REFS'); my $css_element_classes = $self->html_get_css_elements_classes($filename); my $css_import_lines = $self->css_get_info('imports'); my $css_rule_lines = $self->css_get_info('rules'); return '' if !@$css_import_lines and !@$css_element_classes and !@$css_rule_lines and (!defined($css_refs) or !@$css_refs); my $css_text = "\n"; foreach my $ref (@$css_refs) { $css_text .= $self->close_html_lone_element( '; last if (!defined($input_line)); my $line = Encode::decode($input_perl_encoding, $input_line); $line_nr++; if ($line_nr == 1) { # should always be the first line if ($line =~ /^\@charset *"([^"]+)" *; *$/) { my $charset = $1; my $Encode_encoding_object = find_encoding($charset); if (defined($Encode_encoding_object)) { my $perl_encoding = $Encode_encoding_object->name(); if (defined($perl_encoding) and $perl_encoding ne '') { $input_perl_encoding = $perl_encoding; } } next; } } #print STDERR "Line: $line"; if ($in_rules) { push @$rules, $line; next; } my $text = ''; while (1) { #sleep 1; #print STDERR "${text}!in_comment $in_comment in_rules $in_rules in_import $in_import in_string $in_string: $line"; if ($in_comment) { if ($line =~ s/^(.*?\*\/)//) { $text .= $1; $in_comment = 0; } else { push @$imports, $text . $line; last; } } elsif (!$in_string and $line =~ s/^\///) { if ($line =~ s/^\*//) { $text .= '/*'; $in_comment = 1; } else { push (@$imports, $text. "\n") if ($text ne ''); push (@$rules, '/' . $line); $in_rules = 1; last; } } elsif (!$in_string and $in_import and $line =~ s/^([\"\'])//) { # strings outside of import start rules $text .= "$1"; $in_string = quotemeta("$1"); } elsif ($in_string and $line =~ s/^(\\$in_string)//) { $text .= $1; } elsif ($in_string and $line =~ s/^($in_string)//) { $text .= $1; $in_string = 0; } elsif ((! $in_string and !$in_import) and ($line =~ s/^([\\]?\@import)$// or $line =~ s/^([\\]?\@import\s+)//)) { $text .= $1; $in_import = 1; } elsif (!$in_string and $in_import and $line =~ s/^\;//) { $text .= ';'; $in_import = 0; } elsif (($in_import or $in_string) and $line =~ s/^(.)//) { $text .= $1; } elsif (!$in_import and $line =~ s/^([^\s])//) { push (@$imports, $text. "\n") if ($text ne ''); push (@$rules, $1 . $line); $in_rules = 1; last; } elsif ($line =~ s/^(\s)//) { $text .= $1; } elsif ($line eq '') { push (@$imports, $text); last; } } } $self->converter_line_warn(__("string not closed in css file"), {'file_name' => $file, 'line_nr' => $line_nr}) if ($in_string); $self->converter_line_warn(__("--css-include ended in comment"), {'file_name' => $file, 'line_nr' => $line_nr}) if ($in_comment); $self->converter_line_warn(__("\@import not finished in css file"), {'file_name' => $file, 'line_nr' => $line_nr}) if ($in_import and !$in_comment and !$in_string); return ($imports, $rules); } sub _prepare_css($) { my $self = shift; $self->{'files_css_rule_lines'} = []; $self->{'files_css_import_lines'} = []; return if ($self->get_conf('NO_CSS')); my @css_import_lines; my @css_rule_lines; my $css_files = $self->get_conf('CSS_FILES'); foreach my $css_file (@$css_files) { my $css_file_fh; my $css_file_path; if ($css_file eq '-') { $css_file_fh = \*STDIN; $css_file_path = '-'; } else { $css_file_path = Texinfo::Common::locate_include_file($css_file, $self->get_conf('INCLUDE_DIRECTORIES')); unless (defined($css_file_path)) { my $css_input_file_name; my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); if (defined($encoding)) { $css_input_file_name = decode($encoding, $css_file); } else { $css_input_file_name = $css_file; } $self->converter_document_warn(sprintf( __("CSS file %s not found"), $css_input_file_name)); next; } unless (open(CSSFILE, $css_file_path)) { my $css_file_name; my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); if (defined($encoding)) { $css_file_name = decode($encoding, $css_file_path); } else { $css_file_name = $css_file_path; } $self->converter_document_warn(sprintf(__( "could not open --include-file %s: %s"), $css_file_name, $!)); next; } $css_file_fh = \*CSSFILE; } my ($import_lines, $rules_lines); ($import_lines, $rules_lines) = _process_css_file($self, $css_file_fh, $css_file_path); if (!close($css_file_fh)) { my $css_file_name; my $encoding = $self->get_conf('COMMAND_LINE_ENCODING'); if (defined($encoding)) { $css_file_name = decode($encoding, $css_file_path); } else { $css_file_name = $css_file_path; } $self->converter_document_warn( sprintf(__("error on closing CSS file %s: %s"), $css_file_name, $!)); } push @css_import_lines, @$import_lines; push @css_rule_lines, @$rules_lines; } if ($self->get_conf('DEBUG')) { if (@css_import_lines) { print STDERR "# css import lines\n"; foreach my $line (@css_import_lines) { print STDERR "$line"; } } if (@css_rule_lines) { print STDERR "# css rule lines\n"; foreach my $line (@css_rule_lines) { print STDERR "$line"; } } } push @{$self->{'files_css_import_lines'}}, @css_import_lines; push @{$self->{'files_css_rule_lines'}}, @css_rule_lines; } # Get the name of a file containing a label, as well as the identifier within # that file to link to that label. $normalized is the normalized label name # and $label_element is the label contents element. Labels are typically # associated to @node, @*anchor or @float and to external nodes. sub _normalized_label_id_file($$$) { my ($self, $normalized, $label_element) = @_; my $target; if (!defined($normalized) and defined($label_element)) { $normalized = Texinfo::Convert::NodeNameNormalization::convert_to_node_identifier( $label_element); } if (defined($normalized)) { $target = _normalized_to_id($normalized); } else { $target = ''; } # to find out the Top node, one could check $normalized if (defined($self->{'file_id_setting'}->{'label_target_name'})) { $target = &{$self->{'file_id_setting'}->{'label_target_name'}}($self, $normalized, $label_element, $target); } my $filename = $self->node_information_filename($normalized, $label_element); return ($filename, $target); } sub _register_id($$) { my ($self, $id) = @_; $self->{'seen_ids'}->{$id} = 1; } sub _id_is_registered($$) { my ($self, $id) = @_; if (exists($self->{'seen_ids'}->{$id})) { return 1; } else { return 0; } } sub _unique_target($$) { my ($self, $target_base) = @_; my $nr=1; my $target = $target_base; while (_id_is_registered($self, $target)) { $target = $target_base.'-'.$nr; $nr++; # Avoid integer overflow die if ($nr == 0); } return $target; } sub _new_sectioning_command_target($$) { my ($self, $command) = @_; my ($normalized_name, $filename) = $self->normalized_sectioning_command_filename($command); my $target_base = _normalized_to_id($normalized_name); if ($target_base !~ /\S/ and $command->{'cmdname'} eq 'top') { # @top is allowed to be empty. In that case it gets this target name $target_base = 'SEC_Top'; $normalized_name = $target_base; } my $nr=1; my $target = $target_base; if ($target_base ne '') { $target = _unique_target($self, $target_base); } else { $target = ''; } # These are undefined if the $target is set to ''. my $target_contents; my $target_shortcontents; if (exists($sectioning_heading_commands{$command->{'cmdname'}})) { if ($target ne '') { my $target_base_contents = 'toc-'.$normalized_name; $target_contents = _unique_target($self, $target_base_contents); my $target_base_shortcontents = 'stoc-'.$normalized_name; $target_shortcontents = _unique_target($self, $target_base_shortcontents); } } if (defined($self->{'file_id_setting'}->{'sectioning_command_target_name'})) { ($target, $target_contents, $target_shortcontents, $filename) = &{$self->{'file_id_setting'}->{'sectioning_command_target_name'}}($self, $command, $target, $target_contents, $target_shortcontents, $filename); } if ($self->get_conf('DEBUG')) { print STDERR "Register $command->{'cmdname'} $target\n"; } $self->{'targets'}->{$command} = { 'target' => $target, 'section_filename' => $filename, }; _register_id($self, $target); if (defined($target_contents)) { $self->{'targets'}->{$command}->{'contents_target'} = $target_contents; _register_id($self, $target_contents); } else { $self->{'targets'}->{$command}->{'contents_target'} = ''; } if (defined($target_shortcontents)) { $self->{'targets'}->{$command}->{'shortcontents_target'} = $target_shortcontents; _register_id($self, $target_shortcontents); } else { $self->{'targets'}->{$command}->{'shortcontents_target'} = ''; } } # This set with two different codes # * the target information, id and normalized filename of 'identifiers_target', # ie everything that may be the target of a ref: @node, @float label, # @anchor, @namedanchor. # * The target information of sectioning elements # @node and section commands targets are therefore both set. # # conversion to HTML is done on-demand, upon call to command_text # and similar functions. # Note that 'node_filename', which is set here for Top target information # too, is not used later for Top anchors or links, see the NOTE below # associated with setting TOP_NODE_FILE_TARGET. sub _set_root_commands_targets_node_files($) { my $self = shift; my $sections_list; my $labels_list; if (exists($self->{'document'})) { $sections_list = $self->{'document'}->sections_list(); $labels_list = $self->{'document'}->labels_list(); } if (defined($labels_list)) { my $extension = ''; $extension = '.'.$self->get_conf('EXTENSION') if (defined($self->get_conf('EXTENSION')) and $self->get_conf('EXTENSION') ne ''); foreach my $target_element (@$labels_list) { next if (not exists($target_element->{'extra'}) or not $target_element->{'extra'}->{'is_target'}); my $label_element = Texinfo::Common::get_label_element($target_element); my ($node_filename, $target) = _normalized_label_id_file($self, $target_element->{'extra'} ->{'normalized'}, $label_element); $node_filename .= $extension; if (defined($self->{'file_id_setting'}->{'node_file_name'})) { # a non defined filename is ok if called with convert, but not # if output in files. We reset if undef, silently unless verbose # in case called by convert. my $user_node_filename = &{$self->{'file_id_setting'}->{'node_file_name'}}( $self, $target_element, $node_filename); if (defined($user_node_filename)) { $node_filename = $user_node_filename; } elsif ($self->get_conf('VERBOSE')) { $self->converter_document_warn(sprintf(__( "user-defined node file name not set for `%s'"), $node_filename)); } elsif ($self->get_conf('DEBUG')) { warn "user-defined node file name undef for `$node_filename'\n"; } } if ($self->get_conf('DEBUG')) { print STDERR 'Label' # uncomment to get the perl object names #."($target_element)" ." \@$target_element->{'cmdname'} $target, $node_filename\n"; } $self->{'targets'}->{$target_element} = {'target' => $target, 'node_filename' => $node_filename}; _register_id($self, $target); } } if (defined($sections_list)) { foreach my $section_relations (@{$sections_list}) { my $section_element = $section_relations->{'element'}; _new_sectioning_command_target($self, $section_element); } } } sub _set_heading_commands_targets($) { my $self = shift; my $global_commands; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); } if (defined($global_commands)) { foreach my $cmdname (sort(keys(%sectioning_heading_commands)), 'xrefname') { if (!exists($root_commands{$cmdname}) and exists($global_commands->{$cmdname})) { foreach my $command (@{$global_commands->{$cmdname}}) { _new_sectioning_command_target($self, $command); } } } } } sub _html_get_tree_root_element($$;$); # If $FIND_CONTAINER is set, the element that holds the command output # is found, otherwise the element that holds the command is found. This is # mostly relevant for footnote only. # If no known root element type is found, the returned root element is undef, # and not set to the element at the tree root sub _html_get_tree_root_element($$;$) { my ($self, $command, $find_container) = @_; # can be used to debug/understand what is going on #my $debug = 1; my $current = $command; #print STDERR "START ".Texinfo::Common::debug_print_element($current)."\n" if ($debug); my ($output_unit, $root_command); while (1) { if (exists($current->{'type'}) and $current->{'type'} eq 'special_unit_element') { return ($current->{'associated_unit'}, $current); } if (exists($current->{'cmdname'})) { if (exists($root_commands{$current->{'cmdname'}})) { $root_command = $current; #print STDERR "CMD ROOT $current->{'cmdname'}\n" if ($debug); } elsif (exists($block_commands{$current->{'cmdname'}}) and $block_commands{$current->{'cmdname'}} eq 'region') { if ($current->{'cmdname'} eq 'copying' and exists($self->{'document'})) { my $global_commands = $self->{'document'}->global_commands_information(); if (defined($global_commands) and exists($global_commands->{'insertcopying'})) { foreach my $insertcopying (@{$global_commands ->{'insertcopying'}}) { #print STDERR "INSERTCOPYING\n" if ($debug); my ($output_unit, $root_command) = _html_get_tree_root_element($self, $insertcopying, $find_container); return ($output_unit, $root_command) if (defined($output_unit) or defined($root_command)); } } } elsif ($current->{'cmdname'} eq 'titlepage' and $self->get_conf('USE_TITLEPAGE_FOR_TITLE') and $self->get_conf('SHOW_TITLE')) { #print STDERR "FOR titlepage document_units [0]\n" if ($debug); return ($self->{'document_units'}->[0], $self->{'document_units'}->[0]->{'unit_command'}); } die "Problem $output_unit, $root_command" if (defined($output_unit) or defined($root_command)); return (undef, undef); } elsif ($find_container) { # @footnote and possibly @*contents when a separate element is set my ($special_unit_variety, $special_unit, $class_base, $special_unit_direction) = $self->command_name_special_unit_information($current->{'cmdname'}); if (defined($special_unit)) { #print STDERR "SPECIAL $current->{'cmdname'}: $special_unit_variety ($special_unit_direction)\n" if ($debug); return ($special_unit, undef); } } } if (exists($current->{'associated_unit'})) { #print STDERR "ASSOCIATED_UNIT ".Texinfo::Common::debug_print_output_unit($current->{'associated_unit'})."\n" if ($debug); return ($current->{'associated_unit'}, $root_command); } elsif (exists($current->{'parent'})) { #print STDERR "PARENT ".Texinfo::Common::debug_print_element($current->{'parent'})."\n" if ($debug); $current = $current->{'parent'}; } else { #print STDERR "UNKNOWN ROOT ".Texinfo::Common::debug_print_element($current)."\n" if ($debug); return (undef, $root_command); } } } sub _html_set_pages_files($$$$$$$$) { my ($self, $output_units, $special_units, $associated_output_units, $output_file, $destination_directory, $output_filename, $document_name) = @_; my @filenames_order; my %unit_file_name_paths; # associate a file to the source information leading to set the file # name. Use the first element source information associated to a file. # The source information can be either a tree element associated to # the 'file_info_element' key, with a 'file_info_type' 'node' or # 'section'... or a specific source associated to the 'file_info_name' # key with 'file_info_type' 'special_file', or a source set if # nothing was found, with 'file_info_type' 'stand_in_file' and a # 'file_info_name'. Redirection files are added in the output() # function. my %files_source_info = (); if (!$self->get_conf('SPLIT')) { push @filenames_order, $output_filename; foreach my $output_unit (@$output_units) { $unit_file_name_paths{$output_unit} = $output_filename; } $files_source_info{$output_filename} = {'file_info_type' => 'special_file', 'file_info_name' => 'non_split', 'file_info_path' => $output_file}; } else { my $identifiers_target; if (exists($self->{'document'})) { $identifiers_target = $self->{'document'}->labels_information(); } # first determine the top node file name. my $node_top; $node_top = $identifiers_target->{'Top'} if (defined($identifiers_target)); my $top_node_filename = $self->top_node_filename($document_name); my $node_top_output_unit; if (defined($node_top) and defined($top_node_filename)) { $node_top_output_unit = $node_top->{'associated_unit'}; die "BUG: No output unit for top node" if (!defined($node_top_output_unit)); push @filenames_order, $top_node_filename; $unit_file_name_paths{$node_top_output_unit} = $top_node_filename; $files_source_info{$top_node_filename} = {'file_info_type' => 'special_file', 'file_info_name' => 'Top', 'file_info_path' => undef}; } my $file_nr = 0; my $extension = ''; $extension = '.'.$self->get_conf('EXTENSION') if (defined($self->get_conf('EXTENSION')) and $self->get_conf('EXTENSION') ne ''); foreach my $output_unit (@$output_units) { # For Top node. next if ($node_top_output_unit and $output_unit eq $node_top_output_unit); my $file_output_unit = $output_unit->{'first_in_page'}; if (!defined($file_output_unit)) { cluck ("No first_in_page for $output_unit\n"); } if (not exists($unit_file_name_paths{$file_output_unit})) { my $node_filename; foreach my $root_command (@{$file_output_unit->{'unit_contents'}}) { if (exists($root_command->{'cmdname'}) and $root_command->{'cmdname'} eq 'node') { # double node are not normalized, they are handled here if (!exists($root_command->{'extra'}) or !exists($root_command->{'extra'}->{'normalized'}) or !exists($identifiers_target->{ $root_command->{'extra'}->{'normalized'}})) { $node_filename = 'unknown_node'; $node_filename .= $extension; if (!exists($files_source_info{$node_filename})) { push @filenames_order, $node_filename; $files_source_info{$node_filename} = {'file_info_type' => 'stand_in_file', 'file_info_name' => 'unknown_node', 'file_info_path' => undef}; } } else { # Nodes with {'extra'}->{'is_target'} should always be in # 'identifiers_target', and thus in targets. It is a bug otherwise. $node_filename = $self->{'targets'}->{$root_command}->{'node_filename'}; if (not exists($files_source_info{$node_filename}) or $files_source_info{$node_filename} ->{'file_info_type'} ne 'stand_in_file') { push @filenames_order, $node_filename unless ($files_source_info{$node_filename}); $files_source_info{$node_filename} = {'file_info_type' => 'node', 'file_info_element' => $root_command, 'file_info_path' => undef}; } } $unit_file_name_paths{$file_output_unit} = $node_filename; last; } } if (not defined($node_filename)) { # use section to do the file name if there is no node my $command = $file_output_unit->{'unit_section'}; if (defined($command)) { if ($command->{'element'}->{'cmdname'} eq 'top' and !defined($node_top) and defined($top_node_filename)) { $unit_file_name_paths{$file_output_unit} = $top_node_filename; # existing top_node_filename can happen, see # html_tests.t top_file_name_and_node_name_collision push @filenames_order, $top_node_filename unless exists($files_source_info{$top_node_filename}); $files_source_info{$top_node_filename} = {'file_info_type' => 'special_file', 'file_info_name' => 'Top', 'file_info_path' => undef}; } else { my $section_filename = $self->{'targets'}->{$command->{'element'}} ->{'section_filename'}; $unit_file_name_paths{$file_output_unit} = $section_filename; if (not exists($files_source_info{$section_filename}) or $files_source_info{$section_filename} ->{'file_info_type'} ne 'stand_in_file') { push @filenames_order, $section_filename unless (exists($files_source_info{$section_filename})); $files_source_info{$section_filename} = {'file_info_type' => 'section', 'file_info_element' => $command->{'element'}, 'file_info_path' => undef}; } } } else { # when everything else has failed if ($file_nr == 0 and !defined($node_top) and defined($top_node_filename)) { $unit_file_name_paths{$file_output_unit} = $top_node_filename; unless (exists($files_source_info{$top_node_filename})) { push @filenames_order, $top_node_filename; $files_source_info{$top_node_filename} = {'file_info_type' => 'stand_in_file', 'file_info_name' => 'Top', 'file_info_path' => undef}; } } else { my $filename = $document_name . "_$file_nr"; $filename .= $extension; $unit_file_name_paths{$file_output_unit} = $filename; unless (exists($files_source_info{$filename})) { push @filenames_order, $filename; $files_source_info{$filename} = {'file_info_type' => 'stand_in_file', 'file_info_name' => 'unknown', 'file_info_path' => undef}; } } $file_nr++; } } } if ($output_unit ne $file_output_unit) { $unit_file_name_paths{$output_unit} = $unit_file_name_paths{$file_output_unit} } } } foreach my $output_unit (@$output_units) { my $filename = $unit_file_name_paths{$output_unit}; my $file_source_info = $files_source_info{$filename}; # check if (!defined($file_source_info)) { print STDERR "BUG: no files_source_info: $filename\n"; } my $filepath = $file_source_info->{'file_info_path'}; if (defined($self->{'file_id_setting'}->{'unit_file_name'})) { # NOTE the information that it is associated with @top or @node Top # may be determined with $self->unit_is_top_output_unit($output_unit); my ($user_filename, $user_filepath) = &{$self->{'file_id_setting'}->{'unit_file_name'}}( $self, $output_unit, $filename, $filepath); if (defined($user_filename)) { my $user_file_source_info; if (exists($files_source_info{$user_filename})) { $user_file_source_info = $files_source_info{$user_filename}; my $previous_filepath = $user_file_source_info->{'file_info_path'}; # It is likely that setting different paths for the same file is # not intended, so we warn. if (defined($user_filepath) and defined($previous_filepath) and $user_filepath ne $previous_filepath) { $self->converter_document_warn( sprintf(__("resetting %s file path %s to %s"), $user_filename, $previous_filepath, $user_filepath)); } elsif (defined($user_filepath) and !defined($previous_filepath)) { $self->converter_document_warn( sprintf(__("resetting %s file path from a relative path to %s"), $user_filename, $user_filepath)); } elsif (!defined($user_filepath) and defined($previous_filepath)) { $self->converter_document_warn( sprintf(__("resetting %s file path from %s to a relative path"), $user_filename, $previous_filepath)); } } $filename = $user_filename; push @filenames_order, $filename unless (defined($user_file_source_info)); $files_source_info{$filename} = {'file_info_type' => 'special_file', 'file_info_name' => 'user_defined', 'file_info_path' => $user_filepath}; } } $self->set_output_unit_file($output_unit, $filename); my $output_unit_filename = $output_unit->{'unit_filename'}; $self->{'file_counters'}->{$output_unit_filename} = 0 if (!exists($self->{'file_counters'}->{$output_unit_filename})); $self->{'file_counters'}->{$output_unit_filename}++; print STDERR 'Page ' # uncomment for perl object name #."$output_unit " .Texinfo::OutputUnits::output_unit_texi($output_unit) .": $output_unit_filename($self->{'file_counters'}->{$output_unit_filename})\n" if ($self->get_conf('DEBUG')); } if (defined($special_units)) { foreach my $special_unit (@$special_units) { my $unit_command = $special_unit->{'unit_command'}; my $filename = $self->{'targets'}->{$unit_command}->{'special_unit_filename'}; # Associate the special elements that have no page with the main page. # This may only happen if not split. if (!defined($filename) and defined($output_units->[0]->{'unit_filename'})) { $filename = $output_units->[0]->{'unit_filename'}; } if (defined($filename)) { push @filenames_order, $filename unless exists($files_source_info{$filename}); $self->set_output_unit_file($special_unit, $filename); $self->{'file_counters'}->{$filename} = 0 if (!exists($self->{'file_counters'}->{$filename})); $self->{'file_counters'}->{$filename}++; print STDERR 'Special page' # uncomment for perl object name #." $special_unit" .": $filename($self->{'file_counters'}->{$filename})\n" if ($self->get_conf('DEBUG')); my $file_source_info = {'file_info_element' => $unit_command, 'file_info_type' => 'special_unit', 'file_info_path' => undef}; $files_source_info{$filename} = $file_source_info unless(exists($files_source_info{$filename}) and $files_source_info{$filename}->{'file_info_type'} ne 'stand_in_file'); } } } foreach my $filename (@filenames_order) { $self->set_file_path($filename, $destination_directory, $files_source_info{$filename}->{'file_info_path'}); } # to be able to associate to the output unit file the associated # output units will be output into, this is done after document output # units got files. # In practice only used for contents and shortcontents. if (defined($associated_output_units) and scalar(@$associated_output_units)) { foreach my $special_unit (@$associated_output_units) { my $associated_output_unit = $special_unit->{'associated_document_unit'}; my $unit_command = $special_unit->{'unit_command'}; my $filename; my $command_target = $self->{'targets'}->{$unit_command}; # set by the user if (defined($command_target->{'special_unit_filename'})) { $filename = $command_target->{'special_unit_filename'}; } else { $filename = $associated_output_unit->{'unit_filename'} if ($associated_output_unit); $command_target->{'special_unit_filename'} = $filename; } # set here the file name, but do not associate a counter as it is already # set for the output unit the special output unit is in. $self->set_output_unit_file($special_unit, $filename) if (defined($filename)); } } return \%files_source_info; } # $ROOT is a parsed Texinfo tree. Return a list of the "elements" we need to # output in the HTML file(s). Each "element" is what can go in one HTML file, # such as the content between @node lines in the Texinfo source. # Also setup targets associated to tree elements and to elements associated # to special units. sub _prepare_conversion_units($$$) { my ($self, $document, $document_name) = @_; my ($output_units, $special_units, $associated_special_units); if ($self->get_conf('USE_NODES')) { $output_units = Texinfo::OutputUnits::split_by_node($document); } else { $output_units = Texinfo::OutputUnits::split_by_section($document); } # Needs to be set early in case it would be needed to find some region # command associated root command. $self->{'document_units'} = $output_units; # configuration used to determine if a special element is to be done # (in addition to contents) my @conf_for_special_units = ('footnotestyle'); $self->set_global_document_commands('last', \@conf_for_special_units); # NOTE if the last value of footnotestyle is separate, all the footnotes # formatted text are set to the special element set in _prepare_special_units # as _html_get_tree_root_element uses the Footnote direction for every # footnote. Therefore if @footnotestyle separate is set late in the # document the current value may not be consistent with the link obtained # for the footnote formatted text. This is not an issue, as the manual # says that @footnotestyle should only appear in the preamble, and it # makes sense to have something consistent in the whole document for # footnotes position. ($special_units, $associated_special_units) = _prepare_special_units($self, $output_units); # reset to the default $self->set_global_document_commands('before', \@conf_for_special_units); # Do that before the other elements, to be sure that special page ids # are registered before elements id are. _set_special_units_targets_files($self, $special_units, $document_name); _prepare_associated_special_units_targets($self, $associated_special_units); _set_root_commands_targets_node_files($self); _prepare_index_entries_targets($self); _prepare_footnotes_targets($self); _set_heading_commands_targets($self); $self->register_output_units_lists([$output_units, $special_units, $associated_special_units]); return ($output_units, $special_units, $associated_special_units); } sub _prepare_units_directions_files($$$$$$$$) { my ($self, $output_units, $special_units, $associated_special_units, $output_file, $destination_directory, $output_filename, $document_name) = @_; my $identifiers_target; my $nodes_list; if (exists($self->{'document'})) { $identifiers_target = $self->{'document'}->labels_information(); $nodes_list = $self->{'document'}->nodes_list(); } _prepare_output_units_global_targets($self, $output_units, $special_units, $associated_special_units); Texinfo::OutputUnits::split_pages($output_units, $nodes_list, $self->get_conf('SPLIT')); # reset even if $output_file eq '', as 'file_counters' is accessed # below, so it needs to be empty in the case of $output_file eq ''. $self->initialize_output_units_files(); # determine file names associated with the different pages, and setup # the counters for special element pages. my $files_source_info; if ($output_file ne '') { $files_source_info = _html_set_pages_files($self, $output_units, $special_units, $associated_special_units, $output_file, $destination_directory, $output_filename, $document_name); } # do output units directions. Texinfo::OutputUnits::units_directions($identifiers_target, $nodes_list, $output_units, $self->get_conf('DEBUG')); _prepare_special_units_directions($self, $special_units); # do output units directions related to files. # Here such that PrevFile and NextFile can be set. Texinfo::OutputUnits::units_file_directions($output_units); # elements_in_file_count is only set in HTML, not in # Texinfo::Convert::Converter $self->{'elements_in_file_count'} = {}; # condition could also be based on $output_file ne '' if (exists($self->{'file_counters'})) { # 'file_counters' is dynamic, decreased when the element is encountered # 'elements_in_file_count' is not modified afterwards foreach my $filename (keys(%{$self->{'file_counters'}})) { $self->{'elements_in_file_count'}->{$filename} = $self->{'file_counters'}->{$filename}; } } #if (1 or $self->get_conf('DEBUG') >= 30) { # if ($self->{'document'}) { # my $tree = $self->{'document'}->tree(); # my $use_filename = 0; # if ($self->get_conf('TEST')) { # $use_filename = 1; # } # my $output_units_output # = Texinfo::OutputUnits::print_output_units_tree_details($output_units, # $tree, $use_filename); # } #} return $files_source_info; } sub _register_special_unit($$) { my ($self, $special_unit_variety) = @_; my $special_unit = {'unit_type' => 'special_unit', 'special_unit_variety' => $special_unit_variety, 'directions' => {}}; # a "virtual" out of tree element used for targets my $unit_command = Texinfo::TreeElement::new({'type' => 'special_unit_element', 'associated_unit' => $special_unit}); $special_unit->{'unit_command'} = $unit_command; return $special_unit; } # prepare both special output units in separate output units, and # special output units associated to a regular document output unit, # output as part of regular output but also possible target of # special output unit direction. In practice, only contents and # shortcontents are associated with special output unit directions # and can be output as part of document output units. sub _prepare_special_units($$) { my ($self, $output_units) = @_; my $global_commands; my $sections_list; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); $sections_list = $self->{'document'}->sections_list(); } # for separate special output units my %do_special; # for associated special output units my $associated_special_units = []; if (defined($sections_list) and scalar(@{$sections_list}) > 1) { foreach my $cmdname ('shortcontents', 'contents') { my $special_unit_variety = $contents_command_special_unit_variety{$cmdname}; if ($self->get_conf($cmdname)) { my $contents_location = $self->get_conf('CONTENTS_OUTPUT_LOCATION'); if ($contents_location eq 'separate_element') { $do_special{$special_unit_variety} = 1; } else { my $associated_output_unit; if ($contents_location eq 'after_title') { $associated_output_unit = $output_units->[0]; } elsif ($contents_location eq 'after_top') { if (defined($global_commands) and exists($global_commands->{'top'})) { my $section_top = $global_commands->{'top'}; if (exists($section_top->{'associated_unit'})) { $associated_output_unit = $section_top->{'associated_unit'}; } } next unless ($associated_output_unit); } elsif ($contents_location eq 'inline') { if (defined($global_commands) and exists($global_commands->{$cmdname})) { foreach my $command(@{$global_commands->{$cmdname}}) { my $root_command; ($associated_output_unit, $root_command) = _html_get_tree_root_element($self, $command); if (defined($associated_output_unit)) { last; } } } else { next; } } else { # only happens with an unknown CONTENTS_OUTPUT_LOCATION next; } my $special_unit = _register_special_unit($self, $special_unit_variety); $special_unit->{'associated_document_unit'} = $associated_output_unit; push @$associated_special_units, $special_unit; } } } } if (defined($global_commands) and exists($global_commands->{'footnote'}) and scalar(@$output_units) > 1) { my $footnotestyle = $self->get_conf('footnotestyle'); if (defined($footnotestyle) and $footnotestyle eq 'separate') { $do_special{'footnotes'} = 1; } } if ((!defined($self->get_conf('DO_ABOUT')) and scalar(@$output_units) > 1 and ($self->get_conf('SPLIT') or $self->get_conf('HEADERS'))) or ($self->get_conf('DO_ABOUT'))) { $do_special{'about'} = 1; } my $special_units = []; # sort special elements according to their index order from # special_unit_info 'order'. # First reverse the hash, using arrays in case some elements are at the # same index, and sort to get alphabetically sorted special element # varieties that are at the same index. my %special_units_indices; foreach my $special_unit_variety (sort($self->get_special_unit_info_varieties('order'))) { next unless ($do_special{$special_unit_variety}); my $index = $self->special_unit_info('order', $special_unit_variety); $special_units_indices{$index} = [] if (not exists($special_units_indices{$index})); push @{$special_units_indices{$index}}, $special_unit_variety; } # now sort according to indices my @sorted_elements_varieties; foreach my $index (sort { $a <=> $b } (keys(%special_units_indices))) { push @sorted_elements_varieties, @{$special_units_indices{$index}}; } # Setup separate special output units my $previous_output_unit; $previous_output_unit = $output_units->[-1]; foreach my $special_unit_variety (@sorted_elements_varieties) { my $special_unit = _register_special_unit($self, $special_unit_variety); push @$special_units, $special_unit; if (defined($previous_output_unit)) { $special_unit->{'tree_unit_directions'} = {}; $previous_output_unit->{'tree_unit_directions'} = {} if (not exists($previous_output_unit->{'tree_unit_directions'})); $special_unit->{'tree_unit_directions'}->{'prev'} = $previous_output_unit; $previous_output_unit->{'tree_unit_directions'}->{'next'} = $special_unit; } $previous_output_unit = $special_unit; } return $special_units, $associated_special_units; } sub _set_special_units_targets_files($$$) { my ($self, $special_units, $document_name) = @_; my $extension = ''; $extension = $self->get_conf('EXTENSION') if (defined($self->get_conf('EXTENSION'))); foreach my $special_unit (@$special_units) { my $special_unit_variety = $special_unit->{'special_unit_variety'}; # it may be undef'ined in user customization code my $target = $self->special_unit_info('target', $special_unit_variety); next if (!defined($target)); my $default_filename; if ($self->get_conf('SPLIT') or !$self->get_conf('MONOLITHIC') # in general $document_name not defined means called through convert and defined($document_name)) { my $special_unit_file_string = $self->special_unit_info('file_string', $special_unit_variety); $special_unit_file_string = '' if (!defined($special_unit_file_string)); $default_filename = $document_name . $special_unit_file_string; $default_filename .= '.'.$extension if (defined($extension)); } else { $default_filename = undef; } my $filename; if (defined($self->{'file_id_setting'}->{'special_unit_target_file_name'})) { ($target, $filename) = &{$self->{'file_id_setting'}->{'special_unit_target_file_name'}}( $self, $special_unit, $target, $default_filename); } $filename = $default_filename if (!defined($filename)); if ($self->get_conf('DEBUG')) { my $fileout = $filename; $fileout = 'UNDEF' if (!defined($fileout)); print STDERR 'Add special' # uncomment for the perl object name #." $special_unit" ." $special_unit_variety: target $target,\n". " filename $fileout\n"; } my $unit_command = $special_unit->{'unit_command'}; $self->{'targets'}->{$unit_command} = {'target' => $target, 'special_unit_filename' => $filename, }; _register_id($self, $target); } } sub _prepare_associated_special_units_targets($$) { my ($self, $associated_output_units) = @_; return unless (defined($associated_output_units)); foreach my $special_unit (@$associated_output_units) { my $special_unit_variety = $special_unit->{'special_unit_variety'}; # it may be undef'ined in user customization code my $target = $self->special_unit_info('target', $special_unit_variety); my $default_filename; my $filename; if (defined($self->{'file_id_setting'}->{'special_unit_target_file_name'})) { ($target, $filename) = &{$self->{'file_id_setting'}->{'special_unit_target_file_name'}}( $self, $special_unit, $target, $default_filename); } $filename = $default_filename if (!defined($filename)); if ($self->get_conf('DEBUG')) { my $str_filename = $filename; $str_filename = 'UNDEF (default)' if (not defined($str_filename)); my $str_target = $target; $str_target = 'UNDEF' if (not defined($str_target)); print STDERR 'Add content' # uncomment to get the perl object name #." $special_unit" ." $special_unit_variety: target $str_target,\n". " filename $str_filename\n"; } my $unit_command = $special_unit->{'unit_command'}; my $command_target = {'target' => $target}; $self->{'targets'}->{$unit_command} = $command_target; if (defined($target)) { _register_id($self, $target); } if (defined ($filename)) { $command_target->{'special_unit_filename'} = $filename; } } } sub _prepare_special_units_directions($$) { my ($self, $special_units) = @_; return unless(defined($special_units)); foreach my $special_unit (@$special_units) { $special_unit->{'directions'}->{'This'} = $special_unit; } } # Associate output units to the global targets, First, Last, Top, Index. sub _prepare_output_units_global_targets($$$$) { my ($self, $output_units, $special_units, $associated_special_units) = @_; $self->{'global_units_directions'}->{'First'} = $output_units->[0]; $self->{'global_units_directions'}->{'Last'} = $output_units->[-1]; $self->{'global_units_directions'}->{'Top'} = _get_top_unit($self, $output_units); my $global_commands; my $nodes_list; my $sections_list; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); $nodes_list = $self->{'document'}->nodes_list(); $sections_list = $self->{'document'}->sections_list(); } # Associate Index with the last @printindex. According to Werner Lemberg, # "the most general index is normally the last one, not the first" # https://lists.gnu.org/archive/html/bug-texinfo/2025-01/msg00019.html # # It is always the last printindex, even if it is not output (for example # it is in @copying and @titlepage, which are certainly wrong constructs). if (defined($global_commands) and exists($global_commands->{'printindex'})) { # Here document_unit can only be a document unit, or maybe undef if there # are no document unit at all my ($document_unit, $root_command) = _html_get_tree_root_element($self, $global_commands->{'printindex'}->[-1]); if (defined($document_unit)) { if (defined($root_command)) { my $section_relations; if ($root_command->{'cmdname'} eq 'node') { if (defined($nodes_list)) { my $node_relations = $nodes_list->[$root_command->{'extra'}->{'node_number'} -1]; if (exists($node_relations->{'associated_section'})) { $section_relations = $node_relations->{'associated_section'}; } } } else { $section_relations = $sections_list->[$root_command->{'extra'}->{'section_number'} -1]; } # find the first level 1 sectioning element to associate the printindex # with. May not work correctly if structuring was not done if ($section_relations) { my $current_command = $section_relations->{'element'}; while (exists($current_command->{'extra'}) and defined($current_command->{'extra'}->{'section_level'}) and $current_command->{'extra'}->{'section_level'} > 1 and exists($section_relations->{'section_directions'}) and exists($section_relations->{'section_directions'}->{'up'}) and exists($section_relations->{'section_directions'}->{'up'} ->{'element'}->{'associated_unit'})) { $section_relations = $section_relations->{'section_directions'}->{'up'}; $current_command = $section_relations->{'element'}; $document_unit = $current_command->{'associated_unit'}; } } } $self->{'global_units_directions'}->{'Index'} = $document_unit; } } if ($self->{'customized_global_directions'}) { foreach my $direction (sort(keys(%{$self->{'customized_global_directions'}}))) { my $node_texi_name = $self->{'customized_global_directions'}->{$direction}; if (defined($node_texi_name) and not defined($self->global_direction_text($direction))) { # FIXME check that relative directions are not replaced by # global_units_directions (as done in C)? It may not be an issue. # Determine the document unit corresponding to the direction # node name Texinfo code # Parse the customized direction node name Texinfo code my $node_element; my $parser = Texinfo::Parser::parser({'NO_INDEX' => 1, 'NO_USER_COMMANDS' => 1,}); my $tree = $parser->parse_texi_line($node_texi_name, undef, 1); my $errors = $parser->errors(); my $errors_count = Texinfo::Report::count_errors($errors); if ($errors_count) { warn "Global $direction node name parsing $errors_count error(s)\n"; warn "node name: $node_texi_name\n"; warn "Error messages: \n"; foreach my $error_message (@$errors) { warn $error_message->{'error_line'}; } } # convert to identifier and determine the node element target if ($tree) { my $normalized_node = Texinfo::Convert::NodeNameNormalization::convert_to_node_identifier($tree); if ($normalized_node ne '' and $normalized_node =~ /[^-]/) { $node_element = $self->label_command($normalized_node); } } if (!defined($node_element)) { $self->converter_document_warn( sprintf(__("could not find %s node `%s'"), $direction, $node_texi_name)); } else { $self->{'global_units_directions'}->{$direction} = $node_element->{'associated_unit'}; } } } } if ($self->get_conf('DEBUG')) { # TODO also show the global directions added by the user print STDERR "GLOBAL DIRECTIONS:\n"; foreach my $global_direction (@global_directions_order) { if (defined($self->global_direction_unit($global_direction))) { my $global_unit = $self->global_direction_unit($global_direction); print STDERR " $global_direction" # uncomment to get the perl object name # ."($global_unit)" .': '. Texinfo::OutputUnits::output_unit_texi($global_unit)."\n"; } } print STDERR "\n"; } # TODO show the special units directions if DEBUG is set? foreach my $units_list ($special_units, $associated_special_units) { if (defined($units_list) and scalar(@$units_list)) { foreach my $special_unit (@$units_list) { my $special_unit_variety = $special_unit->{'special_unit_variety'}; my $special_unit_direction = $self->special_unit_info('direction', $special_unit_variety); $self->{'global_units_directions'}->{$special_unit_direction} = $special_unit; } } } } sub _prepare_index_entries_targets($) { my $self = shift; my $indices_information; if (exists($self->{'document'})) { $indices_information = $self->{'document'}->indices_information(); } if (defined($indices_information)) { my $no_unidecode; $no_unidecode = 1 if (defined($self->get_conf('USE_UNIDECODE')) and !$self->get_conf('USE_UNIDECODE')); my $in_test; $in_test = 1 if ($self->get_conf('TEST')); foreach my $index_name (sort(keys(%$indices_information))) { foreach my $index_entry (@{$indices_information->{$index_name} ->{'index_entries'}}) { my $main_entry_element = $index_entry->{'entry_element'}; # does not refer to the document my $seeentry = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seeentry'); next if (defined($seeentry)); my $seealso = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seealso'); next if (defined($seealso)); my $region = ''; $region = "$main_entry_element->{'extra'}->{'element_region'}-" if (defined($main_entry_element->{'extra'}->{'element_region'})); my $entry_reference_content_element = Texinfo::Common::index_content_element($main_entry_element, 1); # construct element to convert to a normalized identifier to use as # hrefs target my $normalize_index_element = Texinfo::TreeElement::new( {'contents' => [$entry_reference_content_element]}); my $subentries_tree = Texinfo::Convert::Utils::comma_index_subentries_tree( $main_entry_element, ' '); if (defined($subentries_tree)) { push @{$normalize_index_element->{'contents'}}, @{$subentries_tree->{'contents'}}; } my $normalized_index = Texinfo::Convert::NodeNameNormalization::normalize_transliterate_texinfo( $normalize_index_element, $in_test, $no_unidecode); my $target_base = "index-" . $region .$normalized_index; my $target = _unique_target($self, $target_base); _register_id($self, $target); my $target_element = $main_entry_element; $target_element = $index_entry->{'entry_associated_element'} if ($index_entry->{'entry_associated_element'}); $self->{'targets'}->{$target_element} = {'target' => $target, }; } } } } sub _prepare_footnotes_targets($) { my $self = shift; my $footid_base = 'FOOT'; my $docid_base = 'DOCF'; my $global_commands; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); } if (defined($global_commands) and exists($global_commands->{'footnote'})) { my $footnote_nr = 0; foreach my $footnote (@{$global_commands->{'footnote'}}) { $footnote_nr++; my $nr = $footnote_nr; # anchor for the footnote text my $footid = $footid_base.$nr; # anchor for the location of the @footnote in the document my $docid = $docid_base.$nr; while (_id_is_registered($self, $docid) or _id_is_registered($self, $footid)) { $nr++; $footid = $footid_base.$nr; $docid = $docid_base.$nr; # Avoid integer overflow die if ($nr == 0); } _register_id($self, $footid); _register_id($self, $docid); $self->{'targets'}->{$footnote} = { 'target' => $footid }; $self->{'special_targets'}->{'footnote_location'}->{$footnote} = { 'target' => $docid }; print STDERR 'Enter footnote' # uncomment for the perl object name #." $footnote" .": target $footid, nr $footnote_nr\n" .Texinfo::Convert::Texinfo::convert_to_texinfo($footnote)."\n" if ($self->get_conf('DEBUG')); } } } sub _source_info_id($) { my $source_info = shift; my $result; if (exists($source_info->{'file_name'})) { $result = $source_info->{'file_name'}; } else { $result = ''; } $result .= '-'; if (exists($source_info->{'macro'})) { $result .= $source_info->{'macro'}; } $result .= '-'; if (exists($source_info->{'line_nr'})) { $result .= $source_info->{'line_nr'}; } else { $result .= '0'; } return $result; } sub _check_htmlxref_already_warned($$$) { my ($self, $manual_name, $source_info) = @_; my $node_manual_key; if (defined($source_info)) { $node_manual_key = _source_info_id($source_info).'-'.$manual_name; } else { $node_manual_key = 'UNDEF-'.$manual_name; } if (exists($self->{'check_htmlxref_already_warned'}->{$node_manual_key})) { return 1; } else { $self->{'check_htmlxref_already_warned'}->{$node_manual_key} = 1; return 0; } } # returns file base name, extension and anchor associated to node # (anchor, float...) command adhering strictly to the HTML Xref specification. # The $CROSSREF_EXTENSION argument should be the external crossreference # filename extension, if undef, the $EXTENSION argument is used. sub standard_label_id_file($$$$$) { my ($self, $normalized, $label_element, $crossref_extension, $extension) = @_; my $target; my $filename; if (!defined($normalized) and defined($label_element)) { $normalized = Texinfo::Convert::NodeNameNormalization::convert_to_node_identifier( $label_element); } my $options = \%Texinfo::Options::converter_customization_options; if (defined($normalized)) { $target = _normalized_to_id($normalized); # use default, not user-defined value my $basefilename_length = $options->{'BASEFILENAME_LENGTH'}; $filename = substr($normalized, 0, $basefilename_length); } else { $target = ''; $filename = ''; } # to find out the Top node, one could check $normalized if (defined($self->{'file_id_setting'}->{'label_target_name'})) { $target = &{$self->{'file_id_setting'}->{'label_target_name'}}($self, $normalized, $label_element, $target); } my $file_extension = ''; my $external_extension = $crossref_extension; $external_extension = $extension if (not defined($external_extension)); $file_extension = '.' . $external_extension if (defined($external_extension) and $external_extension ne ''); return ($filename, $file_extension, $target); } sub _external_node_href($$$) { my ($self, $external_node, # for messages only $source_command) = @_; my $normalized = $external_node->{'extra'}->{'normalized'}; my $node_contents = $external_node->{'extra'}->{'node_content'}; #print STDERR "external_node: ".join('|', keys(%$external_node))."\n"; my ($target_filebase, $external_file_extension, $target) = $self->standard_label_id_file($normalized, $node_contents, $self->get_conf('EXTERNAL_CROSSREF_EXTENSION'), $defaults{'EXTENSION'}); # always undef if conversion is called through convert() my $default_target_split = $self->get_conf('EXTERNAL_CROSSREF_SPLIT'); # initialize to $default_target_split my $is_target_split; if ($default_target_split) { $is_target_split = 1; } else { $is_target_split = 0; } # used if !$is_target_split my $file = ''; # used if $is_target_split my $directory = ''; if (exists($external_node->{'extra'}->{'manual_content'})) { Texinfo::Convert::Text::set_options_code($self->{'convert_text_options'}); my $manual_name = Texinfo::Convert::Text::convert_to_text( $external_node->{'extra'}->{'manual_content'}, $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_code($self->{'convert_text_options'}); if ($self->get_conf('IGNORE_REF_TO_TOP_NODE_UP') and $target eq '') { my $top_node_up = $self->get_conf('TOP_NODE_UP'); if (defined($top_node_up) and "($manual_name)" eq $top_node_up) { return ''; } } my $manual_base = $manual_name; # in 2023 there were manuals with .info. Warning added in 2024. if ($manual_base =~ s/(\.info?)$//) { $self->converter_line_warn(sprintf(__( "do not set %s suffix in reference for manual `%s'"), $1, $manual_name), $source_command->{'source_info'}); } $manual_base =~ s/^.*\///; my $split_found; my $htmlxref_href; my $htmlxref_mode = $self->get_conf('HTMLXREF_MODE'); if (!defined($htmlxref_mode) or $htmlxref_mode ne 'none') { if (exists($self->{'htmlxref'}->{$manual_base})) { my $htmlxref_info = $self->{'htmlxref'}->{$manual_base}; my $document_split = $self->get_conf('SPLIT'); $document_split = 'mono' if (!$document_split); foreach my $split_ordered (@{$htmlxref_entries{$document_split}}) { if (exists($htmlxref_info->{$split_ordered})) { $split_found = $split_ordered; if ($htmlxref_info->{$split_ordered} ne '') { $htmlxref_href = $self->url_protect_url_text($htmlxref_info->{$split_ordered}); } last; } } } if (defined($split_found)) { if ($split_found eq 'mono') { $is_target_split = 0; } else { $is_target_split = 1; } } else { # nothing specified for that manual, use default if ($self->get_conf('CHECK_HTMLXREF')) { if (defined($source_command) and $source_command->{'source_info'}) { if (!_check_htmlxref_already_warned($self, $manual_name, $source_command->{'source_info'})) { $self->converter_line_warn(sprintf(__( "no HTML cross-references entry found for `%s'"), $manual_name), $source_command->{'source_info'}); } } else { if (!_check_htmlxref_already_warned($self, $manual_name, undef)) { $self->converter_document_warn(sprintf(__( "no HTML cross-references entry found for `%s'"), $manual_name), ); cluck; } } } } } if ($is_target_split) { if (defined($htmlxref_href)) { $directory = $htmlxref_href; } else { if (defined($self->get_conf('EXTERNAL_DIR'))) { $directory = $self->get_conf('EXTERNAL_DIR')."/$manual_base"; } elsif ($self->get_conf('SPLIT')) { $directory = "../$manual_base"; } my $output_format = $self->get_conf('TEXINFO_OUTPUT_FORMAT'); if (defined($output_format) and $output_format ne '') { $directory .= '_'.$output_format; } $directory = $self->url_protect_file_text($directory); } $directory .= "/"; } else {# target not split if (defined($htmlxref_href)) { $file = $htmlxref_href; } else { if (defined($self->get_conf('EXTERNAL_DIR'))) { $file = $self->get_conf('EXTERNAL_DIR')."/$manual_base"; } elsif ($self->get_conf('SPLIT')) { $file = "../$manual_base"; } else { $file = $manual_base; } $file .= $external_file_extension; $file = $self->url_protect_file_text($file); } } } if ($is_target_split) { my $file_name; if (($target eq 'Top' or $target eq '') and defined($self->get_conf('TOP_NODE_FILE_TARGET'))) { $file_name = $self->get_conf('TOP_NODE_FILE_TARGET'); } else { $file_name = $target_filebase . $external_file_extension; } if (defined($self->{'file_id_setting'}->{'external_target_split_name'})) { ($target, $directory, $file_name) = &{$self->{'file_id_setting'}->{'external_target_split_name'}}($self, $normalized, $external_node, $target, $directory, $file_name); $directory = '' if (!defined($directory)); $file_name = '' if (!defined($file_name)); $target = '' if (!defined($target)); } my $result = $directory . $file_name; if ($target ne '') { $result .= '#' . $target; } return $result; } else { if ($target eq '') { $target = 'Top'; } if (defined($self->{'file_id_setting'}->{ 'external_target_non_split_name'})) { ($target, $file) = &{$self->{'file_id_setting'}->{'external_target_non_split_name'}}($self, $normalized, $external_node, $target, $file); $file = '' if (!defined($file)); $target = '' if (!defined($target)); } my $result = $file; if ($target ne '') { $result .= '#' . $target; } return $result; } } # Output a list of the nodes immediately below this one sub _mini_toc($$) { my ($self, $section_relations) = @_; my $result = ''; my $entry_index = 0; if (defined($section_relations) and exists($section_relations->{'section_children'}) and scalar(@{$section_relations->{'section_children'}})) { $result .= $self->html_attribute_class('ul', ['mini-toc']).">\n"; foreach my $section_relations (@{$section_relations->{'section_children'}}) { my $section = $section_relations->{'element'}; my $text = $self->command_text($section, 'text_nonumber'); # could happen with empty sectioning command next if (!defined($text) or $text eq ''); $entry_index++; my $accesskey = ''; $accesskey = " accesskey=\"$entry_index\"" if ($self->get_conf('USE_ACCESSKEY') and $entry_index < 10); my $href = $self->command_href($section); $result .= "
  • "; if (defined($href)) { $result .= "$text"; } else { $result .= $text; } $result .= "
  • \n"; } $result .= "\n"; } return $result; } sub _default_format_contents($$;$$) { my ($self, $cmdname, $command, $filename) = @_; $filename = $self->current_filename() if (!defined($filename)); my $document = $self->get_info('document'); my $sections_list; my $sectioning_root; if (defined($document)) { $sections_list = $document->sections_list(); $sectioning_root = $document->sectioning_root(); } return '' if (!defined($sections_list) or !scalar(@$sections_list) # this should not happen with $sections_list as set from Structuring # sectioning_structure, but could happen with another source. # We consider that if sectioning_root is set as usual, all the # fields are set consistently with what sectioning_structure would # have set. or !defined($sectioning_root)); my $is_contents; $is_contents = 1 if ($cmdname eq 'contents'); my $min_root_level = $sectioning_root->{'section_children'}->[0] ->{'element'}->{'extra'}->{'section_level'}; my $max_root_level = $min_root_level; foreach my $top_relations (@{$sectioning_root->{'section_children'}}) { my $top_section = $top_relations->{'element'}; $min_root_level = $top_section->{'extra'}->{'section_level'} if ($top_section->{'extra'}->{'section_level'} < $min_root_level); $max_root_level = $top_section->{'extra'}->{'section_level'} if ($top_section->{'extra'}->{'section_level'} > $max_root_level); } # chapter level elements are considered top-level here. $max_root_level = 1 if ($max_root_level < 1); #print STDERR "ROOT_LEVEL Max: $max_root_level, Min: $min_root_level\n"; my @toc_ul_classes; push @toc_ul_classes, 'toc-numbered-mark' if ($self->get_conf('NUMBER_SECTIONS')); my $result = ''; if ($is_contents and !defined($self->get_conf('BEFORE_TOC_LINES')) or (!$is_contents and !defined($self->get_conf('BEFORE_SHORT_TOC_LINES')))) { $result .= $self->html_attribute_class('div', [$cmdname]).">\n"; } elsif($is_contents) { $result .= $self->get_conf('BEFORE_TOC_LINES'); } else { $result .= $self->get_conf('BEFORE_SHORT_TOC_LINES'); } my $has_toplevel_contents; if (@{$sectioning_root->{'section_children'}} > 1) { $result .= $self->html_attribute_class('ul', \@toc_ul_classes) .">\n"; $has_toplevel_contents = 1; } my $link_to_toc = (!$is_contents and $self->get_conf('SHORT_TOC_LINK_TO_TOC') and ($self->get_conf('contents')) and ($self->get_conf('CONTENTS_OUTPUT_LOCATION') ne 'inline' or _has_contents_or_shortcontents($self))); foreach my $top_relations (@{$sectioning_root->{'section_children'}}) { my $section_relations = $top_relations; SECTION: while (defined($section_relations)) { my $section = $section_relations->{'element'}; if ($section->{'cmdname'} ne 'top') { my $text = $self->command_text($section); my $href; if ($link_to_toc) { $href = $self->command_contents_href($section, 'contents', $filename); } else { $href = $self->command_href($section, $filename); } my $toc_id = $self->command_contents_target($section, $cmdname); if ($text ne '') { # no indenting for shortcontents $result .= (' ' x (2*($section->{'extra'}->{'section_level'} - $min_root_level))) if ($is_contents); $result .= "
  • "; if ($toc_id ne '' or defined($href)) { $result .= "{'associated_node'}) and $section_relations->{'associated_node'} ->{'element'}->{'extra'}->{'isindex'}) { $result .= ' rel="index"'; } $result .= ">$text"; } else { $result .= $text; } } } elsif (exists($section_relations->{'section_children'}) and scalar(@{$section_relations->{'section_children'}}) and $has_toplevel_contents) { $result .= "
  • "; } # for shortcontents don't do child if child is not toplevel if (exists($section_relations->{'section_children'}) and ($is_contents or $section->{'extra'}->{'section_level'} < $max_root_level)) { # no indenting for shortcontents $result .= "\n" . ' ' x (2*($section->{'extra'}->{'section_level'} - $min_root_level)) if ($is_contents); $result .= $self->html_attribute_class('ul', \@toc_ul_classes) .">\n"; $section_relations = $section_relations->{'section_children'}->[0]; } elsif (exists($section_relations->{'section_directions'}) and exists($section_relations->{'section_directions'}->{'next'}) and $section->{'cmdname'} ne 'top') { $result .= "
  • \n"; last if ($section_relations eq $top_relations); $section_relations = $section_relations->{'section_directions'}->{'next'}; } else { #last if ($section eq $top_section); if ($section_relations eq $top_relations) { $result .= "\n" unless ($section->{'cmdname'} eq 'top'); last; } while (exists($section_relations->{'section_directions'}) and exists($section_relations->{'section_directions'}->{'up'})) { $section_relations = $section_relations->{'section_directions'}->{'up'}; $section = $section_relations->{'element'}; $result .= "\n" . ' ' x (2*($section->{'extra'}->{'section_level'} - $min_root_level)) . ""; if ($section_relations eq $top_relations) { $result .= "\n" if ($has_toplevel_contents); last SECTION; } if (exists($section_relations->{'section_directions'}) and exists($section_relations->{'section_directions'} ->{'next'})) { $result .= "\n"; $section_relations = $section_relations->{'section_directions'}->{'next'}; last; } } } } } if (scalar(@{$sectioning_root->{'section_children'}}) > 1) { $result .= "\n"; } if ($is_contents and !defined($self->get_conf('AFTER_TOC_LINES')) or (!$is_contents and !defined($self->get_conf('AFTER_SHORT_TOC_LINES')))) { $result .= "\n\n"; } elsif ($is_contents) { $result .= $self->get_conf('AFTER_TOC_LINES'); } else { $result .= $self->get_conf('AFTER_SHORT_TOC_LINES'); } return $result; } sub _default_format_program_string($) { my $self = shift; if (defined($self->get_conf('PROGRAM')) and $self->get_conf('PROGRAM') ne '' and defined($self->get_conf('PACKAGE_URL'))) { return $self->convert_tree( $self->cdt('This document was generated on @emph{@today{}} using @uref{{program_homepage}, @emph{{program}}}.', { 'program_homepage' => Texinfo::TreeElement::new( {'text' => $self->get_conf('PACKAGE_URL')}), 'program' => Texinfo::TreeElement::new( {'text' => $self->get_conf('PROGRAM')}) }), 'Tr program string program'); } else { return $self->convert_tree( $self->cdt('This document was generated on @emph{@today{}}.'), 'Tr program string date'); } } sub _default_format_end_file($$$) { my ($self, $filename, $output_unit) = @_; my $result = ''; if ($self->get_conf('PROGRAM_NAME_IN_FOOTER')) { $result .= "

    \n "; my $open = $self->html_attribute_class('span', ['program-in-footer']); $result .= $open.'>' if ($open ne ''); my $program_string = &{$self->formatting_function('format_program_string')}($self); $result .= $program_string; $result .= '' if ($open ne ''); $result .= "\n

    "; } $result .= "\n\n"; my $pre_body_close = $self->get_conf('PRE_BODY_CLOSE'); $result .= $pre_body_close if (defined($pre_body_close)); my $jslicenses = $self->get_info('jslicenses'); if (defined($jslicenses) and ((exists($jslicenses->{'infojs'}) and scalar(keys %{$jslicenses->{'infojs'}})) or (($self->get_file_information('mathjax', $filename) or !$self->get_conf('SPLIT')) and (exists($jslicenses->{'mathjax'}) and scalar(keys %{$jslicenses->{'mathjax'}}))))) { my $js_setting = $self->get_conf('JS_WEBLABELS'); my $js_path = $self->get_conf('JS_WEBLABELS_FILE'); if (defined($js_setting) and defined($js_path) and ($js_setting eq 'generate' or $js_setting eq 'reference')) { $result .= '' .$self->convert_tree($self->cdt('JavaScript license information'), 'Tr JS license header') .''; } } return "$result "; } sub _root_html_element_attributes_string($) { my $self = shift; if (defined($self->get_conf('HTML_ROOT_ELEMENT_ATTRIBUTES')) and $self->get_conf('HTML_ROOT_ELEMENT_ATTRIBUTES') ne '') { return ' '.$self->get_conf('HTML_ROOT_ELEMENT_ATTRIBUTES'); } return ''; } # This is used for normal output files and other files, like # redirection file headers. $COMMAND is the tree element for # a @node that is being output in the file. sub _file_header_information($$;$) { my ($self, $command, $filename) = @_; my $title; my $command_description; my $title_string = $self->get_info('title_string'); if (defined($command)) { my $command_string; if ($self->get_conf('SECTION_NAME_IN_TITLE') and exists($command->{'cmdname'}) and $command->{'cmdname'} eq 'node') { my $associated_title_command; my $document = $self->get_info('document'); if (defined($document)) { my $nodes_list = $document->nodes_list(); my $node_relations = $nodes_list->[$command->{'extra'}->{'node_number'} -1]; $associated_title_command = $node_relations->{'associated_title_command'}; if (defined($associated_title_command)) { $command_string = $self->command_text($associated_title_command, 'string'); } } } if (!defined($command_string) or $command_string eq '') { # happens for @node and special_unit_element. # Also for @anchor and @namedanchor for redirection files. # TODO use command_name? If SECTION_NAME_IN_TITLE? $command_string = $self->command_text($command, 'string'); } if (defined($command_string) and $command_string ne '' and $command_string ne $title_string) { # TRANSLATORS: sectioning element title for the page header my $title_tree = $self->cdt('{element_text} ({title})', {'title' => {'text' => $title_string, 'type' => '_converted'}, 'element_text' => {'text' => $command_string, 'type' => '_converted'}}); my $context_str = 'file_header_title-element-'; if (exists($command->{'cmdname'})) { $context_str .= '@'.$command->{'cmdname'}; } elsif (exists($command->{'type'})) { $context_str .= $command->{'type'}; } # NOTE 'element_title' is not unique although this could be called # for each file. We are in string context, though, so it is # probably not important. $title = $self->convert_tree_new_formatting_context($title_tree, $context_str, $CTXF_string, 'element_title'); } $command_description = $self->command_description($command, 'string'); } $title = $title_string if (!defined($title)); my $keywords = $command_description; $keywords = $title if (not defined($keywords) or $keywords eq ''); my $description = $self->get_info('documentdescription_string'); $description = $command_description if (not defined($description) or $description eq ''); $description = $title if (not defined($description) or $description eq ''); $description = $self->close_html_lone_element( "close_html_lone_element( "get_conf('OUTPUT_ENCODING_NAME')) and ($self->get_conf('OUTPUT_ENCODING_NAME') ne '')); my $date = ''; if ($self->get_conf('DATE_IN_HEADER')) { my $today = $self->convert_tree_new_formatting_context( Texinfo::TreeElement::new({'cmdname' => 'today'}), 'DATE_IN_HEADER'); $date = $self->close_html_lone_element( "formatting_function('format_css_lines')}($self, $filename); my $doctype = $self->get_conf('DOCTYPE'); $doctype = '' if (!defined($doctype)); my $root_html_element_attributes = _root_html_element_attributes_string($self); my $body_attributes = $self->get_conf('BODY_ELEMENT_ATTRIBUTES'); if (!defined($body_attributes)) { $body_attributes = ''; } elsif ($body_attributes =~ /^[^ ]/) { $body_attributes = ' '.$body_attributes; } if ($self->get_conf('HTML_MATH') and $self->get_conf('HTML_MATH') eq 'mathjax' and $self->get_file_information('mathjax', $filename)) { $body_attributes .= ' class="tex2jax_ignore"'; } my $copying_comment = $self->get_info('copying_comment'); $copying_comment = '' if (not defined($copying_comment)); my $after_body_open = $self->get_conf('AFTER_BODY_OPEN'); $after_body_open = '' if (!defined($after_body_open)); my $program_and_version = $self->get_conf('PACKAGE_AND_VERSION'); $program_and_version = '' if (!defined($program_and_version)); my $program_homepage = $self->get_conf('PACKAGE_URL'); $program_homepage = '' if (!defined($program_homepage)); my $program = $self->get_conf('PROGRAM'); my $generator = ''; if (defined($program) and $program ne '') { $generator = $self->close_html_lone_element( "get_conf('EXTRA_HEAD') if (defined($self->get_conf('EXTRA_HEAD'))); if (defined($self->get_conf('INFO_JS_DIR'))) { if (!$self->get_conf('SPLIT')) { $self->converter_document_error( sprintf(__("%s not meaningful for non-split output"), 'INFO_JS_DIR')); } else { my $jsdir = $self->get_conf('INFO_JS_DIR'); if ($jsdir eq '.') { $jsdir = ''; } else { $jsdir =~ s,/*$,/,; # append a single slash } my $protected_jsdir = $self->url_protect_url_text($jsdir); $extra_head .= $self->close_html_lone_element( ''."\n" .''; } } if ((defined($self->get_conf('HTML_MATH')) and $self->get_conf('HTML_MATH') eq 'mathjax') and ($self->get_file_information('mathjax', $filename))) { my $mathjax_script = $self->get_conf('MATHJAX_SCRIPT'); my $default_mathjax_configuration = " options: { skipHtmlTags: {'[-]': ['pre']}, // do not skip pre ignoreHtmlClass: 'tex2jax_ignore', processHtmlClass: 'tex2jax_process' }, tex: { processEscapes: false, // do not use \\\$ to produce a literal dollar sign processEnvironments: false, // do not process \\begin{xxx}...\\end{xxx} outside math mode processRefs: false, // do not process \\ref{...} outside of math mode displayMath: [ // start/end delimiter pairs for display math ['\\\\[', '\\\\]'] ], },"; $extra_head .= "'; } return ($title, $description, $keywords, $encoding, $date, $css_lines, $doctype, $root_html_element_attributes, $body_attributes, $copying_comment, $after_body_open, $extra_head, $program_and_version, $program_homepage, $program, $generator); } sub _get_links($$$$) { my ($self, $filename, $output_unit, $node_command) = @_; my $links = ''; if ($self->get_conf('USE_LINKS')) { my $link_directions = $self->get_conf('LINKS_DIRECTIONS'); return $links if (!defined($link_directions)); foreach my $link_direction (@$link_directions) { my $link_href = $self->from_element_direction($link_direction, 'href', $output_unit, $filename, $node_command); #print STDERR "$link_direction -> " # .(defined($link_href) ? $link_href : 'UNDEF')."\n"; if (defined($link_href) and $link_href ne '') { my $link_string = $self->from_element_direction($link_direction, 'string', $output_unit); my $link_title = ''; $link_title = " title=\"$link_string\"" if (defined($link_string)); my $rel = ''; my $button_rel = $self->direction_string($link_direction, 'rel', 'string'); $rel = " rel=\"".$button_rel.'"' if (defined($button_rel)); $links .= $self->close_html_lone_element( "{'unit_node'})) { $node_command = $output_unit->{'unit_node'}->{'element'}; } my $element_command = $output_unit->{'unit_command'}; if ($self->get_conf('SPLIT') and defined($element_command)) { $command_for_title = $element_command; } } my ($title, $description, $keywords, $encoding, $date, $css_lines, $doctype, $root_html_element_attributes, $body_attributes, $copying_comment, $after_body_open, $extra_head, $program_and_version, $program_homepage, $program, $generator) = _file_header_information($self, $command_for_title, $filename); my $links = _get_links($self, $filename, $output_unit, $node_command); my $keywords_output = ''; if (defined($keywords)) { $keywords_output = $self->close_html_lone_element( "close_html_lone_element( "close_html_lone_element( "close_html_lone_element( " $after_body_open"; return $result; } sub _default_format_node_redirection_page($$;$) { my ($self, $command, $filename) = @_; # TODO use command_name? Only if xrefautomaticsectiontitle is on? my $name = $self->command_text($command); my $href = $self->command_href($command, $filename); my $direction = "$name"; my $string = $self->convert_tree( $self->cdt('The node you are looking for is at {href}.', { 'href' => Texinfo::TreeElement::new({'type' => '_converted', 'text' => $direction })}), 'Tr redirection sentence'); my ($title, $description, $keywords, $encoding, $date, $css_lines, $doctype, $root_html_element_attributes, $body_attributes, $copying_comment, $after_body_open, $extra_head, $program_and_version, $program_homepage, $program, $generator) = _file_header_information($self, $command, $filename); my $keywords_output = ''; if (defined($keywords)) { $keywords_output = $self->close_html_lone_element( "close_html_lone_element( "close_html_lone_element( "close_html_lone_element( "close_html_lone_element( " $after_body_open

    $string

    "; return $result; } sub _default_format_single_footnote($$$$$$) { my ($self, $command, $footid, $number_in_doc, $href, $mark) = @_; my $footnote_text = $self->convert_tree_new_formatting_context($command->{'contents'}->[0], "$command->{'cmdname'} $number_in_doc $footid"); chomp ($footnote_text); $footnote_text .= "\n"; return $self->html_attribute_class('h5', ['footnote-body-heading']) . '>'. "($mark)\n" . $footnote_text; } sub _default_format_footnotes_sequence($) { my $self = shift; my $pending_footnotes = $self->get_pending_footnotes(); my $result = ''; foreach my $pending_footnote_info_array (@$pending_footnotes) { my ($command, $footid, $docid, $number_in_doc, $footnote_location_filename, $multi_expanded_region) = @$pending_footnote_info_array; my $footnote_location_href = $self->footnote_location_href($command, undef, $docid, $footnote_location_filename); my $footnote_mark; if ($self->get_conf('NUMBER_FOOTNOTES')) { $footnote_mark = $number_in_doc; } else { $footnote_mark = $self->get_conf('NO_NUMBER_FOOTNOTE_SYMBOL'); $footnote_mark = '' if (!defined($footnote_mark)); } # NOTE the @-commands in @footnote that are formatted differently depending # on multi_expanded_region($self) cannot know that the original context # of the @footnote in the main document was $multi_expanded_region. # We do not want to set multi_expanded in customizable code. However, it # could be possible to set a shared_conversion_state based on $multi_expanded_region # and have all the conversion functions calling multi_expanded_region($self) # also check the shared_conversion_state. The special situations # with those @-commands in @footnote in multi expanded # region do not justify this additional code and complexity. The consequences # should only be redundant anchors HTML elements. $result .= &{$self->formatting_function('format_single_footnote')}($self, $command, $footid, $number_in_doc, $footnote_location_href, $footnote_mark); } return $result; } sub _default_format_footnotes_segment($) { my $self = shift; my $foot_lines = &{$self->formatting_function('format_footnotes_sequence')}($self); return '' if ($foot_lines eq ''); my $class = $self->special_unit_info('class', 'footnotes'); my $result = $self->html_attribute_class('div', [$class.'-segment']).">\n"; $result .= $self->get_conf('DEFAULT_RULE') . "\n" if (defined($self->get_conf('DEFAULT_RULE')) and $self->get_conf('DEFAULT_RULE') ne ''); my $footnote_heading = $self->special_unit_info_text('heading', 'footnotes'); my $level = $self->get_conf('FOOTNOTE_END_HEADER_LEVEL'); $result .= &{$self->formatting_function('format_heading_text')}($self, undef, [$class.'-heading'], $footnote_heading, $level)."\n"; $result .= $foot_lines; $result .= "\n"; return $result; } sub _default_format_special_body_about($$$) { my ($self, $special_type, $element) = @_; my $about = ''; if ($self->get_conf('PROGRAM_NAME_IN_ABOUT')) { $about .= "

    \n "; $about .= &{$self->formatting_function('format_program_string')}($self); $about .= "\n

    \n"; } $about .= "

    \n"; my $buttons = $self->get_conf('SECTION_BUTTONS'); if (!defined($buttons)) { $about .= $self->convert_tree( $self->cdt('There are no buttons for this document.')). "\n"; $about .= "

    \n"; return $about; } $about .= $self->convert_tree( $self->cdt(' The buttons in the navigation panels have the following meaning:'), 'ABOUT') . "\n"; my $table = $self->html_attribute_class('table', ['direction-about']).'>'; $about .= < $table EOT my $button_th = $self->html_attribute_class('th', ['button-direction-about']).'>'; my $name_th = $self->html_attribute_class('th', ['name-direction-about']).'>'; my $description_th = $self->html_attribute_class('th', ['description-direction-about']).'>'; my $example_th = $self->html_attribute_class('th', ['example-direction-about']).'>'; # TRANSLATORS: direction column header in the navigation help $about .= " $button_th " . $self->convert_tree($self->cdt('Button'), 'ABOUT') ." \n". # TRANSLATORS: button label column header in the navigation help " $name_th " . $self->convert_tree($self->cdt('Name'), 'ABOUT') . " \n" . # TRANSLATORS: direction description column header in the navigation help " $description_th " . $self->convert_tree($self->cdt('Go to'), 'ABOUT') . " \n" . # TRANSLATORS: section reached column header in the navigation help " $example_th " . $self->convert_tree($self->cdt('From 1.2.3 go to'), 'ABOUT') ."\n". " \n"; my $active_icons; if ($self->get_conf('ICONS')) { $active_icons = $self->get_conf('ACTIVE_ICONS'); } foreach my $button_spec (@{$buttons}) { next if (defined($self->global_direction_text($button_spec)) or ref($button_spec) eq 'CODE' or ref($button_spec) eq 'SCALAR' or (ref($button_spec) eq 'ARRAY' and scalar(@$button_spec) != 2)); my $direction; if (ref($button_spec) eq 'ARRAY') { $direction = $button_spec->[0]; } else { $direction = $button_spec; } $about .= " \n ".$self->html_attribute_class('td', ['button-direction-about']) .'>'; # if the button spec is an array we do not know what the button # looks like, so we do not show the button but still show explanations. if (ref($button_spec) ne 'ARRAY') { if ($active_icons and $active_icons->{$direction}) { my $button_name_string = $self->direction_string($direction, 'button', 'string'); $about .= &{$self->formatting_function('format_button_icon_img')}($self, $button_name_string, $active_icons->{$direction}) } else { my $direction_text = $self->direction_string($direction, 'text'); $direction_text = '' if (!defined($direction_text)); $about .= ' ['. $direction_text .'] '; } } $about .= "\n"; # same order for getting the direction strings as in C code my $button_name = $self->direction_string($direction, 'button'); $button_name = '' if (!defined($button_name)); my $direction_description = $self->direction_string($direction, 'description'); $direction_description = '' if (!defined($direction_description)); my $direction_example = $self->direction_string($direction, 'example'); $direction_example = '' if (!defined($direction_example)); my $description_td = $self->html_attribute_class('td', ['description-direction-about']).'>'; my $example_td = $self->html_attribute_class('td', ['example-direction-about']).'>'; $about .= ' '.$self->html_attribute_class('td', ['name-direction-about']).'>' ."$button_name ${description_td}$direction_description ${example_td}$direction_example "; } $about .= <

    EOT $about .= $self->convert_tree( $self->cdt(' where the @strong{ Example } assumes that the current position is at @strong{ Subsubsection One-Two-Three } of a document of the following structure:'), 'ABOUT') . "\n"; # where the Example assumes that the current position # is at Subsubsection One-Two-Three of a document of # the following structure: $about .= <

      EOT my $non_breaking_space = $self->get_info('non_breaking_space'); # TRANSLATORS: example name of section for section 1 $about .= '
    • 1. ' . $self->convert_tree($self->cdt('Section One'), 'ABOUT') . "\n" . "
        \n" . # TRANSLATORS: example name of section for section 1.1 '
      • 1.1 ' . $self->convert_tree($self->cdt('Subsection One-One'), 'ABOUT') . "\n"; $about .= <
      • ...
    • EOT $about .= '
    • 1.2 ' . # TRANSLATORS: example name of section for section 1.2 $self->convert_tree($self->cdt('Subsection One-Two'), 'ABOUT') . "\n" . "
        \n" . '
      • 1.2.1 ' . # TRANSLATORS: example name of section for section 1.2.1 $self->convert_tree($self->cdt('Subsubsection One-Two-One'), 'ABOUT') . "
      • \n" . '
      • 1.2.2 ' . # TRANSLATORS: example name of section for section 1.2.2 $self->convert_tree($self->cdt('Subsubsection One-Two-Two'), 'ABOUT') . "
      • \n" . '
      • 1.2.3 ' . # TRANSLATORS: example name of section for section 1.2.3 $self->convert_tree($self->cdt('Subsubsection One-Two-Three'), 'ABOUT') . " $non_breaking_space $non_breaking_space\n" . ' <== ' . $self->convert_tree($self->cdt('Current Position'), 'ABOUT') . "
      • \n" . '
      • 1.2.4 ' . # TRANSLATORS: example name of section for section 1.2.4 $self->convert_tree($self->cdt('Subsubsection One-Two-Four'), 'ABOUT') . "
      • \n" . "
      \n" . "
    • \n" . '
    • 1.3 ' . # TRANSLATORS: example name of section for section 1.3 $self->convert_tree($self->cdt('Subsection One-Three'), 'ABOUT') . "\n"; $about .= <
    • ...
    EOT $about .= '
  • 1.4 ' . # TRANSLATORS: example name of section for section 1.4 $self->convert_tree($self->cdt('Subsection One-Four'), 'ABOUT') . "
  • \n"; $about .= < EOT return $about; } sub _default_format_special_body_contents($$$) { my ($self, $special_type, $element) = @_; return &{$self->formatting_function('format_contents')}($self, 'contents'); } sub _default_format_special_body_shortcontents($$$) { my ($self, $special_type, $element) = @_; return &{$self->formatting_function('format_contents')}($self, 'shortcontents'); } sub _default_format_special_body_footnotes($$$) { my ($self, $special_type, $element) = @_; return &{$self->formatting_function('format_footnotes_sequence')}($self); } sub _do_jslicenses_file($$) { my ($self, $destination_directory) = @_; my $setting = $self->get_conf('JS_WEBLABELS'); my $path = $self->get_conf('JS_WEBLABELS_FILE'); # Possible settings: # 'generate' - create file at JS_WEBLABELS_FILE # 'reference' - reference file at JS_WEBLABELS_FILE but do not create it # 'omit' - do nothing return if (!defined($setting) or $setting ne 'generate' or !defined($path) or $path eq ''); if (File::Spec->file_name_is_absolute($path) or $path =~ /^[A-Za-z]*:/ or $path eq '-') { $self->converter_document_warn(sprintf( __("cannot use absolute path or URL `%s' for JS_WEBLABELS_FILE when generating web labels file"), $path)); return; } my $doctype = $self->get_conf('DOCTYPE'); $doctype = '' if (!defined($doctype)); my $root_html_element_attributes = _root_html_element_attributes_string($self); my $a = $doctype . "\n" ."" .'jslicense labels '; my $jslicenses = $self->get_info('jslicenses'); foreach my $category (sort(keys(%$jslicenses))) { foreach my $file (sort(keys(%{$jslicenses->{$category}}))) { my $file_info = $jslicenses->{$category}->{$file}; $a .= "\n"; $a .= '\n"; $a .= '\n"; $a .= '\n"; $a .= "\n"; } } $a .= "
    $file$file_info->[0]$file_info->[2]
    \n\n"; my $license_file; if ($destination_directory ne '') { $license_file = join('/', ($destination_directory, $path)); } else { $license_file = $path; } # sequence of bytes my ($licence_file_path, $path_encoding) = $self->encoded_output_file_name($license_file); my ($fh, $error_message_licence_file, $overwritten_file) = Texinfo::Convert::Utils::output_files_open_out( $self->output_files_information(), $licence_file_path, undef, $self->get_conf('OUTPUT_ENCODING_NAME')); if ($overwritten_file) { $self->converter_document_warn( sprintf(__("overwriting output file with js licences: %s"), $license_file)); } if (defined($fh)) { print $fh $a; Texinfo::Convert::Utils::output_files_register_closed( $self->output_files_information(), $licence_file_path); if (!close ($fh)) { $self->converter_document_error( sprintf(__("error on closing %s: %s"), $license_file, $!)); } } else { $self->converter_document_error( sprintf(__("could not open %s for writing: %s"), $license_file, $error_message_licence_file)); } } sub _has_contents_or_shortcontents($) { my $self = shift; my $global_commands; my $document = $self->get_info('document'); if (defined($document)) { $global_commands = $document->global_commands_information(); } foreach my $cmdname ('contents', 'shortcontents') { if (defined($global_commands) and exists($global_commands->{$cmdname})) { return 1; } } return 0; } # to be called before starting conversion. # NOTE not called directly by convert_tree, which means that convert_tree # needs to be called from a converter which would have had this function # called already. # This function initializes states that are initialized either in XS # or in Perl. Called as early as possible in the conversion functions. # $CONTEXT is the first conversion context name. # $DOCUMENT is the converted Texinfo parsed document. sub conversion_initialization($$;$) { my ($self, $context, $document) = @_; $self->{'converter_info'} = {}; if (defined($document)) { $self->set_document($document); $self->{'converter_info'}->{'document'} = $document; } # needed if a converter is reused $self->set_global_document_commands('before', \@informative_global_commands); $self->set_global_document_commands('before', \@contents_commands); $self->{'css_import_lines'} = [@{$self->{'files_css_import_lines'}}]; $self->{'css_rule_lines'} = [@{$self->{'files_css_rule_lines'}}]; $self->{'shared_conversion_state'} = {}; $self->{'document_context'} = []; $self->{'document_global_context_counter'} = 0; $self->{'associated_inline_content'} = {}; foreach my $cmdname (keys(%default_shared_conversion_states)) { foreach my $state_name (keys(%{$default_shared_conversion_states{$cmdname}})) { $self->define_shared_conversion_state($cmdname, $state_name, $default_shared_conversion_states{$cmdname}->{$state_name}); } } # even if there is no actual file, this is needed if the API is used. $self->{'html_files_information'} = {}; # Needed for CSS gathering, even if nothing related to CSS is output $self->{'document_global_context_css'} = {}; $self->{'page_css'} = {}; # targets # used for diverse tree elements: nodes and sectioning commands, indices, # footnotes, special output units elements... $self->{'targets'} = {}; # for footnotes $self->{'special_targets'} = {'footnote_location' => {}}; $self->{'seen_ids'} = {}; # other $self->{'pending_footnotes'} = []; $self->{'pending_closes'} = {}; delete $self->{'current_node'}; delete $self->{'current_root_command'}; # for user-defined translation results. Always reset such as not # to get a cached translation obtained for a previous conversion. $self->{'translation_cache'} = {}; my %special_characters_set; my $output_encoding = $self->get_conf('OUTPUT_ENCODING_NAME'); foreach my $special_character (keys(%special_characters)) { my ($default_entity, $unicode_point) = @{$special_characters{$special_character}}; if ($self->get_conf('OUTPUT_CHARACTERS') and Texinfo::Convert::Unicode::unicode_point_decoded_in_encoding( $output_encoding, $unicode_point)) { $special_characters_set{$special_character} = charnames::vianame("U+$unicode_point"); } elsif ($self->get_conf('USE_NUMERIC_ENTITY')) { $special_characters_set{$special_character} = '&#'.hex($unicode_point).';'; } else { $special_characters_set{$special_character} = $default_entity; } } $self->{'converter_info'}->{'non_breaking_space'} = $special_characters_set{'non_breaking_space'}; $self->{'converter_info'}->{'paragraph_symbol'} = $special_characters_set{'paragraph_symbol'}; if (not defined($self->get_conf('OPEN_QUOTE_SYMBOL'))) { my $set = $self->set_conf('OPEN_QUOTE_SYMBOL', $special_characters_set{'left_quote'}); # override undef set in init file/command line $self->force_conf('OPEN_QUOTE_SYMBOL', '') if (!$set); } if (not defined($self->get_conf('CLOSE_QUOTE_SYMBOL'))) { my $set = $self->set_conf('CLOSE_QUOTE_SYMBOL', $special_characters_set{'right_quote'}); # override undef set in init file/command line $self->force_conf('CLOSE_QUOTE_SYMBOL', '') if (!$set); } if (not defined($self->get_conf('MENU_SYMBOL'))) { my $set = $self->set_conf('MENU_SYMBOL', $special_characters_set{'bullet'}); # override undef set in init file/command line $self->force_conf('MENU_SYMBOL', '') if (!$set); } my $line_break_element; if ($self->get_conf('USE_XML_SYNTAX')) { foreach my $customization_variable ('BIG_RULE', 'DEFAULT_RULE') { my $variable_value = $self->get_conf($customization_variable); if (defined($variable_value)) { my $closed_lone_element = _xhtml_re_close_lone_element($variable_value); if ($closed_lone_element ne $variable_value) { $self->force_conf($customization_variable, $closed_lone_element); } } } $line_break_element = '
    '; } else { $line_break_element = '
    '; } $self->{'converter_info'}->{'line_break_element'} = $line_break_element; # duplicate such as not to modify the defaults my $conf_default_no_arg_commands_formatting_normal = Storable::dclone($default_no_arg_commands_formatting{'normal'}); my $non_breaking_space = $self->get_info('non_breaking_space'); if ($non_breaking_space ne $xml_named_entity_nbsp) { foreach my $space_command (' ', "\t", "\n", 'tie') { $conf_default_no_arg_commands_formatting_normal->{$space_command}->{'text'} = $non_breaking_space; } } if ($self->get_conf('USE_NUMERIC_ENTITY')) { foreach my $cmdname (keys(%Texinfo::Convert::Unicode::unicode_entities)) { $conf_default_no_arg_commands_formatting_normal->{$cmdname}->{'text'} = $Texinfo::Convert::Unicode::unicode_entities{$cmdname}; } } $conf_default_no_arg_commands_formatting_normal->{'*'}->{'text'} = $self->get_info('line_break_element'); # NOTE need to be before the call to css_set_selector_style just below %{$self->{'css_element_class_styles'}} = %default_css_element_class_styles; # initialized here and not with the converter because it may depend on # the document encoding. $self->{'no_arg_commands_formatting'} = {}; foreach my $cmdname (keys(%{$default_no_arg_commands_formatting{'normal'}})) { $self->{'no_arg_commands_formatting'}->{$cmdname} = {}; # For now, there are no 'translated_to_convert' Texinfo code for # no args commands in the default case. The translated command # is common to diverse output formats if (defined($self->{'customized_no_arg_commands_formatting'} ->{$cmdname}->{'translated_to_convert'})) { $self->{'no_arg_commands_formatting'}->{$cmdname} ->{'translated_to_convert'} = $self->{'customized_no_arg_commands_formatting'} ->{$cmdname}->{'translated_to_convert'}; } foreach my $context (@no_args_commands_contexts) { my $no_arg_command_customized_formatting = $self->{'customized_no_arg_commands_formatting'} ->{$cmdname}->{$context}; if (defined($no_arg_command_customized_formatting)) { $self->{'no_arg_commands_formatting'}->{$cmdname}->{$context} = $no_arg_command_customized_formatting; } else { my $context_default_default_no_arg_commands_formatting; if ($context eq 'normal') { $context_default_default_no_arg_commands_formatting = $conf_default_no_arg_commands_formatting_normal; } else { $context_default_default_no_arg_commands_formatting = $default_no_arg_commands_formatting{$context}; } if (defined($context_default_default_no_arg_commands_formatting ->{$cmdname})) { if ($self->get_conf('OUTPUT_CHARACTERS') and Texinfo::Convert::Unicode::brace_no_arg_command( $cmdname, $self->get_conf('OUTPUT_ENCODING_NAME'))) { $self->{'no_arg_commands_formatting'}->{$cmdname}->{$context} = { 'text' => Texinfo::Convert::Unicode::brace_no_arg_command( $cmdname, $self->get_conf('OUTPUT_ENCODING_NAME'))}; # reset CSS for itemize command arguments if ($context eq 'css_string' and exists($brace_commands{$cmdname}) and $cmdname ne 'bullet' and $cmdname ne 'w' and not $special_list_mark_css_string_no_arg_command{$cmdname}) { my $css_string = $self->{'no_arg_commands_formatting'} ->{$cmdname}->{$context}->{'text'}; $css_string = '"'.$css_string.'"'; css_set_selector_style($self, "ul.mark-$cmdname", "list-style-type: $css_string"); } } else { $self->{'no_arg_commands_formatting'}->{$cmdname}->{$context} = $context_default_default_no_arg_commands_formatting->{$cmdname}; } } else { # not possible for the normal context because the commands come # from the normal context defaults, and in addition there should # be normal context defaults for all the nobrace brace commands # except for @today #if ($context eq 'normal') { # die "Brace no args command normal no set for $cmdname"; #} $self->{'no_arg_commands_formatting'}->{$cmdname}->{$context} = {'unset' => 1}; } } } } # set sane defaults in case there is none and the default formatting # function is used foreach my $cmdname (keys(%{$default_no_arg_commands_formatting{'normal'}})) { if (exists($self->{'commands_conversion'}->{$cmdname}) and $self->{'commands_conversion'}->{$cmdname} eq $default_commands_conversion{$cmdname}) { _complete_no_arg_commands_formatting($self, $cmdname); } } # for global directions always set, and for directions to special elements, # only filled if special elements are actually used. $self->{'global_units_directions'} = {}; # three types of direction strings: # * strings not translated, already converted # * strings translated # - strings already converted # - strings not already converted $self->{'directions_strings'} = {}; # The strings not translated, already converted are # initialized here and not with the converter because # substitute_html_non_breaking_space is used and it depends on the document. foreach my $string_type (keys(%default_converted_directions_strings)) { $self->{'directions_strings'}->{$string_type} = {}; foreach my $direction (keys(%{$self->{'all_directions'}})) { $self->{'directions_strings'}->{$string_type}->{$direction} = {}; my $string_contexts; if (exists($self->{'customized_direction_strings'}->{$string_type}) and exists($self->{'customized_direction_strings'}->{$string_type} ->{$direction})) { if (defined($self->{'customized_direction_strings'}->{$string_type} ->{$direction}->{'converted'})) { $string_contexts = $self->{'customized_direction_strings'}->{$string_type} ->{$direction}->{'converted'}; } else { $string_contexts = {'normal' => undef }; } } else { my $string = $default_converted_directions_strings{$string_type}->{$direction}; $string_contexts = {'normal' => $string}; } $string_contexts->{'string'} = $string_contexts->{'normal'} if (not defined($string_contexts->{'string'})); foreach my $context (keys(%$string_contexts)) { if (defined($string_contexts->{$context})) { $self->{'directions_strings'}->{$string_type} ->{$direction}->{$context} = $self->substitute_html_non_breaking_space( $string_contexts->{$context}); } else { $self->{'directions_strings'}->{$string_type} ->{$direction}->{$context} = undef; } } } } # direction strings foreach my $string_type (keys(%default_translated_directions_strings)) { # those will be determined from translatable strings $self->{'directions_strings'}->{$string_type} = {}; }; # to avoid infinite recursions when a section refers to itself, possibly # indirectly $self->{'referred_command_stack'} = []; $self->{'check_htmlxref_already_warned'} = {}; $self->{'converter_info'}->{'expanded_formats'} = $self->{'expanded_formats'}; $self->{'multiple_pass'} = []; if (not defined($self->get_conf('NODE_NAME_IN_INDEX'))) { $self->set_conf('NODE_NAME_IN_INDEX', $self->get_conf('USE_NODES')); } if ($self->get_conf('HTML_MATH') and not defined($self->get_conf('CONVERT_TO_LATEX_IN_MATH'))) { $self->set_conf('CONVERT_TO_LATEX_IN_MATH', 1); } if ($self->get_conf('CONVERT_TO_LATEX_IN_MATH')) { $self->{'options_latex_math'} = { Texinfo::Convert::LaTeX::copy_options_for_convert_to_latex_math($self) }; } if ($self->get_conf('NO_TOP_NODE_OUTPUT') and not defined($self->get_conf('SHOW_TITLE'))) { $self->set_conf('SHOW_TITLE', 1); } my $use_accesskey = $self->get_conf('USE_ACCESSKEY'); if (!defined($use_accesskey) and $self->get_conf('SPLIT') and $self->get_conf('SPLIT') eq 'node') { $self->set_conf('USE_ACCESSKEY', 1); } _new_document_context($self, $context); } sub conversion_finalization($) { my $self = shift; _pop_document_context($self); } sub _prepare_title_titlepage($$$$) { my ($self, $output_file, $output_filename, $output_units) = @_; # set file name to be the first file name for formatting of title page. # The title page prepared here is thus only fit to be used in the first # output unit. if ($output_file ne '') { $self->{'current_filename'} = $output_units->[0]->{'unit_filename'}; } else { $self->{'current_filename'} = $output_filename; } # title $self->{'converter_info'}->{'title_titlepage'} = &{$self->formatting_function('format_title_titlepage')}($self); delete $self->{'current_filename'}; } sub _html_convert_convert($$$$) { my ($self, $document, $output_units, $special_units) = @_; my $result = ''; $self->{'current_filename'} = ''; my $unit_nr = 0; # NOTE there is no rule before the footnotes special element in # case of separate footnotes in this setting. foreach my $output_unit (@$output_units, @$special_units) { print STDERR "\nC UNIT $unit_nr\n" if ($self->get_conf('DEBUG')); my $output_unit_text = $self->convert_output_unit($output_unit, "convert unit $unit_nr"); $result .= $output_unit_text; $unit_nr++; } delete $self->{'current_filename'}; return $result; } sub _prepare_simpletitle($) { my $self = shift; if (exists($self->{'document'})) { my $global_commands = $self->{'document'}->global_commands_information(); if (defined($global_commands)) { foreach my $simpletitle_command ('settitle', 'shorttitlepage') { if (exists($global_commands->{$simpletitle_command})) { my $command = $global_commands->{$simpletitle_command}; next if (!exists($command->{'contents'}->[0]->{'contents'})); $self->{'converter_info'}->{'simpletitle_tree'} = $command->{'contents'}->[0]; $self->{'converter_info'}->{'simpletitle_command_name'} = $simpletitle_command; last; } } } } } # Common to output and convert, run after the first handler in output. sub _init_conversion_after_setup_handler($) { my $self = shift; # the presence of contents elements in the document is used in diverse # places, set it once for all here my @contents_elements_options = grep {Texinfo::Common::valid_customization_option($_)} sort(keys(%contents_command_special_unit_variety)); $self->set_global_document_commands('last', \@contents_elements_options); # cache, as it is checked for each text element if ($self->get_conf('OUTPUT_CHARACTERS') and $self->get_conf('OUTPUT_ENCODING_NAME') and $self->get_conf('OUTPUT_ENCODING_NAME') eq 'utf-8') { $self->{'use_unicode_text'} = 1; } } sub _setup_convert($) { my $self = shift; _init_conversion_after_setup_handler($self); } # Conversion to a string, mostly used in tests. # $SELF is the output converter object of class Texinfo::Convert::HTML (this # module), and $DOCUMENT is the parsed document from the parser and structuring sub convert($$) { my ($self, $document) = @_; $self->conversion_initialization('_convert', $document); _setup_convert($self); my ($output_units, $special_units, $associated_special_units) = _prepare_conversion_units($self, $document, undef); # setup global targets. It is not clearly relevant to have those # global targets when called as convert, but the Top global # unit directions is often referred to in code, so at least this # global target needs to be setup. # Since the relative directions are not set, this leads to lone # global direction buttons such as [Contents] or [Index] appearing # in otherwise empty navigation headings if those global directions # are set and present in the buttons, as is the case in the default # buttons. For example in converters_tests/ref_in_sectioning # or converters_tests/sections_and_printindex. _prepare_output_units_global_targets($self, $output_units, $special_units, $associated_special_units); # setup untranslated strings _translate_names($self); _prepare_simpletitle($self); # title. Not often set in the default case, as convert() is only # used in the *.t tests, and a title requires both simpletitle_tree # and SHOW_TITLE set, with the default formatting function. _prepare_title_titlepage($self, '', '', $output_units); # main conversion here my $result = _html_convert_convert($self, $document, $output_units, $special_units); $self->conversion_finalization(); return $result; } sub convert_output_unit($$;$) { my ($self, $output_unit, # only used for debug $explanation) = @_; $debug = $self->get_conf('DEBUG') if !defined($debug); my $unit_type_name = $output_unit->{'unit_type'}; if (exists($self->{'output_units_conversion'}->{$unit_type_name}) and !defined($self->{'output_units_conversion'}->{$unit_type_name})) { if ($debug) { print STDERR "IGNORED OU $unit_type_name\n"; } return ''; } if ($debug) { print STDERR "UNIT($explanation) -> ou: $unit_type_name '" .Texinfo::OutputUnits::output_unit_texi($output_unit)."'\n"; } $self->{'current_output_unit'} = $output_unit; my $content_formatted = ''; if (exists($output_unit->{'unit_contents'})) { my $content_idx = 0; foreach my $content (@{$output_unit->{'unit_contents'}}) { $content_formatted .= _convert($self, $content, "$unit_type_name c[$content_idx]"); $content_idx++; } } my $result = ''; if (defined($self->{'output_units_conversion'}->{$unit_type_name})) { $result .= &{$self->{'output_units_conversion'}->{$unit_type_name}} ($self, $unit_type_name, $output_unit, $content_formatted); } elsif (defined($content_formatted)) { $result .= $content_formatted; } delete $self->{'current_output_unit'}; print STDERR "DOUNIT ($unit_type_name) => `$result'\n" if $debug; return $result; } # This is called from the main program on the converter. sub output_internal_links($) { my $self = shift; my $out_string = ''; foreach my $output_unit (@{$self->{'document_units'}}) { my $text; my $href; my $command = $output_unit->{'unit_command'}; if (defined($command)) { # Use '' for filename, to force a filename in href. $href = $self->command_href($command, ''); my ($tree, $in_code) = _internal_command_tree($self, $command, 0); if (defined($tree)) { $text = Texinfo::Convert::Text::convert_to_text($tree, $self->{'convert_text_options'}); } if (defined($href) or defined($text)) { $out_string .= $href if (defined($href)); $out_string .= "\tunit\t"; $out_string .= $text if (defined($text)); $out_string .= "\n"; } } } if (exists($self->{'document'})) { my $sections_list = $self->{'document'}->sections_list(); foreach my $section_relations (@{$sections_list}) { my $command = $section_relations->{'element'}; my $href = $self->command_href($command, ''); my ($tree, $in_code) = _internal_command_tree($self, $command, 0); my $text; if (defined($tree)) { $text = Texinfo::Convert::Text::convert_to_text($tree, $self->{'convert_text_options'}); } if (defined($href) or defined($text)) { $out_string .= $href if (defined($href)); $out_string .= "\tsection\t"; my $command_name = Texinfo::Structuring::section_level_adjusted_command_name($command); $out_string .= $command_name.' '; $out_string .= $text if (defined($text)); $out_string .= "\n"; } } my $labels_list = $self->{'document'}->labels_list(); if (defined($labels_list)) { my %commands_lists; foreach my $target_element (@$labels_list) { next if (not exists($target_element->{'extra'}) or not $target_element->{'extra'}->{'is_target'}); my $cmdname = $target_element->{'cmdname'}; if (!exists($commands_lists{$cmdname})) { $commands_lists{$cmdname} = []; } push @{$commands_lists{$cmdname}}, $target_element; } foreach my $cmdtype ('node', 'anchor', 'namedanchor', 'float') { next unless (exists($commands_lists{$cmdtype})); foreach my $target_element (@{$commands_lists{$cmdtype}}) { my $label_element = Texinfo::Common::get_label_element($target_element); my $href = $self->command_href($target_element, ''); my $text; if (defined($label_element)) { $text = Texinfo::Convert::Text::convert_to_text($label_element, $self->{'convert_text_options'}); } if (defined($href) or defined($text)) { $out_string .= $href if (defined($href)); $out_string .= "\t${cmdtype}\t"; $out_string .= $text if (defined($text)); $out_string .= "\n"; } } } } } my $index_entries_by_letter = $self->get_converter_indices_sorted_by_letter(); if (defined($index_entries_by_letter)) { my $indices_information; if (exists($self->{'document'})) { $indices_information = $self->{'document'}->indices_information(); } foreach my $index_name (sort(keys(%{$index_entries_by_letter}))) { foreach my $letter_entry (@{$index_entries_by_letter->{$index_name}}) { foreach my $index_entry (@{$letter_entry->{'entries'}}) { my $main_entry_element = $index_entry->{'entry_element'}; # does not refer to the document my $seeentry = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seeentry'); next if (defined($seeentry)); my $seealso = Texinfo::Common::index_entry_referred_entry($main_entry_element, 'seealso'); next if (defined($seealso)); my $href; $href = $self->command_href($main_entry_element, ''); # Obtain term by converting to text my $in_code = $indices_information->{$index_entry->{'index_name'}}->{'in_code'}; if ($in_code) { Texinfo::Convert::Text::set_options_code( $self->{'convert_text_options'}); } my $entry_reference_content_element = Texinfo::Common::index_content_element($main_entry_element); my @contents = ($entry_reference_content_element); my $subentries_tree = Texinfo::Convert::Utils::comma_index_subentries_tree( $main_entry_element); if (defined($subentries_tree)) { push @contents, @{$subentries_tree->{'contents'}}; } my $index_term = Texinfo::Convert::Text::convert_to_text( Texinfo::TreeElement::new({'contents' => \@contents}), $self->{'convert_text_options'}); if ($in_code) { Texinfo::Convert::Text::reset_options_code( $self->{'convert_text_options'}); } if (defined($index_term) and $index_term =~ /\S/) { $out_string .= $href if (defined($href)); $out_string .= "\t$index_name\t"; $out_string .= $index_term; $out_string .= "\n"; } } } } } if ($out_string ne '') { return $out_string; } else { return undef; } } sub _run_stage_handlers($$$$) { my ($converter, $stage_handlers, $document, $stage) = @_; return 0 if (!defined($stage_handlers->{$stage})); my $handler_idx = 1; foreach my $handler_and_priority (@{$stage_handlers->{$stage}}) { my ($handler, $priority) = @$handler_and_priority; if ($converter->get_conf('DEBUG')) { print STDERR "RUN handler $handler_idx: stage $stage, priority $priority\n"; } my $status = &{$handler}($converter, $document, $stage); if (!defined($status) or ref($status) ne '' or $status !~ /^\d+$/) { $converter->converter_document_error( sprintf(__("handler %d of stage %s priority %s: non-numeric status"), $handler_idx, $stage, $priority)); $status = $converter->get_conf('HANDLER_FATAL_ERROR_LEVEL') +1; } if ($status != 0) { if ($status < 0) { $converter->converter_document_error( sprintf(__("handler %d of stage %s priority %s failed"), $handler_idx, $stage, $priority)); } else { # the handler is supposed to have output an error message # already if $status > 0 if ($converter->get_conf('VERBOSE') or $converter->get_conf('DEBUG')) { print STDERR "FAIL handler $handler_idx: stage $stage, " ."priority $priority, status $status\n"; } } return $status; } $handler_idx++; } return 0; } sub _do_js_files($$) { my ($self, $destination_directory) = @_; if ($self->get_conf('INFO_JS_DIR')) { my $info_js_dir = $self->get_conf('INFO_JS_DIR'); my $jsdir; if ($destination_directory ne '') { $jsdir = join('/', ($destination_directory, $info_js_dir)); } else { $jsdir = $info_js_dir; } my ($encoded_jsdir, $dir_encoding) = $self->encoded_output_file_name($jsdir); my $succeeded = $self->create_destination_directory($encoded_jsdir, $jsdir); # Copy JS files. if ($succeeded) { if (!$self->get_conf('TEST')) { my $jssrcdir; if (!$Texinfo::ModulePath::texinfo_uninstalled) { $jssrcdir = join('/', ( $Texinfo::ModulePath::converter_datadir, 'js')); } else { $jssrcdir = join('/', ( $Texinfo::ModulePath::t2a_srcdir, $updir, 'js')); } for my $f ('info.js', 'modernizr.js', 'info.css') { my $from = join('/', ($jssrcdir, $f)); if (!copy($from, $jsdir)) { $self->converter_document_error( sprintf(__("error on copying %s into %s"), $from, $jsdir)); } } } else { # create empty files for tests to keep results stable. foreach my $f ('info.js', 'modernizr.js', 'info.css') { my $filename = join('/', ($jsdir, $f)); if (!open(FH, '>', $filename)) { $self->converter_document_error( sprintf(__("error on creating empty %s: %s"), $filename, $!)); } else { if (!close(FH)) { $self->converter_document_error( sprintf(__("error on closing empty %s: %s"), $filename, $!)); } } } } } } my $jslicenses = $self->get_info('jslicenses'); if (defined($jslicenses) and scalar(%$jslicenses)) { _do_jslicenses_file($self, $destination_directory); } } sub _prepare_converted_output_info($$$$) { my ($self, $output_file, $output_filename, $output_units) = @_; my $stage_handlers = $self->{'stage_handlers'}; my $structure_status = _run_stage_handlers($self, $stage_handlers, $self->{'document'}, 'structure'); my $handler_fatal_error_level = $self->get_conf('HANDLER_FATAL_ERROR_LEVEL'); unless ($structure_status < $handler_fatal_error_level and $structure_status > -$handler_fatal_error_level) { return 0; } my $default_document_language = $self->get_conf('documentlanguage'); $self->set_global_document_commands('preamble', ['documentlanguage']); my $preamble_document_language = $self->get_conf('documentlanguage'); if (not (!defined($default_document_language) and !defined($preamble_document_language)) and (!defined($default_document_language) or !defined($preamble_document_language) or $default_document_language ne $preamble_document_language)) { _translate_names($self); } # prepare title. fulltitle uses more possibility than simpletitle for # title, including @-commands found in @titlepage only. Therefore # simpletitle is more in line with what makeinfo in C did. _prepare_simpletitle($self); my $global_commands; if (exists($self->{'document'})) { $global_commands = $self->{'document'}->global_commands_information(); } my $fulltitle_tree; if (defined($global_commands)) { foreach my $fulltitle_command ('settitle', 'title', 'shorttitlepage') { if (exists($global_commands->{$fulltitle_command})) { my $command = $global_commands->{$fulltitle_command}; next if (!exists($command->{'contents'}->[0]->{'contents'})); $fulltitle_tree = $command->{'contents'}->[0]; last; } } if (!defined($fulltitle_tree) and exists($global_commands->{'top'})) { # arguments_line type element my $arguments_line = $global_commands->{'top'}->{'contents'}->[0]; my $line_arg = $arguments_line->{'contents'}->[0]; if (exists($line_arg->{'contents'})) { $fulltitle_tree = $line_arg; } } if (!defined($fulltitle_tree) and exists($global_commands->{'titlefont'}) and exists($global_commands->{'titlefont'}->[0]->{'contents'}) and exists($global_commands->{'titlefont'}->[0]->{'contents'}->[0] ->{'contents'})) { $fulltitle_tree = $global_commands->{'titlefont'}->[0]; } } my $html_title_string; my $title_tree; if (defined($fulltitle_tree)) { $title_tree = $fulltitle_tree; $html_title_string = $self->convert_tree_new_formatting_context($title_tree, 'title_string', $CTXF_string); if ($html_title_string !~ /\S/) { $html_title_string = undef; } } if (!defined($html_title_string)) { my $default_title = $self->cdt('Untitled Document'); $title_tree = $default_title; $self->{'converter_info'}->{'title_tree'} = $title_tree; $self->{'converter_info'}->{'title_string'} = $self->convert_tree_new_formatting_context($title_tree, 'title_string', $CTXF_string); my $input_file_name; if (exists($self->{'document'})) { my $document_info = $self->{'document'}->global_information(); if (defined($document_info)) { $input_file_name = $document_info->{'input_file_name'}; } } if (defined($input_file_name)) { $self->converter_line_warn(__( "must specify a title with a title command or \@top"), {'file_name' => $input_file_name}); } else { $self->converter_document_warn(__( "must specify a title with a title command or \@top")); } } else { $self->{'converter_info'}->{'title_tree'} = $title_tree; $self->{'converter_info'}->{'title_string'} = $html_title_string; } # copying comment if (defined($global_commands) and exists($global_commands->{'copying'})) { my $copying_comment = Texinfo::Convert::Text::convert_to_text( Texinfo::TreeElement::new( {'contents' => $global_commands->{'copying'}->{'contents'}}), $self->{'convert_text_options'}); if ($copying_comment ne '') { $self->{'converter_info'}->{'copying_comment'} = &{$self->formatting_function('format_comment')}($self, $copying_comment); } } # documentdescription if (defined($self->get_conf('documentdescription'))) { $self->{'converter_info'}->{'documentdescription_string'} = $self->get_conf('documentdescription'); } elsif (defined($global_commands) and exists($global_commands->{'documentdescription'})) { my $tmp = Texinfo::TreeElement::new({'contents' => $global_commands->{'documentdescription'}->{'contents'}}); my $documentdescription_string = $self->convert_tree_new_formatting_context($tmp, 'documentdescription', $CTXF_string); chomp($documentdescription_string); $self->{'converter_info'}->{'documentdescription_string'} = $documentdescription_string; } # TODO document that this stage handler is called with end of preamble # documentlanguage when it is certain that this will not change ever. my $init_status = _run_stage_handlers($self, $stage_handlers, $self->{'document'}, 'init'); unless ($init_status < $handler_fatal_error_level and $init_status > -$handler_fatal_error_level) { return 0; } _prepare_title_titlepage($self, $output_file, $output_filename, $output_units); $self->set_global_document_commands('before', ['documentlanguage']); if (not (!defined($default_document_language) and !defined($preamble_document_language)) and (!defined($default_document_language) or !defined($preamble_document_language) or $default_document_language ne $preamble_document_language)) { _translate_names($self); } # reset in case the user changed customization variables in handlers $self->{'convert_text_options'} = Texinfo::Convert::Text::copy_options_for_convert_text($self); return 1; } # units or root conversion sub _html_convert_output($$$$$$$$) { my ($self, $output_file, $destination_directory, $output_filename, $document_name, $document, $output_units, $special_units) = @_; my ($encoded_destination_directory, $dir_encoding) = $self->encoded_output_file_name($destination_directory); my $succeeded = $self->create_destination_directory($encoded_destination_directory, $destination_directory); if (!$succeeded) { return undef; } my $text_output = ''; if ($output_file eq '') { $self->{'current_filename'} = $output_filename; my $body = ''; my $unit_nr = 0; # NOTE there is no rule before the footnotes special element in # case of separate footnotes in this setting. foreach my $output_unit (@$output_units, @$special_units) { print STDERR "\nUNIT NO-PAGE $unit_nr\n" if ($self->get_conf('DEBUG')); my $output_unit_text = $self->convert_output_unit($output_unit, "no-page output unit $unit_nr"); $body .= $output_unit_text; $unit_nr++; } # do end file first, in case it needs some CSS my $file_end = &{$self->formatting_function('format_end_file')}($self, $output_filename, undef); my $file_beginning = &{$self->formatting_function('format_begin_file')}($self, $output_filename, undef); $text_output .= $file_beginning; $text_output .= $body; $text_output .= $file_end; } else { # output with pages print STDERR "DO Units with filenames\n" if ($self->get_conf('DEBUG')); my %files; my $unit_nr = -1; # Now do the output, converting each output units and special output units # in turn $special_units = [] if (!defined($special_units)); foreach my $output_unit (@$output_units, @$special_units) { $unit_nr++; my $output_unit_filename = $output_unit->{'unit_filename'}; $self->{'current_filename'} = $output_unit_filename; # convert body before header in case this affects the header # and, for special output unit, to avoid outputting anything if empty. my $body; if ($output_unit->{'unit_type'} eq 'special_unit') { print STDERR "\nUNIT SPECIAL " ."$output_unit->{'special_unit_variety'} $unit_nr\n" if ($self->get_conf('DEBUG')); $body = $self->convert_output_unit($output_unit, "output s-unit $unit_nr"); if ($body eq '') { $body = undef; } } else { print STDERR "\nUNIT $unit_nr\n" if ($self->get_conf('DEBUG')); $body = $self->convert_output_unit($output_unit, "output unit $unit_nr"); } $self->{'file_counters'}->{$output_unit_filename}--; # register the output but do not print anything. Printing # only when file_counters reach 0, to be sure that all the # elements have been converted before headers are done. if (defined($body)) { if (!exists($files{$output_unit_filename})) { $files{$output_unit_filename} = {'first_unit' => $output_unit, 'body' => ''}; } $files{$output_unit_filename}->{'body'} .= $body; } else { next if (!exists($files{$output_unit_filename}) or $files{$output_unit_filename}->{'body'} eq ''); } if ($self->{'file_counters'}->{$output_unit_filename} == 0) { my $out_filepath = $self->{'out_filepaths'}->{$output_unit_filename}; my $file_output_unit = $files{$output_unit_filename}->{'first_unit'}; my ($encoded_out_filepath, $path_encoding) = $self->encoded_output_file_name($out_filepath); # the third return information, set if the file has already been used # in this files_information is not checked as this cannot happen. my ($file_fh, $error_message) = Texinfo::Convert::Utils::output_files_open_out( $self->output_files_information(), $encoded_out_filepath, undef, $self->get_conf('OUTPUT_ENCODING_NAME')); if (!defined($file_fh)) { $self->converter_document_error( sprintf(__("could not open %s for writing: %s"), $out_filepath, $error_message)); delete $self->{'current_filename'}; return undef; } # do end file first in case it requires some CSS my $end_file = &{$self->formatting_function('format_end_file')}($self, $output_unit_filename, $output_unit); print $file_fh "".&{$self->formatting_function('format_begin_file')}( $self, $output_unit_filename, $file_output_unit); print $file_fh "".$files{$output_unit_filename}->{'body'}; # end file print $file_fh "". $end_file; # Do not close STDOUT now such that the file descriptor is not reused # by open, which uses the lowest-numbered file descriptor not open, # for another filehandle. Closing STDOUT is handled by the caller. if ($out_filepath ne '-') { Texinfo::Convert::Utils::output_files_register_closed( $self->output_files_information(), $encoded_out_filepath); if (!close($file_fh)) { $self->converter_document_error( sprintf(__("error on closing %s: %s"), $out_filepath, $!)); delete $self->{'current_filename'}; return undef; } } } } } delete $self->{'current_filename'}; return $text_output; } # as a function for XS override sub _prepare_node_redirection_page($$$) { my ($self, $target_element, $redirection_filename) = @_; $self->{'current_filename'} = $redirection_filename; my $redirection_page = &{$self->formatting_function('format_node_redirection_page')}($self, $target_element, $redirection_filename); delete $self->{'current_filename'}; return $redirection_page; } sub _node_redirections($$$$) { my ($self, $output_file, $destination_directory, $files_source_info) = @_; my $labels_list; if (exists($self->{'document'})) { $labels_list = $self->{'document'}->labels_list(); } my $redirection_files_done = 0; # do node redirection pages delete $self->{'current_filename'}; if ($self->get_conf('NODE_FILES') and defined($labels_list) and $output_file ne '') { my $add_translit_redirection = 0; my $added_translit_extension; if ($self->get_conf('TRANSLITERATE_FILE_NAMES')) { $add_translit_redirection = 1; $added_translit_extension = ''; $added_translit_extension = '.'.$self->get_conf('EXTENSION') if (defined($self->get_conf('EXTENSION')) and $self->get_conf('EXTENSION') ne ''); } my %redirection_filenames; foreach my $target_element (@$labels_list) { next if (not exists($target_element->{'extra'}) or not $target_element->{'extra'}->{'is_target'}); my $label_element = Texinfo::Common::get_label_element($target_element); # filename may not be defined in case of an @anchor or similar in # @titlepage, and @titlepage is not used. my $filename = $self->command_filename($target_element); next if (!defined($filename)); my $node_filename; my $normalized = $target_element->{'extra'}->{'normalized'}; # NOTE 'node_filename' is not used for Top, TOP_NODE_FILE_TARGET # is. The other manual must use the same convention to get it # right. We do not do 'node_filename' as a redirection file # either. if ($normalized eq 'Top' and defined($self->get_conf('TOP_NODE_FILE_TARGET'))) { $node_filename = $self->get_conf('TOP_NODE_FILE_TARGET'); } else { my ($target_filebase, $external_file_extension, $id) = $self->standard_label_id_file($normalized, $label_element, $self->get_conf('EXTERNAL_CROSSREF_EXTENSION'), $defaults{'EXTENSION'}); $node_filename = $target_filebase.$external_file_extension; } my @redirection_files; my $node_redirection_filename = $self->register_normalize_case_filename($node_filename); if ($node_filename ne $filename) { # first condition finds conflict with tree elements if ($self->count_elements_in_filename('total', $node_redirection_filename) or exists($redirection_filenames{$node_redirection_filename})) { $self->converter_line_warn( sprintf(__("\@%s `%s' file %s for redirection exists"), $target_element->{'cmdname'}, Texinfo::Convert::Texinfo::convert_to_texinfo( Texinfo::TreeElement::new( {'contents' => $label_element->{'contents'}})), $node_redirection_filename), $target_element->{'source_info'}); my $file_source = $files_source_info->{$node_redirection_filename}; my $file_info_type = $file_source->{'file_info_type'}; if ($file_info_type eq 'special_file' or $file_info_type eq 'stand_in_file') { my $name = $file_source->{'file_info_name'}; if ($name eq 'non_split') { # This cannot actually happen, as the @anchor/@node/@float # with potentially conflicting name will also be in the # non-split output document and therefore does not need # a redirection. $self->converter_document_warn( __("conflict with whole document file"), 1); } elsif ($name eq 'Top') { $self->converter_document_warn( __("conflict with Top file"), 1); } elsif ($name eq 'user_defined') { $self->converter_document_warn( __("conflict with user-defined file"), 1); } elsif ($name eq 'unknown_node') { $self->converter_document_warn( __("conflict with unknown node file"), 1); } elsif ($name eq 'unknown') { $self->converter_document_warn( __("conflict with file without known source"), 1); } } elsif ($file_info_type eq 'node') { my $conflicting_node = $file_source->{'file_info_element'}; my $label_element = Texinfo::Common::get_label_element($conflicting_node); $self->converter_line_warn( sprintf(__p('conflict of redirection file with file based on node name', "conflict with \@%s `%s' file"), $conflicting_node->{'cmdname'}, Texinfo::Convert::Texinfo::convert_to_texinfo( Texinfo::TreeElement::new( {'contents' => $label_element->{'contents'}})) ), $conflicting_node->{'source_info'}, 1); } elsif ($file_info_type eq 'redirection') { my $conflicting_node = $file_source->{'file_info_element'}; my $conflicting_label_element = $file_source->{'file_info_label_element'}; $self->converter_line_warn( sprintf(__("conflict with \@%s `%s' redirection file"), $conflicting_node->{'cmdname'}, Texinfo::Convert::Texinfo::convert_to_texinfo( Texinfo::TreeElement::new( {'contents' => $conflicting_label_element->{'contents'}})) ), $conflicting_node->{'source_info'}, 1); } elsif ($file_info_type eq 'section') { my $conflicting_section = $file_source->{'file_info_element'}; # arguments_line type element my $arguments_line = $conflicting_section->{'contents'}->[0]; my $line_arg = $arguments_line->{'contents'}->[0]; $self->converter_line_warn( sprintf(__p('conflict of redirection file with file based on section name', "conflict with \@%s `%s' file"), $conflicting_section->{'cmdname'}, Texinfo::Convert::Texinfo::convert_to_texinfo( Texinfo::TreeElement::new( {'contents' => $line_arg->{'contents'}})), ), $conflicting_section->{'source_info'}, 1); } elsif ($file_info_type eq 'special_unit') { my $unit_command = $file_source->{'file_info_element'}; my $special_unit = $unit_command->{'associated_unit'}; my $output_unit_variety = $special_unit->{'special_unit_variety'}; $self->converter_document_warn( sprintf(__("conflict with %s special element"), $output_unit_variety), 1); } } else { push @redirection_files, $node_redirection_filename; } } if ($add_translit_redirection and $normalized ne 'Top') { # based on Texinfo::Convert::Converter node_information_filename my $no_unidecode; $no_unidecode = 1 if (defined($self->get_conf('USE_UNIDECODE')) and !$self->get_conf('USE_UNIDECODE')); my $in_test; $in_test = 1 if ($self->get_conf('TEST')); my $translit_filename = Texinfo::Convert::NodeNameNormalization::normalize_transliterate_texinfo( Texinfo::TreeElement::new( {'contents' => $label_element->{'contents'}}), $in_test, $no_unidecode); $translit_filename = $self->_id_to_filename($translit_filename); $translit_filename = $translit_filename.$added_translit_extension; if ($translit_filename ne $node_redirection_filename and $translit_filename ne $filename) { my $translit_redirection_filename = $self->register_normalize_case_filename($translit_filename); if (!$self->count_elements_in_filename('total', $translit_redirection_filename) and not exists( $redirection_filenames{$translit_redirection_filename})) { push @redirection_files, $translit_redirection_filename; } } } if (defined($self->{'file_id_setting'}->{'redirection_file_names'})) { my $reference_redirection_files = [@redirection_files]; # NOTE there are no checks on returned redirection files not # replacing other files, it is up to the user to do what they # should. @redirection_files = &{$self->{'file_id_setting'}->{'redirection_file_names'}}($self, $target_element, $filename, $node_redirection_filename, $reference_redirection_files); } foreach my $redirection_filename (@redirection_files) { $redirection_filenames{$redirection_filename} = $target_element; $files_source_info->{$redirection_filename} = {'file_info_type' => 'redirection', 'file_info_element' => $target_element, 'file_info_path' => undef, 'file_info_label_element' => $label_element}; my $redirection_page = _prepare_node_redirection_page($self, $target_element, $redirection_filename); my $out_filepath; if ($destination_directory ne '') { $out_filepath = join('/', ($destination_directory, $redirection_filename)); } else { $out_filepath = $redirection_filename; } my ($encoded_out_filepath, $path_encoding) = $self->encoded_output_file_name($out_filepath); # the third return information, set if the file has already been used # in this files_information is not checked as this cannot happen. my ($file_fh, $error_message) = Texinfo::Convert::Utils::output_files_open_out( $self->output_files_information(), $encoded_out_filepath, undef, $self->get_conf('OUTPUT_ENCODING_NAME')); if (!defined($file_fh)) { $self->converter_document_error(sprintf(__( "could not open %s for writing: %s"), $out_filepath, $error_message)); } else { print $file_fh $redirection_page; Texinfo::Convert::Utils::output_files_register_closed( $self->output_files_information(), $encoded_out_filepath); if (!close($file_fh)) { $self->converter_document_error(sprintf(__( "error on closing redirection node file %s: %s"), $out_filepath, $!)); $self->conversion_finalization(); return undef; } } $redirection_files_done++; # NOTE failure to open a file does not stop the processing } } } return $redirection_files_done; } sub _setup_output($) { my $self = shift; # Should not actually be needed, as it is already deleted after conversion # and each time it is set out of the conversion. delete $self->{'current_filename'}; # no splitting when writing to the null device or to stdout or returning # a string if (defined($self->get_conf('OUTFILE')) and ($Texinfo::Common::null_device_file{$self->get_conf('OUTFILE')} or $self->get_conf('OUTFILE') eq '-' or $self->get_conf('OUTFILE') eq '')) { $self->force_conf('SPLIT', ''); $self->force_conf('MONOLITHIC', 1); } if ($self->get_conf('SPLIT')) { $self->set_conf('NODE_FILES', 1); } $self->set_conf('EXTERNAL_CROSSREF_SPLIT', $self->get_conf('SPLIT')); my $handler_fatal_error_level = $self->get_conf('HANDLER_FATAL_ERROR_LEVEL'); if (!defined($handler_fatal_error_level)) { $handler_fatal_error_level = $Texinfo::Options::converter_customization_options{ 'HANDLER_FATAL_ERROR_LEVEL'}; $self->force_conf('HANDLER_FATAL_ERROR_LEVEL', $handler_fatal_error_level); } if ($self->get_conf('HTML_MATH') and $self->get_conf('HTML_MATH') eq 'mathjax') { # See https://www.gnu.org/licenses/javascript-labels.html # # The link to the source for mathjax does not strictly follow the advice # there: instead we link to instructions for obtaining the full source in # its preferred form of modification. my $mathjax_script = $self->get_conf('MATHJAX_SCRIPT'); if (! defined($mathjax_script)) { $mathjax_script = 'https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-svg.js'; $self->set_conf('MATHJAX_SCRIPT', $mathjax_script); } my $mathjax_source = $self->get_conf('MATHJAX_SOURCE'); if (! defined($mathjax_source)) { $mathjax_source = 'http://docs.mathjax.org/en/latest/web/hosting.html#getting-mathjax-via-git'; $self->set_conf('MATHJAX_SOURCE', $mathjax_source); } } my $setup_status = _run_stage_handlers($self, $self->{'stage_handlers'}, $self->{'document'}, 'setup'); if ($setup_status < $handler_fatal_error_level and $setup_status > -$handler_fatal_error_level) { } else { return undef; } # the settable commands configuration has potentially been modified for # this output file especially in setup handler. Update the corresponding # initial configuration. my $conf = $self->{'conf'}; foreach my $settable_command ( keys(%Texinfo::Common::document_settable_at_commands)) { if (exists($conf->{$settable_command})) { $self->{'commands_init_conf'}->{$settable_command} = $conf->{$settable_command}; } } # set BODY_ELEMENT_ATTRIBUTES $self->set_global_document_commands('preamble', ['documentlanguage']); my $body_lang = $self->get_conf('documentlanguage'); if (defined($body_lang)) { $self->set_conf('BODY_ELEMENT_ATTRIBUTES', 'lang="'.$body_lang.'"'); } else { #$self->set_conf('BODY_ELEMENT_ATTRIBUTES', 'lang=""'); # Note: HTML 5 documentation specifies that the lang attribute can # take an empty string as its value to specify that the language # is unknown. However, outputting lang="" is unnecessary. } $self->set_global_document_commands('before', ['documentlanguage']); _init_conversion_after_setup_handler($self); my $jslicenses = {}; if ($self->get_conf('HTML_MATH') and $self->get_conf('HTML_MATH') eq 'mathjax') { # See https://www.gnu.org/licenses/javascript-labels.html my $mathjax_script = $self->get_conf('MATHJAX_SCRIPT'); my $mathjax_source = $self->get_conf('MATHJAX_SOURCE'); $jslicenses->{'mathjax'} = { $mathjax_script => [ 'Apache License, Version 2.0.', 'https://www.apache.org/licenses/LICENSE-2.0', $mathjax_source ]}; } if ($self->get_conf('INFO_JS_DIR')) { $jslicenses->{'infojs'} = { 'js/info.js' => [ 'GNU General Public License 3.0 or later', 'https://www.gnu.org/licenses/gpl-3.0.html', 'js/info.js' ], 'js/modernizr.js' => [ 'Expat', 'http://www.jclark.com/xml/copying.txt', 'js/modernizr.js' ]}; } $self->{'converter_info'}->{'jslicenses'} = $jslicenses; # this sets output_file (based on OUTFILE), to be used if not split, # but also the corresponding 'output_filename' that is useful in # particular when output_file is '', 'destination_directory' that # is mainly useful when split and 'document_name' that is generally useful. my ($output_file, $destination_directory, $output_filename, $document_name) = $self->determine_files_and_directory( $self->get_conf('TEXINFO_OUTPUT_FORMAT')); # set for init files $self->{'converter_info'}->{'document_name'} = $document_name; $self->{'converter_info'}->{'destination_directory'} = $destination_directory; return [$output_file, $destination_directory, $output_filename, $document_name]; } # return 0 on failure, 1 on success. sub _finish_output($$$$) { my ($self, $output_file, $destination_directory, $files_source_info) = @_; _do_js_files($self, $destination_directory); my $stage_handlers = $self->{'stage_handlers'}; my $handler_fatal_error_level = $self->get_conf('HANDLER_FATAL_ERROR_LEVEL'); my $finish_status = _run_stage_handlers($self, $stage_handlers, $self->{'document'}, 'finish'); unless ($finish_status < $handler_fatal_error_level and $finish_status > -$handler_fatal_error_level) { return 0; } # undef status means an error occured my $node_redirections_status = _node_redirections($self, $output_file, $destination_directory, $files_source_info); if (!defined($node_redirections_status)) { return 0; } return 1; } # Main function for outputting a manual in HTML. # $SELF is the output converter object of class Texinfo::Convert::HTML (this # module), and $DOCUMENT is the parsed document from the parser and structuring sub output($$) { my ($self, $document) = @_; $self->conversion_initialization('_output', $document); my $paths = _setup_output($self); if (!defined($paths)) { $self->conversion_finalization(); return undef; } my ($output_file, $destination_directory, $output_filename, $document_name) = @$paths; # Get the list of output units to be processed. my ($output_units, $special_units, $associated_special_units) = _prepare_conversion_units($self, $document, $document_name); # setup untranslated strings _translate_names($self); my $files_source_info = _prepare_units_directions_files($self, $output_units, $special_units, $associated_special_units, $output_file, $destination_directory, $output_filename, $document_name); my $succeeded = _prepare_converted_output_info($self, $output_file, $output_filename, $output_units); if (!$succeeded) { $self->conversion_finalization(); return undef; } # conversion my $text_output = _html_convert_output($self, $output_file, $destination_directory, $output_filename, $document_name, $document, $output_units, $special_units); if (!defined($text_output)) { $self->conversion_finalization(); return undef; } if ($text_output ne '' and $output_file eq '') { # $output_file eq '' should always be true, as $text_output is only # filled in that case. if (!$self->get_conf('TEST')) { # This case is unlikely to happen, as there is no output file # only if formatting is called as convert, which only happens in tests. _do_js_files($self, $destination_directory); } $self->conversion_finalization(); return $text_output; } my $finish_succeeded = _finish_output($self, $output_file, $destination_directory, $files_source_info); if (!$finish_succeeded) { $self->conversion_finalization(); return undef; } $self->conversion_finalization(); return undef; } #my $characters_replaced_from_class_names = quotemeta('[](),~#:/\\@+=!;.,?* '); # Not clear what character should be allowed and which ones replaced # besides space. Not really important as the caller should themselves # sanitize the class names already. my $characters_replaced_from_class_names = quotemeta(' '); sub _protect_class_name($$) { my ($self, $class_name) = @_; $class_name =~ s/[$characters_replaced_from_class_names]/-/g; # API info: using the API to allow for customization would be: # return &{$self->formatting_function('format_protect_text')}($self, $class_name); return _default_format_protect_text($self, $class_name); } sub _open_command_update_context($$) { my ($self, $command_name) = @_; my $convert_to_latex; if (exists($brace_commands{$command_name}) and $brace_commands{$command_name} eq 'context') { _new_document_context($self, $command_name); } if (exists($format_context_commands{$command_name})) { push @{$self->{'document_context'}->[-1]->{'formatting_context'}}, {'context_name' => '@'.$command_name}; } if (exists($block_commands{$command_name})) { push @{$self->{'document_context'}->[-1]->{'block_commands'}}, $command_name; } my $preformatted = 0; if (exists($pre_class_commands{$command_name})) { push @{$self->{'document_context'}->[-1]->{'preformatted_classes'}}, $pre_class_commands{$command_name}; if (exists($preformatted_commands{$command_name})) { $self->{'document_context'}->[-1]->{'inside_preformatted'}++; $preformatted = 1; } elsif ($block_commands{$command_name} eq 'menu' and $self->{'document_context'}->[-1]->{'inside_preformatted'}) { $preformatted = 1; } } if (exists($composition_context_commands{$command_name})) { push @{$self->{'document_context'}->[-1]->{'composition_context'}}, $command_name; push @{$self->{'document_context'}->[-1]->{'preformatted_context'}}, $preformatted; } if (exists($format_raw_commands{$command_name})) { $self->{'document_context'}->[-1]->{'raw'}++; } elsif ($command_name eq 'verbatim') { $self->{'document_context'}->[-1]->{'verbatim'}++; } if (exists($brace_code_commands{$command_name}) or exists($preformatted_code_commands{$command_name})) { push @{$self->{'document_context'}->[-1]->{'monospace'}}, 1; } elsif (exists($brace_commands{$command_name}) and $brace_commands{$command_name} eq 'style_no_code') { push @{$self->{'document_context'}->[-1]->{'monospace'}}, 0; } elsif ($self->{'upper_case_commands'}->{$command_name}) { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'upper_case'}++; } elsif (exists($math_commands{$command_name})) { $self->{'document_context'}->[-1]->{'math'}++; $convert_to_latex = 1 if ($self->get_conf('CONVERT_TO_LATEX_IN_MATH')); } if ($command_name eq 'verb') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'space_protected'}++; } elsif ($command_name eq 'w') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'no_break'}++; } return $convert_to_latex; } sub _convert_command_update_context($$) { my ($self, $command_name) = @_; if (exists($composition_context_commands{$command_name})) { pop @{$self->{'document_context'}->[-1]->{'composition_context'}}; pop @{$self->{'document_context'}->[-1]->{'preformatted_context'}}; } if (exists($pre_class_commands{$command_name})) { pop @{$self->{'document_context'}->[-1]->{'preformatted_classes'}}; if (exists($preformatted_commands{$command_name})) { $self->{'document_context'}->[-1]->{'inside_preformatted'}--; } } if (exists($preformatted_code_commands{$command_name}) or (exists($brace_commands{$command_name}) and $brace_commands{$command_name} eq 'style_no_code') or exists($brace_code_commands{$command_name})) { pop @{$self->{'document_context'}->[-1]->{'monospace'}}; } elsif ($self->{'upper_case_commands'}->{$command_name}) { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'upper_case'}--; } elsif (exists($math_commands{$command_name})) { $self->{'document_context'}->[-1]->{'math'}--; } if ($command_name eq 'verb') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'space_protected'}--; } elsif ($command_name eq 'w') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'no_break'}--; } if (exists($format_raw_commands{$command_name})) { $self->{'document_context'}->[-1]->{'raw'}--; } elsif ($command_name eq 'verbatim') { $self->{'document_context'}->[-1]->{'verbatim'}--; } if (exists($block_commands{$command_name})) { pop @{$self->{'document_context'}->[-1]->{'block_commands'}}; } if (exists($format_context_commands{$command_name})) { pop @{$self->{'document_context'}->[-1]->{'formatting_context'}}; } if (exists($brace_commands{$command_name}) and $brace_commands{$command_name} eq 'context') { _pop_document_context($self); } } sub _open_type_update_context($$) { my ($self, $type_name) = @_; if ($type_name eq 'paragraph') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'paragraph_number'}++; } elsif ($type_name eq 'preformatted' or $type_name eq 'rawpreformatted') { $self->{'document_context'}->[-1]->{'formatting_context'}->[-1] ->{'preformatted_number'}++; } elsif ($self->{'pre_class_types'}->{$type_name}) { push @{$self->{'document_context'}->[-1]->{'preformatted_classes'}}, $self->{'pre_class_types'}->{$type_name}; push @{$self->{'document_context'}->[-1]->{'preformatted_context'}}, 1; push @{$self->{'document_context'}->[-1]->{'composition_context'}}, $type_name; } if ($self->{'code_types'}->{$type_name}) { push @{$self->{'document_context'}->[-1]->{'monospace'}}, 1; } } sub _convert_type_update_context($$) { my ($self, $type_name) = @_; if ($self->{'code_types'}->{$type_name}) { pop @{$self->{'document_context'}->[-1]->{'monospace'}}; } if ($self->{'pre_class_types'}->{$type_name}) { pop @{$self->{'document_context'}->[-1]->{'preformatted_classes'}}; pop @{$self->{'document_context'}->[-1]->{'composition_context'}}; pop @{$self->{'document_context'}->[-1]->{'preformatted_context'}}; } } sub _debug_print_html_contexts($) { my $self = shift; my @document_contexts = map {defined($_->{'context'}) ? $_->{'context'}: 'UNDEF'} @{$self->{'document_context'}}; my @contexts_names = map {defined($_->{'context_name'}) ? $_->{'context_name'}: 'UNDEF'} @{$self->{'document_context'}->[-1]->{'formatting_context'}}; return "[".join('|',@document_contexts)."](".join('|',@contexts_names).")"; } # Convert tree element $ELEMENT, and return HTML text for the output files. # $EXPLANATION is only used for debug. sub _convert($$;$); sub _convert($$;$) { my ($self, $element, $explanation) = @_; if (!defined($element)) { cluck('BUG: _convert: element UNDEF'); return ''; } # to help debug and trace my $command_type = ''; if (exists($element->{'cmdname'})) { $command_type = "\@$element->{'cmdname'} "; } if (exists($element->{'type'})) { $command_type .= $element->{'type'}; } $debug = $self->get_conf('DEBUG') if !defined($debug); # cache return value of get_conf for speed if ($debug) { #cluck() if (!defined($explanation)); $explanation = 'NO EXPLANATION' if (!defined($explanation)); my $contexts_str = _debug_print_html_contexts($self); print STDERR "ELEMENT($explanation) ".$contexts_str.", ->"; print STDERR " cmd: $element->{'cmdname'}," if (exists($element->{'cmdname'})); print STDERR " type: $element->{'type'}" if (exists($element->{'type'})); if (exists($element->{'text'})) { if ($element->{'text'} eq '') { print STDERR ' text(EMPTY)'; } else { my $text = $element->{'text'}; $text =~ s/\n/\\n/; print STDERR " text: $text"; } } print STDERR "\n"; } if (ref($element) ne 'HASH' and ref($element) ne 'Texinfo::TreeElement') { cluck "_convert: tree element not a HASH\n"; return ''; } if ((exists($element->{'type'}) and exists($self->{'types_conversion'}->{$element->{'type'}}) and !defined($self->{'types_conversion'}->{$element->{'type'}})) or (exists($element->{'cmdname'}) and exists($self->{'commands_conversion'}->{$element->{'cmdname'}}) and !defined($self->{'commands_conversion'}->{$element->{'cmdname'}}))) { if ($debug) { print STDERR "IGNORED $command_type\n"; } return ''; } # Process text if (exists($element->{'text'})) { my $result; # already converted to html, keep it as is if (exists($element->{'type'}) and $element->{'type'} eq '_converted') { $result = $element->{'text'}; } else { $result = &{$self->{'types_conversion'}->{'text'}} ($self, $element->{'type'}, $element, $element->{'text'}); } print STDERR "DO TEXT => `$result'\n" if $debug; return $result; } # commands like @deffnx have both a cmdname and a def_line type. It is # better to consider them as a def_line type, as the whole point of the # def_line type is to handle the same the def*x and def* line formatting. if (exists($element->{'cmdname'}) and !((exists($element->{'type'}) and $element->{'type'} eq 'definfoenclose_command') or (exists($element->{'type'}) and $element->{'type'} eq 'index_entry_command'))) { my $command_name = $element->{'cmdname'}; my $data_command_name; if ($command_name eq 'item' and exists($element->{'contents'}) and exists($element->{'contents'}->[0]->{'type'}) and $element->{'contents'}->[0]->{'type'} eq 'line_arg') { $data_command_name = 'item_LINE'; } else { $data_command_name = $command_name; } if (exists($root_commands{$command_name})) { $self->{'current_root_command'} = $element; } if (exists($self->{'commands_conversion'}->{$command_name})) { my $convert_to_latex = _open_command_update_context($self, $command_name); my $result = ''; if (defined($self->{'commands_open'}->{$command_name})) { $result .= &{$self->{'commands_open'}->{$command_name}}($self, $command_name, $element); } my $content_formatted = ''; if (exists($element->{'contents'}) and (exists($root_commands{$command_name}) or exists($block_commands{$command_name}) or $command_name eq 'tab' or $command_name eq 'headitem' or $data_command_name eq 'item')) { if ($convert_to_latex) { # displaymath $content_formatted = Texinfo::Convert::LaTeX::convert_to_latex_math(undef, Texinfo::TreeElement::new({'contents' => $element->{'contents'}}), $self->{'options_latex_math'}); } else { my $contents_nr = scalar(@{$element->{'contents'}}); for (my $idx = 0; $idx < $contents_nr; $idx++) { $content_formatted .= _convert($self, $element->{'contents'}->[$idx], "$command_type c[$idx]"); } } } my $args_formatted; # contents could be not set for brace commands without braces if ((exists($brace_commands{$command_name}) and exists($element->{'contents'})) or (exists($line_commands{$command_name}) and $line_commands{$command_name} eq 'line') or (($command_name eq 'item' or $command_name eq 'itemx') and exists($element->{'contents'}) and exists($element->{'contents'}->[0]->{'type'}) and $element->{'contents'}->[0]->{'type'} eq 'line_arg') or ($command_name eq 'quotation' or $command_name eq 'smallquotation') or $command_name eq 'float' or $command_name eq 'cartouche') { my $arguments_list; if (exists($element->{'contents'}->[0]->{'type'}) and $element->{'contents'}->[0]->{'type'} eq 'arguments_line') { $arguments_list = $element->{'contents'}->[0]->{'contents'}; } else { $arguments_list = $element->{'contents'}; } $args_formatted = []; my @args_specification; @args_specification = @{$html_default_commands_args{$command_name}} if (exists($html_default_commands_args{$command_name})); my $spec_nr = scalar(@args_specification); my $arg_idx = -1; foreach my $arg (@{$arguments_list}) { $arg_idx++; my $arg_spec; if ($arg_idx < $spec_nr) { $arg_spec = $args_specification[$arg_idx]; } if (!exists($arg->{'contents'})) { push @$args_formatted, undef; next; } # NOTE here commands with empty array reference in array # reference associated to command in html_default_commands_args # do not have $arg_spec reset to normal, such that their argument # is not converted here $arg_spec = ['normal'] if (!defined($arg_spec)); my $arg_formatted = {'arg_tree' => $arg}; foreach my $arg_type (@$arg_spec) { my $explanation = "$command_type A[$arg_idx]$arg_type"; if ($arg_type eq 'normal') { if ($convert_to_latex) { $arg_formatted->{'normal'} = Texinfo::Convert::LaTeX::convert_to_latex_math(undef, $arg, $self->{'options_latex_math'}); } else { $arg_formatted->{'normal'} = _convert($self, $arg, $explanation); } } elsif ($arg_type eq 'monospace') { _set_code_context($self, 1); $arg_formatted->{$arg_type} = _convert($self, $arg, $explanation); _pop_code_context($self); } elsif ($arg_type eq 'string') { _new_document_context($self, $command_type, $CTXF_string); $arg_formatted->{$arg_type} = _convert($self, $arg, $explanation); #_unset_string_context($self); _pop_document_context($self); } elsif ($arg_type eq 'monospacestring') { _new_document_context($self, $command_type, $CTXF_string | $CTXF_code); $arg_formatted->{$arg_type} = _convert($self, $arg, $explanation); _pop_document_context($self); } elsif ($arg_type eq 'monospacetext') { Texinfo::Convert::Text::set_options_code( $self->{'convert_text_options'}); $arg_formatted->{$arg_type} = Texinfo::Convert::Text::convert_to_text($arg, $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_code( $self->{'convert_text_options'}); } elsif ($arg_type eq 'filenametext') { Texinfo::Convert::Text::set_options_code( $self->{'convert_text_options'}); # Always use encoded characters for file names Texinfo::Convert::Text::set_options_encoding_if_not_ascii($self, $self->{'convert_text_options'}); $arg_formatted->{$arg_type} = Texinfo::Convert::Text::convert_to_text($arg, $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_code( $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_encoding( $self->{'convert_text_options'}); } elsif ($arg_type eq 'url') { Texinfo::Convert::Text::set_options_code( $self->{'convert_text_options'}); # set the encoding to UTF-8 to always have a string that # is suitable for percent encoding. Texinfo::Convert::Text::set_options_encoding( $self->{'convert_text_options'}, 'utf-8'); $arg_formatted->{$arg_type} = Texinfo::Convert::Text::convert_to_text($arg, $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_code( $self->{'convert_text_options'}); Texinfo::Convert::Text::reset_options_encoding( $self->{'convert_text_options'}); } elsif ($arg_type eq 'raw') { _set_raw_context($self); $arg_formatted->{$arg_type} = _convert($self, $arg, $explanation); _unset_raw_context($self); } } push @$args_formatted, $arg_formatted; } } _convert_command_update_context($self, $command_name); # TODO remove some time in the future of 2024, it is not used # in texi2any and have never been documented. It may be used in 3rd # party codes, though. if ($element->{'cmdname'} eq 'node') { $self->{'current_node'} = $element; } # args are formatted, now format the command itself if (defined($args_formatted)) { if (!defined($self->{'commands_conversion'}->{$command_name})) { print STDERR "No command_conversion for $command_name\n"; } else { $result .= &{$self->{'commands_conversion'}->{$command_name}}($self, $command_name, $element, $args_formatted, $content_formatted); } } else { $result .= &{$self->{'commands_conversion'}->{$command_name}}($self, $command_name, $element, undef, $content_formatted); } if ($command_name eq 'documentlanguage') { _translate_names($self); } return $result; } else { print STDERR "Command not converted: $command_name\n" if ($self->get_conf('VERBOSE') or $self->get_conf('DEBUG')); if (exists($root_commands{$command_name})) { delete $self->{'current_root_command'}; } return ''; } } elsif (exists($element->{'type'})) { my $result = ''; my $type_name = $element->{'type'}; _open_type_update_context($self, $type_name); if (defined($self->{'types_open'}->{$type_name})) { $result .= &{$self->{'types_open'}->{$type_name}}($self, $type_name, $element); } my $content_formatted = ''; if ($type_name eq 'definfoenclose_command') { if (exists($element->{'contents'})) { $content_formatted = _convert($self, $element->{'contents'}->[0], "DEFINFOENCLOSE_ARG"); } } elsif (exists($element->{'contents'}) and $type_name ne 'untranslated_def_line_arg') { my $content_idx = 0; foreach my $content (@{$element->{'contents'}}) { $content_formatted .= _convert($self, $content, "$command_type c[$content_idx]"); $content_idx++; } } _convert_type_update_context($self, $type_name); if (defined($self->{'types_conversion'}->{$type_name})) { $result .= &{$self->{'types_conversion'}->{$type_name}} ($self, $type_name, $element, $content_formatted); } else { $result .= $content_formatted; } print STDERR "DO type ($type_name) => `$result'\n" if $debug; return $result; # no type, no cmdname, but contents. } elsif (exists($element->{'contents'})) { # this happens inside accents, for section/node names, for @images. my $content_formatted = ''; my $content_idx = 0; foreach my $content (@{$element->{'contents'}}) { $content_formatted .= _convert($self, $content, "$command_type C[$content_idx]"); $content_idx++; } print STDERR "UNNAMED HOLDER => `$content_formatted'\n" if $debug; return $content_formatted; } else { print STDERR "UNNAMED empty\n" if $debug; if (defined($self->{'types_conversion'}->{''})) { return &{$self->{'types_conversion'}->{''}} ($self, $element); } else { return ''; } } print STDERR "DEBUG: HERE!($element)\n"; } sub _set_variables_texi2html($) { my $options = shift; my @texi2html_options = ( ['SECTION_BUTTONS', ['FastBack', 'Back', 'Up', 'Forward', 'FastForward', 'Space', 'Space', 'Space', 'Space', 'Top', 'Contents', 'Index', 'About' ]], ['TOP_BUTTONS', ['Back', 'Forward', 'Space', 'Contents', 'Index', 'About']], ['TOP_FOOTER_BUTTONS', ['Back', 'Forward', 'Space', 'Contents', 'Index', 'About']], ['MISC_BUTTONS', [ 'Top', 'Contents', 'Index', 'About' ]], ['CHAPTER_BUTTONS', [ 'FastBack', 'FastForward', 'Space', 'Space', 'Space', 'Space', 'Space', 'Top', 'Contents', 'Index', 'About', ]], ['SECTION_FOOTER_BUTTONS', [ 'FastBack', 'FirstInFileBack', 'FirstInFileUp', 'Forward', 'FastForward' ]], ['CHAPTER_FOOTER_BUTTONS', [ 'FastBack', 'FastForward', 'Space', 'Space', 'Space', 'Space', 'Space', 'Top', 'Contents', 'Index', 'About', ]], ['NODE_FOOTER_BUTTONS', [ 'FastBack', 'Back', 'Up', 'Forward', 'FastForward', 'Space', 'Space', 'Space', 'Space', 'Top', 'Contents', 'Index', 'About' ]], ); my $regular_texi2html_options = Texinfo::Options::get_regular_options('texi2html'); foreach my $option (keys(%$regular_texi2html_options)) { $options->{$option} = $regular_texi2html_options->{$option}; } foreach my $option (@texi2html_options) { $options->{$option->[0]} = $option->[1]; } } 1; # The documentation of the customization API is in the texi2any_api # Texinfo manual. POD format is not suitable for such a documentation, because # of the module documentation style, the language limitations, and also because # the customization API involves multiple modules as well as the main program. __END__ # Automatically generated from Convert_format_template.pod =head1 NAME Texinfo::Convert::HTML - Convert Texinfo tree to HTML =head1 SYNOPSIS my $converter = Texinfo::Convert::HTML->converter({'NUMBER_SECTIONS' => 0}); # output to files $converter->output($document); # no header nor footer output my $converted = $converter->convert($document); $converter->output_internal_links(); # HTML only =head1 NOTES The Texinfo Perl module main purpose is to be used in C to convert Texinfo to other formats. There is no promise of API stability. =head1 DESCRIPTION Texinfo::Convert::HTML converts a Texinfo tree to HTML. =head1 METHODS =over =item $converter = Texinfo::Convert::HTML->converter($options) Initialize converter from Texinfo to HTML. The I<$options> hash reference holds Texinfo customization options for the converter. These options should be Texinfo customization options that can be passed to the converter. Most of the customization options are described in the Texinfo manual or in the customization API manual. Those customization options, when appropriate, override the document content. See L for more information. =item $converter->output($document) Convert a Texinfo parsed document I<$document> and output the result in files as described in the Texinfo manual. =item $result = $converter->convert($document) Convert a Texinfo parsed document I<$document> and return the resulting output. =item $result = $converter->convert_tree($tree) Convert a Texinfo tree portion I<$tree> and return the resulting output. This function does not try to output a full document but only portions. In general it is better to call this function when conversion is already ongoing, as it requires an association to a document and a suitably initialized converter formatting state. =item $result = $converter->output_internal_links() X> Returns text representing the links in the document. The format should follow the C<--internal-links> option of the C specification. This is only supported in (and relevant for) HTML. =back =head1 AUTHOR Patrice Dumas, Ebug-texinfo@gnu.orgE =head1 COPYRIGHT AND LICENSE Copyright 2010- Free Software Foundation, Inc. See the source file for all copyright years. This library is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. =cut