.\" -*- mode: troff; coding: utf-8 -*- .\" Automatically generated by Pod::Man 5.01 (Pod::Simple 3.43) .\" .\" Standard preamble: .\" ======================================================================== .de Sp \" Vertical space (when we can't use .PP) .if t .sp .5v .if n .sp .. .de Vb \" Begin verbatim text .ft CW .nf .ne \\$1 .. .de Ve \" End verbatim text .ft R .fi .. .\" \*(C` and \*(C' are quotes in nroff, nothing in troff, for use with C<>. .ie n \{\ . ds C` "" . ds C' "" 'br\} .el\{\ . ds C` . ds C' 'br\} .\" .\" Escape single quotes in literal strings from groff's Unicode transform. .ie \n(.g .ds Aq \(aq .el .ds Aq ' .\" .\" If the F register is >0, we'll generate index entries on stderr for .\" titles (.TH), headers (.SH), subsections (.SS), items (.Ip), and index .\" entries marked with X<> in POD. Of course, you'll have to process the .\" output yourself in some meaningful fashion. .\" .\" Avoid warning from groff about undefined register 'F'. .de IX .. .nr rF 0 .if \n(.g .if rF .nr rF 1 .if (\n(rF:(\n(.g==0)) \{\ . if \nF \{\ . de IX . tm Index:\\$1\t\\n%\t"\\$2" .. . if !\nF==2 \{\ . nr % 0 . nr F 2 . \} . \} .\} .rr rF .\" ======================================================================== .\" .IX Title "LOCALE::PO4A::TRANSTRACTOR.3PM 1" .TH LOCALE::PO4A::TRANSTRACTOR.3PM 1 2024-06-26 "perl v5.38.2" "User Contributed Perl Documentation" .\" For nroff, turn off justification. Always turn off hyphenation; it makes .\" way too many mistakes in technical documents. .if n .ad l .nh .SH NAME Locale::Po4a::TransTractor \- generic trans(lator ex)tractor. .SH DESCRIPTION .IX Header "DESCRIPTION" The po4a (PO for anything) project goal is to ease translations (and more interestingly, the maintenance of translations) using gettext tools on areas where they were not expected like documentation. .PP This class is the ancestor of every po4a parser used to parse a document, to search translatable strings, to extract them to a PO file and to replace them by their translation in the output document. .PP More formally, it takes the following arguments as input: .IP \- 2 a document to translate; .IP \- 2 a PO file containing the translations to use. .PP As output, it produces: .IP \- 2 another PO file, resulting of the extraction of translatable strings from the input document; .IP \- 2 a translated document, with the same structure than the one in input, but with all translatable strings replaced with the translations found in the PO file provided in input. .PP Here is a graphical representation of this: .PP .Vb 6 \& Input document \-\-\e /\-\-\-> Output document \& \e / (translated) \& +\-> parse() function \-\-\-\-\-+ \& / \e \& Input PO \-\-\-\-\-\-\-\-/ \e\-\-\-> Output PO \& (extracted) .Ve .SH "FUNCTIONS YOUR PARSER SHOULD OVERRIDE" .IX Header "FUNCTIONS YOUR PARSER SHOULD OVERRIDE" .IP \fBparse()\fR 4 .IX Item "parse()" This is where all the work takes place: the parsing of input documents, the generation of output, and the extraction of the translatable strings. This is pretty simple using the provided functions presented in the section \&\fBINTERNAL FUNCTIONS\fR below. See also the \fBSYNOPSIS\fR, which presents an example. .Sp This function is called by the \fBprocess()\fR function below, but if you choose to use the \fBnew()\fR function, and to add content manually to your document, you will have to call this function yourself. .IP \fBdocheader()\fR 4 .IX Item "docheader()" This function returns the header we should add to the produced document, quoted properly to be a comment in the target language. See the section \&\fBEducating developers about translations\fR, from \fBpo4a\fR\|(7), for what it is good for. .SH SYNOPSIS .IX Header "SYNOPSIS" The following example parses a list of paragraphs beginning with "
". For the sake of simplicity, we assume that the document is well formatted, i.e. that '
' tags are the only tags present, and that this tag is at the very beginning of each paragraph. .PP .Vb 2 \& sub parse { \& my $self = shift; \& \& PARAGRAPH: while (1) { \& my ($paragraph,$pararef)=("",""); \& my $first=1; \& my ($line,$lref)=$self\->shiftline(); \& while (defined($line)) { \& if ($line =~ m/
/ && !$first\-\-; ) { \& # Not the first time we see
. \& # Reput the current line in input, \& # and put the built paragraph to output \& $self\->unshiftline($line,$lref); \& \& # Now that the document is formed, translate it: \& # \- Remove the leading tag \& $paragraph =~ s/^
//s; \& \& # \- push to output the leading tag (untranslated) and the \& # rest of the paragraph (translated) \& $self\->pushline( "
"
\& . $self\->translate($paragraph,$pararef)
\& );
\&
\& next PARAGRAPH;
\& } else {
\& # Append to the paragraph
\& $paragraph .= $line;
\& $pararef = $lref unless(length($pararef));
\& }
\&
\& # Reinit the loop
\& ($line,$lref)=$self\->shiftline();
\& }
\& # Did not get a defined line? End of input file.
\& return;
\& }
\& }
.Ve
.PP
Once you've implemented the parse function, you can use your document
class, using the public interface presented in the next section.
.SH "PUBLIC INTERFACE for scripts using your parser"
.IX Header "PUBLIC INTERFACE for scripts using your parser"
.SS Constructor
.IX Subsection "Constructor"
.IP process(%) 4
.IX Item "process(%)"
This function can do all you need to do with a po4a document in one
invocation. Its arguments must be packed as a hash. ACTIONS:
.RS 4
.IP a. 3
.IX Item "a."
Reads all the PO files specified in po_in_name
.IP b. 3
.IX Item "b."
Reads all original documents specified in file_in_name
.IP c. 3
.IX Item "c."
Parses the document
.IP d. 3
.IX Item "d."
Reads and applies all the addenda specified
.IP e. 3
.IX Item "e."
Writes the translated document to file_out_name (if given)
.IP f. 3
.IX Item "f."
Writes the extracted PO file to po_out_name (if given)
.RE
.RS 4
.Sp
ARGUMENTS, beside the ones accepted by \fBnew()\fR (with expected type):
.IP "file_in_name (@)" 4
.IX Item "file_in_name (@)"
List of filenames where we should read the input document.
.IP "file_in_charset ($)" 4
.IX Item "file_in_charset ($)"
Charset used in the input document (if it isn't specified, use UTF\-8).
.IP "file_out_name ($)" 4
.IX Item "file_out_name ($)"
Filename where we should write the output document.
.IP "file_out_charset ($)" 4
.IX Item "file_out_charset ($)"
Charset used in the output document (if it isn't specified, use UTF\-8).
.IP "po_in_name (@)" 4
.IX Item "po_in_name (@)"
List of filenames where we should read the input PO files from, containing
the translation which will be used to translate the document.
.IP "po_out_name ($)" 4
.IX Item "po_out_name ($)"
Filename where we should write the output PO file, containing the strings
extracted from the input document.
.IP "addendum (@)" 4
.IX Item "addendum (@)"
List of filenames where we should read the addenda from.
.IP "addendum_charset ($)" 4
.IX Item "addendum_charset ($)"
Charset for the addenda.
.RE
.RS 4
.RE
.IP new(%) 4
.IX Item "new(%)"
Create a new po4a document. Accepted options (in the hash passed as a parameter):
.RS 4
.IP "verbose ($)" 4
.IX Item "verbose ($)"
Sets the verbosity.
.IP "debug ($)" 4
.IX Item "debug ($)"
Sets the debugging.
.IP "wrapcol ($)" 4
.IX Item "wrapcol ($)"
The column at which we should wrap text in output document (default: 76).
.Sp
The negative value means not to wrap lines at all.
.RE
.RS 4
.Sp
Also it accepts next options for underlying Po-files: \fBporefs\fR,
\&\fBcopyright-holder\fR, \fBmsgid-bugs-address\fR, \fBpackage-name\fR,
\&\fBpackage-version\fR, \fBwrap-po\fR.
.RE
.SS "Manipulating document files"
.IX Subsection "Manipulating document files"
.IP read($$$) 4
.IX Item "read($$$)"
Add another input document data at the end of the existing array \f(CW\*(C`@{$self\->{TT}{doc_in}}\*(C'\fR.
.Sp
This function takes two mandatory arguments and an optional one.
* The filename to read on disk;
* The name to use as filename when building the reference in the PO file;
* The charset to use to read that file (UTF\-8 by default)
.Sp
This array \f(CW\*(C`@{$self\->{TT}{doc_in}}\*(C'\fR holds this input document data as an
array of strings with alternating meanings.
* The string \f(CW$textline\fR holding each line of the input text data.
* The string \f(CW\*(C`$filename:$linenum\*(C'\fR holding its location and called as
"reference" (\f(CW\*(C`linenum\*(C'\fR starts with 1).
.Sp
Please note that it does not parse anything. You should use the \fBparse()\fR
function when you're done with packing input files into the document.
.IP write($) 4
.IX Item "write($)"
Write the translated document to the given filename.
.Sp
This translated document data are provided by:
* \f(CW\*(C`$self\->docheader()\*(C'\fR holding the header text for the plugin, and
* \f(CW\*(C`@{$self\->{TT}{doc_out}}\*(C'\fR holding each line of the main translated text in the array.
.SS "Manipulating PO files"
.IX Subsection "Manipulating PO files"
.IP readpo($) 4
.IX Item "readpo($)"
Add the content of a file (which name is passed as argument) to the
existing input PO. The old content is not discarded.
.IP writepo($) 4
.IX Item "writepo($)"
Write the extracted PO file to the given filename.
.IP \fBstats()\fR 4
.IX Item "stats()"
Returns some statistics about the translation done so far. Please note that
it's not the same statistics than the one printed by msgfmt
\&\-\-statistic. Here, it's stats about recent usage of the PO file, while
msgfmt reports the status of the file. It is a wrapper to the
Locale::Po4a::Po::stats_get function applied to the input PO file. Example
of use:
.Sp
.Vb 1
\& [normal use of the po4a document...]
\&
\& ($percent,$hit,$queries) = $document\->stats();
\& print "We found translations for $percent\e% ($hit from $queries) of strings.\en";
.Ve
.SS "Manipulating addenda"
.IX Subsection "Manipulating addenda"
.IP addendum($) 4
.IX Item "addendum($)"
Please refer to \fBpo4a\fR\|(7) for more information on what addenda are,
and how translators should write them. To apply an addendum to the translated
document, simply pass its filename to this function and you are done ;)
.Sp
This function returns a non-null integer on error.
.SH "INTERNAL FUNCTIONS used to write derivative parsers"
.IX Header "INTERNAL FUNCTIONS used to write derivative parsers"
.SS "Getting input, providing output"
.IX Subsection "Getting input, providing output"
Four functions are provided to get input and return output. They are very
similar to shift/unshift and push/pop of Perl.
.PP
.Vb 4
\& * Perl shift returns the first array item and drop it from the array.
\& * Perl unshift prepends an item to the array as the first array item.
\& * Perl pop returns the last array item and drop it from the array.
\& * Perl push appends an item to the array as the last array item.
.Ve
.PP
The first pair is about input, while the second is about output. Mnemonic: in
input, you are interested in the first line, what shift gives, and in output
you want to add your result at the end, like push does.
.IP \fBshiftline()\fR 4
.IX Item "shiftline()"
This function returns the first line to be parsed and its corresponding
reference (packed as an array) from the array \f(CW\*(C`@{$self\->{TT}{doc_in}}\*(C'\fR and
drop these first 2 array items. Here, the reference is provided by a string
\&\f(CW\*(C`$filename:$linenum\*(C'\fR.
.IP unshiftline($$) 4
.IX Item "unshiftline($$)"
Unshifts the last shifted line of the input document and its corresponding
reference back to the head of \f(CW\*(C`{$self\->{TT}{doc_in}}\*(C'\fR.
.IP pushline($) 4
.IX Item "pushline($)"
Push a new line to the end of \f(CW\*(C`{$self\->{TT}{doc_out}}\*(C'\fR.
.IP \fBpopline()\fR 4
.IX Item "popline()"
Pop the last pushed line from the end of \f(CW\*(C`{$self\->{TT}{doc_out}}\*(C'\fR.
.SS "Marking strings as translatable"
.IX Subsection "Marking strings as translatable"
One function is provided to handle the text which should be translated.
.IP translate($$$) 4
.IX Item "translate($$$)"
Mandatory arguments:
.RS 4
.IP \- 2
A string to translate
.IP \- 2
The reference of this string (i.e. position in inputfile)
.IP \- 2
The type of this string (i.e. the textual description of its structural role;
used in \fBLocale::Po4a::Po::gettextization()\fR; see also \fBpo4a\fR\|(7),
section \fBGettextization: how does it work?\fR)
.RE
.RS 4
.Sp
This function can also take some extra arguments. They must be organized as
a hash. For example:
.Sp
.Vb 2
\& $self\->translate("string","ref","type",
\& \*(Aqwrap\*(Aq => 1);
.Ve
.IP \fBwrap\fR 4
.IX Item "wrap"
boolean indicating whether we can consider that whitespaces in string are
not important. If yes, the function canonizes the string before looking for
a translation or extracting it, and wraps the translation.
.IP \fBwrapcol\fR 4
.IX Item "wrapcol"
the column at which we should wrap (default: the value of \fBwrapcol\fR specified
during creation of the TransTractor or 76).
.Sp
The negative value will be substracted from the default.
.IP \fBcomment\fR 4
.IX Item "comment"
an extra comment to add to the entry.
.RE
.RS 4
.Sp
Actions:
.IP \- 2
Pushes the string, reference and type to po_out.
.IP \- 2
Returns the translation of the string (as found in po_in) so that the
parser can build the doc_out.
.IP \- 2
Handles the charsets to recode the strings before sending them to
po_out and before returning the translations.
.RE
.RS 4
.RE
.SS "Misc functions"
.IX Subsection "Misc functions"
.IP \fBverbose()\fR 4
.IX Item "verbose()"
Returns if the verbose option was passed during the creation of the
TransTractor.
.IP \fBdebug()\fR 4
.IX Item "debug()"
Returns if the debug option was passed during the creation of the
TransTractor.
.IP \fBget_in_charset()\fR 4
.IX Item "get_in_charset()"
This function return the charset that was provided as master charset
.IP \fBget_out_charset()\fR 4
.IX Item "get_out_charset()"
This function will return the charset that should be used in the output
document (usually useful to substitute the input document's detected charset
where it has been found).
.Sp
It will use the output charset specified in the command line. If it wasn't
specified, it will use the input PO's charset, and if the input PO has the
default "CHARSET", it will return the input document's charset, so that no
encoding is performed.
.SH "FUTURE DIRECTIONS"
.IX Header "FUTURE DIRECTIONS"
One shortcoming of the current TransTractor is that it can't handle
translated document containing all languages, like debconf templates, or
\&.desktop files.
.PP
To address this problem, the only interface changes needed are:
.IP \- 2
take a hash as po_in_name (a list per language)
.IP \- 2
add an argument to translate to indicate the target language
.IP \- 2
make a pushline_all function, which would make pushline of its content for
all languages, using a map-like syntax:
.Sp
.Vb 3
\& $self\->pushline_all({ "Description[".$langcode."]=".
\& $self\->translate($line,$ref,$langcode)
\& });
.Ve
.PP
Will see if it's enough ;)
.SH AUTHORS
.IX Header "AUTHORS"
.Vb 3
\& Denis Barbier