LOCALE::PO4A::TRANSTRACTOR.3PM(1) User Contributed Perl Documentation LOCALE::PO4A::TRANSTRACTOR.3PM(1)

名前

Locale::Po4a::TransTractor - 汎用翻訳抽出機構。

説明

po4a (PO for anything) プロジェクトは、gettext ツールが想定していないドキュメントのような領域で翻訳をしやすくすること (またより興味深いのは、翻訳文の保守がしやすくなること) を目標にしています。

このクラスは、翻訳可能な文字列を検索するためのドキュメントのパース、PO ファイルへの抽出、出力したドキュメントへの翻訳した文字列の置換に使用する、すべての po4a パーサの祖先になります。

もっと形式張って言うと、入力として以下の引数を取ります:

  • 翻訳する文書;
  • 使用する翻訳を含むPO ファイル。

以下を出力します:

  • 入力ドキュメントから翻訳可能な文字列を抽出した、別の PO ファイル。
  • 入力したものと同じ構造で、入力した PO ファイルにある翻訳で翻訳可能な文字列を置換した、翻訳済みドキュメント。

これを視覚的に表すと次のようになります:

入力ドキュメント-\                             /---> 出力ドキュメント
                  \                           /          (翻訳済)
                   +-> parse() 関数 ---------+
                  /                           \
入力 PO  --------/                             \---> 出力 PO
                                                      (抽出済)

個別のパーサでオーバーライドするべき関数

ここに、入力ドキュメントのパース、出力の生成、翻訳可能文字列の抽出といった、すべての動作を定義しています。後述する 内部関数 節で説明する提供された関数を使用するのはかなり簡単です。サンプル付きの 書式 も参照してください。

この関数は後述の process() 関数から呼ばれますが、new() 関数を使用してドキュメントに内容を手で追加するのを選んだ場合、この関数自体を呼ばねばなりません。

この関数は、ターゲットの言語でコメントにするために適切にクォートした、生成したドキュメントに追加するべきヘッダを返します。この何がよいのかは、po4a(7)翻訳についての開発者教育 節を参照してください。

書式

以下の例は、"<p>" で始まる段落のリストをパースします。簡単にするために、ドキュメントは整形されている、すなわち、現れるタグは '<p>' のみで、各段落はこのタグで必ず始まると仮定します。

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/<p>/ && !$first--; ) {
              # Not the first time we see <p>.
              # 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/^<p>//s;
              #   - (翻訳されていない)先頭のタグと
              #     (翻訳された)残りの段落を出力に押し込みます
              $self->pushline(  "<p>"
                              . $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;
  }
}

parse 関数を実装したら、次節で説明するパブリックインターフェースを用いて document クラスを使用できます。

パーサで使用するスクリプトのパブリックインターフェース

コンストラクタ

この関数は、po4a ドキュメントで行うのに必要なすべてを、一度の実行で行います。引数はハッシュとしてパックしなくてはなりません。動作は以下のようになります:
po_in_name で指定したすべての PO ファイルの読み込み
file_in_name で指定したすべてのオリジナルドキュメントの読み込み
ドキュメントのパース
指定したすべての追加内容の読み込みと適用
翻訳したドキュメントの file_out_name への書き出し (与えられた場合)
抽出した PO ファイルの po_out_name への書き出し (与えられた場合)

new() で受け付けるもの以外の引数 (と想定する型):

読み込むべき入力ドキュメントのファイル名のリストです。
入力文書で使用する文字集合です(指定しない場合、UTF-8を使用)。
書き出すべき出力ドキュメントのファイル名です。
出力文書で使用する文字集合です(指定しない場合、UTF-8を使用)。
読み込むべき入力 PO ファイル (ドキュメントの翻訳に使用する翻訳) のファイル名のリストです。
入力ドキュメントから抽出した文字列を含む、書き出すべき出力 PO ファイルのファイル名です。
読み込むべき追加内容のファイル名のリストです。
追加内容の文字セット。
新規 po4a ドキュメントを作成します。以下の(引数として渡されたハッシュの)オプションを受け取ります:
冗長表示を有効にします。
デバッグを有効にします。

ドキュメントファイルの操作

既存の配列"@{$self->{TT}{doc_in}}"の末尾に他の入力文書データを加えます。引数は読み込むファイル名です。もし2つ目の引数が与えられていれば、それは参照で使うファイル名です。

この配列 "@{$self->{TT}{doc_in}}" にはこの入力文書のデータが別の意味合いを持つ文字列の配列として保管されています。
* 文字列 $textline には入力テキストデータのそれぞれの行が含まれます。
* 文字列 "$filename:$linenum" には位置が含まれており「参照」と呼ばれます("linenum"は1始まりです)。

パースは一切行わないことに注意してください。入力ファイルがドキュメントに格納した時点で parse() 関数を使用するべきです。

与えたファイル名で翻訳済みドキュメントを書き出します。

この翻訳された文書データは以下によって与えられます。
* "$self->docheader()" はプラグインのヘッダテキストを保有します。また、
* "@{$self->{TT}{doc_out}}" は配列に翻訳されたメインのテキストのそれぞれの行を保有します。

既存の入力 PO に、(引数で渡した名前の) ファイルの内容を追加します。古い内容は破棄されません。
抽出した PO ファイルを与えたファイル名で書き出します。
現在までに翻訳した内容に関する統計を返します。msgfmt --statistic が出力する統計とは同じとは限らないことに注意してください。ここでは、PO ファイルの最新の使用法についての統計ですが、msgfmt が報告するのは、ファイルの状態についてです。これは、Locale::Po4a::Po::stats_get 関数を入力した PO ファイルに適用するラッパーです。サンプルは以下のようになります:
[po4a ドキュメントの通常の使用...]
($percent,$hit,$queries) = $document->stats();
print "$percent\%($queries中$hit)の文字列の翻訳があります。\n";

追加内容の操作

追加内容とは何か、や、翻訳者はどのように書いたらよいのかといった情報は、po4a(7) を参照してください。翻訳したドキュメントに追加内容を適用するには、この関数に単純にファイル名を渡し、実行するだけです ;)

この関数は、エラー時に null 以外の数値を返します。

派生パーサ作成時に使用する内部関数

入力と出力

入力を取得し出力を返すために4つの関数が提供されています。これらはPerlのshift/unshiftとpush/popに大変似ています。

* Perlのshiftは最初の配列の要素を返し配列からその要素を切り落とします。
* Perlのunshiftは配列に要素をその配列の最初の要素として前置します。
* Perlのpopは配列の最後の要素を返しそれを配列から切り落とします。
* Perlのpushは配列にその配列の最後の要素として後置します。

1つ目の対は入力についてのもので、2つ目は出力についてのものです。覚え方としては、入力ではshiftにより得られる最初の行に関心があり、出力ではpushがするように結果を末尾に加えたいのだ、とできます。

この関数は解析される最初の行と配列"@{$self->{TT}{doc_in}}"からの(配列として詰め込まれた)参照を返し、これらの配列の最初2つの要素を除きます。ここで参照は文字列"$filename:$linenum"によって与えられます。
入力文書の最後にshiftされた行をunshiftして対応する参照を"{$self->{TT}{doc_in}}"の先頭に戻します。
"{$self->{{TT}{doc_out}}"の末尾に開業を押し込みます。
"{$self->{TT}{doc_out}}"の末尾から最後に押し込んだ行を取り出します。

文字列を翻訳可能としてマーク

翻訳するべきテキストを扱う関数を一つ用意しています。

必須の引数:
  • 翻訳する文字列
  • この文字列の参照 (言い換えると、入力ファイルの場所)
  • 文字列の型 (つまり構造上の役割をテキストで説明したもの。Locale::Po4a::Po::gettextization() で使用します。 po4a(7)gettext 化: どのように動作しますか? 節も参照してください)

この関数は、いくつか追加引数を取れます。ハッシュとしてまとめなければなりません。例えば:

$self->translate("string","ref","type",
                 'wrap' => 1);
文字列中の空白が重要でないとして扱うかどうかを示す真偽値です。重要でない場合、この関数は、翻訳を探したり抽出したりする前の文字列を納め、翻訳を折り返します。
改行を行う幅です (デフォルト: 76)。
エントリに追加する更なるコメント。

動作:

  • 文字列、参照、型を po_out に push します。
  • パーサが doc_out をビルドできるように、文字列の翻訳 (po_in に見つかったもの) を返します。
  • 文字列を po_out に送る前や翻訳を返す前に、文字列を再コード化する文字セットを扱います。

その他の関数

TransTractor の生成時に verbose オプションが渡された場合、返します。
TransTractor の生成時に debug オプションが渡された場合、返します。
この関数はマスター文書の文字集合として与えられる文字集合を返します
この関数は、出力文書で使用する文字セットを返します(この関数により見付かった入力文書で検出された文字セットを置き換えるのに、大抵の場合で便利です)。

コマンドラインで指定した出力文字セットが使われます。指定しない場合は、入力 PO ファイルの文字セットを使用します。入力 PO ファイルにデフォルトの "CHARSET" がある場合は、入力ドキュメントの文字セットを返します。そして、エンコーディングの変換は行われません。

将来の方向性

現在の TransTractor の欠点の一つに、(debconf テンプレートや、.desktop ファイルのような) すべての言語を含む翻訳済みドキュメントを扱えないというものがあります。

この問題に対処するには、以下のようにインターフェースのみを変更することが必要です:

  • po_in_name (言語ごとのリスト) としてハッシュを取ります
  • 対象言語を示すための翻訳する引数を追加します
  • pushline_all 関数を作ります。この関数により全ての言語を対象とする内容のpushlineを生み出します。これにはmapのような構文が使われます。
    $self->pushline_all({ "Description[".$langcode."]=".
                          $self->translate($line,$ref,$langcode)
                        });
    

これで十分だといいのですが ;)

著者

Denis Barbier <barbier@linuxfr.org>
Martin Quinson (mquinson#debian.org)
Jordi Vilalta <jvprat@gmail.com>

訳者

倉澤 望 <nabetaro@debian.or.jp>
Debian JP Documentation ML <debian-doc@debian.or.jp>
2024-04-28 perl v5.38.2