mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
VDR developer version 1.7.30 is now available at ftp://ftp.tvdr.de/vdr/Developer/vdr-1.7.30.tar.bz2 A 'diff' against the previous version is available at ftp://ftp.tvdr.de/vdr/Developer/vdr-1.7.29-1.7.30.diff MD5 checksums: c6d75f2962bc3e22d9313c0ee4fa113a vdr-1.7.30.tar.bz2 a63098efcc58bc697d6b890097d9c370 vdr-1.7.29-1.7.30.diff WARNING: ======== This is a developer version. Even though I use it in my productive environment. I strongly recommend that you only use it under controlled conditions and for testing and debugging. The default skin "LCARS" displays the signal strengths and qualities of all devices in its main menu. For devices that have an stb0899 frontend chip (like the TT-budget S2-3200) retrieving this information from the driver is rather slow, which results in a sluggish response to user input in the main menu. To speed this up you may want to apply the patches from ftp://ftp.tvdr.de/vdr/Developer/Driver-Patches to the LinuxDVB driver source. From the HISTORY file: - Fixed sorting recordings in the top level video directory. - Fixed handling control characters in SI data in case of UTF-8 encoded strings (thanks to Mehdi Karamnejad for reporting a problem with garbled UTF-8 EPG data and helping to debug it). - Updated the Finnish OSD texts (thanks to Rolf Ahrenberg). - When checking whether a video directory is empty, file names that start with a dot ('.') are now ignored and will be implicitly removed if the directory contains no other files. This fixes the leftover ".sort" files that were introduced in version 1.7.29. - Added IsUpdate() to the EPG handler interface (thanks to Jörg Wendel). - Fixed detecting transfer mode on full featured DVB cards (thanks to Stefan Huelswitt for reporting a problem with updating CA descriptors in such cases). - Fixed a race condition when zapping in transfer mode (reported by Reinhard Nissl). This involves a slight change in the semantics of the cReceiver::Activate() function, which is now called with 'false' after the receiver has been detached from the device. - The new function cDevice::ReadFilter() can be used by devices to implement their own way of retrieving section filter data (thanks to Deti Fliegl). - The new function cDevice::HasInternalCam() can be implemented by devices that provide encrypted channels in an already decrypted form, without requiring explicit handling of a CAM (thanks to Tobias Grimm). - VDR can now be built according to the FHS ("File system Hierarchy Standard") by activating the line #USEFHS = 1 in a copy of the file Make.config.template (thanks to Dennis Bendlin, as well as Christopher Reimer and Udo Richter for contributing to the patch). - By default (without FHS support) the config directory is now set to the value given in the -v option if only -v and no -c is given. - Fixed a long delay at the end when replaying a recording that has stopped recording less than an hour ago (typically time shift mode or a freshly edited recording). - Fixed getting the file size and number of frames of ongoing recordings (only the timestamp of the recording's directory was checked, while it should have been that of the index file). - Fixed sluggish response when manipulating editing marks while a cutting thread is running (reported by Torsten Lang). - The new setup options "OSD/Color key [0123]" can be used to adjust the sequence of the color buttons displayed in the menus to that of the color keys on your remote control (based on a patch from Oliver Schinagl). Authors of plugins that implement skins may want to adapt their SetButtons() function in order to make use of this new feature. See, for instance, the function cSkinClassicDisplayMenu::SetButtons() in skinclassic.c for details.
301 lines
8.8 KiB
C
301 lines
8.8 KiB
C
/*
|
|
* i18n.c: Internationalization
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: i18n.c 2.5 2012/09/01 10:53:43 kls Exp $
|
|
*/
|
|
|
|
/*
|
|
* In case an English phrase is used in more than one context (and might need
|
|
* different translations in other languages) it can be preceded with an
|
|
* arbitrary string to describe its context, separated from the actual phrase
|
|
* by a '$' character (see for instance "Button$Stop" vs. "Stop").
|
|
* Of course this means that no English phrase may contain the '$' character!
|
|
* If this should ever become necessary, the existing '$' would have to be
|
|
* replaced with something different...
|
|
*/
|
|
|
|
#include "i18n.h"
|
|
#include <ctype.h>
|
|
#include <libintl.h>
|
|
#include <locale.h>
|
|
#include <unistd.h>
|
|
#include "tools.h"
|
|
|
|
// TRANSLATORS: The name of the language, as written natively
|
|
const char *LanguageName = trNOOP("LanguageName$English");
|
|
// TRANSLATORS: The 3-letter code of the language
|
|
const char *LanguageCode = trNOOP("LanguageCode$eng");
|
|
|
|
// List of known language codes with aliases.
|
|
// Actually we could list all codes from http://www.loc.gov/standards/iso639-2
|
|
// here, but that would be several hundreds - and for most of them it's unlikely
|
|
// they're ever going to be used...
|
|
|
|
const char *LanguageCodeList[] = {
|
|
"eng,dos",
|
|
"deu,ger",
|
|
"slv,slo",
|
|
"ita",
|
|
"dut,nla,nld",
|
|
"prt",
|
|
"fra,fre",
|
|
"nor",
|
|
"fin,suo",
|
|
"pol",
|
|
"esl,spa",
|
|
"ell,gre",
|
|
"sve,swe",
|
|
"rom,rum",
|
|
"hun",
|
|
"cat,cln",
|
|
"rus",
|
|
"srb,srp,scr,scc",
|
|
"hrv",
|
|
"est",
|
|
"dan",
|
|
"cze,ces",
|
|
"tur",
|
|
"ukr",
|
|
"ara",
|
|
NULL
|
|
};
|
|
|
|
static cString I18nLocaleDir;
|
|
|
|
static cStringList LanguageLocales;
|
|
static cStringList LanguageNames;
|
|
static cStringList LanguageCodes;
|
|
|
|
static int NumLocales = 1;
|
|
static int CurrentLanguage = 0;
|
|
|
|
static bool ContainsCode(const char *Codes, const char *Code)
|
|
{
|
|
while (*Codes) {
|
|
int l = 0;
|
|
for ( ; l < 3 && Code[l]; l++) {
|
|
if (Codes[l] != tolower(Code[l]))
|
|
break;
|
|
}
|
|
if (l == 3)
|
|
return true;
|
|
Codes++;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static const char *SkipContext(const char *s)
|
|
{
|
|
const char *p = strchr(s, '$');
|
|
return p ? p + 1 : s;
|
|
}
|
|
|
|
static void SetEnvLanguage(const char *Locale)
|
|
{
|
|
setenv("LANGUAGE", Locale, 1);
|
|
extern int _nl_msg_cat_cntr;
|
|
++_nl_msg_cat_cntr;
|
|
}
|
|
|
|
void I18nInitialize(const char *LocaleDir)
|
|
{
|
|
I18nLocaleDir = LocaleDir;
|
|
LanguageLocales.Append(strdup(I18N_DEFAULT_LOCALE));
|
|
LanguageNames.Append(strdup(SkipContext(LanguageName)));
|
|
LanguageCodes.Append(strdup(LanguageCodeList[0]));
|
|
textdomain("vdr");
|
|
bindtextdomain("vdr", I18nLocaleDir);
|
|
cFileNameList Locales(I18nLocaleDir, true);
|
|
if (Locales.Size() > 0) {
|
|
char *OldLocale = strdup(setlocale(LC_MESSAGES, NULL));
|
|
for (int i = 0; i < Locales.Size(); i++) {
|
|
cString FileName = cString::sprintf("%s/%s/LC_MESSAGES/vdr.mo", *I18nLocaleDir, Locales[i]);
|
|
if (access(FileName, F_OK) == 0) { // found a locale with VDR texts
|
|
if (NumLocales < I18N_MAX_LANGUAGES - 1) {
|
|
SetEnvLanguage(Locales[i]);
|
|
const char *TranslatedLanguageName = gettext(LanguageName);
|
|
if (TranslatedLanguageName != LanguageName) {
|
|
NumLocales++;
|
|
if (strstr(OldLocale, Locales[i]) == OldLocale)
|
|
CurrentLanguage = LanguageLocales.Size();
|
|
LanguageLocales.Append(strdup(Locales[i]));
|
|
LanguageNames.Append(strdup(TranslatedLanguageName));
|
|
const char *Code = gettext(LanguageCode);
|
|
for (const char **lc = LanguageCodeList; *lc; lc++) {
|
|
if (ContainsCode(*lc, Code)) {
|
|
Code = *lc;
|
|
break;
|
|
}
|
|
}
|
|
LanguageCodes.Append(strdup(Code));
|
|
}
|
|
}
|
|
else {
|
|
esyslog("ERROR: too many locales - increase I18N_MAX_LANGUAGES!");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
SetEnvLanguage(LanguageLocales[CurrentLanguage]);
|
|
free(OldLocale);
|
|
dsyslog("found %d locales in %s", NumLocales - 1, *I18nLocaleDir);
|
|
}
|
|
// Prepare any known language codes for which there was no locale:
|
|
for (const char **lc = LanguageCodeList; *lc; lc++) {
|
|
bool Found = false;
|
|
for (int i = 0; i < LanguageCodes.Size(); i++) {
|
|
if (strcmp(*lc, LanguageCodes[i]) == 0) {
|
|
Found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!Found) {
|
|
dsyslog("no locale for language code '%s'", *lc);
|
|
LanguageLocales.Append(strdup(I18N_DEFAULT_LOCALE));
|
|
LanguageNames.Append(strdup(*lc));
|
|
LanguageCodes.Append(strdup(*lc));
|
|
}
|
|
}
|
|
}
|
|
|
|
void I18nRegister(const char *Plugin)
|
|
{
|
|
cString Domain = cString::sprintf("vdr-%s", Plugin);
|
|
bindtextdomain(Domain, I18nLocaleDir);
|
|
}
|
|
|
|
void I18nSetLocale(const char *Locale)
|
|
{
|
|
if (Locale && *Locale) {
|
|
int i = LanguageLocales.Find(Locale);
|
|
if (i >= 0) {
|
|
CurrentLanguage = i;
|
|
SetEnvLanguage(Locale);
|
|
}
|
|
else
|
|
dsyslog("unknown locale: '%s'", Locale);
|
|
}
|
|
}
|
|
|
|
int I18nCurrentLanguage(void)
|
|
{
|
|
return CurrentLanguage;
|
|
}
|
|
|
|
void I18nSetLanguage(int Language)
|
|
{
|
|
if (Language < LanguageNames.Size()) {
|
|
CurrentLanguage = Language;
|
|
I18nSetLocale(I18nLocale(CurrentLanguage));
|
|
}
|
|
}
|
|
|
|
int I18nNumLanguagesWithLocale(void)
|
|
{
|
|
return NumLocales;
|
|
}
|
|
|
|
const cStringList *I18nLanguages(void)
|
|
{
|
|
return &LanguageNames;
|
|
}
|
|
|
|
const char *I18nTranslate(const char *s, const char *Plugin)
|
|
{
|
|
if (!s)
|
|
return s;
|
|
if (CurrentLanguage) {
|
|
const char *t = Plugin ? dgettext(Plugin, s) : gettext(s);
|
|
if (t != s)
|
|
return t;
|
|
}
|
|
return SkipContext(s);
|
|
}
|
|
|
|
const char *I18nLocale(int Language)
|
|
{
|
|
return 0 <= Language && Language < LanguageLocales.Size() ? LanguageLocales[Language] : NULL;
|
|
}
|
|
|
|
const char *I18nLanguageCode(int Language)
|
|
{
|
|
return 0 <= Language && Language < LanguageCodes.Size() ? LanguageCodes[Language] : NULL;
|
|
}
|
|
|
|
int I18nLanguageIndex(const char *Code)
|
|
{
|
|
for (int i = 0; i < LanguageCodes.Size(); i++) {
|
|
if (ContainsCode(LanguageCodes[i], Code))
|
|
return i;
|
|
}
|
|
//dsyslog("unknown language code: '%s'", Code);
|
|
return -1;
|
|
}
|
|
|
|
const char *I18nNormalizeLanguageCode(const char *Code)
|
|
{
|
|
for (int i = 0; i < 3; i++) {
|
|
if (Code[i]) {
|
|
// ETSI EN 300 468 defines language codes as consisting of three letters
|
|
// according to ISO 639-2. This means that they are supposed to always consist
|
|
// of exactly three letters in the range a-z - no digits, UTF-8 or other
|
|
// funny characters. However, some broadcasters apparently don't have a
|
|
// copy of the DVB standard (or they do, but are perhaps unable to read it),
|
|
// so they put all sorts of non-standard stuff into the language codes,
|
|
// like nonsense as "2ch" or "A 1" (yes, they even go as far as using
|
|
// blanks!). Such things should go into the description of the EPG event's
|
|
// ComponentDescriptor.
|
|
// So, as a workaround for this broadcaster stupidity, let's ignore
|
|
// language codes with unprintable characters...
|
|
if (!isprint(Code[i])) {
|
|
//dsyslog("invalid language code: '%s'", Code);
|
|
return "???";
|
|
}
|
|
// ...and replace blanks with underlines (ok, this breaks the 'const'
|
|
// of the Code parameter - but hey, it's them who started this):
|
|
if (Code[i] == ' ')
|
|
*((char *)&Code[i]) = '_';
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
int n = I18nLanguageIndex(Code);
|
|
return n >= 0 ? I18nLanguageCode(n) : Code;
|
|
}
|
|
|
|
bool I18nIsPreferredLanguage(int *PreferredLanguages, const char *LanguageCode, int &OldPreference, int *Position)
|
|
{
|
|
int pos = 1;
|
|
bool found = false;
|
|
while (LanguageCode) {
|
|
int LanguageIndex = I18nLanguageIndex(LanguageCode);
|
|
for (int i = 0; i < LanguageCodes.Size(); i++) {
|
|
if (PreferredLanguages[i] < 0)
|
|
break; // the language is not a preferred one
|
|
if (PreferredLanguages[i] == LanguageIndex) {
|
|
if (OldPreference < 0 || i < OldPreference) {
|
|
OldPreference = i;
|
|
if (Position)
|
|
*Position = pos;
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if ((LanguageCode = strchr(LanguageCode, '+')) != NULL) {
|
|
LanguageCode++;
|
|
pos++;
|
|
}
|
|
else if (pos == 1 && Position)
|
|
*Position = 0;
|
|
}
|
|
if (OldPreference < 0) {
|
|
OldPreference = LanguageCodes.Size(); // higher than the maximum possible value
|
|
return true; // if we don't find a preferred one, we take the first one
|
|
}
|
|
return found;
|
|
}
|