190 lines
4.1 KiB
Plaintext
190 lines
4.1 KiB
Plaintext
|
#!/usr/bin/perl -w
|
||
|
# vi:wrap:
|
||
|
|
||
|
# See Qt I18N documentation for usage information.
|
||
|
|
||
|
use POSIX qw(strftime);
|
||
|
|
||
|
$projectid='PROJECT VERSION';
|
||
|
$datetime = strftime "%Y-%m-%d %X %Z", localtime;
|
||
|
$charset='iso-8859-1';
|
||
|
$translator='FULLNAME <EMAIL@ADDRESS>';
|
||
|
$revision_date='YYYY-MM-DD';
|
||
|
|
||
|
$real_mark = "tr";
|
||
|
$noop_mark = "QT_TR_NOOP";
|
||
|
$scoped_mark = "qApp->translate";
|
||
|
$noop_scoped_mark = "QT_TRANSLATE_NOOP";
|
||
|
|
||
|
$header=
|
||
|
'# This is a Qt message file in .po format. Each msgid starts with
|
||
|
# a scope. This scope should *NOT* be translated - eg. translating
|
||
|
# from French to English, "Foo::Bar" would be translated to "Pub",
|
||
|
# not "Foo::Pub".
|
||
|
msgid ""
|
||
|
msgstr ""
|
||
|
"Project-Id-Version: '.$projectid.'\n"
|
||
|
"POT-Creation-Date: '.$datetime.'\n"
|
||
|
"PO-Revision-Date: '.$revision_date.'\n"
|
||
|
"Last-Translator: '.$translator.'\n"
|
||
|
"Content-Type: text/plain; charset='.$charset.'\n"
|
||
|
|
||
|
';
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
$scope = "";
|
||
|
|
||
|
if ( $#ARGV < 0 ) {
|
||
|
print STDERR "Usage: findtr sourcefile ... >project.po\n";
|
||
|
exit 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
sub outmsg {
|
||
|
my ($file, $line, $scope, $msgid) = @_;
|
||
|
# unesc
|
||
|
$msgid =~ s/$esc:$esc:$esc/::/gs;
|
||
|
$msgid =~ s|$esc/$esc/$esc|//|gs;
|
||
|
|
||
|
# Remove blank lines
|
||
|
$msgid =~ s/\n\n+/\n/gs;
|
||
|
$msgid = "\"\"\n$msgid" if $msgid =~ /\n/s;
|
||
|
print "#: $file:$line\n";
|
||
|
$msgid =~ s/^"//; #"emacs bug
|
||
|
print "msgid \"${scope}::$msgid\n";
|
||
|
#print "msgstr \"$msgid\n";
|
||
|
print "msgstr \"\"\n";
|
||
|
print "\n";
|
||
|
}
|
||
|
|
||
|
sub justlines {
|
||
|
my $l = @_;
|
||
|
$l =~ tr|\n||dc;
|
||
|
return $l;
|
||
|
}
|
||
|
|
||
|
print $header;
|
||
|
|
||
|
|
||
|
foreach $file ( @ARGV ) {
|
||
|
next unless open( I, "< $file" );
|
||
|
|
||
|
$source = join( "", <I> );
|
||
|
|
||
|
# Find esc. Avoid bad case 1/// -> 1/1/1/1 -> ///1
|
||
|
$esc = 1;
|
||
|
while ( $source =~ m!(?:$esc/$esc/$esc)|(?:$esc///)
|
||
|
|(?:$esc:$esc:$esc)|(?:$esc:\:\:)! ) {
|
||
|
$esc++;
|
||
|
}
|
||
|
|
||
|
# Hide quoted :: in practically all strings
|
||
|
$source =~ s/\"([^"\n]*)::([^"\n]*)\"/\"$1$esc:$esc:$esc$2\"/g;
|
||
|
|
||
|
# Hide quoted // in practically all strings
|
||
|
$source =~ s|\"([^"\n]*)//([^"\n]*)\"|\"$1$esc/$esc/$esc$2\"|g;
|
||
|
|
||
|
|
||
|
# strip comments -- does not handle "/*" in strings
|
||
|
while( $source =~ s|/\*(.*?)\*/|justlines($1)|ges ) { }
|
||
|
while( $source =~ s|//(.*?)\n|\n|g ) { }
|
||
|
|
||
|
while( $source =~ /
|
||
|
(?:
|
||
|
# Some doublequotes are "escaped" to help vim syntax highlight
|
||
|
|
||
|
# $1 = scope; $2 = parameters etc.
|
||
|
(?:
|
||
|
# Scoped function name ($1 is scope).
|
||
|
(\w+)::(?:\w+)
|
||
|
\s*
|
||
|
# Parameters etc up to open-curly - no semicolons
|
||
|
\(([^();]*)\)
|
||
|
\s*
|
||
|
(?:\{|:)
|
||
|
)
|
||
|
|
|
||
|
# $3 - one-argument msgid
|
||
|
(?:\b
|
||
|
# One of the marks
|
||
|
(?:$real_mark|$noop_mark)
|
||
|
\s*
|
||
|
# The parameter
|
||
|
\(\s*((?:"(?:[^"]|[^\\]\\")*"\s*)+)\)
|
||
|
)
|
||
|
|
|
||
|
# $4,$5 - two-argument msgid
|
||
|
(?:\b
|
||
|
# One of the scoped marks
|
||
|
(?:$scoped_mark|$noop_scoped_mark)
|
||
|
\s*
|
||
|
# The parameters
|
||
|
\(
|
||
|
# The scope parameter
|
||
|
\s*"([^\"]*)"
|
||
|
\s*,\s*
|
||
|
# The msgid parameter
|
||
|
\s*((?:\"(?:[^"]|[^\\]\\")*"\s*)+) #"emacs
|
||
|
\)
|
||
|
)
|
||
|
|
|
||
|
# $6,$7 - scoped one-argument msgid
|
||
|
(?:\b
|
||
|
# The scope
|
||
|
(\w+)::
|
||
|
# One of the marks
|
||
|
(?:$real_mark)
|
||
|
\s*
|
||
|
# The parameter
|
||
|
\(\s*((?:"(?:[^"]|[^\\]\\")*"\s*)+)\)
|
||
|
)
|
||
|
)/gsx )
|
||
|
{
|
||
|
@lines = split /^/m, "$`";
|
||
|
$line = @lines;
|
||
|
if ( defined( $1 ) ) {
|
||
|
if ( $scope ne $1 ) {
|
||
|
$sc=$1;
|
||
|
$etc=$2;
|
||
|
# remove strings
|
||
|
$etc =~ s/"(?:[^"]|[^\\]\\")"//g;
|
||
|
# count ( and )
|
||
|
@open = split /\(/m, $etc;
|
||
|
@close = split /\)/m, $etc;
|
||
|
if ( $#open == $#close ) {
|
||
|
$scope = $sc;
|
||
|
}
|
||
|
}
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
if ( defined( $3 ) ) {
|
||
|
$this_scope = $scope;
|
||
|
$msgid = $3;
|
||
|
} elsif ( defined( $4 ) ) {
|
||
|
$this_scope = $4;
|
||
|
$msgid = $5;
|
||
|
} elsif ( defined( $6 ) ) {
|
||
|
$this_scope = $6;
|
||
|
$msgid = $7;
|
||
|
} else {
|
||
|
next;
|
||
|
}
|
||
|
|
||
|
$msgid =~ s/^\s*//;
|
||
|
$msgid =~ s/\s*$//;
|
||
|
|
||
|
# Might still be non-unique eg. tr("A" "B") vs. tr("A" "B").
|
||
|
|
||
|
$location{"${this_scope}::${msgid}"} = "$file:$line";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for $scoped_msgid ( sort keys %location ) {
|
||
|
($scope,$msgid) = $scoped_msgid =~ m/([^:]*)::(.*)/s;
|
||
|
($file,$line) = $location{$scoped_msgid} =~ m/([^:]*):(.*)/s;
|
||
|
outmsg($file,$line,$scope,$msgid);
|
||
|
}
|