/*++ Copyright (c) 1991-1997, Microsoft Corporation All rights reserved. Module Name: charmap.c Abstract: This module contains the main routines for the Charmap utility, an interface for selecting special characters. Revision History: --*/ // // Include Files. // #define WIN31 #include "windows.h" #include #include "charmap.h" #include "stdlib.h" #include "tchar.h" #ifdef UNICODE #include "wchar.h" #else #include "stdio.h" #endif #include "commctrl.h" #include #pragma warning( disable : 4242) // // Macros. // #define FMagData(psycm) ((psycm)->xpMagCurr != 0) #define abs(x) (((x) >= 0) ? (x) : (-(x))) // // Constant Declarations. // #define STATUSPOINTSIZE 8 // point size of status bar font #define FE_STATUSPOINTSIZE 10 // FE point size of status bar font #define DX_BITMAP 20 // width of TT bitmap #define DY_BITMAP 12 // height of TT bitmap #define BACKGROUND 0x000000FF // bright blue #define BACKGROUNDSEL 0x00FF00FF // bright purple #define BUTTONFACE 0x00C0C0C0 // bright grey #define BUTTONSHADOW 0x00808080 // dark grey #define TOOLBARPOINTSIZE 21 // height of toolbar in points // Font types #define PS_OPENTYPE_FONT 0x0001 #define TT_OPENTYPE_FONT 0x0002 #define TRUETYPE_FONT 0x0004 #define TYPE1_FONT 0x0008 // // Debug Print Code. // #if 0 TCHAR szDOUT[3] = TEXT("A\n"); TCHAR szDbgBuf[256]; #define DOUTL(p) OutputDebugString(TEXT(p)) #define DOUTCHN(ch) if(0){}else {szDOUT[0] = ch; OutputDebugString(szDOUT);} #define DPRINT(p) if(0){}else {wsprintf p; OutputDebugString(szDbgBuf);} #else #define DOUTL(p) #define DOUTCHN(ch) #define DPRINT(p) #endif // // Global Variables. // HANDLE hInst; INT cchSymRow = 32; // number of characters across the character grid INT cchSymCol = 8; // number of rows in the character grid UTCHAR chSymFirst = 32; UTCHAR chSymLast = 255; UTCHAR chRangeFirst = 32; UTCHAR chRangeLast = 255; SYCM sycm; // tons of data need to do char grid painting WORD wCFRichText = 0; // private clipboard format, rich text format HFONT hFontClipboard = NULL; // tells us which font is in the clipboard HANDLE hstrClipboard = NULL; // contains the string which is in the clipboard BOOL fDelClipboardFont = FALSE; // the clipboard font needs to be deleted INT iControl = ID_CHARGRID; // index indicating which control has focus HBITMAP hbmFont = NULL; // TT bitmap drawn before font facenames in combo LONG lEditSel = 0; // contains the selection range of the EC HBRUSH hStaticBrush; // used for static controls during WM_CTLCOLOR // // Currently there is no defined interface for querying what character // ranges a Unicode font supports. For now, this table only has the subsets // that contain characters supported by the Lucida Sans Unicode font // uncommented. When we get an API that allows querying the font driver for // ranges of Unicode characters supported (and whether or not a font is a // Unicode font!) then all entries can be uncommented. // USUBSET aSubsetData[] = { { 0x0020, 0x00ff, IDS_WINDOWS }, { 0x0020, 0x00ff, IDS_LATIN1 }, { 0x0100, 0x017f, IDS_LATINEXA }, { 0x0180, 0x024f, IDS_LATINEXB }, { 0x0250, 0x02af, IDS_IPAEX }, { 0x02b0, 0x02ff, IDS_SPACINGMODIFIERS }, { 0x0300, 0x036f, IDS_COMBININGDIACRITICS }, { 0x0370, 0x03cf, IDS_BASICGREEK }, { 0x03d0, 0x03ff, IDS_GREEKSYMBOLS }, { 0x0400, 0x04ff, IDS_CYRILLIC }, { 0x0530, 0x058f, IDS_ARMENIAN }, { 0x0590, 0x05ff, IDS_HEBREW }, { 0x0600, 0x0652, IDS_BASICARABIC }, { 0x0653, 0x06ff, IDS_ARABICEX }, { 0x0900, 0x097f, IDS_DEVANAGARI }, { 0x0980, 0x09ff, IDS_BENGALI }, { 0x0a00, 0x0a7f, IDS_GURMUKHI }, { 0x0a80, 0x0aff, IDS_GUJARATI }, { 0x0b00, 0x0b7f, IDS_ORIYA }, { 0x0b80, 0x0bff, IDS_TAMIL }, { 0x0c00, 0x0c7f, IDS_TELUGU }, { 0x0c80, 0x0cff, IDS_KANNADA }, { 0x0d00, 0x0d7f, IDS_MALAYALAM }, { 0x0e00, 0x0e7f, IDS_THAI }, { 0x0e80, 0x0eff, IDS_LAO }, { 0x10d0, 0x10ff, IDS_BASICGEORGIAN }, { 0x10a0, 0x10cf, IDS_GEORGIANEX }, { 0x1100, 0x11ff, IDS_HANGULJAMO }, { 0x1e00, 0x1eff, IDS_LATINEXADDITIONAL }, { 0x1f00, 0x1fff, IDS_GREEKEX }, { 0x2000, 0x206f, IDS_GENERALPUNCTUATION }, { 0x2070, 0x209f, IDS_SUPERANDSUBSCRIPTS }, { 0x20a0, 0x20cf, IDS_CURRENCYSYMBOLS }, { 0x20d0, 0x20ff, IDS_COMBININGDIACRITICSFORSYMBOLS }, { 0x2100, 0x214f, IDS_LETTERLIKESYMBOLS }, { 0x2150, 0x218f, IDS_NUMBERFORMS }, { 0x2190, 0x21ff, IDS_ARROWS }, { 0x2200, 0x22ff, IDS_MATHEMATICALOPS }, { 0x2300, 0x23ff, IDS_MISCTECHNICAL }, { 0x2400, 0x243f, IDS_CONTROLPICTURES }, { 0x2440, 0x245f, IDS_OPTICALCHAR }, { 0x2460, 0x24ff, IDS_ENCLOSEDALPHANUM }, { 0x2500, 0x257f, IDS_BOXDRAWING }, { 0x2580, 0x259f, IDS_BLOCKELEMENTS }, { 0x25a0, 0x25ff, IDS_GEOMETRICSHAPES }, { 0x2600, 0x26ff, IDS_MISCDINGBATS }, { 0x2700, 0x27bf, IDS_DINGBATS }, { 0x3000, 0x303f, IDS_CJKSYMBOLSANDPUNC }, { 0x3040, 0x309f, IDS_HIRAGANA }, { 0x30a0, 0x30ff, IDS_KATAKANA }, { 0x3100, 0x312f, IDS_BOPOMOFO }, { 0x3130, 0x318f, IDS_HANGULCOMPATIBILITYJAMO }, { 0x3190, 0x319f, IDS_CJKMISC }, { 0x3200, 0x32ff, IDS_ENCLOSEDCJKLETTERSANDMONTHS }, { 0x3300, 0x33ff, IDS_CJKCOMPATIBILITY }, { 0x4e00, 0x9fff, IDS_CJKUNIFIEDIDEOGRAPHS }, { 0xac00, 0xd7a3, IDS_HANGUL }, { 0xe000, 0xf8ff, IDS_PRIVATEUSEAREA }, { 0xf900, 0xfaff, IDS_CJKCOMPATIBILITYIDEOGRAPHS }, { 0xfb00, 0xfb4f, IDS_ALPAHPRESENTATIONFORMS }, { 0xfb50, 0xfdff, IDS_ARABICPRESENTATIONFORMSA }, { 0xfe30, 0xfe4f, IDS_CJKCOMPFORMS }, { 0xfe50, 0xfe6f, IDS_SMALLFORMVARIANTS }, { 0xfe70, 0xfefe, IDS_ARABICPRESENTATIONFORMSB }, { 0xff00, 0xffef, IDS_HALFANDFULLWIDTHFORMS }, { 0xfff0, 0xfffd, IDS_SPECIALS } }; INT cSubsets = sizeof(aSubsetData) / sizeof(USUBSET); INT iCurSubset = 0; // index of current Unicode subset - default to Latin-1 // // Useful window handles. // HWND hwndDialog; HWND hwndCharGrid; // // Data used to draw the status bar. // RECT rcStatusLine; // bounding rect for status bar RECT rcToolbar[2]; // bounding rects for toolbars INT dyStatus; // height of status bar INT dyToolbar[2]; // height of tool bars INT dxHelpField; // width of help window INT dxKeystrokeField; // width of keystroke window TCHAR szKeystrokeText[MAX_PATH]; // buffer for keystroke text TCHAR szKeystrokeLabel[30]; // buffer for keystroke label TCHAR szSpace[15]; // strings for keystroke description TCHAR szCtrl[15]; TCHAR szCtrlAlt[25]; TCHAR szShiftCtrlAlt[25]; TCHAR szAlt[15]; TCHAR szUnicodeLabel[23]; // buffer for Unicode label INT iKeystrokeTextStart; // place to start appending text to above INT iUnicodeLabelStart; // place to start appending text to above HFONT hfontStatus; // font used for text of status bar //////////////////////////////////////////////////////////////////////////// // // WinMain // // Calls initialization function, processes message loop, cleanup. // //////////////////////////////////////////////////////////////////////////// INT WinMain( HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { MSG msg; if (!InitApplication(hInstance)) { return (FALSE); } // // Perform initialization for this instance. // if (!InitInstance(hInstance, nCmdShow)) { return (FALSE); } while (GetMessage(&msg, NULL, 0, 0)) { // // Filter for possible tabs now to implement context sensitive help. // if (msg.message == WM_KEYDOWN) { if (!UpdateHelpText(&msg, NULL)) { continue; } } // // Main message loop. // if (!IsDialogMessage(hwndDialog, &msg)) { TranslateMessage(&msg); DispatchMessage(&msg); } } // // Free up some stuff. // if (hfontStatus) { DeleteObject(hfontStatus); } if (hbmFont) { DeleteObject(hbmFont); } return (msg.wParam); } //////////////////////////////////////////////////////////////////////////// // // InitApplication // // Initializes window data and registers window class. // //////////////////////////////////////////////////////////////////////////// BOOL InitApplication( HANDLE hInstance) { WNDCLASS wc; // // Register a window class that we will use to draw the character // grid into. // wc.style = CS_DBLCLKS; wc.lpfnWndProc = CharGridWndProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = hInstance; wc.hIcon = NULL; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wc.lpszMenuName = NULL; wc.lpszClassName = TEXT("CharGridWClass"); if (!RegisterClass(&wc)) { return (FALSE); } wc.style = 0; wc.lpfnWndProc = DefDlgProc; wc.cbClsExtra = 0; wc.cbWndExtra = DLGWINDOWEXTRA; wc.hInstance = hInstance; wc.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDIC_CHARMAP)); wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1); wc.lpszMenuName = NULL; wc.lpszClassName = TEXT("MyDlgClass"); if (!RegisterClass(&wc)) { return (FALSE); } return TRUE; } //////////////////////////////////////////////////////////////////////////// // // InitInstance // // Does some initialization and creates main window which is a dialog. // //////////////////////////////////////////////////////////////////////////// BOOL InitInstance( HANDLE hInstance, INT nCmdShow) { INT i; CHARSETINFO csi; DWORD dw = GetACP(); LANGID PrimaryLangId = (PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale()))); BOOL bFE = ((PrimaryLangId == LANG_JAPANESE) || (PrimaryLangId == LANG_KOREAN) || (PrimaryLangId == LANG_CHINESE)); // // Save the instance handle in a global variable. // hInst = hInstance; // // This font will be used to paint the status line. // if (!TranslateCharsetInfo((DWORD*)dw, &csi, TCI_SRCCODEPAGE)) { csi.ciCharset = ANSI_CHARSET; } hfontStatus = CreateFont( -PointsToHeight(bFE ? FE_STATUSPOINTSIZE : STATUSPOINTSIZE), 0, 0, 0, 400, 0, 0, 0, csi.ciCharset, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, VARIABLE_PITCH, TEXT("MS Shell Dlg") ); dyStatus = 2 * PointsToHeight(STATUSPOINTSIZE); dyToolbar[0] = PointsToHeight(TOOLBARPOINTSIZE); dyToolbar[1] = PointsToHeight(TOOLBARPOINTSIZE); // // Load the Unicode subset names before initializing the main window. // for (i = 0; i < cSubsets; i++) { if (!LoadString( hInst, aSubsetData[i].StringResId, (LPTSTR)aSubsetData[i].Name, LF_SUBSETSIZE)) { return (FALSE); } } // // Create a main window for this application instance. // if (!(hwndDialog = CreateDialog( hInstance, TEXT("CharMap"), NULL, CharMapDlgProc ))) { return (FALSE); } // // Initialize some strings used for the Keystroke status bar field. // For international purposes, this string could be length 0. // LoadString( hInst, IDS_KEYSTROKE, (LPTSTR)szKeystrokeLabel, BTOC(sizeof(szKeystrokeLabel)) ); if (!LoadString( hInst, IDS_UNICODELABEL, (LPTSTR)szUnicodeLabel, BTOC(sizeof(szUnicodeLabel)) )) { if (!LoadString( hInst, IDS_SPACE, (LPTSTR)szSpace, BTOC(sizeof(szSpace)) )) { return (FALSE); } } if (!LoadString( hInst, IDS_CTRL, (LPTSTR)szCtrl, BTOC(sizeof(szCtrl)) )) { return (FALSE); } if (!LoadString( hInst, IDS_CTRLALT, (LPTSTR)szCtrlAlt, BTOC(sizeof(szCtrlAlt)) )) { return (FALSE); } if (!LoadString( hInst, IDS_SHIFTCTRLALT, (LPTSTR)szShiftCtrlAlt, BTOC(sizeof(szShiftCtrlAlt)) )) { return (FALSE); } if (!LoadString( hInst, IDS_ALT, (LPTSTR)szAlt, BTOC(sizeof(szAlt)) )) { return (FALSE); } // // Store the index to where we start adding status line text changes. // iKeystrokeTextStart = lstrlen(szKeystrokeLabel); iUnicodeLabelStart = lstrlen(szUnicodeLabel); // // Initialize keystroke text, make the window visible, // update its client area, and return "success". // UpdateKeystrokeText(NULL, sycm.fAnsiFont, sycm.chCurr, FALSE); ShowWindow(hwndDialog, nCmdShow); UpdateWindow(hwndDialog); return (TRUE); } //////////////////////////////////////////////////////////////////////////// // // ConvertANSIFontToUnicode // //////////////////////////////////////////////////////////////////////////// WCHAR ConvertANSIFontToUnicode( HWND hwnd, HFONT hFont, CHAR ch) { WORD cp = CP_ACP; WCHAR wch; HDC hdc; hdc = GetDC(hwnd); if (hdc != NULL) { HFONT hfOld; TEXTMETRIC tm; CHARSETINFO csi; DWORD cs; hfOld = SelectObject(hdc, hFont); if (GetTextMetrics(hdc, &tm)) { cs = MAKELONG(tm.tmCharSet, 0); if (TranslateCharsetInfo((DWORD *)cs, &csi, TCI_SRCCHARSET)) { cp = csi.ciACP; } else { DPRINT(( szDbgBuf, TEXT("CvtAtoW: TranslateCharsetInfo(cset=%d) returned 0! (GetLastErr=%d), using CP_ACP\n"), cs, GetLastError() )); } } SelectObject(hdc, hfOld); ReleaseDC(hwnd, hdc); } if (MultiByteToWideChar(cp, 0, &ch, 1, &wch, 1) != 1) { if (MultiByteToWideChar(CP_ACP, 0, &ch, 1, &wch, 1) != 1) { wch = (WCHAR)(BYTE)ch; } } DPRINT(( szDbgBuf, TEXT("CvtAtoW: 0x%02x '%c' (CP:%d) -> U'%04X'\n"), (DWORD)(BYTE)ch, ch, cp, (DWORD)wch )); return (wch); } //////////////////////////////////////////////////////////////////////////// // // EnumChildProc // // Gets called during init for each child window. // //////////////////////////////////////////////////////////////////////////// BOOL CALLBACK EnumChildProc( HWND hwnd, LPARAM lParam) { LONG st; TCHAR szClass[MAX_PATH]; // // Get control class. // GetClassName(hwnd, szClass, MAX_PATH); if (lstrcmpi(szClass, TEXT("button")) == 0 ) { // // If it is a button, set the ex style to NOTIFYPARENT. // st = GetWindowLong(hwnd, GWL_EXSTYLE); st = st & ~WS_EX_NOPARENTNOTIFY; SetWindowLong(hwnd, GWL_EXSTYLE, st); } return (TRUE); } //////////////////////////////////////////////////////////////////////////// // // CharMapDlgProc // // Processes messages for the main window. This window is a dialog box. // //////////////////////////////////////////////////////////////////////////// INT_PTR APIENTRY CharMapDlgProc( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case ( WM_CTLCOLORSTATIC ) : { POINT point; SetBkColor((HDC)wParam, GetSysColor(COLOR_BTNFACE)); UnrealizeObject(hStaticBrush); point.x = point.y = 0; ClientToScreen(hWnd, &point); return ((INT_PTR)hStaticBrush); break; } case ( WM_INITDIALOG ) : { RECT rectParent, rectTopRightControl, rect; POINT pt; INT iSubset; HWND hwndCMSB; // // Set buttons to send WM_PARENTNOTIFY. // EnumChildWindows(hWnd, EnumChildProc, (LPARAM)NULL ); // // Create the character grid with dimensions which just fit // inside the space allowed in the dialog. When it processes // the WM_CREATE message it will be sized and centered more // accurately. // GetClientRect(hWnd, &rectParent); GetWindowRect(GetDlgItem(hWnd, ID_CLOSE), &rectTopRightControl); ScreenToClient(hWnd, (LPPOINT)&(rectTopRightControl.left)); ScreenToClient(hWnd, (LPPOINT)&(rectTopRightControl.right)); if (!(hwndCharGrid = CreateWindow( TEXT("CharGridWClass"), NULL, WS_CHILD | WS_VISIBLE | WS_TABSTOP, 1, rectParent.top + dyToolbar[0] + dyToolbar[1], rectParent.right - 1, rectParent.bottom - rectParent.top - dyStatus - dyToolbar[0] - dyToolbar[1] - 1, hWnd, (HMENU)ID_CHARGRID, hInst, NULL ))) { DestroyWindow(hWnd); break; } GetWindowRect(hwndCharGrid, &rect); pt.x = rect.right; pt.y = rect.top; ScreenToClient(hWnd, &pt); hwndCMSB = CreateWindowEx( 0L, TEXT("SCROLLBAR"), NULL, WS_CHILD | SBS_VERT | WS_VISIBLE | WS_TABSTOP, pt.x + 1, pt.y + 1, sycm.dxpBox, sycm.dypCM, hWnd, (HMENU)ID_MAPSCROLL, hInst, NULL ); hStaticBrush = CreateSolidBrush(GetSysColor(COLOR_BTNFACE)); // // Initialize the status line data. // dxHelpField = 21 * rectParent.right / 32; dxKeystrokeField = 9 * rectParent.right / 32; rcStatusLine = rectParent; rcStatusLine.top = rcStatusLine.bottom - dyStatus; // // Initialize the toolbars. // rcToolbar[0] = rectParent; rcToolbar[0].bottom = rcToolbar[0].top + dyToolbar[0]; rcToolbar[1] = rcToolbar[0]; rcToolbar[1].top = rcToolbar[0].bottom + GetSystemMetrics(SM_CYBORDER); rcToolbar[1].bottom = rcToolbar[1].top + dyToolbar[1]; // // Disable Copy button. // EnableWindow(GetDlgItem(hWnd, ID_COPY), FALSE); // // Fill "Subset" list box. // for (iSubset = 0; iSubset < cSubsets; iSubset++) { SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_ADDSTRING, 0, (DWORD)aSubsetData[iSubset].Name ); } iCurSubset = SelectInitialSubset(hWnd); // // Fall through to WM_FONTCHANGE... // } case ( WM_FONTCHANGE ) : { HDC hdc = GetDC(hWnd); // // Get the fonts from the system and put them in the font // selection combo box. // if (message == WM_FONTCHANGE) { SaveCurrentFont(hWnd); SendDlgItemMessage(hWnd, ID_FONT, CB_RESETCONTENT, 0, 0L); } EnumFontFamilies(hdc, NULL, (FONTENUMPROC)FontLoadProc, (LPARAM)hWnd); ReleaseDC(hWnd, hdc); // // Setup character dimensions and select this font. // RecalcCharMap( hWnd, &sycm, SelectInitialFont(hWnd), (message == WM_FONTCHANGE) ); SetEditCtlFont(hWnd, ID_STRING, sycm.hFont); if (message == WM_INITDIALOG) { SetFocus(hwndCharGrid); // // Fall through to WM_SYSCOLORCHANGE... // } else { break; } } case ( WM_SYSCOLORCHANGE ) : { if (hbmFont) { DeleteObject(hbmFont); } hbmFont = LoadBitmaps(IDBM_TT); DeleteObject(hStaticBrush); hStaticBrush = CreateSolidBrush(GetSysColor(COLOR_BTNFACE)); break; } case ( WM_PARENTNOTIFY ) : { POINTS points; DWORD dwMsgPos; POINT point; DPRINT(( szDbgBuf, TEXT("WM_PARENTNOTIFY: lParam:0x%08lX, wParam:0x%08lX\n"), (DWORD)lParam, (DWORD)wParam )); // // We process this message to implement the context sensitive // help. Downclicks to controls are found here, the help // message is updated in the status bar. // // The parameters with this message are unreliable! // if (LOWORD(wParam) == WM_LBUTTONDOWN) { dwMsgPos = GetMessagePos(); points = MAKEPOINTS(dwMsgPos); point.x = points.x; point.y = points.y; UpdateHelpText(NULL, WindowFromPoint(point)); } break; } case ( WM_VSCROLL ) : { ProcessScrollMsg(hWnd, LOWORD(wParam), HIWORD(wParam)); break; } case ( WM_PAINT ) : { HBRUSH hBrush; RECT rcTemp, rectNextButton; INT dyBorder, dxBorder; PAINTSTRUCT ps; HDC hdc; // // This code implements painting of the status bar. // hdc = BeginPaint(hWnd, &ps); rcTemp = rcStatusLine; dyBorder = GetSystemMetrics(SM_CYBORDER); dxBorder = GetSystemMetrics(SM_CXBORDER); // // Make the whole thing grey. // if (hBrush = CreateSolidBrush(GetSysColor(COLOR_BTNFACE))) { FillRect(hdc, &rcTemp, hBrush); rcTemp.left = rcToolbar[0].left; rcTemp.top = rcToolbar[0].top; rcTemp.right = rcToolbar[1].right; rcTemp.bottom = rcToolbar[1].bottom; FillRect(hdc, &rcTemp, hBrush); DeleteObject(hBrush); } GetWindowRect(GetDlgItem(hWnd, ID_TOPLEFT), &rectNextButton); ScreenToClient(hWnd, (LPPOINT)&(rectNextButton.left)); ScreenToClient(hWnd, (LPPOINT)&(rectNextButton.right)); // // Solid black line across bottom of toolbar. // if (hBrush = CreateSolidBrush(GetSysColor(COLOR_WINDOWFRAME))) { #ifdef USE_MIRRORING DWORD dwLayout; GetProcessDefaultLayout(&dwLayout); if(dwLayout & LAYOUT_RTL) { // // Interchange the right and left values. // int tmp = rectNextButton.left; rectNextButton.left = rectNextButton.right; rectNextButton.right = tmp; } #endif rcTemp = rcToolbar[0]; rcTemp.top = rcTemp.bottom; rcTemp.bottom += dyBorder; rcTemp.left = rectNextButton.left - 2 - dxBorder; FillRect(hdc, &rcTemp, hBrush); rcTemp = rcToolbar[1]; rcTemp.top = rcTemp.bottom; rcTemp.bottom += dyBorder; FillRect(hdc, &rcTemp, hBrush); // // Vertical line. // rcTemp.top = rcToolbar[0].top; rcTemp.bottom = rcToolbar[1].bottom; rcTemp.left = rectNextButton.left - 2 - dxBorder; rcTemp.right = rectNextButton.left - 2; FillRect(hdc, &rcTemp, hBrush); DeleteObject(hBrush); } if (hBrush = CreateSolidBrush(GetSysColor(COLOR_BTNSHADOW))) { // // Status line top. // rcTemp.left = 8 * dyBorder; rcTemp.right = rcTemp.left + dxHelpField; rcTemp.top = rcStatusLine.top + dyBorder * 2; rcTemp.bottom = rcTemp.top + dyBorder; FillRect(hdc, &rcTemp, hBrush); // // Keystroke line top. // rcTemp.right = rcStatusLine.right - 8 * dyBorder; rcTemp.left = rcTemp.right - dxKeystrokeField; FillRect(hdc, &rcTemp, hBrush); // // Status line left side. // rcTemp = rcStatusLine; rcTemp.left = 8 * dyBorder; rcTemp.right = rcTemp.left + dyBorder; rcTemp.top += dyBorder * 2; rcTemp.bottom -= dyBorder * 2; FillRect(hdc, &rcTemp, hBrush); // // Keystroke line left side. // rcTemp.left = rcStatusLine.right - 9 * dyBorder - dxKeystrokeField; rcTemp.right = rcTemp.left + dyBorder; FillRect(hdc, &rcTemp, hBrush); DeleteObject(hBrush); } if (hBrush = CreateSolidBrush(GetSysColor(COLOR_BTNHIGHLIGHT))) { // // Status line bottom. // rcTemp.left = 8 * dyBorder; rcTemp.right = rcTemp.left + dxHelpField; rcTemp.top = rcStatusLine.bottom - 3 * dyBorder; rcTemp.bottom = rcTemp.top + dyBorder; FillRect(hdc, &rcTemp, hBrush); // // Keystroke line bottom. // rcTemp.right = rcStatusLine.right - 8 * dyBorder; rcTemp.left = rcTemp.right - dxKeystrokeField; FillRect(hdc, &rcTemp, hBrush); // // Status line right side. // rcTemp = rcStatusLine; rcTemp.left = 8 * dyBorder + dxHelpField; rcTemp.right = rcTemp.left + dyBorder; rcTemp.top += dyBorder * 2; rcTemp.bottom -= dyBorder * 2; FillRect(hdc, &rcTemp, hBrush); // // Keystroke line right side. // rcTemp.left = rcStatusLine.right - 8 * dyBorder; rcTemp.right = rcTemp.left + dyBorder; FillRect(hdc, &rcTemp, hBrush); DeleteObject(hBrush); } // // Solid black line across top. // if (hBrush = CreateSolidBrush(GetSysColor(COLOR_WINDOWFRAME))) { rcTemp = rcStatusLine; rcTemp.bottom = rcTemp.top; rcTemp.top -= dyBorder; FillRect(hdc, &rcTemp, hBrush); DeleteObject(hBrush); } PaintStatusLine(hdc, TRUE, TRUE); EndPaint(hWnd, &ps); return (TRUE); } case ( WM_MEASUREITEM ) : { HDC hDC; HFONT hFont; TEXTMETRIC tm; hDC = GetDC(NULL); hFont = (HFONT)SendMessage(hWnd, WM_GETFONT, 0, 0L); if (hFont) { hFont = SelectObject(hDC, hFont); } GetTextMetrics(hDC, &tm); if (hFont) { SelectObject(hDC, hFont); } ReleaseDC(NULL, hDC); ((LPMEASUREITEMSTRUCT)lParam)->itemHeight = max(tm.tmHeight, DY_BITMAP); break; } case ( WM_DRAWITEM ) : { if (((LPDRAWITEMSTRUCT)lParam)->itemID != -1) { DrawFamilyComboItem((LPDRAWITEMSTRUCT)lParam); } break; } case ( WM_ASKCBFORMATNAME ) : { LoadString(hInst, IDS_RTF, (LPTSTR)lParam, wParam); return (TRUE); } case ( WM_PAINTCLIPBOARD ) : { LPPAINTSTRUCT lpPS; HANDLE hFont; LPTSTR lpstrText; if (hstrClipboard) { // // Setup. // lpPS = (LPPAINTSTRUCT)GlobalLock((HANDLE)lParam); lpstrText = (LPTSTR)GlobalLock(hstrClipboard); // // Paint. // hFont = SelectObject(lpPS->hdc, hFontClipboard); TextOut(lpPS->hdc, 0, 0, lpstrText, lstrlen(lpstrText)); SelectObject(lpPS->hdc, hFont); // // Cleanup. // GlobalUnlock(hstrClipboard); GlobalUnlock((HANDLE)lParam); } return (TRUE); } case ( WM_CLOSE ) : { DestroyWindow(hWnd); return (TRUE); } case ( WM_COMMAND ) : { switch (GET_WM_COMMAND_ID(wParam, lParam)) { case ( IDCANCEL ) : case ( ID_CLOSE ) : { DestroyWindow(hWnd); return (TRUE); break; } case ( ID_SELECT ) : { WCHAR wch = sycm.chCurr; if (sycm.fAnsiFont) { wch = ConvertANSIFontToUnicode( hWnd, sycm.hFont, (char)wch ); } SendDlgItemMessage(hWnd, ID_STRING, WM_CHAR, (WPARAM)wch, 0L); break; } case ( ID_COPY ) : { CopyString(hWnd); return (TRUE); break; } case ( ID_FONT ) : { if (HIWORD(wParam) == CBN_SELCHANGE) { RecalcCharMap( hWnd, &sycm, (INT)SendDlgItemMessage( hWnd, ID_FONT, CB_GETCURSEL, 0, 0L ), TRUE ); SetEditCtlFont(hWnd, ID_STRING, sycm.hFont); } else if (HIWORD(wParam) == CBN_SETFOCUS) { // // Necessary if hotkey is used to get to the CB. // UpdateHelpText(NULL, (HWND)lParam); } return (TRUE); break; } case ( ID_UNICODESUBSET ) : { if (HIWORD(wParam) == CBN_SELCHANGE) { INT iSubset; INT cEntries; iSubset = (INT)SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_GETCURSEL, 0, 0 ); SubSetChanged( hWnd, iSubset, aSubsetData[iSubset].BeginRange, aSubsetData[iSubset].EndRange ); cEntries = (INT)SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_GETCOUNT, 0, 0 ) - 1; EnableWindow( GetDlgItem(hWnd, ID_PREVSUBSET), iSubset > 0 ); EnableWindow( GetDlgItem(hWnd, ID_NEXTSUBSET), iSubset < cEntries ); } else if (HIWORD(wParam) == CBN_SETFOCUS) { // // Necessary if hotkey is used to get to the CB. // UpdateHelpText(NULL, (HWND)lParam); } return (0L); break; } case ( ID_NEXTSUBSET ) : { INT iCurSelection, iNumEntries; iCurSelection = (INT)SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_GETCURSEL, 0, 0 ); if (iCurSelection == CB_ERR) { return (0L); } iNumEntries = (INT)SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_GETCOUNT, 0, 0 ); if (iNumEntries == CB_ERR) { return (0L); } if (iCurSelection++ < (iNumEntries - 1)) { if (iCurSelection == 1) { // // Enable Previous button. // EnableWindow(GetDlgItem(hWnd, ID_PREVSUBSET), TRUE); } SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_SETCURSEL, iCurSelection, 0 ); SubSetChanged( hWnd, iCurSelection, aSubsetData[iCurSelection].BeginRange, aSubsetData[iCurSelection].EndRange ); if (iCurSelection == (iNumEntries - 1)) { HWND hwndButton; EnableWindow(GetDlgItem(hWnd, ID_NEXTSUBSET), FALSE); // // Only reset the button style and focus if // the "Next" button currently has it. // if (iControl == ID_NEXTSUBSET) { SendDlgItemMessage( hwndDialog, ID_PREVSUBSET, BM_SETSTYLE, BS_DEFPUSHBUTTON, 1 ); SendDlgItemMessage( hwndDialog, ID_NEXTSUBSET, BM_SETSTYLE, BS_PUSHBUTTON, 1 ); hwndButton = GetDlgItem(hWnd, ID_PREVSUBSET); SetFocus(hwndButton); UpdateHelpText(NULL, hwndButton); } } } return (0L); break; } case ( ID_PREVSUBSET ) : { INT iCurSelection; iCurSelection = (INT)SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_GETCURSEL, 0, 0 ); if (iCurSelection == CB_ERR) { return (0L); } if (iCurSelection > 0) { iCurSelection--; if (iCurSelection == (cSubsets - 2)) { // // Enable Next button. // EnableWindow(GetDlgItem(hWnd, ID_NEXTSUBSET), TRUE); } SendDlgItemMessage( hWnd, ID_UNICODESUBSET, CB_SETCURSEL, iCurSelection, 0 ); SubSetChanged( hWnd, iCurSelection, aSubsetData[iCurSelection].BeginRange, aSubsetData[iCurSelection].EndRange ); if (iCurSelection == 0) { HWND hwndButton; EnableWindow(GetDlgItem(hWnd, ID_PREVSUBSET), FALSE); // // Only reset the button style and focus if // the "Previous" button currently has it. // if (iControl == ID_PREVSUBSET) { SendDlgItemMessage( hwndDialog, ID_NEXTSUBSET, BM_SETSTYLE, BS_DEFPUSHBUTTON, 1 ); SendDlgItemMessage( hwndDialog, ID_PREVSUBSET, BM_SETSTYLE, BS_PUSHBUTTON, 1 ); hwndButton = GetDlgItem(hWnd, ID_NEXTSUBSET); SetFocus(hwndButton); UpdateHelpText(NULL, hwndButton); } } } return (0L); break; } case ( ID_STRING ) : { if (HIWORD(wParam) == EN_SETFOCUS) { // // Necessary if hotkey is used to get to the EC. // UpdateHelpText(NULL, (HWND)lParam); } else if (HIWORD(wParam) == EN_CHANGE) { // // Disable Copy button if there are no chars in EC. // INT iLength; iLength = GetWindowTextLength((HWND)lParam); EnableWindow(GetDlgItem(hWnd, ID_COPY), (BOOL)iLength); } break; } case ( ID_HELP ) : { DoHelp(hWnd, TRUE); break; } } break; } case ( WM_DESTROY ) : { SaveCurrentFont(hWnd); SaveCurrentSubset(hWnd); DoHelp(hWnd, FALSE); DeleteObject(hStaticBrush); PostQuitMessage(0); break; } case ( WM_ACTIVATEAPP ) : { if (wParam) { SendDlgItemMessage( hWnd, ID_STRING, EM_SETSEL, LOWORD(lEditSel), HIWORD(lEditSel) ); } else { lEditSel = SendDlgItemMessage(hWnd, ID_STRING, EM_GETSEL, 0, 0L); SendDlgItemMessage(hWnd, ID_STRING, EM_SETSEL, 0, 0L); } break; } } return (0L); } //////////////////////////////////////////////////////////////////////////// // // CharGridWndProc // // Processes messages for the character grid window. // //////////////////////////////////////////////////////////////////////////// LRESULT APIENTRY CharGridWndProc( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case ( WM_CREATE ) : { RECT rect; HDC hdcScrn; POINT point1, point2; // // Setup global. // hwndCharGrid = hWnd; GetClientRect(hWnd, &rect); // // Calculate metrics for the character grid and the // magnify window. // sycm.dxpBox = (rect.right - 1) / (cchSymRow + 2); sycm.dypBox = (rect.bottom - 2) / (cchSymCol + 1); sycm.dxpCM = sycm.dxpBox * cchSymRow + 1; sycm.dypCM = sycm.dypBox * cchSymCol + 1; // space inside for border if ((PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale())) == LANG_CHINESE)) { sycm.dxpMag = sycm.dxpBox * 3 + 5; } else { sycm.dxpMag = sycm.dxpBox * 2 + 4; // twice the size + 2 bit border } sycm.dypMag = sycm.dypBox * 2 + 4; sycm.chCurr = chSymFirst; sycm.hFontMag = NULL; sycm.hFont = NULL; sycm.hdcMag = NULL; sycm.hbmMag = NULL; sycm.ypDest = 0; sycm.fFocusState = sycm.fMouseDn = sycm.fCursorOff = FALSE; // // Size the window precisely so the grid fits and is centered. // MoveWindow( hWnd, (rect.right - sycm.dxpCM + 1) / 2, (rect.bottom - sycm.dypCM + 1) / 2 + ((LPCREATESTRUCT)lParam)->y - 2, sycm.dxpCM + 2, sycm.dypCM + 2, FALSE ); // // Figure out what the offsets are between the dialog // and the character grid window. // point1.x = point1.y = point2.x = point2.y = 0; ClientToScreen(hWnd, &point1); ClientToScreen(((LPCREATESTRUCT)lParam)->hwndParent, &point2); #ifdef USE_MIRRORING sycm.xpCM = (abs(point1.x - point2.x)) - (sycm.dxpMag - sycm.dxpBox) / 2; #else sycm.xpCM = (point1.x - point2.x) - (sycm.dxpMag - sycm.dxpBox) / 2; #endif sycm.ypCM = (point1.y - point2.y) - (sycm.dypMag - sycm.dypBox) / 2; // // Create dc and bitmap for the magnify window. // if ((hdcScrn = GetWindowDC(hWnd)) != NULL) { if ((sycm.hdcMag = CreateCompatibleDC(hdcScrn)) != NULL) { SetTextColor( sycm.hdcMag, GetSysColor(COLOR_WINDOWTEXT) ); SetBkColor( sycm.hdcMag, GetSysColor(COLOR_WINDOW) ); SetBkMode(sycm.hdcMag, OPAQUE); if ((sycm.hbmMag = CreateCompatibleBitmap( hdcScrn, sycm.dxpMag, sycm.dypMag * 2 )) == NULL) { DeleteObject(sycm.hdcMag); } else { SelectObject(sycm.hdcMag, sycm.hbmMag); } } ReleaseDC(hWnd, hdcScrn); } break; } case ( WM_DESTROY ) : { if (sycm.fMouseDn) { ExitMagnify(hWnd, &sycm); } if (fDelClipboardFont) { DeleteObject(hFontClipboard); } if (sycm.hFont != NULL) { DeleteObject(sycm.hFont); } if (sycm.hFontMag != NULL) { DeleteObject(sycm.hFontMag); } if (sycm.hdcMag != NULL) { DeleteDC(sycm.hdcMag); } if (sycm.hbmMag != NULL) { DeleteObject(sycm.hbmMag); } break; } case ( WM_SETFOCUS ) : case ( WM_KILLFOCUS ) : { RestoreSymMag(&sycm); DrawSymChOutlineHwnd( &sycm, hWnd, sycm.chCurr, TRUE, message == WM_SETFOCUS ); break; } case ( WM_LBUTTONDOWN ) : { RECT rect; DOUTL("WM_LBUTTONDOWN: In\n"); // // Don't draw anything if there's an update region pending. // if (GetUpdateRect(hWnd, (LPRECT)&rect, FALSE) != 0) { DOUTL("WM_LBUTTONDOWN: No upd rect\n"); break; } SetFocus(hWnd); SetCapture(hWnd); sycm.fMouseDn = TRUE; if (!FMagData(&sycm)) { DOUTL("WM_LBUTTONDOWN: Drawing sym outline\n"); DrawSymChOutlineHwnd(&sycm, hWnd, sycm.chCurr, FALSE, FALSE); } // // Fall through to WM_MOUSEMOVE... // } case ( WM_MOUSEMOVE ) : { DOUTL("WM_MOUSEMOVE: In\n"); if (sycm.fMouseDn) { POINT pt; UINT chMouseSymbol; DOUTL("WM_MOUSEMOVE: mouse is down\n"); pt.x = LOWORD(lParam); pt.y = HIWORD(lParam); ClientToScreen(hWnd, (LPPOINT)&pt); if (WindowFromPoint(pt) == hWnd) { ScreenToClient(hWnd, (LPPOINT)&pt); // // Convert back to a 'points'-like thing. // lParam = MAKELONG((WORD)pt.x, (WORD)pt.y); chMouseSymbol = (UINT)ChFromSymLParam(&sycm, lParam); if (chMouseSymbol > (UINT)chSymLast) { // // We're outside of current character range (but // still within the grid). Restore cursor and // leave magnified character. // if (sycm.fCursorOff) { sycm.fCursorOff = FALSE; ShowCursor(TRUE); } } else { // // We're in the grid and within the range of currently // displayed characters, display magnified character. // DOUTL("WM_MOUSEMOVE: in grid and subrange\n"); if (!sycm.fCursorOff) { sycm.fCursorOff = TRUE; ShowCursor(FALSE); } DOUTL("WM_MOUSEMOVE: movsymsel "); DOUTCHN( (UTCHAR)chMouseSymbol ); MoveSymbolSel(&sycm, (UTCHAR)chMouseSymbol); } } else { // // Left grid, leave magnified character and restore // cursor. // if (sycm.fCursorOff) { sycm.fCursorOff = FALSE; ShowCursor(TRUE); } } } DOUTL("WM_MOUSEMOVE: Leaving\n"); break; } case ( WM_CANCELMODE ) : case ( WM_LBUTTONUP ) : { if (sycm.fMouseDn) { ExitMagnify(hWnd, &sycm); } break; } case ( WM_LBUTTONDBLCLK ) : { WCHAR wch = sycm.chCurr; // // Send this character to the entry field. // if (sycm.fAnsiFont) { wch = ConvertANSIFontToUnicode(hWnd, sycm.hFont, (char)wch); } SendDlgItemMessage(hwndDialog, ID_STRING, WM_CHAR, (WPARAM)wch, 0L); break; } case ( WM_GETDLGCODE ) : { // // Necessary to obtain arrow and tab messages. // return (DLGC_WANTARROWS | DLGC_WANTCHARS); break; } case ( WM_KEYDOWN ) : { UTCHAR chNew = sycm.chCurr; INT cchMoved; if (sycm.fMouseDn) { break; } switch (wParam) { case ( VK_LEFT ) : { if (--chNew < chSymFirst) { return (0L); } break; } case ( VK_UP ) : { if ((chNew -= cchSymRow) < chSymFirst) { if (!ScrollMap(GetParent(hWnd), -cchSymRow, TRUE)) { return (0L); } RestoreSymMag(&sycm); } break; } case ( VK_RIGHT ) : { if (++chNew > chSymLast) { return (0L); } break; } case ( VK_DOWN ) : { if ((chNew += cchSymRow) > chSymLast) { if (!ScrollMap(GetParent(hWnd), cchSymRow, TRUE)) { return (0L); } RestoreSymMag(&sycm); } break; } case ( VK_NEXT ) : { if ((cchMoved = ScrollMapPage(GetParent(hWnd), FALSE, TRUE)) == 0) { return (0L); } // // We scrolled the map! Bump the char so it is // still in the window. // RestoreSymMag(&sycm); chNew += cchMoved; break; } case ( VK_PRIOR ) : { if ((cchMoved = ScrollMapPage( GetParent(hWnd), TRUE, TRUE )) == 0) { return (0L); } // // We scrolled the map! Bump the char so it is // still in the window. // RestoreSymMag(&sycm); chNew += cchMoved; break; } default : { return (0L); } } if (!FMagData(&sycm)) { DrawSymChOutlineHwnd(&sycm, hWnd, sycm.chCurr, FALSE, FALSE); } MoveSymbolSel(&sycm, (UTCHAR)chNew); break; } case ( WM_CHAR ) : { WCHAR wch = (WCHAR)wParam; char ch; if (sycm.fMouseDn) { break; } if (sycm.fAnsiFont) { if (WideCharToMultiByte( CP_ACP, 0, &wch, 1, &ch, 1, NULL, NULL ) != 1) { break; } wch = (WCHAR)(BYTE)ch; } if ((wch >= chSymFirst) && (wch <= chSymLast)) { if (!FMagData(&sycm)) { DrawSymChOutlineHwnd(&sycm, hWnd, sycm.chCurr, FALSE, FALSE); } MoveSymbolSel(&sycm, (UTCHAR)wch); SendDlgItemMessage(hwndDialog, ID_STRING, WM_CHAR, wParam, 0L); } break; } case ( WM_PAINT ) : { HDC hdc; PAINTSTRUCT ps; DOUTL("WM_PAINT: In\n"); hdc = BeginPaint(hWnd, &ps); DOUTL("WM_PAINT: drawing map\n"); DrawSymbolMap(&sycm, hdc); EndPaint(hWnd, &ps); DOUTL("WM_PAINT: Leaving\n"); return (TRUE); } default : { return (DefWindowProc(hWnd, message, wParam, lParam)); } } return (0L); } //////////////////////////////////////////////////////////////////////////// // // ProcessScrollMsg // //////////////////////////////////////////////////////////////////////////// VOID ProcessScrollMsg( HWND hwndDlg, int nCode, int nPos) { UTCHAR chNew = sycm.chCurr; HWND hwndGrid = GetDlgItem(hwndDlg, ID_CHARGRID); int cchScroll; switch( nCode ) { case ( SB_LINEUP ) : { cchScroll = -cchSymRow; break; } case ( SB_LINEDOWN ) : { cchScroll = cchSymRow; break; } case ( SB_PAGEUP ) : { cchScroll = (int)TRUE; break; } case ( SB_PAGEDOWN ) : { cchScroll = (int)FALSE; break; } case ( SB_THUMBTRACK ) : case ( SB_THUMBPOSITION ) : { cchScroll = (nPos * cchSymRow + chRangeFirst) - chSymFirst; break; } default : { return; } } if (nCode == SB_PAGEUP || nCode == SB_PAGEDOWN) { if (!ScrollMapPage(hwndDlg, (BOOL)cchScroll, FALSE)) { return; } // // ScrollMapPage will do the right thing to sycm.chCurr. // chNew = sycm.chCurr; } else { if (cchScroll == 0 || !ScrollMap(hwndDlg, cchScroll, FALSE)) { return; } // // Keep the current symbol inside the window. // while (chNew > chSymLast) { chNew -= cchSymRow; } while (chNew < chSymFirst) { chNew += cchSymRow; } } #if 0 if (!FMagData(&sycm)) { DrawSymChOutlineHwnd(&sycm, hwndGrid, sycm.chCurr, FALSE, FALSE); } MoveSymbolSel(&sycm, (UTCHAR)chNew); #else sycm.chCurr = chNew; InvalidateRect(hwndGrid, NULL, TRUE); #endif } //////////////////////////////////////////////////////////////////////////// // // ScrollMapPage // // Scrolls the map up or down by a page. See ScrollMap(). // //////////////////////////////////////////////////////////////////////////// INT ScrollMapPage( HWND hwndDlg, BOOL fUp, BOOL fRePaint) { INT cchScroll = cchFullMap; if (fUp) { cchScroll = -cchScroll; } if ((chSymFirst + cchScroll) < chRangeFirst) { cchScroll = (chRangeFirst - chSymFirst); } else if ((chSymLast + cchScroll) > chRangeLast) { cchScroll = (chRangeLast - chSymLast); } return (ScrollMap(hwndDlg, cchScroll, fRePaint) ? cchScroll : 0); } //////////////////////////////////////////////////////////////////////////// // // ScrollMap // // Scrolls the map up or down if there are too many chars to fit in the // chargrid. // //////////////////////////////////////////////////////////////////////////// BOOL ScrollMap( HWND hwndDlg, INT cchScroll, BOOL fRePaint) { HWND hwndSB, hwndCharGrid; INT chFirst = chSymFirst + cchScroll; INT chLast = chSymLast + cchScroll; HDC hdc; if ((chFirst < chRangeFirst) || (chLast > chRangeLast)) { return (FALSE); } hwndCharGrid = GetDlgItem(hwndDlg, ID_CHARGRID); hwndSB = GetDlgItem(hwndDlg, ID_MAPSCROLL); SetScrollPos(hwndSB, SB_CTL, (chFirst - chRangeFirst) / cchSymRow, TRUE); UpdateSymbolRange(hwndDlg, chFirst, chLast); if ((hwndDlg != NULL) && ((hdc = GetDC(hwndDlg)) != NULL)) { LPINT lpdxp; HFONT hFont; UINT ch; hFont = SelectObject(hdc, sycm.hFont); lpdxp = (LPINT)sycm.rgdxp; if (sycm.fAnsiFont) { GetCharWidth32A(hdc, chSymFirst, chSymLast, lpdxp); } else { GetCharWidth32(hdc, chSymFirst, chSymLast, lpdxp); } SelectObject(hdc, hFont); for (ch = (UINT) chSymFirst; ch <= (UINT) chSymLast; ch++, lpdxp++) { *lpdxp = (sycm.dxpBox - *lpdxp) / 2 - 1; } ReleaseDC(hwndDlg, hdc); } if (fRePaint) { InvalidateRect(hwndCharGrid, NULL, TRUE); } return (TRUE); } //////////////////////////////////////////////////////////////////////////// // // ChFromSymLParam // // Determines the character to select from the mouse position (lParam). // //////////////////////////////////////////////////////////////////////////// INT ChFromSymLParam( PSYCM psycm, LPARAM lParam) { return (min( cchSymRow - 1, max(0, ((INT)LOWORD(lParam) - 1) / psycm->dxpBox) ) + min( cchSymCol - 1, max(0, ((INT)HIWORD(lParam) - 1) / psycm->dypBox) ) * cchSymRow + chSymFirst); } //////////////////////////////////////////////////////////////////////////// // // DrawSymChOutlineHwnd // // Gets a DC for hwnd, calls DrawSymChOutline. // //////////////////////////////////////////////////////////////////////////// VOID DrawSymChOutlineHwnd( PSYCM psycm, HWND hwnd, UTCHAR ch, BOOL fVisible, BOOL fFocus) { HDC hdc = GetDC(hwnd); DrawSymChOutline(psycm, hdc, ch, fVisible, fFocus); ReleaseDC(hwnd, hdc); } //////////////////////////////////////////////////////////////////////////// // // RecalcCharMap // // Recalculates fixed character map data (font info, sizes, etc.). // //////////////////////////////////////////////////////////////////////////// VOID RecalcCharMap( HWND hwndDlg, PSYCM psycm, INT iCombo, BOOL fRedraw) { HDC hdc; TEXTMETRIC tm; UINT ch; LPINT lpdxp; HFONT hFont; LOGFONT LogFont; ITEMDATA ItemData; LONG iCurSel; // // Get rid of the old font handles. // if (hFontClipboard && (hFontClipboard == psycm->hFont)) { fDelClipboardFont = TRUE; } if (psycm->hFont && (hFontClipboard != psycm->hFont)) { DeleteObject(psycm->hFont); } if (psycm->hFontMag) { DeleteObject(psycm->hFontMag); } hdc = GetDC(hwndCharGrid); // // Set up the LogFont structure. // Make sure it fits in the grid. // if (PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale())) == LANG_CHINESE) { LogFont.lfHeight = 16; } else { LogFont.lfHeight = psycm->dypBox - 3; // Allow for whitespace. } // // Set these to zero. // LogFont.lfWidth = LogFont.lfEscapement = LogFont.lfOrientation = LogFont.lfWeight = 0; LogFont.lfItalic = LogFont.lfUnderline = LogFont.lfStrikeOut = LogFont.lfOutPrecision = LogFont.lfClipPrecision = LogFont.lfQuality = LogFont.lfPitchAndFamily = 0; // // Let the facename and size define the font. // // LogFont.lfCharSet = DEFAULT_CHARSET; // Work around the GDI bug that assumes the font's default charset // is always the system default locale. // *(DWORD *)&ItemData = SendDlgItemMessage( hwndDlg, ID_FONT, CB_GETITEMDATA, iCombo, 0L ); LogFont.lfCharSet = ItemData.CharSet; // // Get the facename from the combo box. // SendDlgItemMessage( hwndDlg, ID_FONT, CB_GETLBTEXT, iCombo, (LONG)(LPTSTR)LogFont.lfFaceName ); // // Enable Block listbox and set defaults appropriately. // EnableWindow(GetDlgItem(hwndDlg, ID_UNICODESUBSET), TRUE); iCurSel = SendDlgItemMessage( hwndDlg, ID_UNICODESUBSET, CB_GETCURSEL, 0, 0L ); UpdateSymbolSelection( hwndDlg, aSubsetData[iCurSel].BeginRange, aSubsetData[iCurSel].EndRange ); // // Enable Previous button if not on first subset. // if (iCurSel > 0) { EnableWindow(GetDlgItem(hwndDlg, ID_PREVSUBSET), TRUE); } else { EnableWindow(GetDlgItem(hwndDlg, ID_PREVSUBSET), FALSE); } // // Enable Next button if not on last subset. // if (iCurSel < (cSubsets - 1)) { EnableWindow(GetDlgItem(hwndDlg, ID_NEXTSUBSET), TRUE); } else { EnableWindow(GetDlgItem(hwndDlg, ID_NEXTSUBSET), FALSE); } // // The first sub sel is the ANSI code page. // psycm->fAnsiFont = (iCurSel == 0); // // Create the font. // psycm->hFont = CreateFontIndirect(&LogFont); hFont = SelectObject(hdc, psycm->hFont); // // Create the magnify font. // LogFont.lfHeight = psycm->dypMag - 5; // Allow for whitespace. psycm->hFontMag = CreateFontIndirect(&LogFont); // // Calculate new values and place in window data structure. // GetTextMetrics(hdc, &tm); psycm->xpCh = 2; psycm->ypCh = (4 + psycm->dypBox - tm.tmHeight) / 2; lpdxp = (LPINT)psycm->rgdxp; if (psycm->fAnsiFont) { GetCharWidth32A(hdc, chSymFirst, chSymLast, lpdxp); } else { GetCharWidth32(hdc, chSymFirst, chSymLast, lpdxp); } SelectObject(hdc, hFont); for (ch = (UINT) chSymFirst; ch <= (UINT) chSymLast; ch++, lpdxp++) { *lpdxp = (psycm->dxpBox - *lpdxp) / 2 - 1; } ReleaseDC(hwndCharGrid, hdc); psycm->xpMagCurr = 0; // No magnification data if (fRedraw) { InvalidateRect(hwndCharGrid, NULL, TRUE); } } //////////////////////////////////////////////////////////////////////////// // // DrawSymbolMap // // Draws all of the pieces of the symbol character map. // //////////////////////////////////////////////////////////////////////////// VOID DrawSymbolMap( PSYCM psycm, HDC hdc) { BOOL fFocus; DrawSymbolGrid(psycm, hdc); DrawSymbolChars(psycm, hdc); // // We need to force the focus rect to paint if we have the focus // since the old focus rect has been drawn over already. // if (fFocus = psycm->fFocusState) { psycm->fFocusState = FALSE; } DrawSymChOutline(psycm, hdc, psycm->chCurr, TRUE, fFocus); } //////////////////////////////////////////////////////////////////////////// // // DrawSymbolGrid // // Draws the symbol character map grid. // //////////////////////////////////////////////////////////////////////////// VOID DrawSymbolGrid( PSYCM psycm, HDC hdc) { INT cli; // count of lines INT xp, yp; INT dxpBox = psycm->dxpBox; INT dypBox = psycm->dypBox; HPEN hpenOld; hpenOld = SelectObject(hdc, CreatePen( PS_SOLID, 1, GetSysColor(COLOR_WINDOWFRAME) )); // // Draw horizontal lines. // xp = psycm->dxpCM + 1; yp = 1; cli = cchSymCol+1; while (cli--) { MoveToEx(hdc, 1, yp, NULL); LineTo(hdc, xp, yp); yp += dypBox; } // // Draw vertical lines. // yp = psycm->dypCM; xp = 1; cli = cchSymRow+1; while (cli--) { MoveToEx(hdc, xp, 1, NULL); LineTo(hdc, xp, yp); xp += dxpBox; } DeleteObject(SelectObject(hdc, hpenOld)); } //////////////////////////////////////////////////////////////////////////// // // DrawSymbolChars // // Draws the symbol character map. // //////////////////////////////////////////////////////////////////////////// VOID DrawSymbolChars( PSYCM psycm, HDC hdc) { INT dxpBox = psycm->dxpBox; INT dypBox = psycm->dypBox; INT cch; INT x, y; INT yp; TCHAR ch; HFONT hFontOld; RECT rect; LPRECT lprect = (LPRECT)▭ LPINT lpdxp; // // Setup the font and colors. // hFontOld = (HFONT)SelectObject(hdc, psycm->hFont); SetTextColor(hdc, GetSysColor(COLOR_WINDOWTEXT)); SetBkColor(hdc, GetSysColor(COLOR_WINDOW)); SetBkMode(hdc, OPAQUE); // // Draw characters. // cch = 1; ch = chSymFirst; lpdxp = (LPINT)psycm->rgdxp; rect.top = 2; yp = psycm->ypCh; rect.bottom = rect.top + dypBox - 1; for (y = 0; y++ < cchSymCol;) { rect.left = psycm->xpCh; rect.right = rect.left + dxpBox - 1; for (x = 0; (x++ < cchSymRow) && (ch <= chSymLast);) { if (psycm->fAnsiFont) { ExtTextOutA( hdc, rect.left + (*lpdxp++), yp, ETO_OPAQUE | ETO_CLIPPED, lprect, &(CHAR)ch, 1, NULL ); } else { ExtTextOutW( hdc, rect.left + (*lpdxp++), yp, ETO_OPAQUE | ETO_CLIPPED, lprect, &ch, 1, NULL ); } ch++; rect.left += dxpBox; rect.right += dxpBox; } yp += dypBox; rect.top += dypBox; rect.bottom += dypBox; } SelectObject(hdc, hFontOld); } //////////////////////////////////////////////////////////////////////////// // // DrawSymChOutline // // Draws an outline around the symbol in the character map. If fVisible, // then it draws the outline, otherwise it erases it. // //////////////////////////////////////////////////////////////////////////// VOID DrawSymChOutline( PSYCM psycm, HDC hdc, UTCHAR ch, BOOL fVisible, BOOL fFocus) { HBRUSH hbrOld; RECT rc; INT dxpBox = psycm->dxpBox; INT dypBox = psycm->dypBox; hbrOld = SelectObject( hdc, CreateSolidBrush(GetSysColor( fVisible ? COLOR_WINDOWFRAME : COLOR_WINDOW )) ); ch -= chSymFirst; rc.left = (ch % cchSymRow) * dxpBox + 2; rc.right = rc.left + dxpBox - 1; rc.top = (ch / cchSymRow) * dypBox + 2; rc.bottom = rc.top + dypBox - 1; // // Draw selection rectangle. // PatBlt(hdc, rc.left, rc.top - 2, dxpBox - 1, 1, PATCOPY); PatBlt(hdc, rc.left, rc.bottom + 1, dxpBox - 1, 1, PATCOPY); PatBlt(hdc, rc.left - 2, rc.top, 1, dypBox - 1, PATCOPY); PatBlt(hdc, rc.right + 1, rc.top, 1, dypBox - 1, PATCOPY); DeleteObject(SelectObject(hdc, GetStockObject(NULL_BRUSH))); // // Deal with the focus rectangle. // if (fFocus != psycm->fFocusState) { DrawFocusRect(hdc, &rc); psycm->fFocusState = fFocus; } SelectObject(hdc, hbrOld); } //////////////////////////////////////////////////////////////////////////// // // MoveSymbolSel // // Changes the current symbol selection. Handles drawing of magnified // characters. // //////////////////////////////////////////////////////////////////////////// VOID MoveSymbolSel( PSYCM psycm, UTCHAR chNew) { HDC hdc; HDC hdcMag = psycm->hdcMag; RECT rc; HFONT hFontOld; HFONT hFontMag; // old font in memory dc HPEN hpenOld; UTCHAR chNorm = chNew - chSymFirst + 32; INT dxpMag = psycm->dxpMag; // for quick reference INT dypMag = psycm->dypMag; INT ypMemSrc = psycm->ypDest; INT ypMemDest = ypMemSrc ^ dypMag; INT xpCurr = psycm->xpMagCurr; INT ypCurr = psycm->ypMagCurr; INT xpNew = psycm->xpCM + (psycm->dxpBox * (chNorm % cchSymRow)); INT ypNew = psycm->ypCM + (psycm->dypBox * ((chNorm / cchSymRow) - 1)); INT dxpCh; // width of extra character space (used to center char in box) INT dypCh; SIZE sz; DOUTL("MoveSymbolSel: In\n"); if (((chNew == (UTCHAR)psycm->chCurr) && FMagData(psycm))) { DOUTL("MoveSymbolSel: ch == cur && fMag... exiting\n"); return; } // // Don't draw a magnified character if the char grid has an update // region or is not visible. // if (!IsWindowVisible(hwndCharGrid) || GetUpdateRect(hwndCharGrid, &rc, FALSE)) { DOUTL("MoveSymbolSel: not vis or upd rect... exiting\n"); return; } hdc = GetDC(hwndDialog); // // Setup the magnified font character. // hFontMag = SelectObject(hdcMag, psycm->hFontMag); if (psycm->fAnsiFont) { char chANSINew = (char)chNew; GetTextExtentPointA(hdcMag, &chANSINew, 1, &sz); } else { GetTextExtentPointW(hdcMag, &chNew, 1, &sz); } if (PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale())) == LANG_CHINESE) { dxpCh = (dxpMag - (INT)sz.cx) / 2 - 2; dypCh = (dypMag - (INT)sz.cy) / 2 - 2; } else { dxpCh = (dxpMag - (INT)sz.cx) / 2 - 1; dypCh = (dypMag - (INT)sz.cy) / 2 - 1; } hpenOld = SelectObject(hdc, CreatePen( PS_SOLID, 1, GetSysColor(COLOR_WINDOWFRAME) )); hFontOld = SelectObject(hdc, psycm->hFontMag); // // Copy screen data to offscreen bitmap. // BitBlt(hdcMag, 0, ypMemDest, dxpMag, dypMag, hdc, xpNew, ypNew, SRCCOPY); // // Setup DC. // SetTextColor(hdc, GetSysColor(COLOR_WINDOWTEXT)); SetBkColor(hdc, GetSysColor(COLOR_WINDOW)); SetBkMode(hdc, OPAQUE); if (FMagData(psycm)) { INT xpT = xpNew - xpCurr; // point of overlap in offscreen data INT ypT = ypNew - ypCurr; INT dxpT = dxpMag - abs(xpT); // size of overlap INT dypT = dypMag - abs(ypT); DOUTL("MoveSymbolSel: FMagData\n"); if ((dxpT > 0) && (dypT > 0)) { INT xpTmax, ypTmax; // max(0, xpT); INT xpTmin, ypTmin; // min(0, xpT); INT xpTnmin, ypTnmin; // min(0, -xpT); DOUTL("MoveSymbolSel: dxpT > 0 && dypT > 0\n"); if (xpT < 0) { xpTnmin = - (xpTmin = xpT); xpTmax = 0; } else { xpTmax = xpT; xpTnmin = xpTmin = 0; } if (ypT < 0) { ypTnmin = - (ypTmin = ypT); ypTmax = 0; } else { ypTmax = ypT; ypTnmin = ypTmin = 0; } rc.left = xpTmax; rc.right = xpTmin + dxpMag; rc.top = ypTmax + ypMemSrc; rc.bottom= ypTmin + dypMag + ypMemSrc; // // Copy overlapping offscreen data. // BitBlt( hdcMag, xpTnmin, ypTnmin + ypMemDest, dxpT, dypT, hdcMag, xpTmax, ypTmax + ypMemSrc, SRCCOPY ); // // Print part of char over old screen data. // if (psycm->fAnsiFont) { ExtTextOutA( hdcMag, xpT + dxpCh, ypT + dypCh + ypMemSrc, ETO_OPAQUE | ETO_CLIPPED, (LPRECT)&rc, &(CHAR)chNew, 1, NULL ); } else { ExtTextOutW( hdcMag, xpT + dxpCh, ypT + dypCh + ypMemSrc, ETO_OPAQUE | ETO_CLIPPED, (LPRECT)&rc, &chNew, 1, NULL ); } } // // Restore old screen data. // BitBlt(hdc, xpCurr, ypCurr, dxpMag, dypMag, hdcMag, 0, ypMemSrc, SRCCOPY); } rc.right = (psycm->xpMagCurr = rc.left = xpNew) + dxpMag - 2; rc.bottom = (psycm->ypMagCurr = rc.top = ypNew) + dypMag - 2; // // The rectangle. // MoveToEx(hdc, rc.left, rc.top, NULL); LineTo(hdc, rc.left, rc.bottom - 1); LineTo(hdc, rc.right - 1, rc.bottom - 1); LineTo(hdc, rc.right - 1, rc.top); LineTo(hdc, rc.left, rc.top); // // The shadow. // MoveToEx(hdc, rc.right, rc.top + 1, NULL); LineTo(hdc, rc.right, rc.bottom); LineTo(hdc, rc.left, rc.bottom); MoveToEx(hdc, rc.right + 1, rc.top + 2, NULL); LineTo(hdc, rc.right + 1, rc.bottom + 1); LineTo(hdc, rc.left + 1, rc.bottom + 1); rc.left++; rc.top++; rc.right--; rc.bottom--; // // Draw magnified character on screen. // if (psycm->fAnsiFont) { ExtTextOutA( hdc, xpNew + dxpCh, ypNew + dypCh, ETO_OPAQUE | ETO_CLIPPED, (LPRECT)&rc, &(CHAR)chNew, 1, NULL ); } else { ExtTextOutW( hdc, xpNew + dxpCh, ypNew + dypCh, ETO_OPAQUE | ETO_CLIPPED, (LPRECT)&rc, &chNew, 1, NULL ); } psycm->ypDest = ypMemDest; DeleteObject(SelectObject(hdc, hpenOld)); SelectObject(hdc, hFontOld); SelectObject(hdcMag, hFontMag); UpdateKeystrokeText(hdc, psycm->fAnsiFont, chNew, TRUE); ReleaseDC(hwndDialog, hdc); psycm->chCurr = chNew; DOUTL("MoveSymbolSel: Leaving\n"); } //////////////////////////////////////////////////////////////////////////// // // RestoreSymMag // // Restores the screen data under the magnifier. // //////////////////////////////////////////////////////////////////////////// VOID RestoreSymMag( PSYCM psycm) { if (FMagData(psycm)) { HDC hdc = GetDC(hwndDialog); BitBlt( hdc, psycm->xpMagCurr, psycm->ypMagCurr, psycm->dxpMag, psycm->dypMag, psycm->hdcMag, 0, psycm->ypDest, SRCCOPY ); ReleaseDC(hwndDialog, hdc); psycm->xpMagCurr = 0; // flag - no data offscreen (see FMagData) } } //////////////////////////////////////////////////////////////////////////// // // FontLoadProc // // Used by EnumFonts to load our combo box with all the fonts installed // in the system. // //////////////////////////////////////////////////////////////////////////// INT APIENTRY FontLoadProc( LPLOGFONT lpLogFont, NEWTEXTMETRICEX* lpTextMetric, DWORD nFontType, LPARAM lpData) { INT iPos; TCHAR szFace[LF_FACESIZE]; // // Check for duplicates. // iPos = (INT)SendDlgItemMessage( (HWND)lpData, ID_FONT, CB_FINDSTRING, (WPARAM)-1, (DWORD)&lpLogFont->lfFaceName ); if (iPos == CB_ERR) { NotInListYet: // // Doesn't exist, insert the facename into the combo box. // iPos = (INT)SendDlgItemMessage( (HWND)lpData, ID_FONT, CB_ADDSTRING, 0, (DWORD)&lpLogFont->lfFaceName ); } else { // // Make sure it is not just a substring (want a full match). // SendDlgItemMessage( (HWND)lpData, ID_FONT, CB_GETLBTEXT, iPos, (LONG)(LPTSTR)szFace ); if (lstrcmpi(szFace, lpLogFont->lfFaceName)) { goto NotInListYet; } // // Already exists, blow out now if this is not a true type font. // if (!(nFontType & TRUETYPE_FONTTYPE)) { return (1); } } // // Store the pertinant font information in the combo item data. // if ((iPos != CB_ERR) && (iPos != CB_ERRSPACE)) { ITEMDATA ItemData; DWORD ntmFlags = lpTextMetric->ntmTm.ntmFlags; SHORT sFontType = 0; if (ntmFlags & NTM_PS_OPENTYPE) { sFontType = PS_OPENTYPE_FONT; } else if (ntmFlags & NTM_TYPE1) { sFontType = TYPE1_FONT; } else if (nFontType & TRUETYPE_FONTTYPE) { if (ntmFlags & NTM_TT_OPENTYPE) sFontType = TT_OPENTYPE_FONT; else sFontType = TRUETYPE_FONT; } ItemData.FontType = sFontType; ItemData.CharSet = lpLogFont->lfCharSet; ItemData.PitchAndFamily = lpLogFont->lfPitchAndFamily; SendDlgItemMessage( (HWND)lpData, ID_FONT, CB_SETITEMDATA, iPos, *(DWORD *)&ItemData ); } // // Continue enumeration. // return (1); } //////////////////////////////////////////////////////////////////////////// // // GetEditText // // Returns HANDLE containing the text in the edit control. // // NOTE: Caller is responsible for freeing this handle! // //////////////////////////////////////////////////////////////////////////// HANDLE GetEditText( HWND hwndDlg) { INT cchText; HWND hwndEditCtl; HANDLE hmem; LPTSTR lpstrText; DWORD dwSel; hwndEditCtl = GetDlgItem(hwndDlg, ID_STRING); cchText = GetWindowTextLength(hwndEditCtl); hmem = GlobalAlloc(0, CTOB((cchText + 1))); lpstrText = (LPTSTR)GlobalLock(hmem); cchText = GetWindowText(hwndEditCtl, lpstrText, cchText+1); dwSel = SendMessage(hwndEditCtl, EM_GETSEL, 0, 0L); if (LOWORD(dwSel) != HIWORD(dwSel)) { // // If there is a selection, then only get the selected text. // *(lpstrText + HIWORD(dwSel)) = TEXT('\0'); lstrcpy(lpstrText, lpstrText + LOWORD(dwSel)); } GlobalUnlock(hmem); if (cchText == 0) { hmem = GlobalFree(hmem); } return (hmem); } //////////////////////////////////////////////////////////////////////////// // // CopyString // // Implements the copy function. // //////////////////////////////////////////////////////////////////////////// VOID CopyString( HWND hwndDlg) { HANDLE hmem; LPTSTR lpstrText; if (hmem = GetEditText(hwndDlg)) { lpstrText = (LPTSTR)GlobalLock(hmem); // // Copying string to clipboard. // if (OpenClipboard(hwndDlg)) { EmptyClipboard(); SendRTFToClip(hwndDlg, lpstrText); #ifdef UNICODE SetClipboardData(CF_UNICODETEXT, hmem); #else SetClipboardData(CF_TEXT, hmem); #endif CloseClipboard(); } else { // // If we couldn't open the clipboard, then we need to free memory. // GlobalUnlock(hmem); GlobalFree(hmem); } } } //////////////////////////////////////////////////////////////////////////// // // SendRTFToClip // // Puts the string in the clipboard using Rich Text Format. This assumes // that the clipboard has already been opened. // //////////////////////////////////////////////////////////////////////////// VOID SendRTFToClip( HWND hwndDlg, LPTSTR lpstrText) { INT iCurrFont; ITEMDATA ItemData; TCHAR szFaceName[LF_FACESIZE]; HANDLE hmemRTF, hmemClip; LPTSTR lpstrClipString; TCHAR achHeaderTmpl[] = TEXT("{\\rtf1\\ansi\\ansicpg%d {\\fonttbl{\\f0\\"); TCHAR achHeader[sizeof(achHeaderTmpl) / sizeof(TCHAR) + 20]; TCHAR achMiddle[] = TEXT(";}}\\sectd\\pard\\plain\\f0 "); INT cchUC; #ifndef UNICODE_RTF LPWSTR pszRTFW; #endif #define MAXLENGTHFONTFAMILY 8 #define ALITTLEEXTRA 10 // covers extra characters + length of font size iCurrFont = (INT)SendDlgItemMessage(hwndDlg, ID_FONT, CB_GETCURSEL, 0, 0L); // // Get the item data - contains fonttype, charset, and pitchandfamily. // *(DWORD *)&ItemData = SendDlgItemMessage( hwndDlg, ID_FONT, CB_GETITEMDATA, iCurrFont, 0L ); // // Get the facename from the combo box. // SendDlgItemMessage( hwndDlg, ID_FONT, CB_GETLBTEXT, iCurrFont, (LPARAM)(LPTSTR)szFaceName ); wsprintf(achHeader, achHeaderTmpl, (INT)(SHORT)GetACP()); // // 16 times in case they're all > 7 bits (each chr -> \uc1\uddddddd\'xx) // and room for the second byte of DBCS. // hmemRTF = GlobalAlloc( 0, CTOB(lstrlen((LPTSTR)achHeader) + MAXLENGTHFONTFAMILY + lstrlen(szFaceName) + lstrlen((LPTSTR)achMiddle) + 2 * 16 * lstrlen(lpstrText) + ALITTLEEXTRA) ); if (hmemRTF == NULL) { return; } // // Allocate memory for local storage of clipboard string for owner draw. // if (hmemClip = GlobalAlloc(0, CTOB(lstrlen(lpstrText) + 1))) { // // Get rid of old ones. // if (hstrClipboard) { GlobalFree(hstrClipboard); } if (fDelClipboardFont) { fDelClipboardFont = FALSE; DeleteObject(hFontClipboard); } // // Save this stuff away for owner drawing in a clipboard viewer. // hFontClipboard = sycm.hFont; hstrClipboard = hmemClip; lstrcpy(GlobalLock(hstrClipboard), lpstrText); GlobalUnlock(hstrClipboard); } else { GlobalFree(hmemRTF); return; } lpstrClipString = GlobalLock(hmemRTF); #ifndef UNICODE_RTF pszRTFW = lpstrClipString; #endif lstrcpy(lpstrClipString, achHeader); if (ItemData.CharSet == SYMBOL_CHARSET) { lstrcat(lpstrClipString, (LPTSTR)TEXT("ftech ")); } else { // // Top four bits specify family. // switch (ItemData.PitchAndFamily & 0xf0) { case ( FF_DECORATIVE ) : { lstrcat(lpstrClipString, (LPTSTR)TEXT("fdecor ")); break; } case ( FF_MODERN ) : { lstrcat(lpstrClipString, (LPTSTR)TEXT("fmodern ")); break; } case ( FF_ROMAN ) : { lstrcat(lpstrClipString, (LPTSTR)TEXT("froman ")); break; } case ( FF_SCRIPT ) : { lstrcat(lpstrClipString, (LPTSTR)TEXT("fscript ")); break; } case ( FF_SWISS ) : { lstrcat(lpstrClipString, (LPTSTR)TEXT("fswiss ")); break; } default : { break; } } } lstrcat(lpstrClipString, szFaceName); lstrcat(lpstrClipString, (LPTSTR)achMiddle); // // We need to do the text character by character, making sure // that we output a special sequence \'hh for characters bigger // than 7 bits long! // lpstrClipString = (LPTSTR)(lpstrClipString + lstrlen(lpstrClipString)); cchUC = 0; while (*lpstrText) { if ((UTCHAR)*lpstrText < 128) { if (*lpstrText == TEXT('\\') || *lpstrText == TEXT('{') || *lpstrText == TEXT('}')) { // // Need to preface these symbols with a '\' since they are // special control characters for RTF. // *lpstrClipString++ = TEXT('\\'); } *lpstrClipString++ = *lpstrText++; } else { unsigned char achTmp[2]; unsigned char *pTmp = achTmp; int cch; cch = WideCharToMultiByte( CP_ACP, 0, lpstrText, 1, pTmp, 2, NULL, NULL ); // // Put in a \uc# to tell Unicode reader how many bytes to skip // and the \uN code to indicate the real unicode value. // if (cch != cchUC ) { cchUC = cch; lpstrClipString += wsprintf( lpstrClipString, TEXT("\\uc%d"), (INT)(SHORT)cchUC ); } lpstrClipString += wsprintf( lpstrClipString, TEXT("\\u%d"), (INT)(SHORT)*lpstrText ); // // Now put the \'xx string in to indicate the actual character. // lpstrText++; while (cch--) { *lpstrClipString++ = TEXT('\\'); *lpstrClipString++ = TEXT('\''); wsprintf(achMiddle, TEXT("%x"), (INT)*pTmp++); *lpstrClipString++ = achMiddle[0]; *lpstrClipString++ = achMiddle[1]; } } } *lpstrClipString++ = TEXT('}'); *lpstrClipString++ = TEXT('\0'); if (!wCFRichText) { TCHAR szRTF[80]; LoadString(hInst, IDS_RTF, szRTF, BTOC(sizeof(szRTF)) - 1); wCFRichText = RegisterClipboardFormat(szRTF); } #ifndef UNICODE_RTF { // // RTF is only defined for ANSI, not for Unicode, therefore // we need to convert the buffer before we put it on the // clipboard. Eventually, we should add autoconversion code // to USER to handle this for us. // int cch; HANDLE hmemRTFA; LPSTR pszRTFA; cch = WideCharToMultiByte( CP_ACP, 0, pszRTFW, lpstrClipString - pszRTFW, NULL, 0, NULL, NULL ); if (cch != 0 && (hmemRTFA = GlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE,cch)) != NULL) { pszRTFA = GlobalLock(hmemRTFA); WideCharToMultiByte( CP_ACP, 0, pszRTFW, lpstrClipString - pszRTFW, pszRTFA, cch, NULL, NULL ); GlobalUnlock(hmemRTFA); GlobalUnlock(hmemRTF); GlobalFree(hmemRTF); hmemRTF = hmemRTFA; } } #endif // // Put RTF and OwnerDisplay formats in the clipboard. // SetClipboardData(wCFRichText, hmemRTF); SetClipboardData(CF_OWNERDISPLAY, NULL); } //////////////////////////////////////////////////////////////////////////// // // PointsToHeight // // Calculates the height in pixels of the specified point size for the // current display. // //////////////////////////////////////////////////////////////////////////// INT PointsToHeight( INT iPoints) { HDC hdc; INT iHeight; hdc = GetDC(HWND_DESKTOP); iHeight = MulDiv(iPoints, GetDeviceCaps(hdc, LOGPIXELSY), 72); ReleaseDC(HWND_DESKTOP, hdc); return (iHeight); } //////////////////////////////////////////////////////////////////////////// // // UpdateKeystrokeText // // Calculates and updates the text string displayed in the Keystroke // field of the status bar. It repaints the status field if fRedraw is // TRUE. // //////////////////////////////////////////////////////////////////////////// VOID UpdateKeystrokeText( HDC hdc, BOOL fANSI, UTCHAR chNew, BOOL fRedraw) { TCHAR szUnshifted[CCH_KEYNAME]; INT vkRes; LONG lParam; if (!fANSI) { lstrcpy(szKeystrokeText, szUnicodeLabel); wsprintf( (LPTSTR)(szKeystrokeText + iUnicodeLabelStart), TEXT("%04x"), chNew ); } else { lstrcpy(szKeystrokeText, szKeystrokeLabel); vkRes = VkKeyScan(chNew); // // Map the virtual key code into an unshifted character value. // lParam = MapVirtualKey(LOBYTE(vkRes), 0) << 16; GetKeyNameText(lParam, szUnshifted, CCH_KEYNAME - 1); switch (HIBYTE(vkRes)) { case ( 0 ) : // unshifted char case ( 1 ) : // character is shifted, just display the shifted char { if (chNew != TEXT(' ')) { szKeystrokeText[iKeystrokeTextStart] = chNew; szKeystrokeText[iKeystrokeTextStart + 1] = TEXT('\0'); } else { lstrcat(szKeystrokeText, szUnshifted); } break; } case ( 2 ) : // character is control character { lstrcpy((LPTSTR)(szKeystrokeText + iKeystrokeTextStart), szCtrl); lstrcat(szKeystrokeText, (LPTSTR)szUnshifted); break; } case ( 6 ) : // character is CONTROL+ALT { lstrcpy((LPTSTR)(szKeystrokeText + iKeystrokeTextStart), szCtrlAlt); lstrcat(szKeystrokeText, (LPTSTR)szUnshifted); break; } case ( 7 ) : // character is SHIFT+CONTROL+ALT { lstrcpy((LPTSTR)(szKeystrokeText + iKeystrokeTextStart), szShiftCtrlAlt); lstrcat(szKeystrokeText, (LPTSTR)szUnshifted); break; } default : // Character created via Alt + Numpad { lstrcpy((LPTSTR)(szKeystrokeText + iKeystrokeTextStart), szAlt); wsprintf( (LPTSTR)(szKeystrokeText + lstrlen(szKeystrokeText)), TEXT("%d"), chNew ); break; } } } if (fRedraw) { PaintStatusLine(hdc, FALSE, TRUE); } } //////////////////////////////////////////////////////////////////////////// // // UpdateHelpText // // Calculates if the Help string needs to be updated, and does so if // necessary. // // If hwndCtrl is not NULL, then it specifies the window handle of the // control gaining focus, and lpmsg is ignored. // // If hwndCtrl is NULL, then lpmsg must point to a valid message structure. // If it is a tab character, then we calculate what the next control is // that will receive the focus. // //////////////////////////////////////////////////////////////////////////// BOOL UpdateHelpText( LPMSG lpmsg, HWND hwndCtrl) { HDC hdc; BOOL fPaintStatus = FALSE; BOOL fRet = TRUE; DPRINT((szDbgBuf, TEXT("UpdHlpTxt: lpmsg:0x%08lX, hwnd:0x%08lX\n"), (DWORD)lpmsg, (DWORD)hwndCtrl)); if (hwndCtrl != NULL) { fPaintStatus = TRUE; iControl = GetDlgCtrlID(hwndCtrl); } else if (lpmsg->message == WM_KEYDOWN) { if (lpmsg->wParam == VK_TAB) { fPaintStatus = TRUE; hwndCtrl = GetNextDlgTabItem( hwndDialog, GetDlgItem(hwndDialog, iControl), (BOOL)(GetKeyState(VK_SHIFT) & 0x8000) ); iControl = GetDlgCtrlID(hwndCtrl); if (iControl == ID_STRING) { // // Do this ourselves, otherwise default action will select // the whole edit control. // SetFocus(hwndCtrl); fRet = FALSE; } if (iControl == ID_CHARGRID) { // // Set the default button back to "Select". The default // might have changed to the "Next" or "Previous" button. // SendMessage(hwndDialog, DM_SETDEFID, ID_SELECT, 0); } } else if (lpmsg->wParam == VK_F1) { PostMessage(hwndDialog, WM_COMMAND, ID_HELP, 0L); } } if (fPaintStatus) { hdc = GetDC(hwndDialog); PaintStatusLine(hdc, TRUE, FALSE); ReleaseDC(hwndDialog, hdc); } return (fRet); } //////////////////////////////////////////////////////////////////////////// // // SubSetChanged // // Sets the ANSI bit if appropriate and then calls UpdateSymbolSelection // and then repaints the window. // // Repaints Keystroke field if HWND != NULL. // Sets sycm->fAnsiFont if 'Windows Chars' is the subset. // Redraws the char grid. // //////////////////////////////////////////////////////////////////////////// VOID SubSetChanged( HWND hwnd, INT iSubSet, INT ichFirst, INT ichLast) { HDC hdc; BOOL fANSI = (iSubSet == 0); if (fANSI != sycm.fAnsiFont) { sycm.fAnsiFont = fANSI; } UpdateSymbolSelection(hwnd, ichFirst, ichLast); if ((hwnd != NULL) && ((hdc = GetDC(hwnd)) != NULL)) { LPINT lpdxp; HFONT hFont; UINT ch; hFont = SelectObject(hdc, sycm.hFont); lpdxp = (LPINT)sycm.rgdxp; if (iSubSet == 0) { GetCharWidth32A(hdc, chSymFirst, chSymLast, lpdxp); } else { GetCharWidth32(hdc, chSymFirst, chSymLast, lpdxp); } SelectObject(hdc, hFont); for (ch = (UINT) chSymFirst; ch <= (UINT) chSymLast; ch++, lpdxp++) { *lpdxp = (sycm.dxpBox - *lpdxp) / 2 - 1; } ReleaseDC(hwnd, hdc); } InvalidateRect(hwndCharGrid, NULL, TRUE); } //////////////////////////////////////////////////////////////////////////// // // UpdateSymbolSelection // // Updates the values of the following global values: // chRangeFirst // chRangeLast // Subsets in the Unicode character set have different numbers of // characters. We have to do some bounds checking in order to set an // appropriate sycm.chCurr value. The "Keystroke" status field is // updated. // // Repaints Keystroke field if HWND != NULL. // //////////////////////////////////////////////////////////////////////////// VOID UpdateSymbolSelection( HWND hwnd, INT FirstChar, INT LastChar) { int iCmd = SW_HIDE; HWND hwndSB; UINT chFirst, chLast; chRangeFirst = FirstChar; chRangeLast = LastChar; chFirst = chRangeFirst; chLast = chFirst + cchFullMap - 1; chLast = min(chLast, chRangeLast); hwndSB = GetDlgItem(hwnd, ID_MAPSCROLL); if (chLast != chRangeLast) { int i; iCmd = SW_SHOW; SetScrollPos(hwndSB, SB_CTL, 0, FALSE); i = (chRangeLast - chRangeFirst + 1) - cchFullMap; if (i < 0) { i = 1; } else { i = i / cchSymRow; } SetScrollRange(hwndSB, SB_CTL, 0, i, FALSE); InvalidateRect(hwndSB, NULL, FALSE); } ShowWindow(hwndSB, iCmd); UpdateSymbolRange(hwnd, chFirst, chLast); } //////////////////////////////////////////////////////////////////////////// // // UpdateSymbolRange // // Updates the values of the following global values: // chSymFirst // chSymLast // sycm.chCurr // Subsets in the Unicode character set have different numbers of // characters. We have to do some bounds checking in order to set an // appropriate sycm.chCurr value. The "Keystroke" status field is // updated. // // Repaints Keystroke field if HWND != NULL. // //////////////////////////////////////////////////////////////////////////// VOID UpdateSymbolRange( HWND hwnd, INT FirstChar, INT LastChar) { UTCHAR chSymOffset; chSymOffset = sycm.chCurr - chSymFirst; chSymFirst = FirstChar; chSymLast = LastChar; sycm.chCurr = chSymOffset + chSymFirst; if (sycm.chCurr > chSymLast) { sycm.chCurr = chSymFirst; } if (hwnd != NULL) { HDC hdc; hdc = GetDC(hwnd); UpdateKeystrokeText(hdc, sycm.fAnsiFont, sycm.chCurr, TRUE); ReleaseDC(hwnd, hdc); } else { UpdateKeystrokeText(NULL, sycm.fAnsiFont, sycm.chCurr, FALSE); } } //////////////////////////////////////////////////////////////////////////// // // PaintStatusLine // // Paints the Help and Keystroke fields in the status bar. // // Repaints Help field if fHelp == TRUE. // Repaints Keystroke field if fKeystroke == TRUE. // //////////////////////////////////////////////////////////////////////////// VOID PaintStatusLine( HDC hdc, BOOL fHelp, BOOL fKeystroke) { HFONT hfontOld = NULL; RECT rect; INT dyBorder; TCHAR szHelpText[100]; dyBorder = GetSystemMetrics(SM_CYBORDER); if (hfontStatus) { hfontOld = SelectObject(hdc, hfontStatus); } // // Set the text and background colors. // SetTextColor(hdc, GetSysColor(COLOR_BTNTEXT)); SetBkColor(hdc, GetSysColor(COLOR_BTNFACE)); if (fHelp) { // // Now the help text, with a gray background. // rect.top = rcStatusLine.top + 3 * dyBorder; rect.bottom = rcStatusLine.bottom - 3 * dyBorder; rect.left = 9 * dyBorder; rect.right = rect.left + dxHelpField - 2 * dyBorder; LoadString(hInst, iControl, szHelpText, BTOC(sizeof(szHelpText)) - 1); ExtTextOut( hdc, rect.left + dyBorder * 2, rect.top, ETO_OPAQUE | ETO_CLIPPED, &rect, szHelpText, lstrlen(szHelpText), NULL ); } if (fKeystroke) { // // Now the keystroke text, with a gray background. // rect.top = rcStatusLine.top + 3 * dyBorder; rect.bottom = rcStatusLine.bottom - 3 * dyBorder; rect.right = rcStatusLine.right - 9 * dyBorder; rect.left = rect.right - dxKeystrokeField + 2 * dyBorder; ExtTextOut( hdc, rect.left + dyBorder * 2, rect.top, ETO_OPAQUE | ETO_CLIPPED, &rect, szKeystrokeText, lstrlen(szKeystrokeText), NULL ); } if (hfontOld) { SelectObject(hdc, hfontOld); } } //////////////////////////////////////////////////////////////////////////// // // DrawFamilyComboItem // // Paints the font facenames and TT bitmap in the font combo box. // //////////////////////////////////////////////////////////////////////////// BOOL DrawFamilyComboItem( LPDRAWITEMSTRUCT lpdis) { HDC hDC, hdcMem; DWORD rgbBack, rgbText; TCHAR szFace[LF_FACESIZE]; HBITMAP hOld; INT dy; SHORT sFontType; hDC = lpdis->hDC; if (lpdis->itemState & ODS_SELECTED) { rgbBack = SetBkColor(hDC, GetSysColor(COLOR_HIGHLIGHT)); rgbText = SetTextColor(hDC, GetSysColor(COLOR_HIGHLIGHTTEXT)); } else { rgbBack = SetBkColor(hDC, GetSysColor(COLOR_WINDOW)); rgbText = SetTextColor(hDC, GetSysColor(COLOR_WINDOWTEXT)); } SendMessage( lpdis->hwndItem, CB_GETLBTEXT, lpdis->itemID, (LONG)(LPTSTR)szFace ); ExtTextOut( hDC, lpdis->rcItem.left + DX_BITMAP, lpdis->rcItem.top, ETO_OPAQUE | ETO_CLIPPED, &lpdis->rcItem, szFace, lstrlen(szFace), NULL ); hdcMem = CreateCompatibleDC(hDC); if (hdcMem) { if (hbmFont) { hOld = SelectObject(hdcMem, hbmFont); sFontType = ((ITEMDATA FAR *)&(lpdis->itemData))->FontType; if (sFontType) { int xSrc; dy = ((lpdis->rcItem.bottom - lpdis->rcItem.top) - DY_BITMAP) / 2; if (sFontType & TRUETYPE_FONT) xSrc = 0; else if (sFontType & TT_OPENTYPE_FONT) xSrc = 2; else if(sFontType & PS_OPENTYPE_FONT) xSrc = 3; else if (sFontType & TYPE1_FONT) xSrc = 4; BitBlt( hDC, lpdis->rcItem.left, lpdis->rcItem.top + dy, DX_BITMAP, DY_BITMAP, hdcMem, xSrc * DX_BITMAP, lpdis->itemState & ODS_SELECTED ? DY_BITMAP : 0, SRCCOPY ); } SelectObject(hdcMem, hOld); } DeleteDC(hdcMem); } SetTextColor(hDC, rgbText); SetBkColor(hDC, rgbBack); return (TRUE); } //////////////////////////////////////////////////////////////////////////// // // LoadBitmaps // // Loads DIB bitmaps and "fixes up" their color tables so that we get the // desired result for the device we are on. // // This routine requires: // - the DIB is a 16 color DIB authored with the standard windows colors // - bright blue (00 00 FF) is converted to the background color // - light grey (C0 C0 C0) is replaced with the button face color // - dark grey (80 80 80) is replaced with the button shadow color // // This means you can't have any of these colors in your bitmap. // //////////////////////////////////////////////////////////////////////////// HBITMAP LoadBitmaps( INT id) { HDC hdc; HANDLE h, hRes; DWORD *p; LPBYTE lpBits; LPBITMAPINFOHEADER lpBitmapInfo; INT numcolors; DWORD rgbSelected, rgbUnselected; HBITMAP hbm; rgbSelected = GetSysColor(COLOR_HIGHLIGHT); // // Flip the colors. // rgbSelected = RGB( GetBValue(rgbSelected), GetGValue(rgbSelected), GetRValue(rgbSelected) ); rgbUnselected = GetSysColor(COLOR_WINDOW); // // Flip the colors. // rgbUnselected = RGB( GetBValue(rgbUnselected), GetGValue(rgbUnselected), GetRValue(rgbUnselected) ); h = FindResource(hInst, MAKEINTRESOURCE(id), RT_BITMAP); hRes = LoadResource(hInst, h); // // Lock the bitmap and get a pointer to the color table. // lpBitmapInfo = (LPBITMAPINFOHEADER)LockResource(hRes); if (!lpBitmapInfo) { return (FALSE); } p = (DWORD *)((LPSTR)(lpBitmapInfo) + lpBitmapInfo->biSize); // // Search for the Solid Blue entry and replace it with the current // background RGB. // numcolors = 16; while (numcolors-- > 0) { if (*p == BACKGROUND) { *p = rgbUnselected; } else if (*p == BACKGROUNDSEL) { *p = rgbSelected; } p++; } UnlockResource(hRes); // // Now create the DIB. // lpBitmapInfo = (LPBITMAPINFOHEADER)LockResource(hRes); // // First skip over the header structure. // lpBits = (LPBYTE)(lpBitmapInfo + 1); // // Skip the color table entries, if any. // lpBits += (1 << (lpBitmapInfo->biBitCount)) * sizeof(RGBQUAD); // // Create a color bitmap compatible with the display device. // hdc = GetDC(NULL); hbm = CreateDIBitmap( hdc, lpBitmapInfo, (DWORD)CBM_INIT, lpBits, (LPBITMAPINFO)lpBitmapInfo, DIB_RGB_COLORS ); ReleaseDC(NULL, hdc); GlobalUnlock(hRes); FreeResource(hRes); return (hbm); } //////////////////////////////////////////////////////////////////////////// // // DoHelp // // Invokes help if fInvokeHelp is true, or dismisses help if fInvokeHelp // is FALSE. // //////////////////////////////////////////////////////////////////////////// VOID DoHelp( HWND hWnd, BOOL fInvokeHelp) { TCHAR szHelp[80]; if (LoadString(hInst, IDS_HELP, szHelp, BTOC(sizeof(szHelp)) - 1)) { if (fInvokeHelp) { // APPCOMPAT: an error in HtmlHelp prevents the unicode version from working // This is a HACK to get around the problem. Remove this hack when the problem is fixed. HtmlHelpA(GetDesktopWindow(), "charmap.chm", HH_DISPLAY_TOPIC, 0L); } } } //////////////////////////////////////////////////////////////////////////// // // SaveFont // // Saves the current font facename in win.ini, so that it can be selected // the next time charmap comes up. // //////////////////////////////////////////////////////////////////////////// VOID SaveCurrentFont( HWND hWndDlg) { TCHAR szFaceName[LF_FACESIZE] = TEXT(""); SendDlgItemMessage( hWndDlg, ID_FONT, CB_GETLBTEXT, (WORD)SendDlgItemMessage( hWndDlg, ID_FONT, CB_GETCURSEL, 0, 0L ), (LONG)(LPTSTR)szFaceName ); WriteProfileString(TEXT("MSCharMap"), TEXT("Font"), (LPTSTR)szFaceName); } //////////////////////////////////////////////////////////////////////////// // // SelectInitialFont // // Selects the initial font by getting a saved facename from win.ini and // selecting it in the combo box. // // Returns index to font selected. // //////////////////////////////////////////////////////////////////////////// INT SelectInitialFont( HWND hWndDlg) { TCHAR szFaceName[LF_FACESIZE] = TEXT(""); INT iIndex; if ((GetProfileString( TEXT("MSCharMap"), TEXT("Font"), NULL, (LPTSTR)szFaceName, BTOC(sizeof(szFaceName)) ) == 0) || ((iIndex = (INT)SendDlgItemMessage( hWndDlg, ID_FONT, CB_SELECTSTRING, (WPARAM)-1, (LONG)(LPTSTR)szFaceName )) == CB_ERR)) { // // If there was no profile or the selection failed then try selecting // the symbol font, if that fails then select the first one. // if ((iIndex = (INT)SendDlgItemMessage( hWndDlg, ID_FONT, CB_SELECTSTRING, (WPARAM)-1, (LONG)(LPTSTR)TEXT("Symbol") )) == CB_ERR) { SendDlgItemMessage(hWndDlg, ID_FONT, CB_SETCURSEL, iIndex = 0, 0L); } } return (iIndex); } //////////////////////////////////////////////////////////////////////////// // // SaveCurrentSubset // // Saves the current subset name in win.ini, so that it can be selected // the next time charmap comes up. // //////////////////////////////////////////////////////////////////////////// VOID SaveCurrentSubset( HWND hWndDlg) { TCHAR szSubsetName[LF_SUBSETSIZE] = TEXT(""); SendDlgItemMessage( hWndDlg, ID_UNICODESUBSET, CB_GETLBTEXT, (WORD)SendDlgItemMessage( hWndDlg, ID_UNICODESUBSET, CB_GETCURSEL, 0, 0L ), (LONG)(LPTSTR)szSubsetName ); WriteProfileString(TEXT("MSCharMap"), TEXT("Block"), (LPTSTR)szSubsetName); } //////////////////////////////////////////////////////////////////////////// // // SelectInitialSubset // // Selects the initial Unicode subset by getting a saved block name from // win.ini. // // Returns index to subset selected. // //////////////////////////////////////////////////////////////////////////// INT SelectInitialSubset( HWND hWndDlg) { TCHAR szSubsetName[LF_SUBSETSIZE] = TEXT(""); INT iIndex; if ((GetProfileString( TEXT("MSCharMap"), TEXT("Block"), NULL, (LPTSTR)szSubsetName, BTOC(sizeof(szSubsetName)) ) == 0) || ((iIndex = (INT)SendDlgItemMessage( hWndDlg, ID_UNICODESUBSET, CB_SELECTSTRING, (WPARAM)-1, (LONG)(LPTSTR)szSubsetName )) == CB_ERR)) { // // If there was no profile or the selection failed then try selecting // the Basic Latin block, if that fails then select the first one. // if ((iIndex = (INT)SendDlgItemMessage( hWndDlg, ID_UNICODESUBSET, CB_SELECTSTRING, (WPARAM)-1, (LONG)(LPTSTR)TEXT("Basic Latin") )) == CB_ERR) { SendDlgItemMessage( hWndDlg, ID_UNICODESUBSET, CB_SETCURSEL, iIndex = 0, 0L ); } } chSymFirst = aSubsetData[iIndex].BeginRange; chSymLast = aSubsetData[iIndex].EndRange; sycm.chCurr = chSymFirst; return (iIndex); } //////////////////////////////////////////////////////////////////////////// // // ExitMagnify // // Releases mouse capture, exits magnify mode, and restores the cursor. // //////////////////////////////////////////////////////////////////////////// VOID ExitMagnify( HWND hWnd, PSYCM psycm) { // // Release capture, remove magnified character, restore cursor. // ReleaseCapture(); RestoreSymMag(psycm); DrawSymChOutlineHwnd(psycm, hWnd, psycm->chCurr, TRUE, TRUE); if (psycm->fCursorOff) { ShowCursor(TRUE); } psycm->fMouseDn = psycm->fCursorOff = FALSE; } //////////////////////////////////////////////////////////////////////////// // // SetEditCtlFont // // Creates a font for the Edit control that visually matches the handle // given, but is guaranteed not to be bigger than the size of the edit // control. // //////////////////////////////////////////////////////////////////////////// void SetEditCtlFont( HWND hwndDlg, int idCtl, HFONT hfont) { static HFONT hfNew = NULL; LOGFONT lfNew; HWND hwndCtl = GetDlgItem(hwndDlg, idCtl); RECT rc; if (hfNew != NULL) { DeleteObject(hfNew); } GetWindowRect(hwndCtl, &rc); if (GetObject(hfont, sizeof(lfNew), &lfNew) != 0) { lfNew.lfHeight = rc.bottom - rc.top - 8; lfNew.lfWidth = lfNew.lfEscapement = lfNew.lfOrientation = lfNew.lfWeight = 0; hfNew = CreateFontIndirect(&lfNew); } else { hfNew = hfont; } SendMessage(hwndCtl, WM_SETFONT, (WPARAM)hfNew, (LPARAM)TRUE); if (hfNew == hfont) { hfNew = NULL; } }