2013-10-15 21:58:27 +00:00
|
|
|
/*
|
|
|
|
* Rufus: The Reliable USB Formatting Utility
|
|
|
|
* Localization functions, a.k.a. "Everybody is doing it wrong but me!"
|
|
|
|
* Copyright © 2013 Pete Batard <pete@akeo.ie>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Memory leaks detection - define _CRTDBG_MAP_ALLOC as preprocessor macro */
|
|
|
|
#ifdef _CRTDBG_MAP_ALLOC
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <crtdbg.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <wchar.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
#include "rufus.h"
|
|
|
|
#include "resource.h"
|
|
|
|
#include "msapi_utf8.h"
|
|
|
|
#include "localization.h"
|
|
|
|
#include "localization_data.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* List of supported locale commands, with their parameter syntax:
|
|
|
|
* c control ID (no space, no quotes)
|
|
|
|
* s: quoted string
|
|
|
|
* i: 32 bit signed integer
|
|
|
|
* u: 32 bit unsigned CSV list
|
|
|
|
* Remember to update the size of the array in localization.h when adding/removing elements
|
|
|
|
*/
|
|
|
|
const loc_parse parse_cmd[9] = {
|
|
|
|
// Translation name and Windows LCIDs it should apply to
|
|
|
|
{ 'l', LC_LOCALE, "ssu" }, // l "en_US" "English (US)" 0x0009,0x1009
|
|
|
|
// Base translation to add on top of (eg. "English (UK)" can be used to build on top of "English (US)"
|
|
|
|
{ 'b', LC_BASE, "s" }, // b "en_US"
|
|
|
|
// Version to use for the localization commandset and API
|
|
|
|
{ 'v', LC_VERSION, "ii" }, // v 1.0 // TODO: NOT IMPLEMENTED YET
|
|
|
|
// Translate the text control associated with an ID
|
|
|
|
{ 't', LC_TEXT, "cs" }, // t IDC_CONTROL "Translation"
|
|
|
|
// Set the section/dialog to which the next commands should apply
|
|
|
|
{ 'g', LC_GROUP, "c" }, // g IDD_DIALOG
|
|
|
|
// Resize a dialog (dx dy pixel increment)
|
|
|
|
{ 's', LC_SIZE, "cii" }, // s IDC_CONTROL +10 +10
|
|
|
|
// Move a dialog (dx dy pixed displacement)
|
|
|
|
{ 'm', LC_MOVE, "cii" }, // m IDC_CONTROL -5 0
|
|
|
|
// Set the font to use for the text controls that follow
|
|
|
|
// Use f "Default" 0 to reset the font
|
|
|
|
{ 'f', LC_FONT, "si" }, // f "MS Dialog" 10
|
|
|
|
// Set the direction to use for the text controls that follow
|
|
|
|
// 0 = Left to right, 1 = Right to left
|
|
|
|
{ 'd', LC_DIRECTION, "i" }, // d 1 // TODO: NOT IMPLEMENTED YET
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Globals */
|
|
|
|
int loc_line_nr;
|
|
|
|
struct list_head locale_list = {NULL, NULL};
|
|
|
|
char *loc_filename = NULL, *embedded_loc_filename = "[embedded] rufus.loc";
|
|
|
|
|
2013-10-18 22:08:49 +00:00
|
|
|
/*
|
|
|
|
* Hash table functions - modified From glibc 2.3.2:
|
|
|
|
* [Aho,Sethi,Ullman] Compilers: Principles, Techniques and Tools, 1986
|
|
|
|
* [Knuth] The Art of Computer Programming, part 3 (6.4)
|
|
|
|
*/
|
|
|
|
typedef struct htab_entry {
|
|
|
|
uint32_t used;
|
|
|
|
char* str;
|
|
|
|
loc_cmd* dlg_cmd;
|
|
|
|
} htab_entry;
|
|
|
|
htab_entry* htab_table = NULL;
|
|
|
|
size_t htab_size, htab_filled;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For the used double hash method the table size has to be a prime. To
|
|
|
|
* correct the user given table size we need a prime test. This trivial
|
|
|
|
* algorithm is adequate because the code is called only during init and
|
|
|
|
* the number is likely to be small
|
|
|
|
*/
|
|
|
|
static uint32_t isprime(uint32_t number)
|
|
|
|
{
|
|
|
|
// no even number will be passed
|
|
|
|
uint32_t divider = 3;
|
|
|
|
|
|
|
|
while((divider * divider < number) && (number % divider != 0))
|
|
|
|
divider += 2;
|
|
|
|
|
|
|
|
return (number % divider != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Before using the hash table we must allocate memory for it.
|
|
|
|
* We allocate one element more as the found prime number says.
|
|
|
|
* This is done for more effective indexing as explained in the
|
|
|
|
* comment for the hash function.
|
|
|
|
*/
|
|
|
|
static BOOL htab_create(uint32_t nel)
|
|
|
|
{
|
|
|
|
if (htab_table != NULL) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Change nel to the first prime number not smaller as nel.
|
|
|
|
nel |= 1;
|
|
|
|
while(!isprime(nel))
|
|
|
|
nel += 2;
|
|
|
|
|
|
|
|
htab_size = nel;
|
|
|
|
uprintf("localization: using %d entries hash table\n", nel);
|
|
|
|
htab_filled = 0;
|
|
|
|
|
|
|
|
// allocate memory and zero out.
|
|
|
|
htab_table = (htab_entry*)calloc(htab_size + 1, sizeof(htab_entry));
|
|
|
|
if (htab_table == NULL) {
|
|
|
|
uprintf("localization: could not allocate space for hash table\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* After using the hash table it has to be destroyed. */
|
|
|
|
static void htab_destroy(void)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
if (htab_table == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i=0; i<htab_size; i++) {
|
|
|
|
if (htab_table[i].used) {
|
|
|
|
safe_free(htab_table[i].str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
safe_free(htab_table);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is the search function. It uses double hashing with open addressing.
|
|
|
|
* We use a trick to speed up the lookup. The table is created with one
|
|
|
|
* more element available. This enables us to use the index zero special.
|
|
|
|
* This index will never be used because we store the first hash index in
|
|
|
|
* the field used where zero means not used. Every other value means used.
|
|
|
|
* The used field can be used as a first fast comparison for equality of
|
|
|
|
* the stored and the parameter value. This helps to prevent unnecessary
|
|
|
|
* expensive calls of strcmp.
|
|
|
|
*/
|
|
|
|
static uint32_t htab_hash(char* str)
|
|
|
|
{
|
|
|
|
uint32_t hval, hval2;
|
|
|
|
uint32_t idx;
|
|
|
|
uint32_t r = 5381;
|
|
|
|
int c;
|
|
|
|
char* sz = str;
|
|
|
|
|
|
|
|
if (str == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Compute main hash value (algorithm suggested by Nokia)
|
|
|
|
// TODO: THIS ALGORITHM SUUUUUUUUUUUUUUCKS!!!!
|
|
|
|
// Uncomment 'uprintf("hash collision' below to find out why...
|
|
|
|
while ((c = *sz++) != 0)
|
|
|
|
r = ((r << 5) + r) + c;
|
|
|
|
if (r == 0)
|
|
|
|
++r;
|
|
|
|
|
|
|
|
// compute table hash: simply take the modulus
|
|
|
|
hval = r % htab_size;
|
|
|
|
if (hval == 0)
|
|
|
|
++hval;
|
|
|
|
|
|
|
|
// Try the first index
|
|
|
|
idx = hval;
|
|
|
|
|
|
|
|
if (htab_table[idx].used) {
|
|
|
|
if ( (htab_table[idx].used == hval)
|
|
|
|
&& (safe_strcmp(str, htab_table[idx].str) == 0) ) {
|
|
|
|
// existing hash
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
// uprintf("hash collision ('%s' vs '%s')\n", str, htab_table[idx].str);
|
|
|
|
|
|
|
|
// Second hash function, as suggested in [Knuth]
|
|
|
|
hval2 = 1 + hval % (htab_size - 2);
|
|
|
|
|
|
|
|
do {
|
|
|
|
// Because size is prime this guarantees to step through all available indexes
|
|
|
|
if (idx <= hval2) {
|
|
|
|
idx = htab_size + idx - hval2;
|
|
|
|
} else {
|
|
|
|
idx -= hval2;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we visited all entries leave the loop unsuccessfully
|
|
|
|
if (idx == hval) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If entry is found use it.
|
|
|
|
if ( (htab_table[idx].used == hval)
|
|
|
|
&& (safe_strcmp(str, htab_table[idx].str) == 0) ) {
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (htab_table[idx].used);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Not found => New entry
|
|
|
|
|
|
|
|
// If the table is full return an error
|
|
|
|
if (htab_filled >= htab_size) {
|
|
|
|
uprintf("localization: hash table is full (%d entries)", htab_size);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
safe_free(htab_table[idx].str);
|
|
|
|
htab_table[idx].used = hval;
|
|
|
|
htab_table[idx].str = (char*) malloc(safe_strlen(str)+1);
|
|
|
|
if (htab_table[idx].str == NULL) {
|
|
|
|
uprintf("localization: could not duplicate string for hash table\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
memcpy(htab_table[idx].str, str, safe_strlen(str)+1);
|
|
|
|
++htab_filled;
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2013-10-15 21:58:27 +00:00
|
|
|
/*
|
|
|
|
* Add a localization command to a dialog/section
|
|
|
|
*/
|
|
|
|
void add_dialog_command(int index, loc_cmd* lcmd)
|
|
|
|
{
|
2013-10-18 22:08:49 +00:00
|
|
|
char str[128];
|
|
|
|
uint32_t i;
|
2013-10-15 21:58:27 +00:00
|
|
|
if ((lcmd == NULL) || (index < 0) || (index >= ARRAYSIZE(loc_dlg))) {
|
2013-10-18 22:08:49 +00:00
|
|
|
uprintf("localization: invalid parameter for add_dialog_command\n");
|
2013-10-15 21:58:27 +00:00
|
|
|
return;
|
|
|
|
}
|
2013-10-18 22:08:49 +00:00
|
|
|
|
|
|
|
// A dialog command must be unique, so we use a hash to identify any
|
|
|
|
// command that may already have been populated, and ensure it is replaced
|
|
|
|
// with the new one.
|
2013-10-19 14:48:22 +00:00
|
|
|
// Two dialogs may have different "m IDC_CONTROL" lines, and also
|
|
|
|
// "m IDC_CONTROL" and "t IDC_CONTROL" are separate, so we compute two more
|
|
|
|
// unique identifiers for dialog and command at the beginning of our string
|
|
|
|
str[0] = index + 0x30;
|
|
|
|
str[1] = lcmd->command + 0x30;
|
|
|
|
safe_strcpy(&str[2], sizeof(str)-2, lcmd->txt[0]);
|
2013-10-18 22:08:49 +00:00
|
|
|
i = htab_hash(str);
|
|
|
|
if (i != 0) {
|
|
|
|
if (htab_table[i].dlg_cmd != NULL) {
|
|
|
|
list_del(&(htab_table[i].dlg_cmd->list));
|
|
|
|
free_loc_cmd(htab_table[i].dlg_cmd);
|
|
|
|
}
|
|
|
|
htab_table[i].dlg_cmd = lcmd;
|
|
|
|
}
|
2013-10-15 21:58:27 +00:00
|
|
|
list_add(&lcmd->list, &loc_dlg[index].list);
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_loc_cmd(loc_cmd* lcmd)
|
|
|
|
{
|
|
|
|
if (lcmd == NULL)
|
|
|
|
return;
|
|
|
|
safe_free(lcmd->txt[0]);
|
|
|
|
safe_free(lcmd->txt[1]);
|
|
|
|
safe_free(lcmd->unum);
|
|
|
|
free(lcmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_dialog_list(void)
|
|
|
|
{
|
|
|
|
size_t i = 0;
|
|
|
|
loc_cmd *lcmd, *next;
|
|
|
|
|
|
|
|
for (i=0; i<ARRAYSIZE(loc_dlg); i++) {
|
|
|
|
if (list_empty(&loc_dlg[i].list))
|
|
|
|
continue;
|
|
|
|
list_for_each_entry_safe(lcmd, next, &loc_dlg[i].list, loc_cmd, list) {
|
|
|
|
list_del(&lcmd->list);
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_locale_list(void)
|
|
|
|
{
|
|
|
|
loc_cmd *lcmd, *next;
|
|
|
|
|
|
|
|
list_for_each_entry_safe(lcmd, next, &locale_list, loc_cmd, list) {
|
|
|
|
list_del(&lcmd->list);
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Init/destroy our various localization lists
|
2013-10-24 21:57:34 +00:00
|
|
|
* keep the locale list and filename on reinit
|
2013-10-15 21:58:27 +00:00
|
|
|
*/
|
2013-10-24 21:57:34 +00:00
|
|
|
void _init_localization(BOOL reinit) {
|
2013-10-15 21:58:27 +00:00
|
|
|
size_t i;
|
|
|
|
for (i=0; i<ARRAYSIZE(loc_dlg); i++)
|
|
|
|
list_init(&loc_dlg[i].list);
|
2013-10-24 21:57:34 +00:00
|
|
|
if (!reinit)
|
|
|
|
list_init(&locale_list);
|
2013-10-18 22:08:49 +00:00
|
|
|
htab_create(LOC_HTAB_SIZE);
|
2013-10-15 21:58:27 +00:00
|
|
|
}
|
|
|
|
|
2013-10-24 21:57:34 +00:00
|
|
|
void _exit_localization(BOOL reinit) {
|
|
|
|
if (!reinit) {
|
|
|
|
free_locale_list();
|
|
|
|
if (loc_filename != embedded_loc_filename)
|
|
|
|
safe_free(loc_filename);
|
|
|
|
}
|
2013-10-15 21:58:27 +00:00
|
|
|
free_dialog_list();
|
2013-10-18 22:08:49 +00:00
|
|
|
htab_destroy();
|
2013-10-15 21:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Validate and store localization command data
|
|
|
|
*
|
|
|
|
* TODO: Do we need to store a revert for every action we execute here,
|
|
|
|
* or do we want to reinstantiate the dialogs?
|
|
|
|
*/
|
|
|
|
BOOL dispatch_loc_cmd(loc_cmd* lcmd)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
static int dlg_index = 0;
|
|
|
|
loc_cmd* base_locale = NULL;
|
|
|
|
|
|
|
|
if (lcmd == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (lcmd->command <= LC_TEXT) {
|
|
|
|
// Any command up to LC_TEXT takes a control ID in text[0]
|
|
|
|
for (i=0; i<ARRAYSIZE(control_id); i++) {
|
|
|
|
if (safe_strcmp(lcmd->txt[0], control_id[i].name) == 0) {
|
|
|
|
lcmd->ctrl_id = control_id[i].id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (lcmd->ctrl_id < 0) {
|
|
|
|
luprintf("unknown control '%s'\n", lcmd->txt[0]);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(lcmd->command) {
|
|
|
|
// NB: For commands that take an ID, ctrl_id is always a valid index at this stage
|
|
|
|
case LC_TEXT:
|
|
|
|
case LC_MOVE:
|
|
|
|
case LC_SIZE:
|
|
|
|
add_dialog_command(dlg_index, lcmd);
|
|
|
|
break;
|
|
|
|
case LC_GROUP:
|
|
|
|
if ((lcmd->ctrl_id-IDD_DIALOG) > ARRAYSIZE(loc_dlg)) {
|
|
|
|
luprintf("'%s' is not a group ID\n", lcmd->txt[0]);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
dlg_index = lcmd->ctrl_id - IDD_DIALOG;
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
break;
|
|
|
|
case LC_VERSION:
|
|
|
|
luprintf("GOT VERSION: %d.%d\n", lcmd->num[0], lcmd->num[1]);
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
break;
|
|
|
|
case LC_BASE:
|
2013-10-17 19:12:16 +00:00
|
|
|
base_locale = get_locale_from_name(lcmd->txt[0], FALSE);
|
2013-10-15 21:58:27 +00:00
|
|
|
if (base_locale != NULL) {
|
2013-10-17 19:12:16 +00:00
|
|
|
uprintf("localization: using locale base '%s'\n", lcmd->txt[0]);
|
2013-10-15 21:58:27 +00:00
|
|
|
get_loc_data_file(NULL, (long)base_locale->num[0], (long)base_locale->num[1], base_locale->line_nr);
|
|
|
|
} else {
|
2013-10-17 19:12:16 +00:00
|
|
|
luprintf("locale base '%s' not found - ignoring", lcmd->txt[0]);
|
2013-10-15 21:58:27 +00:00
|
|
|
}
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
err:
|
|
|
|
free_loc_cmd(lcmd);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Apply stored localization commands to a specific dialog
|
|
|
|
* If hDlg is NULL, apply the commands against an active Window
|
|
|
|
* TODO: if dlg_id is <0, apply all
|
|
|
|
*/
|
|
|
|
void apply_localization(int dlg_id, HWND hDlg)
|
|
|
|
{
|
|
|
|
loc_cmd* lcmd;
|
|
|
|
HWND hCtrl = NULL;
|
|
|
|
int id_start = IDD_DIALOG, id_end = IDD_DIALOG + ARRAYSIZE(loc_dlg);
|
|
|
|
LONG_PTR style;
|
|
|
|
BOOL left_to_right = FALSE;
|
|
|
|
|
|
|
|
if ((dlg_id >= id_start) && (dlg_id < id_end)) {
|
|
|
|
// If we have a valid dialog_id, just process that one dialog
|
|
|
|
id_start = dlg_id;
|
|
|
|
id_end = dlg_id + 1;
|
|
|
|
if (hDlg != NULL) {
|
|
|
|
loc_dlg[dlg_id-IDD_DIALOG].hDlg = hDlg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (dlg_id = id_start; dlg_id < id_end; dlg_id++) {
|
|
|
|
hDlg = loc_dlg[dlg_id-IDD_DIALOG].hDlg;
|
|
|
|
if ((!IsWindow(hDlg)) || (list_empty(&loc_dlg[dlg_id-IDD_DIALOG].list)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// TODO: storing the messages in an array indexed on the message ID - 3000 would be faster
|
|
|
|
list_for_each_entry(lcmd, &loc_dlg[dlg_id-IDD_DIALOG].list, loc_cmd, list) {
|
|
|
|
if (lcmd->command <= LC_TEXT) { // TODO: should always be the case
|
|
|
|
if (lcmd->ctrl_id == dlg_id) {
|
|
|
|
if ((dlg_id == IDD_DIALOG) && (lcmd->txt[1] != NULL) && (lcmd->txt[1][0] != 0)) {
|
|
|
|
loc_line_nr = lcmd->line_nr;
|
|
|
|
luprint("operation forbidden (main dialog title cannot be changed)");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
hCtrl = hDlg;
|
2013-10-17 19:42:03 +00:00
|
|
|
if (dlg_id == IDD_DIALOG)
|
|
|
|
hDlg = NULL;
|
2013-10-15 21:58:27 +00:00
|
|
|
} else {
|
|
|
|
hCtrl = GetDlgItem(hDlg, lcmd->ctrl_id);
|
|
|
|
}
|
|
|
|
if (hCtrl == NULL) {
|
|
|
|
loc_line_nr = lcmd->line_nr;
|
|
|
|
luprintf("control '%s' is not part of dialog '%s'\n",
|
|
|
|
lcmd->txt[0], control_id[dlg_id-IDD_DIALOG].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(lcmd->command) {
|
|
|
|
// NB: For commands that take an ID, ctrl_id is always a valid index at this stage
|
|
|
|
case LC_TEXT:
|
|
|
|
if (hCtrl != NULL) {
|
|
|
|
if ((lcmd->txt[1] != NULL) && (lcmd->txt[1][0] != 0))
|
|
|
|
SetWindowTextU(hCtrl, lcmd->txt[1]);
|
|
|
|
if (left_to_right) {
|
|
|
|
style = GetWindowLongPtr(hCtrl, GWL_EXSTYLE);
|
|
|
|
style |= WS_EX_LAYOUTRTL; // WS_EX_RIGHT | WS_EX_RTLREADING
|
|
|
|
SetWindowLongPtr(hCtrl, GWL_EXSTYLE, style);
|
|
|
|
InvalidateRect(hCtrl, NULL, TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LC_MOVE:
|
|
|
|
if (hCtrl != NULL) {
|
|
|
|
ResizeMoveCtrl(hDlg, hCtrl, lcmd->num[0], lcmd->num[1], 0, 0);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case LC_SIZE:
|
|
|
|
if (hCtrl != NULL) {
|
|
|
|
ResizeMoveCtrl(hDlg, hCtrl, 0, 0, lcmd->num[0], lcmd->num[1]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function should be called when a localized dialog is destroyed
|
|
|
|
* NB: we can't use isWindow() against our existing HWND to avoid this call
|
|
|
|
* as handles are recycled.
|
|
|
|
*/
|
|
|
|
void reset_localization(int dlg_id)
|
|
|
|
{
|
|
|
|
loc_dlg[dlg_id-IDD_DIALOG].hDlg = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Produce a formatted localized message.
|
|
|
|
* Like printf, this call takes a variable number of argument, and uses
|
|
|
|
* the message ID to identify the formatted message to use.
|
|
|
|
* Uses a rolling list of buffers to allow concurrency
|
|
|
|
* TODO: use dynamic realloc'd buffer in case LOC_MESSAGE_SIZE is not enough
|
|
|
|
*/
|
|
|
|
char* lmprintf(int msg_id, ...)
|
|
|
|
{
|
|
|
|
static int buf_id = 0;
|
|
|
|
static char buf[LOC_MESSAGE_NB][LOC_MESSAGE_SIZE];
|
|
|
|
char *format = NULL;
|
|
|
|
va_list args;
|
|
|
|
loc_cmd* lcmd;
|
|
|
|
buf_id %= LOC_MESSAGE_NB;
|
|
|
|
buf[buf_id][0] = 0;
|
|
|
|
list_for_each_entry(lcmd, &loc_dlg[IDD_MESSAGES-IDD_DIALOG].list, loc_cmd, list) {
|
|
|
|
if ((lcmd->command == LC_TEXT) && (lcmd->ctrl_id == msg_id) && (lcmd->txt[1] != NULL)) {
|
|
|
|
format = lcmd->txt[1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (format == NULL) {
|
|
|
|
safe_sprintf(buf[buf_id], LOC_MESSAGE_SIZE-1, "MSG_%03d UNTRANSLATED", msg_id - MSG_000);
|
|
|
|
} else {
|
|
|
|
va_start(args, msg_id);
|
|
|
|
safe_vsnprintf(buf[buf_id], LOC_MESSAGE_SIZE-1, format, args);
|
|
|
|
va_end(args);
|
|
|
|
buf[buf_id][LOC_MESSAGE_SIZE-1] = '\0';
|
|
|
|
}
|
|
|
|
return buf[buf_id++];
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These 2 functions are used to set the current locale
|
2013-10-18 22:08:49 +00:00
|
|
|
* If fallback is true, the call will fall back to use the first
|
|
|
|
* translation listed in the loc file
|
2013-10-15 21:58:27 +00:00
|
|
|
*/
|
2013-10-18 22:08:49 +00:00
|
|
|
loc_cmd* get_locale_from_lcid(int lcid, BOOL fallback)
|
2013-10-15 21:58:27 +00:00
|
|
|
{
|
|
|
|
loc_cmd* lcmd = NULL;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (list_empty(&locale_list)) {
|
|
|
|
uprintf("localization: the locale list is empty!\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
list_for_each_entry(lcmd, &locale_list, loc_cmd, list) {
|
|
|
|
for (i=0; i<lcmd->unum_size; i++) {
|
|
|
|
if (lcmd->unum[i] == lcid) {
|
|
|
|
return lcmd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-18 22:08:49 +00:00
|
|
|
if (!fallback)
|
|
|
|
return NULL;
|
|
|
|
|
2013-10-15 21:58:27 +00:00
|
|
|
lcmd = list_entry(locale_list.next, loc_cmd, list);
|
|
|
|
// If we couldn't find a supported locale, just pick the first one (usually English)
|
|
|
|
uprintf("localization: could not find locale for LCID: 0x%04X. Will default to '%s'\n", lcid, lcmd->txt[0]);
|
|
|
|
return lcmd;
|
|
|
|
}
|
|
|
|
|
2013-10-18 22:08:49 +00:00
|
|
|
loc_cmd* get_locale_from_name(char* locale_name, BOOL fallback)
|
2013-10-15 21:58:27 +00:00
|
|
|
{
|
|
|
|
loc_cmd* lcmd = NULL;
|
|
|
|
|
|
|
|
if (list_empty(&locale_list)) {
|
|
|
|
uprintf("localization: the locale list is empty!\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
list_for_each_entry(lcmd, &locale_list, loc_cmd, list) {
|
|
|
|
if (safe_strcmp(lcmd->txt[0], locale_name) == 0)
|
|
|
|
return lcmd;
|
|
|
|
}
|
|
|
|
|
2013-10-18 22:08:49 +00:00
|
|
|
if (!fallback)
|
2013-10-17 19:12:16 +00:00
|
|
|
return NULL;
|
|
|
|
|
2013-10-15 21:58:27 +00:00
|
|
|
lcmd = list_entry(locale_list.next, loc_cmd, list);
|
|
|
|
uprintf("localization: could not find locale for name '%s'. Will default to '%s'\n", locale_name, lcmd->txt[0]);
|
|
|
|
return lcmd;
|
|
|
|
}
|