2007-01-25 23:36:38 +01:00
|
|
|
/**
|
2007-03-22 22:50:20 +01:00
|
|
|
* vim: set ts=4 :
|
2007-08-15 08:19:30 +02:00
|
|
|
* =============================================================================
|
2007-08-01 04:12:47 +02:00
|
|
|
* SourceMod
|
|
|
|
* Copyright (C) 2004-2007 AlliedModders LLC. All rights reserved.
|
2007-08-15 08:19:30 +02:00
|
|
|
* =============================================================================
|
2007-01-25 23:36:38 +01:00
|
|
|
*
|
2007-08-15 08:19:30 +02:00
|
|
|
* This program is free software; you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU General Public License, version 3.0, as published by the
|
|
|
|
* Free Software Foundation.
|
2007-08-01 04:12:47 +02:00
|
|
|
*
|
2007-08-15 08:19:30 +02:00
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
|
|
* details.
|
2007-08-01 04:12:47 +02:00
|
|
|
*
|
2007-08-15 08:19:30 +02:00
|
|
|
* You should have received a copy of the GNU General Public License along with
|
|
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
2007-08-01 04:12:47 +02:00
|
|
|
*
|
2007-08-15 08:19:30 +02:00
|
|
|
* As a special exception, AlliedModders LLC gives you permission to link the
|
|
|
|
* code of this program (as well as its derivative works) to "Half-Life 2," the
|
|
|
|
* "Source Engine," the "SourcePawn JIT," and any Game MODs that run on software
|
|
|
|
* by the Valve Corporation. You must obey the GNU General Public License in
|
|
|
|
* all respects for all other code used. Additionally, AlliedModders LLC grants
|
|
|
|
* this exception to all derivative works. AlliedModders LLC defines further
|
|
|
|
* exceptions, found in LICENSE.txt (as of this writing, version JULY-31-2007),
|
|
|
|
* or <http://www.sourcemod.net/license.php>.
|
2007-01-25 23:36:38 +01:00
|
|
|
*
|
|
|
|
* Version: $Id$
|
|
|
|
*/
|
|
|
|
|
2007-01-07 09:47:38 +01:00
|
|
|
#include <stdarg.h>
|
2007-01-16 21:36:09 +01:00
|
|
|
#include <stdlib.h>
|
2007-01-25 10:19:38 +01:00
|
|
|
#include <ctype.h>
|
2007-03-10 22:18:07 +01:00
|
|
|
#include "Translator.h"
|
2007-03-10 22:26:04 +01:00
|
|
|
#include "Logger.h"
|
2007-01-07 09:47:38 +01:00
|
|
|
#include "LibrarySys.h"
|
|
|
|
#include "sm_stringutil.h"
|
2007-01-16 21:36:09 +01:00
|
|
|
#include "sourcemod.h"
|
2007-08-19 15:25:42 +02:00
|
|
|
#include "PlayerManager.h"
|
2007-01-07 09:47:38 +01:00
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
Translator g_Translator;
|
2007-02-17 10:00:14 +01:00
|
|
|
CPhraseFile *g_pCorePhrases = NULL;
|
2007-10-21 22:06:30 +02:00
|
|
|
unsigned int g_pCorePhraseID = 0;
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
struct trans_t
|
|
|
|
{
|
|
|
|
int stridx;
|
|
|
|
int fmt_order;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct phrase_t
|
|
|
|
{
|
|
|
|
int fmt_list;
|
|
|
|
unsigned int fmt_count;
|
|
|
|
unsigned int fmt_bytes;
|
|
|
|
int trans_tbl;
|
|
|
|
unsigned int translations;
|
|
|
|
};
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
CPhraseFile::CPhraseFile(Translator *pTranslator, const char *file)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
m_pStringTab = pTranslator->GetStringTable();
|
|
|
|
m_pMemory = m_pStringTab->GetMemTable();
|
|
|
|
m_LangCount = pTranslator->GetLanguageCount();
|
|
|
|
m_File.assign(file);
|
|
|
|
m_pTranslator = pTranslator;
|
|
|
|
m_pPhraseLookup = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
CPhraseFile::~CPhraseFile()
|
|
|
|
{
|
|
|
|
if (m_pPhraseLookup)
|
|
|
|
{
|
|
|
|
sm_trie_destroy(m_pPhraseLookup);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CPhraseFile::ParseError(const char *message, ...)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, message);
|
|
|
|
vsnprintf(buffer, sizeof(buffer), message, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
m_ParseError.assign(buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CPhraseFile::ParseWarning(const char *message, ...)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, message);
|
|
|
|
vsnprintf(buffer, sizeof(buffer), message, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
if (!m_FileLogged)
|
|
|
|
{
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] Warning(s) encountered in translation file \"%s\"", m_File.c_str());
|
2007-01-20 03:51:30 +01:00
|
|
|
m_FileLogged = true;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] %s", message);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CPhraseFile::ReparseFile()
|
|
|
|
{
|
|
|
|
if (m_pPhraseLookup)
|
|
|
|
{
|
|
|
|
sm_trie_destroy(m_pPhraseLookup);
|
|
|
|
}
|
|
|
|
m_pPhraseLookup = sm_trie_create();
|
|
|
|
|
|
|
|
m_LangCount = m_pTranslator->GetLanguageCount();
|
|
|
|
|
|
|
|
if (!m_LangCount)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCError err;
|
2007-03-22 22:12:17 +01:00
|
|
|
char path[PLATFORM_MAX_PATH];
|
2007-01-19 03:14:50 +01:00
|
|
|
g_SourceMod.BuildPath(Path_SM, path, PLATFORM_MAX_PATH, "translations/%s", m_File.c_str());
|
2007-06-24 23:04:20 +02:00
|
|
|
|
2007-06-24 23:14:06 +02:00
|
|
|
//backwards compatibility shim
|
|
|
|
/* :HACKHACK: Change .cfg/.txt and vice versa for compatibility */
|
2007-06-24 23:04:20 +02:00
|
|
|
if (!g_LibSys.PathExists(path))
|
|
|
|
{
|
2007-06-24 23:14:06 +02:00
|
|
|
if (m_File.compare("common.cfg") == 0)
|
|
|
|
{
|
|
|
|
UTIL_ReplaceAll(path, sizeof(path), "common.cfg", "common.phrases.txt");
|
|
|
|
} else if (strstr(path, ".cfg")) {
|
|
|
|
UTIL_ReplaceAll(path, sizeof(path), ".cfg", ".txt");
|
|
|
|
} else if (strstr(path, ".txt")) {
|
|
|
|
UTIL_ReplaceAll(path, sizeof(path), ".txt", ".cfg");
|
|
|
|
}
|
2007-06-24 23:04:20 +02:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCStates states;
|
2007-01-07 09:47:38 +01:00
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
if ((err=textparsers->ParseFile_SMC(path, this, &states)) != SMCError_Okay)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-09-11 17:10:18 +02:00
|
|
|
const char *msg = textparsers->GetSMCErrorString(err);
|
2007-01-07 09:47:38 +01:00
|
|
|
if (!msg)
|
|
|
|
{
|
|
|
|
msg = m_ParseError.c_str();
|
|
|
|
}
|
|
|
|
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] Fatal error encountered parsing translation file \"%s\"", m_File.c_str());
|
2007-10-31 06:04:07 +01:00
|
|
|
g_Logger.LogError("[SM] Error (line %d, column %d): %s", states.line, states.col, msg);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CPhraseFile::ReadSMC_ParseStart()
|
|
|
|
{
|
|
|
|
m_CurPhrase = -1;
|
|
|
|
m_ParseState = PPS_None;
|
|
|
|
m_FileLogged = false;
|
|
|
|
m_LastPhraseString.clear();
|
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult CPhraseFile::ReadSMC_NewSection(const SMCStates *states, const char *name)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
bool recognized = false;
|
|
|
|
if (m_ParseState == PPS_None)
|
|
|
|
{
|
|
|
|
if (strcmp(name, "Phrases") == 0)
|
|
|
|
{
|
|
|
|
m_ParseState = PPS_Phrases;
|
|
|
|
recognized = true;
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (m_ParseState == PPS_Phrases)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
m_ParseState = PPS_InPhrase;
|
|
|
|
recognized = true;
|
|
|
|
|
|
|
|
phrase_t *pPhrase;
|
|
|
|
m_CurPhrase = m_pMemory->CreateMem(sizeof(phrase_t), (void **)&pPhrase);
|
|
|
|
|
|
|
|
/* Create the reverse lookup */
|
|
|
|
if (!sm_trie_insert(m_pPhraseLookup, name, reinterpret_cast<void *>(m_CurPhrase)))
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Skipping duplicate phrase \"%s\" on line %d.", name, states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
/* :IDEA: prevent memory waste by seeking backwards in memtable? */
|
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
/* Initialize new phrase */
|
|
|
|
trans_t *pTrans;
|
|
|
|
|
|
|
|
pPhrase->fmt_count = 0;
|
|
|
|
pPhrase->fmt_list = -1;
|
2007-06-06 19:34:16 +02:00
|
|
|
|
|
|
|
int trans_tbl = m_pMemory->CreateMem(sizeof(trans_t) * m_LangCount, (void **)&pTrans);
|
|
|
|
pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
|
|
|
pPhrase->trans_tbl = trans_tbl;
|
|
|
|
|
2007-01-07 09:47:38 +01:00
|
|
|
pPhrase->translations = 0;
|
|
|
|
pPhrase->fmt_bytes = 0;
|
|
|
|
|
|
|
|
for (unsigned int i=0; i<m_LangCount; i++)
|
|
|
|
{
|
|
|
|
pTrans[i].stridx = -1;
|
|
|
|
}
|
|
|
|
m_LastPhraseString.assign(name);
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (m_ParseState == PPS_InPhrase)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
ParseError("Phrase sections may not have sub-sections");
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_HaltFail;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!recognized)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Ignoring invalid section \"%s\" on line %d.", name, states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult CPhraseFile::ReadSMC_KeyValue(const SMCStates *states, const char *key, const char *value)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
/* See if we are ignoring a phrase */
|
2007-06-15 03:58:13 +02:00
|
|
|
if (m_CurPhrase == -1)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
phrase_t *pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
|
|
|
|
|
|
|
if (key[0] == '#' && strcmp(key, "#format") == 0)
|
|
|
|
{
|
|
|
|
if (pPhrase->fmt_list != -1)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Ignoring duplicated #format property on line %d", states->line);
|
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pPhrase->translations > 0)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("#format property should come before translations on line %d, ignoring", states->line);
|
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Do an initial browsing for error checking and what not */
|
|
|
|
enum ParseStates
|
|
|
|
{
|
|
|
|
Parse_None,
|
|
|
|
Parse_Index,
|
|
|
|
Parse_Format,
|
|
|
|
};
|
|
|
|
|
|
|
|
ParseStates state = Parse_None;
|
|
|
|
|
|
|
|
const char *old_value = value;
|
2007-06-15 03:58:13 +02:00
|
|
|
const char *last_value = value;
|
2007-01-07 09:47:38 +01:00
|
|
|
while (*value != '\0')
|
|
|
|
{
|
|
|
|
if (state == Parse_None)
|
|
|
|
{
|
|
|
|
if (*value == '{')
|
|
|
|
{
|
|
|
|
pPhrase->fmt_count++;
|
|
|
|
state = Parse_Index;
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (*value == ',')
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
/* Do nothing */
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-09-11 17:10:18 +02:00
|
|
|
unsigned int bytes = textparsers->GetUTF8CharBytes(value);
|
2007-01-07 09:47:38 +01:00
|
|
|
if (bytes != 1 || !isalpha(*value))
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Invalid token '%c' in #format property on line %d.", *value, states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (state == Parse_Index)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
if (*value == ':')
|
|
|
|
{
|
|
|
|
state = Parse_Format;
|
2007-06-15 03:58:13 +02:00
|
|
|
if (value - last_value >= 15)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Too many digits in format index on line %d, phrase will be ignored.", states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-09-11 17:10:18 +02:00
|
|
|
unsigned int bytes = textparsers->GetUTF8CharBytes(value);
|
2007-01-07 09:47:38 +01:00
|
|
|
if (bytes != 1 || !isdigit(*value))
|
|
|
|
{
|
|
|
|
ParseWarning("Token '%c' in #format property on line %d is not a digit, phrase will be ignored.",
|
|
|
|
*value,
|
2007-10-31 06:04:07 +01:00
|
|
|
states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (state == Parse_Format)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
if (*value == '}')
|
|
|
|
{
|
|
|
|
state = Parse_None;
|
2007-06-15 03:58:13 +02:00
|
|
|
last_value = value + 1;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
value++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state != Parse_None)
|
|
|
|
{
|
|
|
|
/* Moose clam cow. */
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Unterminated format string on line %d, phrase will be ignored.", states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
value = old_value;
|
|
|
|
|
|
|
|
/* Allocate the format table */
|
|
|
|
char fmt_buf[16];
|
|
|
|
int *fmt_list;
|
2007-07-19 04:08:42 +02:00
|
|
|
int tmp = m_pMemory->CreateMem(sizeof(int) * pPhrase->fmt_count, (void **)&fmt_list);
|
|
|
|
|
|
|
|
/* Update the phrase pointer in case it changed */
|
|
|
|
pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
|
|
|
pPhrase->fmt_list = tmp;
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
/* Initialize */
|
|
|
|
for (size_t i=0; i<pPhrase->fmt_count; i++)
|
|
|
|
{
|
|
|
|
fmt_list[i] = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now we need to read again... this time into the format buffer */
|
|
|
|
const char *in_ptr = value;
|
|
|
|
const char *idx_ptr = NULL;
|
|
|
|
char *out_ptr = NULL;
|
|
|
|
unsigned int cur_idx = 0;
|
|
|
|
state = Parse_None;
|
|
|
|
while (*in_ptr != '\0')
|
|
|
|
{
|
|
|
|
if (state == Parse_None)
|
|
|
|
{
|
|
|
|
if (*in_ptr == '{')
|
|
|
|
{
|
|
|
|
state = Parse_Index;
|
|
|
|
idx_ptr = NULL;
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (state == Parse_Index)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
if (*in_ptr == ':')
|
|
|
|
{
|
|
|
|
/* Check the number! */
|
|
|
|
if (!idx_ptr)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Format property contains unindexed format string on line %d, phrase will be ignored.", states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
long idx = strtol(idx_ptr, NULL, 10);
|
|
|
|
if (idx < 1 || idx > (long)pPhrase->fmt_count)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Format property contains invalid index '%d' on line %d, phrase will be ignored.", idx, states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
|
|
|
}
|
|
|
|
else if (fmt_list[idx - 1] != -1)
|
|
|
|
{
|
|
|
|
ParseWarning("Format property contains duplicated index '%d' on line %d, phrase will be ignored.", idx, states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
cur_idx = (unsigned int)idx;
|
|
|
|
state = Parse_Format;
|
|
|
|
out_ptr = NULL;
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (!idx_ptr)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
idx_ptr = in_ptr;
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (state == Parse_Format)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
if (*in_ptr == '}')
|
|
|
|
{
|
|
|
|
if (!out_ptr)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Format property contains empty format string on line %d, phrase will be ignored.", states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
*out_ptr = '\0';
|
|
|
|
state = Parse_None;
|
|
|
|
/* Now, add this to our table */
|
2007-07-19 05:22:10 +02:00
|
|
|
int tmp_idx = m_pStringTab->AddString(fmt_buf);
|
|
|
|
/* Update pointers and update necessary variables */
|
2007-07-19 04:08:42 +02:00
|
|
|
pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
2007-01-07 09:47:38 +01:00
|
|
|
pPhrase->fmt_bytes += strlen(fmt_buf);
|
2007-07-19 05:22:10 +02:00
|
|
|
fmt_list = (int *)m_pMemory->GetAddress(pPhrase->fmt_list);
|
|
|
|
fmt_list[cur_idx - 1] = tmp_idx;
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
if (!out_ptr)
|
|
|
|
{
|
|
|
|
out_ptr = fmt_buf;
|
|
|
|
*out_ptr++ = '%';
|
|
|
|
}
|
|
|
|
/* Check length ... */
|
2007-01-25 10:19:38 +01:00
|
|
|
if ((unsigned)(out_ptr - fmt_buf) >= sizeof(fmt_buf) - 1)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
ParseWarning("Format property contains format string that exceeds maximum length on line %d, phrase will be ignored.",
|
2007-10-31 06:04:07 +01:00
|
|
|
states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
*out_ptr++ = *in_ptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
in_ptr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we've gotten here, we only accepted unique indexes in a range.
|
|
|
|
* Therefore, the list has to be completely filled. Double check anyway.
|
|
|
|
*/
|
|
|
|
for (size_t i=0; i<pPhrase->fmt_count; i++)
|
|
|
|
{
|
|
|
|
if (fmt_list[i] == -1)
|
|
|
|
{
|
|
|
|
ParseWarning("Format property contains no string for index %d on line %d, phrase will be ignored.",
|
|
|
|
i + 1,
|
2007-10-31 06:04:07 +01:00
|
|
|
states->line);
|
2007-01-07 09:47:38 +01:00
|
|
|
m_CurPhrase = -1;
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
}
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
size_t len = strlen(key);
|
|
|
|
if (len != 2)
|
|
|
|
{
|
2007-10-31 06:04:07 +01:00
|
|
|
ParseWarning("Ignoring translation to invalid language \"%s\" on line %d.", key, states->line);
|
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int lang;
|
|
|
|
if (!m_pTranslator->GetLanguageByCode(key, &lang))
|
|
|
|
{
|
|
|
|
/* Ignore if we don't have a language.
|
|
|
|
* :IDEA: issue a one-time alert?
|
|
|
|
*/
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* See how many bytes we need for this string, then allocate.
|
|
|
|
* NOTE: THIS SHOULD GUARANTEE THAT WE DO NOT NEED TO NEED TO SIZE CHECK
|
|
|
|
*/
|
|
|
|
len = strlen(value) + pPhrase->fmt_bytes + 1;
|
|
|
|
char *out_buf;
|
|
|
|
int out_idx;
|
|
|
|
|
|
|
|
out_idx = m_pMemory->CreateMem(len, (void **)&out_buf);
|
|
|
|
|
2007-07-19 04:08:42 +02:00
|
|
|
/* Update pointer */
|
|
|
|
pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
|
|
|
|
2007-01-07 09:47:38 +01:00
|
|
|
int *fmt_order;
|
|
|
|
int *fmt_list = (int *)m_pMemory->GetAddress(pPhrase->fmt_list);
|
|
|
|
trans_t *pTrans = (trans_t *)m_pMemory->GetAddress(pPhrase->trans_tbl);
|
|
|
|
|
|
|
|
pTrans = &pTrans[lang];
|
|
|
|
pTrans->stridx = out_idx;
|
|
|
|
|
|
|
|
bool params[MAX_TRANSLATE_PARAMS];
|
|
|
|
|
|
|
|
/* Build the format order list, if necessary */
|
|
|
|
if (fmt_list)
|
|
|
|
{
|
2007-07-19 04:08:42 +02:00
|
|
|
int tmp = m_pMemory->CreateMem(pPhrase->fmt_count * sizeof(int), (void **)&fmt_order);
|
|
|
|
|
|
|
|
/* Update pointers */
|
|
|
|
pPhrase = (phrase_t *)m_pMemory->GetAddress(m_CurPhrase);
|
|
|
|
pTrans = (trans_t *)m_pMemory->GetAddress(pPhrase->trans_tbl);
|
|
|
|
fmt_list = (int *)m_pMemory->GetAddress(pPhrase->fmt_list);
|
2007-10-22 01:50:37 +02:00
|
|
|
out_buf = (char *)m_pMemory->GetAddress(out_idx);
|
2007-07-19 04:08:42 +02:00
|
|
|
pTrans = &pTrans[lang];
|
|
|
|
|
|
|
|
/* Now it's safe to save the index */
|
|
|
|
pTrans->fmt_order = tmp;
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
for (unsigned int i=0; i<pPhrase->fmt_count; i++)
|
|
|
|
{
|
|
|
|
fmt_order[i] = -1;
|
|
|
|
}
|
|
|
|
memset(¶ms[0], 0, sizeof(params));
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *in_ptr = value;
|
|
|
|
char *out_ptr = out_buf;
|
|
|
|
unsigned int order_idx = 0;
|
|
|
|
while (*in_ptr != '\0')
|
|
|
|
{
|
|
|
|
if (*in_ptr == '\\')
|
|
|
|
{
|
|
|
|
switch (*(in_ptr + 1))
|
|
|
|
{
|
|
|
|
case '\n':
|
|
|
|
*out_ptr++ = '\n';
|
|
|
|
break;
|
|
|
|
case '\t':
|
|
|
|
*out_ptr++ = '\t';
|
|
|
|
break;
|
|
|
|
case '\r':
|
|
|
|
*out_ptr++ = '\r';
|
|
|
|
break;
|
|
|
|
case '{':
|
|
|
|
*out_ptr++ = '{';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* Copy both bytes since we don't know what's going on */
|
|
|
|
*out_ptr++ = *in_ptr++;
|
|
|
|
*out_ptr++ = *in_ptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Skip past the last byte read */
|
|
|
|
in_ptr++;
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (*in_ptr == '{' && fmt_list != NULL)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
/* Search for parameters if this is a formatted string */
|
|
|
|
const char *scrap_in_point = in_ptr;
|
|
|
|
const char *digit_start = ++in_ptr;
|
|
|
|
unsigned int bytes;
|
|
|
|
while (*in_ptr != '\0')
|
|
|
|
{
|
2007-09-11 17:10:18 +02:00
|
|
|
bytes = textparsers->GetUTF8CharBytes(in_ptr);
|
2007-01-07 09:47:38 +01:00
|
|
|
if (bytes != 1)
|
|
|
|
{
|
|
|
|
goto scrap_point;
|
|
|
|
}
|
|
|
|
if (*in_ptr == '}')
|
|
|
|
{
|
|
|
|
/* Did we get an index? */
|
|
|
|
if (in_ptr == digit_start)
|
|
|
|
{
|
|
|
|
goto scrap_point;
|
|
|
|
}
|
|
|
|
/* Is it valid? */
|
|
|
|
long idx = strtol(digit_start, NULL, 10);
|
|
|
|
if (idx < 1 || idx > (int)pPhrase->fmt_count)
|
|
|
|
{
|
|
|
|
goto scrap_point;
|
|
|
|
}
|
|
|
|
if (params[idx-1])
|
|
|
|
{
|
|
|
|
goto scrap_point;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We're safe to insert the string. First mark order. */
|
|
|
|
fmt_order[order_idx++] = (int)idx - 1;
|
|
|
|
/* Now concatenate */
|
|
|
|
out_ptr += sprintf(out_ptr, "%s", m_pStringTab->GetString(fmt_list[idx-1]));
|
|
|
|
/* Mark as used */
|
|
|
|
params[idx-1] = true;
|
|
|
|
|
|
|
|
goto cont_loop;
|
|
|
|
}
|
|
|
|
in_ptr++;
|
|
|
|
}
|
|
|
|
scrap_point:
|
|
|
|
/* Pretend none of this ever happened. Move along now! */
|
|
|
|
in_ptr = scrap_in_point;
|
|
|
|
*out_ptr++ = *in_ptr;
|
|
|
|
} else {
|
|
|
|
*out_ptr++ = *in_ptr;
|
|
|
|
}
|
|
|
|
cont_loop:
|
|
|
|
in_ptr++;
|
|
|
|
}
|
|
|
|
*out_ptr = '\0';
|
|
|
|
pPhrase->translations++;
|
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult CPhraseFile::ReadSMC_LeavingSection(const SMCStates *states)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
if (m_ParseState == PPS_InPhrase)
|
|
|
|
{
|
|
|
|
if (m_CurPhrase == -1 && m_LastPhraseString.size())
|
|
|
|
{
|
|
|
|
sm_trie_delete(m_pPhraseLookup, m_LastPhraseString.c_str());
|
|
|
|
}
|
|
|
|
m_CurPhrase = -1;
|
|
|
|
m_ParseState = PPS_Phrases;
|
|
|
|
m_LastPhraseString.assign("");
|
2007-10-31 06:04:07 +01:00
|
|
|
}
|
|
|
|
else if (m_ParseState == PPS_Phrases)
|
|
|
|
{
|
2007-01-07 09:47:38 +01:00
|
|
|
m_ParseState = PPS_None;
|
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CPhraseFile::ReadSMC_ParseEnd(bool halted, bool failed)
|
|
|
|
{
|
|
|
|
/* Check to see if we have any dangling phrases that weren't completed, and scrap them */
|
|
|
|
if ((halted || failed) && m_LastPhraseString.size())
|
|
|
|
{
|
|
|
|
sm_trie_delete(m_pPhraseLookup, m_LastPhraseString.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-26 23:37:54 +01:00
|
|
|
TransError CPhraseFile::GetTranslation(const char *szPhrase, unsigned int lang_id, Translation *pTrans)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
if (lang_id >= m_LangCount)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
return Trans_BadLanguage;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void *object;
|
2007-01-26 23:37:54 +01:00
|
|
|
if (!sm_trie_retrieve(m_pPhraseLookup, szPhrase, &object))
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
return Trans_BadPhrase;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
phrase_t *pPhrase = (phrase_t *)m_pMemory->GetAddress(reinterpret_cast<int>(object));
|
|
|
|
trans_t *trans = (trans_t *)m_pMemory->GetAddress(pPhrase->trans_tbl);
|
|
|
|
|
2007-01-26 23:37:54 +01:00
|
|
|
trans = &trans[lang_id];
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
if (trans->stridx == -1)
|
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
return Trans_BadPhraseLanguage;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-01-26 23:37:54 +01:00
|
|
|
pTrans->fmt_count = pPhrase->fmt_count;
|
2007-10-31 06:04:07 +01:00
|
|
|
pTrans->fmt_order = pTrans->fmt_count > 0 ? (int *)m_pMemory->GetAddress(trans->fmt_order) : NULL;
|
2007-01-26 23:37:54 +01:00
|
|
|
pTrans->szPhrase = m_pStringTab->GetString(trans->stridx);
|
|
|
|
|
|
|
|
return Trans_Okay;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *CPhraseFile::GetFilename()
|
|
|
|
{
|
|
|
|
return m_File.c_str();
|
|
|
|
}
|
|
|
|
|
2007-02-17 10:00:14 +01:00
|
|
|
/**************************
|
|
|
|
** MAIN TRANSLATOR CODE **
|
|
|
|
**************************/
|
|
|
|
|
2007-04-05 12:55:40 +02:00
|
|
|
Translator::Translator() : m_ServerLang(LANGUAGE_ENGLISH)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
m_pStringTab = new BaseStringTable(2048);
|
|
|
|
m_pLCodeLookup = sm_trie_create();
|
2007-04-05 12:55:40 +02:00
|
|
|
strncopy(m_InitialLang, "en", sizeof(m_InitialLang));
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
Translator::~Translator()
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
for (size_t i=0; i<m_Files.size(); i++)
|
|
|
|
{
|
|
|
|
delete m_Files[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i=0; i<m_Languages.size(); i++)
|
|
|
|
{
|
|
|
|
delete m_Languages[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
sm_trie_destroy(m_pLCodeLookup);
|
|
|
|
|
|
|
|
delete m_pStringTab;
|
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
ConfigResult Translator::OnSourceModConfigChanged(const char *key,
|
|
|
|
const char *value,
|
|
|
|
ConfigSource source,
|
|
|
|
char *error,
|
|
|
|
size_t maxlength)
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 07:45:35 +02:00
|
|
|
if (strcasecmp(key, "ServerLang") == 0)
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 07:25:11 +02:00
|
|
|
if (source == ConfigSource_Console)
|
|
|
|
{
|
|
|
|
unsigned int index;
|
|
|
|
if (!GetLanguageByCode(value, &index))
|
|
|
|
{
|
|
|
|
UTIL_Format(error, maxlength, "Language code \"%s\" is not registered", value);
|
|
|
|
return ConfigResult_Reject;
|
|
|
|
}
|
|
|
|
|
2007-04-05 12:55:40 +02:00
|
|
|
m_ServerLang = index;
|
|
|
|
} else {
|
|
|
|
strncopy(m_InitialLang, value, sizeof(m_InitialLang));
|
|
|
|
}
|
2007-04-05 07:25:11 +02:00
|
|
|
|
|
|
|
return ConfigResult_Accept;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-05 07:25:11 +02:00
|
|
|
return ConfigResult_Ignore;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
|
|
|
|
2007-04-07 05:58:20 +02:00
|
|
|
void Translator::OnSourceModLevelChange(const char *mapName)
|
|
|
|
{
|
|
|
|
/* Refresh language stuff */
|
|
|
|
char path[PLATFORM_MAX_PATH];
|
|
|
|
g_SourceMod.BuildPath(Path_SM, path, sizeof(path), "configs/languages.cfg");
|
|
|
|
RebuildLanguageDatabase(path);
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Translator::OnSourceModAllInitialized()
|
2007-02-17 10:00:14 +01:00
|
|
|
{
|
2007-02-17 20:03:18 +01:00
|
|
|
AddLanguage("en", "English");
|
2007-06-24 23:04:20 +02:00
|
|
|
|
|
|
|
unsigned int id;
|
|
|
|
|
2007-12-04 16:58:07 +01:00
|
|
|
id = FindOrAddPhraseFile("core.phrases.txt");
|
2007-10-21 22:06:30 +02:00
|
|
|
g_pCorePhraseID = id;
|
2007-02-17 10:00:14 +01:00
|
|
|
g_pCorePhrases = GetFileByIndex(id);
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
bool Translator::GetLanguageByCode(const char *code, unsigned int *index)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
void *_index;
|
|
|
|
|
|
|
|
if (!sm_trie_retrieve(m_pLCodeLookup, code, &_index))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index)
|
|
|
|
{
|
|
|
|
*index = reinterpret_cast<unsigned int>(_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-08-19 15:25:42 +02:00
|
|
|
bool Translator::GetLanguageByName(const char *name, unsigned int *index)
|
|
|
|
{
|
|
|
|
CVector<Language *>::iterator iter;
|
|
|
|
unsigned int id = 0;
|
|
|
|
|
|
|
|
for (iter=m_Languages.begin(); iter!=m_Languages.end(); iter++, id++)
|
|
|
|
{
|
|
|
|
if (strcasecmp(m_pStringTab->GetString((*iter)->m_FullName), name) == 0)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iter == m_Languages.end())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index)
|
|
|
|
{
|
|
|
|
*index = id;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
unsigned int Translator::GetLanguageCount()
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
return (unsigned int)m_Languages.size();
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
BaseStringTable *Translator::GetStringTable()
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
return m_pStringTab;
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
unsigned int Translator::FindOrAddPhraseFile(const char *phrase_file)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
for (size_t i=0; i<m_Files.size(); i++)
|
|
|
|
{
|
|
|
|
if (strcmp(m_Files[i]->GetFilename(), phrase_file) == 0)
|
|
|
|
{
|
|
|
|
return (unsigned int)i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CPhraseFile *pFile = new CPhraseFile(this, phrase_file);
|
|
|
|
unsigned int idx = (unsigned int)m_Files.size();
|
|
|
|
|
|
|
|
m_Files.push_back(pFile);
|
|
|
|
|
|
|
|
pFile->ReparseFile();
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Translator::RebuildLanguageDatabase(const char *lang_header_file)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
/* Erase everything we have */
|
|
|
|
sm_trie_destroy(m_pLCodeLookup);
|
|
|
|
m_pLCodeLookup = sm_trie_create();
|
|
|
|
m_pStringTab->Reset();
|
|
|
|
|
|
|
|
for (size_t i=0; i<m_Languages.size(); i++)
|
|
|
|
{
|
|
|
|
delete m_Languages[i];
|
|
|
|
}
|
|
|
|
m_Languages.clear();
|
|
|
|
|
|
|
|
/* Start anew */
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCError err;
|
|
|
|
SMCStates states;
|
|
|
|
if ((err=textparsers->ParseFile_SMC(lang_header_file, this, &states)) != SMCError_Okay)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-09-11 17:10:18 +02:00
|
|
|
const char *str_err = textparsers->GetSMCErrorString(err);
|
2007-01-07 09:47:38 +01:00
|
|
|
if (!str_err)
|
|
|
|
{
|
|
|
|
str_err = m_CustomError.c_str();
|
|
|
|
}
|
|
|
|
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] Failed to parse language header file: \"%s\"", lang_header_file);
|
2007-10-31 06:04:07 +01:00
|
|
|
g_Logger.LogError("[SM] Parse error (line %d, column %d): %s", states.line, states.col, str_err);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-04-05 12:55:40 +02:00
|
|
|
void *serverLang;
|
|
|
|
|
|
|
|
if (!sm_trie_retrieve(m_pLCodeLookup, m_InitialLang, &serverLang))
|
|
|
|
{
|
|
|
|
g_Logger.LogError("Server language was set to bad language \"%s\" -- reverting to English", m_InitialLang);
|
|
|
|
|
|
|
|
strncopy(m_InitialLang, "en", sizeof(m_InitialLang));
|
|
|
|
m_ServerLang = LANGUAGE_ENGLISH;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_ServerLang = reinterpret_cast<unsigned int>(serverLang);
|
|
|
|
|
2007-01-07 09:47:38 +01:00
|
|
|
if (!m_Languages.size())
|
|
|
|
{
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] Fatal error, no languages found! Translation will not work.");
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i=0; i<m_Files.size(); i++)
|
|
|
|
{
|
|
|
|
m_Files[i]->ReparseFile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
void Translator::ReadSMC_ParseStart()
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
m_InLanguageSection = false;
|
|
|
|
m_CustomError.clear();
|
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult Translator::ReadSMC_NewSection(const SMCStates *states, const char *name)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
if (!m_InLanguageSection)
|
|
|
|
{
|
|
|
|
if (strcmp(name, "Languages") == 0)
|
|
|
|
{
|
|
|
|
m_InLanguageSection = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!m_InLanguageSection)
|
|
|
|
{
|
2007-01-20 03:51:30 +01:00
|
|
|
g_Logger.LogError("[SM] Warning: Unrecognized section \"%s\" in languages.cfg", name);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult Translator::ReadSMC_LeavingSection(const SMCStates *states)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
m_InLanguageSection = false;
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
SMCResult Translator::ReadSMC_KeyValue(const SMCStates *states, const char *key, const char *value)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
|
|
|
size_t len = strlen(key);
|
|
|
|
|
|
|
|
if (len != 2)
|
|
|
|
{
|
2007-01-17 07:48:52 +01:00
|
|
|
g_Logger.LogError("[SM] Warning encountered parsing languages.cfg file.");
|
2007-01-20 03:51:30 +01:00
|
|
|
g_Logger.LogError("[SM] Invalid language code \"%s\" is being ignored.", key);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-02-17 20:03:18 +01:00
|
|
|
AddLanguage(key, value);
|
|
|
|
|
2007-10-31 06:04:07 +01:00
|
|
|
return SMCResult_Continue;
|
2007-02-17 20:03:18 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
bool Translator::AddLanguage(const char *langcode, const char *description)
|
2007-02-17 20:03:18 +01:00
|
|
|
{
|
|
|
|
if (sm_trie_retrieve(m_pLCodeLookup, langcode, NULL))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-01-07 09:47:38 +01:00
|
|
|
Language *pLanguage = new Language;
|
|
|
|
unsigned int idx = m_Languages.size();
|
|
|
|
|
2007-02-17 20:03:18 +01:00
|
|
|
pLanguage->m_code2[0] = langcode[0];
|
|
|
|
pLanguage->m_code2[1] = langcode[1];
|
|
|
|
pLanguage->m_code2[2] = langcode[2];
|
|
|
|
pLanguage->m_FullName = m_pStringTab->AddString(description);
|
2007-01-07 09:47:38 +01:00
|
|
|
|
2007-02-17 20:03:18 +01:00
|
|
|
sm_trie_insert(m_pLCodeLookup, langcode, reinterpret_cast<void *>(idx));
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
m_Languages.push_back(pLanguage);
|
|
|
|
|
2007-02-17 20:03:18 +01:00
|
|
|
return true;
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
CPhraseFile *Translator::GetFileByIndex(unsigned int index)
|
2007-01-27 05:21:11 +01:00
|
|
|
{
|
|
|
|
if (index >= m_Files.size())
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_Files[index];
|
|
|
|
}
|
|
|
|
|
2007-03-15 05:45:17 +01:00
|
|
|
size_t Translator::Translate(char *buffer, size_t maxlength, void **params, const Translation *pTrans)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
void *new_params[MAX_TRANSLATE_PARAMS];
|
2007-01-07 09:47:38 +01:00
|
|
|
|
|
|
|
/* Rewrite the parameter order */
|
2007-01-26 23:37:54 +01:00
|
|
|
for (unsigned int i=0; i<pTrans->fmt_count; i++)
|
2007-01-07 09:47:38 +01:00
|
|
|
{
|
2007-01-26 23:37:54 +01:00
|
|
|
new_params[i] = params[pTrans->fmt_order[i]];
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
|
|
|
|
2007-01-26 23:37:54 +01:00
|
|
|
return gnprintf(buffer, maxlength, pTrans->szPhrase, new_params);
|
2007-01-07 09:47:38 +01:00
|
|
|
}
|
2007-02-17 20:03:18 +01:00
|
|
|
|
2008-03-12 04:33:52 +01:00
|
|
|
TransError Translator::CoreTransEx(CPhraseFile *pFile,
|
|
|
|
int client,
|
|
|
|
char *buffer,
|
|
|
|
size_t maxlength,
|
|
|
|
const char *phrase,
|
|
|
|
void **params,
|
|
|
|
size_t *outlen)
|
2007-02-17 20:03:18 +01:00
|
|
|
{
|
|
|
|
Translation trans;
|
|
|
|
TransError err;
|
2008-03-12 04:33:52 +01:00
|
|
|
|
2007-04-05 12:59:26 +02:00
|
|
|
/* Using server lang temporarily until client lang stuff is implemented */
|
2008-03-12 04:33:52 +01:00
|
|
|
if ((err = pFile->GetTranslation(phrase, m_ServerLang, &trans)) != Trans_Okay)
|
2007-02-17 20:03:18 +01:00
|
|
|
{
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t len = Translate(buffer, maxlength, params, &trans);
|
|
|
|
|
|
|
|
if (outlen)
|
|
|
|
{
|
|
|
|
*outlen = len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Trans_Okay;
|
|
|
|
}
|
2007-04-05 05:02:00 +02:00
|
|
|
|
2008-03-12 04:33:52 +01:00
|
|
|
TransError Translator::CoreTrans(int client,
|
|
|
|
char *buffer,
|
|
|
|
size_t maxlength,
|
|
|
|
const char *phrase,
|
|
|
|
void **params,
|
|
|
|
size_t *outlen)
|
|
|
|
{
|
|
|
|
if (!g_pCorePhrases)
|
|
|
|
{
|
|
|
|
return Trans_BadPhraseFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CoreTransEx(g_pCorePhrases, client, buffer, maxlength, phrase, params, outlen);
|
|
|
|
}
|
|
|
|
|
2007-04-05 12:55:40 +02:00
|
|
|
unsigned int Translator::GetServerLanguage()
|
2007-04-05 05:02:00 +02:00
|
|
|
{
|
2007-04-05 12:55:40 +02:00
|
|
|
return m_ServerLang;
|
2007-04-05 05:02:00 +02:00
|
|
|
}
|
2007-07-14 03:04:15 +02:00
|
|
|
|
|
|
|
unsigned int Translator::GetClientLanguage(int client)
|
|
|
|
{
|
2007-08-19 15:25:42 +02:00
|
|
|
CPlayer *pPlayer = g_Players.GetPlayerByIndex(client);
|
|
|
|
return pPlayer->GetLanguageId();
|
2007-07-14 03:04:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Translator::GetLanguageInfo(unsigned int number, const char **code, const char **name)
|
|
|
|
{
|
|
|
|
if (number >= GetLanguageCount())
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Language *l = m_Languages[number];
|
|
|
|
if (code)
|
|
|
|
{
|
|
|
|
*code = l->m_code2;
|
|
|
|
}
|
|
|
|
if (name)
|
|
|
|
{
|
|
|
|
*name = m_pStringTab->GetString(l->m_FullName);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|