#include "Core.h"

#ifdef PLATFORM_WIN32
#include <wingdi.h>
#include <winnls.h>
#endif
#ifdef PLATFORM_POSIX
#include <locale.h>
#include <langinfo.h>
#endif

NAMESPACE_UPP

#define LLOG(x) // LOG(x)

String LNGAsText(int d)
{
	String result;
	int c = (d >> 15) & 31;
	if(c) {
		result.Cat(c + 'A' - 1);
		c = (d >> 10) & 31;
		if(c) {
			result.Cat(c + 'A' - 1);
			c = (d >> 5) & 31;
			if(c) {
				result.Cat('-');
				result.Cat(c + 'A' - 1);
				c = d & 31;
				if(c) result.Cat(c + 'A' - 1);
			}
		}
	}
	c = (d >> 20) & 255;
	if(c)
		result << ' ' << CharsetName(c);
	return result;
}

byte GetLNGCharset(int d)
{
	byte cs = byte(d >> 20);
	return cs ? cs : CHARSET_UTF8;
}

int  SetLNGCharset(int lng, byte chrset)
{
	return (lng & ~(0xffffffff << 20)) | (chrset << 20);
}

int LNGFromText(const char *s)
{
	int l = 0;
	if(IsAlpha(*s)) {
		l = (ToUpper(*s++) - 'A' + 1) << 15;
		if(IsAlpha(*s)) {
			l |= (ToUpper(*s++) - 'A' + 1) << 10;
			if(*s && !IsAlpha(*s))
				s++;
			if(IsAlpha(*s)) {
				l |= (ToUpper(*s++) - 'A' + 1) << 5;
				if(IsAlpha(*s)) {
					l |= (ToUpper(*s++) - 'A' + 1);
					while(*s && *s != ' ')
						s++;
					if(*s == ' ') {
						s++;
						int cs = CharsetByName(s);
						if(cs > 0)
							l |= (cs << 20);
						else
							return 0;
					}
					return l;
				}
			}
		}
	}
	return 0;
}

#ifdef PLATFORM_WIN32

String GetUserLocale(dword type)
{
#ifdef PLATFORM_WINCE
	wchar h[256];
	int n = ::GetLocaleInfo(GetUserDefaultLCID(), type, h, 256);
	return n ? WString(h, n - 1).ToString() : String();
#else
	char h[256];
	int n =:: GetLocaleInfo(GetUserDefaultLCID(), type, h, 256);
	return n ? String(h, n - 1) : String();
#endif
}

int GetSystemLNG()
{
	static int lang;
	ONCELOCK {
		lang = LNGFromText(GetUserLocale(LOCALE_SISO639LANGNAME) + GetUserLocale(LOCALE_SISO3166CTRYNAME));
		if(!lang)
			lang = LNG_ENGLISH;
		int cs = atoi(GetUserLocale(LOCALE_IDEFAULTANSICODEPAGE));
		if(cs >= 1250 && cs <= 1258)
			lang = SetLNGCharset(lang, CHARSET_WIN1250 + cs - 1250);
	}
	return lang;
}
#endif

#ifdef PLATFORM_POSIX
int GetSystemLNG() {
	static int lang;
	ONCELOCK {
		String s = Environment().Get("LANG", Null);
		lang = LNGFromText(s);
		if(!lang)
			lang = LNG_ENGLISH;
		const char *q = strchr(s, '.');
		if(q)
			lang = SetLNGCharset(lang, CharsetByName(q + 1));
	};
	return lang;
};

#endif

class LangConvertClass : public Convert {
	virtual Value  Format(const Value& q) const {
		return LNGAsText((int)q);
	}

	virtual Value  Scan(const Value& text) const {
		if(IsNull(text)) return 0;
		int q = LNGFromText((String)text);
		if(!q) return ErrorValue(t_("Invalid language specification."));
		return (int) q;
	}

	virtual int    Filter(int chr) const {
		return chr == ' ' || chr == '-' || IsDigit(chr) ? chr : IsAlpha(chr) ? ToUpper(chr) : 0;
	}
};

Convert& LNGConvert()
{
	return Single<LangConvertClass>();
}

static int sCurrentLanguage = -1;

int  GetCurrentLanguage() {
	return sCurrentLanguage;
}

#ifdef PLATFORM_WIN32

typedef VectorMap<int, LCID> LCIDMap;
GLOBAL_VAR(LCIDMap, GetLCIDMap)

#ifdef PLATFORM_WINCE //TODO?
String GetLocaleInfoA(LCID lcid, LCTYPE lctype)
{
	wchar cbuf[1000];
	::GetLocaleInfoW(lcid, lctype, cbuf, __countof(cbuf));
	return FromSystemCharset(cbuf);
}
#else
String GetLocaleInfoA(LCID lcid, LCTYPE lctype)
{
	if(IsWinNT()) {
		wchar cbuf[1000];
		UnicodeWin32().GetLocaleInfoW(lcid, lctype, cbuf, __countof(cbuf));
		return FromSystemCharsetW(cbuf);
	}
	else {
		char cbuf[1000];
		::GetLocaleInfoA(lcid, lctype, cbuf, __countof(cbuf));
		return FromSystemCharset(cbuf);
	}
}
#endif

WString GetLocaleInfoW(LCID lcid, LCTYPE lctype)
{
	union {
		wchar wbuf[1000];
		char abuf[1000];
	};
	Zero(wbuf);
	if(::GetLocaleInfoW(lcid, lctype, (WCHAR *)wbuf, __countof(wbuf)))
		return wbuf;
#ifdef PLATFORM_WINCE
	return Null;
#else
	::GetLocaleInfoA(lcid, lctype, abuf, __countof(abuf));
	return ToUnicode(abuf, CHARSET_DEFAULT);
#endif
}

#ifdef PLATFORM_WINCE
static BOOL CALLBACK sEnumLocale(wchar *locale_string)
#else
static BOOL CALLBACK sEnumLocale(char *locale_string)
#endif
{
	LLOG(locale_string);
	LCID lcid = stou(locale_string, NULL, 16);
#ifdef PLATFORM_WINCE
	wchar buffer[10];
#else
	char buffer[10];
#endif
	::GetLocaleInfo(lcid, LOCALE_SISO639LANGNAME, buffer, 10);
	int language = (ToUpper(buffer[0]) << 24) + (ToUpper(buffer[1]) << 16);
	::GetLocaleInfo(lcid, LOCALE_SISO3166CTRYNAME, buffer, 10);
	language += (ToUpper(buffer[0]) << 8) + (ToUpper(buffer[1]) << 0);
	LLOG(FormatIntHex(language, 8) << ", " << LNGAsText(language) << "->" << FormatIntHex(lcid, 8));
	GetLCIDMap().GetAdd(language, lcid);
	return TRUE;
}

LCID GetLanguageLCID(int language)
{
	if(language == 0)
		return 0x400;
	ONCELOCK {
		EnumSystemLocales(&sEnumLocale, LCID_SUPPORTED);
	}
	return GetLCIDMap().Get(language, MAKELCID(MAKELANGID(LANG_NEUTRAL, SUBLANG_NEUTRAL), SORT_DEFAULT));
}
#endif

void SetLanguage(int lang) {
	if(lang != LNG_CURRENT) {
		sCurrentLanguage = lang;
		SetDefaultCharset(GetLNGCharset(lang));
	}
	SetCurrentLanguage(lang);
}

const int *GetAllLanguages() {
	static int all_langs[] = {
		LNG_('E', 'N', 'U', 'S'),
		LNG_('E', 'N', 'G', 'B'),
		LNG_('E', 'N', 'A', 'U'),
		LNG_('E', 'N', 'C', 'A'),
		LNG_('E', 'N', 'N', 'Z'),
		LNG_('E', 'N', 'I', 'E'),
		LNG_('E', 'N', 'Z', 'A'),
		LNG_('E', 'N', 'J', 'M'),
		LNG_('E', 'N', 'C', 'B'),
		LNG_('E', 'N', 'B', 'Z'),
		LNG_('E', 'N', 'T', 'T'),
		LNG_('B', 'G', 'B', 'G'),
		LNG_('C', 'S', 'C', 'Z'),
		LNG_('D', 'A', 'D', 'K'),
		LNG_('D', 'E', 'D', 'E'),
		LNG_('D', 'E', 'C', 'H'),
		LNG_('D', 'E', 'A', 'T'),
		LNG_('D', 'E', 'L', 'U'),
		LNG_('D', 'E', 'L', 'I'),
		LNG_('E', 'L', 'G', 'R'),
		LNG_('E', 'S', 'E', 'S'),
		LNG_('E', 'S', 'M', 'X'),
		LNG_('E', 'S', 'E', 'S'),
		LNG_('E', 'S', 'G', 'T'),
		LNG_('E', 'S', 'C', 'R'),
		LNG_('E', 'S', 'P', 'A'),
		LNG_('E', 'S', 'D', 'O'),
		LNG_('E', 'S', 'V', 'E'),
		LNG_('E', 'S', 'C', 'O'),
		LNG_('E', 'S', 'P', 'E'),
		LNG_('E', 'S', 'A', 'R'),
		LNG_('E', 'S', 'E', 'C'),
		LNG_('E', 'S', 'C', 'L'),
		LNG_('E', 'S', 'U', 'Y'),
		LNG_('E', 'S', 'P', 'Y'),
		LNG_('E', 'S', 'B', 'O'),
		LNG_('E', 'S', 'S', 'V'),
		LNG_('E', 'S', 'H', 'N'),
		LNG_('E', 'S', 'N', 'I'),
		LNG_('E', 'S', 'P', 'R'),
		LNG_('F', 'I', 'F', 'I'),
		LNG_('F', 'R', 'F', 'R'),
		LNG_('F', 'R', 'B', 'E'),
		LNG_('F', 'R', 'C', 'A'),
		LNG_('F', 'R', 'C', 'H'),
		LNG_('F', 'R', 'L', 'U'),
		LNG_('H', 'U', 'H', 'U'),
		LNG_('I', 'S', 'I', 'S'),
		LNG_('I', 'T', 'I', 'T'),
		LNG_('I', 'T', 'C', 'H'),
		LNG_('N', 'L', 'N', 'L'),
		LNG_('N', 'L', 'B', 'E'),
		LNG_('N', 'O', 'N', 'O'),
		LNG_('N', 'O', 'N', 'O'),
		LNG_('P', 'L', 'P', 'L'),
		LNG_('P', 'T', 'B', 'R'),
		LNG_('P', 'T', 'P', 'T'),
		LNG_('R', 'O', 'R', 'O'),
		LNG_('R', 'U', 'R', 'U'),
		LNG_('H', 'R', 'H', 'R'),
		LNG_('S', 'R', 'S', 'P'),
		LNG_('S', 'R', 'S', 'P'),
		LNG_('S', 'K', 'S', 'K'),
		LNG_('S', 'V', 'S', 'E'),
		LNG_('S', 'V', 'F', 'I'),
		LNG_('T', 'R', 'T', 'R'),
		LNG_('S', 'L', 'S', 'I'),
		LNG_('A', 'F', 'Z', 'A'),
		LNG_('S', 'Q', 'A', 'L'),
		LNG_('E', 'U', 'E', 'S'),
		LNG_('B', 'E', 'B', 'Y'),
		LNG_('C', 'A', 'E', 'S'),
		LNG_('E', 'T', 'E', 'E'),
		LNG_('F', 'O', 'F', 'O'),
		LNG_('I', 'D', 'I', 'D'),
		LNG_('L', 'V', 'L', 'V'),
		LNG_('L', 'T', 'L', 'T'),
		LNG_('U', 'K', 'U', 'A'),
		LNG_('Z', 'H', 'C', 'N'),
		LNG_('Z', 'H', 'T', 'W'),
		LNG_('K', 'O', 'K', 'R'),
		LNG_('J', 'A', 'J', 'P'),
		0
	};
	return all_langs;
}

String GetLangName(int language)
{
	return GetLanguageInfo(language).english_name;
}

/*
	char buffer[200];
	if(language & 0x1000000) {
		language &= 15;
		static char *name[] = {
			"ANSI",
			"RUSSIAN",
			"EASTEUROPE",
			"GREEK",
			"TURKISH",
			"BALTIC",
			"HEBREW",
			"ARABIC"
		};
		return String(name[language & 7]) + (language < 8 ? " I" : " II");
	}
	return GetLocaleInfo(MAKELCID(language, SORT_DEFAULT), LOCALE_SENGLANGUAGE, buffer, 200)
	       ? buffer : "";
*/

String GetNativeLangName(int language) {
	return GetLanguageInfo(language).native_name.ToString();
/*
	char buffer[200];
	return GetLocaleInfo(MAKELCID(language, SORT_DEFAULT), LOCALE_SNATIVELANGNAME, buffer, 200)
	       ? buffer : ~GetLangName(language);
*/
}

byte GetLangCharset(int language) {
	return GetLanguageInfo(language).charset;
/*
	static struct { byte charset; dword codepage; } tab[] = {
		{ CHARSET_WIN1252,    1252 },
		{ CHARSET_WIN1251,    1251 },
		{ CHARSET_WIN1250,    1250 },
		{ CHARSET_WIN1253,    1253 },
		{ CHARSET_WIN1254,    1254 },
		{ CHARSET_WIN1257,    1257 },
		{ CHARSET_WIN1255,    1255 },
		{ CHARSET_WIN1256,    1256 },
//		{ SHIFTJIS_CHARSET,    932 },
//		{ HANGEUL_CHARSET,     949 },
//		{ GB2312_CHARSET,      936 },
//		{ CHINESEBIG5_CHARSET, 950 },
	};
	if(language & 0x1000000)
		return tab[language & 7].charset;
	char buffer[20];
	GetLocaleInfo(MAKELCID(language, SORT_DEFAULT), LOCALE_IDEFAULTANSICODEPAGE, buffer, 20);
	int codepage = atoi(buffer);
	for(int i = 0; i < __countof(tab); i++)
		if(tab[i].codepage == codepage)
			return tab[i].charset;
	return CHARSET_DEFAULT;
*/
}
LanguageInfo::LanguageInfo(int lang_)
: language(lang_)
{
	charset = CHARSET_DEFAULT;

#ifdef PLATFORM_WIN32
	LCID lcid = GetLanguageLCID(lang_);
	english_name = GetLocaleInfoA(lcid, LOCALE_SENGLANGUAGE);
	native_name = GetLocaleInfoW(lcid, LOCALE_SNATIVELANGNAME);
	thousand_separator = GetLocaleInfoA(lcid, LOCALE_STHOUSAND);
	decimal_point = GetLocaleInfoA(lcid, LOCALE_SDECIMAL);
	static const LCTYPE months[] =
	{
		LOCALE_SMONTHNAME1,  LOCALE_SMONTHNAME2,  LOCALE_SMONTHNAME3,
		LOCALE_SMONTHNAME4,  LOCALE_SMONTHNAME5,  LOCALE_SMONTHNAME6,
		LOCALE_SMONTHNAME7,  LOCALE_SMONTHNAME8,  LOCALE_SMONTHNAME9,
		LOCALE_SMONTHNAME10, LOCALE_SMONTHNAME11, LOCALE_SMONTHNAME12,
	};
	static const LCTYPE smonths[] =
	{
		LOCALE_SABBREVMONTHNAME1,  LOCALE_SABBREVMONTHNAME2,  LOCALE_SABBREVMONTHNAME3,
		LOCALE_SABBREVMONTHNAME4,  LOCALE_SABBREVMONTHNAME5,  LOCALE_SABBREVMONTHNAME6,
		LOCALE_SABBREVMONTHNAME7,  LOCALE_SABBREVMONTHNAME8,  LOCALE_SABBREVMONTHNAME9,
		LOCALE_SABBREVMONTHNAME10, LOCALE_SABBREVMONTHNAME11, LOCALE_SABBREVMONTHNAME12,
	};
	ASSERT(__countof(months) == __countof(month_names));
	int i;
	for(i = 0; i < __countof(months); i++)
	{
		month_names[i] = GetLocaleInfoW(lcid, months[i]);
		short_month_names[i] = GetLocaleInfoW(lcid, smonths[i]);
	}
	static const LCTYPE days[] =
	{
		LOCALE_SDAYNAME1, LOCALE_SDAYNAME2, LOCALE_SDAYNAME3, LOCALE_SDAYNAME4,
		LOCALE_SDAYNAME5, LOCALE_SDAYNAME6, LOCALE_SDAYNAME7,
	};
	static const LCTYPE sdays[] =
	{
		LOCALE_SABBREVDAYNAME1, LOCALE_SABBREVDAYNAME2, LOCALE_SABBREVDAYNAME3, LOCALE_SABBREVDAYNAME4,
		LOCALE_SABBREVDAYNAME5, LOCALE_SABBREVDAYNAME6, LOCALE_SABBREVDAYNAME7,
	};
	ASSERT(__countof(days) == __countof(day_names));
	for(i = 0; i < __countof(days); i++)
	{
		day_names[i] = GetLocaleInfoW(lcid, days[i]);
		short_day_names[i] = GetLocaleInfoW(lcid, sdays[i]);
	}

	static struct { byte charset; dword codepage; } tab[] =
	{
		{ CHARSET_WIN1252,    1252 },
		{ CHARSET_WIN1251,    1251 },
		{ CHARSET_WIN1250,    1250 },
		{ CHARSET_WIN1253,    1253 },
		{ CHARSET_WIN1254,    1254 },
		{ CHARSET_WIN1257,    1257 },
		{ CHARSET_WIN1255,    1255 },
		{ CHARSET_WIN1256,    1256 },
//		{ SHIFTJIS_CHARSET,    932 },
//		{ HANGEUL_CHARSET,     949 },
//		{ GB2312_CHARSET,      936 },
//		{ CHINESEBIG5_CHARSET, 950 },
	};
	int codepage = atoi(GetLocaleInfoA(lcid, LOCALE_IDEFAULTANSICODEPAGE));
	for(i = 0; i < __countof(tab); i++)
		if(tab[i].codepage == codepage)
		{
			charset = tab[i].charset;
			break;
		}
#endif

#ifdef PLATFORM_POSIX
	String langtext = LNGAsText(language);
	char ltext[6];
	ltext[0] = ToLower(langtext[0]);
	ltext[1] = ToLower(langtext[1]);
	ltext[2] = '_';
	ltext[3] = ToUpper(langtext[3]);
	ltext[4] = ToUpper(langtext[4]);
	ltext[5] = 0;
	String oldloc = setlocale(LC_ALL, NULL);
//	puts(String() << "setting locale " << ltext << ", old locale = " << oldloc);
	if(setlocale(LC_ALL, ltext))
	{
//		puts(String() << "set locale " << ltext << " succeeded");
		const struct lconv *lc = localeconv();
		decimal_point = lc->decimal_point;
		thousand_separator = lc->thousands_sep;
		//lc->grouping - controls thousands grouping
		static const int months[] =
		{
			MON_1, MON_2, MON_3, MON_4, MON_5, MON_6, MON_7, MON_8, MON_9, MON_10, MON_11, MON_12,
		};
		static const int smonths[] =
		{
			ABMON_1, ABMON_2, ABMON_3, ABMON_4, ABMON_5, ABMON_6, ABMON_7, ABMON_8, ABMON_9, ABMON_10, ABMON_11, ABMON_12,
		};
		ASSERT(__countof(months) == __countof(month_names) && __countof(smonths) == __countof(month_names));
		int i;
		for(i = 0; i < __countof(months); i++)
		{
			month_names[i] = nl_langinfo(months[i]);
			short_month_names[i] = nl_langinfo(smonths[i]);
		}
		static const int days[] =
		{ // Linux locale starts with Sunday
			DAY_2, DAY_3, DAY_4, DAY_5, DAY_6, DAY_7, DAY_1,
		};
		static const int sdays[] =
		{
			ABDAY_2, ABDAY_3, ABDAY_4, ABDAY_5, ABDAY_6, ABDAY_7, ABDAY_1,
		};
		ASSERT(__countof(days) == __countof(day_names) && __countof(sdays) == __countof(day_names));
		for(i = 0; i < __countof(days); i++)
		{
			day_names[i] = nl_langinfo(days[i]);
			short_day_names[i] = nl_langinfo(sdays[i]);
		}

		setlocale(LC_ALL, oldloc);
	}
#endif
}

String LanguageInfo::ToString() const
{
	String out;
	out << "LANGUAGE={" << LNGAsText(language) << "}\n"
#ifdef PLATFORM_WIN32
		<< NFormat("LCID={%08x}\n", (int)GetLanguageLCID(language))
#endif
		<< "ENGLISH_NAME={" << english_name << "}\n"
		"NATIVE_NAME={" << FromUnicode(native_name) << "}\n"
		"CHARSET={" << charset << "}\n"
		"THOUSAND_SEPARATOR={" << thousand_separator << "}\n"
		"DECIMAL_POINT={" << decimal_point << "}\n"
		"MONTH_NAMES={\n";
	int i;
	for(i = 0; i < __countof(month_names); i++)
		out << "    {" << FromUnicode(month_names[i]) << "} / {" << FromUnicode(short_month_names[i]) << "}\n";
	out << "}\n"
		"DAY_NAMES={\n";
	for(i = 0; i < __countof(day_names); i++)
		out << "    {" << FromUnicode(day_names[i]) << "} / {" << FromUnicode(short_day_names[i]) << "}\n";
	out << "}\n";
	return out;
}

static const char *NlsFindDigits(const char *src, String& dest)
{
	if(*src && !IsDigit(*src))
	{
		const char *start = src;
		while(*++src && !IsDigit(*src))
			;
		dest.Cat(start, (int)(src - start));
	}
	return src;
}

static const char *NlsCopyDigits(const char *src, String& dest, String thousands)
{
	if(IsDigit(*(src = NlsFindDigits(src, dest))))
	{
		const char *p = src;
		while(IsDigit(*++src))
			;
		int first = ((int)(src - p) + 2) % 3 + 1;
		while(p < src)
		{
			dest.Cat(p, first);
			if((p += first) < src)
			{
				dest.Cat(thousands);
				first = 3;
			}
		}
	}
	return src;
}

static String NlsFormatRaw(const char *n, String thousands, String decimals)
{
	if(*n == 0)
		return Null;
//	puts(String() << "NlsFormatRaw, n = <" << n << ">, thousands <" << thousands << ">, decimal <" << decimals << ">");
	String result;
	n = NlsCopyDigits(n, result, thousands);
	if(*n == (wchar)'.')
	{ // decimal separator
		n++;
		result.Cat(decimals);
		const char *s = n;
		while(IsDigit(*n))
			n++;
		result.Cat(s, (int)(n - s));
	}
	if(*(n = NlsCopyDigits(n, result, thousands)))
		result.Cat(n);
	return result;
}

String LanguageInfo::FormatInt(int value) const
{
	if(IsNull(value))
		return Null;
	String dest;
	String is = UPP::FormatInt(value);
	const char *p = NlsCopyDigits(is, dest, thousand_separator);
	if(*p)
		dest.Cat(p);
	return dest;
}

String LanguageInfo::FormatDouble(double value, int digits, int FD_flags, int fill_exp) const
{
	if(IsNull(value))
		return Null;
//	puts(String() << "LanguageInfo(" << LNGAsText(language) << "): thousands <" << thousand_separator << ">, decimal <" << decimal_point << ">");
	return NlsFormatRaw(UPP::FormatDouble(value, digits, FD_flags, fill_exp), thousand_separator, decimal_point);
}

String LanguageInfo::FormatDate(Date date) const
{
	return UPP::FormatDate(date, date_format, language);
}

String LanguageInfo::FormatTime(Time time) const
{
	return UPP::FormatTime(time, time_format, language);
}

/*
int LanguageInfo::ScanInt(const char *text, const char **endptr) const
{
	return ScanInt(text, endptr); //!! todo
}
*/

/*
double LanguageInfo::ScanDouble(const char *text, const char **endptr) const
{
	return ScanDouble(text, endptr); //!! todo
}
*/

/*
Date LanguageInfo::ScanDate(const char *text, const char **endptr, Date base_date) const
{
	return UPP::ScanDate(text, endptr, date_format, language, base_date);
}
*/

/*
Time LanguageInfo::ScanTime(const char *text, const char **endptr, Time base_time) const
{
	return UPP::ScanTime(text, endptr, time_format, language, base_time);
}
*/

WString LanguageInfo::GetIndexLetter(const wchar *text) const
{
	return IsLetter(*text) ? WString(text, 1) : WString(Null);
}

static int LangCompareDigits(const wchar *&a, const wchar *&b, const wchar *e1, const wchar *e2)
{
	const wchar *p1 = a, *p2 = b;
	while(p1 < e1 && *p1 == '0')
		p1++;
	while(p2 < e2 && *p2 == '0')
		p2++;
	const wchar *x1 = p1, *x2 = p2;
	while(p1 < e1 && IsDigit(*p1))
		p1++;
	while(p2 < e2 && IsDigit(*p2))
		p2++;
	if(p1 - x1 != p2 - x2)
		return cmp(p1 - x1, p2 - x2);
	for(; x1 != p1; x1++, x2++)
		if(*x1 != *x2)
			return *x1 > *x2 ? 1 : -1;
	a = p1;
	b = p2;
	return 0;
}

int LanguageInfo::Compare(const wchar *a, const wchar *b, int a_length, int b_length) const
{
	ASSERT(language != LNG_CZECH); // use RegisterLanguageInfoCS to register true czech compare

	int  little = 0, middle = 0;
	const wchar *p1 = a, *e1 = a + a_length, *p2 = b, *e2 = b + b_length;

	while(p1 < e1 && p2 < e2)
	{
		wchar c1 = *p1++;
		wchar c2 = *p2++;

		int level1 = (IsLetter(c1) ? 3 : IsDigit(c1) ? 2 : c1 == ' ' ? 0 : 1);
		int level2 = (IsLetter(c2) ? 3 : IsDigit(c2) ? 2 : c2 == ' ' ? 0 : 1);
		if(level1 != level2)
			return cmp(level1, level2);
		if(level1 <= 1)
		{
			if(c1 != c2)
				return cmp(c1, c2);
			continue;
		}
		if(level1 == 2)
		{ // digits
			const wchar *dp1 = --p1, *dp2 = --p2;
			int res = LangCompareDigits(dp1, dp2, e1, e2);
			if(res)
				return res;
			p1 = dp1;
			p2 = dp2;
			continue;
		}

		int u1, u2, i1, i2;

		i1 = ToAscii(u1 = ToUpper(c1));
		i2 = ToAscii(u2 = ToUpper(c2));

		if(i1 != i2)
			return i1 >= i2 ? 1 : -1;

		if(u1 != u2) // different diacritics
			if(middle == 0)
				middle = u1 - u2;

		if(c1 != c2) // different case
		{
			if(little == 0)
				little = (u1 != c1) - (u2 != c2);
		}
	}
	little += 4 * middle;
	if(little == 0)
		little = a_length - b_length;
	return sgn(little);
}

/*
static bool ContainsAccents(const wchar *s)
{
	for(; *s; s++)
		if(*s != ToAscii(*s))
			return true;
	return false;
}
*/

class DefaultWildcardCompare : public LanguageInfo::WildcardCompare
{
public:
	DefaultWildcardCompare(const wchar *templ)
	{
		raw_templ = 0;
//		cvt_ascii = true;
		if(*templ == 0)
			return;
//		if((*templ == '.' && templ[1] != 0 && *++templ != '.') || ContainsAccents(templ))
//			cvt_ascii = true;
		raw_templ = templ;
//		if(cvt_ascii)
//			cvt_templ = ToUpper(ToAscii(raw_templ));
//		else
			cvt_templ = ToUpper(raw_templ);
	}

	virtual bool Matches(const wchar *s) const
	{
		return !raw_templ || RawMatches(s, cvt_templ);
	}

private:
	bool RawMatches(const wchar *s, const wchar *templ) const
	{
		for(;;)
			switch(*templ++)
			{
			case 0: return true;
			case '.': if(*templ == 0) return *s == 0; // force end of string
			case '?': if(*s++ == 0) return false; break;
			case '*':
				do
					if(RawMatches(s, templ))
						return true;
				while(*s++);
				return false;
			case '\\':
				if(*templ == 0 || *s++ != *templ++)
					return false;
				break;

			default:
//				if(templ[-1] != ToUpper(cvt_ascii ? ToAscii(*s) : *s))
				if(templ[-1] != ToUpper(*s))
					return false;
				s++;
				break;
			}
		return true;
	}

private:
	const wchar *raw_templ;
//	bool         cvt_ascii;
	WString      cvt_templ;
};

One<LanguageInfo::WildcardCompare> LanguageInfo::GetWildcardCompare(const wchar *wildcard_text) const
{
	return new DefaultWildcardCompare(wildcard_text);
}

typedef ArrayMap<int, LanguageInfo> LanguageInfoMap;
GLOBAL_VAR(LanguageInfoMap, LanguageInfo::Map)
static StaticMutex sMapMutex;

void LanguageInfo::Register(One<LanguageInfo> info)
{
	INTERLOCKED_(sMapMutex) {
		int lang = info->language;
		int f = Map().Find(lang);
		if(f >= 0)
			Map().Set(f, -info);
		else
			Map().Add(lang, -info);
	}
}

class LanguageInfoCS : public LanguageInfo
{
public:
	LanguageInfoCS();

	virtual WString GetIndexLetter(const wchar *text) const;

	virtual int     Compare(const wchar *a, const wchar *b, int a_length, int b_length) const;
	virtual One<WildcardCompare> GetWildcardCompare(const wchar *wildcard_text) const;

public:
};

LanguageInfoCS::LanguageInfoCS()
: LanguageInfo(LNG_CZECH)
{
	english_name = "Czech";
	native_name = ToUnicode("??sky", CHARSET_WIN1250); // __FILE__CHARSET__
//	charset = CHARSET_WIN1250;
//	thousand_separator = L" ";
//	decimal_point = L",";
}

WString LanguageInfoCS::GetIndexLetter(const wchar *s) const
{
	wchar temp[3];
	temp[0] = temp[1] = temp[2] = 0;
	if(*s <= 2047 && IsLetter(*s)) // IsLetter
	{
		temp[0] = ToUpper(*s);
		if(s[1] <= 2047 && IsLetter(s[1]))
			temp[1] = ToLower(s[1]);
		if(temp[0] != 'C' || temp[1] != 'h')
			temp[1] = 0;
		switch(ToUpper(ToAscii(*s)))
		{
		case 'A': case 'E': case 'I': case 'N':
		case 'O': case 'T': case 'U': case 'Y':
			temp[0] = ToAscii(temp[0]);
			break;
		}
	}
	return temp;
}

int LanguageInfoCS::Compare(const wchar *a, const wchar *b, int a_length, int b_length) const
{
	int  little = 0, middle = 0;
	const wchar *p1 = a, *e1 = a + a_length, *p2 = b, *e2 = b + b_length;

	while(p1 < e1 && p2 < e2)
	{
		wchar c1 = *p1++;
		wchar c2 = *p2++;

		int level1 = (IsLetter(c1) ? 3 : IsDigit(c1) ? 2 : c1 == ' ' ? 0 : 1);
		int level2 = (IsLetter(c2) ? 3 : IsDigit(c2) ? 2 : c2 == ' ' ? 0 : 1);
		if(level1 != level2)
			return cmp(level1, level2);
		if(level1 <= 1)
		{
			if(c1 != c2)
				return cmp(c1, c2);
			continue;
		}
		if(level1 == 2)
		{ // digits
			const wchar *dp1 = --p1, *dp2 = --p2;
			int res = LangCompareDigits(dp1, dp2, e1, e2);
			if(res)
				return res;
			p1 = dp1;
			p2 = dp2;
			continue;
		}

		int u1, u2, i1, i2;

		if((c1 == 'C' || c1 == 'c') && (p1 < e1 && (*p1 == 'H' || *p1 == 'h')))
		{ // CH
			i1 = 'H';
			u1 = 65535;
			p1++;
		}
		else
			i1 = ToAscii(u1 = ToUpper(c1));
		if((c2 == 'C' || c2 == 'c') && (p2 < e2 && (*p2 == 'H' || *p2 == 'h')))
		{ // CH
			i2 = 'H';
			u2 = 65535;
			p2++;
		}
		else
			i2 = ToAscii(u2 = ToUpper(c2));

		if(i1 != i2)
			return i1 >= i2 ? 1 : -1;

		if(u1 != u2) // different diacritics
			switch(i1)
			{
			case 'A': case 'E': case 'I': case 'N':
			case 'O': case 'T': case 'U': case 'Y':
				if(middle == 0)
					middle = u1 - u2;
				continue;
			default:
				return u1 >= u2 ? 1 : -1;
			}
		if(c1 != c2) // different case
		{
			if(little == 0)

				little = (u1 != c1) - (u2 != c2);
		}
	}
	little += 4 * middle;
	if(little == 0)
		little = a_length - b_length;
	return sgn(little);
}

class WildcardCompareCS : public LanguageInfo::WildcardCompare
{
public:
	WildcardCompareCS(const wchar *templ)
	{
		raw_templ = 0;
//		cvt_ascii = false;
		if(*templ == 0)
			return;
//		if((*templ == '.' && templ[1] != 0 && *++templ != '.') || ContainsAccents(templ))
//		{
//			exclude_ch = (ToUpper(*templ) == 'C' && templ[1] == 0);
//			cvt_ascii = false;
//		}
		raw_templ = templ;
//		if(cvt_ascii)
//			cvt_templ = ToUpper(ToAscii(raw_templ));
//		else
			cvt_templ = ToUpper(raw_templ);
	}

	virtual bool Matches(const wchar *s) const
	{
//		if(exclude_ch && *s && (s[1] == 'h' || s[1] == 'H'))
//			return false;
		return !raw_templ || RawMatches(s, cvt_templ);
	}

private:
	bool RawMatches(const wchar *s, const wchar *templ) const
	{
		for(;;)
			switch(*templ++)
			{
			case 0: return true;
			case '.': if(*templ == 0) return *s == 0; // force end of string
			case '?': if(*s++ == 0) return false; break;
			case '*':
				do
					if(RawMatches(s, templ))
						return true;
				while(*s++);
				return false;
			case '\\': if(*templ == 0 || *templ++ != *s++) return false; break;
			default: if(templ[-1] != ToUpper(*s++)) return false;
				break;
			}
		return true;
	}

private:
	const wchar *raw_templ;
//	bool         exclude_ch;
//	bool         cvt_ascii;
	WString      cvt_templ;
};

One<LanguageInfo::WildcardCompare> LanguageInfoCS::GetWildcardCompare(const wchar *wildcard_text) const
{
	return new WildcardCompareCS(wildcard_text);
}

class LanguageInfoEN : public LanguageInfo
{
public:
	LanguageInfoEN()
		: LanguageInfo(LNG_('E', 'N', 'U', 'S'))
	{
		english_name = "American English";
		native_name = ToUnicode("American English", CHARSET_DEFAULT);
//		thousand_separator = " ";
//		decimal_point = ".";
	}
};

class LanguageInfoEB : public LanguageInfo
{
public:
	LanguageInfoEB()
		: LanguageInfo(LNG_('E', 'N', 'G', 'B'))
	{
		english_name = "British English";
		native_name = ToUnicode("British English", CHARSET_DEFAULT);
//		thousand_separator = " ";
//		decimal_point = ".";
	}
};

class LanguageInfoEA : public LanguageInfo
{
public:
	LanguageInfoEA()
		: LanguageInfo(LNG_('E', 'N', 'A', 'U'))
	{
		english_name = "Australian English";
		native_name = ToUnicode("Australian English", CHARSET_DEFAULT);
//		thousand_separator = " ";
//		decimal_point = ".";
	}
};

class LanguageInfoEC : public LanguageInfo
{
public:
	LanguageInfoEC()
		: LanguageInfo(LNG_('E', 'N', 'C', 'A'))
	{
		english_name = "Canadian English";
		native_name = ToUnicode("Canadian English", CHARSET_DEFAULT);
//		thousand_separator = " ";
//		decimal_point = ".";
	}
};

class LanguageInfoEZ : public LanguageInfo
{
public:
	LanguageInfoEZ()
		: LanguageInfo(LNG_('E', 'N', 'N', 'Z'))
	{
		english_name = "New Zealand English";
		native_name = ToUnicode("New Zealand English", CHARSET_DEFAULT);
	}
};

class LanguageInfoEI : public LanguageInfo
{
public:
	LanguageInfoEI()
		: LanguageInfo(LNG_('E', 'N', 'I', 'E'))
	{
		english_name = "Irish English";
		native_name = ToUnicode("Irish English", CHARSET_DEFAULT);
	}
};

class LanguageInfoEO : public LanguageInfo
{
public:
	LanguageInfoEO()
		: LanguageInfo(LNG_('E', 'N', 'Z', 'A'))
	{
		english_name = "South African English";
		native_name = ToUnicode("South African English", CHARSET_DEFAULT);
	}
};

class LanguageInfoEJ : public LanguageInfo
{
public:
	LanguageInfoEJ()
		: LanguageInfo(LNG_('E', 'N', 'J', 'M'))
	{
		english_name = "Jamaican English";
		native_name = ToUnicode("Jamaican English", CHARSET_DEFAULT);
	}
};

class LanguageInfoER : public LanguageInfo
{
public:
	LanguageInfoER()
		: LanguageInfo(LNG_('E', 'N', 'C', 'B'))
	{
		english_name = "Caribbean English";
		native_name = ToUnicode("Caribbean English", CHARSET_DEFAULT);
	}
};

class LanguageInfoEY : public LanguageInfo
{
public:
	LanguageInfoEY()
		: LanguageInfo(LNG_('E', 'N', 'B', 'Z'))
	{
		english_name = "Belizian English";
		native_name = ToUnicode("Belizian English", CHARSET_DEFAULT);
	}
};

class LanguageInfoET : public LanguageInfo
{
public:
	LanguageInfoET()
		: LanguageInfo(LNG_('E', 'N', 'T', 'T'))
	{
		english_name = "Trinidadian English";
		native_name = ToUnicode("Trinidadian English", CHARSET_DEFAULT);
	}
};

class LanguageInfoBG : public LanguageInfo
{
public:
	LanguageInfoBG()
		: LanguageInfo(LNG_('B', 'G', 'B', 'G'))
	{
		english_name = "Bulgarian";
		native_name = ToUnicode("Blugarian", CHARSET_DEFAULT);
	}
};

class LanguageInfoDA : public LanguageInfo
{
public:
	LanguageInfoDA()
		: LanguageInfo(LNG_('D', 'A', 'D', 'K'))
	{
		english_name = "Danish";
		native_name = ToUnicode("Danish", CHARSET_DEFAULT);
	}
};

class LanguageInfoGE : public LanguageInfo
{
public:
	LanguageInfoGE()
		: LanguageInfo(LNG_('D', 'E', 'D', 'E'))
	{
		english_name = "German";
		native_name = ToUnicode("Deutsch", CHARSET_WIN1250);
	}
};

class LanguageInfoGS : public LanguageInfo
{
public:
	LanguageInfoGS()
		: LanguageInfo(LNG_('D', 'E', 'C', 'H'))
	{
		english_name = "Swiss German";
		native_name = ToUnicode("Schweizerisch Deutsch", CHARSET_WIN1250);
	}
};

class LanguageInfoGA : public LanguageInfo
{
public:
	LanguageInfoGA()
		: LanguageInfo(LNG_('D', 'E', 'A', 'T'))
	{
		english_name = "Austrian German";
		native_name = "\xC3\x96\x73\x74\x65\x72\x72\x65\x69\x63\x68\x69\x73\x63\x68\x65\x73\x20\x44\x65\x75\x74\x73\x68";
	}
};

class LanguageInfoGL : public LanguageInfo
{
public:
	LanguageInfoGL()
		: LanguageInfo(LNG_('D', 'E', 'L', 'U'))
	{
		english_name = "Luxemburgish";
		native_name = ToUnicode("Luxemburgisch", CHARSET_WIN1250);
	}
};

class LanguageInfoGI : public LanguageInfo
{
public:
	LanguageInfoGI()
		: LanguageInfo(LNG_('D', 'E', 'L', 'I'))
	{
		english_name = "Liechtensteinian German";
		native_name = ToUnicode("Liechtensteinian Deutsch", CHARSET_WIN1250);
	}
};

class LanguageInfoGK : public LanguageInfo
{
public:
	LanguageInfoGK()
		: LanguageInfo(LNG_('E', 'L', 'G', 'R'))
	{
		english_name = "Greek";
		native_name = "\xCE\xB5\xCE\xBB\xCE\xBB\xCE\xB7\xCE\xBD\xCE\xB9\xCE\xBA\xCF\x8C\x82";
	}
};

class LanguageInfoFR : public LanguageInfo
{
public:
	LanguageInfoFR()
		: LanguageInfo(LNG_('F', 'R', 'F', 'R'))
	{
		english_name = "French";
		native_name = "Français";
	}
};

class LanguageInfoFB : public LanguageInfo
{
public:
	LanguageInfoFB()
		: LanguageInfo(LNG_('F', 'R', 'B', 'E'))
	{
		english_name = "Belgian French";
		native_name = ToUnicode("Français de Belgique", CHARSET_DEFAULT);
	}
};

class LanguageInfoFA : public LanguageInfo
{
public:
	LanguageInfoFA()
		: LanguageInfo(LNG_('F', 'R', 'C', 'A'))
	{
		english_name = "Canadian French";
		native_name = ToUnicode("Français Canadien", CHARSET_DEFAULT);
	}
};

class LanguageInfoFC : public LanguageInfo
{
public:
	LanguageInfoFC()
		: LanguageInfo(LNG_('F', 'R', 'C', 'H'))
	{
		english_name = "Swiss French";
		native_name = ToUnicode("Français Suisse", CHARSET_DEFAULT);
	}
};

class LanguageInfoFL : public LanguageInfo
{
public:
	LanguageInfoFL()
		: LanguageInfo(LNG_('F', 'R', 'L', 'U'))
	{
		english_name = "Luxemburgisch French";
		native_name = ToUnicode("Français Luxembourgeois", CHARSET_DEFAULT);
	}
};

class LanguageInfoHU : public LanguageInfo
{
public:
	LanguageInfoHU()
		: LanguageInfo(LNG_('H', 'U', 'H', 'U'))
	{
		english_name = "Hungarian";
		native_name = ToUnicode("Hungarian", CHARSET_DEFAULT);
	}
};

class LanguageInfoIS : public LanguageInfo
{
public:
	LanguageInfoIS()
		: LanguageInfo(LNG_('I', 'S', 'I', 'S'))
	{
		english_name = "Icelandic";
		native_name = ToUnicode("Icelandic", CHARSET_DEFAULT);
	}
};

class LanguageInfoIT : public LanguageInfo
{
public:
	LanguageInfoIT()
		: LanguageInfo(LNG_('I', 'T', 'I', 'T'))
	{
		english_name = "Italian";
		native_name = ToUnicode("Italiano", CHARSET_DEFAULT);
	}
};

class LanguageInfoIC : public LanguageInfo
{
public:
	LanguageInfoIC()
		: LanguageInfo(LNG_('I', 'T', 'C', 'H'))
	{
		english_name = "Swiss Italian";
		native_name = ToUnicode("Svizzera italiana", CHARSET_DEFAULT);
	}
};

class LanguageInfoES : public LanguageInfo
{
public:
	LanguageInfoES()
		: LanguageInfo(LNG_('E', 'S', 'E', 'S'))
	{
		english_name = "Spanish";
		native_name = ToUnicode("Español", CHARSET_WIN1250);
	}
};

class LanguageInfoSM : public LanguageInfo
{
public:
	LanguageInfoSM()
		: LanguageInfo(LNG_('E', 'S', 'M', 'X'))
	{
		english_name = "Mexican Spanish";
		native_name = ToUnicode("Español mexicano", CHARSET_WIN1250);
	}
};

class LanguageInfoSG : public LanguageInfo
{
public:
	LanguageInfoSG()
		: LanguageInfo(LNG_('E', 'S', 'G', 'T'))
	{
		english_name = "Guatemalan Spanish";
		native_name = ToUnicode("Español guatemalteco", CHARSET_WIN1250);
	}
};

class LanguageInfoSC : public LanguageInfo
{
public:
	LanguageInfoSC()
		: LanguageInfo(LNG_('E', 'S', 'C', 'R'))
	{
		english_name = "Costa Rican Spanish";
		native_name = ToUnicode("Español costarricense", CHARSET_WIN1250);
	}
};

class LanguageInfoSP : public LanguageInfo
{
public:
	LanguageInfoSP()
		: LanguageInfo(LNG_('E', 'S', 'P', 'A'))
	{
		english_name = "Panamanian Spanish";
		native_name = ToUnicode("Español panameño", CHARSET_WIN1250);
	}
};

class LanguageInfoSD : public LanguageInfo
{
public:
	LanguageInfoSD()
		: LanguageInfo(LNG_('E', 'S', 'D', 'O'))
	{
		english_name = "Dominican Spanish";
		native_name = ToUnicode("Español dominicano", CHARSET_WIN1250);
	}
};

class LanguageInfoSZ : public LanguageInfo
{
public:
	LanguageInfoSZ()
		: LanguageInfo(LNG_('E', 'S', 'V', 'E'))
	{
		english_name = "Venezuelan Spanish";
		native_name = ToUnicode("Español venezolano", CHARSET_WIN1250);
	}
};

class LanguageInfoSO : public LanguageInfo
{
public:
	LanguageInfoSO()
		: LanguageInfo(LNG_('E', 'S', 'C', 'O'))
	{
		english_name = "Colombian Spanish";
		native_name = ToUnicode("Español colombiano", CHARSET_WIN1250);
	}
};

class LanguageInfoSE : public LanguageInfo
{
public:
	LanguageInfoSE()
		: LanguageInfo(LNG_('E', 'S', 'P', 'E'))
	{
		english_name = "Peruvian Spanish";
		native_name = ToUnicode("Español peruano", CHARSET_WIN1250);
	}
};

class LanguageInfoSA : public LanguageInfo
{
public:
	LanguageInfoSA()
		: LanguageInfo(LNG_('E', 'S', 'A', 'R'))
	{
		english_name = "Argentinian Spanish";
		native_name = ToUnicode("Español argentino", CHARSET_WIN1250);
	}
};

class LanguageInfoSR : public LanguageInfo
{
public:
	LanguageInfoSR()
		: LanguageInfo(LNG_('E', 'S', 'E', 'C'))
	{
		english_name = "Ecuadorian Spanish";
		native_name = ToUnicode("Español ecuatoriano", CHARSET_WIN1250);
	}
};

class LanguageInfoSL : public LanguageInfo
{
public:
	LanguageInfoSL()
		: LanguageInfo(LNG_('E', 'S', 'C', 'L'))
	{
		english_name = "Chilian Spanish";
		native_name = ToUnicode("Español chileno", CHARSET_WIN1250);
	}
};

class LanguageInfoSU : public LanguageInfo
{
public:
	LanguageInfoSU()
		: LanguageInfo(LNG_('E', 'S', 'U', 'Y'))
	{
		english_name = "Uruguayan Spanish";
		native_name = ToUnicode("Español uruguayo", CHARSET_WIN1250);
	}
};

class LanguageInfoSY : public LanguageInfo
{
public:
	LanguageInfoSY()
		: LanguageInfo(LNG_('E', 'S', 'P', 'Y'))
	{
		english_name = "Paraguayan Spanish";
		native_name = ToUnicode("Español paraguayo", CHARSET_WIN1250);
	}
};

class LanguageInfoSB : public LanguageInfo
{
public:
	LanguageInfoSB()
		: LanguageInfo(LNG_('E', 'S', 'B', 'O'))
	{
		english_name = "Bolivian Spanish";
		native_name = ToUnicode("Español boliviano", CHARSET_WIN1250);
	}
};

class LanguageInfoSV : public LanguageInfo
{
public:
	LanguageInfoSV()
		: LanguageInfo(LNG_('E', 'S', 'S', 'V'))
	{
		english_name = "Salvadorean Spanish";
		native_name = ToUnicode("Español salvadoreño", CHARSET_WIN1250);
	}
};

class LanguageInfoSH : public LanguageInfo
{
public:
	LanguageInfoSH()
		: LanguageInfo(LNG_('E', 'S', 'H', 'N'))
	{
		english_name = "Honduranian Spanish";
		native_name = ToUnicode("Español hondureño", CHARSET_WIN1250);
	}
};

class LanguageInfoSN : public LanguageInfo
{
public:
	LanguageInfoSN()
		: LanguageInfo(LNG_('E', 'S', 'N', 'I'))
	{
		english_name = "Nicaraguan Spanish";
		native_name = ToUnicode("Español nicaragüense", CHARSET_WIN1250);
	}
};

class LanguageInfoSI : public LanguageInfo
{
public:
	LanguageInfoSI()
		: LanguageInfo(LNG_('E', 'S', 'P', 'R'))
	{
		english_name = "Puerto Rican Spanish";
		native_name = ToUnicode("Español puertorriqueño", CHARSET_WIN1250);
	}
};

class LanguageInfoFI : public LanguageInfo
{
public:
	LanguageInfoFI()
		: LanguageInfo(LNG_('F', 'I', 'F', 'I'))
	{
		english_name = "Finnish";
		native_name = ToUnicode("Finnish", CHARSET_DEFAULT);
	}
};

class LanguageInfoRU : public LanguageInfo
{
public:
	LanguageInfoRU()
		: LanguageInfo(LNG_('R', 'U', 'R', 'U'))
	{
		english_name = "Russian";
		native_name = "\xD0\xA0\xD1\x83\xD1\x81\xD1\x81\xD0\xBA\xD0\xB8\xD0\xB9";
	}
};

class LanguageInfoUK : public LanguageInfo
{
public:
	LanguageInfoUK()
		: LanguageInfo(LNG_('U', 'K', 'U', 'A'))
	{
		english_name = "Ukrainian";
		native_name = "\xD0\xA3\xD0\xBA\xD1\x80\xD0\xB0\xD1\x97\xD0\xBD\xD1\x81\xD1\x8C\xD0\xBA\xD0\xB0";
	}
};

class LanguageInfoZC : public LanguageInfo
{
public:
	LanguageInfoZC()
		: LanguageInfo(LNG_('Z', 'H', 'C', 'N'))
	{
		english_name = "Simplified Chinese";
		native_name = "\xE7\xAE\x80\xE4\xBD\x93\xE4\xB8\xAD\xE6\x96\x87";
	}
};

class LanguageInfoZT : public LanguageInfo
{
public:
	LanguageInfoZT()
		: LanguageInfo(LNG_('Z', 'H', 'T', 'W'))
	{
		english_name = "Traditional Chinese";
		native_name = "\xE7\xB9\x81\xE9\xAB\x94\xE4\xB8\xAD\xE6\x96\x87";
	}
};

class LanguageInfoJP : public LanguageInfo
{
public:
	LanguageInfoJP()
		: LanguageInfo(LNG_('J', 'A', 'J', 'P'))
	{
		english_name = "Japanese";
		native_name = "\xE6\x97\xA5\xE8\xAA\x9E";
	}
};

class LanguageInfoEU : public LanguageInfo
{
public:
	LanguageInfoEU()
		: LanguageInfo(LNG_('E', 'U', '\0', '\0'))
	{
		english_name = "Basque";
		native_name = ToUnicode("Euskara", CHARSET_DEFAULT);
	}
};

const LanguageInfo& GetLanguageInfo(int lcode)
{
	static bool init_std = false;
	if(!init_std)
	{
		init_std = true;
		LanguageInfo::Register(new LanguageInfoCS);
		LanguageInfo::Register(new LanguageInfoEN);
		LanguageInfo::Register(new LanguageInfoEB);
		LanguageInfo::Register(new LanguageInfoEA);
		LanguageInfo::Register(new LanguageInfoEC);
		LanguageInfo::Register(new LanguageInfoEZ);
		LanguageInfo::Register(new LanguageInfoEI);
		LanguageInfo::Register(new LanguageInfoEO);
		LanguageInfo::Register(new LanguageInfoEJ);
		LanguageInfo::Register(new LanguageInfoER);
		LanguageInfo::Register(new LanguageInfoEY);
		LanguageInfo::Register(new LanguageInfoET);
		LanguageInfo::Register(new LanguageInfoBG);
		LanguageInfo::Register(new LanguageInfoDA);
		LanguageInfo::Register(new LanguageInfoGE);
		LanguageInfo::Register(new LanguageInfoGS);
		LanguageInfo::Register(new LanguageInfoGA);
		LanguageInfo::Register(new LanguageInfoGL);
		LanguageInfo::Register(new LanguageInfoGI);
		LanguageInfo::Register(new LanguageInfoGK);
		LanguageInfo::Register(new LanguageInfoFR);
		LanguageInfo::Register(new LanguageInfoFB);
		LanguageInfo::Register(new LanguageInfoFA);
		LanguageInfo::Register(new LanguageInfoFC);
		LanguageInfo::Register(new LanguageInfoFL);
		LanguageInfo::Register(new LanguageInfoHU);
		LanguageInfo::Register(new LanguageInfoIS);
		LanguageInfo::Register(new LanguageInfoIT);
		LanguageInfo::Register(new LanguageInfoIC);
		LanguageInfo::Register(new LanguageInfoES);
		LanguageInfo::Register(new LanguageInfoSM);
		LanguageInfo::Register(new LanguageInfoSG);
		LanguageInfo::Register(new LanguageInfoSC);
		LanguageInfo::Register(new LanguageInfoSP);
		LanguageInfo::Register(new LanguageInfoSD);
		LanguageInfo::Register(new LanguageInfoSZ);
		LanguageInfo::Register(new LanguageInfoSO);
		LanguageInfo::Register(new LanguageInfoSE);
		LanguageInfo::Register(new LanguageInfoSA);
		LanguageInfo::Register(new LanguageInfoSR);
		LanguageInfo::Register(new LanguageInfoSL);
		LanguageInfo::Register(new LanguageInfoSU);
		LanguageInfo::Register(new LanguageInfoSY);
		LanguageInfo::Register(new LanguageInfoSB);
		LanguageInfo::Register(new LanguageInfoSV);
		LanguageInfo::Register(new LanguageInfoSH);
		LanguageInfo::Register(new LanguageInfoSN);
		LanguageInfo::Register(new LanguageInfoSI);
		LanguageInfo::Register(new LanguageInfoFI);
		LanguageInfo::Register(new LanguageInfoRU);
		LanguageInfo::Register(new LanguageInfoUK);
		LanguageInfo::Register(new LanguageInfoZC);
		LanguageInfo::Register(new LanguageInfoZT);
		LanguageInfo::Register(new LanguageInfoJP);
		LanguageInfo::Register(new LanguageInfoEU);
	}

	static int recent = 0;
	LanguageInfo *rinfo = 0;
	if(rinfo && lcode == recent)
		return *rinfo;
	if(lcode == 0)
		lcode = GetCurrentLanguage(); //!! todo - decide on default / neutral language code
	INTERLOCKED_(sMapMutex) {
		LanguageInfoMap& map = LanguageInfo::Map();
		int f = map.Find(lcode);
		if(f < 0)
		{
			f  = map.GetCount();
			map.Add(lcode, new LanguageInfo(lcode));
		}
		recent = lcode;
		return *(rinfo = &map[f]);
	}
	return *rinfo;
}

const LanguageInfo& GetLanguageInfo()
{
	return GetLanguageInfo(GetCurrentLanguage());
}

END_UPP_NAMESPACE
