mirror of
https://github.com/reactos/wine.git
synced 2024-12-14 15:19:28 +00:00
cafb192ecc
This means the resulting rectangle will be short, but we don't have much choice. Some apps don't cope properly with the one-past-the-end character. For example, Excel 2007 gets stuck in an infinite loop.
1602 lines
51 KiB
C
1602 lines
51 KiB
C
/*
|
|
* The IME for interfacing with Mac input methods
|
|
*
|
|
* Copyright 2008, 2013 CodeWeavers, Aric Stewart
|
|
* Copyright 2013 Ken Thomases for CodeWeavers Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
/*
|
|
* Notes:
|
|
* The normal flow for IMM/IME Processing is as follows.
|
|
* 1) The Keyboard Driver generates key messages which are first passed to
|
|
* the IMM and then to IME via ImeProcessKey. If the IME returns 0 then
|
|
* it does not want the key and the keyboard driver then generates the
|
|
* WM_KEYUP/WM_KEYDOWN messages. However if the IME is going to process the
|
|
* key it returns non-zero.
|
|
* 2) If the IME is going to process the key then the IMM calls ImeToAsciiEx to
|
|
* process the key. the IME modifies the HIMC structure to reflect the
|
|
* current state and generates any messages it needs the IMM to process.
|
|
* 3) IMM checks the messages and send them to the application in question. From
|
|
* here the IMM level deals with if the application is IME aware or not.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include "macdrv.h"
|
|
#include "winuser.h"
|
|
#include "imm.h"
|
|
#include "ddk/imm.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(imm);
|
|
|
|
#define FROM_MACDRV ((HIMC)0xcafe1337)
|
|
|
|
typedef struct _IMEPRIVATE {
|
|
BOOL bInComposition;
|
|
BOOL bInternalState;
|
|
HFONT textfont;
|
|
HWND hwndDefault;
|
|
|
|
UINT repeat;
|
|
} IMEPRIVATE, *LPIMEPRIVATE;
|
|
|
|
typedef struct _tagTRANSMSG {
|
|
UINT message;
|
|
WPARAM wParam;
|
|
LPARAM lParam;
|
|
} TRANSMSG, *LPTRANSMSG;
|
|
|
|
static const WCHAR UI_CLASS_NAME[] = {'W','i','n','e',' ','M','a','c',' ','I','M','E',0};
|
|
|
|
static HIMC *hSelectedFrom = NULL;
|
|
static INT hSelectedCount = 0;
|
|
|
|
/* MSIME messages */
|
|
static UINT WM_MSIME_SERVICE;
|
|
static UINT WM_MSIME_RECONVERTOPTIONS;
|
|
static UINT WM_MSIME_MOUSE;
|
|
static UINT WM_MSIME_RECONVERTREQUEST;
|
|
static UINT WM_MSIME_RECONVERT;
|
|
static UINT WM_MSIME_QUERYPOSITION;
|
|
static UINT WM_MSIME_DOCUMENTFEED;
|
|
|
|
|
|
static HIMC RealIMC(HIMC hIMC)
|
|
{
|
|
if (hIMC == FROM_MACDRV)
|
|
{
|
|
INT i;
|
|
HWND wnd = GetFocus();
|
|
HIMC winHimc = ImmGetContext(wnd);
|
|
for (i = 0; i < hSelectedCount; i++)
|
|
if (winHimc == hSelectedFrom[i])
|
|
return winHimc;
|
|
return NULL;
|
|
}
|
|
else
|
|
return hIMC;
|
|
}
|
|
|
|
static LPINPUTCONTEXT LockRealIMC(HIMC hIMC)
|
|
{
|
|
HIMC real_imc = RealIMC(hIMC);
|
|
if (real_imc)
|
|
return ImmLockIMC(real_imc);
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
static BOOL UnlockRealIMC(HIMC hIMC)
|
|
{
|
|
HIMC real_imc = RealIMC(hIMC);
|
|
if (real_imc)
|
|
return ImmUnlockIMC(real_imc);
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
static HIMCC ImeCreateBlankCompStr(void)
|
|
{
|
|
HIMCC rc;
|
|
LPCOMPOSITIONSTRING ptr;
|
|
rc = ImmCreateIMCC(sizeof(COMPOSITIONSTRING));
|
|
ptr = ImmLockIMCC(rc);
|
|
memset(ptr, 0, sizeof(COMPOSITIONSTRING));
|
|
ptr->dwSize = sizeof(COMPOSITIONSTRING);
|
|
ImmUnlockIMCC(rc);
|
|
return rc;
|
|
}
|
|
|
|
static int updateField(DWORD origLen, DWORD origOffset, DWORD currentOffset,
|
|
LPBYTE target, LPBYTE source, DWORD* lenParam,
|
|
DWORD* offsetParam, BOOL wchars)
|
|
{
|
|
if (origLen > 0 && origOffset > 0)
|
|
{
|
|
int truelen = origLen;
|
|
if (wchars)
|
|
truelen *= sizeof(WCHAR);
|
|
|
|
memcpy(&target[currentOffset], &source[origOffset], truelen);
|
|
|
|
*lenParam = origLen;
|
|
*offsetParam = currentOffset;
|
|
currentOffset += truelen;
|
|
}
|
|
return currentOffset;
|
|
}
|
|
|
|
static HIMCC updateCompStr(HIMCC old, LPCWSTR compstr, DWORD len, DWORD *flags)
|
|
{
|
|
/* we need to make sure the CompStr, CompClaus and CompAttr fields are all
|
|
* set and correct */
|
|
int needed_size;
|
|
HIMCC rc;
|
|
LPBYTE newdata = NULL;
|
|
LPBYTE olddata = NULL;
|
|
LPCOMPOSITIONSTRING new_one;
|
|
LPCOMPOSITIONSTRING lpcs = NULL;
|
|
INT current_offset = 0;
|
|
|
|
TRACE("%s, %i\n", debugstr_wn(compstr, len), len);
|
|
|
|
if (old == NULL && compstr == NULL && len == 0)
|
|
return NULL;
|
|
|
|
if (compstr == NULL && len != 0)
|
|
{
|
|
ERR("compstr is NULL however we have a len! Please report\n");
|
|
len = 0;
|
|
}
|
|
|
|
if (old != NULL)
|
|
{
|
|
olddata = ImmLockIMCC(old);
|
|
lpcs = (LPCOMPOSITIONSTRING)olddata;
|
|
}
|
|
|
|
needed_size = sizeof(COMPOSITIONSTRING) + len * sizeof(WCHAR) +
|
|
len + sizeof(DWORD) * 2;
|
|
|
|
if (lpcs != NULL)
|
|
{
|
|
needed_size += lpcs->dwCompReadAttrLen;
|
|
needed_size += lpcs->dwCompReadClauseLen;
|
|
needed_size += lpcs->dwCompReadStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwResultReadClauseLen;
|
|
needed_size += lpcs->dwResultReadStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwResultClauseLen;
|
|
needed_size += lpcs->dwResultStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwPrivateSize;
|
|
}
|
|
rc = ImmCreateIMCC(needed_size);
|
|
newdata = ImmLockIMCC(rc);
|
|
new_one = (LPCOMPOSITIONSTRING)newdata;
|
|
|
|
new_one->dwSize = needed_size;
|
|
current_offset = sizeof(COMPOSITIONSTRING);
|
|
if (lpcs != NULL)
|
|
{
|
|
current_offset = updateField(lpcs->dwCompReadAttrLen,
|
|
lpcs->dwCompReadAttrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadAttrLen,
|
|
&new_one->dwCompReadAttrOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompReadClauseLen,
|
|
lpcs->dwCompReadClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadClauseLen,
|
|
&new_one->dwCompReadClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompReadStrLen,
|
|
lpcs->dwCompReadStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadStrLen,
|
|
&new_one->dwCompReadStrOffset, TRUE);
|
|
|
|
/* new CompAttr, CompClause, CompStr, dwCursorPos */
|
|
new_one->dwDeltaStart = 0;
|
|
new_one->dwCursorPos = lpcs->dwCursorPos;
|
|
|
|
current_offset = updateField(lpcs->dwResultReadClauseLen,
|
|
lpcs->dwResultReadClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultReadClauseLen,
|
|
&new_one->dwResultReadClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwResultReadStrLen,
|
|
lpcs->dwResultReadStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultReadStrLen,
|
|
&new_one->dwResultReadStrOffset, TRUE);
|
|
|
|
current_offset = updateField(lpcs->dwResultClauseLen,
|
|
lpcs->dwResultClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultClauseLen,
|
|
&new_one->dwResultClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwResultStrLen,
|
|
lpcs->dwResultStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultStrLen,
|
|
&new_one->dwResultStrOffset, TRUE);
|
|
|
|
current_offset = updateField(lpcs->dwPrivateSize,
|
|
lpcs->dwPrivateOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwPrivateSize,
|
|
&new_one->dwPrivateOffset, FALSE);
|
|
}
|
|
else
|
|
{
|
|
new_one->dwCursorPos = len;
|
|
*flags |= GCS_CURSORPOS;
|
|
}
|
|
|
|
/* set new data */
|
|
/* CompAttr */
|
|
new_one->dwCompAttrLen = len;
|
|
if (len > 0)
|
|
{
|
|
new_one->dwCompAttrOffset = current_offset;
|
|
memset(&newdata[current_offset], ATTR_INPUT, len);
|
|
current_offset += len;
|
|
}
|
|
|
|
/* CompClause */
|
|
if (len > 0)
|
|
{
|
|
new_one->dwCompClauseLen = sizeof(DWORD) * 2;
|
|
new_one->dwCompClauseOffset = current_offset;
|
|
*(DWORD*)&newdata[current_offset] = 0;
|
|
current_offset += sizeof(DWORD);
|
|
*(DWORD*)&newdata[current_offset] = len;
|
|
current_offset += sizeof(DWORD);
|
|
}
|
|
|
|
/* CompStr */
|
|
new_one->dwCompStrLen = len;
|
|
if (len > 0)
|
|
{
|
|
new_one->dwCompStrOffset = current_offset;
|
|
memcpy(&newdata[current_offset], compstr, len * sizeof(WCHAR));
|
|
}
|
|
|
|
|
|
ImmUnlockIMCC(rc);
|
|
if (lpcs)
|
|
ImmUnlockIMCC(old);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static HIMCC updateResultStr(HIMCC old, LPWSTR resultstr, DWORD len)
|
|
{
|
|
/* we need to make sure the ResultStr and ResultClause fields are all
|
|
* set and correct */
|
|
int needed_size;
|
|
HIMCC rc;
|
|
LPBYTE newdata = NULL;
|
|
LPBYTE olddata = NULL;
|
|
LPCOMPOSITIONSTRING new_one;
|
|
LPCOMPOSITIONSTRING lpcs = NULL;
|
|
INT current_offset = 0;
|
|
|
|
TRACE("%s, %i\n", debugstr_wn(resultstr, len), len);
|
|
|
|
if (old == NULL && resultstr == NULL && len == 0)
|
|
return NULL;
|
|
|
|
if (resultstr == NULL && len != 0)
|
|
{
|
|
ERR("resultstr is NULL however we have a len! Please report\n");
|
|
len = 0;
|
|
}
|
|
|
|
if (old != NULL)
|
|
{
|
|
olddata = ImmLockIMCC(old);
|
|
lpcs = (LPCOMPOSITIONSTRING)olddata;
|
|
}
|
|
|
|
needed_size = sizeof(COMPOSITIONSTRING) + len * sizeof(WCHAR) +
|
|
sizeof(DWORD) * 2;
|
|
|
|
if (lpcs != NULL)
|
|
{
|
|
needed_size += lpcs->dwCompReadAttrLen;
|
|
needed_size += lpcs->dwCompReadClauseLen;
|
|
needed_size += lpcs->dwCompReadStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwCompAttrLen;
|
|
needed_size += lpcs->dwCompClauseLen;
|
|
needed_size += lpcs->dwCompStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwResultReadClauseLen;
|
|
needed_size += lpcs->dwResultReadStrLen * sizeof(DWORD);
|
|
needed_size += lpcs->dwPrivateSize;
|
|
}
|
|
rc = ImmCreateIMCC(needed_size);
|
|
newdata = ImmLockIMCC(rc);
|
|
new_one = (LPCOMPOSITIONSTRING)newdata;
|
|
|
|
new_one->dwSize = needed_size;
|
|
current_offset = sizeof(COMPOSITIONSTRING);
|
|
if (lpcs != NULL)
|
|
{
|
|
current_offset = updateField(lpcs->dwCompReadAttrLen,
|
|
lpcs->dwCompReadAttrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadAttrLen,
|
|
&new_one->dwCompReadAttrOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompReadClauseLen,
|
|
lpcs->dwCompReadClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadClauseLen,
|
|
&new_one->dwCompReadClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompReadStrLen,
|
|
lpcs->dwCompReadStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompReadStrLen,
|
|
&new_one->dwCompReadStrOffset, TRUE);
|
|
|
|
current_offset = updateField(lpcs->dwCompAttrLen,
|
|
lpcs->dwCompAttrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompAttrLen,
|
|
&new_one->dwCompAttrOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompClauseLen,
|
|
lpcs->dwCompClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompClauseLen,
|
|
&new_one->dwCompClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwCompStrLen,
|
|
lpcs->dwCompStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwCompStrLen,
|
|
&new_one->dwCompStrOffset, TRUE);
|
|
|
|
new_one->dwCursorPos = lpcs->dwCursorPos;
|
|
new_one->dwDeltaStart = 0;
|
|
|
|
current_offset = updateField(lpcs->dwResultReadClauseLen,
|
|
lpcs->dwResultReadClauseOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultReadClauseLen,
|
|
&new_one->dwResultReadClauseOffset, FALSE);
|
|
|
|
current_offset = updateField(lpcs->dwResultReadStrLen,
|
|
lpcs->dwResultReadStrOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwResultReadStrLen,
|
|
&new_one->dwResultReadStrOffset, TRUE);
|
|
|
|
/* new ResultClause , ResultStr */
|
|
|
|
current_offset = updateField(lpcs->dwPrivateSize,
|
|
lpcs->dwPrivateOffset,
|
|
current_offset, newdata, olddata,
|
|
&new_one->dwPrivateSize,
|
|
&new_one->dwPrivateOffset, FALSE);
|
|
}
|
|
|
|
/* set new data */
|
|
/* ResultClause */
|
|
if (len > 0)
|
|
{
|
|
new_one->dwResultClauseLen = sizeof(DWORD) * 2;
|
|
new_one->dwResultClauseOffset = current_offset;
|
|
*(DWORD*)&newdata[current_offset] = 0;
|
|
current_offset += sizeof(DWORD);
|
|
*(DWORD*)&newdata[current_offset] = len;
|
|
current_offset += sizeof(DWORD);
|
|
}
|
|
|
|
/* ResultStr */
|
|
new_one->dwResultStrLen = len;
|
|
if (len > 0)
|
|
{
|
|
new_one->dwResultStrOffset = current_offset;
|
|
memcpy(&newdata[current_offset], resultstr, len * sizeof(WCHAR));
|
|
}
|
|
ImmUnlockIMCC(rc);
|
|
if (lpcs)
|
|
ImmUnlockIMCC(old);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void GenerateIMEMessage(HIMC hIMC, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
LPTRANSMSG lpTransMsg;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
lpIMC->hMsgBuf = ImmReSizeIMCC(lpIMC->hMsgBuf, (lpIMC->dwNumMsgBuf + 1) * sizeof(TRANSMSG));
|
|
if (!lpIMC->hMsgBuf)
|
|
return;
|
|
|
|
lpTransMsg = ImmLockIMCC(lpIMC->hMsgBuf);
|
|
if (!lpTransMsg)
|
|
return;
|
|
|
|
lpTransMsg += lpIMC->dwNumMsgBuf;
|
|
lpTransMsg->message = msg;
|
|
lpTransMsg->wParam = wParam;
|
|
lpTransMsg->lParam = lParam;
|
|
|
|
ImmUnlockIMCC(lpIMC->hMsgBuf);
|
|
lpIMC->dwNumMsgBuf++;
|
|
|
|
ImmGenerateMessage(RealIMC(hIMC));
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
static void GenerateIMECHARMessages(HIMC hIMC, LPWSTR String, DWORD length)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
LPTRANSMSG lpTransMsg;
|
|
DWORD i;
|
|
|
|
if (length <= 0)
|
|
return;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
lpIMC->hMsgBuf = ImmReSizeIMCC(lpIMC->hMsgBuf, (lpIMC->dwNumMsgBuf + length) * sizeof(TRANSMSG));
|
|
if (!lpIMC->hMsgBuf)
|
|
return;
|
|
|
|
lpTransMsg = ImmLockIMCC(lpIMC->hMsgBuf);
|
|
if (!lpTransMsg)
|
|
return;
|
|
|
|
lpTransMsg += lpIMC->dwNumMsgBuf;
|
|
for (i = 0; i < length; i++)
|
|
{
|
|
lpTransMsg->message = WM_IME_CHAR;
|
|
lpTransMsg->wParam = String[i];
|
|
lpTransMsg->lParam = 1;
|
|
lpTransMsg++;
|
|
}
|
|
|
|
ImmUnlockIMCC(lpIMC->hMsgBuf);
|
|
lpIMC->dwNumMsgBuf += length;
|
|
|
|
ImmGenerateMessage(RealIMC(hIMC));
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
static BOOL GenerateMessageToTransKey(LPDWORD lpTransBuf, UINT *uNumTranMsgs,
|
|
UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LPTRANSMSG ptr;
|
|
|
|
if (*uNumTranMsgs + 1 >= (UINT)*lpTransBuf)
|
|
return FALSE;
|
|
|
|
ptr = (LPTRANSMSG)(lpTransBuf + 1 + *uNumTranMsgs * 3);
|
|
ptr->message = msg;
|
|
ptr->wParam = wParam;
|
|
ptr->lParam = lParam;
|
|
(*uNumTranMsgs)++;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static BOOL IME_RemoveFromSelected(HIMC hIMC)
|
|
{
|
|
int i;
|
|
for (i = 0; i < hSelectedCount; i++)
|
|
{
|
|
if (hSelectedFrom[i] == hIMC)
|
|
{
|
|
if (i < hSelectedCount - 1)
|
|
memmove(&hSelectedFrom[i], &hSelectedFrom[i + 1], (hSelectedCount - i - 1) * sizeof(HIMC));
|
|
hSelectedCount--;
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void IME_AddToSelected(HIMC hIMC)
|
|
{
|
|
hSelectedCount++;
|
|
if (hSelectedFrom)
|
|
hSelectedFrom = HeapReAlloc(GetProcessHeap(), 0, hSelectedFrom, hSelectedCount * sizeof(HIMC));
|
|
else
|
|
hSelectedFrom = HeapAlloc(GetProcessHeap(), 0, sizeof(HIMC));
|
|
hSelectedFrom[hSelectedCount - 1] = hIMC;
|
|
}
|
|
|
|
static void UpdateDataInDefaultIMEWindow(HIMC hIMC, HWND hwnd, BOOL showable)
|
|
{
|
|
LPCOMPOSITIONSTRING compstr;
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
if (lpIMC->hCompStr)
|
|
compstr = ImmLockIMCC(lpIMC->hCompStr);
|
|
else
|
|
compstr = NULL;
|
|
|
|
if (compstr == NULL || compstr->dwCompStrLen == 0)
|
|
ShowWindow(hwnd, SW_HIDE);
|
|
else if (showable)
|
|
ShowWindow(hwnd, SW_SHOWNOACTIVATE);
|
|
|
|
RedrawWindow(hwnd, NULL, NULL, RDW_ERASENOW | RDW_INVALIDATE);
|
|
|
|
if (compstr != NULL)
|
|
ImmUnlockIMCC(lpIMC->hCompStr);
|
|
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
BOOL WINAPI ImeConfigure(HKL hKL, HWND hWnd, DWORD dwMode, LPVOID lpData)
|
|
{
|
|
FIXME("(%p, %p, %d, %p): stub\n", hKL, hWnd, dwMode, lpData);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return FALSE;
|
|
}
|
|
|
|
DWORD WINAPI ImeConversionList(HIMC hIMC, LPCWSTR lpSource, LPCANDIDATELIST lpCandList,
|
|
DWORD dwBufLen, UINT uFlag)
|
|
|
|
{
|
|
FIXME("(%p, %s, %p, %d, %d): stub\n", hIMC, debugstr_w(lpSource), lpCandList,
|
|
dwBufLen, uFlag);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return 0;
|
|
}
|
|
|
|
BOOL WINAPI ImeDestroy(UINT uForce)
|
|
{
|
|
TRACE("\n");
|
|
HeapFree(GetProcessHeap(), 0, hSelectedFrom);
|
|
hSelectedFrom = NULL;
|
|
hSelectedCount = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
LRESULT WINAPI ImeEscape(HIMC hIMC, UINT uSubFunc, LPVOID lpData)
|
|
{
|
|
TRACE("%x %p\n", uSubFunc, lpData);
|
|
return 0;
|
|
}
|
|
|
|
BOOL WINAPI ImeProcessKey(HIMC hIMC, UINT vKey, LPARAM lKeyData, const LPBYTE lpbKeyState)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
BOOL inIME;
|
|
|
|
TRACE("hIMC %p vKey 0x%04x lKeyData 0x%08lx lpbKeyState %p\n", hIMC, vKey, lKeyData, lpbKeyState);
|
|
|
|
switch (vKey)
|
|
{
|
|
case VK_SHIFT:
|
|
case VK_CONTROL:
|
|
case VK_CAPITAL:
|
|
case VK_MENU:
|
|
return FALSE;
|
|
}
|
|
|
|
inIME = macdrv_using_input_method();
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC)
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
|
|
if (inIME && !myPrivate->bInternalState)
|
|
ImmSetOpenStatus(RealIMC(FROM_MACDRV), TRUE);
|
|
else if (!inIME && myPrivate->bInternalState)
|
|
{
|
|
ShowWindow(myPrivate->hwndDefault, SW_HIDE);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = ImeCreateBlankCompStr();
|
|
ImmSetOpenStatus(RealIMC(FROM_MACDRV), FALSE);
|
|
}
|
|
|
|
myPrivate->repeat = (lKeyData >> 30) & 0x1;
|
|
|
|
myPrivate->bInternalState = inIME;
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
}
|
|
UnlockRealIMC(hIMC);
|
|
|
|
return inIME;
|
|
}
|
|
|
|
BOOL WINAPI ImeSelect(HIMC hIMC, BOOL fSelect)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
TRACE("%p %s\n", hIMC, fSelect ? "TRUE" : "FALSE");
|
|
|
|
if (hIMC == FROM_MACDRV)
|
|
{
|
|
ERR("ImeSelect should never be called from Cocoa\n");
|
|
return FALSE;
|
|
}
|
|
|
|
if (!hIMC)
|
|
return TRUE;
|
|
|
|
/* not selected */
|
|
if (!fSelect)
|
|
return IME_RemoveFromSelected(hIMC);
|
|
|
|
IME_AddToSelected(hIMC);
|
|
|
|
/* Initialize our structures */
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC != NULL)
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
myPrivate->bInComposition = FALSE;
|
|
myPrivate->bInternalState = FALSE;
|
|
myPrivate->textfont = NULL;
|
|
myPrivate->hwndDefault = NULL;
|
|
myPrivate->repeat = 0;
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL WINAPI ImeSetActiveContext(HIMC hIMC, BOOL fFlag)
|
|
{
|
|
FIXME("(%p, %x): stub\n", hIMC, fFlag);
|
|
return TRUE;
|
|
}
|
|
|
|
UINT WINAPI ImeToAsciiEx(UINT uVKey, UINT uScanCode, const LPBYTE lpbKeyState,
|
|
LPDWORD lpdwTransKey, UINT fuState, HIMC hIMC)
|
|
{
|
|
UINT vkey;
|
|
LPINPUTCONTEXT lpIMC;
|
|
LPIMEPRIVATE myPrivate;
|
|
HWND hwndDefault;
|
|
UINT repeat;
|
|
INT rc;
|
|
|
|
TRACE("uVKey 0x%04x uScanCode 0x%04x fuState %u hIMC %p\n", uVKey, uScanCode, fuState, hIMC);
|
|
|
|
vkey = LOWORD(uVKey);
|
|
|
|
if (vkey == VK_KANA || vkey == VK_KANJI || vkey == VK_MENU)
|
|
{
|
|
TRACE("Skipping metakey\n");
|
|
return 0;
|
|
}
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
if (!myPrivate->bInternalState)
|
|
{
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
UnlockRealIMC(hIMC);
|
|
return 0;
|
|
}
|
|
|
|
repeat = myPrivate->repeat;
|
|
hwndDefault = myPrivate->hwndDefault;
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
UnlockRealIMC(hIMC);
|
|
|
|
TRACE("Processing Mac 0x%04x\n", vkey);
|
|
rc = macdrv_process_text_input(uVKey, uScanCode, repeat, lpbKeyState, hIMC);
|
|
|
|
if (!rc)
|
|
{
|
|
UINT msgs = 0;
|
|
UINT msg = (uScanCode & 0x8000) ? WM_KEYUP : WM_KEYDOWN;
|
|
|
|
/* KeyStroke not processed by the IME
|
|
* so we need to rebuild the KeyDown message and pass it on to WINE
|
|
*/
|
|
if (!GenerateMessageToTransKey(lpdwTransKey, &msgs, msg, vkey, MAKELONG(0x0001, uScanCode)))
|
|
GenerateIMEMessage(hIMC, msg, vkey, MAKELONG(0x0001, uScanCode));
|
|
|
|
return msgs;
|
|
}
|
|
else
|
|
UpdateDataInDefaultIMEWindow(hIMC, hwndDefault, FALSE);
|
|
return 0;
|
|
}
|
|
|
|
BOOL WINAPI NotifyIME(HIMC hIMC, DWORD dwAction, DWORD dwIndex, DWORD dwValue)
|
|
{
|
|
BOOL bRet = FALSE;
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
TRACE("%p %i %i %i\n", hIMC, dwAction, dwIndex, dwValue);
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return FALSE;
|
|
|
|
switch (dwAction)
|
|
{
|
|
case NI_OPENCANDIDATE: FIXME("NI_OPENCANDIDATE\n"); break;
|
|
case NI_CLOSECANDIDATE: FIXME("NI_CLOSECANDIDATE\n"); break;
|
|
case NI_SELECTCANDIDATESTR: FIXME("NI_SELECTCANDIDATESTR\n"); break;
|
|
case NI_CHANGECANDIDATELIST: FIXME("NI_CHANGECANDIDATELIST\n"); break;
|
|
case NI_SETCANDIDATE_PAGESTART: FIXME("NI_SETCANDIDATE_PAGESTART\n"); break;
|
|
case NI_SETCANDIDATE_PAGESIZE: FIXME("NI_SETCANDIDATE_PAGESIZE\n"); break;
|
|
case NI_CONTEXTUPDATED:
|
|
switch (dwValue)
|
|
{
|
|
case IMC_SETCOMPOSITIONWINDOW: FIXME("NI_CONTEXTUPDATED: IMC_SETCOMPOSITIONWINDOW\n"); break;
|
|
case IMC_SETCONVERSIONMODE: FIXME("NI_CONTEXTUPDATED: IMC_SETCONVERSIONMODE\n"); break;
|
|
case IMC_SETSENTENCEMODE: FIXME("NI_CONTEXTUPDATED: IMC_SETSENTENCEMODE\n"); break;
|
|
case IMC_SETCANDIDATEPOS: FIXME("NI_CONTEXTUPDATED: IMC_SETCANDIDATEPOS\n"); break;
|
|
case IMC_SETCOMPOSITIONFONT:
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
TRACE("NI_CONTEXTUPDATED: IMC_SETCOMPOSITIONFONT\n");
|
|
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
if (myPrivate->textfont)
|
|
{
|
|
DeleteObject(myPrivate->textfont);
|
|
myPrivate->textfont = NULL;
|
|
}
|
|
myPrivate->textfont = CreateFontIndirectW(&lpIMC->lfFont.W);
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
}
|
|
break;
|
|
case IMC_SETOPENSTATUS:
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
TRACE("NI_CONTEXTUPDATED: IMC_SETOPENSTATUS\n");
|
|
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
if (lpIMC->fOpen != myPrivate->bInternalState && myPrivate->bInComposition)
|
|
{
|
|
if(lpIMC->fOpen == FALSE)
|
|
{
|
|
GenerateIMEMessage(hIMC, WM_IME_ENDCOMPOSITION, 0, 0);
|
|
myPrivate->bInComposition = FALSE;
|
|
}
|
|
else
|
|
{
|
|
GenerateIMEMessage(hIMC, WM_IME_STARTCOMPOSITION, 0, 0);
|
|
GenerateIMEMessage(hIMC, WM_IME_COMPOSITION, 0, 0);
|
|
}
|
|
}
|
|
myPrivate->bInternalState = lpIMC->fOpen;
|
|
bRet = TRUE;
|
|
}
|
|
break;
|
|
default: FIXME("NI_CONTEXTUPDATED: Unknown\n"); break;
|
|
}
|
|
break;
|
|
case NI_COMPOSITIONSTR:
|
|
switch (dwIndex)
|
|
{
|
|
case CPS_COMPLETE:
|
|
{
|
|
HIMCC newCompStr;
|
|
DWORD cplen = 0;
|
|
LPWSTR cpstr;
|
|
LPCOMPOSITIONSTRING cs = NULL;
|
|
LPBYTE cdata = NULL;
|
|
LPIMEPRIVATE myPrivate;
|
|
|
|
TRACE("NI_COMPOSITIONSTR: CPS_COMPLETE\n");
|
|
|
|
/* clear existing result */
|
|
newCompStr = updateResultStr(lpIMC->hCompStr, NULL, 0);
|
|
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
|
|
if (lpIMC->hCompStr)
|
|
{
|
|
cdata = ImmLockIMCC(lpIMC->hCompStr);
|
|
cs = (LPCOMPOSITIONSTRING)cdata;
|
|
cplen = cs->dwCompStrLen;
|
|
cpstr = (LPWSTR)&cdata[cs->dwCompStrOffset];
|
|
ImmUnlockIMCC(lpIMC->hCompStr);
|
|
}
|
|
if (cplen > 0)
|
|
{
|
|
WCHAR param = cpstr[0];
|
|
DWORD flags = GCS_COMPSTR;
|
|
|
|
newCompStr = updateResultStr(lpIMC->hCompStr, cpstr, cplen);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
newCompStr = updateCompStr(lpIMC->hCompStr, NULL, 0, &flags);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
|
|
GenerateIMEMessage(hIMC, WM_IME_COMPOSITION, 0, flags);
|
|
|
|
GenerateIMEMessage(hIMC, WM_IME_COMPOSITION, param,
|
|
GCS_RESULTSTR | GCS_RESULTCLAUSE);
|
|
}
|
|
|
|
GenerateIMEMessage(hIMC, WM_IME_ENDCOMPOSITION, 0, 0);
|
|
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
myPrivate->bInComposition = FALSE;
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
|
|
bRet = TRUE;
|
|
}
|
|
break;
|
|
case CPS_CONVERT: FIXME("NI_COMPOSITIONSTR: CPS_CONVERT\n"); break;
|
|
case CPS_REVERT: FIXME("NI_COMPOSITIONSTR: CPS_REVERT\n"); break;
|
|
case CPS_CANCEL:
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
|
|
TRACE("NI_COMPOSITIONSTR: CPS_CANCEL\n");
|
|
|
|
if (lpIMC->hCompStr)
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
|
|
lpIMC->hCompStr = ImeCreateBlankCompStr();
|
|
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
if (myPrivate->bInComposition)
|
|
{
|
|
GenerateIMEMessage(hIMC, WM_IME_ENDCOMPOSITION, 0, 0);
|
|
myPrivate->bInComposition = FALSE;
|
|
}
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
bRet = TRUE;
|
|
}
|
|
break;
|
|
default: FIXME("NI_COMPOSITIONSTR: Unknown\n"); break;
|
|
}
|
|
break;
|
|
default: FIXME("Unknown Message\n"); break;
|
|
}
|
|
|
|
UnlockRealIMC(hIMC);
|
|
return bRet;
|
|
}
|
|
|
|
BOOL WINAPI ImeRegisterWord(LPCWSTR lpszReading, DWORD dwStyle, LPCWSTR lpszRegister)
|
|
{
|
|
FIXME("(%s, %d, %s): stub\n", debugstr_w(lpszReading), dwStyle, debugstr_w(lpszRegister));
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WINAPI ImeUnregisterWord(LPCWSTR lpszReading, DWORD dwStyle, LPCWSTR lpszUnregister)
|
|
{
|
|
FIXME("(%s, %d, %s): stub\n", debugstr_w(lpszReading), dwStyle, debugstr_w(lpszUnregister));
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return FALSE;
|
|
}
|
|
|
|
UINT WINAPI ImeGetRegisterWordStyle(UINT nItem, LPSTYLEBUFW lpStyleBuf)
|
|
{
|
|
FIXME("(%d, %p): stub\n", nItem, lpStyleBuf);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return 0;
|
|
}
|
|
|
|
UINT WINAPI ImeEnumRegisterWord(REGISTERWORDENUMPROCW lpfnEnumProc, LPCWSTR lpszReading,
|
|
DWORD dwStyle, LPCWSTR lpszRegister, LPVOID lpData)
|
|
{
|
|
FIXME("(%p, %s, %d, %s, %p): stub\n", lpfnEnumProc, debugstr_w(lpszReading), dwStyle,
|
|
debugstr_w(lpszRegister), lpData);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return 0;
|
|
}
|
|
|
|
static BOOL IME_SetCompositionString(void* hIMC, DWORD dwIndex, LPCVOID lpComp, DWORD dwCompLen, DWORD cursor_pos, BOOL cursor_valid)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
DWORD flags = 0;
|
|
WCHAR wParam = 0;
|
|
LPIMEPRIVATE myPrivate;
|
|
|
|
TRACE("(%p, %d, %p, %d):\n", hIMC, dwIndex, lpComp, dwCompLen);
|
|
|
|
/*
|
|
* Explanation:
|
|
* this sets the composition string in the imm32.dll level
|
|
* of the composition buffer.
|
|
* TODO: set the Cocoa window's marked text string and tell text input context
|
|
*/
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
|
|
if (lpIMC == NULL)
|
|
return FALSE;
|
|
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
|
|
if (dwIndex == SCS_SETSTR)
|
|
{
|
|
HIMCC newCompStr;
|
|
|
|
if (!myPrivate->bInComposition)
|
|
{
|
|
GenerateIMEMessage(hIMC, WM_IME_STARTCOMPOSITION, 0, 0);
|
|
myPrivate->bInComposition = TRUE;
|
|
}
|
|
|
|
flags = GCS_COMPSTR;
|
|
|
|
if (dwCompLen && lpComp)
|
|
{
|
|
newCompStr = updateCompStr(lpIMC->hCompStr, (LPCWSTR)lpComp, dwCompLen / sizeof(WCHAR), &flags);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
|
|
wParam = ((const WCHAR*)lpComp)[0];
|
|
flags |= GCS_COMPCLAUSE | GCS_COMPATTR | GCS_DELTASTART;
|
|
}
|
|
else
|
|
{
|
|
newCompStr = updateCompStr(lpIMC->hCompStr, NULL, 0, &flags);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
}
|
|
|
|
if (cursor_valid)
|
|
{
|
|
LPCOMPOSITIONSTRING compstr;
|
|
compstr = ImmLockIMCC(lpIMC->hCompStr);
|
|
compstr->dwCursorPos = cursor_pos;
|
|
ImmUnlockIMCC(lpIMC->hCompStr);
|
|
flags |= GCS_CURSORPOS;
|
|
}
|
|
|
|
}
|
|
|
|
GenerateIMEMessage(hIMC, WM_IME_COMPOSITION, wParam, flags);
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
UnlockRealIMC(hIMC);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL WINAPI ImeSetCompositionString(HIMC hIMC, DWORD dwIndex, LPCVOID lpComp, DWORD dwCompLen,
|
|
LPCVOID lpRead, DWORD dwReadLen)
|
|
{
|
|
TRACE("(%p, %d, %p, %d, %p, %d):\n", hIMC, dwIndex, lpComp, dwCompLen, lpRead, dwReadLen);
|
|
|
|
if (lpRead && dwReadLen)
|
|
FIXME("Reading string unimplemented\n");
|
|
|
|
return IME_SetCompositionString(hIMC, dwIndex, lpComp, dwCompLen, 0, FALSE);
|
|
}
|
|
|
|
DWORD WINAPI ImeGetImeMenuItems(HIMC hIMC, DWORD dwFlags, DWORD dwType, LPIMEMENUITEMINFOW lpImeParentMenu,
|
|
LPIMEMENUITEMINFOW lpImeMenu, DWORD dwSize)
|
|
{
|
|
FIXME("(%p, %x %x %p %p %x): stub\n", hIMC, dwFlags, dwType, lpImeParentMenu, lpImeMenu, dwSize);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return 0;
|
|
}
|
|
|
|
static void IME_NotifyComplete(void* hIMC)
|
|
{
|
|
NotifyIME(hIMC, NI_COMPOSITIONSTR, CPS_COMPLETE, 0);
|
|
}
|
|
|
|
/*****
|
|
* Internal functions to help with IME window management
|
|
*/
|
|
static void PaintDefaultIMEWnd(HIMC hIMC, HWND hwnd)
|
|
{
|
|
PAINTSTRUCT ps;
|
|
RECT rect;
|
|
HDC hdc;
|
|
LPCOMPOSITIONSTRING compstr;
|
|
LPBYTE compdata = NULL;
|
|
HMONITOR monitor;
|
|
MONITORINFO mon_info;
|
|
INT offX = 0, offY = 0;
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
hdc = BeginPaint(hwnd, &ps);
|
|
|
|
GetClientRect(hwnd, &rect);
|
|
FillRect(hdc, &rect, (HBRUSH)(COLOR_WINDOW + 1));
|
|
|
|
compdata = ImmLockIMCC(lpIMC->hCompStr);
|
|
compstr = (LPCOMPOSITIONSTRING)compdata;
|
|
|
|
if (compstr->dwCompStrLen && compstr->dwCompStrOffset)
|
|
{
|
|
SIZE size;
|
|
POINT pt;
|
|
HFONT oldfont = NULL;
|
|
LPWSTR CompString;
|
|
LPIMEPRIVATE myPrivate;
|
|
|
|
CompString = (LPWSTR)(compdata + compstr->dwCompStrOffset);
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
|
|
if (myPrivate->textfont)
|
|
oldfont = SelectObject(hdc, myPrivate->textfont);
|
|
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
|
|
GetTextExtentPoint32W(hdc, CompString, compstr->dwCompStrLen, &size);
|
|
pt.x = size.cx;
|
|
pt.y = size.cy;
|
|
LPtoDP(hdc, &pt, 1);
|
|
|
|
/*
|
|
* How this works based on tests on windows:
|
|
* CFS_POINT: then we start our window at the point and grow it as large
|
|
* as it needs to be for the string.
|
|
* CFS_RECT: we still use the ptCurrentPos as a starting point and our
|
|
* window is only as large as we need for the string, but we do not
|
|
* grow such that our window exceeds the given rect. Wrapping if
|
|
* needed and possible. If our ptCurrentPos is outside of our rect
|
|
* then no window is displayed.
|
|
* CFS_FORCE_POSITION: appears to behave just like CFS_POINT
|
|
* maybe because the default MSIME does not do any IME adjusting.
|
|
*/
|
|
if (lpIMC->cfCompForm.dwStyle != CFS_DEFAULT)
|
|
{
|
|
POINT cpt = lpIMC->cfCompForm.ptCurrentPos;
|
|
ClientToScreen(lpIMC->hWnd, &cpt);
|
|
rect.left = cpt.x;
|
|
rect.top = cpt.y;
|
|
rect.right = rect.left + pt.x;
|
|
rect.bottom = rect.top + pt.y;
|
|
monitor = MonitorFromPoint(cpt, MONITOR_DEFAULTTOPRIMARY);
|
|
}
|
|
else /* CFS_DEFAULT */
|
|
{
|
|
/* Windows places the default IME window in the bottom left */
|
|
HWND target = lpIMC->hWnd;
|
|
if (!target) target = GetFocus();
|
|
|
|
GetWindowRect(target, &rect);
|
|
rect.top = rect.bottom;
|
|
rect.right = rect.left + pt.x + 20;
|
|
rect.bottom = rect.top + pt.y + 20;
|
|
offX=offY=10;
|
|
monitor = MonitorFromWindow(target, MONITOR_DEFAULTTOPRIMARY);
|
|
}
|
|
|
|
if (lpIMC->cfCompForm.dwStyle == CFS_RECT)
|
|
{
|
|
RECT client;
|
|
client =lpIMC->cfCompForm.rcArea;
|
|
MapWindowPoints(lpIMC->hWnd, 0, (POINT *)&client, 2);
|
|
IntersectRect(&rect, &rect, &client);
|
|
/* TODO: Wrap the input if needed */
|
|
}
|
|
|
|
if (lpIMC->cfCompForm.dwStyle == CFS_DEFAULT)
|
|
{
|
|
/* make sure we are on the desktop */
|
|
mon_info.cbSize = sizeof(mon_info);
|
|
GetMonitorInfoW(monitor, &mon_info);
|
|
|
|
if (rect.bottom > mon_info.rcWork.bottom)
|
|
{
|
|
int shift = rect.bottom - mon_info.rcWork.bottom;
|
|
rect.top -= shift;
|
|
rect.bottom -= shift;
|
|
}
|
|
if (rect.left < 0)
|
|
{
|
|
rect.right -= rect.left;
|
|
rect.left = 0;
|
|
}
|
|
if (rect.right > mon_info.rcWork.right)
|
|
{
|
|
int shift = rect.right - mon_info.rcWork.right;
|
|
rect.left -= shift;
|
|
rect.right -= shift;
|
|
}
|
|
}
|
|
|
|
SetWindowPos(hwnd, HWND_TOPMOST, rect.left, rect.top, rect.right - rect.left,
|
|
rect.bottom - rect.top, SWP_NOACTIVATE);
|
|
|
|
TextOutW(hdc, offX, offY, CompString, compstr->dwCompStrLen);
|
|
|
|
if (oldfont)
|
|
SelectObject(hdc, oldfont);
|
|
}
|
|
|
|
ImmUnlockIMCC(lpIMC->hCompStr);
|
|
|
|
EndPaint(hwnd, &ps);
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
static void DefaultIMEComposition(HIMC hIMC, HWND hwnd, LPARAM lParam)
|
|
{
|
|
TRACE("IME message WM_IME_COMPOSITION 0x%lx\n", lParam);
|
|
if (lParam & GCS_RESULTSTR)
|
|
{
|
|
LPCOMPOSITIONSTRING compstr;
|
|
LPBYTE compdata;
|
|
LPWSTR ResultStr;
|
|
HIMCC newCompStr;
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
TRACE("Posting result as IME_CHAR\n");
|
|
compdata = ImmLockIMCC(lpIMC->hCompStr);
|
|
compstr = (LPCOMPOSITIONSTRING)compdata;
|
|
ResultStr = (LPWSTR)(compdata + compstr->dwResultStrOffset);
|
|
GenerateIMECHARMessages(hIMC, ResultStr, compstr->dwResultStrLen);
|
|
ImmUnlockIMCC(lpIMC->hCompStr);
|
|
|
|
/* clear the buffer */
|
|
newCompStr = updateResultStr(lpIMC->hCompStr, NULL, 0);
|
|
ImmDestroyIMCC(lpIMC->hCompStr);
|
|
lpIMC->hCompStr = newCompStr;
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
else
|
|
UpdateDataInDefaultIMEWindow(hIMC, hwnd, TRUE);
|
|
}
|
|
|
|
static void DefaultIMEStartComposition(HIMC hIMC, HWND hwnd)
|
|
{
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC == NULL)
|
|
return;
|
|
|
|
TRACE("IME message WM_IME_STARTCOMPOSITION\n");
|
|
lpIMC->hWnd = GetFocus();
|
|
ShowWindow(hwnd, SW_SHOWNOACTIVATE);
|
|
UnlockRealIMC(hIMC);
|
|
}
|
|
|
|
static LRESULT ImeHandleNotify(HIMC hIMC, HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (wParam)
|
|
{
|
|
case IMN_OPENSTATUSWINDOW:
|
|
FIXME("WM_IME_NOTIFY:IMN_OPENSTATUSWINDOW\n");
|
|
break;
|
|
case IMN_CLOSESTATUSWINDOW:
|
|
FIXME("WM_IME_NOTIFY:IMN_CLOSESTATUSWINDOW\n");
|
|
break;
|
|
case IMN_OPENCANDIDATE:
|
|
FIXME("WM_IME_NOTIFY:IMN_OPENCANDIDATE\n");
|
|
break;
|
|
case IMN_CHANGECANDIDATE:
|
|
FIXME("WM_IME_NOTIFY:IMN_CHANGECANDIDATE\n");
|
|
break;
|
|
case IMN_CLOSECANDIDATE:
|
|
FIXME("WM_IME_NOTIFY:IMN_CLOSECANDIDATE\n");
|
|
break;
|
|
case IMN_SETCONVERSIONMODE:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETCONVERSIONMODE\n");
|
|
break;
|
|
case IMN_SETSENTENCEMODE:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETSENTENCEMODE\n");
|
|
break;
|
|
case IMN_SETOPENSTATUS:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETOPENSTATUS\n");
|
|
break;
|
|
case IMN_SETCANDIDATEPOS:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETCANDIDATEPOS\n");
|
|
break;
|
|
case IMN_SETCOMPOSITIONFONT:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETCOMPOSITIONFONT\n");
|
|
break;
|
|
case IMN_SETCOMPOSITIONWINDOW:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETCOMPOSITIONWINDOW\n");
|
|
break;
|
|
case IMN_GUIDELINE:
|
|
FIXME("WM_IME_NOTIFY:IMN_GUIDELINE\n");
|
|
break;
|
|
case IMN_SETSTATUSWINDOWPOS:
|
|
FIXME("WM_IME_NOTIFY:IMN_SETSTATUSWINDOWPOS\n");
|
|
break;
|
|
default:
|
|
FIXME("WM_IME_NOTIFY:<Unknown 0x%lx>\n", wParam);
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static LRESULT WINAPI IME_WindowProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT rc = 0;
|
|
HIMC hIMC;
|
|
|
|
TRACE("Incoming Message 0x%x (0x%08lx, 0x%08lx)\n", msg, wParam, lParam);
|
|
|
|
/*
|
|
* Each UI window contains the current Input Context.
|
|
* This Input Context can be obtained by calling GetWindowLong
|
|
* with IMMGWL_IMC when the UI window receives a WM_IME_xxx message.
|
|
* The UI window can refer to this Input Context and handles the
|
|
* messages.
|
|
*/
|
|
|
|
hIMC = (HIMC)GetWindowLongPtrW(hwnd, IMMGWL_IMC);
|
|
if (!hIMC)
|
|
hIMC = RealIMC(FROM_MACDRV);
|
|
|
|
/* if we have no hIMC there are many messages we cannot process */
|
|
if (hIMC == NULL)
|
|
{
|
|
switch (msg) {
|
|
case WM_IME_STARTCOMPOSITION:
|
|
case WM_IME_ENDCOMPOSITION:
|
|
case WM_IME_COMPOSITION:
|
|
case WM_IME_NOTIFY:
|
|
case WM_IME_CONTROL:
|
|
case WM_IME_COMPOSITIONFULL:
|
|
case WM_IME_SELECT:
|
|
case WM_IME_CHAR:
|
|
return 0L;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (msg)
|
|
{
|
|
case WM_CREATE:
|
|
{
|
|
LPIMEPRIVATE myPrivate;
|
|
LPINPUTCONTEXT lpIMC;
|
|
|
|
SetWindowTextA(hwnd, "Wine Ime Active");
|
|
|
|
lpIMC = LockRealIMC(hIMC);
|
|
if (lpIMC)
|
|
{
|
|
myPrivate = ImmLockIMCC(lpIMC->hPrivate);
|
|
myPrivate->hwndDefault = hwnd;
|
|
ImmUnlockIMCC(lpIMC->hPrivate);
|
|
}
|
|
UnlockRealIMC(hIMC);
|
|
|
|
return TRUE;
|
|
}
|
|
case WM_PAINT:
|
|
PaintDefaultIMEWnd(hIMC, hwnd);
|
|
return FALSE;
|
|
|
|
case WM_NCCREATE:
|
|
return TRUE;
|
|
|
|
case WM_SETFOCUS:
|
|
if (wParam)
|
|
SetFocus((HWND)wParam);
|
|
else
|
|
FIXME("Received focus, should never have focus\n");
|
|
break;
|
|
case WM_IME_COMPOSITION:
|
|
DefaultIMEComposition(hIMC, hwnd, lParam);
|
|
break;
|
|
case WM_IME_STARTCOMPOSITION:
|
|
DefaultIMEStartComposition(hIMC, hwnd);
|
|
break;
|
|
case WM_IME_ENDCOMPOSITION:
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_IME_ENDCOMPOSITION", wParam, lParam);
|
|
ShowWindow(hwnd, SW_HIDE);
|
|
break;
|
|
case WM_IME_SELECT:
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_IME_SELECT", wParam, lParam);
|
|
break;
|
|
case WM_IME_CONTROL:
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_IME_CONTROL", wParam, lParam);
|
|
rc = 1;
|
|
break;
|
|
case WM_IME_NOTIFY:
|
|
rc = ImeHandleNotify(hIMC, hwnd, msg, wParam, lParam);
|
|
break;
|
|
default:
|
|
TRACE("Non-standard message 0x%x\n", msg);
|
|
}
|
|
/* check the MSIME messages */
|
|
if (msg == WM_MSIME_SERVICE)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_SERVICE", wParam, lParam);
|
|
rc = FALSE;
|
|
}
|
|
else if (msg == WM_MSIME_RECONVERTOPTIONS)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_RECONVERTOPTIONS", wParam, lParam);
|
|
}
|
|
else if (msg == WM_MSIME_MOUSE)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_MOUSE", wParam, lParam);
|
|
}
|
|
else if (msg == WM_MSIME_RECONVERTREQUEST)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_RECONVERTREQUEST", wParam, lParam);
|
|
}
|
|
else if (msg == WM_MSIME_RECONVERT)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_RECONVERT", wParam, lParam);
|
|
}
|
|
else if (msg == WM_MSIME_QUERYPOSITION)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_QUERYPOSITION", wParam, lParam);
|
|
}
|
|
else if (msg == WM_MSIME_DOCUMENTFEED)
|
|
{
|
|
TRACE("IME message %s, 0x%lx, 0x%lx\n", "WM_MSIME_DOCUMENTFEED", wParam, lParam);
|
|
}
|
|
/* DefWndProc if not an IME message */
|
|
if (!rc && !((msg >= WM_IME_STARTCOMPOSITION && msg <= WM_IME_KEYLAST) ||
|
|
(msg >= WM_IME_SETCONTEXT && msg <= WM_IME_KEYUP)))
|
|
rc = DefWindowProcW(hwnd, msg, wParam, lParam);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static BOOL WINAPI register_classes( INIT_ONCE *once, void *param, void **context )
|
|
{
|
|
WNDCLASSW wndClass;
|
|
ZeroMemory(&wndClass, sizeof(WNDCLASSW));
|
|
wndClass.style = CS_GLOBALCLASS | CS_IME | CS_HREDRAW | CS_VREDRAW;
|
|
wndClass.lpfnWndProc = (WNDPROC) IME_WindowProc;
|
|
wndClass.cbClsExtra = 0;
|
|
wndClass.cbWndExtra = 2 * sizeof(LONG_PTR);
|
|
wndClass.hInstance = macdrv_module;
|
|
wndClass.hCursor = LoadCursorW(NULL, (LPWSTR)IDC_ARROW);
|
|
wndClass.hIcon = LoadIconW(NULL, (LPWSTR)IDI_APPLICATION);
|
|
wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
|
|
wndClass.lpszMenuName = 0;
|
|
wndClass.lpszClassName = UI_CLASS_NAME;
|
|
|
|
RegisterClassW(&wndClass);
|
|
|
|
WM_MSIME_SERVICE = RegisterWindowMessageA("MSIMEService");
|
|
WM_MSIME_RECONVERTOPTIONS = RegisterWindowMessageA("MSIMEReconvertOptions");
|
|
WM_MSIME_MOUSE = RegisterWindowMessageA("MSIMEMouseOperation");
|
|
WM_MSIME_RECONVERTREQUEST = RegisterWindowMessageA("MSIMEReconvertRequest");
|
|
WM_MSIME_RECONVERT = RegisterWindowMessageA("MSIMEReconvert");
|
|
WM_MSIME_QUERYPOSITION = RegisterWindowMessageA("MSIMEQueryPosition");
|
|
WM_MSIME_DOCUMENTFEED = RegisterWindowMessageA("MSIMEDocumentFeed");
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL WINAPI ImeInquire(LPIMEINFO lpIMEInfo, LPWSTR lpszUIClass, LPCWSTR lpszOption)
|
|
{
|
|
static INIT_ONCE init_once = INIT_ONCE_STATIC_INIT;
|
|
|
|
TRACE("\n");
|
|
InitOnceExecuteOnce( &init_once, register_classes, NULL, NULL );
|
|
lpIMEInfo->dwPrivateDataSize = sizeof(IMEPRIVATE);
|
|
lpIMEInfo->fdwProperty = IME_PROP_UNICODE | IME_PROP_AT_CARET;
|
|
lpIMEInfo->fdwConversionCaps = IME_CMODE_NATIVE | IME_CMODE_FULLSHAPE;
|
|
lpIMEInfo->fdwSentenceCaps = IME_SMODE_AUTOMATIC;
|
|
lpIMEInfo->fdwUICaps = UI_CAP_2700;
|
|
/* Tell App we cannot accept ImeSetCompositionString calls */
|
|
/* FIXME: Can we? */
|
|
lpIMEInfo->fdwSCSCaps = 0;
|
|
lpIMEInfo->fdwSelectCaps = SELECT_CAP_CONVERSION;
|
|
|
|
lstrcpyW(lpszUIClass, UI_CLASS_NAME);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Interfaces to other parts of the Mac driver */
|
|
|
|
/***********************************************************************
|
|
* macdrv_im_set_text
|
|
*/
|
|
void macdrv_im_set_text(const macdrv_event *event)
|
|
{
|
|
HWND hwnd = macdrv_get_window_hwnd(event->window);
|
|
void *himc = event->im_set_text.data;
|
|
|
|
TRACE("win %p/%p himc %p text %s complete %u\n", hwnd, event->window, himc,
|
|
debugstr_cf(event->im_set_text.text), event->im_set_text.complete);
|
|
|
|
if (!himc) himc = RealIMC(FROM_MACDRV);
|
|
|
|
if (event->im_set_text.text)
|
|
{
|
|
CFIndex length = CFStringGetLength(event->im_set_text.text);
|
|
const UniChar *chars = CFStringGetCharactersPtr(event->im_set_text.text);
|
|
UniChar *buffer = NULL;
|
|
|
|
if (!chars)
|
|
{
|
|
buffer = HeapAlloc(GetProcessHeap(), 0, length * sizeof(*buffer));
|
|
CFStringGetCharacters(event->im_set_text.text, CFRangeMake(0, length), buffer);
|
|
chars = buffer;
|
|
}
|
|
|
|
if (himc)
|
|
IME_SetCompositionString(himc, SCS_SETSTR, chars, length * sizeof(*chars),
|
|
event->im_set_text.cursor_pos, !event->im_set_text.complete);
|
|
else
|
|
{
|
|
INPUT input;
|
|
CFIndex i;
|
|
|
|
input.type = INPUT_KEYBOARD;
|
|
input.ki.wVk = 0;
|
|
input.ki.time = 0;
|
|
input.ki.dwExtraInfo = 0;
|
|
|
|
for (i = 0; i < length; i++)
|
|
{
|
|
input.ki.wScan = chars[i];
|
|
input.ki.dwFlags = KEYEVENTF_UNICODE;
|
|
__wine_send_input(hwnd, &input);
|
|
|
|
input.ki.dwFlags = KEYEVENTF_UNICODE | KEYEVENTF_KEYUP;
|
|
__wine_send_input(hwnd, &input);
|
|
}
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, buffer);
|
|
}
|
|
|
|
if (event->im_set_text.complete)
|
|
IME_NotifyComplete(himc);
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* query_ime_char_rect
|
|
*/
|
|
BOOL query_ime_char_rect(macdrv_query* query)
|
|
{
|
|
HWND hwnd = macdrv_get_window_hwnd(query->window);
|
|
void *himc = query->ime_char_rect.data;
|
|
CFRange* range = &query->ime_char_rect.range;
|
|
CGRect* rect = &query->ime_char_rect.rect;
|
|
IMECHARPOSITION charpos;
|
|
BOOL ret = FALSE;
|
|
|
|
TRACE("win %p/%p himc %p range %ld-%ld\n", hwnd, query->window, himc, range->location,
|
|
range->length);
|
|
|
|
if (!himc) himc = RealIMC(FROM_MACDRV);
|
|
|
|
charpos.dwSize = sizeof(charpos);
|
|
charpos.dwCharPos = range->location;
|
|
if (ImmRequestMessageW(himc, IMR_QUERYCHARPOSITION, (ULONG_PTR)&charpos))
|
|
{
|
|
int i;
|
|
|
|
*rect = CGRectMake(charpos.pt.x, charpos.pt.y, 0, charpos.cLineHeight);
|
|
|
|
/* iterate over rest of length to extend rect */
|
|
for (i = 1; i < range->length; i++)
|
|
{
|
|
charpos.dwSize = sizeof(charpos);
|
|
charpos.dwCharPos = range->location + i;
|
|
if (!ImmRequestMessageW(himc, IMR_QUERYCHARPOSITION, (ULONG_PTR)&charpos) ||
|
|
charpos.pt.y != rect->origin.y)
|
|
{
|
|
range->length = i;
|
|
break;
|
|
}
|
|
|
|
rect->size.width = charpos.pt.x - rect->origin.x;
|
|
}
|
|
|
|
ret = TRUE;
|
|
}
|
|
|
|
if (!ret)
|
|
{
|
|
LPINPUTCONTEXT ic = ImmLockIMC(himc);
|
|
|
|
if (ic)
|
|
{
|
|
LPIMEPRIVATE private = ImmLockIMCC(ic->hPrivate);
|
|
LPBYTE compdata = ImmLockIMCC(ic->hCompStr);
|
|
LPCOMPOSITIONSTRING compstr = (LPCOMPOSITIONSTRING)compdata;
|
|
LPWSTR str = (LPWSTR)(compdata + compstr->dwCompStrOffset);
|
|
|
|
if (private->hwndDefault && compstr->dwCompStrOffset &&
|
|
IsWindowVisible(private->hwndDefault))
|
|
{
|
|
HDC dc = GetDC(private->hwndDefault);
|
|
HFONT oldfont = NULL;
|
|
SIZE size;
|
|
|
|
if (private->textfont)
|
|
oldfont = SelectObject(dc, private->textfont);
|
|
|
|
if (range->location > compstr->dwCompStrLen)
|
|
range->location = compstr->dwCompStrLen;
|
|
if (range->location + range->length > compstr->dwCompStrLen)
|
|
range->length = compstr->dwCompStrLen - range->location;
|
|
|
|
GetTextExtentPoint32W(dc, str, range->location, &size);
|
|
charpos.rcDocument.left = size.cx;
|
|
charpos.rcDocument.top = 0;
|
|
GetTextExtentPoint32W(dc, str, range->location + range->length, &size);
|
|
charpos.rcDocument.right = size.cx;
|
|
charpos.rcDocument.bottom = size.cy;
|
|
|
|
if (ic->cfCompForm.dwStyle == CFS_DEFAULT)
|
|
OffsetRect(&charpos.rcDocument, 10, 10);
|
|
|
|
LPtoDP(dc, (POINT*)&charpos.rcDocument, 2);
|
|
MapWindowPoints(private->hwndDefault, 0, (POINT*)&charpos.rcDocument, 2);
|
|
*rect = cgrect_from_rect(charpos.rcDocument);
|
|
ret = TRUE;
|
|
|
|
if (oldfont)
|
|
SelectObject(dc, oldfont);
|
|
ReleaseDC(private->hwndDefault, dc);
|
|
}
|
|
|
|
ImmUnlockIMCC(ic->hCompStr);
|
|
ImmUnlockIMCC(ic->hPrivate);
|
|
}
|
|
|
|
ImmUnlockIMC(himc);
|
|
}
|
|
|
|
if (!ret)
|
|
{
|
|
GUITHREADINFO gti;
|
|
gti.cbSize = sizeof(gti);
|
|
if (GetGUIThreadInfo(0, >i))
|
|
{
|
|
MapWindowPoints(gti.hwndCaret, 0, (POINT*)>i.rcCaret, 2);
|
|
*rect = cgrect_from_rect(gti.rcCaret);
|
|
ret = TRUE;
|
|
}
|
|
}
|
|
|
|
if (ret && range->length && !rect->size.width)
|
|
rect->size.width = 1;
|
|
|
|
TRACE(" -> %s range %ld-%ld rect %s\n", ret ? "TRUE" : "FALSE", range->location,
|
|
range->length, wine_dbgstr_cgrect(*rect));
|
|
|
|
return ret;
|
|
}
|