blob: d486e1eb9f0fd2eeab1b2fc14e86028d90ae6d05 [file] [log] [blame]
# IXIN.pm: output tree as IXIN.
#
# Copyright 2013 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 <http://www.gnu.org/licenses/>.
#
# Original author: Patrice Dumas <pertusus@free.fr>
#
# This module implements abstract functions that output the IXIN format
# using lower level formatting funtions, here adapted to lisp like
# output. For other output, the output specific functions should be
# redefined. This module is not enough to output IXIN format, a module
# inheriting both from a converter module and this module is required.
package Texinfo::Convert::IXIN;
use 5.00405;
use strict;
use MIME::Base64;
use Texinfo::Convert::TexinfoSXML;
use Texinfo::Common;
use Carp qw(cluck);
require Exporter;
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK %EXPORT_TAGS);
@ISA = qw(Exporter Texinfo::Convert::Converter);
# Items to export into callers namespace by default. Note: do not export
# names by default without a very good reason. Use EXPORT_OK instead.
# Do not simply export all your public functions/methods/constants.
# This allows declaration use Texinfo::Convert::IXIN ':all';
# If you do not need this, moving things directly into @EXPORT or @EXPORT_OK
# will save memory.
%EXPORT_TAGS = ( 'all' => [ qw(
output_ixin
) ] );
@EXPORT_OK = ( @{ $EXPORT_TAGS{'all'} } );
@EXPORT = qw(
);
$VERSION = '6.1';
my $ixin_version = 1;
sub _ixin_version($)
{
my $self = shift;
return $ixin_version;
}
my %additional_setting_commands;
# FIXME pagesizes is line
foreach my $command ('pagesizes', 'everyheading', 'everyfooting',
'evenheading', 'evenfooting', 'oddheading', 'oddfooting',
'documentencoding', 'documentlanguage', 'clickstyle') {
$additional_setting_commands{$command} = 1;
}
# Here are all the commands that are misc_commands with type matching \d
# and are also global_unique_commands/global_multiple_commands in Parser.pm
# but are not setting commands.
my %global_misc_not_setting_commands = (
'printindex' => 1,
);
my @image_files_extensions = ('eps', 'gif', 'jpg', 'jpeg', 'pdf', 'png', 'svg',
'txt');
my %extension_mime_mapping = (
'eps' => 'application/postscript',
'gif' => 'image/gif',
'jpg' => 'image/jpeg',
'jpeg' => 'image/jpeg',
'pdf' => 'application/pdf',
'png' => 'image/png',
'svg' => 'image/svg+xml',
'txt' => 'text/plain',
'tiff' => 'image/tiff',
'' => 'image/unknown',
);
# output specific
sub ixin_header($)
{
my $self = shift;
my $header = 'ixin '.$self->_ixin_version().';';
if ($self->get_conf('OUTPUT_ENCODING_NAME')) {
$header .= ' -*- coding: '. $self->get_conf('OUTPUT_ENCODING_NAME') .'-*-;';
}
$header .= "\n";
}
my %attribute_string_names = (
'nodeentry' => {'name' => 1},
'nodelabel' => {'name' => 1},
'floatentry' => {'name' => 1},
'label' => {'name' => 1},
'filename' => {'name' => 1},
'settingvalue' => {'value' => 1},
'nodetweakvalue' => {'value' => 1},
'floatindex' => {'type' => 1},
'blobentry' => {'mimetype' => 1, 'filename' => 1},
);
sub _ixin_attributes($$$)
{
my $self = shift;
my $name = shift;
my $attributes = shift;
my $result = '';
if ($attributes) {
for (my $i = 0; $i < scalar(@$attributes); $i += 2) {
if ($attribute_string_names{$name}
and $attribute_string_names{$name}->{$attributes->[$i]}) {
$result .= '"'
.Texinfo::Convert::TexinfoSXML->protect_text($attributes->[$i+1]).'"';
} else {
$result .= $attributes->[$i+1];
}
$result .= ' ';
}
}
return $result;
}
sub ixin_open_element($$;$)
{
my $self = shift;
my $name = shift;
my $attributes = shift;
my $result = '(';
$result .= $self->_ixin_attributes($name, $attributes);
return $result;
}
sub ixin_list_element($$$)
{
my $self = shift;
my $name = shift;
my $attributes = shift;
my $result = $self->_ixin_attributes($name, $attributes);
$result =~ s/ $//;
return $result;
}
sub ixin_close_element($$)
{
my $self = shift;
my $name = shift;
return ')';
#return "|$name)";
}
sub ixin_element($$;$)
{
my $self = shift;
my $name = shift;
my $attributes = shift;
my $opening = $self->ixin_open_element($name, $attributes);
$opening =~ s/ $//;
return $opening . $self->ixin_close_element($name);
}
sub ixin_symbol_element($$$)
{
my $self = shift;
my $name = shift;
my $string = shift;
return $string;
}
sub ixin_none_element($$)
{
my $self = shift;
my $name = shift;
return ' - ';
}
# end output specific subs
# FIXME this is rather non specific. Move to Converter?
sub _get_element($$);
sub _get_element($$)
{
my $self = shift;
my $current = shift;
my ($element, $root_command);
while (1) {
#print STDERR Texinfo::Common::_print_current($current);
if ($current->{'type'}) {
if ($current->{'type'} eq 'element') {
return ($current, $root_command);
}
}
if ($current->{'cmdname'}) {
if ($Texinfo::Common::root_commands{$current->{'cmdname'}}) {
$root_command = $current;
return ($element, $root_command) if defined($element);
}
}
if ($current->{'parent'}) {
$current = $current->{'parent'};
} else {
return ($element, $root_command);
}
}
}
sub _count_bytes($$)
{
my $self = shift;
my $string = shift;
return Texinfo::Common::count_bytes($self, $string);
}
sub _associated_node_id($$$;$)
{
my $self = shift;
my $command = shift;
my $node_label_number = shift;
my $node_command = shift;
if (!defined($node_command)) {
my ($element, $root_command) = $self->_get_element($command);
if ($root_command) {
if (!$root_command->{'cmdname'} or $root_command->{'cmdname'} ne 'node') {
if ($element->{'extra'}->{'element_command'}
and $element->{'extra'}->{'element_command'}
and $element->{'extra'}->{'element_command'}->{'cmdname'}
and $element->{'extra'}->{'element_command'}->{'cmdname'} eq 'node') {
$node_command = $element->{'extra'}->{'element_command'};
}
} else {
$node_command = $root_command;
}
}
}
my $associated_node_id;
if (defined($node_command)
and defined($node_command->{'extra'}->{'normalized'})) {
$associated_node_id
= $node_label_number->{$node_command->{'extra'}->{'normalized'}};
} else {
$associated_node_id = -1;
}
return $associated_node_id;
}
sub _index_font_name($$)
{
my $self = shift;
my $in_code = shift;
if ($in_code) {
return 'code';
} else {
return 'r';
}
}
my @node_directions = ('Next', 'Prev', 'Up');
sub output_ixin($$)
{
my $self = shift;
my $root = shift;
$self->_set_outfile();
return undef unless $self->_create_destination_directory();
my $fh;
if (! $self->{'output_file'} eq '') {
$fh = $self->Texinfo::Common::open_out($self->{'output_file'});
if (!$fh) {
$self->document_error(sprintf($self->__("could not open %s for writing: %s"),
$self->{'output_file'}, $!));
return undef;
}
}
$self->_set_global_multiple_commands(-1);
# we ignore everything before the first node
$self->_set_ignored_type('text_root');
my $result = $self->ixin_header();
$result .= $self->ixin_open_element('meta');
$result .= $self->ixin_open_element('xid');
my $output_file = $self->ixin_none_element('filename');
if ($self->{'output_file'} ne '') {
$result .= $self->ixin_list_element('filename',
['name', $self->{'output_file'}]);
}
my $lang = $self->get_conf('documentlanguage');
#my $lang_code = $lang;
#my $region_code;
#if ($lang =~ /^([a-z]+)_([A-Z]+)/) {
# $lang_code = $1;
# $region_code = $2;
#}
$result .= ' ';
$result .= $self->ixin_list_element('lang', ['name', $lang]);
# FIXME title: use simpletitle or fulltitle
if ($self->{'info'}->{'dircategory_direntry'}) {
my $current_category;
foreach my $dircategory_direntry (@{$self->{'info'}->{'dircategory_direntry'}}) {
if ($dircategory_direntry->{'cmdname'} and $dircategory_direntry->{'cmdname'} eq 'dircategory') {
if ($current_category) {
$result .= $self->ixin_close_element('category');
}
$current_category = $dircategory_direntry;
$result .= $self->ixin_open_element('category');
# FIXME wait for Thien-Thi input on renderable or string.
} elsif ($dircategory_direntry->{'cmdname'}
and $dircategory_direntry->{'cmdname'} eq 'direntry') {
# FIXME wait for Thien-Thi input on renderable or string and node
# rendering
}
}
if ($current_category) {
$result .= $self->ixin_close_element('category');
}
}
$result .= $self->ixin_close_element('xid');
# FIXME vars: wait for Thien-Thi answer.
my $elements = Texinfo::Structuring::split_by_node($root);
# setting_commands is for @-commands appearing before the first node,
# while end_of_nodes_setting_commands holds, for @-commands names, the
# last @-command element.
my %setting_commands;
my %end_of_nodes_setting_commands;
my %setting_commands_defaults;
foreach my $global_command (keys(%{$self->{'extra'}})) {
if ((($Texinfo::Common::misc_commands{$global_command}
and $Texinfo::Common::misc_commands{$global_command} =~ /^\d/)
or $additional_setting_commands{$global_command})
and !$global_misc_not_setting_commands{$global_command}) {
if (ref($self->{'extra'}->{$global_command}) eq 'ARRAY') {
if (defined($Texinfo::Common::document_settable_at_commands{$global_command})) {
$setting_commands_defaults{$global_command}
= $Texinfo::Common::document_settable_at_commands{$global_command};
}
foreach my $command (@{$self->{'extra'}->{$global_command}}) {
my ($element, $root_command) = _get_element($self, $command);
# before first node
if (!defined($root_command->{'extra'})
and !defined($root_command->{'extra'}->{'normalized'})) {
$setting_commands{$global_command} = $command;
} else {
# register the setting value at the end of the node
$end_of_nodes_setting_commands{$root_command->{'extra'}->{'normalized'}}->{$global_command}
= $command;
}
#print STDERR "$element $root_command->{'extra'} $global_command\n";
}
} else {
if (defined($Texinfo::Common::document_settable_unique_at_commands{$global_command})) {
$setting_commands_defaults{$global_command}
= $Texinfo::Common::document_settable_unique_at_commands{$global_command};
}
$setting_commands{$global_command} = $self->{'extra'}->{$global_command};
}
}
}
my %settings;
foreach my $setting_command_name (keys(%setting_commands)) {
my $setting_command = $setting_commands{$setting_command_name};
$setting_command_name = 'shortcontents'
if ($setting_command_name eq 'summarycontents');
my $value = $self->_informative_command_value($setting_command);
#print STDERR "$setting_command_name $value\n";
# do not register settings if sete at the default value.
if (defined($value)
and !(defined($setting_commands_defaults{$setting_command_name})
and $setting_commands_defaults{$setting_command_name} eq $value)) {
$settings{$setting_command_name} = $value;
}
}
$result .= ' ';
$result .= $self->ixin_open_element('settings');
if (scalar(keys(%settings))) {
foreach my $command_name (sort(keys(%settings))) {
$result .= $self->ixin_open_element('setting');
$result .= $self->ixin_symbol_element('settingname', $command_name);
$result .= ' ';
if ($Texinfo::Common::misc_commands{$command_name} eq 'lineraw') {
$result .= $self->ixin_list_element('settingvalue',
['value', $settings{$command_name}]);
} else {
$result .= $self->ixin_symbol_element('settingvalue', $settings{$command_name});
}
$result .= $self->ixin_close_element('setting');
}
}
$result .= $self->ixin_close_element('settings');
foreach my $region ('copying', 'titlepage') {
if ($self->{'extra'}->{$region}) {
$result .= $self->convert_tree($self->{'extra'}->{$region});
} else {
$result .= $self->ixin_none_element($region);
}
}
# FIXME toc: wait for Thien-Thi answer.
$result .= $self->ixin_close_element('meta');
$result .= "\n";
# to do the nodes index, one need the size of each node.
# to do the counts list, one need to know the sizze of the node index.
# So we have to start by the node data.
my $node_nr = 0;
my %current_settings;
my %node_label_number;
my %node_byte_sizes;
my %node_tweaks;
my @nodes;
my $document_output = '';
if ($elements) {
foreach my $node_element (@$elements) {
next if ($node_element->{'extra'}->{'no_node'});
$node_nr++;
my $node = $node_element->{'extra'}->{'element_command'};
push @nodes, $node;
my $normalized_node_name = $node->{'extra'}->{'normalized'};
foreach my $setting_command_name (keys(%current_settings)) {
$node_tweaks{$normalized_node_name}->{$setting_command_name}
= $current_settings{$setting_command_name};
}
$node_label_number{$normalized_node_name} = $node_nr;
my $node_result = $self->convert_tree($node_element)."\n";
$document_output .= $node_result;
# get node length.
$node_byte_sizes{$normalized_node_name}
= $self->_count_bytes($node_result);
# update current settings
if (defined($end_of_nodes_setting_commands{$normalized_node_name})) {
foreach my $setting_command_name (keys(%{$end_of_nodes_setting_commands{$normalized_node_name}})) {
my $value = $self->_informative_command_value(
$end_of_nodes_setting_commands{$normalized_node_name}->{$setting_command_name});
if ((defined($settings{$setting_command_name})
and $settings{$setting_command_name} eq $value)
or (!defined($settings{$setting_command_name})
and defined($setting_commands_defaults{$setting_command_name})
and $setting_commands_defaults{$setting_command_name} eq $value)) {
delete $current_settings{$setting_command_name};
} else {
$current_settings{$setting_command_name} = $value;
}
}
}
}
} else {
# not a full document.
}
my $nodes_index = $self->ixin_open_element('nodesindex');
foreach my $node (@nodes) {
my $normalized_node_name = $node->{'extra'}->{'normalized'};
# FIXME name should be a renderable sequence
my @attributes = ('name', $normalized_node_name,
'length', $node_byte_sizes{$normalized_node_name});
foreach my $direction (@node_directions) {
if ($node->{'node_'.lc($direction)}) {
my $node_direction = $node->{'node_'.lc($direction)};
if ($node_direction->{'extra'}->{'manual_content'}) {
# FIXME?
push @attributes, ('node'.lc($direction), -2);
} else {
push @attributes, ('node'.lc($direction),
$node_label_number{$node_direction->{'extra'}->{'normalized'}})
}
} else {
push @attributes, ('node'.lc($direction), -1);
}
}
$nodes_index .= $self->ixin_open_element('nodeentry', \@attributes);
if ($node_tweaks{$normalized_node_name}) {
$nodes_index .= $self->ixin_open_element('nodetweaks');
foreach my $command_name (sort(keys(%{$node_tweaks{$normalized_node_name}}))) {
$nodes_index .= $self->ixin_open_element('nodetweak');
$nodes_index .= $self->ixin_symbol_element('nodetweakname', $command_name);
$nodes_index .= ' ';
if ($Texinfo::Common::misc_commands{$command_name} eq 'lineraw') {
$nodes_index .= $self->ixin_list_element('nodetweakvalue',
['value', $node_tweaks{$normalized_node_name}->{$command_name}]);
} else {
$nodes_index .= $self->ixin_symbol_element('nodetweakvalue',
$node_tweaks{$normalized_node_name}->{$command_name});
}
$nodes_index .= $self->ixin_close_element('nodetweak');
}
$nodes_index .= $self->ixin_close_element('nodetweaks');
}
$nodes_index .= $self->ixin_close_element('nodeentry');
}
$nodes_index .= $self->ixin_close_element('nodesindex');
$nodes_index .= "\n";
# do sectioning tree
my $sectioning_tree = '';
$sectioning_tree .= $self->ixin_open_element('sectioningtree');
if ($self->{'structuring'} and $self->{'structuring'}->{'sectioning_root'}) {
my $section_root = $self->{'structuring'}->{'sectioning_root'};
foreach my $top_section (@{$section_root->{'section_childs'}}) {
my $section = $top_section;
SECTION:
while ($section) {
my $associated_node_id = $self->_associated_node_id($section,
\%node_label_number);
my @attributes = ('nodeid', $associated_node_id, 'type',
$self->_level_corrected_section($section));
$sectioning_tree .= $self->ixin_open_element('sectionentry',
\@attributes);
$sectioning_tree .= $self->ixin_open_element('sectiontitle');
if ($section->{'args'} and $section->{'args'}->[0]) {
$sectioning_tree .= $self->convert_tree($section->{'args'}->[0]);
}
$sectioning_tree .= $self->ixin_close_element('sectiontitle');
# top is special and never considered to contain anything. So
# it is closed here and not below.
if ($section->{'cmdname'} eq 'top') {
$sectioning_tree .= $self->ixin_close_element('sectionentry');
}
if ($section->{'section_childs'}) {
$section = $section->{'section_childs'}->[0];
} elsif ($section->{'section_next'}) {
$sectioning_tree .= $self->ixin_close_element('sectionentry');
last if ($section eq $top_section);
$section = $section->{'section_next'};
} else {
if ($section eq $top_section) {
$sectioning_tree .= $self->ixin_close_element('sectionentry')
unless ($section->{'cmdname'} eq 'top');
last;
}
while ($section->{'section_up'}) {
$section = $section->{'section_up'};
$sectioning_tree .= $self->ixin_close_element('sectionentry');
if ($section eq $top_section) {
$sectioning_tree .= $self->ixin_close_element('sectionentry')
unless ($section->{'cmdname'} eq 'top');
last SECTION;
}
if ($section->{'section_next'}) {
$sectioning_tree .= $self->ixin_close_element('sectionentry');
$section = $section->{'section_next'};
last;
}
}
}
}
}
}
$sectioning_tree .= $self->ixin_close_element('sectioningtree') . "\n";
# do labels
my $non_node_labels_text = '';
my $labels_nr = 0;
my %floats_associated_node_id;
if ($self->{'labels'}) {
foreach my $label (sort(keys(%{$self->{'labels'}}))) {
my $command = $self->{'labels'}->{$label};
next if ($command->{'cmdname'} eq 'node');
$labels_nr++;
my $associated_node_id = $self->_associated_node_id($command,
\%node_label_number);
$non_node_labels_text .= $self->ixin_element('label', ['name', $label,
'nodeid', $associated_node_id,
'type', $command->{'cmdname'}]);
# register floats to avoid doing it twice for the float specific index
if ($command->{'cmdname'} eq 'float') {
$floats_associated_node_id{$command} = $associated_node_id;
}
}
}
my $labels_text = $self->ixin_open_element('labels', ['count', $labels_nr]);
foreach my $node (@nodes) {
$labels_text .= $self->ixin_list_element('nodelabel', ['name',
$node->{'extra'}->{'normalized'}]);
$labels_text .= ' ';
}
$labels_text .= $non_node_labels_text
. $self->ixin_close_element('labels')."\n";
# do document-term sets (indices counts and indices)
my %dts_information;
if ($self->{'parser'}) {
my ($index_names, $merged_indices)
= $self->{'parser'}->indices_information();
my $merged_index_entries
= Texinfo::Structuring::merge_indices($index_names);
my $entries
= $self->Texinfo::Structuring::sort_indices($merged_index_entries,
$index_names);
# first do the dts_text as the counts are needed for the dts index
foreach my $index_name (sort(keys(%$entries))) {
my $dts_text_result = '';
my $dts_entries_nr = 0;
my $dts_in_code = $index_names->{$index_name}->{'in_code'};
foreach my $dts_entry (@{$entries->{$index_name}}) {
my $node = $dts_entry->{'node'};
my $associated_node_id;
if (defined($node)) {
$associated_node_id = $self->_associated_node_id(undef,
\%node_label_number, $node);
} else {
$associated_node_id = -1;
}
my $entry = $self->convert_tree({'contents' => $dts_entry->{'content'}});
$dts_text_result .= $self->ixin_open_element('dtsentry',
['nodeid', $associated_node_id]);
$dts_text_result .= $self->ixin_open_element('dtsterm');
$dts_text_result .= $entry;
$dts_text_result .= $self->ixin_close_element('dtsterm');
if ($dts_entry->{'in_code'} != $dts_in_code) {
my $font_name = $self->_index_font_name($dts_entry->{'in_code'});
$dts_text_result .= ' ';
$dts_text_result .= $self->ixin_list_element('dtsfont', ['font',
$font_name]);
}
$dts_text_result .= $self->ixin_close_element('dtsentry');
$dts_entries_nr++;
}
my $dts_opening = $self->ixin_open_element('dts', ['count', $dts_entries_nr,
'font', $self->_index_font_name($dts_in_code)]);
$dts_text_result = $dts_opening . $dts_text_result
. $self->ixin_close_element('dts') . "\n";
$dts_information{$index_name}->{'dts_text'} = $dts_text_result;
}
}
# Gather informations on printindex @-commands associated node id
if ($self->{'extra'}->{'printindex'}) {
foreach my $command (@{$self->{'extra'}->{'printindex'}}) {
my $associated_node_id = $self->_associated_node_id($command,
\%node_label_number);
if ($command->{'extra'} and $command->{'extra'}->{'misc_args'}
and defined($command->{'extra'}->{'misc_args'}->[0])) {
my $index_name = $command->{'extra'}->{'misc_args'}->[0];
push @{$dts_information{$index_name}->{'node_id'}}, $associated_node_id;
}
}
}
# now construct dts_index and dts_text
my $dts_index = '';
my $dts_text = $self->ixin_open_element('dtssets');
foreach my $index_name (sort(keys(%dts_information))) {
my $dts_len = 0;
if (exists($dts_information{$index_name}->{'dts_text'})) {
$dts_len = $self->_count_bytes($dts_information{$index_name}->{'dts_text'});
$dts_text .= $dts_information{$index_name}->{'dts_text'};
}
my @attributes = ('name', $index_name, 'dtslen', $dts_len);
$dts_index .= $self->ixin_open_element('dtsindexentry', \@attributes);
if ($dts_information{$index_name}->{'node_id'}) {
foreach my $node_id (sort(@{$dts_information{$index_name}->{'node_id'}})) {
$dts_index .= $self->ixin_list_element('dtsnodeid', ['nodeid', $node_id]);
$dts_index .= ' ';
}
}
$dts_index =~ s/ $//;
$dts_index .= $self->ixin_close_element('dtsindexentry');
}
$dts_text .= $self->ixin_close_element('dtssets') ."\n";
if ($dts_index ne '') {
$dts_index = $self->ixin_open_element('dtsindex', ['dtsindexlen',
$self->_count_bytes($dts_text)])
. $dts_index . $self->ixin_close_element('dtsindex');
} else {
$dts_index = $self->ixin_none_element('dtsindex')
}
# do floats
my %floats_information;
# collect all float types corresponding to float commands
if ($self->{'floats'}) {
foreach my $type (keys(%{$self->{'floats'}})) {
$floats_information{$type} = {};
}
}
# collect listoffloats information
if ($self->{'extra'}->{'listoffloats'}) {
foreach my $command (@{$self->{'extra'}->{'listoffloats'}}) {
my $associated_node_id = $self->_associated_node_id($command,
\%node_label_number);
my $type = $command->{'extra'}->{'type'}->{'normalized'};
if ($command->{'extra'}->{'type'}->{'content'}) {
$floats_information{$type}->{'type'}
= $self->convert_tree({'contents'
=> $command->{'extra'}->{'type'}->{'content'}});
}
push @{$floats_information{$type}->{'node_id'}}, $associated_node_id;
}
}
# now do the floats sets and the floats index
my $floats_text = $self->ixin_open_element('floatsset');
my $floats_index = '';
foreach my $type (sort(keys(%floats_information))) {
my $float_text_len = 0;
if ($self->{'floats'}->{$type}) {
my $float_nr = 0;
my $float_text = '';
foreach my $float (@{$self->{'floats'}->{$type}}) {
$float_nr++;
my $associated_node_id;
# associated node already found when collecting labels
if (exists($floats_associated_node_id{$float})) {
$associated_node_id = $floats_associated_node_id{$float};
} else {
$associated_node_id = $self->_associated_node_id($float,
\%node_label_number);
}
my @attribute = ('nodeid', $associated_node_id);
$float_text .= $self->ixin_open_element('floatentry', \@attribute);
if ($float->{'extra'}->{'normalized'}) {
$float_text .= $self->ixin_list_element('floatlabel',
['name', $float->{'extra'}->{'normalized'}]);
$float_text .= ' ';
} else {
$float_text .= $self->ixin_none_element('floatlabel');
}
if ($float->{'extra'}->{'node_content'}) {
$float_text .= $self->ixin_open_element('floatname');
$float_text .= $self->convert_tree({'contents'
=> $float->{'extra'}->{'node_content'}});
$float_text .= $self->ixin_close_element('floatname');
} else {
$float_text .= $self->ixin_none_element('floatname');
}
if ($float->{'extra'}->{'shortcaption'}) {
$float_text .= $self->convert_tree($float->{'extra'}->{'shortcaption'});
} elsif ($float->{'extra'}->{'caption'}) {
$float_text .= $self->convert_tree($float->{'extra'}->{'caption'});
} else {
$float_text .= $self->ixin_none_element('caption');
}
$float_text .= $self->ixin_close_element('floatentry')."\n";
}
$float_text = $self->ixin_open_element('floatset', ['count', $float_nr])
. $float_text .$self->ixin_close_element('floatset')."\n";
$float_text_len = $self->_count_bytes($float_text);
$floats_text .= $float_text;
# determine type expandable string from first float if it was not
# already determined from listoffloats
if (!defined($floats_information{$type}->{'type'})) {
my $command = $self->{'floats'}->{$type}->[0];
if ($command->{'extra'}->{'type'}
and $command->{'extra'}->{'type'}->{'content'}) {
$floats_information{$type}->{'type'}
= $self->convert_tree({'contents'
=> $command->{'extra'}->{'type'}->{'content'}});
}
}
}
my @attribute = ('type', $type, 'floatentrylen', $float_text_len);
$floats_index .= $self->ixin_open_element('floatindex', \@attribute);
if ($floats_information{$type}->{'type'}) {
$floats_index .= $self->ixin_open_element('floatindextype');
$floats_index .= $floats_information{$type}->{'type'};
$floats_index .= $self->ixin_close_element('floatindextype');
} else {
$floats_index .= $self->ixin_none_element('floatindextype');
}
if ($floats_information{$type}->{'node_id'}) {
foreach my $associated_node_id (@{$floats_information{$type}->{'node_id'}}) {
$floats_index .= ' ';
$floats_index .= $self->ixin_list_element('floatindexnode',
['nodeid', $associated_node_id]);
}
}
$floats_index .= $self->ixin_close_element('floatindex');
}
$floats_text .= $self->ixin_close_element('floatsset')."\n";
if ($floats_index ne '') {
$floats_index = $self->ixin_open_element('floatsindex', ['floatsindexlen',
$self->_count_bytes($floats_text)])
.$floats_index .$self->ixin_close_element('floatsindex');
} else {
$floats_index = $self->ixin_none_element('floatsindex');
}
# do blobs
my $blobs = '';
my $blobs_index = '';
my $blob_nr = 0;
if ($self->{'extra'}->{'image'}) {
foreach my $command (@{$self->{'extra'}->{'image'}}) {
my @extension;
my $basefile;
my $extension;
if (defined($command->{'extra'}->{'brace_command_contents'}->[0])) {
$basefile = Texinfo::Convert::Text::convert(
{'contents' => $command->{'extra'}->{'brace_command_contents'}->[0]},
{'code' => 1, Texinfo::Common::_convert_text_options($self)});
}
if (defined($command->{'extra'}->{'brace_command_contents'}->[4])) {
$extension = Texinfo::Convert::Text::convert(
{'contents' => $command->{'extra'}->{'brace_command_contents'}->[0]},
{'code' => 1, Texinfo::Common::_convert_text_options($self)});
$extension =~ s/^\.//;
@extension = ($extension);
}
foreach my $extension (@extension, @image_files_extensions) {
my $filename = $basefile.'.'.$extension;
my $file = $self->Texinfo::Common::locate_include_file($filename);
if (defined($file)) {
my $filehandle = do { local *FH };
if (open ($filehandle, $file)) {
$blob_nr++;
if ($extension eq 'txt') {
binmode($filehandle, ":encoding("
.$self->get_conf('INPUT_PERL_ENCODING').")")
if (defined($self->get_conf('INPUT_PERL_ENCODING')));
}
my $file_content;
if (-z $file) {
$file_content = '';
} else {
$file_content = <$filehandle>;
}
my $encoded_file = encode_base64($file_content);
$blobs .= $encoded_file;
my $blob_len = $self->_count_bytes($encoded_file);
my $mime_type;
if ($extension_mime_mapping{lc($extension)}) {
$mime_type = $extension_mime_mapping{lc($extension)};
} else {
$mime_type = $extension_mime_mapping{''};
}
$blobs_index .= $self->ixin_element('blobentry',
['bloblen', $blob_len, 'encoding', 'base64',
'mimetype', $mime_type, 'filename', $filename]) ."\n";
}
}
}
#print STDERR "$basefile\n";
}
}
$blobs_index = $self->ixin_open_element('blobsindex',
['count', $blob_nr])
.$blobs_index.$self->ixin_close_element('blobsindex')."\n";
my @counts_attributes = ('nodeindexlen', $self->_count_bytes($nodes_index),
'nodecounts', $node_nr,
'sectioningtreelen', $self->_count_bytes($sectioning_tree),
'labelslen', $self->_count_bytes($labels_text),
'blobsindexlen', $self->_count_bytes($blobs_index));
my $output = $self->_output_text($result, $fh);
my $counts_text = $self->ixin_open_element('counts', \@counts_attributes);
$counts_text .= $dts_index;
$counts_text .= $floats_index;
$counts_text .= $self->ixin_close_element('counts') . "\n";
$output .= $self->_output_text($counts_text, $fh);
$output .= $self->_output_text($nodes_index, $fh);
$output .= $self->_output_text($sectioning_tree, $fh);
$output .= $self->_output_text($labels_text, $fh);
$output .= $self->_output_text($dts_text, $fh);
$output .= $self->_output_text($floats_text, $fh);
$output .= $self->_output_text($blobs_index, $fh);
$output .= $self->_output_text($document_output, $fh);
$output .= $self->_output_text($blobs, $fh);
if ($fh and $self->{'output_file'} ne '-') {
$self->register_close_file($self->{'output_file'});
if (!close ($fh)) {
$self->document_error(sprintf($self->__("error on closing %s: %s"),
$self->{'output_file'}, $!));
}
}
return $output;
}
1;