/*++ Copyright (c) 1990-1998 Microsoft Corporation, All Rights Reserved Module Name: IMEEUDC.c ++*/ #include #include #include #include "imeeudc.h" #define STRSAFE_NO_DEPRECATE #include "strsafe.h" #ifndef ARRAYSIZE #define ARRAYSIZE(a) (sizeof(a)/sizeof(a[0])) #endif // // J. J. Lee 9-29-1994 // // UI of this program // +--------------------------------------------------+ // +--------------------------------------------------+ // | | // | +------------+ | // | | FA40 | | // | +------------+ | // | | // | +------------+ +-----+ | // | (IMEName iStartIME + 0) | GOLD | | ^ | | // | +------------+ | | | | // | | | | // | +------------+ |scrol| | // | (IMEName iStartIME + 1) | WOOD | | bar | | // | +------------+ | | | // | | | | // | +------------+ | | | | // | (IMEName iStartIME + 2) | WATER | | V | | // | +------------+ +-----+ | // | | // | +----------+ +-----------+ | // | | Register | | Abort | | // | +----------+ +-----------+ | // | | // +--------------------------------------------------+ // // The scroll bar only appear when there are more than 3 IMEs // // This is a sample code for EUDC regsiter a new created word into IMEs typedef struct _tagREGWORDSTRUCT { HKL hKL; BOOL bUpdate; TCHAR szIMEName[16]; UINT uIMENameLen; TCHAR szReading[14]; DWORD dwReadingLen; } REGWORDSTRUCT; typedef REGWORDSTRUCT FAR *LPREGWORDSTRUCT; typedef struct _tagIMELINKREGWORD { HIMC hOldIMC; HIMC hRegWordIMC; BOOL fCompMsg; UINT nEudcIMEs; UINT nCurrIME; TCHAR szEudcCodeString[4]; REGWORDSTRUCT sRegWordStruct[1]; } IMELINKREGWORD; typedef IMELINKREGWORD FAR *LPIMELINKREGWORD; typedef struct _tagIMERADICALRECT { UINT nStartIME; UINT nPerPageIMEs; SIZE lTextSize; SIZE lCurrReadingExtent; HWND hRegWordButton; HWND hScrollWnd; RECT rcRadical[1]; } IMERADICALRECT; typedef IMERADICALRECT FAR *LPIMERADICALRECT; static const TCHAR szAppName[] = TEXT("EUDC"); static const TCHAR szMenuName[] = TEXT("ImeEudcMenu"); static const TCHAR szRegWordCls[] = TEXT("Radical"); static const TCHAR szImeLinkDlg[] = TEXT("ImeLinkDlg"); typedef struct _tagCOUNTRYSETTING { UINT uCodePage; LPCTSTR szCodePage; } COUNTRYSETTING; static const COUNTRYSETTING sCountry[] = { { BIG5_CP, TEXT("BIG5") } , { ALT_BIG5_CP, TEXT("BIG5") } #if defined(UNICODE) , { UNICODE_CP, TEXT("UNICODE") } #endif , { GB2312_CP, TEXT("GB2312") } }; static HINSTANCE hAppInst; /************************************************************/ /* SwitchToThisIME */ /************************************************************/ void SwitchToThisIME( HWND hWnd, UINT uIndex) { LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; LPIMERADICALRECT lpImeLinkRadical; DWORD fdwConversionMode, fdwSentenceMode; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); if (lpImeLinkRegWord->nCurrIME == uIndex) { return; } if (uIndex >= lpImeLinkRegWord->nEudcIMEs) { MessageBeep((UINT)-1); return; } lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); if (uIndex < lpImeLinkRadical->nStartIME) { lpImeLinkRadical->nStartIME = uIndex; } else if ((uIndex - lpImeLinkRadical->nStartIME) >= lpImeLinkRadical->nPerPageIMEs) { lpImeLinkRadical->nStartIME = uIndex - (lpImeLinkRadical->nPerPageIMEs - 1); } else { } // avoid clear composition string SendMessage(hWnd, WM_EUDC_COMPMSG, 0, FALSE); lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[uIndex]; // switch to this IME ActivateKeyboardLayout(lpRegWordStructTmp->hKL, 0); ImmGetConversionStatus(lpImeLinkRegWord->hRegWordIMC, &fdwConversionMode, &fdwSentenceMode); fdwConversionMode = (fdwConversionMode | IME_CMODE_EUDC | IME_CMODE_NATIVE) | (fdwConversionMode & IME_CMODE_SOFTKBD); ImmSetConversionStatus(lpImeLinkRegWord->hRegWordIMC, fdwConversionMode, fdwSentenceMode); SendMessage(hWnd, WM_EUDC_COMPMSG, 0, TRUE); lpImeLinkRegWord->nCurrIME = uIndex; if(lpImeLinkRadical->hScrollWnd){ SCROLLINFO scInfo; scInfo.cbSize = sizeof(SCROLLINFO); scInfo.fMask = SIF_POS; scInfo.nPos = lpImeLinkRegWord->nCurrIME; SetScrollInfo(lpImeLinkRadical->hScrollWnd, SB_CTL, &scInfo, FALSE); } InvalidateRect(hWnd, NULL, TRUE); *(LPTSTR)&lpRegWordStructTmp->szReading[ lpRegWordStructTmp->dwReadingLen] = '\0'; ImmSetCompositionString(lpImeLinkRegWord->hRegWordIMC, SCS_SETSTR, NULL, 0, lpRegWordStructTmp->szReading, lpRegWordStructTmp->dwReadingLen * sizeof(TCHAR)); SetFocus(hWnd); return; } /************************************************************/ /* RegWordCreate */ /************************************************************/ LPIMELINKREGWORD RegWordCreate( HWND hWnd) { HWND hEudcEditWnd; UINT nLayouts; HKL FAR *lphKL; UINT i, nIMEs; DWORD dwSize; LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; TCHAR szStrBuf[16]; HDC hDC; SIZE lTextSize; RECT rcRect; LPIMERADICALRECT lpImeLinkRadical; TCHAR szTitle[32]; TCHAR szMessage[256]; hEudcEditWnd = GetWindow(GetParent(hWnd), GW_OWNER); nLayouts = GetKeyboardLayoutList(0, NULL); lphKL = GlobalAlloc(GPTR, sizeof(HKL) * nLayouts); if (!lphKL) { return (NULL); } lpImeLinkRegWord = NULL; // get all keyboard layouts, it will include all IMEs GetKeyboardLayoutList(nLayouts, lphKL); for (i = 0, nIMEs = 0; i < nLayouts; i++) { BOOL fRet; HKL hKL; TCHAR szImeEudcDic[80]; hKL = *(lphKL + i); fRet = ImmIsIME(hKL); if (!fRet) { // this is not an IME continue; } szImeEudcDic[0] = '\0'; fRet = (BOOL) ImmEscape(hKL, (HIMC)NULL, IME_ESC_GET_EUDC_DICTIONARY, szImeEudcDic); if (!fRet) { continue; } else if (szImeEudcDic[0]) { } else { continue; } *(lphKL + nIMEs) = hKL; // write back to the same buffer nIMEs++; } if (!nIMEs) { LoadString(hAppInst, IDS_NOIME_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOIME_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hEudcEditWnd, szMessage, szTitle, MB_OK); goto RegWordCreateFreeHKL; } // now there are nIMEs can support IME EUDC dictionary dwSize = sizeof(IMELINKREGWORD) - sizeof(REGWORDSTRUCT) + sizeof(REGWORDSTRUCT) * nIMEs; lpImeLinkRegWord = (LPIMELINKREGWORD)GlobalAlloc(GPTR, dwSize); if (!lpImeLinkRegWord) { LoadString(hAppInst, IDS_NOMEM_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOMEM_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hEudcEditWnd, szMessage, szTitle, MB_OK); goto RegWordCreateFreeHKL; } lpImeLinkRegWord->nEudcIMEs = nIMEs; lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[0]; for (i = 0; i < nIMEs; i++) { LRESULT lRet; #ifndef UNICODE UINT j, uInternal; #endif UINT uReadingSize; lpRegWordStructTmp->hKL = *(lphKL + i); lRet = ImmEscape(lpRegWordStructTmp->hKL, (HIMC)NULL, IME_ESC_MAX_KEY, NULL); if (!lRet) { // error message - can not support this IME! lpImeLinkRegWord->nEudcIMEs--; continue; } uReadingSize = sizeof(TCHAR); #ifndef UNICODE for (j = 0; j < 256; j++) { uInternal = ImmEscape(lpRegWordStructTmp->hKL, (HIMC)NULL, IME_ESC_SEQUENCE_TO_INTERNAL, &j); if (uInternal > 255) { uReadingSize = sizeof(WCHAR); break; } } #endif if (lRet * uReadingSize > sizeof(lpRegWordStructTmp->szReading) - sizeof(TCHAR)) { // error case, we can not support this IME // we should count this into data structure // error message - the reading of this IME is too long! lpImeLinkRegWord->nEudcIMEs--; continue; } lRet = ImmEscape(lpRegWordStructTmp->hKL, (HIMC)NULL, IME_ESC_IME_NAME, lpRegWordStructTmp->szIMEName); if (!lRet) { // error message - can not support this IME! lpImeLinkRegWord->nEudcIMEs--; continue; } // avoid length problem lpRegWordStructTmp->szIMEName[ sizeof(lpRegWordStructTmp->szIMEName) / sizeof(TCHAR) - 1] = '\0'; lpRegWordStructTmp->uIMENameLen = lstrlen(lpRegWordStructTmp->szIMEName); lpRegWordStructTmp++; } if (!lpImeLinkRegWord->nEudcIMEs) { LoadString(hAppInst, IDS_NOIME_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOIME_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hEudcEditWnd, szMessage, szTitle, MB_OK); goto RegWordCreateFreeRegWord; } LoadString(hAppInst, IDS_CHINESE_CHAR, szStrBuf, sizeof(szStrBuf) / sizeof(TCHAR)); hDC = GetDC(NULL); GetTextExtentPoint(hDC, szStrBuf, lstrlen(szStrBuf), &lTextSize); ReleaseDC(NULL, hDC); // decide the rectangle of IME radical GetWindowRect(hWnd, &rcRect); // we can show how many IME per page nIMEs = (rcRect.bottom - rcRect.top) / (2 * lTextSize.cy); if (lpImeLinkRegWord->nEudcIMEs <= nIMEs) { // all IMEs can fit in one page nIMEs = lpImeLinkRegWord->nEudcIMEs; } dwSize = sizeof(IMERADICALRECT) - sizeof(RECT) + sizeof(RECT) * RECT_NUMBER * nIMEs; lpImeLinkRadical = (LPIMERADICALRECT)GlobalAlloc(GPTR, dwSize); if (!lpImeLinkRadical) { // we can not handle any IME lpImeLinkRegWord->nEudcIMEs = 0; LoadString(hAppInst, IDS_NOMEM_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOMEM_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hEudcEditWnd, szMessage, szTitle, MB_OK); goto RegWordCreateFreeRegWord; } lpImeLinkRadical->nStartIME = 0; lpImeLinkRadical->nPerPageIMEs = nIMEs; lpImeLinkRadical->lTextSize = lTextSize; if (lpImeLinkRegWord->nEudcIMEs > nIMEs) { // IMEs more than one page, add scroll bar SCROLLINFO scInfo; // IMEs more than one page, add scroll bar lpImeLinkRadical->hScrollWnd = CreateWindowEx(0, TEXT("scrollbar"), NULL, WS_CHILD|WS_VISIBLE|SBS_VERT, rcRect.right - rcRect.left - lTextSize.cx, 0, lTextSize.cx, rcRect.bottom - rcRect.top, hWnd, 0, hAppInst, NULL); scInfo.cbSize = sizeof(SCROLLINFO); scInfo.fMask = SIF_ALL; scInfo.nMin = 0; scInfo.nMax = lpImeLinkRegWord->nEudcIMEs - 1 + (nIMEs - 1); scInfo.nPage = nIMEs; scInfo.nPos = 0; scInfo.nTrackPos = 0; SetScrollInfo(lpImeLinkRadical->hScrollWnd, SB_CTL, &scInfo, FALSE); } // decide the UI dimension for (i = 0; i < nIMEs; i++) { UINT j, k; // rectangle for IME name j = i * RECT_NUMBER + RECT_IMENAME; lpImeLinkRadical->rcRadical[j].left = lTextSize.cx; // add UI margin - UI_MARGIN lpImeLinkRadical->rcRadical[j].top = lTextSize.cy * (i * 4 + 1) / 2 - UI_MARGIN; lpImeLinkRadical->rcRadical[j].right = lpImeLinkRadical->rcRadical[j].left + lTextSize.cx * 4; // add UI margin - UI_MARGIN * 2 lpImeLinkRadical->rcRadical[j].bottom = lpImeLinkRadical->rcRadical[j].top + lTextSize.cy + UI_MARGIN * 2; // rectangle for radical k = i * RECT_NUMBER + RECT_RADICAL; lpImeLinkRadical->rcRadical[k].left = lpImeLinkRadical->rcRadical[j].right + lTextSize.cx; // add UI margin - UI_MARGIN lpImeLinkRadical->rcRadical[k].top = lpImeLinkRadical->rcRadical[j].top; lpImeLinkRadical->rcRadical[k].right = lpImeLinkRadical->rcRadical[k].left + lTextSize.cx * (sizeof(lpRegWordStructTmp->szReading) / sizeof(TCHAR) / 2 - 1); // add UI margin - UI_MARGIN * 2 lpImeLinkRadical->rcRadical[k].bottom = lpImeLinkRadical->rcRadical[k].top + lTextSize.cy + UI_MARGIN * 2; } SetWindowLongPtr(hWnd, GWL_RADICALRECT, (LONG_PTR)lpImeLinkRadical); RegWordCreateFreeRegWord: if (!lpImeLinkRegWord->nEudcIMEs) { GlobalFree((HGLOBAL)lpImeLinkRegWord); lpImeLinkRegWord = NULL; } RegWordCreateFreeHKL: GlobalFree((HGLOBAL)lphKL); return (lpImeLinkRegWord); } /************************************************************/ /* WmImeComposition */ /************************************************************/ void WmImeComposition( HWND hWnd, LPARAM lParam) { LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; LONG lRet; BOOL bUpdate; TCHAR szReading[sizeof(lpRegWordStructTmp->szReading)]; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[ lpImeLinkRegWord->nCurrIME]; lRet = ImmGetCompositionString(lpImeLinkRegWord->hRegWordIMC, GCS_COMPREADSTR, szReading, sizeof(szReading)); if (lRet < 0) { lpRegWordStructTmp->bUpdate = UPDATE_ERROR; return; } if (lRet > (sizeof(szReading) - sizeof(TCHAR))) { lRet = sizeof(szReading) - sizeof(TCHAR); } szReading[lRet / sizeof(TCHAR)] = '\0'; if (lpRegWordStructTmp->dwReadingLen != (DWORD)lRet / sizeof(TCHAR)) { bUpdate = TRUE; } else if (lstrcmp(lpRegWordStructTmp->szReading, szReading)) { bUpdate = TRUE; } else { bUpdate = FALSE; } if (bUpdate) { LPIMERADICALRECT lpImeLinkRadical; UINT i; UINT j, k; lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); (void)StringCchCopy(lpRegWordStructTmp->szReading, ARRAYSIZE(lpRegWordStructTmp->szReading), szReading); if (lParam & GCS_RESULTSTR) { lpRegWordStructTmp->bUpdate = UPDATE_FINISH; } else { lpRegWordStructTmp->bUpdate = UPDATE_START; } lpRegWordStructTmp->dwReadingLen = (DWORD)lRet / sizeof(TCHAR); if (!IsWindowEnabled(lpImeLinkRadical->hRegWordButton)) { EnableWindow(lpImeLinkRadical->hRegWordButton, TRUE); } i = lpImeLinkRegWord->nCurrIME - lpImeLinkRadical->nStartIME; j = i * RECT_NUMBER + RECT_IMENAME; InvalidateRect(hWnd, &lpImeLinkRadical->rcRadical[j], FALSE); k = i * RECT_NUMBER + RECT_RADICAL; InvalidateRect(hWnd, &lpImeLinkRadical->rcRadical[k], FALSE); } else if (lParam & GCS_RESULTSTR) { LPIMERADICALRECT lpImeLinkRadical; UINT i; UINT j, k; lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); if (lpRegWordStructTmp->bUpdate) { lpRegWordStructTmp->bUpdate = UPDATE_FINISH; } i = lpImeLinkRegWord->nCurrIME - lpImeLinkRadical->nStartIME; j = i * RECT_NUMBER + RECT_IMENAME; InvalidateRect(hWnd, &lpImeLinkRadical->rcRadical[j], FALSE); k = i * RECT_NUMBER + RECT_RADICAL; InvalidateRect(hWnd, &lpImeLinkRadical->rcRadical[k], FALSE); } else { } return; } /************************************************************/ /* lstrcmpn */ /************************************************************/ int lstrcmpn( LPCTSTR lpctszStr1, LPCTSTR lpctszStr2, int cCount) { int i; for (i = 0; i < cCount; i++) { int iCmp = *lpctszStr1++ - *lpctszStr2++; if (iCmp) { return iCmp; } } return 0; } /************************************************************/ /* EnumReading */ /************************************************************/ int CALLBACK EnumReading( LPCTSTR lpszReading, DWORD dwStyle, LPCTSTR lpszString, LPREGWORDSTRUCT lpRegWordStructTmp) { int iLen; DWORD dwZeroSeq; LRESULT lRet; TCHAR tszZeroSeq[8]; iLen = lstrlen(lpszReading); if (iLen * sizeof(TCHAR) > sizeof(lpRegWordStructTmp->szReading) - sizeof(WORD)) { return (0); } lpRegWordStructTmp->dwReadingLen = (DWORD)iLen; lstrcpy(lpRegWordStructTmp->szReading, lpszReading); dwZeroSeq = 0; lRet = ImmEscape(lpRegWordStructTmp->hKL, (HIMC)NULL, IME_ESC_SEQUENCE_TO_INTERNAL, &dwZeroSeq); if (!lRet) { return (1); } iLen = 0; if (LOWORD(lRet)) { #ifdef UNICODE tszZeroSeq[iLen++] = LOWORD(lRet); #else if (LOWORD(lRet) > 0xFF) { tszZeroSeq[iLen++] = HIBYTE(LOWORD(lRet)); tszZeroSeq[iLen++] = LOBYTE(LOWORD(lRet)); } else { tszZeroSeq[iLen++] = LOBYTE(LOWORD(lRet)); } #endif } if (HIWORD(lRet) == 0xFFFF) { // This is caused by sign extent in Win9x in the return value of // ImmEscape, it causes an invalid internal code. } else if (HIWORD(lRet)) { #ifdef UNICODE tszZeroSeq[iLen++] = HIWORD(lRet); #else if (HIWORD(lRet) > 0xFF) { tszZeroSeq[iLen++] = HIBYTE(HIWORD(lRet)); tszZeroSeq[iLen++] = LOBYTE(HIWORD(lRet)); } else { tszZeroSeq[iLen++] = LOBYTE(HIWORD(lRet)); } #endif } else { } for (; lpRegWordStructTmp->dwReadingLen > 0; lpRegWordStructTmp->dwReadingLen -= iLen) { if (lstrcmpn(&lpRegWordStructTmp->szReading[ lpRegWordStructTmp->dwReadingLen - iLen], tszZeroSeq, iLen) != 0) { break; } } lpRegWordStructTmp->szReading[lpRegWordStructTmp->dwReadingLen] = '\0'; return (1); } /************************************************************/ /* EudcCode */ /************************************************************/ void EudcCode( HWND hWnd, UINT uCode) { LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; UINT i; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); #ifdef UNICODE lpImeLinkRegWord->szEudcCodeString[0] = (WCHAR)uCode; #else lpImeLinkRegWord->szEudcCodeString[0] = HIBYTE(uCode); lpImeLinkRegWord->szEudcCodeString[1] = LOBYTE(uCode); #endif lpImeLinkRegWord->szEudcCodeString[2] = lpImeLinkRegWord->szEudcCodeString[3] = '\0'; lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[0]; for (i = 0; i < lpImeLinkRegWord->nEudcIMEs; i++) { lpRegWordStructTmp->bUpdate = UPDATE_NONE; lpRegWordStructTmp->szReading[0] = '\0'; lpRegWordStructTmp->dwReadingLen = 0; ImmEnumRegisterWord(lpRegWordStructTmp->hKL, EnumReading, NULL, IME_REGWORD_STYLE_EUDC, lpImeLinkRegWord->szEudcCodeString, lpRegWordStructTmp); lpRegWordStructTmp->dwReadingLen = lstrlen(lpRegWordStructTmp->szReading); lpRegWordStructTmp++; } lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[ lpImeLinkRegWord->nCurrIME]; ImmSetCompositionString(lpImeLinkRegWord->hRegWordIMC, SCS_SETSTR, NULL, 0, lpRegWordStructTmp->szReading, lpRegWordStructTmp->dwReadingLen * sizeof(TCHAR)); InvalidateRect(hWnd, NULL, FALSE); return; } /************************************************************/ /* ChangeToOtherIME */ /************************************************************/ void ChangeToOtherIME( HWND hWnd, LPARAM lMousePos) { POINT ptMouse; LPIMERADICALRECT lpImeLinkRadical; UINT i; BOOL bFound; ptMouse.x = LOWORD(lMousePos); ptMouse.y = HIWORD(lMousePos); lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); bFound = FALSE; for (i = 0; i < lpImeLinkRadical->nPerPageIMEs; i++) { UINT j; j = i * RECT_NUMBER + RECT_RADICAL; if (PtInRect(&lpImeLinkRadical->rcRadical[j], ptMouse)) { bFound = TRUE; break; } } if (!bFound) { return; } SwitchToThisIME(hWnd, lpImeLinkRadical->nStartIME + i); return; } /************************************************************/ /* ScrollIME */ /************************************************************/ void ScrollIME( HWND hWnd, WPARAM wParam) { LPIMELINKREGWORD lpImeLinkRegWord; LPIMERADICALRECT lpImeLinkRadical; int iLines; UINT uIndex; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); switch (LOWORD(wParam)) { case SB_PAGEDOWN: // scroll (page size - 1) iLines = lpImeLinkRadical->nPerPageIMEs - 1; break; case SB_LINEDOWN: iLines = 1; break; case SB_PAGEUP: // scroll (page size - 1) iLines = 1 - lpImeLinkRadical->nPerPageIMEs; break; case SB_LINEUP: iLines = -1; break; case SB_TOP: // swicth to the first one SwitchToThisIME(hWnd, 0); return; case SB_BOTTOM: // swicth to the last one SwitchToThisIME(hWnd, lpImeLinkRegWord->nEudcIMEs - 1); return; case SB_THUMBPOSITION: SwitchToThisIME(hWnd, HIWORD(wParam)); return; default: return; } uIndex = lpImeLinkRegWord->nCurrIME; if (iLines > 0) { uIndex += (UINT)iLines; if (uIndex >= lpImeLinkRegWord->nEudcIMEs) { // should not exceed the total IMEs uIndex = lpImeLinkRegWord->nEudcIMEs - 1; } } else { UINT uLines; uLines = -iLines; if (uLines > uIndex) { uIndex = 0; } else { uIndex -= uLines; } } SwitchToThisIME(hWnd, uIndex); return; } /************************************************************/ /* ScrollIMEByKey */ /************************************************************/ void ScrollIMEByKey( HWND hWnd, WPARAM wParam) { switch (wParam) { case VK_NEXT: ScrollIME(hWnd, SB_PAGEDOWN); break; case VK_DOWN: // can not work because dialog do not pass this key to us ScrollIME(hWnd, SB_LINEDOWN); break; case VK_PRIOR: ScrollIME(hWnd, SB_PAGEUP); break; case VK_UP: // can not work because dialog do not pass this key to us ScrollIME(hWnd, SB_LINEUP); break; default: return; } return; } /************************************************************/ /* RegWordGetFocus */ /************************************************************/ void RegWordGetFocus( HWND hWnd) { LPIMELINKREGWORD lpImeLinkRegWord; LPIMERADICALRECT lpImeLinkRadical; UINT i; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); CreateCaret(hWnd, NULL, 2, lpImeLinkRadical->lTextSize.cy + CARET_MARGIN * 2); if (lpImeLinkRegWord->nCurrIME < lpImeLinkRadical->nStartIME) { lpImeLinkRegWord->nCurrIME = lpImeLinkRadical->nStartIME; } else if ((lpImeLinkRegWord->nCurrIME - lpImeLinkRadical->nStartIME) >= lpImeLinkRadical->nPerPageIMEs) { lpImeLinkRegWord->nCurrIME = lpImeLinkRadical->nStartIME + lpImeLinkRadical->nPerPageIMEs - 1; } else { } i = lpImeLinkRegWord->nCurrIME - lpImeLinkRadical->nStartIME; i = (i * RECT_NUMBER) + RECT_RADICAL; SetCaretPos(lpImeLinkRadical->rcRadical[i].left + lpImeLinkRadical->lCurrReadingExtent.cx + 2, lpImeLinkRadical->rcRadical[i].top + UI_MARGIN - CARET_MARGIN); ShowCaret(hWnd); return; } /************************************************************/ /* RegWordPaint */ /************************************************************/ void RegWordPaint( HWND hWnd) { LPIMERADICALRECT lpImeLinkRadical; LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; HDC hDC; PAINTSTRUCT ps; UINT i; UINT nShowIMEs; lpImeLinkRadical = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[ lpImeLinkRadical->nStartIME]; HideCaret(hWnd); hDC = BeginPaint(hWnd, &ps); // we only can show up to the final one nShowIMEs = lpImeLinkRegWord->nEudcIMEs - lpImeLinkRadical->nStartIME; if (nShowIMEs > lpImeLinkRadical->nPerPageIMEs) { // we only can show one page a time nShowIMEs = lpImeLinkRadical->nPerPageIMEs; } for (i = 0; i < nShowIMEs; i++) { RECT rcSunken; UINT j, k; k = i * RECT_NUMBER + RECT_RADICAL; rcSunken = lpImeLinkRadical->rcRadical[k]; rcSunken.left -= 2; rcSunken.top -= 2; rcSunken.right += 2; rcSunken.bottom += 2; DrawEdge(hDC, &rcSunken, BDR_SUNKENOUTER, BF_RECT); SetBkColor(hDC, GetSysColor(COLOR_BTNFACE)); if (lpRegWordStructTmp->bUpdate == UPDATE_ERROR) { // red text for error SetTextColor(hDC, RGB(0xFF, 0x00, 0x00)); } else if (lpRegWordStructTmp->bUpdate == UPDATE_START) { // yellow text for not finished SetTextColor(hDC, RGB(0xFF, 0xFF, 0x00)); } else if (lpRegWordStructTmp->bUpdate == UPDATE_REGISTERED) { // green text for registered SetTextColor(hDC, RGB(0x00, 0x80, 0x00)); } else { SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT)); } j = i * RECT_NUMBER + RECT_IMENAME; ExtTextOut(hDC, lpImeLinkRadical->rcRadical[j].left, lpImeLinkRadical->rcRadical[j].top, ETO_OPAQUE|ETO_CLIPPED, &lpImeLinkRadical->rcRadical[j], lpRegWordStructTmp->szIMEName, lpRegWordStructTmp->uIMENameLen, NULL); if ((lpImeLinkRegWord->nCurrIME - lpImeLinkRadical->nStartIME) == i) { SetBkColor(hDC, GetSysColor(COLOR_WINDOW)); SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT)); GetTextExtentPoint(hDC, lpRegWordStructTmp->szReading, lpRegWordStructTmp->dwReadingLen, &lpImeLinkRadical->lCurrReadingExtent); SetCaretPos(lpImeLinkRadical->rcRadical[k].left + lpImeLinkRadical->lCurrReadingExtent.cx + 2, lpImeLinkRadical->rcRadical[k].top + UI_MARGIN - CARET_MARGIN); } else { SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT)); } ExtTextOut(hDC, lpImeLinkRadical->rcRadical[k].left, lpImeLinkRadical->rcRadical[k].top + UI_MARGIN, ETO_OPAQUE, &lpImeLinkRadical->rcRadical[k], lpRegWordStructTmp->szReading, lpRegWordStructTmp->dwReadingLen, NULL); lpRegWordStructTmp++; } EndPaint(hWnd, &ps); ShowCaret(hWnd); return; } /************************************************************/ /* RegWordWndProc */ /************************************************************/ LRESULT CALLBACK RegWordWndProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch (uMsg) { case WM_CREATE: { LPIMELINKREGWORD lpImeLinkRegWord; UINT uIndex; // initialize to 0 SetWindowLongPtr(hWnd, GWL_IMELINKREGWORD, 0L); SetWindowLongPtr(hWnd, GWL_RADICALRECT, 0L); lpImeLinkRegWord = RegWordCreate(hWnd); if (!lpImeLinkRegWord) { return (-1); } lpImeLinkRegWord->fCompMsg = TRUE; lpImeLinkRegWord->nCurrIME = 0xFFFFFFFF; lpImeLinkRegWord->hRegWordIMC = ImmCreateContext(); if (!lpImeLinkRegWord->hRegWordIMC) { return (-1); } lpImeLinkRegWord->hOldIMC = ImmAssociateContext(hWnd, lpImeLinkRegWord->hRegWordIMC); SetWindowLongPtr(hWnd, GWL_IMELINKREGWORD, (LONG_PTR)lpImeLinkRegWord); uIndex = 0; SwitchToThisIME(hWnd, 0); // the switch will fail, if the window is disable, try again PostMessage(hWnd, WM_EUDC_SWITCHIME, 0, uIndex); } break; case WM_EUDC_COMPMSG: { LPIMELINKREGWORD lpImeLinkRegWord; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpImeLinkRegWord->fCompMsg = (BOOL)lParam; } break; case WM_EUDC_SWITCHIME: { LPIMELINKREGWORD lpImeLinkRegWord; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpImeLinkRegWord->nCurrIME = 0xFFFFFFFF; SwitchToThisIME(hWnd, (UINT)lParam); } break; case WM_IME_STARTCOMPOSITION: case WM_IME_ENDCOMPOSITION: break; case WM_IME_COMPOSITION: { LPIMELINKREGWORD lpImeLinkRegWord; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); if (lpImeLinkRegWord->fCompMsg) { WmImeComposition(hWnd, lParam); } } break; case WM_IME_NOTIFY: switch (wParam) { case IMN_OPENSTATUSWINDOW: case IMN_CLOSESTATUSWINDOW: case IMN_OPENCANDIDATE: case IMN_CHANGECANDIDATE: case IMN_CLOSECANDIDATE: break; default: return DefWindowProc(hWnd, uMsg, wParam, lParam); } break; case WM_IME_SETCONTEXT: return DefWindowProc(hWnd, uMsg, wParam, lParam & ~(ISC_SHOWUIALL)); case WM_EUDC_REGISTER_BUTTON: { LPIMERADICALRECT lpImeRadicalRect; lpImeRadicalRect = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); lpImeRadicalRect->hRegWordButton = (HWND)lParam; } break; case WM_EUDC_CODE: EudcCode(hWnd, (UINT)lParam); break; case WM_LBUTTONDOWN: ChangeToOtherIME(hWnd, lParam); break; case WM_VSCROLL: ScrollIME(hWnd, wParam); break; case WM_KEYDOWN: ScrollIMEByKey(hWnd, wParam); break; case WM_SETFOCUS: RegWordGetFocus(hWnd); break; case WM_KILLFOCUS: DestroyCaret(); break; case WM_PAINT: RegWordPaint(hWnd); break; case WM_DESTROY: { LPIMERADICALRECT lpImeRadicalRect; LPIMELINKREGWORD lpImeLinkRegWord; lpImeRadicalRect = (LPIMERADICALRECT)GetWindowLongPtr(hWnd, GWL_RADICALRECT); if (lpImeRadicalRect) { GlobalFree((HGLOBAL)lpImeRadicalRect); } lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); if (!lpImeLinkRegWord) { break; } ImmAssociateContext(hWnd, lpImeLinkRegWord->hOldIMC); ImmDestroyContext(lpImeLinkRegWord->hRegWordIMC); GlobalFree((HGLOBAL)lpImeLinkRegWord); } break; default: return DefWindowProc(hWnd, uMsg, wParam, lParam); } return (0L); } /************************************************************/ /* RegisterThisEudc */ /************************************************************/ int RegisterThisEudc( HWND hWnd) { LPIMELINKREGWORD lpImeLinkRegWord; LPREGWORDSTRUCT lpRegWordStructTmp; UINT i; int iRet; lpImeLinkRegWord = (LPIMELINKREGWORD)GetWindowLongPtr(hWnd, GWL_IMELINKREGWORD); lpRegWordStructTmp = &lpImeLinkRegWord->sRegWordStruct[0]; iRet = -1; for (i = 0; i < lpImeLinkRegWord->nEudcIMEs; i++, lpRegWordStructTmp++) { if (lpRegWordStructTmp->bUpdate == UPDATE_NONE) { } else if (lpRegWordStructTmp->bUpdate != UPDATE_FINISH) { TCHAR szStrBuf[128]; int iYesNo; if (iRet != -1) { continue; } LoadString(hAppInst, IDS_QUERY_NOTFINISH, szStrBuf, sizeof(szStrBuf) / sizeof(TCHAR)); iYesNo = MessageBox(hWnd, szStrBuf, lpRegWordStructTmp->szIMEName, MB_APPLMODAL|MB_YESNO|MB_DEFBUTTON1); if (iYesNo == IDYES) { iRet = i; } } else { BOOL fRet; TCHAR szStrBuf[128]; int iYesNo; fRet = ImmRegisterWord(lpRegWordStructTmp->hKL, lpRegWordStructTmp->szReading, IME_REGWORD_STYLE_EUDC, lpImeLinkRegWord->szEudcCodeString); if (fRet) { lpRegWordStructTmp->bUpdate = UPDATE_REGISTERED; continue; } else { lpRegWordStructTmp->bUpdate = UPDATE_ERROR; } if (iRet != -1) { continue; } LoadString(hAppInst, IDS_QUERY_REGISTER, szStrBuf, sizeof(szStrBuf) / sizeof(TCHAR)); iYesNo = MessageBox(hWnd, szStrBuf, lpRegWordStructTmp->szIMEName, MB_APPLMODAL|MB_YESNO|MB_DEFBUTTON1); if (iYesNo == IDYES) { iRet = i; } } } InvalidateRect(hWnd, NULL, FALSE); return (iRet); } /************************************************************/ /* CodePageInfo() */ /************************************************************/ int CodePageInfo( UINT uCodePage) { int i; for (i = 0; i < sizeof(sCountry) / sizeof(COUNTRYSETTING); i++) { if (sCountry[i].uCodePage == uCodePage) { return(i); } } return (-1); } /************************************************************/ /* ImeLinkDlgProc */ /************************************************************/ INT_PTR CALLBACK ImeLinkDlgProc( HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch (uMsg) { case WM_INITDIALOG: { HWND hRadicalWnd, hRegWordButton; int cbString; #ifdef UNICODE // this code could not run under non NATIVE platforms UINT uCodePage, uNativeCode; int i; #endif TCHAR szTitle[128]; cbString = GetWindowText(hDlg, szTitle, sizeof(szTitle) / sizeof(TCHAR)); #ifdef UNICODE uCodePage = GetACP(); i = CodePageInfo(uCodePage); if (uCodePage == UNICODE_CP || i == -1) { wsprintf(&szTitle[cbString], TEXT("%4X"), (UINT)lParam); } else { uNativeCode = 0; WideCharToMultiByte(uCodePage, WC_COMPOSITECHECK, (LPCWSTR)&lParam, 1, (LPSTR)&uNativeCode, sizeof(uNativeCode), NULL, NULL); // convert to multi byte string uNativeCode = LOBYTE(uNativeCode) << 8 | HIBYTE(uNativeCode); wsprintf(&szTitle[cbString], TEXT("%4X (%s - %4X)"), (UINT)lParam, sCountry[i].szCodePage, (UINT)uNativeCode); } #else wsprintf(&szTitle[cbString], TEXT("%4X"), (UINT)lParam); #endif SetWindowText(hDlg, szTitle); hRadicalWnd = GetDlgItem(hDlg, IDD_RADICAL); SendMessage(hRadicalWnd, WM_EUDC_CODE, 0, lParam); hRegWordButton = GetDlgItem(hDlg, IDOK); EnableWindow(hRegWordButton, FALSE); SendMessage(hRadicalWnd, WM_EUDC_REGISTER_BUTTON, 0, (LPARAM)hRegWordButton); } return (TRUE); // do not want to set focus to special control case WM_COMMAND: switch (wParam) { case IDOK: { HWND hRadicalWnd; hRadicalWnd = GetDlgItem(hDlg, IDD_RADICAL); if (RegisterThisEudc(hRadicalWnd) == -1) { EndDialog(hDlg, TRUE); } else { SetFocus(hRadicalWnd); } } break; case IDCANCEL: EndDialog(hDlg, FALSE); break; default: return (FALSE); } return (TRUE); case WM_IME_NOTIFY: // we need to hook these messages from frame window also // otherwise sometime the OPENSTATUS will send to the frame // window and the child - hRadicalWnd will not get these messages switch (wParam) { case IMN_OPENSTATUSWINDOW: case IMN_CLOSESTATUSWINDOW: return (TRUE); default: return (FALSE); } default: return (FALSE); } return (TRUE); } /************************************************************/ /* ImeLink */ /************************************************************/ void ImeLink( HWND hWnd, UINT uCode) { static BOOL bFirstTime = TRUE; UINT nLayouts; HKL FAR *lphKL; TCHAR szTitle[32]; TCHAR szMessage[256]; UINT i, nIMEs; WNDCLASSEX wcClass; HKL hOldKL; nLayouts = GetKeyboardLayoutList(0, NULL); lphKL = GlobalAlloc(GPTR, sizeof(HKL) * nLayouts); if (!lphKL) { LoadString(hAppInst, IDS_NOMEM_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOMEM_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hWnd, szMessage, szTitle, MB_OK); return; } // MSVC may have problem for recusive modal dialog box, // I mean create a modal dialog box within a modal dialog box. // so we need to move the code from RegWordCreate() to here to // prevent creating a modal dialog form a modal dialog. // The ImmConfigureIME API call is possible to create a modal dialog. // get all keyboard layouts, it will include all IMEs GetKeyboardLayoutList(nLayouts, lphKL); for (i = 0, nIMEs = 0; i < nLayouts; i++) { BOOL fRet; HKL hKL; TCHAR szImeEudcDic[80]; hKL = *(lphKL + i); fRet = ImmIsIME(hKL); if (!fRet) { // this is not an IME continue; } szImeEudcDic[0] = '\0'; fRet = (BOOL) ImmEscape(hKL, (HIMC)NULL, IME_ESC_GET_EUDC_DICTIONARY, szImeEudcDic); if (!fRet) { continue; } if (szImeEudcDic[0]) { fRet = TRUE; } else if (!bFirstTime) { } else { fRet = ImmConfigureIME(hKL, hWnd, IME_CONFIG_SELECTDICTIONARY, NULL); } if (!fRet) { // this IME do not have an IME EUDC dictionary continue; } if (szImeEudcDic[0] == '\0') { // check whether we really get a dictionary fRet = (BOOL) ImmEscape(hKL, (HIMC)NULL, IME_ESC_GET_EUDC_DICTIONARY, szImeEudcDic); if (!fRet) { continue; } else if (szImeEudcDic[0] == '\0') { continue; } else { } } else { } nIMEs++; } GlobalFree((HGLOBAL)lphKL); if (bFirstTime) { bFirstTime = FALSE; } if (!nIMEs) { LoadString(hAppInst, IDS_NOIME_TITLE, szTitle, sizeof(szTitle) / sizeof(TCHAR)); LoadString(hAppInst, IDS_NOIME_MSG, szMessage, sizeof(szMessage) / sizeof(TCHAR)); MessageBox(hWnd, szMessage, szTitle, MB_OK); return; } if (!GetClassInfoEx(hAppInst, szRegWordCls, &wcClass)) { wcClass.cbSize = sizeof(WNDCLASSEX); wcClass.style = CS_HREDRAW|CS_VREDRAW; wcClass.lpfnWndProc = RegWordWndProc; wcClass.cbClsExtra = 0; wcClass.cbWndExtra = GWL_SIZE; wcClass.hInstance = hAppInst; wcClass.hIcon = NULL; wcClass.hCursor = LoadCursor(NULL, IDC_ARROW); wcClass.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1); wcClass.lpszMenuName = NULL; wcClass.lpszClassName = szRegWordCls; wcClass.hIconSm = NULL; RegisterClassEx(&wcClass); } hOldKL = GetKeyboardLayout(0); DialogBoxParam(hAppInst, szImeLinkDlg, hWnd, ImeLinkDlgProc, (LPARAM)uCode); ActivateKeyboardLayout(hOldKL, 0); return; } /************************************************************/ /* MatchImeName() */ /************************************************************/ HKL MatchImeName( LPCTSTR szStr) { TCHAR szImeName[16]; int nLayout; HKL hKL; HGLOBAL hMem; HKL FAR * lpMem; int i; nLayout = GetKeyboardLayoutList(0, NULL); // alloc temp buffer hMem = GlobalAlloc(GHND, sizeof(HKL) * nLayout); if (!hMem) { return (NULL); } lpMem = (HKL FAR *)GlobalLock(hMem); if (!lpMem) { GlobalFree(hMem); return (NULL); } // get all keyboard layouts, it includes all IMEs GetKeyboardLayoutList(nLayout, lpMem); for (i = 0; i < nLayout; i++) { BOOL fRet; hKL = *(lpMem + i); fRet = (BOOL) ImmEscape(hKL, (HIMC)NULL, IME_ESC_IME_NAME, szImeName); if (!fRet) { // this hKL can not ask name continue; } if (lstrcmp(szStr, szImeName) == 0) { goto MatchOvr; } } hKL = NULL; MatchOvr: GlobalUnlock(hMem); GlobalFree(hMem); return (hKL); } /************************************************************/ /* RegisterTable() */ /************************************************************/ HKL RegisterTable( HWND hWnd, LPUSRDICIMHDR lpIsvUsrDic, DWORD dwFileSize, UINT uCodePage) { HKL hKL; HDC hDC; SIZE lTextSize; RECT rcProcess; DWORD i; LPBYTE lpCurr, lpEnd; BOOL fRet; TCHAR szStr[16]; TCHAR szProcessFmt[32]; TCHAR szResult[2][32]; TCHAR szProcessInfo[48]; WORD wInternalCode[256]; WORD wAltInternalCode[256]; #ifdef UNICODE if (uCodePage == UNICODE_CP) { LPUNATSTR lpszMethodName; lpszMethodName = (LPUNATSTR)lpIsvUsrDic->achMethodName; for (i = 0; i < sizeof(lpIsvUsrDic->achMethodName) / sizeof(TCHAR); i++) { szStr[i] = *lpszMethodName++; } szStr[i] = '\0'; } else { UINT uLen; uLen = MultiByteToWideChar(uCodePage, MB_PRECOMPOSED, (LPCSTR)lpIsvUsrDic->achMethodName, sizeof(lpIsvUsrDic->achMethodName), szStr, sizeof(szStr) / sizeof(TCHAR)); szStr[uLen] = '\0'; } #else for (i = 0; i < sizeof(lpIsvUsrDic->achMethodName); i++) { szStr[i] = lpIsvUsrDic->achMethodName[i]; } szStr[i] = '\0'; #endif hKL = MatchImeName(szStr); if (!hKL) { return (hKL); } LoadString(hAppInst, IDS_PROCESS_FMT, szProcessFmt, sizeof(szProcessFmt) / sizeof(TCHAR)); LoadString(hAppInst, IDS_RESULT_FAIL, szResult[0], sizeof(szResult[0]) / sizeof(TCHAR)); LoadString(hAppInst, IDS_RESULT_SUCCESS, szResult[1], sizeof(szResult[1]) / sizeof(TCHAR)); LoadString(hAppInst, IDS_CHINESE_CHAR, szStr, sizeof(szStr) / sizeof(TCHAR)); hDC = GetDC(NULL); GetTextExtentPoint(hDC, szStr, sizeof(WORD)/sizeof(TCHAR), &lTextSize); ReleaseDC(NULL, hDC); // show the processing in somewhere, don't need to be same as this rcProcess.left = 1; rcProcess.top = 1; rcProcess.right = rcProcess.left + lTextSize.cx * sizeof(szProcessInfo) / sizeof(WORD); rcProcess.bottom = rcProcess.top + lTextSize.cy; // convert sequence code to internal code for (i = 0; i < sizeof(wInternalCode) / sizeof(WORD); i++) { LRESULT lRet; lRet = ImmEscape(hKL, (HIMC)NULL, IME_ESC_SEQUENCE_TO_INTERNAL, &i); if (HIWORD(lRet) == 0xFFFF) { // This is caused by sign extent in Win9x in the return value of // ImmEscape, it causes an invalid internal code. wAltInternalCode[i] = 0; } else { wAltInternalCode[i] = HIWORD(lRet); } wInternalCode[i] = LOWORD(lRet); #ifndef UNICODE if (wAltInternalCode[i] > 0xFF) { // convert to multi byte string wAltInternalCode[i] = LOBYTE(wAltInternalCode[i]) << 8 | HIBYTE(wAltInternalCode[i]); } if (wInternalCode[i] > 0xFF) { // convert to multi byte string wInternalCode[i] = LOBYTE(wInternalCode[i]) << 8 | HIBYTE(wInternalCode[i]); } #endif } // check for each record and register it // get to the first record and skip the Bank ID lpCurr = (LPBYTE)(lpIsvUsrDic + 1) + sizeof(WORD); lpEnd = (LPBYTE)lpIsvUsrDic + dwFileSize; for (; lpCurr < lpEnd; // internal code + sequence code + Bank ID of next record lpCurr += sizeof(WORD) + lpIsvUsrDic->cMethodKeySize + sizeof(WORD)) { int j; // quick way to init \0 for the register string *(LPDWORD)szStr = 0; #ifdef UNICODE if (uCodePage == UNICODE_CP) { szStr[0] = *(LPUNATSTR)lpCurr; } else { CHAR szMultiByte[4]; szMultiByte[0] = HIBYTE(*(LPUNATSTR)lpCurr); szMultiByte[1] = LOBYTE(*(LPUNATSTR)lpCurr); MultiByteToWideChar(uCodePage, MB_PRECOMPOSED, szMultiByte, 2, szStr, 2); } #else szStr[1] = *lpCurr; szStr[0] = *(lpCurr + 1); #endif for (i = 0, j = 0; i < lpIsvUsrDic->cMethodKeySize; i++) { if (!wAltInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)]) { } else if (wAltInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)] < 0xFF) { *(LPTSTR)&szStr[4 + j] = (TCHAR) wAltInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)]; j += sizeof(TCHAR) / sizeof(TCHAR); } else { *(LPWSTR)&szStr[4 + j] = (WCHAR) wAltInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)]; j += sizeof(WCHAR) / sizeof(TCHAR); } if (wInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)] < 0xFF) { *(LPTSTR)&szStr[4 + j] = (TCHAR) wInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)]; j += sizeof(TCHAR) / sizeof(TCHAR); } else { *(LPWSTR)&szStr[4 + j] = (WCHAR) wInternalCode[*(LPBYTE)(lpCurr + sizeof(WORD) + i)]; j += sizeof(WCHAR) / sizeof(TCHAR); } } szStr[4 + j] = szStr[4 + j + 1] = szStr[4 + j + 2] = '\0'; fRet = ImmRegisterWord(hKL, &szStr[4], IME_REGWORD_STYLE_EUDC, szStr); wsprintf(szProcessInfo, szProcessFmt, (LPTSTR)szStr, (LPTSTR)&szStr[4], szResult[fRet]); hDC = GetDC(hWnd); // show the process information ExtTextOut(hDC, rcProcess.left, rcProcess.top, ETO_OPAQUE, &rcProcess, szProcessInfo, lstrlen(szProcessInfo), NULL); ReleaseDC(NULL, hDC); if (!fRet) { // wait 3 seconds for fail case Sleep(3000); } } return (hKL); } /************************************************************/ /* BatchImeLink() */ /************************************************************/ void BatchImeLink( HWND hWnd) { HANDLE hIsvUsrDicFile, hIsvUsrDic; LPUSRDICIMHDR lpIsvUsrDic; TCHAR chReplace; int i, cbString; DWORD dwSize, dwFileSize; LPTSTR szTitle, szMessage; int iTitle, iMessage; OPENFILENAME ofn; TCHAR szFilter[64]; TCHAR szFileName[MAX_PATH]; TCHAR szDirName[MAX_PATH]; // try to share the buffer szTitle = szFilter; iTitle = sizeof(szFilter) / sizeof(TCHAR); szMessage = szDirName; iMessage = sizeof(szDirName) / sizeof(TCHAR); // internal error, the data structure need byte alignment // it should not use WORD or DWORD alignment if (sizeof(USRDICIMHDR) != 256) { LoadString(hAppInst, IDS_INTERNAL_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_INTERNAL_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); return; } // do we need to set a new file name cbString = LoadString(hAppInst, IDS_ISV_FILE_FILTER, szFilter, sizeof(szFilter) / sizeof(TCHAR)); chReplace = szFilter[cbString - 1]; for (i = 0; szFilter[i]; i++) { if (szFilter[i] == chReplace) { szFilter[i] = '\0'; } } if (!GetWindowsDirectory(szDirName, sizeof(szDirName) / sizeof(TCHAR))) { return; } lstrcpy(szFileName, TEXT("*.TBL")); // prompt a open file dialog ofn.lStructSize = sizeof(OPENFILENAME); ofn.hwndOwner = hWnd; ofn.lpstrFilter = szFilter; ofn.lpstrCustomFilter = NULL; ofn.nMaxCustFilter = 0; ofn.nFilterIndex = 1; ofn.lpstrFile = szFileName; ofn.nMaxFile = sizeof(szFileName) / sizeof(TCHAR); ofn.lpstrFileTitle = NULL; ofn.nMaxFileTitle = 0; ofn.lpstrInitialDir = szDirName; ofn.lpstrTitle = NULL;; ofn.Flags = OFN_NOCHANGEDIR|OFN_HIDEREADONLY|OFN_CREATEPROMPT| OFN_PATHMUSTEXIST; ofn.lpstrDefExt = NULL; if (!GetOpenFileName(&ofn)) { return; } hIsvUsrDicFile = CreateFile(ofn.lpstrFile, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); if (hIsvUsrDicFile == INVALID_HANDLE_VALUE) { LoadString(hAppInst, IDS_NOTOPEN_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_NOTOPEN_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST); return; } #if 0 for (i = 0; i < sizeof(szFileName); i++) { if (szFileName[i] == '\\') { szFileName[i] = ' '; } } #endif hIsvUsrDic = CreateFileMapping((HANDLE)hIsvUsrDicFile, NULL, PAGE_READONLY, 0, 0, NULL); if (!hIsvUsrDic) { LoadString(hAppInst, IDS_NOTOPEN_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_NOTOPEN_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST); goto BatchCloseUsrDicFile; } lpIsvUsrDic = MapViewOfFile(hIsvUsrDic, FILE_MAP_READ, 0, 0, 0); if (!lpIsvUsrDic) { LoadString(hAppInst, IDS_NOTOPEN_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_NOTOPEN_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST); goto BatchCloseUsrDic; } dwSize = lpIsvUsrDic->ulTableCount * (sizeof(WORD) + sizeof(WORD) + lpIsvUsrDic->cMethodKeySize) + 256; dwFileSize = GetFileSize(hIsvUsrDicFile, (LPDWORD)NULL); #if 0 // temp code dwSize = dwFileSize; #endif if (dwSize != dwFileSize) { LoadString(hAppInst, IDS_FILESIZE_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_FILESIZE_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (lpIsvUsrDic->uHeaderSize != 256) { LoadString(hAppInst, IDS_HEADERSIZE_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_HEADERSIZE_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (lpIsvUsrDic->uInfoSize != 13) { LoadString(hAppInst, IDS_INFOSIZE_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_INFOSIZE_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (CodePageInfo(lpIsvUsrDic->idCP) == -1) { LoadString(hAppInst, IDS_CODEPAGE_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_CODEPAGE_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (*(LPUNADWORD)lpIsvUsrDic->idUserCharInfoSign != SIGN_CWIN) { // != CWIN LoadString(hAppInst, IDS_CWINSIGN_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_CWINSIGN_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (*(LPUNADWORD)((LPBYTE)lpIsvUsrDic->idUserCharInfoSign + sizeof(DWORD)) != SIGN__TBL) { // != _TBL LoadString(hAppInst, IDS_CWINSIGN_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_CWINSIGN_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else if (!RegisterTable(hWnd, lpIsvUsrDic, dwFileSize, lpIsvUsrDic->idCP)) { LoadString(hAppInst, IDS_UNMATCHED_TITLE, szTitle, iTitle); LoadString(hAppInst, IDS_UNMATCHED_MSG, szMessage, iMessage); MessageBox(hWnd, szMessage, szTitle, MB_OK); } else { // OK } UnmapViewOfFile(lpIsvUsrDic); BatchCloseUsrDic: CloseHandle(hIsvUsrDic); BatchCloseUsrDicFile: CloseHandle(hIsvUsrDicFile); return; } /************************************************************/ /* WndProc() */ /************************************************************/ LRESULT CALLBACK WndProc( // this is the window procedure of // EUDC editor HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { #ifdef UNICODE static UINT uCode = 0xE000; #else static UINT uCode = 0xFA40; #endif switch (uMsg) { case WM_COMMAND: switch (wParam) { case IDM_NEW_EUDC: uCode++; break; case IDM_IME_LINK: ImeLink(hWnd, uCode); break; case IDM_BATCH_IME_LINK: BatchImeLink(hWnd); break; default: break; } break; case WM_DESTROY: PostQuitMessage(0); return (0L); default: return DefWindowProc(hWnd, uMsg, wParam, lParam); } return (0L); } /************************************************************/ /* WinMain() */ /************************************************************/ int WINAPI WinMain( HINSTANCE hInst, HINSTANCE hPrevInst, LPSTR lpszCmdLine, int nCmdShow) { WNDCLASS wcClass; HWND hWnd; MSG sMsg; hAppInst = hInst; wcClass.style = CS_HREDRAW|CS_VREDRAW; wcClass.lpfnWndProc = WndProc; wcClass.cbClsExtra = 0; wcClass.cbWndExtra = 0; wcClass.hInstance = hAppInst; wcClass.hIcon = LoadIcon(NULL, IDI_APPLICATION); wcClass.hCursor = LoadCursor(NULL, IDC_ARROW); wcClass.hbrBackground = GetStockObject(LTGRAY_BRUSH); wcClass.lpszMenuName = szMenuName; wcClass.lpszClassName = szAppName; RegisterClass(&wcClass); hWnd = CreateWindowEx(WS_EX_WINDOWEDGE, szAppName, TEXT("Fake EUDC Editor"), WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hAppInst, NULL); ShowWindow(hWnd, SW_SHOWDEFAULT); UpdateWindow(hWnd); if (!hWnd) { return (0); } while (GetMessage(&sMsg, NULL, 0, 0)) { TranslateMessage(&sMsg); DispatchMessage(&sMsg); } return ((int) sMsg.wParam); }