2020-09-30 16:53:55 +02:00

367 lines
12 KiB
C++

/****************************************************************************
COMPUI.CPP
Owner: cslim
Copyright (c) 1997-1999 Microsoft Corporation
Composition window UI functions
History:
14-JUL-1999 cslim Copied from IME98 source tree
*****************************************************************************/
#include "precomp.h"
#include "ui.h"
#include "imedefs.h"
#include "names.h"
#include "escape.h"
#include "winex.h"
#include "cicero.h"
#include "debug.h"
#include "resource.h"
PRIVATE VOID PASCAL PaintCompWindow(HWND hCompWnd, HDC hDC);
///////////////////////////////////////////////////////////////////////////////
LRESULT CALLBACK CompWndProc(HWND hCompWnd, UINT uMessage, WPARAM wParam, LPARAM lParam)
{
Dbg(DBGID_UI, TEXT("CompWndProc():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 (0L);
case WM_PAINT:
{
HDC hDC;
PAINTSTRUCT ps;
hDC = BeginPaint(hCompWnd, &ps);
PaintCompWindow(hCompWnd, hDC);
EndPaint(hCompWnd, &ps);
}
break;
default :
return DefWindowProc(hCompWnd, uMessage, wParam, lParam);
}
return (0L);
}
VOID PASCAL PaintCompWindow(HWND hCompWnd, HDC hDC)
{
HWND hWndUI;
HIMC hIMC;
PCIMECtx pImeCtx;
HFONT hFontFix, hOldFont;
HBITMAP hBMComp;
INT iSaveBkMode;
hWndUI = GetWindow(hCompWnd, GW_OWNER);
hIMC = GethImcFromHwnd(hWndUI);
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
return;
Dbg(DBGID_UI, TEXT("PaintCompWindow():CompCh = 0x%X"), pImeCtx->GetCompBufStr());
if (pImeCtx->GetCompBufLen())
{
// Create font
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);
// Draw comp window Bitmap
hBMComp = (HBITMAP)OurLoadImage(MAKEINTRESOURCE(IDB_COMP_WIN), IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE|LR_LOADMAP3DCOLORS );
DrawBitmap(hDC, 0, 0, hBMComp);
iSaveBkMode = SetBkMode(hDC, TRANSPARENT);
// Set text color
SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT));
OurTextOutW(hDC, 3, 3, pImeCtx->GetCompBufStr());
// restore and delete created objects
SelectObject(hDC, hOldFont);
SetBkMode(hDC, iSaveBkMode);
DeleteObject(hBMComp);
DeleteObject(hFontFix);
}
}
// open comp window
VOID PASCAL OpenComp(HWND hUIWnd)
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
HIMC hIMC;
PCIMECtx pImeCtx;
INT nShowCompCmd;
POINT ptClientComp;
CIMEData ImeData;
Dbg(DBGID_UI, TEXT("OpenComp()"));
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
// can not draw comp window
if (!hUIPrivate)
{
DbgAssert(0);
return;
}
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate)
return;
// Check WM_IME_SETCONTEXT lParam
if ((lpUIPrivate->uiShowParam & ISC_SHOWUICOMPOSITIONWINDOW) == 0)
goto OpenCompUnlockUIPriv;
hIMC = GethImcFromHwnd(hUIWnd);
pImeCtx = GetIMECtx(hIMC);
if (pImeCtx == NULL)
{
Dbg(DBGID_UI, TEXT("OpenComp - Invalid hImc"));
ImeData->ptCompPos.x = ImeData->rcWorkArea.right - COMP_SIZEX;
ImeData->ptCompPos.y = ImeData->rcWorkArea.bottom - COMP_SIZEY;
nShowCompCmd = SW_HIDE;
}
else
{
#if 1 // MultiMonitor
RECT rcWorkArea;
ImeMonitorWorkAreaFromWindow(pImeCtx->GetAppWnd(), &rcWorkArea);
// if client window exist in same monitor as status window
if (!IsCicero() && PtInRect(&rcWorkArea, ImeData->ptStatusPos))
{
ImeData->ptCompPos.x =
(ImeData->ptStatusPos.x+ImeData->xStatusWi+UI_GAPX + COMP_SIZEX > ImeData->rcWorkArea.right) ?
ImeData->ptStatusPos.x - UI_GAPX - COMP_SIZEX : ImeData->ptStatusPos.x + ImeData->xStatusWi + UI_GAPX;
ImeData->ptCompPos.y = ImeData->ptStatusPos.y;
Dbg(DBGID_UI, TEXT("OpenComp - PtInRect x = %d, y = %d"), ImeData->ptCompPos.x, ImeData->ptCompPos.y);
}
else
{
// if client window appeared in different monitor from status window resides,
// then display left top of workarea of client window montitor
ImeData->ptCompPos.x = 0;
ImeData->ptCompPos.y = 0;
}
#else
ImeData->ptCompPos.x =
(ImeData->ptStatusPos.x+ImeData->xStatusWi+UI_GAPX + COMP_SIZEX > ImeData->rcWorkArea.right) ?
ImeData->ptStatusPos.x - UI_GAPX - COMP_SIZEX : ImeData->ptStatusPos.x + ImeData->xStatusWi + UI_GAPX;
ImeData->ptCompPos.y = ImeData->ptStatusPos.y;
#endif
// Set Comp wnd position of HIMC in client coordinate
ptClientComp = ImeData->ptCompPos;
ScreenToClient(pImeCtx->GetAppWnd(), &ptClientComp);
pImeCtx->SetCompositionFormPos(ptClientComp);
nShowCompCmd = SW_SHOWNOACTIVATE;
}
if (lpUIPrivate->hCompWnd)
{
SetWindowPos(lpUIPrivate->hCompWnd, NULL,
ImeData->ptCompPos.x, ImeData->ptCompPos.y,
0, 0,
SWP_NOACTIVATE|SWP_NOSIZE|SWP_NOZORDER);
}
else
{
Dbg(DBGID_UI, TEXT("OpenComp - CreateWindow x = %d, y = %d"), ImeData->ptCompPos.x, ImeData->ptCompPos.y);
// create comp window
lpUIPrivate->hCompWnd = CreateWindow(
szCompClassName, TEXT("\0"),
WS_DISABLED | WS_POPUP,
ImeData->ptCompPos.x, ImeData->ptCompPos.y,
COMP_SIZEX, COMP_SIZEX,
hUIWnd, (HMENU)NULL, vpInstData->hInst, NULL);
DbgAssert(lpUIPrivate->hCompWnd != 0);
//if (!lpUIPrivate->hCompWnd)
// goto OpenCompUnlockUIPriv;
}
//if (pImeCtx)
// ShowComp(hUIWnd, SW_SHOWNOACTIVATE);
OpenCompUnlockUIPriv:
GlobalUnlock(hUIPrivate);
return;
}
// Show the composition window
VOID ShowComp(HWND hUIWnd, INT nShowCompCmd)
{
HGLOBAL hUIPrivate;
LPUIPRIV lpUIPrivate;
HIMC hIMC;
PCIMECtx pImeCtx;
Dbg(DBGID_UI, TEXT("ShowComp(): nShowCompCmd = %d"), nShowCompCmd);
hUIPrivate = GethUIPrivateFromHwnd(hUIWnd);
// can not draw comp window
if (!hUIPrivate)
{
DbgAssert(0);
return;
}
lpUIPrivate = (LPUIPRIV)GlobalLock(hUIPrivate);
if (!lpUIPrivate)
{
DbgAssert(0);
return;
}
hIMC = GethImcFromHwnd(hUIWnd);
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
goto SwCompNoChange;
// b#53794
// Some absurd Apps send WM_IME_START_COMPOSITION and WM_IME_COMPOSITION,
// even though they have no composition string
// In this case wrong composition window appears in screen like white square or something.
if (nShowCompCmd == SW_SHOWNOACTIVATE)
{
if (pImeCtx->GetCompBufLen() == 0 || vfWndOpen[COMP_WINDOW] == fFalse)
nShowCompCmd = SW_HIDE;
}
if (lpUIPrivate->nShowCompCmd == nShowCompCmd)
goto SwCompNoChange;
if (lpUIPrivate->hCompWnd)
{
ShowWindow(lpUIPrivate->hCompWnd, nShowCompCmd);
lpUIPrivate->nShowCompCmd = nShowCompCmd;
}
else
lpUIPrivate->nShowCompCmd = SW_HIDE;
SwCompNoChange:
GlobalUnlock(hUIPrivate);
}
BOOL fSetCompWindowPos(HWND hCompWnd)
{
HIMC hIMC;
PCIMECtx pImeCtx;
RECT rcCur;
POINT ptNew;
CIMEData ImeData;
// No composition window
if (hCompWnd == 0)
return fFalse;
hIMC = GethImcFromHwnd(GetWindow(hCompWnd, GW_OWNER));
//if (!hIMC)
// {
// DbgAssert(0);
// return fFalse;
// }
//lpIMC = (LPINPUTCONTEXT)OurImmLockIMC(hIMC);
//if (!lpIMC)
// {
// DbgAssert(0);
// return fFalse;
// }
if ((pImeCtx = GetIMECtx(hIMC)) == NULL)
return fFalse;
Dbg(DBGID_UI, TEXT("fSetCompWindowPos()"));
if (pImeCtx->GetCompositionFormStyle() & CFS_RECT)
{
Dbg(DBGID_UI, TEXT("fSetCompWindowPos(): CFS_RECT"));
pImeCtx->GetCompositionForm(&rcCur);
ptNew.x = rcCur.left;
ptNew.y = rcCur.top;
ClientToScreen(pImeCtx->GetAppWnd(), &ptNew);
}
else
if (pImeCtx->GetCompositionFormStyle() & CFS_POINT)
{
Dbg(DBGID_UI, TEXT("fSetCompWindowPos(): CFS_POINT"));
pImeCtx->GetCompositionForm(&ptNew);
ClientToScreen(pImeCtx->GetAppWnd(), &ptNew);
}
else
{ // For CFS_DEFAULT
Dbg(DBGID_UI, TEXT("fSetCompWindowPos(): CFS_DEFAULT"));
#if 1 // MultiMonitor
RECT rcWorkArea;
ImeMonitorWorkAreaFromWindow(pImeCtx->GetAppWnd(), &rcWorkArea);
// if client window exist in same monitor as status window
if ( PtInRect(&rcWorkArea, ImeData->ptStatusPos) )
{
ptNew.x =
(ImeData->ptStatusPos.x+ImeData->xStatusWi+UI_GAPX + COMP_SIZEX > ImeData->rcWorkArea.right) ?
ImeData->ptStatusPos.x - UI_GAPX - COMP_SIZEX : ImeData->ptStatusPos.x + ImeData->xStatusWi + UI_GAPX;
ptNew.y = ImeData->ptStatusPos.y;
}
else
{ // if client window appeared in different monitor from status window resides,
// then display right bottom of workarea of client window montitor
ptNew.x = rcWorkArea.right - COMP_SIZEX;
ptNew.y = rcWorkArea.bottom - COMP_SIZEY;
}
#else
ptNew.x =
(ImeData->ptStatusPos.x+ImeData->xStatusWi+UI_GAPX + COMP_SIZEX > ImeData->rcWorkArea.right) ?
ImeData->ptStatusPos.x - UI_GAPX - COMP_SIZEX : ImeData->ptStatusPos.x + ImeData->xStatusWi + UI_GAPX;
ptNew.y = ImeData->ptStatusPos.y;
#endif
ImeData->ptCompPos = ptNew;
ScreenToClient(pImeCtx->GetAppWnd(), &ptNew);
pImeCtx->SetCompositionFormPos(ptNew);
}
SetWindowPos(hCompWnd, NULL,
ptNew.x, ptNew.y,
0, 0, SWP_NOACTIVATE|SWP_NOCOPYBITS|SWP_NOSIZE|SWP_NOZORDER);
return (fTrue);
}