870 lines
30 KiB
C++
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
/****************************************************************************
CANDUI.CPP
Owner: cslim
Copyright (c) 1997-1999 Microsoft Corporation
Candidate window UI functions
History:
14-JUL-1999 cslim Copied from IME98 source tree
*****************************************************************************/
#include "precomp.h"
#include "debug.h"
#include "ui.h"
#include "lexheader.h"
#include "names.h"
#include "escape.h"
#include "imedefs.h"
#include "hanja.h"
#include "winex.h"
///////////////////////////////////////////////////////////////////////////////
// Private data
// =====-- START OF SHARED DATA --=====
#pragma data_seg(".MSIMESHR")
PRIVATE RECT rcCandCli = { 0, 0, 319, 29 },
rcLArrow = { 15, 4, 27, 25 }, rcRArrow = { 292, 4, 304, 25 },
s_rcCandBtn[CAND_PAGE_SIZE] = {
{ 30, 4, 57, 25 }, { 59, 4, 86, 25 },
{ 88, 4, 115, 25 }, { 117, 4, 144, 25 },
{ 146, 4, 173, 25 }, { 175, 4, 202, 25 },
{ 204, 4, 231, 25 }, { 233, 4, 260, 25 },
{ 262, 4, 289, 25 } };
#pragma data_seg()
// =====-- END OF SHARED DATA --=====
///////////////////////////////////////////////////////////////////////////////
// Private functions
PRIVATE VOID PASCAL PaintCandWindow(HWND hCandWnd, HDC hDC);
PRIVATE BOOL PASCAL CandOnSetCursor(HWND hCandWnd, WORD Message);
PRIVATE VOID PASCAL AdjustCandBoundry(LPPOINT lpptCandWnd);
PRIVATE VOID PASCAL AdjustCandRectBoundry(PCIMECtx pImeCtx, LPPOINT lpptCaret);
#ifdef NOTUSED
PRIVATE VOID NotifyTooltip( HWND hCandWnd, UINT message, WPARAM wParam, LPARAM lParam )
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
HWND hUIWnd;
MSG msg;
POINT ptCur;
hUIWnd = GetWindow(hCandWnd, GW_OWNER);
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
if (!hUIPrivate)
return;
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate) // can not draw candidate window
return;
ZeroMemory(&msg, sizeof(MSG));
msg.hwnd = hCandWnd; msg.message = message;
msg.wParam = 0; //msg.lParam = 0x00050023L;
if (message == WM_SETCURSOR)
{
GetCursorPos(&ptCur);
ScreenToClient(hCandWnd, &ptCur);
msg.lParam = (ptCur.y << 16) | ptCur.x;
}
else
{
msg.message = message;
msg.wParam = wParam;
msg.lParam = lParam;
}
Dbg(DBGID_Cand, TEXT("CandOnSetCursor(): WM_MOUSEMOVE - msg.lParam= 0x%08lX"), msg.lParam),
OurSendMessage(lpUIPrivate->hCandTTWnd, TTM_RELAYEVENT, 0, (LPARAM) (LPMSG) &msg);
GlobalUnlock(hUIPrivate);
}
#endif
///////////////////////////////////////////////////////////////////////////////
LRESULT CALLBACK CandWndProc(HWND hCandWnd, UINT uMessage, WPARAM wParam, LPARAM lParam)
{
Dbg(DBGID_UI, TEXT("CandWndProc():uMessage = 0x%08lX, wParam = 0x%04X, lParam = 0x%08lX"), uMessage, wParam, lParam);
switch (uMessage)
{
case WM_IME_CHAR : case WM_IME_COMPOSITIONFULL :
case WM_IME_COMPOSITION : case WM_IME_CONTROL :
case WM_IME_SELECT :
case WM_IME_SETCONTEXT : case WM_IME_STARTCOMPOSITION :
case WM_IME_ENDCOMPOSITION :
return 0;
case WM_SETCURSOR:
CandOnSetCursor( (HWND) wParam, HIWORD(lParam) );
return 1;
case WM_PAINT:
{
HDC hDC;
PAINTSTRUCT ps;
hDC = BeginPaint(hCandWnd, &ps);
PaintCandWindow(hCandWnd, hDC);
EndPaint(hCandWnd, &ps);
}
break;
default :
return DefWindowProc(hCandWnd, uMessage, wParam, lParam);
}
return (0L);
}
VOID PASCAL OpenCand(HWND hUIWnd)
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
HIMC hIMC;
PCIMECtx pImeCtx;
POINT ptWnd, ptClientWnd;
CIMEData ImeData;
#if 1 // MultiMonitor
RECT rcWorkArea;
#endif
Dbg(DBGID_Cand, TEXT("OpenCand():"));
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
if (!hUIPrivate) // can not draw candidate window
{
DbgAssert(0);
return;
}
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate) // can not draw candidate window
return;
// Check WM_IME_SETCONTEXT lParam
if ((lpUIPrivate->uiShowParam & ISC_SHOWUICANDIDATEWINDOW)==0)
{
Dbg(DBGID_Cand, TEXT("!!! No ISC_SHOWUICANDIDATEWINDOW bit exit OpenCand()"));
goto OpenCandUnlockUIPriv;
}
hIMC = GethImcFromHwnd(hUIWnd);
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
{
DbgAssert(0);
goto OpenCandUnlockUIPriv;
}
//if (!(fdwImeMsg & MSG_ALREADY_OPEN))
// {
// Sometime the application call ImmNotifyIME to cancel the
// composition before it process IMN_OPENCANDIDATE.
// We should avoid to process this kind of IMN_OPENCANDIDATE.
// goto OpenCandUnlockIMC;
// }
if (pImeCtx->GetCompBufLen() == 0)
{
DbgAssert(0);
goto OpenCandUnlockUIPriv;
}
if (pImeCtx->GetCandidateFormIndex(0) == 0)
{
//ptWnd = lpIMC->cfCandForm[0].ptCurrentPos;
pImeCtx->GetCandidateFormPos(&ptWnd, 0);
ClientToScreen(pImeCtx->GetAppWnd(), &ptWnd);
if (pImeCtx->GetCandidateFormStyle(0) & CFS_FORCE_POSITION)
{
Dbg(DBGID_Cand, TEXT("OpenCand(): CFS_FORCE_POSITION"));
}
else
if (pImeCtx->GetCandidateFormStyle(0) == CFS_EXCLUDE)
{
//RECT rcCand;
Dbg(DBGID_Cand, TEXT("OpenCand(): CFS_EXCLUDE"));
//if (lpUIPrivate->hCandWnd) {
// GetWindowRect(lpUIPrivate->hCandWnd, &rcCand);
//} else {
// *(LPPOINT)&rcCand = ptWnd;
//}
AdjustCandRectBoundry(pImeCtx, &ptWnd);
}
else
if (pImeCtx->GetCandidateFormStyle(0) == CFS_CANDIDATEPOS)
{
Dbg(DBGID_Cand, TEXT("OpenCand(): CFS_CANDIDATEPOS"));
AdjustCandBoundry(&ptWnd);
}
else
{
goto OpenCandDefault;
}
}
else // if (lpIMC->cfCandForm[0].dwIndex != 0)
{
OpenCandDefault:
Dbg(DBGID_Cand, TEXT("OpenCand(): OpenCandDefault"));
/*
if (lpUIPrivate->nShowCompCmd != SW_HIDE) {
ptWnd.x = ptWnd.y = 0;
ClientToScreen(lpUIPrivate->hCompWnd, &ptWnd);
ptWnd.x -= lpImeL->cxCompBorder;
ptWnd.y -= lpImeL->cyCompBorder;
} else {
POINT ptNew;
ptWnd = lpIMC->cfCompForm.ptCurrentPos;
ClientToScreen(lpIMC->hWnd, &ptWnd);
ptWnd.x -= lpImeL->cxCompBorder;
ptWnd.y -= lpImeL->cyCompBorder;
ptNew = ptWnd;
// try to simulate the position of composition window
AdjustCompPosition(lpIMC, &ptWnd, &ptNew);
}
*/
// CalcCandPos(lpIMC, &ptWnd);
#if 1 // MultiMonitor
ImeMonitorWorkAreaFromWindow(pImeCtx->GetAppWnd(), &rcWorkArea);
ptWnd.x = rcWorkArea.right - ImeData->xCandWi;
ptWnd.y = rcWorkArea.bottom - ImeData->yCandHi;
#else
ptWnd.x = ImeData->rcWorkArea.right - ImeData->xCandWi;
ptWnd.y = ImeData->rcWorkArea.bottom - ImeData->yCandHi;
#endif
pImeCtx->SetCandidateFormStyle(CFS_CANDIDATEPOS);
ptClientWnd = ptWnd;
ScreenToClient(pImeCtx->GetAppWnd(), &ptClientWnd);
pImeCtx->SetCandidateFormPos(ptClientWnd);
}
if (lpUIPrivate->hCandWnd)
{
Dbg(DBGID_Cand, TEXT("OpenCand - SetWindowPos"), ptWnd.x, ptWnd.y);
SetWindowPos(lpUIPrivate->hCandWnd, NULL,
ptWnd.x, ptWnd.y,
0, 0, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOZORDER);
}
else
{
Dbg(DBGID_Cand, TEXT("OpenCand - CreateWindowEx x=%d, y=%d"), ptWnd.x, ptWnd.y);
// Create Candidate window
lpUIPrivate->hCandWnd = CreateWindowEx(0,
szCandClassName, NULL,
WS_POPUP|WS_DISABLED,
ptWnd.x, ptWnd.y,
ImeData->xCandWi, ImeData->yCandHi,
hUIWnd,
(HMENU)NULL,
vpInstData->hInst,
NULL);
if (lpUIPrivate->hCandWnd == NULL)
goto OpenCandUnlockUIPriv;
// Create candidate TT
if (IsWinNT())
lpUIPrivate->hCandTTWnd = CreateWindowW(
wszTooltipClassName,
NULL,
TTS_ALWAYSTIP|WS_DISABLED,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
lpUIPrivate->hCandWnd,
(HMENU) NULL,
vpInstData->hInst,
NULL);
else
lpUIPrivate->hCandTTWnd = CreateWindow(
szTooltipClassName,
NULL,
TTS_ALWAYSTIP|WS_DISABLED,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
lpUIPrivate->hCandWnd,
(HMENU) NULL,
vpInstData->hInst,
NULL);
DbgAssert(lpUIPrivate->hCandTTWnd != 0);
if (lpUIPrivate->hCandTTWnd)
{
TOOLINFO ti;
ZeroMemory(&ti, sizeof(TOOLINFO));
ti.cbSize = sizeof(TOOLINFO);
ti.uFlags = 0;
ti.hwnd = lpUIPrivate->hCandWnd;
ti.hinst = vpInstData->hInst;
// Reset Tooltip data
for (INT i=0; i<CAND_PAGE_SIZE; i++)
{
ti.uId = i;
ti.lpszText = "";
ti.rect = s_rcCandBtn[i];
OurSendMessage(lpUIPrivate->hCandTTWnd, TTM_ADDTOOL, 0, (LPARAM) (LPTOOLINFO) &ti);
}
}
}
ShowCand(hUIWnd, SW_SHOWNOACTIVATE);
OpenCandUnlockUIPriv:
GlobalUnlock(hUIPrivate);
return;
}
///////////////////////////////////////////////////////////////////////////////
//
VOID PASCAL AdjustCandBoundry(LPPOINT lpptCandWnd)
{
CIMEData ImeData;
#if 1 // MultiMonitor
RECT rcWorkArea;
RECT rcCandWnd;
#endif
Dbg(DBGID_UI, TEXT("AdjustCandBoundry():"));
#if 1 // MultiMonitor
*(LPPOINT)&rcCandWnd = *(LPPOINT)lpptCandWnd;
rcCandWnd.right = rcCandWnd.left + ImeData->xCandWi;
rcCandWnd.bottom = rcCandWnd.top + ImeData->yCandHi;
ImeMonitorWorkAreaFromRect(&rcCandWnd, &rcWorkArea);
if (lpptCandWnd->x < rcWorkArea.left) {
lpptCandWnd->x = rcWorkArea.left;
} else if (lpptCandWnd->x + ImeData->xCandWi > rcWorkArea.right) {
lpptCandWnd->x = rcWorkArea.right - ImeData->xCandWi;
} else {
}
if (lpptCandWnd->y < rcWorkArea.top) {
lpptCandWnd->y = rcWorkArea.top;
} else if (lpptCandWnd->y + ImeData->yCandHi > rcWorkArea.bottom) {
lpptCandWnd->y = rcWorkArea.bottom - ImeData->yCandHi;
} else {
}
#else
if (lpptCandWnd->x < ImeData->rcWorkArea.left) {
lpptCandWnd->x = ImeData->rcWorkArea.left;
} else if (lpptCandWnd->x + ImeData->xCandWi > ImeData->rcWorkArea.right) {
lpptCandWnd->x = ImeData->rcWorkArea.right - ImeData->xCandWi;
} else {
}
if (lpptCandWnd->y < ImeData->rcWorkArea.top) {
lpptCandWnd->y = ImeData->rcWorkArea.top;
} else if (lpptCandWnd->y + ImeData->yCandHi > ImeData->rcWorkArea.bottom) {
lpptCandWnd->y = ImeData->rcWorkArea.bottom - ImeData->yCandHi;
} else {
}
#endif
}
VOID PASCAL AdjustCandRectBoundry(PCIMECtx pImeCtx, LPPOINT lpptCaret) // the caret position. Screen coord
{
RECT rcExclude, rcCandRect, rcInterSect;
POINT ptCurrentPos;
CIMEData ImeData;
// translate from client coordinate to screen coordinate
// rcExclude = lpIMC->cfCandForm[0].rcArea;
pImeCtx->GetCandidateForm(&rcExclude);
pImeCtx->GetCandidateFormPos(&ptCurrentPos);
rcExclude.left += lpptCaret->x - ptCurrentPos.x;
rcExclude.right += lpptCaret->x - ptCurrentPos.x;
rcExclude.top += lpptCaret->y - ptCurrentPos.y;
rcExclude.bottom += lpptCaret->y - ptCurrentPos.y;
AdjustCandBoundry(lpptCaret);
*(LPPOINT)&rcCandRect = *lpptCaret;
rcCandRect.right = rcCandRect.left + ImeData->xCandWi;
rcCandRect.bottom = rcCandRect.top + ImeData->yCandHi;
if (IntersectRect(&rcInterSect, &rcCandRect, &rcExclude))
{
#if 1 // MultiMonitor
RECT rcWorkArea;
ImeMonitorWorkAreaFromWindow(pImeCtx->GetAppWnd(), &rcWorkArea);
// Adjust y-axis only
if ( (rcExclude.bottom + ImeData->yCandHi) < rcWorkArea.bottom )
lpptCaret->y = rcExclude.bottom;
else
lpptCaret->y = rcExclude.top - ImeData->yCandHi;
#else
// Adjust y-axis only
if ( (rcExclude.bottom + ImeData->yCandHi) < ImeData->rcWorkArea.bottom )
lpptCaret->y = rcExclude.bottom;
else
lpptCaret->y = rcExclude.top - ImeData->yCandHi;
#endif
}
}
BOOL fSetCandWindowPos(HWND hCandWnd)
{
HWND hUIWnd;
HIMC hIMC;
PCIMECtx pImeCtx;
POINT ptWnd;
CIMEData ImeData;
if (hCandWnd == 0)
{
DbgAssert(0);
return fTrue;
}
hUIWnd = GetWindow(hCandWnd, GW_OWNER);
hIMC = GethImcFromHwnd(hUIWnd);
//if (!hIMC)
// {
// DbgAssert(0);
// return fFalse;
// }
//lpIMC = (LPINPUTCONTEXT)OurImmLockIMC(hIMC);
//if (!lpIMC)
// {
// DbgAssert(0);
// return fFalse;
// }
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
return fFalse;
//ptWnd = lpIMC->cfCandForm[0].ptCurrentPos;
pImeCtx->GetCandidateFormPos(&ptWnd);
ClientToScreen(pImeCtx->GetAppWnd(), &ptWnd);
if (pImeCtx->GetCandidateFormStyle() & CFS_FORCE_POSITION)
{
}
else
if (pImeCtx->GetCandidateFormStyle() == CFS_EXCLUDE)
{
RECT rcCand;
GetWindowRect(hCandWnd, &rcCand);
AdjustCandRectBoundry(pImeCtx, &ptWnd);
if (ptWnd.x == rcCand.left && ptWnd.y == rcCand.right)
return (0L);
}
else
if (pImeCtx->GetCandidateFormStyle() == CFS_CANDIDATEPOS)
{
AdjustCandBoundry(&ptWnd);
}
else
if (pImeCtx->GetCandidateFormStyle() == CFS_DEFAULT)
{
#if 1 // MultiMonitor
RECT rcWorkArea;
ImeMonitorWorkAreaFromWindow(pImeCtx->GetAppWnd(), &rcWorkArea);
ptWnd.x = rcWorkArea.right - ImeData->xCandWi;
ptWnd.y = rcWorkArea.bottom - ImeData->yCandHi;
#else
ptWnd.x = ImeData->rcWorkArea.right - ImeData->xCandWi;
ptWnd.y = ImeData->rcWorkArea.bottom - ImeData->yCandHi;
#endif
}
SetWindowPos(hCandWnd, NULL, ptWnd.x, ptWnd.y, 0, 0, SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOZORDER);
return (0L);
}
// Show the candidate window
VOID ShowCand(HWND hUIWnd, INT nShowCandCmd)
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
Dbg(DBGID_Cand, TEXT("ShowCand(): nShowCandCmd = %d"), nShowCandCmd);
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
if (!hUIPrivate) // can not darw candidate window
return;
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate) // can not draw candidate window
return;
if (nShowCandCmd == SW_SHOWNOACTIVATE)
nShowCandCmd = vfWndOpen[CAND_WINDOW] ? SW_SHOWNOACTIVATE : SW_HIDE;
if (lpUIPrivate->nShowCandCmd == nShowCandCmd)
goto SwCandNoChange;
if (lpUIPrivate->hCandWnd)
{
ShowWindow(lpUIPrivate->hCandWnd, nShowCandCmd);
lpUIPrivate->nShowCandCmd = nShowCandCmd;
}
else
lpUIPrivate->nShowCandCmd = SW_HIDE;
SwCandNoChange:
GlobalUnlock(hUIPrivate);
return;
}
BOOL PASCAL CandOnSetCursor(HWND hCandWnd, WORD message)
{
INT iLoop;
POINT ptPos;
SetCursor(LoadCursor(vpInstData->hInst, MAKEINTRESOURCE(IDC_IME_HAND)));
switch (message)
{
case WM_LBUTTONDOWN:
GetCursorPos(&ptPos);
ScreenToClient(hCandWnd, &ptPos);
if (PtInRect((LPRECT)&rcCandCli, ptPos))
{
if (!PtInRect((LPRECT)&rcLArrow, ptPos)
&& !PtInRect((LPRECT)&rcRArrow, ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[0], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[1], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[2], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[3], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[4], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[5], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[6], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[7], ptPos)
&& !PtInRect((LPRECT)&s_rcCandBtn[8], ptPos))
MessageBeep(MB_ICONEXCLAMATION);
}
break;
case WM_LBUTTONUP:
GetCursorPos(&ptPos);
ScreenToClient(hCandWnd, &ptPos);
if (PtInRect((LPRECT)&rcLArrow, ptPos))
{
keybd_event(VK_LEFT, 0, KEYEVENTF_EXTENDEDKEY, 0);
keybd_event(VK_LEFT, 0, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, 0);
}
else if (PtInRect((LPRECT)&rcRArrow, ptPos))
{
keybd_event(VK_RIGHT, 0, KEYEVENTF_EXTENDEDKEY, 0);
keybd_event(VK_RIGHT, 0, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, 0);
}
else
{
for (iLoop = 0; iLoop < CAND_PAGE_SIZE; iLoop++)
if (PtInRect((LPRECT)&s_rcCandBtn[iLoop], ptPos))
{
keybd_event((BYTE)(iLoop + '1'), 0, 0, 0);
keybd_event((BYTE)(iLoop + '1'), 0, KEYEVENTF_KEYUP, 0);
break;
}
}
break;
case WM_MOUSEMOVE:
//case WM_LBUTTONDOWN:
//case WM_LBUTTONUP:
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
HWND hUIWnd;
MSG msg;
POINT ptCur;
hUIWnd = GetWindow(hCandWnd, GW_OWNER);
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
if (!hUIPrivate) {
break;
}
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate) { // can not draw candidate window
break;
}
ZeroMemory(&msg, sizeof(MSG));
msg.message = message;
msg.hwnd = hCandWnd;
msg.wParam = 0; //msg.lParam = 0x00050023L;
GetCursorPos(&ptCur);
ScreenToClient(hCandWnd, &ptCur);
msg.lParam = MAKELONG(ptCur.x, ptCur.y);
Dbg(DBGID_Cand|DBGID_UI, TEXT("CandOnSetCursor(): WM_MOUSEMOVE - msg.lParam= 0x%08lX"), msg.lParam),
OurSendMessage(lpUIPrivate->hCandTTWnd, TTM_RELAYEVENT, 0, (LPARAM) (LPMSG) &msg);
GlobalUnlock(hUIPrivate);
}
break;
}
return fTrue;
}
VOID PASCAL PaintCandWindow(HWND hCandWnd, HDC hDC)
{
HWND hUIWnd;
HIMC hIMC;
PCIMECtx pImeCtx;
LPCANDIDATEINFO lpCandInfo;
LPCANDIDATELIST lpCandList;
// LPSTR lpCandStr;
HFONT hFontFix, hOldFont;
DWORD iLoop, iStart;
HBITMAP hBMCand, hBMCandNum, hBMCandArr1, hBMCandArr2;
INT iSaveBkMode;
TOOLINFO ti;
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
Dbg(DBGID_UI, TEXT("PaintCandWindow"));
hUIWnd = GetWindow(hCandWnd, GW_OWNER);
hIMC = GethImcFromHwnd(hUIWnd);
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
if (!hUIPrivate)
{
DbgAssert(0);
return;
}
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate)
return;
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
{
DbgAssert(0);
goto PaintCandWindowExit;
}
lpCandInfo = pImeCtx->GetPCandInfo();
if (lpCandInfo == NULL)
{
DbgAssert(0);
goto PaintCandWindowExit;
}
lpCandList = (LPCANDIDATELIST)((LPBYTE)lpCandInfo + sizeof(CANDIDATEINFO));
Dbg(DBGID_UI, TEXT("PaintCandWindow - dwCount = %d"), lpCandList->dwCount);
if (lpCandList->dwCount)
{
if (IsWinNT())
hFontFix = CreateFontW(
-16,0,0,0,
0,0,0,0,
HANGUL_CHARSET,
OUT_DEFAULT_PRECIS,
CLIP_DEFAULT_PRECIS,
DEFAULT_QUALITY,
FIXED_PITCH,
wzIMECompFont);
else
hFontFix = CreateFontA(
-16,0,0,0,
0,0,0,0,
HANGUL_CHARSET,
OUT_DEFAULT_PRECIS,
CLIP_DEFAULT_PRECIS,
DEFAULT_QUALITY,
FIXED_PITCH,
szIMECompFont);
hOldFont = (HFONT)SelectObject(hDC, hFontFix);
// Load bitmaps
hBMCand = (HBITMAP)OurLoadImage(MAKEINTRESOURCE(IDB_CAND_WIN),
IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE|LR_LOADMAP3DCOLORS );
hBMCandNum = (HBITMAP)OurLoadImage(MAKEINTRESOURCE(IDB_CAND_NUM),
IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE|LR_LOADMAP3DCOLORS );
hBMCandArr1 = (HBITMAP)OurLoadImage(MAKEINTRESOURCE(IDB_CAND_ARRY1),
IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE|LR_LOADMAP3DCOLORS );
hBMCandArr2 = (HBITMAP)OurLoadImage(MAKEINTRESOURCE(IDB_CAND_ARRY2),
IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE|LR_LOADMAP3DCOLORS );
DrawBitmap(hDC, 0, 0, hBMCand);
iSaveBkMode = SetBkMode(hDC, TRANSPARENT);
iStart = (lpCandList->dwSelection / lpCandList->dwPageSize) * lpCandList->dwPageSize;
ZeroMemory(&ti, sizeof(TOOLINFO));
ti.cbSize = sizeof(TOOLINFO);
ti.uFlags = 0;
ti.hwnd = hCandWnd;
ti.hinst = vpInstData->hInst;
// Paint current page 9 candidate chars
for (iLoop = 0; iLoop < CAND_PAGE_SIZE && iStart+iLoop < lpCandList->dwCount; iLoop++)
{
// Set text color
if ( (iStart + iLoop) >= GetNumOfK0() ) // if K1 Hanja set it blue color
{
// If button face is black
if (GetSysColor(COLOR_3DFACE) == RGB(0,0,0))
SetTextColor(hDC, RGB(0, 128, 255));
else
SetTextColor(hDC, RGB(0, 0, 255));
}
else
SetTextColor(hDC, GetSysColor(COLOR_MENUTEXT));
// lpCandStr = (LPSTR)((LPSTR)lpCandList + lpCandList->dwOffset[iStart + iLoop]);
OurTextOutW(hDC, s_rcCandBtn[iLoop].left + 10, s_rcCandBtn[iLoop].top +3,
pImeCtx->GetCandidateStr(iStart + iLoop));
// (LPWSTR)lpCandStr,
// 1);
// Dbg(DBGID_UI, TEXT("PaintCandWindow - Cand Char = 0x%04x"), *(LPWSTR)lpCandStr);
// Set tooltip info
if (IsWin(lpUIPrivate->hCandTTWnd))
{
CIMEData ImeData;
CHAR szCurSense[MAX_SENSE_LENGTH+6+1]; // 6 reserved for Unicode display(Format "U+0000")
WCHAR wszCurSense[MAX_SENSE_LENGTH+6+1]; // 6 reserved for Unicode display(Format "U+0000")
CHAR szHanjaMeaning[MAX_SENSE_LENGTH+1];
WCHAR wzHangulOfHanja[2]; // Need just one character
CHAR szHangulOfHanja[4]; // One DBCS + One Null + extra byte
LPWSTR pwszMeaning;
// Init local vars
szCurSense[0] = '\0';
wszCurSense[0] = L'\0';
szHanjaMeaning[0] = '\0';
wzHangulOfHanja[0] = L'\0';
szHangulOfHanja[0] = '\0';
ti.uId = iLoop;
// Get the meaning of current Hanja
if (pwszMeaning = pImeCtx->GetCandidateMeaningStr(iStart+iLoop))
{
// Get Hangul pronounciation of current Hanja
wzHangulOfHanja[0] = GetCurrentHangulOfHanja();
wzHangulOfHanja[1] = L'\0';
if (IsWinNT())
{
if (ImeData->fCandUnicodeTT)
wsprintfW(wszCurSense, L"%s %s\r\nU+%04X",
pwszMeaning,
wzHangulOfHanja,
(WORD)pImeCtx->GetCandidateStr(iStart + iLoop));
else
wsprintfW(wszCurSense, L"%s %s", pwszMeaning, wzHangulOfHanja);
ti.lpszText = (LPSTR)wszCurSense;
}
else // If not NT, convert to ANSI
{
if (WideCharToMultiByte(CP_KOREA, 0,
pwszMeaning,
-1,
(LPSTR)szHanjaMeaning,
sizeof(szHanjaMeaning),
NULL,
NULL) == 0)
szHanjaMeaning[0] = 0;
if (WideCharToMultiByte(CP_KOREA, 0,
wzHangulOfHanja,
-1,
(LPSTR)szHangulOfHanja,
sizeof(szHangulOfHanja),
NULL,
NULL) == 0)
szHangulOfHanja[0] = 0;
if (ImeData->fCandUnicodeTT)
wsprintfA(szCurSense, "%s %s\r\nU+%04X",
szHanjaMeaning,
szHangulOfHanja,
(WORD)pImeCtx->GetCandidateStr(iStart + iLoop));
else
wsprintfA(szCurSense, "%s %s", szHanjaMeaning, szHangulOfHanja);
ti.lpszText = szCurSense;
}
}
else
{
if (ImeData->fCandUnicodeTT)
{
wsprintfA(szCurSense, "U+%04X", (WORD)pImeCtx->GetCandidateStr(iStart + iLoop));
ti.lpszText = szCurSense;
}
}
// Set Tooltip Text
if (ti.lpszText)
{
UINT uiMsgUpdateTxt = TTM_UPDATETIPTEXTW;
if (!IsWinNT())
uiMsgUpdateTxt = TTM_UPDATETIPTEXT;
OurSendMessage(lpUIPrivate->hCandTTWnd, uiMsgUpdateTxt, 0, (LPARAM) (LPTOOLINFO) &ti);
// To force the tooltip control to use multiple lines
OurSendMessage(lpUIPrivate->hCandTTWnd, TTM_SETMAXTIPWIDTH, 0, 300);
}
}
}
SetBkMode(hDC, iSaveBkMode);
// Reset blank cand list tooltip
if (iLoop < CAND_PAGE_SIZE)
{
ti.lpszText = NULL;
for (; iLoop < CAND_PAGE_SIZE; iLoop++)
{
ti.uId = iLoop;
OurSendMessage(lpUIPrivate->hCandTTWnd, TTM_UPDATETIPTEXT, 0, (LPARAM) (LPTOOLINFO) &ti);
DrawBitmap(hDC, s_rcCandBtn[iLoop].left + 3, s_rcCandBtn[iLoop].top + 6, hBMCandNum);
}
}
//
if (iStart)
DrawBitmap(hDC, 19, 8, hBMCandArr1);
if (iStart + CAND_PAGE_SIZE < lpCandList->dwCount)
DrawBitmap(hDC, 296, 8, hBMCandArr2);
DeleteObject(hBMCand);
DeleteObject(hBMCandNum);
DeleteObject(hBMCandArr1);
DeleteObject(hBMCandArr2);
SelectObject(hDC, hOldFont);
DeleteObject(hFontFix);
}
PaintCandWindowExit:
GlobalUnlock(hUIPrivate);
}