mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-07 10:21:31 +00:00
521 lines
15 KiB
Perl
Executable File
521 lines
15 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
#
|
|
# This tools is kind of a hack to be able to maintain the credits list of
|
|
# Residual in a single central location. We then generate the various versions
|
|
# of the credits in other places from this source. In particular:
|
|
# - The AUTHORS file
|
|
# - The gui/credits.h header file
|
|
# - The Credits.rtf file used by the Mac OS X port
|
|
# - The credits.xml file, part of the DocBook manual
|
|
# - Finally, credits.xml, for use on the website (different format than the DocBook one)
|
|
#
|
|
# Initial version written by Fingolfin in December 2004.
|
|
#
|
|
|
|
|
|
use strict;
|
|
use Text::Wrap;
|
|
|
|
if ($Text::Wrap::VERSION < 2001.0929) {
|
|
die "Text::Wrap version >= 2001.0929 is required. You have $Text::Wrap::VERSION\n";
|
|
}
|
|
|
|
my $mode = "";
|
|
my $max_name_width;
|
|
|
|
# Count the level in the section hierarchy, i.e. how deep we are currently nested
|
|
# in terms of 'sections'.
|
|
my $section_level = 0;
|
|
|
|
# Count how many sections there have been on this level already
|
|
my @section_count = ( 0, 0, 0 );
|
|
|
|
if ($#ARGV >= 0) {
|
|
$mode = "TEXT" if ($ARGV[0] eq "--text"); # AUTHORS file
|
|
$mode = "XML-WEB" if ($ARGV[0] eq "--xml-website"); # credits.xml (for use on the website)
|
|
$mode = "CPP" if ($ARGV[0] eq "--cpp"); # credits.h (for use by about.cpp)
|
|
$mode = "XML-DOC" if ($ARGV[0] eq "--xml-docbook"); # credits.xml (DocBook)
|
|
$mode = "RTF" if ($ARGV[0] eq "--rtf"); # Credits.rtf (Mac OS X About box)
|
|
}
|
|
|
|
if ($mode eq "") {
|
|
print STDERR "Usage: $0 [--text | --xml-website | --cpp | --xml-docbook | --rtf]\n";
|
|
print STDERR " Just pass --text / --xml-website / --cpp / --xml-docbook / --rtf as parameter, and credits.pl\n";
|
|
print STDERR " will print out the corresponding version of the credits to stdout.\n";
|
|
exit 1;
|
|
}
|
|
|
|
$Text::Wrap::unexpand = 0;
|
|
if ($mode eq "TEXT") {
|
|
$Text::Wrap::columns = 78;
|
|
$max_name_width = 21; # The maximal width of a name.
|
|
} elsif ($mode eq "CPP") {
|
|
$Text::Wrap::columns = 48; # Approx.
|
|
}
|
|
|
|
# Convert HTML entities to ASCII for the plain text mode
|
|
sub html_entities_to_ascii {
|
|
my $text = shift;
|
|
|
|
# For now we hardcode these mappings
|
|
# á -> a
|
|
# é -> e
|
|
# ì -> i
|
|
# ó -> o
|
|
# ø -> o
|
|
# ö -> o / oe
|
|
# ä -> a
|
|
# ü -> ue
|
|
# å -> aa
|
|
# & -> &
|
|
# ł -> l
|
|
# Š -> S
|
|
$text =~ s/á/a/g;
|
|
$text =~ s/é/e/g;
|
|
$text =~ s/ì/i/g;
|
|
$text =~ s/ó/o/g;
|
|
$text =~ s/ø/o/g;
|
|
$text =~ s/ł/l/g;
|
|
$text =~ s/Š/S/g;
|
|
$text =~ s/å/aa/g;
|
|
|
|
$text =~ s/ä/a/g;
|
|
$text =~ s/ü/ue/g;
|
|
# HACK: Torbj*o*rn but G*oe*ffringmann and R*oe*ver and J*oe*rg
|
|
$text =~ s/Torbjörn/Torbjorn/g;
|
|
$text =~ s/ö/oe/g;
|
|
|
|
$text =~ s/&/&/g;
|
|
|
|
return $text;
|
|
}
|
|
|
|
# Convert HTML entities to C++ characters
|
|
sub html_entities_to_cpp {
|
|
my $text = shift;
|
|
|
|
# The numerical values are octal!
|
|
$text =~ s/á/\\341/g;
|
|
$text =~ s/é/\\351/g;
|
|
$text =~ s/ì/\\354/g;
|
|
$text =~ s/ó/\\363/g;
|
|
$text =~ s/ø/\\370/g;
|
|
$text =~ s/ł/l/g;
|
|
$text =~ s/Š/S/g;
|
|
$text =~ s/å/\\345/g;
|
|
|
|
$text =~ s/ä/\\344/g;
|
|
$text =~ s/ö/\\366/g;
|
|
$text =~ s/ü/\\374/g;
|
|
|
|
$text =~ s/&/&/g;
|
|
|
|
return $text;
|
|
}
|
|
|
|
# Convert HTML entities to RTF codes
|
|
# This is using the Mac OS Roman encoding
|
|
sub html_entities_to_rtf {
|
|
my $text = shift;
|
|
|
|
$text =~ s/á/\\'87/g;
|
|
$text =~ s/é/\\'8e/g;
|
|
$text =~ s/ì/\\'93/g;
|
|
$text =~ s/ó/\\'97/g;
|
|
$text =~ s/ø/\\'bf/g;
|
|
$text =~ s/å/\\'8c/g;
|
|
# The following numerical values are octal!
|
|
$text =~ s/ł/\\uc0\\u322 /g;
|
|
$text =~ s/Š/\\uc0\\u540 /g;
|
|
|
|
# Back to hex numbers
|
|
$text =~ s/ä/\\'8a/g;
|
|
$text =~ s/ö/\\'9a/g;
|
|
$text =~ s/ü/\\'9f/g;
|
|
|
|
$text =~ s/&/&/g;
|
|
|
|
return $text;
|
|
}
|
|
|
|
# Convert HTML entities to TeX codes
|
|
sub html_entities_to_tex {
|
|
my $text = shift;
|
|
|
|
$text =~ s/á/\\'a/g;
|
|
$text =~ s/é/\\'e/g;
|
|
$text =~ s/ì/\\`\\i/g;
|
|
$text =~ s/ó/\\'o/g;
|
|
$text =~ s/ø/{\\o}/g;
|
|
$text =~ s/å/\\aa /g;
|
|
$text =~ s/ł/{\\l}/g;
|
|
$text =~ s/Š/{\\v S}/g;
|
|
|
|
$text =~ s/ä/\\"a/g;
|
|
$text =~ s/ö/\\"o/g;
|
|
$text =~ s/ü/\\"u/g;
|
|
|
|
$text =~ s/&/\\&/g;
|
|
|
|
return $text;
|
|
}
|
|
|
|
#
|
|
# Small reference of the RTF commands used here:
|
|
#
|
|
# \fs28 switches to 14 point font (28 = 2 * 14)
|
|
# \pard reset to default paragraph properties
|
|
#
|
|
# \ql left-aligned text
|
|
# \qr right-aligned text
|
|
# \qc centered text
|
|
# \qj justified text
|
|
#
|
|
# \b turn on bold
|
|
# \b0 turn off bold
|
|
#
|
|
# For more information: <http://latex2rtf.sourceforge.net/rtfspec.html>
|
|
#
|
|
|
|
sub begin_credits {
|
|
my $title = shift;
|
|
|
|
if ($mode eq "TEXT") {
|
|
#print html_entities_to_ascii($title)."\n";
|
|
} elsif ($mode eq "RTF") {
|
|
print '{\rtf1\mac\ansicpg10000' . "\n";
|
|
print '{\fonttbl\f0\fswiss\fcharset77 Helvetica-Bold;\f1\fswiss\fcharset77 Helvetica;}' . "\n";
|
|
print '{\colortbl;\red255\green255\blue255;\red0\green128\blue0;\red128\green128\blue128;}' . "\n";
|
|
print '\vieww6920\viewh15480\viewkind0' . "\n";
|
|
print "\n";
|
|
} elsif ($mode eq "CPP") {
|
|
print "// This file was generated by credits.pl. Do not edit by hand!\n";
|
|
print "static const char *credits[] = {\n";
|
|
} elsif ($mode eq "XML-DOC") {
|
|
print "<?xml version='1.0'?>\n";
|
|
print "<!-- This file was generated by credits.pl. Do not edit by hand! -->\n";
|
|
print "<!DOCTYPE appendix PUBLIC '-//OASIS//DTD DocBook XML V4.2//EN'\n";
|
|
print " 'http://www.oasis-open.org/docbook/xml/4.2/docbookx.dtd'>\n";
|
|
print "<appendix id='credits'>\n";
|
|
print " <title>" . $title . "</title>\n";
|
|
print " <informaltable frame='none'>\n";
|
|
print " <tgroup cols='3' align='left' colsep='0' rowsep='0'>\n";
|
|
print " <colspec colname='start' colwidth='0.5cm'/>\n";
|
|
print " <colspec colname='name' colwidth='4cm'/>\n";
|
|
print " <colspec colname='job'/>\n";
|
|
print " <tbody>\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
print "<?xml version='1.0'?>\n";
|
|
print "<!-- This file was generated by credits.pl. Do not edit by hand! -->\n";
|
|
print "<credits>\n";
|
|
}
|
|
}
|
|
|
|
sub end_credits {
|
|
if ($mode eq "TEXT") {
|
|
} elsif ($mode eq "RTF") {
|
|
print "}\n";
|
|
} elsif ($mode eq "CPP") {
|
|
print "};\n";
|
|
} elsif ($mode eq "XML-DOC") {
|
|
print " </tbody>\n";
|
|
print " </tgroup>\n";
|
|
print " </informaltable>\n";
|
|
print "</appendix>\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
print "</credits>\n";
|
|
}
|
|
}
|
|
|
|
sub begin_section {
|
|
my $title = shift;
|
|
|
|
if ($mode eq "TEXT") {
|
|
$title = html_entities_to_ascii($title);
|
|
|
|
if ($section_level >= 2) {
|
|
$title .= ":"
|
|
}
|
|
|
|
print " " x $section_level . $title."\n";
|
|
if ($section_level eq 0) {
|
|
print " " x $section_level . "*" x (length $title)."\n";
|
|
} elsif ($section_level eq 1) {
|
|
print " " x $section_level . "-" x (length $title)."\n";
|
|
}
|
|
} elsif ($mode eq "RTF") {
|
|
$title = html_entities_to_rtf($title);
|
|
|
|
# Center text
|
|
print '\pard\qc' . "\n";
|
|
print '\f0\b';
|
|
if ($section_level eq 0) {
|
|
print '\fs40 ';
|
|
} elsif ($section_level eq 1) {
|
|
print '\fs32 ';
|
|
}
|
|
|
|
# Insert an empty line before this section header, *unless*
|
|
# this is the very first section header in the file.
|
|
if ($section_level > 0 || @section_count[0] > 0) {
|
|
print "\\\n";
|
|
}
|
|
print '\cf2 ' . $title . "\n";
|
|
print '\f1\b0\fs24 \cf0 \\' . "\n";
|
|
} elsif ($mode eq "CPP") {
|
|
if ($section_level eq 0) {
|
|
# TODO: Would be nice to have a 'fat' or 'large' mode for
|
|
# headlines...
|
|
$title = html_entities_to_cpp($title);
|
|
print '"C1""'.$title.'",' . "\n";
|
|
print '"",' . "\n";
|
|
} else {
|
|
$title = html_entities_to_cpp($title);
|
|
print '"C1""'.$title.'",' . "\n";
|
|
}
|
|
} elsif ($mode eq "XML-DOC") {
|
|
print " <row><entry namest='start' nameend='job'>";
|
|
print "<emphasis role='bold'>" . $title . ":</emphasis>";
|
|
print "</entry></row>\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
if ($section_level eq 0) {
|
|
print "\t<section>\n";
|
|
print "\t\t<title>" . $title . "</title>\n";
|
|
} elsif ($section_level eq 1) {
|
|
print "\t\t<subsection>\n";
|
|
print "\t\t\t<title>" . $title . "</title>\n";
|
|
} else {
|
|
#print "\t\t\t<group>" . $title . "</group>\n";
|
|
#print "\t\t\t\t<name>" . $title . "</name>\n";
|
|
}
|
|
}
|
|
|
|
# Implicit start of person list on section level 2
|
|
if ($section_level >= 2) {
|
|
begin_persons($title);
|
|
}
|
|
@section_count[$section_level]++;
|
|
$section_level++;
|
|
@section_count[$section_level] = 0;
|
|
}
|
|
|
|
sub end_section {
|
|
$section_level--;
|
|
|
|
# Implicit end of person list on section level 2
|
|
if ($section_level >= 2) {
|
|
end_persons();
|
|
}
|
|
|
|
if ($mode eq "TEXT") {
|
|
# nothing
|
|
} elsif ($mode eq "RTF") {
|
|
# nothing
|
|
} elsif ($mode eq "CPP") {
|
|
print '"",' . "\n";
|
|
} elsif ($mode eq "XML-DOC") {
|
|
print " <row><entry namest='start' nameend='job'> </entry></row>\n\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
if ($section_level eq 0) {
|
|
print "\t</section>\n";
|
|
} elsif ($section_level eq 1) {
|
|
print "\t\t</subsection>\n";
|
|
} else {
|
|
#print "\t\t\t</group>\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
sub begin_persons {
|
|
my $title = shift;
|
|
if ($mode eq "XML-WEB") {
|
|
print "\t\t\t<group>\n";
|
|
print "\t\t\t\t<name>" . $title . "</name>\n";
|
|
#print "\t\t\t\t<persons>\n";
|
|
}
|
|
}
|
|
|
|
sub end_persons {
|
|
if ($mode eq "TEXT") {
|
|
print "\n";
|
|
} elsif ($mode eq "RTF") {
|
|
# nothing
|
|
} elsif ($mode eq "XML-WEB") {
|
|
#print "\t\t\t\t</persons>\n";
|
|
print "\t\t\t</group>\n";
|
|
}
|
|
}
|
|
|
|
sub add_person {
|
|
my $name = shift;
|
|
my $nick = shift;
|
|
my $desc = shift;
|
|
my $tab;
|
|
|
|
if ($mode eq "TEXT") {
|
|
my $min_name_width = length $desc > 0 ? $max_name_width : 0;
|
|
$name = $nick if $name eq "";
|
|
$name = html_entities_to_ascii($name);
|
|
$desc = html_entities_to_ascii($desc);
|
|
|
|
$tab = " " x ($section_level * 2 + 1);
|
|
printf $tab."%-".$min_name_width.".".$max_name_width."s", $name;
|
|
|
|
# Print desc wrapped
|
|
if (length $desc > 0) {
|
|
my $inner_indent = ($section_level * 2 + 1) + $max_name_width + 3;
|
|
my $multitab = " " x $inner_indent;
|
|
print " - " . substr(wrap($multitab, $multitab, $desc), $inner_indent);
|
|
}
|
|
print "\n";
|
|
} elsif ($mode eq "RTF") {
|
|
$name = $nick if $name eq "";
|
|
$name = html_entities_to_rtf($name);
|
|
|
|
# Center text
|
|
print '\pard\qc' . "\n";
|
|
# Activate 1.5 line spacing mode
|
|
print '\sl360\slmult1';
|
|
# The name
|
|
print $name . "\\\n";
|
|
# Description using italics
|
|
if (length $desc > 0) {
|
|
$desc = html_entities_to_rtf($desc);
|
|
print '\pard\qc' . "\n";
|
|
print "\\cf3\\i " . $desc . "\\i0\\cf0\\\n";
|
|
}
|
|
} elsif ($mode eq "CPP") {
|
|
$name = $nick if $name eq "";
|
|
$name = html_entities_to_cpp($name);
|
|
|
|
print '"C0""'.$name.'",' . "\n";
|
|
|
|
# Print desc wrapped
|
|
if (length $desc > 0) {
|
|
$desc = html_entities_to_cpp($desc);
|
|
print '"C2""'.$desc.'",' . "\n";
|
|
}
|
|
} elsif ($mode eq "XML-DOC") {
|
|
$name = $nick if $name eq "";
|
|
print " <row><entry namest='name'>" . $name . "</entry>";
|
|
print "<entry>" . $desc . "</entry></row>\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
$name = "???" if $name eq "";
|
|
print "\t\t\t\t<person>\n";
|
|
print "\t\t\t\t\t<name>" . $name . "</name>\n";
|
|
print "\t\t\t\t\t<alias>" . $nick . "</alias>\n";
|
|
print "\t\t\t\t\t<description>" . $desc . "</description>\n";
|
|
print "\t\t\t\t</person>\n";
|
|
}
|
|
}
|
|
|
|
sub add_paragraph {
|
|
my $text = shift;
|
|
my $tab;
|
|
|
|
if ($mode eq "TEXT") {
|
|
$tab = " " x ($section_level * 2 + 1);
|
|
print wrap($tab, $tab, html_entities_to_ascii($text))."\n";
|
|
print "\n";
|
|
} elsif ($mode eq "RTF") {
|
|
$text = html_entities_to_rtf($text);
|
|
# Center text
|
|
print '\pard\qc' . "\n";
|
|
print "\\\n";
|
|
print $text . "\\\n";
|
|
} elsif ($mode eq "CPP") {
|
|
$text = html_entities_to_ascii($text);
|
|
my $line_start = '"C0""';
|
|
my $line_end = '",';
|
|
print $line_start . $text . $line_end . "\n";
|
|
print $line_start . $line_end . "\n";
|
|
} elsif ($mode eq "XML-DOC") {
|
|
print " <row><entry namest='start' nameend='job'>" . $text . "</entry></row>\n";
|
|
print " <row><entry namest='start' nameend='job'> </entry></row>\n\n";
|
|
} elsif ($mode eq "XML-WEB") {
|
|
print "\t\t<paragraph>" . $text . "</paragraph>\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Now follows the actual credits data! The format should be clear, I hope.
|
|
# Note that people are sorted by their last name in most cases; in the
|
|
# 'Team' section, they are first grouped by category (Engine; porter; misc).
|
|
#
|
|
|
|
begin_credits("Credits");
|
|
begin_section("Residual Team");
|
|
begin_section("Project Leader");
|
|
begin_persons();
|
|
add_person("Paweł Kołodziejski", "aquadran", "");
|
|
end_persons();
|
|
end_section();
|
|
|
|
begin_section("Engine Teams");
|
|
begin_section("Grim");
|
|
add_person("James Brown", "ender", "Core developer (retired)");
|
|
add_person("Giulio Camuffo", "giucam", "Core developer");
|
|
add_person("Paweł Kołodziejski", "aquadran", "Core developer");
|
|
end_section();
|
|
|
|
begin_section("Grim Contributors");
|
|
add_paragraph(
|
|
"If you have contributed to this engine then you deserve to be on this ".
|
|
"list. Contact us and we'll add you.");
|
|
add_person("Torbjörn Andersson", "eriktorbjorn", "Various code fixes");
|
|
add_person("Ori Avtalion", "salty-horse", "Lipsync, LAF support, various code fixes");
|
|
add_person("Marcus Comstedt", "marcus_c", "Initial Dreamcast port");
|
|
add_person("Andrea Corna", "Yak Bizzarro", "Improved font support, patch extractor");
|
|
add_person("Jonathan Gray", "khalek", "Various code fixes");
|
|
add_person("Vincent Hamm", "yazoo", "Various engine code fixes and improvements");
|
|
add_person("Erich Hoover", "Compholio", "x86-64 fixes, various code fixes and improvements");
|
|
add_person("Travis Howell", "Kirben", "Various code fixes, Windows port");
|
|
add_person("Joost Peters", "joostp", "Various code fixes");
|
|
add_person("Matthieu Milan", "usineur", "Few code improvements");
|
|
add_person("Christian Neumair", "mannythegnome", "Various optimisation patches");
|
|
add_person("Daniel Schepler", "", "Initial engine codebase, LUA support");
|
|
add_person("Einar Johan T. Sømåen", "somaen", "Initial EMI support, various code fixes and improvements");
|
|
add_person("Yaron Tausky", "yaront", "Fixes to subtitles");
|
|
add_person("Joel Teichroeb ", "klusark", "Initial EMI support, various code fixes");
|
|
add_person("Pino Toscano", "pinotree", "Debian GNU/Linux package files");
|
|
add_person("Lionel Ulmer", "bbrox", "OpenGL optimisations");
|
|
add_person("Joni Vähämäki", "Akz", "Various code fixes and improvements");
|
|
add_person("", "vpelletier", "Various code fixes and improvements");
|
|
end_section();
|
|
|
|
end_section();
|
|
end_section();
|
|
|
|
begin_section("ScummVM code");
|
|
add_paragraph(
|
|
"Residual use a lot of ScummVM code. ".
|
|
"For a list of authors look into COPYRIGHT file. ");
|
|
end_section();
|
|
|
|
begin_section("Website code");
|
|
begin_persons();
|
|
add_person("Fredrik Wendel", "", "");
|
|
end_persons();
|
|
end_section();
|
|
|
|
# HACK!
|
|
$max_name_width = 16;
|
|
|
|
begin_section("Special thanks to");
|
|
|
|
add_paragraph(
|
|
"The LUA developers, for creating a nice compact script interpreter. ");
|
|
|
|
add_paragraph(
|
|
"Tim Schafer, for obvious reasons, and everybody else who helped make ".
|
|
"Grim Fandango a brilliant game; and the EMI team for giving it their ".
|
|
"best try.");
|
|
|
|
add_paragraph(
|
|
"Bret Mogilefsky, for managing to create a SPUTM-style 3D LUA engine, ".
|
|
"and avoiding the horrible hack it could have been.");
|
|
|
|
end_section();
|
|
|
|
end_credits();
|