| #! /usr/bin/env perl |
| # pod2texi -- convert Pod to Texinfo. |
| # Copyright 2012-2025 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> |
| |
| use 5.006; |
| |
| use strict; |
| |
| use warnings; |
| |
| use Getopt::Long qw(GetOptions); |
| # for fileparse. |
| use File::Basename; |
| use File::Spec; |
| |
| #use Pod::Simple::SimpleTree; |
| #use Data::Dumper; |
| |
| Getopt::Long::Configure("gnu_getopt"); |
| #use Pod::Simple::Debug (4); |
| |
| BEGIN |
| { |
| my ($real_command_name, $command_directory, $command_suffix) |
| = fileparse($0, '.pl'); |
| my $updir = File::Spec->updir(); |
| |
| my $datadir = '@datadir@'; |
| my $converter = '@CONVERTER@'; |
| my $libdir = '@libdir@'; |
| my $xsdir; |
| |
| my $in_source = 0; |
| # set if we detect that the command comes from a Perl build with Makefile.PL. |
| # In that case we assume that the modules should be found in directories |
| # already registered in paths. |
| my $keep_paths = 0; |
| |
| if (defined($ENV{'TEXINFO_DEV_SOURCE'}) |
| and $ENV{'TEXINFO_DEV_SOURCE'} ne '0') { |
| $in_source = 1; |
| } elsif ($datadir eq '@' .'datadir@') { |
| if ($0 =~ /\.pl$/) { |
| $in_source = 1; |
| } else { |
| $keep_paths = 1; |
| } |
| } |
| # in-source run |
| if ($in_source) { |
| my $t2a_builddir; |
| if (defined($ENV{'t2a_builddir'})) { |
| $t2a_builddir = $ENV{'t2a_builddir'}; |
| } else { |
| if (defined($ENV{'top_builddir'})) { |
| $t2a_builddir = join('/', ($ENV{'top_builddir'}, 'tta')); |
| } else { |
| $t2a_builddir = join('/', ($command_directory, $updir, 'tta')); |
| } |
| } |
| |
| # To find Texinfo::ModulePath |
| unshift @INC, join('/', ($t2a_builddir, 'perl')); |
| |
| my $t2a_srcdir; |
| if (defined($ENV{'t2a_srcdir'})) { |
| $t2a_srcdir = $ENV{'t2a_srcdir'}; |
| } else { |
| if (defined($ENV{'top_srcdir'})) { |
| $t2a_srcdir = join('/', ($ENV{'top_srcdir'}, 'tta')); |
| } else { |
| $t2a_srcdir = join('/', ($command_directory, $updir, 'tta')); |
| } |
| $ENV{'t2a_srcdir'} = $t2a_srcdir; |
| } |
| |
| require Texinfo::ModulePath; |
| # NOTE updirs argument cannot point to the right place, as it is only |
| # valid in texi2any directory. t2a_srcdir and t2a_builddir should be used. |
| Texinfo::ModulePath::init(undef, undef, undef); |
| |
| # To find Pod::Simple::Texinfo |
| if (defined($ENV{'top_srcdir'})) { |
| unshift @INC, join('/', ($ENV{'top_srcdir'}, |
| 'Pod-Simple-Texinfo', 'lib')); |
| } else { |
| unshift @INC, join('/', ($command_directory, 'lib')); |
| } |
| } elsif (!$keep_paths) { |
| # Look for modules in their installed locations. |
| my $modules_dir = join('/', ($datadir, $converter)); |
| # look for package data in the installed location. |
| my $modules_converterdatadir = $modules_dir; |
| |
| # try to make package relocatable, will only work if |
| # standard relative paths are used |
| if (! -f join('/', ($modules_dir, 'Texinfo', 'Parser.pm')) |
| and -f join('/', ($command_directory, $updir, 'share', |
| $converter, 'Texinfo', 'Parser.pm'))) { |
| $modules_dir = join('/', ($command_directory, $updir, |
| 'share', $converter)); |
| $modules_converterdatadir = join('/', ($command_directory, $updir, |
| 'share', $converter)); |
| $xsdir = join('/', ($command_directory, $updir, 'lib', $converter)); |
| } |
| |
| unshift @INC, $modules_dir; |
| require Texinfo::ModulePath; |
| Texinfo::ModulePath::init($modules_dir, $xsdir, $modules_converterdatadir, |
| 'installed' => 1); |
| |
| # To find Pod::Simple::Texinfo |
| unshift @INC, join('/', ($modules_converterdatadir, 'Pod-Simple-Texinfo')); |
| } |
| } |
| |
| use Pod::Simple::Texinfo; |
| use Texinfo::CommandsValues; |
| use Texinfo::Common; |
| use Texinfo::Parser; |
| use Texinfo::Convert::Texinfo; |
| use Texinfo::Convert::NodeNameNormalization; |
| use Texinfo::Document; |
| use Texinfo::Structuring; |
| use Texinfo::Transformations; |
| |
| { |
| # A fake package to be able to use Pod::Simple::PullParser without generating |
| # any output. |
| package Pod::Simple::PullParserRun; |
| |
| our @ISA = qw(Pod::Simple::PullParser); |
| sub new |
| { |
| return shift->SUPER::new(@_); |
| } |
| sub run(){}; |
| } |
| |
| my ($real_command_name, $directories, $suffix) = fileparse($0); |
| |
| sub pod2texi_help() |
| { |
| my $pod2texi_help = __("Usage: pod2texi [OPTION]... POD..."); |
| $pod2texi_help .= "\n\n"; |
| $pod2texi_help .= __("Translate Perl Pod documentation file(s) to Texinfo. There are two |
| basic modes of operation. First, by default, each Pod is translated to |
| a standalone Texinfo manual. |
| |
| Second, if --base-level is set higher than 0, each Pod is translated |
| to a file suitable for \@include, and one more file with a main menu |
| and all the \@include is generated."); |
| $pod2texi_help .= "\n\n"; |
| $pod2texi_help .= __("Options: |
| --appendix-sections use appendix-like sections")."\n"; |
| $pod2texi_help .= __(" --base-level=NUM|NAME level of the head1 commands; default 0")."\n"; |
| $pod2texi_help .= __(" --debug=NUM set debugging level")."\n"; |
| $pod2texi_help .= __(" --generate-setfilename generate \@setfilename for standalone |
| manuals")."\n"; |
| $pod2texi_help .= __(" --headings-as-sections no structuring command for sections")."\n"; |
| $pod2texi_help .= __(" --help display this help and exit")."\n"; |
| $pod2texi_help .= __(" --no-fill-section-gaps do not fill sectioning gaps")."\n"; |
| $pod2texi_help .= __(" --no-section-nodes use anchors for sections instead of nodes")."\n"; |
| $pod2texi_help .= __(" --menus generate node menus")."\n"; |
| $pod2texi_help .= __(" --outdir=NAME output included files in NAME; |
| defaults to --subdir")."\n"; |
| $pod2texi_help .= __(" --output=NAME output to NAME for the first or main manual |
| instead of standard output")."\n"; |
| $pod2texi_help .= __(" --preamble=STR insert STR as beginning boilerplate; |
| defaults to a minimal Texinfo document beginning")."\n"; |
| $pod2texi_help .= __(" --setfilename \@setfilename for the main manual")."\n"; |
| $pod2texi_help .= __(" --subdir=NAME include files from NAME in the main manual")."\n"; |
| $pod2texi_help .= __(" --top top for the main manual")."\n"; |
| $pod2texi_help .= __(" --unnumbered-sections do not number sections")."\n"; |
| $pod2texi_help .= __(" --version display version information and exit"); |
| $pod2texi_help .= "\n\n"; |
| |
| $pod2texi_help .= __("Email bug reports to bug-texinfo\@gnu.org, |
| general questions and discussion to help-texinfo\@gnu.org. |
| Texinfo home page: https://www.gnu.org/software/texinfo/")."\n"; |
| return $pod2texi_help; |
| } |
| |
| my $base_level = 0; |
| my $unnumbered_sections = 0; |
| my $appendix_sections = 0; |
| my $headings_as_sections = 0; |
| my $generate_node_menus = 0; |
| # TODO change to 0 when the time has come. |
| my $generate_setfilename = 1; |
| my $outdir; |
| my $output = '-'; |
| my $top = 'top'; |
| my $setfilename = undef; |
| my $preamble = undef; |
| my $subdir; |
| my $section_nodes = 1; |
| my $fill_sectioning_gaps = 1; |
| my $debug = 0; |
| |
| my $result_options = Getopt::Long::GetOptions ( |
| 'help|h' => sub { print pod2texi_help(); exit 0; }, |
| 'version|V' => sub {print "$real_command_name $Pod::Simple::Texinfo::VERSION\n\n"; |
| printf __("Copyright (C) %s Free Software Foundation, Inc. |
| License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html> |
| This is free software: you are free to change and redistribute it. |
| There is NO WARRANTY, to the extent permitted by law.")."\n", "2024"; |
| exit 0;}, |
| 'base-level=s' => sub { |
| if ($_[1] =~ /^[0-4]$/) { |
| $base_level = $_[1]; |
| } elsif (defined($Texinfo::CommandsValues::command_structuring_level{$_[1]})) { |
| $base_level = $Texinfo::CommandsValues::command_structuring_level{$_[1]}; |
| } else { |
| die sprintf(__("%s: wrong argument for --base-level")."\n", |
| $real_command_name); |
| } |
| }, |
| 'appendix-sections!' => \$appendix_sections, |
| 'fill-section-gaps!' => \$fill_sectioning_gaps, |
| 'generate-setfilename!' => \$generate_setfilename, |
| 'headings-as-sections!' => \$headings_as_sections, |
| 'menus!' => \$generate_node_menus, |
| 'outdir=s' => \$outdir, |
| 'output|o=s' => \$output, |
| 'preamble=s' => \$preamble, |
| 'setfilename=s' => \$setfilename, |
| 'subdir=s' => \$subdir, |
| 'top=s' => \$top, |
| 'section-nodes!' => \$section_nodes, |
| 'unnumbered-sections!' => \$unnumbered_sections, |
| 'debug=i' => \$debug, |
| ); |
| |
| exit 1 if (!$result_options); |
| |
| if (!defined($outdir) and defined($subdir)) { |
| $outdir = $subdir; |
| } |
| |
| if (defined($outdir)) { |
| if (! -d $outdir) { |
| if (!mkdir($outdir)) { |
| die sprintf(__("%s: could not create directory %s: %s"), |
| $real_command_name, $outdir, $!); |
| } |
| } |
| } |
| |
| my $STDOUT_DOCU_NAME = 'stdout'; |
| |
| my @manuals; |
| |
| my @input_files = @ARGV; |
| |
| # use STDIN if not a tty, like makeinfo does |
| @input_files = ('-') if (!scalar(@input_files) and !-t STDIN); |
| die sprintf(__("%s: missing file argument")."\n", $real_command_name) |
| .sprintf(__("Try `%s --help' for more information.")."\n", $real_command_name) |
| unless (scalar(@input_files) >= 1); |
| |
| my %file_manual_title; |
| my %manual_name_file; |
| my %file_manual_name; |
| # First gather all the manual names |
| if ($base_level > 0) { |
| foreach my $file (@input_files) { |
| my $manual_name; |
| # we don't want to read from STDIN, as the input read would be lost |
| # same with named pipe and socket... |
| # TODO are there other file types that have the same problem? |
| if ($file eq '-' or -p $file or -S $file) { |
| # do not read file |
| } else { |
| # not really used, only the manual name is used. |
| my $parser = Pod::Simple::PullParserRun->new(); |
| $parser->parse_file($file); |
| my $short_title = $parser->get_short_title(); |
| if (defined($short_title) and $short_title =~ m/\S/) { |
| $file_manual_title{$file} = $short_title; |
| push @manuals, $short_title; |
| $manual_name = $short_title; |
| #print STDERR "NEW MANUAL: $manual_name\n"; |
| } else { |
| if (!$parser->content_seen) { |
| warn sprintf(__("%s: warning: %s without content")."\n", |
| $real_command_name, $file); |
| } |
| } |
| } |
| if (!defined($manual_name)) { |
| if ($file eq '-') { |
| $manual_name = $STDOUT_DOCU_NAME; |
| } else { |
| my ($file_name, $dir, $suffix) = fileparse($file, ('.pm', '.pod')); |
| $manual_name = $file_name; |
| } |
| } |
| if (exists($manual_name_file{$manual_name}) |
| and $manual_name_file{$manual_name} ne $file) { |
| warn sprintf(__("%s: same manual name `%s' for different files: %s and %s"), |
| $real_command_name, $manual_name, $file, |
| $manual_name_file{$manual_name})."\n"; |
| } else { |
| $manual_name_file{$manual_name} = $file; |
| } |
| $file_manual_name{$file} = $manual_name; |
| } |
| } |
| |
| # return a parser and parsed tree |
| sub _parsed_manual_tree($$$$$) |
| { |
| my $self = shift; |
| my $manual_texi = shift; |
| my $section_nodes = shift; |
| my $fill_gaps_in_sectioning = shift; |
| my $do_node_menus = shift; |
| |
| my $parser_options = {}; |
| if ($debug > 3) { |
| $parser_options->{'DEBUG'} = $debug - 3; |
| } |
| my $texi_parser = Texinfo::Parser::parser($parser_options); |
| my $document = $texi_parser->parse_texi_text($manual_texi); |
| |
| if ($debug > 1) { |
| my ($error_messages, $error_count) = $document->parser_errors(); |
| Pod::Simple::Texinfo::print_texinfo_errors($error_messages, |
| $error_count, '_parsed_manual_tree'); |
| } |
| |
| my $identifier_target = $document->labels_information(); |
| |
| # NOTE the document customization information is not initialized |
| # if debug is not high. |
| # The functions called on the document below, and elsewhere in |
| # the code call get_conf for structuring information and menu |
| # generation on the document. |
| # TODO always call register_document_options(), maybe with an empty hash? |
| if ($debug > 3) { |
| # there is not much debug output in structuring code used with the |
| # document, but it could still be interesting to debug. |
| $document->register_document_options({'DEBUG' => $debug - 3}); |
| } |
| |
| if ($fill_gaps_in_sectioning) { |
| my $commands_heading_content; |
| if ($self->texinfo_sectioning_base_level() > 0) { |
| my $manual_title_texi = Pod::Simple::Texinfo::protect_text( |
| $self->texinfo_short_title(), 1, 1); |
| my $parser = Texinfo::Parser::parser(); |
| $commands_heading_content = $parser->parse_texi_line($manual_title_texi); |
| } |
| Texinfo::Transformations::fill_gaps_in_sectioning_in_document($document, |
| $commands_heading_content); |
| if ($section_nodes) { |
| Texinfo::Transformations::insert_nodes_for_sectioning_commands( |
| $document); |
| } |
| } |
| Texinfo::Structuring::sectioning_structure($document); |
| $document->internal_references_information(); |
| # this is needed to set 'normalized' for menu entries, they are |
| # used in complete_tree_nodes_menus_in_document. |
| Texinfo::Structuring::associate_internal_references($document); |
| Texinfo::Transformations::complete_tree_nodes_menus_in_document($document) |
| if ($section_nodes and $do_node_menus); |
| |
| if ($debug > 1) { |
| my ($error_messages, $error_count) = $document->errors(); |
| Pod::Simple::Texinfo::print_texinfo_errors($error_messages, |
| $error_count, '_parsed_manual_tree document'); |
| } |
| |
| return ($texi_parser, $document, $identifier_target); |
| } |
| |
| sub _fix_texinfo_tree($$$$;$$) |
| { |
| my $self = shift; |
| my $manual_texi = shift; |
| my $section_nodes = shift; |
| my $fill_gaps_in_sectioning = shift; |
| my $do_node_menus = shift; |
| my $do_master_menu = shift; |
| |
| my ($texi_parser, $document, $updated_labels) |
| = _parsed_manual_tree($self, $manual_texi, $section_nodes, |
| $fill_gaps_in_sectioning, |
| $do_node_menus); |
| if ($do_master_menu) { |
| if ($do_node_menus) { |
| # It could be possible to show document errors if debug > 1 |
| # for example, but there should not be any error emitted, |
| # except maybe for errors in translations code, which are very |
| # unlikely. |
| Texinfo::Transformations::regenerate_master_menu($document, |
| $texi_parser); |
| } else { |
| # note that that situation cannot happen with the code as it |
| # is now. When _fix_texinfo_tree is called from _do_top_node_menu |
| # both $do_master_menu and $do_node_menus are set. |
| # _fix_texinfo_tree can also be called from _fix_texinfo_manual, but |
| # _fix_texinfo_manual is never called with a $do_master_menu argument, |
| # so when _fix_texinfo_tree is called from _fix_texinfo_manual, |
| # $do_master_menu cannot be set. |
| |
| # setup another tree with menus to do the master menu as menus are |
| # not done for the main tree |
| my ($texi_parser_menus, $document_menus, $updated_labels_menus) |
| = _parsed_manual_tree($self, $manual_texi, $section_nodes, |
| $fill_gaps_in_sectioning, 1); |
| my $top_node_menus = $updated_labels_menus->{'Top'}; |
| if ($top_node_menus) { |
| my $menus_nodes_list = $document_menus->nodes_list(); |
| my $top_node_menus_relations |
| = $menus_nodes_list->[$top_node_menus->{'extra'}->{'node_number'} -1]; |
| if ($top_node_menus_relations->{'menus'} |
| and scalar(@{$top_node_menus_relations->{'menus'}})) { |
| my $top_node_menus_menu = $top_node_menus_relations->{'menus'}->[0]; |
| my $top_node = $updated_labels->{'Top'}; |
| $top_node_menus_menu->{'parent'} = $top_node; |
| push @{$top_node->{'contents'}}, $top_node_menus_menu; |
| my $nodes_list = $document->nodes_list(); |
| my $top_node_relations |
| = $nodes_list->[$top_node->{'extra'}->{'node_number'} -1]; |
| push @{$top_node_relations->{'menus'}}, $top_node_menus_menu; |
| } |
| } |
| } |
| } |
| return ($texi_parser, $document); |
| } |
| |
| sub _fix_texinfo_manual($$$$;$$) |
| { |
| my $self = shift; |
| my $manual_texi = shift; |
| my $section_nodes = shift; |
| my $fill_gaps_in_sectioning = shift; |
| my $do_node_menus = shift; |
| my $do_master_menu = shift; |
| |
| my ($texi_parser, $document) |
| = _fix_texinfo_tree($self, $manual_texi, $section_nodes, |
| $fill_gaps_in_sectioning, $do_node_menus, |
| $do_master_menu); |
| my $tree = $document->tree(); |
| return Texinfo::Convert::Texinfo::convert_to_texinfo($tree); |
| } |
| |
| sub _do_top_node_menu($) |
| { |
| my $manual_texi = shift; |
| |
| my ($texi_parser, $document) |
| = _fix_texinfo_tree(undef, $manual_texi, 1, 0, 1, 1); |
| my $identifier_target = $document->labels_information(); |
| my $top_node = $identifier_target->{'Top'}; |
| my $nodes_list = $document->nodes_list(); |
| my $top_node_relations |
| = $nodes_list->[$top_node->{'extra'}->{'node_number'} -1]; |
| if ($top_node_relations->{'menus'}) { |
| my $top_node_menu = $top_node_relations->{'menus'}->[0]; |
| if ($top_node_menu) { |
| return Texinfo::Convert::Texinfo::convert_to_texinfo($top_node_menu); |
| } |
| } |
| return ''; |
| } |
| |
| my $file_nr = -1; |
| # Full manual is collected to generate the top node menu, if $section_nodes |
| my $full_manual = ''; |
| my @included; |
| foreach my $file (@input_files) { |
| $file_nr++; |
| my $manual_texi = ''; |
| my $outfile; |
| my $incfile; |
| my $outfile_name; |
| my $manual_name; |
| my $manual_title; |
| $manual_title = $file_manual_title{$file} |
| if (defined($file_manual_title{$file})); |
| if ($base_level == 0 and !$file_nr) { |
| $outfile = $output; |
| } else { |
| $manual_name = $file_manual_name{$file}; |
| if (defined($manual_title)) { |
| $outfile_name = Pod::Simple::Texinfo::pod_title_to_file_name($manual_title); |
| } else { |
| $outfile_name = $manual_name; |
| } |
| $outfile_name .= '.texi'; |
| if (defined($outdir)) { |
| $outfile = join('/', ($outdir, $outfile_name)); |
| } else { |
| $outfile = $outfile_name; |
| } |
| if (defined($subdir)) { |
| # use / in generated Texinfo code to be as portable as possible |
| $incfile = "$subdir/$outfile_name"; |
| } else { |
| $incfile = $outfile_name; |
| } |
| } |
| |
| #my $pod_simple_tree = Pod::Simple::SimpleTree->new->parse_file($file)->root; |
| #print STDERR Data::Dumper->Dump([$pod_simple_tree])."\n"; |
| |
| my $new = Pod::Simple::Texinfo->new(); |
| |
| push @included, [$manual_name, $outfile, $incfile, $file] |
| if ($base_level > 0); |
| my $fh; |
| if ($outfile eq '-') { |
| $fh = *STDOUT; |
| } else { |
| open(OUT, ">$outfile") |
| or die sprintf(__("%s: could not open %s for writing: %s")."\n", |
| $real_command_name, $outfile, $!); |
| $fh = *OUT; |
| } |
| # The Texinfo output from Pod::Simple::Texinfo does not contain |
| # @documentencoding. We output UTF-8 as it is consistent with no |
| # @documentencoding, and it also because is the best choice or encoding. |
| # The =encoding information is not available anyway, but even if it |
| # was it would still be better to output UTF-8. |
| binmode($fh, ':encoding(utf-8)'); |
| |
| # this sets the string that $parser's output will be sent to |
| $new->output_string(\$manual_texi); |
| |
| $new->texinfo_generate_setfilename($generate_setfilename); |
| |
| $new->texinfo_sectioning_base_level($base_level); |
| if ($section_nodes) { |
| $new->texinfo_section_nodes(1); |
| } |
| if ($unnumbered_sections) { |
| $new->texinfo_sectioning_style('unnumbered'); |
| } elsif ($appendix_sections) { |
| $new->texinfo_sectioning_style('appendix'); |
| } elsif ($headings_as_sections) { |
| $new->texinfo_sectioning_style('heading'); |
| } |
| if ($base_level > 0 and @manuals) { |
| # names without formatting from Pod::Simple::PullParser->get_short_title |
| $new->texinfo_internal_pod_manuals(\@manuals); |
| } |
| |
| if ($debug) { |
| $new->texinfo_debug($debug); |
| if ($base_level > 0) { |
| print STDERR "processing $file -> $outfile ($base_level, $manual_name)\n"; |
| } else { |
| print STDERR "processing $file -> $outfile\n"; |
| } |
| } |
| $new->parse_file($file); |
| |
| if ($section_nodes or $fill_sectioning_gaps) { |
| if ($debug > 4) { |
| # print the manual obtained before fixing the Texinfo code to a file |
| open(DBGFILE, ">$outfile-dbg") |
| or die sprintf(__("%s: could not open %s: %s")."\n", |
| $real_command_name, "$outfile-dbg", $!); |
| binmode(DBGFILE, ':encoding(utf-8)'); |
| print DBGFILE $manual_texi; |
| } |
| $manual_texi = _fix_texinfo_manual($new, $manual_texi, $section_nodes, |
| $fill_sectioning_gaps, |
| $generate_node_menus); |
| $full_manual .= $manual_texi if ($section_nodes); |
| } |
| print $fh $manual_texi; |
| |
| if ($outfile ne '-') { |
| close($fh) or die sprintf(__("%s: error on closing %s: %s")."\n", |
| $real_command_name, $outfile, $!); |
| } |
| |
| if ($base_level > 0) { |
| if (!$new->content_seen) { |
| warn sprintf(__("%s: removing %s as input file %s has no content")."\n", |
| $real_command_name, $outfile, $file); |
| unlink ($outfile); |
| pop @included; |
| # if we didn't gather the short title, try now, and rename out file if found |
| } elsif (!defined($manual_title)) { |
| my $short_title = $new->texinfo_short_title; |
| if (defined($short_title) and $short_title =~ /\S/) { |
| push @manuals, $short_title; |
| pop @included; |
| my $new_outfile_name |
| = Pod::Simple::Texinfo::pod_title_to_file_name($short_title); |
| $new_outfile_name .= '.texi'; |
| my $new_outfile; |
| if (defined($outdir)) { |
| $new_outfile = join('/', ($outdir, $new_outfile_name)); |
| } else { |
| $new_outfile = $new_outfile_name; |
| } |
| my $new_incfile; |
| if (defined($subdir)) { |
| # use / in generated Texinfo code to be as portable as possible |
| $new_incfile = "$subdir/$new_outfile_name"; |
| } else { |
| $new_incfile = $new_outfile_name; |
| } |
| |
| if ($new_outfile ne $outfile) { |
| unless (rename ($outfile, $new_outfile)) { |
| die sprintf(__("%s: rename %s failed: %s")."\n", |
| $real_command_name, $outfile, $!); |
| } |
| } |
| push @included, [$short_title, $new_outfile, $new_incfile, $file]; |
| } |
| } |
| } |
| } |
| |
| if ($base_level > 0) { |
| my $fh; |
| if ($output ne '-') { |
| open(OUT, ">$output") |
| or die sprintf(__("%s: could not open %s for writing: %s")."\n", |
| $real_command_name, $output, $!); |
| $fh = *OUT; |
| } else { |
| $fh = *STDOUT; |
| } |
| |
| # We output UTF-8 as it is default for Texinfo and is consistent with no |
| # @documentencoding, and it also because is the best choice for encoding. |
| binmode($fh, ':encoding(utf-8)'); |
| |
| my $setfilename_string = ''; |
| if (defined($setfilename)) { |
| $setfilename_string = '@setfilename ' |
| . Pod::Simple::Texinfo::protect_text($setfilename)."\n"; |
| } |
| |
| my $preamble_result; |
| |
| if (! defined ($preamble)) { |
| $preamble_result = '\input texinfo |
| ' . $setfilename_string |
| . "\@settitle $top |
| \@shorttitlepage $top |
| \@headings on |
| |
| \@contents |
| |
| \@node Top |
| \@top $top\n\n"; |
| } elsif ($preamble eq '-') { |
| $preamble_result = join("", <STDIN>); |
| } else { |
| $preamble_result = $preamble; |
| } |
| |
| print $fh $preamble_result; |
| if ($section_nodes) { |
| #print STDERR "\@node Top\n\@top top\n".$full_manual; |
| my $menu = _do_top_node_menu("\@node Top\n\@top top\n".$full_manual); |
| print $fh $menu."\n"; |
| } |
| foreach my $include (@included) { |
| my $incfile = $include->[2]; |
| print $fh "\@include ".Pod::Simple::Texinfo::protect_text($incfile)."\n"; |
| } |
| print $fh "\n\@bye\n"; |
| |
| if ($output ne '-') { |
| close($fh) or die sprintf(__("%s: error on closing %s: %s")."\n", |
| $real_command_name, $output, $!); |
| } |
| } |
| |
| if (defined($output) and $output eq '-') { |
| close(STDOUT) or die sprintf(__("%s: error on closing stdout: %s")."\n", |
| $real_command_name, $!); |
| } |
| |
| 1; |
| |
| __END__ |
| |
| =head1 NAME |
| |
| pod2texi - convert Pod to Texinfo |
| |
| =head1 SYNOPSIS |
| |
| pod2texi [OPTION]... POD... |
| |
| =head1 DESCRIPTION |
| |
| Translate Pod file(s) to Texinfo. There are two basic modes of |
| operation. First, by default, each Pod is translated to a standalone |
| Texinfo manual. |
| |
| Second, if C<--base-level> is set higher than 0, each Pod is translated |
| to a file suitable for C<@include>, and one more file with a main menu |
| and all the C<@include> is generated. |
| |
| =head1 OPTIONS |
| |
| =begin comment |
| |
| This style used for command line options is a style often seen in |
| Pods. Also often seen is simple =item, or a verbatim block with --help |
| output. More rarely =head2, very rare use of C<>. Use of C<> would |
| have been more in line with Texinfo @option. |
| |
| =end comment |
| |
| =over |
| |
| =item B<--appendix-sections> |
| |
| Use appendix sectioning commands (C<@appendix>, ...) instead of the |
| default numbered sectioning Texinfo @-commands (C<@chapter>, |
| C<@section>, ...). |
| |
| =item B<--base-level>=I<NUM|NAME> |
| |
| Sets the level of the C<head1> commands. It may be an integer or a |
| Texinfo sectioning command (without the C<@>): 1 corresponds to the |
| C<@chapter>/C<@unnumbered> level, 2 to the C<@section> level, and so on. |
| The default is 0, meaning that C<head1> commands are still output as |
| chapters, but the output is arranged as a standalone manual. |
| |
| If the level is not 0, the Pod file is rendered as a fragment of a |
| Texinfo manual suitable for C<@include>. In this case, each Pod file |
| has an additional sectioning command covering the entire file, one level |
| above the C<--base-level> value. Therefore, to make each Pod file a |
| chapter in a large manual, you should use C<section> as the base level. |
| |
| For an example of making Texinfo out of the Perl documentation itself, |
| see C<contrib/perldoc-all> in the Texinfo source distribution. |
| |
| =begin comment |
| |
| with output available at L<http://www.gnu.org/software/perl/manual>. |
| |
| =end comment |
| |
| =item B<--debug>=I<NUM> |
| |
| Set debugging level to I<NUM>. |
| |
| =item B<--generate-setfilename> |
| |
| Generate a C<@setfilename> line for standalone manuals. Can be negated |
| with C<--no-generate-setfilename>. Ignored if C<--base-level> is not 0. |
| |
| =item B<--headings-as-sections> |
| |
| Use headings commands (C<@heading>, ...) instead of the |
| default numbered sectioning Texinfo @-commands (C<@chapter>, |
| C<@section>, ...). The sectioning command covering the entire |
| file output for each Pod file if B<--base-level> is not 0 is a |
| numbered command. |
| |
| =item B<--help> |
| |
| Display help and exit. |
| |
| =item B<--menus> |
| |
| Output node menus. If there is a main manual, its Top node menu |
| is always output, since a master menu is generated. Other nodes |
| menus are not output in the default case. |
| |
| =item B<--outdir>=I<NAME> |
| |
| If there is a main manual with include files (each corresponding to |
| an input Pod file), then the generated Texinfo files are put in |
| directory I<NAME>. Default is based on C<--subdir>. |
| |
| =item B<--output>=I<NAME> |
| |
| Name for the first manual, or the main manual if there is a main manual. |
| Default is to write to standard output. |
| |
| =item B<--no-section-nodes> |
| |
| Use anchors for sections instead of nodes. |
| |
| =item B<--no-fill-section-gaps> |
| |
| Do not fill sectioning gaps with empty C<@unnumbered> files. |
| Ordinarily, it's good to keep the sectioning hierarchy intact. |
| |
| =item B<--preamble>=I<STR> |
| |
| Insert I<STR> as top boilerplate before menu and includes. If I<STR> is |
| set to C<->, read the top boilerplate from the standard input. The default top |
| boilerplate is a minimal beginning for a Texinfo document. |
| |
| =item B<--setfilename>=I<STR> |
| |
| Use I<STR> in top boilerplate before menu and includes for C<@setfilename> |
| for the main manual, if C<--base-level> is not set to 0. Ignored if |
| C<--base-level> is 0. No C<@setfilename> is output in the default case |
| for the main manual. |
| |
| =item B<--subdir>=I<NAME> |
| |
| If there is a main manual with include files (each corresponding to |
| an input Pod file), then those include files are included from I<NAME>. |
| |
| If C<--outdir> is set, I<NAME> should in general be set to the relative |
| directory between the main manual and C<--outdir> argument. |
| |
| =item B<--unnumbered-sections> |
| |
| Use unnumbered sectioning commands (C<@unnumbered>, ...) instead of the |
| default numbered sectioning Texinfo @-commands (C<@chapter>, |
| C<@section>, ...). |
| |
| =item B<--top>=I<TOP> |
| |
| Name of the C<@top> element for the main manual. May contain Texinfo code. |
| |
| =item B<--version> |
| |
| Display version information and exit. |
| |
| =back |
| |
| =head1 SEE ALSO |
| |
| L<Pod::Simple::Texinfo>. L<perlpod>. The Texinfo manual. |
| Texinfo home page: L<https://www.gnu.org/software/texinfo/> |
| |
| =head1 COPYRIGHT AND LICENSE |
| |
| Copyright 2012-2025 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. |
| |
| There is NO WARRANTY, to the extent permitted by law. |
| |
| =head1 AUTHOR |
| |
| Patrice Dumas E<lt>bug-texinfo@gnu.orgE<gt>. |
| |
| =cut |