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

618 lines
19 KiB
C

#include <windows.h>
#include <immdev.h>
#include "imeattr.h"
#include "imedefs.h"
#include "imerc.h"
#include "uniime.h"
BOOL IsBig5Character( WCHAR wChar );
/**********************************************************************/
/* AddPhraseString() */
/**********************************************************************/
DWORD PASCAL AddPhraseString(
LPIMEL lpImeL,
LPCANDIDATELIST lpCandList,
DWORD dwPhraseOffset,
DWORD dwPhraseNextOffset,
DWORD dwStartLen,
DWORD dwEndLen,
DWORD dwAddCandLimit,
BOOL fConvertCP)
{
HANDLE hLCPhraseTbl;
LPWSTR lpLCPhraseTbl;
LPWSTR lpStart, lpEnd;
int iBytes;
DWORD dwMaxCand;
DWORD dwOffset;
BOOL bIsNotBig5Char, bIsBig5OnlyMode;
// put the strings into candidate list
hLCPhraseTbl = OpenFileMapping(FILE_MAP_READ, FALSE,
sImeG.szTblFile[1]);
if (!hLCPhraseTbl) {
return (0);
}
lpLCPhraseTbl = (LPWSTR)MapViewOfFile(hLCPhraseTbl, FILE_MAP_READ,
0, 0, 0);
if (!lpLCPhraseTbl) {
dwOffset = 0;
goto AddPhraseStringUnmap;
}
if (lpImeL->fdwModeConfig & MODE_CONFIG_BIG5ONLY)
bIsBig5OnlyMode = TRUE;
else
bIsBig5OnlyMode = FALSE;
lpStart = lpLCPhraseTbl + dwPhraseOffset;
lpEnd = lpLCPhraseTbl + dwPhraseNextOffset;
if (!lpCandList) {
// ask how many candidate list strings
dwOffset = 0;
for (lpStart; lpStart < lpEnd;) {
bIsNotBig5Char = FALSE;
for (; lpStart < lpEnd; lpStart++) {
WORD uCode;
uCode = *lpStart;
// one string finished
#ifdef UNICODE
if (!uCode) {
#else
if (!(uCode & 0x8000)) {
#endif
lpStart++;
break;
}
if ( bIsBig5OnlyMode ) {
if ( IsBig5Character((WCHAR)uCode) == FALSE )
bIsNotBig5Char = TRUE;
}
}
// if it is in Big5 Only Mode, and there is at least one char which is not
// in Big5 charset, we don't count this string
if ( bIsBig5OnlyMode && bIsNotBig5Char )
continue;
// string count plus 1
dwOffset++;
}
goto AddPhraseStringUnmap;
}
// the offset of dwOffset[0]
dwOffset = (DWORD)((LPBYTE)&lpCandList->dwOffset[0] - (LPBYTE)lpCandList);
if (lpCandList->dwSize < dwOffset) {
return (0);
}
// how many bytes of dwOffset[]
iBytes = lpCandList->dwOffset[0] - dwOffset;
// maybe the size is even smaller than it
for (dwMaxCand = 1; dwMaxCand < lpCandList->dwCount; dwMaxCand++) {
if ((int)(lpCandList->dwOffset[dwMaxCand] - dwOffset) < iBytes) {
iBytes = (int)(lpCandList->dwOffset[dwMaxCand] - dwOffset);
}
}
if (iBytes <= 0) {
return (0);
}
dwMaxCand = (DWORD)iBytes / sizeof(DWORD);
if (dwAddCandLimit < dwMaxCand) {
dwMaxCand = dwAddCandLimit;
}
if (lpCandList->dwCount >= dwMaxCand) {
// Grow memory here and do something,
// if you still want to process it.
return (0);
}
dwOffset = lpCandList->dwOffset[lpCandList->dwCount];
for (lpStart; lpStart < lpEnd;) {
BOOL fStrEnd;
DWORD dwStrLen, dwCharLen, dwStrByteLen, dwCharByteLen;
fStrEnd = FALSE;
bIsNotBig5Char = FALSE;
// get the whole string
dwCharByteLen = sizeof(WCHAR);
dwCharLen = sizeof(WCHAR) / sizeof(TCHAR);
for (dwStrLen = dwStrByteLen = 0; !fStrEnd && (lpStart < lpEnd);
lpStart++, dwStrLen+= dwCharLen, dwStrByteLen += dwCharByteLen) {
WORD uCode;
uCode = *lpStart;
// one string finished
#ifdef UNICODE
if (!uCode) {
#else
if (!(uCode & 0x8000)) {
#endif
fStrEnd = TRUE;
#ifdef UNICODE
lpStart++;
break;
#else
uCode |= 0x8000;
#endif
}
// if it is Big5Only Mode, we need to check if this char is in Big5 charset
if ( bIsBig5OnlyMode ) {
if ( !IsBig5Character((WCHAR)uCode) )
bIsNotBig5Char = TRUE;
}
#ifdef UNICODE
if (fConvertCP) {
CHAR szCode[4];
dwCharLen = dwCharByteLen = WideCharToMultiByte(
sImeG.uAnsiCodePage, WC_COMPOSITECHECK,
(LPCWSTR)&uCode, 1, szCode, sizeof(szCode), NULL, NULL);
// because this BIG5 code, convert to BIG5 string
if (dwCharByteLen >= 2) {
uCode = (BYTE)szCode[0] | ((UINT)(BYTE)szCode[1] << 8);
} else {
uCode = (UINT)szCode[0];
}
}
#else
// swap lead & second byte (as a string), UNICODE don't need it
uCode = HIBYTE(uCode) | (LOBYTE(uCode) << 8);
#endif
if ((dwOffset + dwStrByteLen + dwCharByteLen) >=
lpCandList->dwSize) {
goto AddPhraseStringClose;
}
// add this char into candidate list
#ifdef UNICODE
if (dwCharByteLen == sizeof(WCHAR)) {
*(LPWSTR)((LPBYTE)lpCandList + dwOffset + dwStrByteLen) =
(WCHAR)uCode;
} else {
*(LPSTR)((LPBYTE)lpCandList + dwOffset + dwStrByteLen) =
(CHAR)uCode;
}
#else
*(LPWSTR)((LPBYTE)lpCandList + dwOffset + dwStrByteLen) =
(WCHAR)uCode;
#endif
}
if (dwStrLen < dwStartLen) {
// the found string too short
continue;
} else if (dwStrLen >= dwEndLen) {
// the found string too long
continue;
} else {
}
// if it is in Big5 Only Mode, and there is at least one char which is not in Big5
// charset, we just ingore this string, do not put it into the candidate list
if ( bIsBig5OnlyMode && bIsNotBig5Char ) {
bIsNotBig5Char = FALSE;
continue;
}
if ((dwOffset + dwStrByteLen + sizeof(TCHAR)) >= lpCandList->dwSize) {
goto AddPhraseStringClose;
}
// null terminator
*(LPTSTR)((LPBYTE)lpCandList + dwOffset + dwStrByteLen) = TEXT('\0');
dwOffset += (dwStrByteLen + sizeof(TCHAR));
// add one string into candidate list
lpCandList->dwCount++;
if (lpCandList->dwCount >= dwMaxCand) {
// Grow memory here and do something,
// if you still want to process it.
break;
}
// string length plus size of the null terminator
lpCandList->dwOffset[lpCandList->dwCount] = dwOffset;
}
AddPhraseStringUnmap:
UnmapViewOfFile(lpLCPhraseTbl);
AddPhraseStringClose:
CloseHandle(hLCPhraseTbl);
return (dwOffset);
}
/**********************************************************************/
/* UniSearchPhrasePrediction() */
/* Description: */
/* file format can be changed in different version for */
/* performance consideration, ISVs should not assume its format */
/* and serach these files by themselves */
/**********************************************************************/
DWORD WINAPI UniSearchPhrasePrediction(
LPIMEL lpImeL,
UINT uCodePage,
LPCTSTR lpszStr,
DWORD dwStrLen,
LPCTSTR lpszReadStr, // Phonetic reading string
DWORD dwReadStrLen,
DWORD dwStartLen, // find the string length >= this value
DWORD dwEndLen, // find the string length < this value
DWORD dwAddCandLimit,
LPCANDIDATELIST lpCandList)
{
UINT uCode;
HANDLE hLCPtrTbl;
LPWORD lpLCPtrTbl;
int iLo, iHi, iMid;
BOOL fFound, fConvertCP;
DWORD dwPhraseOffset, dwPhraseNextOffset;
if (uCodePage == NATIVE_CP) {
fConvertCP = FALSE;
#ifdef UNICODE
} else if (uCodePage == sImeG.uAnsiCodePage) {
fConvertCP = TRUE;
#endif
} else {
return (0);
}
if (dwStrLen != sizeof(WCHAR) / sizeof(TCHAR)) {
return (0);
}
if (dwStartLen >= dwEndLen) {
return (0);
}
#ifdef UNICODE
uCode = lpszStr[0];
#else
// swap lead byte & second byte, UNICODE don't need it
uCode = (BYTE)lpszStr[1];
*((LPBYTE)&uCode + 1) = (BYTE)lpszStr[0];
#endif
iLo = 0;
#ifdef UNICODE
iHi = sImeG.uTblSize[0] / 6;
#else
iHi = sImeG.uTblSize[0] / 4;
#endif
iMid = (iHi + iLo) /2;
fFound = FALSE;
// LCPTR.TBL
hLCPtrTbl = OpenFileMapping(FILE_MAP_READ, FALSE, sImeG.szTblFile[0]);
if (!hLCPtrTbl) {
return (0);
}
lpLCPtrTbl = MapViewOfFile(hLCPtrTbl, FILE_MAP_READ, 0, 0, 0);
if (!lpLCPtrTbl) {
goto SrchPhrPredictClose;
}
// binary search on phrase table,
// one is multiple word phrase and the other is the two word phrase
for (; iLo <= iHi;) {
LPWORD lpCurr;
#ifdef UNICODE
lpCurr = lpLCPtrTbl + 3 * iMid;
#else
lpCurr = lpLCPtrTbl + 2 * iMid;
#endif
if (uCode > *lpCurr) {
iLo = iMid + 1;
} else if (uCode < *lpCurr) {
iHi = iMid - 1;
} else {
fFound = TRUE;
// use it on TAB key
#ifdef UNICODE
dwPhraseOffset = *(LPUNADWORD)(lpCurr + 1);
dwPhraseNextOffset = *(LPUNADWORD)(lpCurr + 1 + 3);
#else
dwPhraseOffset = *(lpCurr + 1);
dwPhraseNextOffset = *(lpCurr + 1 + 2);
#endif
break;
}
iMid = (iHi + iLo) /2;
}
UnmapViewOfFile(lpLCPtrTbl);
SrchPhrPredictClose:
CloseHandle(hLCPtrTbl);
if (!fFound) {
return (0);
}
// phrase string
return AddPhraseString(lpImeL,lpCandList, dwPhraseOffset, dwPhraseNextOffset,
dwStartLen, dwEndLen, dwAddCandLimit, fConvertCP);
}
/**********************************************************************/
/* UniSearchPhrasePredictionStub() */
/* Description: */
/* file format can be changed in different version for */
/* performance consideration, ISVs should not assume its format */
/* and serach these files by themselves */
/**********************************************************************/
DWORD WINAPI UniSearchPhrasePredictionStub(
LPIMEL lpImeL,
UINT uCodePage,
LPCSTUBSTR lpszStr,
DWORD dwStrLen,
LPCSTUBSTR lpszReadStr, // Phonetic reading string
DWORD dwReadStrLen,
DWORD dwStartLen, // find the string length >= this value
DWORD dwEndLen, // find the string length < this value
DWORD dwAddCandLimit,
LPCANDIDATELIST lpCandList)
{
#ifdef UNICODE
LPTSTR lpszWideStr, lpszWideReadStr;
DWORD dwWideStrLen, dwWideReadStrLen;
DWORD dwWideStartLen, dwWideEndLen;
DWORD dwWideAddCandList, dwRet;
LPCANDIDATELIST lpWideCandList;
LPBYTE lpbBuf;
if (uCodePage != sImeG.uAnsiCodePage) {
return (0);
}
dwRet = dwStrLen * sizeof(WCHAR) + dwReadStrLen * sizeof(WCHAR);
lpbBuf = (LPBYTE)GlobalAlloc(GPTR, dwRet);
if ( lpbBuf == NULL )
return 0;
if (lpszStr) {
lpszWideStr = (LPTSTR)lpbBuf;
dwWideStrLen = MultiByteToWideChar(sImeG.uAnsiCodePage,
MB_PRECOMPOSED, lpszStr, dwStrLen,
lpszWideStr, dwStrLen);
} else {
lpszWideStr = NULL;
dwWideStrLen = 0;
}
if (lpszReadStr) {
lpszWideReadStr = (LPTSTR)(lpbBuf + dwStrLen * sizeof(WCHAR));
dwWideReadStrLen = MultiByteToWideChar(sImeG.uAnsiCodePage,
MB_PRECOMPOSED, lpszReadStr, dwReadStrLen,
lpszWideReadStr, dwReadStrLen);
} else {
lpszWideReadStr = NULL;
dwWideReadStrLen = 0;
}
dwRet = UniSearchPhrasePrediction(lpImeL,uCodePage, lpszWideStr, dwWideStrLen,
lpszWideReadStr, dwWideReadStrLen, dwStartLen, dwEndLen,
dwAddCandLimit, lpCandList);
// now, start W to A conversion and fliter the real limit here
GlobalFree((HGLOBAL)lpbBuf);
return (dwRet);
#else
return (0);
#endif
}
/**********************************************************************/
/* MemoryLack() */
/**********************************************************************/
void PASCAL MemoryLack(
DWORD fdwErrMsg)
{
TCHAR szErrMsg[64];
TCHAR szIMEName[16];
if (sImeG.fdwErrMsg & fdwErrMsg) {
// message already prompted
return;
}
LoadString(hInst, IDS_MEM_LACK_FAIL, szErrMsg, sizeof(szErrMsg)/sizeof(TCHAR));
LoadString(hInst, IDS_IMENAME, szIMEName, sizeof(szIMEName)/sizeof(TCHAR) );
sImeG.fdwErrMsg |= fdwErrMsg;
MessageBeep((UINT)-1);
MessageBox((HWND)NULL, szErrMsg, szIMEName,
MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST);
return;
}
/**********************************************************************/
/* LoadOneGlobalTable() */
/* Description: */
/* memory handle & size of .TBL file will be assigned to */
/* sImeG */
/* Eeturn Value: */
/* length of directory of the .TBL file */
/**********************************************************************/
UINT PASCAL LoadOneGlobalTable( // load one of table file
LPTSTR szTable, // file name of .TBL
UINT uIndex, // the index of array to store memory handle
UINT uLen, // length of the directory
LPTSTR szPath) // buffer for directory
{
HANDLE hTblFile;
HGLOBAL hMap;
TCHAR szFullPathFile[MAX_PATH];
PSECURITY_ATTRIBUTES psa;
CopyMemory(szFullPathFile, szPath, uLen * sizeof(TCHAR));
psa = CreateSecurityAttributes();
if (uLen) {
CopyMemory(&szFullPathFile[uLen], szTable, sizeof(sImeG.szTblFile[0]));
hTblFile = CreateFile(szFullPathFile, GENERIC_READ,
FILE_SHARE_READ|FILE_SHARE_WRITE,
psa, OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
} else {
// try system directory
uLen = GetSystemDirectory(szFullPathFile, MAX_PATH);
if (szFullPathFile[uLen - 1] != TEXT('\\')) { // consider N:\ ;
szFullPathFile[uLen++] = TEXT('\\');
}
CopyMemory(&szFullPathFile[uLen], szTable, sizeof(sImeG.szTblFile[0]));
hTblFile = CreateFile(szFullPathFile, GENERIC_READ,
FILE_SHARE_READ|FILE_SHARE_WRITE,
psa, OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
if (hTblFile != INVALID_HANDLE_VALUE) {
goto CopyDicPath;
}
// if the work station version, SHARE_WRITE will fail
hTblFile = CreateFile(szFullPathFile, GENERIC_READ,
FILE_SHARE_READ,
psa, OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL);
CopyDicPath:
if (hTblFile != INVALID_HANDLE_VALUE) {
CopyMemory(sImeG.szPhrasePath, szFullPathFile, uLen * sizeof(TCHAR));
sImeG.uPathLen = uLen;
goto OpenDicFile;
}
}
OpenDicFile:
// can not find the table file
if (hTblFile != INVALID_HANDLE_VALUE) { // OK
} else if (sImeG.fdwErrMsg & (ERRMSG_LOAD_0 << uIndex)) {
// already prompt error message before, no more
FreeSecurityAttributes(psa);
return (0);
} else { // prompt error message
TCHAR szIMEName[64];
TCHAR szErrMsg[2 * MAX_PATH];
// temp use szIMEName as format string buffer of error message
LoadString(hInst, IDS_FILE_OPEN_FAIL, szIMEName, sizeof(szIMEName)/sizeof(TCHAR));
wsprintf(szErrMsg, szIMEName, szTable);
LoadString(hInst, IDS_IMENAME, szIMEName, sizeof(szIMEName)/sizeof(TCHAR));
sImeG.fdwErrMsg |= ERRMSG_LOAD_0 << uIndex;
MessageBeep((UINT)-1);
MessageBox((HWND)NULL, szErrMsg, szIMEName,
MB_OK|MB_ICONHAND|MB_TASKMODAL|MB_TOPMOST);
FreeSecurityAttributes(psa);
return (0);
}
sImeG.fdwErrMsg &= ~(ERRMSG_LOAD_0 << uIndex);
// create file mapping for IME tables
hMap = CreateFileMapping((HANDLE)hTblFile, psa, PAGE_READONLY,
0, 0, szTable);
if (!hMap) {
MemoryLack(ERRMSG_MEM_0 << uIndex);
CloseHandle(hTblFile);
FreeSecurityAttributes(psa);
return(0);
}
sImeG.fdwErrMsg &= ~(ERRMSG_MEM_0 << uIndex);
sInstG.hMapTbl[uIndex] = hMap;
// get file length
sImeG.uTblSize[uIndex] = GetFileSize(hTblFile, (LPDWORD)NULL);
CloseHandle(hTblFile);
FreeSecurityAttributes(psa);
return (uLen);
}
/**********************************************************************/
/* LoadPhraseTable() */
/* Return Value: */
/* TRUE - successful, FALSE - failure */
/**********************************************************************/
BOOL PASCAL LoadPhraseTable( // load the phrase tables
UINT uLen, // length of the directory
LPTSTR szPath) // buffer for directory
{
int i;
for (i = 0; i < MAX_PHRASE_TABLES; i++) {
if (!*sImeG.szTblFile[i]) {
} else if (sInstG.hMapTbl[i]) { // already loaded
} else if (uLen = LoadOneGlobalTable(sImeG.szTblFile[i], i,
uLen, szPath)) {
} else {
int j;
for (j = 0; j < i; j++) {
if (sInstG.hMapTbl[j]) {
CloseHandle(sInstG.hMapTbl[j]);
sInstG.hMapTbl[j] = (HANDLE)NULL;
}
}
return (FALSE);
}
}
return (TRUE);
}