2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
|
|
*
|
2012-05-21 04:12:37 -07:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "nsBidiKeyboard.h"
|
|
|
|
#include "prmem.h"
|
2008-08-29 14:26:56 -07:00
|
|
|
#include <tchar.h>
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS1(nsBidiKeyboard, nsIBidiKeyboard)
|
|
|
|
|
|
|
|
nsBidiKeyboard::nsBidiKeyboard() : nsIBidiKeyboard()
|
|
|
|
{
|
2011-10-01 19:16:19 -07:00
|
|
|
mInitialized = false;
|
|
|
|
mHaveBidiKeyboards = false;
|
2007-03-22 10:30:00 -07:00
|
|
|
mLTRKeyboard[0] = '\0';
|
|
|
|
mRTLKeyboard[0] = '\0';
|
|
|
|
mCurrentLocaleName[0] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
nsBidiKeyboard::~nsBidiKeyboard()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-08-22 08:56:38 -07:00
|
|
|
NS_IMETHODIMP nsBidiKeyboard::SetLangFromBidiLevel(uint8_t aLevel)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsresult result = SetupBidiKeyboards();
|
|
|
|
if (NS_FAILED(result))
|
|
|
|
return result;
|
|
|
|
|
|
|
|
// call LoadKeyboardLayout() only if the target keyboard layout is different from the current
|
2008-08-29 14:26:56 -07:00
|
|
|
PRUnichar currentLocaleName[KL_NAMELENGTH];
|
|
|
|
wcsncpy(currentLocaleName, (aLevel & 1) ? mRTLKeyboard : mLTRKeyboard, KL_NAMELENGTH);
|
2007-03-22 10:30:00 -07:00
|
|
|
currentLocaleName[KL_NAMELENGTH-1] = '\0'; // null terminate
|
|
|
|
|
|
|
|
NS_ASSERTION(*currentLocaleName,
|
|
|
|
"currentLocaleName has string length == 0");
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* This implementation of automatic keyboard layout switching is too buggy to be useful
|
|
|
|
and the feature itself is inconsistent with Windows. See Bug 162242 */
|
|
|
|
if (strcmp(mCurrentLocaleName, currentLocaleName)) {
|
|
|
|
if (!::LoadKeyboardLayout(currentLocaleName, KLF_ACTIVATE | KLF_SUBSTITUTE_OK)) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
NS_IMETHODIMP nsBidiKeyboard::IsLangRTL(bool *aIsRTL)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2011-10-01 19:16:19 -07:00
|
|
|
*aIsRTL = false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsresult result = SetupBidiKeyboards();
|
|
|
|
if (NS_FAILED(result))
|
|
|
|
return result;
|
|
|
|
|
|
|
|
HKL currentLocale;
|
|
|
|
|
|
|
|
currentLocale = ::GetKeyboardLayout(0);
|
|
|
|
*aIsRTL = IsRTLLanguage(currentLocale);
|
|
|
|
|
2008-08-29 14:26:56 -07:00
|
|
|
if (!::GetKeyboardLayoutNameW(mCurrentLocaleName))
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
NS_ASSERTION(*mCurrentLocaleName,
|
|
|
|
"GetKeyboardLayoutName return string length == 0");
|
2008-08-29 14:26:56 -07:00
|
|
|
NS_ASSERTION((wcslen(mCurrentLocaleName) < KL_NAMELENGTH),
|
2007-03-22 10:30:00 -07:00
|
|
|
"GetKeyboardLayoutName return string length >= KL_NAMELENGTH");
|
|
|
|
|
|
|
|
// The language set by the user overrides the default language for that direction
|
|
|
|
if (*aIsRTL) {
|
2008-08-29 14:26:56 -07:00
|
|
|
wcsncpy(mRTLKeyboard, mCurrentLocaleName, KL_NAMELENGTH);
|
2007-03-22 10:30:00 -07:00
|
|
|
mRTLKeyboard[KL_NAMELENGTH-1] = '\0'; // null terminate
|
|
|
|
} else {
|
2008-08-29 14:26:56 -07:00
|
|
|
wcsncpy(mLTRKeyboard, mCurrentLocaleName, KL_NAMELENGTH);
|
2007-03-22 10:30:00 -07:00
|
|
|
mLTRKeyboard[KL_NAMELENGTH-1] = '\0'; // null terminate
|
|
|
|
}
|
|
|
|
|
2008-08-29 14:26:56 -07:00
|
|
|
NS_ASSERTION((wcslen(mRTLKeyboard) < KL_NAMELENGTH),
|
2007-03-22 10:30:00 -07:00
|
|
|
"mLTRKeyboard has string length >= KL_NAMELENGTH");
|
2008-08-29 14:26:56 -07:00
|
|
|
NS_ASSERTION((wcslen(mLTRKeyboard) < KL_NAMELENGTH),
|
2007-03-22 10:30:00 -07:00
|
|
|
"mRTLKeyboard has string length >= KL_NAMELENGTH");
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
NS_IMETHODIMP nsBidiKeyboard::GetHaveBidiKeyboards(bool* aResult)
|
2011-09-15 07:54:50 -07:00
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER(aResult);
|
|
|
|
|
|
|
|
nsresult result = SetupBidiKeyboards();
|
|
|
|
if (NS_FAILED(result))
|
|
|
|
return result;
|
|
|
|
|
|
|
|
*aResult = mHaveBidiKeyboards;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// Get the list of keyboard layouts available in the system
|
|
|
|
// Set mLTRKeyboard to the first LTR keyboard in the list and mRTLKeyboard to the first RTL keyboard in the list
|
|
|
|
// These defaults will be used unless the user explicitly sets something else.
|
|
|
|
nsresult nsBidiKeyboard::SetupBidiKeyboards()
|
|
|
|
{
|
|
|
|
if (mInitialized)
|
|
|
|
return mHaveBidiKeyboards ? NS_OK : NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
int keyboards;
|
|
|
|
HKL far* buf;
|
|
|
|
HKL locale;
|
2008-08-29 14:26:56 -07:00
|
|
|
PRUnichar localeName[KL_NAMELENGTH];
|
2011-09-28 23:19:26 -07:00
|
|
|
bool isLTRKeyboardSet = false;
|
|
|
|
bool isRTLKeyboardSet = false;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// GetKeyboardLayoutList with 0 as first parameter returns the number of keyboard layouts available
|
2012-07-30 07:20:58 -07:00
|
|
|
keyboards = ::GetKeyboardLayoutList(0, nullptr);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!keyboards)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
// allocate a buffer to hold the list
|
|
|
|
buf = (HKL far*) PR_Malloc(keyboards * sizeof(HKL));
|
|
|
|
if (!buf)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
// Call again to fill the buffer
|
|
|
|
if (::GetKeyboardLayoutList(keyboards, buf) != keyboards) {
|
|
|
|
PR_Free(buf);
|
|
|
|
return NS_ERROR_UNEXPECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Go through the list and pick a default LTR and RTL keyboard layout
|
|
|
|
while (keyboards--) {
|
|
|
|
locale = buf[keyboards];
|
|
|
|
if (IsRTLLanguage(locale)) {
|
2008-08-29 14:26:56 -07:00
|
|
|
_snwprintf(mRTLKeyboard, KL_NAMELENGTH, L"%.*x", KL_NAMELENGTH - 1,
|
2010-06-25 05:13:17 -07:00
|
|
|
LANGIDFROMLCID((DWORD_PTR)locale));
|
2011-10-01 19:16:19 -07:00
|
|
|
isRTLKeyboardSet = true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
else {
|
2008-08-29 14:26:56 -07:00
|
|
|
_snwprintf(mLTRKeyboard, KL_NAMELENGTH, L"%.*x", KL_NAMELENGTH - 1,
|
2010-06-25 05:13:17 -07:00
|
|
|
LANGIDFROMLCID((DWORD_PTR)locale));
|
2011-10-01 19:16:19 -07:00
|
|
|
isLTRKeyboardSet = true;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
PR_Free(buf);
|
2011-10-01 19:16:19 -07:00
|
|
|
mInitialized = true;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// If there is not at least one keyboard of each directionality, Bidi
|
|
|
|
// keyboard functionality will be disabled.
|
|
|
|
mHaveBidiKeyboards = (isRTLKeyboardSet && isLTRKeyboardSet);
|
|
|
|
if (!mHaveBidiKeyboards)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
// Get the current keyboard layout and use it for either mRTLKeyboard or
|
|
|
|
// mLTRKeyboard as appropriate. If the user has many keyboard layouts
|
|
|
|
// installed this prevents us from arbitrarily resetting the current
|
|
|
|
// layout (bug 80274)
|
|
|
|
locale = ::GetKeyboardLayout(0);
|
2008-08-29 14:26:56 -07:00
|
|
|
if (!::GetKeyboardLayoutNameW(localeName))
|
2007-03-22 10:30:00 -07:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
NS_ASSERTION(*localeName,
|
|
|
|
"GetKeyboardLayoutName return string length == 0");
|
2008-08-29 14:26:56 -07:00
|
|
|
NS_ASSERTION((wcslen(localeName) < KL_NAMELENGTH),
|
2007-03-22 10:30:00 -07:00
|
|
|
"GetKeyboardLayout return string length >= KL_NAMELENGTH");
|
|
|
|
|
|
|
|
if (IsRTLLanguage(locale)) {
|
2008-08-29 14:26:56 -07:00
|
|
|
wcsncpy(mRTLKeyboard, localeName, KL_NAMELENGTH);
|
2007-03-22 10:30:00 -07:00
|
|
|
mRTLKeyboard[KL_NAMELENGTH-1] = '\0'; // null terminate
|
|
|
|
}
|
|
|
|
else {
|
2008-08-29 14:26:56 -07:00
|
|
|
wcsncpy(mLTRKeyboard, localeName, KL_NAMELENGTH);
|
2007-03-22 10:30:00 -07:00
|
|
|
mLTRKeyboard[KL_NAMELENGTH-1] = '\0'; // null terminate
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ASSERTION(*mRTLKeyboard,
|
|
|
|
"mLTRKeyboard has string length == 0");
|
|
|
|
NS_ASSERTION(*mLTRKeyboard,
|
|
|
|
"mLTRKeyboard has string length == 0");
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test whether the language represented by this locale identifier is a
|
|
|
|
// right-to-left language, using bit 123 of the Unicode subset bitfield in
|
|
|
|
// the LOCALESIGNATURE
|
|
|
|
// See http://msdn.microsoft.com/library/default.asp?url=/library/en-us/intl/unicode_63ub.asp
|
2011-09-28 23:19:26 -07:00
|
|
|
bool nsBidiKeyboard::IsRTLLanguage(HKL aLocale)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
LOCALESIGNATURE localesig;
|
2010-06-25 05:13:17 -07:00
|
|
|
return (::GetLocaleInfoW(PRIMARYLANGID((DWORD_PTR)aLocale),
|
2007-03-22 10:30:00 -07:00
|
|
|
LOCALE_FONTSIGNATURE,
|
|
|
|
(LPWSTR)&localesig,
|
|
|
|
(sizeof(localesig)/sizeof(WCHAR))) &&
|
|
|
|
(localesig.lsUsb[3] & 0x08000000));
|
|
|
|
}
|