reactos/dll/cpl/input/settings_page.c
Katayama Hirofumi MZ 4d9026c852 [CPL:INPUT] Return TRUE against WM_INITDIALOG and add WS_TABSTOP
This affects control focus.
2022-10-30 08:01:57 +09:00

668 lines
20 KiB
C

/*
* PROJECT: input.dll
* FILE: dll/cpl/input/settings_page.c
* PURPOSE: input.dll
* PROGRAMMERS: Dmitry Chapyshev (dmitry@reactos.org)
* Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
*/
#include "input.h"
#include "layout_list.h"
#include "locale_list.h"
#include "input_list.h"
static INT s_nAliveLeafCount = 0;
static INT s_nRootCount = 0;
static INT s_iKeyboardImage = -1;
static INT s_iDotImage = -1;
static BOOL s_bDefaultInputChanged = FALSE;
static HICON
CreateLayoutIcon(LANGID LangID)
{
WCHAR szBuf[4];
HDC hdcScreen, hdc;
HBITMAP hbmColor, hbmMono, hBmpOld;
HFONT hFont, hFontOld;
LOGFONTW lf;
RECT rect;
ICONINFO IconInfo;
HICON hIcon;
INT cxIcon = GetSystemMetrics(SM_CXSMICON);
INT cyIcon = GetSystemMetrics(SM_CYSMICON);
/* Getting "EN", "FR", etc. from English, French, ... */
if (GetLocaleInfoW(LangID,
LOCALE_SABBREVLANGNAME | LOCALE_NOUSEROVERRIDE,
szBuf,
ARRAYSIZE(szBuf)) == 0)
{
szBuf[0] = szBuf[1] = L'?';
}
szBuf[2] = UNICODE_NULL; /* Truncate the identifier to two characters: "ENG" --> "EN" etc. */
/* Create hdc, hbmColor and hbmMono */
hdcScreen = GetDC(NULL);
hdc = CreateCompatibleDC(hdcScreen);
hbmColor = CreateCompatibleBitmap(hdcScreen, cxIcon, cyIcon);
ReleaseDC(NULL, hdcScreen);
hbmMono = CreateBitmap(cxIcon, cyIcon, 1, 1, NULL);
/* Checking NULL */
if (!hdc || !hbmColor || !hbmMono)
{
if (hbmMono)
DeleteObject(hbmMono);
if (hbmColor)
DeleteObject(hbmColor);
if (hdc)
DeleteDC(hdc);
return NULL;
}
/* Create a font */
hFont = NULL;
if (SystemParametersInfoW(SPI_GETICONTITLELOGFONT, sizeof(lf), &lf, 0))
{
/* Override the current size with something manageable */
lf.lfHeight = -11;
lf.lfWidth = 0;
hFont = CreateFontIndirectW(&lf);
}
if (!hFont)
hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT);
SetRect(&rect, 0, 0, cxIcon, cyIcon);
/* Draw hbmColor */
hBmpOld = SelectObject(hdc, hbmColor);
SetDCBrushColor(hdc, GetSysColor(COLOR_HIGHLIGHT));
FillRect(hdc, &rect, (HBRUSH)GetStockObject(DC_BRUSH));
hFontOld = SelectObject(hdc, hFont);
SetTextColor(hdc, GetSysColor(COLOR_HIGHLIGHTTEXT));
SetBkMode(hdc, TRANSPARENT);
DrawTextW(hdc, szBuf, 2, &rect, DT_SINGLELINE | DT_CENTER | DT_VCENTER);
SelectObject(hdc, hFontOld);
/* Fill hbmMono with black */
SelectObject(hdc, hbmMono);
PatBlt(hdc, 0, 0, cxIcon, cyIcon, BLACKNESS);
SelectObject(hdc, hBmpOld);
/* Create an icon from hbmColor and hbmMono */
IconInfo.fIcon = TRUE;
IconInfo.xHotspot = IconInfo.yHotspot = 0;
IconInfo.hbmColor = hbmColor;
IconInfo.hbmMask = hbmMono;
hIcon = CreateIconIndirect(&IconInfo);
/* Clean up */
DeleteObject(hFont);
DeleteObject(hbmMono);
DeleteObject(hbmColor);
DeleteDC(hdc);
return hIcon;
}
static VOID InitDefaultLangComboBox(HWND hwndCombo)
{
WCHAR szText[256];
INPUT_LIST_NODE *pNode;
INT iIndex, nCount, iDefault = (INT)SendMessageW(hwndCombo, CB_GETCURSEL, 0, 0);
SendMessageW(hwndCombo, CB_RESETCONTENT, 0, 0);
for (pNode = InputList_GetFirst(); pNode != NULL; pNode = pNode->pNext)
{
if (pNode->wFlags & INPUT_LIST_NODE_FLAG_DELETED)
continue;
StringCchPrintfW(szText, _countof(szText), L"%s - %s",
pNode->pLocale->pszName, pNode->pLayout->pszName);
iIndex = (INT)SendMessageW(hwndCombo, CB_ADDSTRING, 0, (LPARAM)szText);
SendMessageW(hwndCombo, CB_SETITEMDATA, iIndex, (LPARAM)pNode);
if (pNode->wFlags & INPUT_LIST_NODE_FLAG_DEFAULT)
iDefault = iIndex;
}
nCount = (INT)SendMessageW(hwndCombo, CB_GETCOUNT, 0, 0);
if (iDefault >= nCount)
SendMessageW(hwndCombo, CB_SETCURSEL, nCount - 1, 0);
else
SendMessageW(hwndCombo, CB_SETCURSEL, iDefault, 0);
}
static VOID
SetControlsState(HWND hwndDlg)
{
HWND hwndList = GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST);
HWND hwndCombo = GetDlgItem(hwndDlg, IDC_DEFAULT_LANGUAGE);
BOOL bIsLeaf, bCanRemove, bCanProp;
HTREEITEM hSelected = TreeView_GetSelection(hwndList);
TV_ITEM item = { TVIF_PARAM | TVIF_HANDLE };
item.hItem = hSelected;
bIsLeaf = (hSelected && TreeView_GetItem(hwndList, &item) && HIWORD(item.lParam));
bCanRemove = (bIsLeaf && (s_nAliveLeafCount > 1)) || (s_nRootCount > 1);
bCanProp = bIsLeaf;
EnableWindow(GetDlgItem(hwndDlg, IDC_REMOVE_BUTTON), bCanRemove);
EnableWindow(GetDlgItem(hwndDlg, IDC_PROP_BUTTON), bCanProp);
InitDefaultLangComboBox(hwndCombo);
}
static BOOL CALLBACK
EnumResNameProc(HMODULE hModule, LPCWSTR lpszType, LPWSTR lpszName, LONG_PTR lParam)
{
HICON* phIconSm = (HICON*)lParam;
if (*phIconSm)
return FALSE;
*phIconSm = (HICON)LoadImageW(hModule, lpszName, IMAGE_ICON,
GetSystemMetrics(SM_CXSMICON),
GetSystemMetrics(SM_CYSMICON),
0);
return TRUE;
}
static HICON LoadIMEIcon(LPCTSTR pszImeFile)
{
WCHAR szSysDir[MAX_PATH], szPath[MAX_PATH];
HINSTANCE hImeInst;
HICON hIconSm = NULL;
GetSystemDirectoryW(szSysDir, _countof(szSysDir));
StringCchPrintfW(szPath, _countof(szPath), L"%s\\%s", szSysDir, pszImeFile);
hImeInst = LoadLibraryExW(szPath, NULL, DONT_RESOLVE_DLL_REFERENCES);
if (hImeInst == NULL)
return NULL;
EnumResourceNamesW(hImeInst, RT_GROUP_ICON, EnumResNameProc, (LPARAM)&hIconSm);
FreeLibrary(hImeInst);
return hIconSm;
}
static HTREEITEM FindLanguageInList(HWND hwndList, LPCTSTR pszLangName)
{
TV_ITEM item;
TCHAR szText[128];
HTREEITEM hItem;
hItem = TreeView_GetRoot(hwndList);
while (hItem)
{
szText[0] = 0;
item.mask = TVIF_TEXT | TVIF_HANDLE;
item.pszText = szText;
item.cchTextMax = _countof(szText);
item.hItem = hItem;
TreeView_GetItem(hwndList, &item);
if (_wcsicmp(szText, pszLangName) == 0)
return hItem;
hItem = TreeView_GetNextSibling(hwndList, hItem);
}
return NULL;
}
static VOID
AddToInputListView(HWND hwndList, INPUT_LIST_NODE *pInputNode)
{
TV_ITEM item;
TV_INSERTSTRUCT insert;
HIMAGELIST hImageList = TreeView_GetImageList(hwndList, TVSIL_NORMAL);
WCHAR szKeyboard[64];
HTREEITEM hItem;
BOOL bBold = !!(pInputNode->wFlags & INPUT_LIST_NODE_FLAG_DEFAULT);
hItem = FindLanguageInList(hwndList, pInputNode->pLocale->pszName);
if (hItem == NULL)
{
// Language icon
INT LangImageIndex = -1;
HICON hLangIcon = CreateLayoutIcon(LOWORD(pInputNode->pLocale->dwId));
if (hLangIcon)
{
LangImageIndex = ImageList_AddIcon(hImageList, hLangIcon);
DestroyIcon(hLangIcon);
}
// Language
ZeroMemory(&item, sizeof(item));
item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_PARAM | TVIF_SELECTEDIMAGE | TVIF_STATE;
item.pszText = pInputNode->pLocale->pszName;
item.iImage = LangImageIndex;
item.iSelectedImage = LangImageIndex;
item.lParam = LOWORD(pInputNode->pLocale->dwId); // HIWORD(item.lParam) == 0
if (bBold)
{
item.state = item.stateMask = TVIS_BOLD;
}
insert.hParent = TVI_ROOT;
insert.hInsertAfter = TVI_LAST;
insert.item = item;
hItem = TreeView_InsertItem(hwndList, &insert);
// The type of input method (currently keyboard only)
LoadStringW(hApplet, IDS_KEYBOARD, szKeyboard, _countof(szKeyboard));
ZeroMemory(&item, sizeof(item));
item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_PARAM | TVIF_SELECTEDIMAGE;
item.pszText = szKeyboard;
item.iImage = s_iKeyboardImage;
item.iSelectedImage = s_iKeyboardImage;
item.lParam = 0; // HIWORD(item.lParam) == 0
insert.hParent = hItem;
insert.hInsertAfter = TVI_LAST;
insert.item = item;
hItem = TreeView_InsertItem(hwndList, &insert);
}
else
{
// Language
ZeroMemory(&item, sizeof(item));
item.mask = TVIF_STATE | TVIF_HANDLE;
item.hItem = hItem;
item.stateMask = TVIS_BOLD;
if (TreeView_GetItem(hwndList, &item) && bBold && !(item.state & TVIS_BOLD))
{
// Make the item bold
item.mask = TVIF_STATE | TVIF_HANDLE;
item.hItem = hItem;
item.state = item.stateMask = TVIS_BOLD;
TreeView_SetItem(hwndList, &item);
}
// The type of input method (currently keyboard only)
hItem = TreeView_GetChild(hwndList, hItem);
}
// Input method
if (hItem)
{
INT ImeImageIndex = s_iDotImage;
if (IS_IME_HKL(pInputNode->hkl) && pInputNode->pLayout->pszImeFile) // IME?
{
HICON hImeIcon = LoadIMEIcon(pInputNode->pLayout->pszImeFile);
if (hImeIcon)
{
ImeImageIndex = ImageList_AddIcon(hImageList, hImeIcon);
DestroyIcon(hImeIcon);
}
}
ZeroMemory(&item, sizeof(item));
item.mask = TVIF_TEXT | TVIF_IMAGE | TVIF_PARAM | TVIF_SELECTEDIMAGE | TVIF_STATE;
item.pszText = pInputNode->pLayout->pszName;
item.iImage = ImeImageIndex;
item.iSelectedImage = ImeImageIndex;
item.lParam = (LPARAM)pInputNode; // HIWORD(item.lParam) != 0
if (bBold)
{
item.state = item.stateMask = TVIS_BOLD; // Make the item bold
}
insert.hParent = hItem;
insert.hInsertAfter = TVI_LAST;
insert.item = item;
hItem = TreeView_InsertItem(hwndList, &insert);
}
}
static VOID ExpandTreeItem(HWND hwndTree, HTREEITEM hItem)
{
TreeView_Expand(hwndTree, hItem, TVE_EXPAND);
hItem = TreeView_GetChild(hwndTree, hItem);
while (hItem)
{
ExpandTreeItem(hwndTree, hItem);
hItem = TreeView_GetNextSibling(hwndTree, hItem);
}
}
static VOID
UpdateInputListView(HWND hwndList)
{
INPUT_LIST_NODE *pNode;
HIMAGELIST hImageList = TreeView_GetImageList(hwndList, TVSIL_NORMAL);
HTREEITEM hItem;
HICON hKeyboardIcon, hDotIcon;
ImageList_RemoveAll(hImageList);
TreeView_DeleteAllItems(hwndList);
// Add keyboard icon
s_iKeyboardImage = -1;
hKeyboardIcon = (HICON)LoadImageW(hApplet, MAKEINTRESOURCEW(IDI_KEYBOARD), IMAGE_ICON,
GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON),
0);
if (hKeyboardIcon)
{
s_iKeyboardImage = ImageList_AddIcon(hImageList, hKeyboardIcon);
DestroyIcon(hKeyboardIcon);
}
// Add dot icon
s_iDotImage = -1;
hDotIcon = (HICON)LoadImageW(hApplet, MAKEINTRESOURCEW(IDI_DOT), IMAGE_ICON,
GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON),
0);
if (hDotIcon)
{
s_iDotImage = ImageList_AddIcon(hImageList, hDotIcon);
DestroyIcon(hDotIcon);
}
InputList_Sort();
s_nAliveLeafCount = InputList_GetAliveCount();
// Add items to the list
for (pNode = InputList_GetFirst(); pNode; pNode = pNode->pNext)
{
if (pNode->wFlags & INPUT_LIST_NODE_FLAG_DELETED)
continue;
AddToInputListView(hwndList, pNode);
}
// Expand all (with counting s_nRootCount)
s_nRootCount = 0;
hItem = TreeView_GetRoot(hwndList);
while (hItem)
{
++s_nRootCount;
ExpandTreeItem(hwndList, hItem);
hItem = TreeView_GetNextSibling(hwndList, hItem);
}
// Redraw
InvalidateRect(hwndList, NULL, TRUE);
}
static VOID
OnInitSettingsPage(HWND hwndDlg)
{
HWND hwndInputList = GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST);
HIMAGELIST hLayoutImageList, hOldImageList;
LayoutList_Create();
LocaleList_Create();
InputList_Create();
EnableWindow(GetDlgItem(hwndDlg, IDC_LANGUAGE_BAR), FALSE);
hLayoutImageList = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
GetSystemMetrics(SM_CYSMICON),
ILC_COLOR8 | ILC_MASK, 0, 0);
if (hLayoutImageList != NULL)
{
hOldImageList = TreeView_SetImageList(hwndInputList, hLayoutImageList, TVSIL_NORMAL);
ImageList_Destroy(hOldImageList);
}
UpdateInputListView(hwndInputList);
SetControlsState(hwndDlg);
}
static VOID
OnDestroySettingsPage(HWND hwndDlg)
{
LayoutList_Destroy();
LocaleList_Destroy();
InputList_Destroy();
}
VOID
OnCommandSettingsPage(HWND hwndDlg, WPARAM wParam)
{
switch (LOWORD(wParam))
{
case IDC_ADD_BUTTON:
{
if (DialogBoxW(hApplet,
MAKEINTRESOURCEW(IDD_ADD),
hwndDlg,
AddDialogProc) == IDOK)
{
UpdateInputListView(GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST));
SetControlsState(hwndDlg);
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
}
}
break;
case IDC_REMOVE_BUTTON:
{
HWND hwndList = GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST);
if (hwndList)
{
HTREEITEM hItem = TreeView_GetSelection(hwndList);
TV_ITEM item = { TVIF_HANDLE | TVIF_PARAM };
item.hItem = hItem;
if (hItem && TreeView_GetItem(hwndList, &item))
{
if (item.lParam == 0) // Branch? (currently branch is keyboard only)
{
// Get root of branch
item.hItem = TreeView_GetParent(hwndList, hItem);
TreeView_GetItem(hwndList, &item);
}
if (HIWORD(item.lParam)) // Leaf?
{
if (InputList_Remove((INPUT_LIST_NODE*)item.lParam))
s_bDefaultInputChanged = TRUE;
}
else // Root?
{
if (InputList_RemoveByLang(LOWORD(item.lParam)))
s_bDefaultInputChanged = TRUE;
}
UpdateInputListView(hwndList);
SetControlsState(hwndDlg);
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
}
}
}
break;
case IDC_PROP_BUTTON:
{
HWND hwndList = GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST);
if (hwndList)
{
HTREEITEM hItem = TreeView_GetSelection(hwndList);
TV_ITEM item = { TVIF_HANDLE | TVIF_PARAM };
item.hItem = hItem;
if (hItem && TreeView_GetItem(hwndList, &item) && HIWORD(item.lParam))
{
if (DialogBoxParamW(hApplet,
MAKEINTRESOURCEW(IDD_INPUT_LANG_PROP),
hwndDlg,
EditDialogProc,
item.lParam) == IDOK)
{
UpdateInputListView(hwndList);
SetControlsState(hwndDlg);
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
}
}
}
}
break;
case IDC_KEY_SET_BTN:
{
DialogBoxW(hApplet,
MAKEINTRESOURCEW(IDD_KEYSETTINGS),
hwndDlg,
KeySettingsDialogProc);
}
break;
case IDC_LANGUAGE_BAR:
{
// FIXME
break;
}
case IDC_DEFAULT_LANGUAGE:
{
if (HIWORD(wParam) == CBN_SELENDOK)
{
HWND hwndList = GetDlgItem(hwndDlg, IDC_KEYLAYOUT_LIST);
HWND hwndCombo = GetDlgItem(hwndDlg, IDC_DEFAULT_LANGUAGE);
INT iSelected = (INT)SendMessageW(hwndCombo, CB_GETCURSEL, 0, 0);
if (iSelected != CB_ERR)
{
LPARAM lParam = SendMessageW(hwndCombo, CB_GETITEMDATA, iSelected, 0);
if (lParam)
{
INPUT_LIST_NODE* pNode = (INPUT_LIST_NODE*)lParam;
if (!(pNode->wFlags & INPUT_LIST_NODE_FLAG_DEFAULT))
{
s_bDefaultInputChanged = TRUE;
InputList_SetDefault(pNode);
UpdateInputListView(hwndList);
SetControlsState(hwndDlg);
PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
}
}
}
}
}
}
}
static BOOL IsRebootNeeded(VOID)
{
INPUT_LIST_NODE *pNode;
if (s_bDefaultInputChanged)
return TRUE;
for (pNode = InputList_GetFirst(); pNode != NULL; pNode = pNode->pNext)
{
if (IS_IME_HKL(pNode->hkl)) /* IME? */
{
return TRUE;
}
}
return FALSE;
}
BOOL EnableProcessPrivileges(LPCWSTR lpPrivilegeName, BOOL bEnable)
{
HANDLE hToken;
LUID luid;
TOKEN_PRIVILEGES tokenPrivileges;
BOOL Ret;
Ret = OpenProcessToken(GetCurrentProcess(),
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
&hToken);
if (!Ret)
return Ret; // failure
Ret = LookupPrivilegeValueW(NULL, lpPrivilegeName, &luid);
if (Ret)
{
tokenPrivileges.PrivilegeCount = 1;
tokenPrivileges.Privileges[0].Luid = luid;
tokenPrivileges.Privileges[0].Attributes = bEnable ? SE_PRIVILEGE_ENABLED : 0;
Ret = AdjustTokenPrivileges(hToken, FALSE, &tokenPrivileges, 0, 0, 0);
}
CloseHandle(hToken);
return Ret;
}
static INT_PTR
OnNotifySettingsPage(HWND hwndDlg, LPARAM lParam)
{
LPNMHDR header = (LPNMHDR)lParam;
switch (header->code)
{
case TVN_SELCHANGED:
{
SetControlsState(hwndDlg);
break;
}
case TVN_ITEMEXPANDING:
{
// FIXME: Prevent collapse (COMCTL32 is buggy)
// https://bugs.winehq.org/show_bug.cgi?id=53727
NM_TREEVIEW* pTreeView = (NM_TREEVIEW*)lParam;
if ((pTreeView->action & TVE_TOGGLE) == TVE_COLLAPSE)
{
SetWindowLongPtrW(hwndDlg, DWLP_MSGRESULT, TRUE);
return TRUE;
}
break;
}
case PSN_APPLY:
{
BOOL bRebootNeeded = IsRebootNeeded();
/* Write Input Methods list to registry */
if (InputList_Process() && bRebootNeeded)
{
/* Needs reboot */
WCHAR szNeedsReboot[128], szLanguage[64];
LoadStringW(hApplet, IDS_REBOOT_NOW, szNeedsReboot, _countof(szNeedsReboot));
LoadStringW(hApplet, IDS_LANGUAGE, szLanguage, _countof(szLanguage));
if (MessageBoxW(hwndDlg, szNeedsReboot, szLanguage,
MB_ICONINFORMATION | MB_YESNOCANCEL) == IDYES)
{
EnableProcessPrivileges(SE_SHUTDOWN_NAME, TRUE);
ExitWindowsEx(EWX_REBOOT | EWX_FORCE, 0);
}
}
break;
}
}
return 0;
}
INT_PTR CALLBACK
SettingsPageProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_INITDIALOG:
OnInitSettingsPage(hwndDlg);
return TRUE;
case WM_DESTROY:
OnDestroySettingsPage(hwndDlg);
break;
case WM_COMMAND:
OnCommandSettingsPage(hwndDlg, wParam);
break;
case WM_NOTIFY:
return OnNotifySettingsPage(hwndDlg, lParam);
}
return FALSE;
}