/* * editec.c - Edit controls rewrite. Version II of edit controls. * Copyright (c) 1985 - 1999, Microsoft Corporation * Created: 24-Jul-88 davidds */ /* Warning: The single line editcontrols contain internal styles and API which * are need to support comboboxes. They are defined in combcom.h/combcom.inc * and may be redefined or renumbered as needed. */ #include "precomp.h" #pragma hdrstop LOOKASIDE EditLookaside; ICH ECFindTabA(LPSTR lpstr, ICH cch); ICH ECFindTabW(LPWSTR lpstr, ICH cch); #define umin(a, b) ((unsigned)(a) < (unsigned)(b) ? (unsigned)(a) : (unsigned)(b)) #define umax(a, b) ((unsigned)(a) > (unsigned)(b) ? (unsigned)(a) : (unsigned)(b)) #define UNICODE_CARRIAGERETURN ((WCHAR)0x0d) #define UNICODE_LINEFEED ((WCHAR)0x0a) #define UNICODE_TAB ((WCHAR)0x09) // IME Menu IDs #define ID_IMEOPENCLOSE 10001 #define ID_SOFTKBDOPENCLOSE 10002 #define ID_RECONVERTSTRING 10003 typedef struct { DWORD fDisableCut : 1; DWORD fDisablePaste : 1; DWORD fNeedSeparatorBeforeImeMenu : 1; DWORD fIME : 1; } EditMenuItemState; /* * Handlers common to both single and multi line edit controls. * ECLock * History: */ PSTR ECLock( PED ped) { PSTR ptext = LOCALLOCK(ped->hText, ped->hInstance); ped->iLockLevel++; /* * If this is the first lock of the text and the text is encoded * decode the text. */ //RIPMSG2(RIP_VERBOSE, "lock : %d '%10s'\n", ped->iLockLevel, ptext); if (ped->iLockLevel == 1 && ped->fEncoded) { /* * rtlrundecode can't handle zero length strings */ if (ped->cch != 0) { STRING string; string.Length = string.MaximumLength = (USHORT)(ped->cch * ped->cbChar); string.Buffer = ptext; RtlRunDecodeUnicodeString(ped->seed, (PUNICODE_STRING)&string); //RIPMSG1(RIP_VERBOSE, "Decoding: '%10s'\n", ptext); } ped->fEncoded = FALSE; } return ptext; } /* * ECUnlock * History: */ void ECUnlock( PED ped) { /* * if we are removing the last lock on the text and the password * character is set then encode the text */ //RIPMSG1(RIP_VERBOSE, "unlock: %d '%10s'\n", ped->iLockLevel, ped->ptext); if (ped->charPasswordChar && ped->iLockLevel == 1 && ped->cch != 0) { UNICODE_STRING string; string.Length = string.MaximumLength = (USHORT)(ped->cch * ped->cbChar); string.Buffer = LOCALLOCK(ped->hText, ped->hInstance); RtlRunEncodeUnicodeString(&(ped->seed), &string); //RIPMSG1(RIP_VERBOSE, "Encoding: '%10s'\n", ped->ptext); ped->fEncoded = TRUE; LOCALUNLOCK(ped->hText, ped->hInstance); } LOCALUNLOCK(ped->hText, ped->hInstance); ped->iLockLevel--; } /* * GetActualNegA() * For a given strip of text, this function computes the negative A width * for the whole strip and returns the value as a postive number. * It also fills the NegAInfo structure with details about the postion * of this strip that results in this Negative A. */ UINT GetActualNegA( HDC hdc, PED ped, int x, LPSTR lpstring, ICH ichString, int nCount, LPSTRIPINFO NegAInfo) { int iCharCount, i; int iLeftmostPoint = x; PABC pABCwidthBuff; UINT wCharIndex; int xStartPoint = x; ABC abc; // To begin with, let us assume that there is no negative A width for // this strip and initialize accodingly. NegAInfo->XStartPos = x; NegAInfo->lpString = lpstring; NegAInfo->nCount = 0; NegAInfo->ichString = ichString; // If the current font is not a TrueType font, then there can not be any // negative A widths. if (!ped->fTrueType) { if(!ped->charOverhang) { return 0; } else { NegAInfo->nCount = min(nCount, (int)ped->wMaxNegAcharPos); return ped->charOverhang; } } // How many characters are to be considered for computing Negative A ? iCharCount = min(nCount, (int)ped->wMaxNegAcharPos); // Do we have the info on individual character's widths? if(!ped->charWidthBuffer) { // No! So, let us tell them to consider all the characters. NegAInfo->nCount = iCharCount; return(iCharCount * ped->aveCharWidth); } pABCwidthBuff = (PABC) ped->charWidthBuffer; if (ped->fAnsi) { for (i = 0; i < iCharCount; i++) { wCharIndex = (UINT)(*((unsigned char *)lpstring)); if (*lpstring == VK_TAB) { // To play it safe, we assume that this tab results in a tab length of // 1 pixel because this is the minimum possible tab length. x++; } else { if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) x += pABCwidthBuff[wCharIndex].abcA; // Add the 'A' width. else { GetCharABCWidthsA(hdc, wCharIndex, wCharIndex, &abc) ; x += abc.abcA; } if (x < iLeftmostPoint) iLeftmostPoint = x; // Reset the leftmost point. if (x < xStartPoint) NegAInfo->nCount = i+1; // 'i' is index; To get the count add 1. if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) { x += pABCwidthBuff[wCharIndex].abcB + pABCwidthBuff[wCharIndex].abcC; } else { x += abc.abcB + abc.abcC; } } lpstring++; } } else { // Unicode LPWSTR lpwstring = (LPWSTR) lpstring ; for (i = 0; i < iCharCount; i++) { wCharIndex = *lpwstring ; if (*lpwstring == VK_TAB) { // To play it safe, we assume that this tab results in a tab length of // 1 pixel because this is the minimum possible tab length. x++; } else { if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) x += pABCwidthBuff[wCharIndex].abcA; // Add the 'A' width. else { GetCharABCWidthsW(hdc, wCharIndex, wCharIndex, &abc) ; x += abc.abcA ; } if (x < iLeftmostPoint) iLeftmostPoint = x; // Reset the leftmost point. if (x < xStartPoint) NegAInfo->nCount = i+1; // 'i' is index; To get the count add 1. if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) x += pABCwidthBuff[wCharIndex].abcB + pABCwidthBuff[wCharIndex].abcC; else x += abc.abcB + abc.abcC ; } lpwstring++; } } // Let us return the negative A for the whole strip as a positive value. return((UINT)(xStartPoint - iLeftmostPoint)); } /* * ECIsAncestorActive() * Returns whether or not we're the child of an "active" window. Looks for * the first parent window that has a caption. * This is a function because we might use it elsewhere when getting left * clicked on, etc. */ BOOL ECIsAncestorActive(HWND hwnd) { // We want to return TRUE always for top level windows. That's because // of how WM_MOUSEACTIVATE works. If we see the click at all, the // window is active. However, if we reach a child ancestor that has // a caption, return the frame-on style bit. // Note that calling FlashWindow() will have an effect. If the user // clicks on an edit field in a child window that is flashed off, nothing // will happen unless the window stops flashing and ncactivates first. while (hwnd) { PWND pwnd = ValidateHwnd( hwnd ); // Bail out if some parent window isn't 4.0 compatible or we've // reached the top. Fixes compatibility problems with 3.x apps, // especially MFC samples. if (!TestWF(pwnd, WFWIN40COMPAT) || !TestWF(pwnd, WFCHILD)) hwnd = NULL; // to break us out of the loop else if (TestWF(pwnd, WFCPRESENT)) return(TestWF(pwnd, WFFRAMEON) != 0); else hwnd = GetParent(hwnd); } return(TRUE); } /* * ECSetIMEMenu() * support IME specific context menu * Create: 30-Apr-97 Hiroyama : Ported from Memphis */ BOOL ECSetIMEMenu( HMENU hMenu, HWND hwnd, EditMenuItemState state) { MENUITEMINFO mii; HIMC hIMC; HKL hKL; HMENU hmenuSub; WCHAR szRes[32]; int nPrevLastItem; int nItemsAdded = 0; UserAssert(IS_IME_ENABLED() && state.fIME); hKL = THREAD_HKL(); if (!fpImmIsIME(hKL)) return TRUE; hIMC = fpImmGetContext(hwnd); if (hIMC == NULL) { // early out return FALSE; } hmenuSub = GetSubMenu(hMenu, 0); if (hmenuSub == NULL) { return FALSE; } nPrevLastItem = GetMenuItemCount(hmenuSub); if (hIMC) { if (LOWORD(HandleToUlong(hKL)) != 0x412) { // If Korean, do not show open/close menus if (fpImmGetOpenStatus(hIMC)) LoadString(hmodUser, STR_IMECLOSE, szRes, sizeof(szRes)); else LoadString(hmodUser, STR_IMEOPEN, szRes, sizeof(szRes)); mii.cbSize = sizeof(MENUITEMINFO); mii.fMask = MIIM_STRING | MIIM_ID; mii.dwTypeData = szRes; mii.cch = 0xffff; mii.wID = ID_IMEOPENCLOSE; InsertMenuItem(hmenuSub, 0xffff, TRUE, &mii); ++nItemsAdded; } if (fpImmGetProperty(hKL, IGP_CONVERSION) & IME_CMODE_SOFTKBD) { DWORD fdwConversion; fpImmGetConversionStatus(hIMC, &fdwConversion, NULL); if (fdwConversion & IME_CMODE_SOFTKBD) LoadString(hmodUser, STR_SOFTKBDCLOSE, szRes, sizeof(szRes)); else LoadString(hmodUser, STR_SOFTKBDOPEN, szRes, sizeof(szRes)); mii.cbSize = sizeof(MENUITEMINFO); mii.fMask = MIIM_STRING | MIIM_ID; mii.dwTypeData = szRes; mii.cch = 0xffff; mii.wID = ID_SOFTKBDOPENCLOSE; InsertMenuItem(hmenuSub, 0xffff, TRUE, &mii); ++nItemsAdded; } if (LOWORD(HandleToUlong(hKL)) != 0x412) { // If Korean, do not show reconversion menus DWORD dwSCS = fpImmGetProperty(hKL, IGP_SETCOMPSTR); LoadString(hmodUser, STR_RECONVERTSTRING, szRes, sizeof(szRes)); mii.cbSize = sizeof(MENUITEMINFO); mii.fMask = MIIM_STRING | MIIM_ID | MIIM_STATE; mii.dwTypeData = szRes; mii.fState = 0; mii.cch = 0xffff; mii.wID = ID_RECONVERTSTRING; if (state.fDisableCut || !(dwSCS & SCS_CAP_SETRECONVERTSTRING) || !(dwSCS & SCS_CAP_MAKEREAD)) { mii.fState |= MFS_GRAYED; } InsertMenuItem(hmenuSub, 0xffff, TRUE, &mii); ++nItemsAdded; } } // Add or remove the menu separator if (state.fNeedSeparatorBeforeImeMenu && nItemsAdded != 0) { // If the menu for Middle East has left a separator, // fNeedSeparatorBeforeImeMenu is FALSE. // I.e. we don't need to add more. mii.cbSize = sizeof(MENUITEMINFO); mii.fMask = MIIM_FTYPE; mii.fType = MFT_SEPARATOR; InsertMenuItem(hmenuSub, nPrevLastItem, TRUE, &mii); } else if (!state.fNeedSeparatorBeforeImeMenu && nItemsAdded == 0) { // Extra separator is left by ME menus. Remove it. UserVerify(NtUserDeleteMenu(hmenuSub, nPrevLastItem - 1, MF_BYPOSITION)); } fpImmReleaseContext(hwnd, hIMC); return TRUE; } void ECInOutReconversionMode(PED ped, BOOL fIn) { UserAssert(fIn == TRUE || fIn == FALSE); if (fIn == ped->fInReconversion) { return; } ped->fInReconversion = fIn; if (ped->fFocus) { (fIn ? NtUserHideCaret: NtUserShowCaret)(ped->hwnd); } return; } /* * ECDoIMEMenuCommand() * support IME specific context menu * Create: 30-Apr-97 Hiroyama : Ported from Memphis */ BOOL NEAR ECDoIMEMenuCommand(PED ped, int cmd, HWND hwnd) { HIMC hIMC; // early out switch (cmd) { case ID_IMEOPENCLOSE: case ID_SOFTKBDOPENCLOSE: case ID_RECONVERTSTRING: break; default: return FALSE; } // everybody needs hIMC, so get it here hIMC = fpImmGetContext(hwnd); if (hIMC == NULL) { // indicate to caller, that no further command processing needed return TRUE; } switch (cmd) { case ID_IMEOPENCLOSE: { // switch IME Open/Close status BOOL fOpen = fpImmGetOpenStatus(hIMC); fpImmSetOpenStatus(hIMC, !fOpen); } break; case ID_SOFTKBDOPENCLOSE: { DWORD fdwConversion; if (fpImmGetConversionStatus(hIMC, &fdwConversion, NULL)) { // Toggle soft keyboard Open/Close status fpImmEnumInputContext(0, SyncSoftKbdState, (fdwConversion & IME_CMODE_SOFTKBD) != IME_CMODE_SOFTKBD); } } break; case ID_RECONVERTSTRING: { DWORD dwStrLen; // holds TCHAR count of recionversion string DWORD cbLen; // holds BYTE SIZE of reconversion string DWORD dwSize; LPRECONVERTSTRING lpRCS; // pass current selection to IME for reconversion dwStrLen = ped->ichMaxSel - ped->ichMinSel; cbLen = dwStrLen * ped->cbChar; dwSize = cbLen + sizeof(RECONVERTSTRING) + 8; lpRCS = (LPRECONVERTSTRING)UserLocalAlloc(0, dwSize); if (lpRCS) { LPBYTE pText; pText = ECLock(ped); if (pText != NULL) { LPBYTE lpDest; BOOL (WINAPI* fpSetCompositionStringAW)(HIMC, DWORD, LPCVOID, DWORD, LPCVOID, DWORD); lpRCS->dwSize = dwSize; lpRCS->dwVersion = 0; lpRCS->dwStrLen = lpRCS->dwCompStrLen = lpRCS->dwTargetStrLen = dwStrLen; lpRCS->dwStrOffset = sizeof(RECONVERTSTRING); lpRCS->dwCompStrOffset = lpRCS->dwTargetStrOffset = 0; lpDest = (LPBYTE)lpRCS + sizeof(RECONVERTSTRING); RtlCopyMemory(lpDest, pText + ped->ichMinSel * ped->cbChar, cbLen); if (ped->fAnsi) { LPBYTE psz = (LPBYTE)lpDest; psz[cbLen] = '\0'; fpSetCompositionStringAW = fpImmSetCompositionStringA; } else { LPWSTR pwsz = (LPWSTR)lpDest; pwsz[dwStrLen] = L'\0'; fpSetCompositionStringAW = fpImmSetCompositionStringW; } ECUnlock(ped); UserAssert(fpSetCompositionStringAW != NULL); ECInOutReconversionMode(ped, TRUE); ECImmSetCompositionWindow(ped, 0, 0); // x and y will be overriden anyway fpSetCompositionStringAW(hIMC, SCS_SETRECONVERTSTRING, lpRCS, dwSize, NULL, 0); } // pText UserLocalFree(lpRCS); } } break; default: // should never reach here. RIPMSG1(RIP_ERROR, "ECDoIMEMenuCommand: unknown command id %d; should never reach here.", cmd); return FALSE; } UserAssert(hIMC != NULL); fpImmReleaseContext(hwnd, hIMC); return TRUE; } /* * ECMenu() * Handles context menu for edit fields. Disables inappropriate commands. * Note that this is NOT subclassing friendly, like most of our functions, * for speed and convenience. */ void ECMenu( HWND hwnd, PED ped, LPPOINT pt) { HMENU hMenu; int cmd = 0; int x; int y; EditMenuItemState state = { FALSE, // fDisableCut TRUE, // fDisablePaste TRUE, // fNeedSeparatorBeforeImeMenu IS_IME_ENABLED() && fpImmIsIME(THREAD_HKL()), // fIME }; // Set focus if we don't have it. if (!ped->fFocus) NtUserSetFocus(hwnd); // Grab the menu from USER's resources... if (!(hMenu = LoadMenu( hmodUser, MAKEINTRESOURCE( ID_EC_PROPERTY_MENU )))) return ; // Undo -- not allowed if we have no saved undo info if (ped->undoType == UNDO_NONE) EnableMenuItem(hMenu, WM_UNDO, MF_BYCOMMAND | MFS_GRAYED); if (ped->fReadOnly || ped->charPasswordChar) { // Cut and Delete -- not allowed if read-only or password state.fDisableCut = TRUE; } else { // Cut, Delete -- not allowed if there's no selection if (ped->ichMinSel == ped->ichMaxSel) state.fDisableCut = TRUE; } // Paste -- not allowed if there's no text on the clipboard // (this works for both OEM and Unicode) // Used to be always disabled for password edits MCostea #221035 if (NtUserIsClipboardFormatAvailable(CF_TEXT)) state.fDisablePaste = FALSE; if (state.fDisableCut) { EnableMenuItem(hMenu, WM_CUT, MF_BYCOMMAND | MFS_GRAYED); EnableMenuItem(hMenu, WM_CLEAR, MF_BYCOMMAND | MFS_GRAYED); } if (state.fDisablePaste) EnableMenuItem(hMenu, WM_PASTE, MF_BYCOMMAND | MFS_GRAYED); // Copy -- not allowed if there's no selection or password ec if ((ped->ichMinSel == ped->ichMaxSel) || (ped->charPasswordChar)) EnableMenuItem(hMenu, WM_COPY, MF_BYCOMMAND | MFS_GRAYED); // Select All -- not allowed if there's no text or if everything is // selected. Latter case takes care of first one. if ((ped->ichMinSel == 0) && (ped->ichMaxSel == ped->cch)) EnableMenuItem(hMenu, EM_SETSEL, MF_BYCOMMAND | MFS_GRAYED); if (ped->pLpkEditCallout) { ped->pLpkEditCallout->EditSetMenu(ped, hMenu); } else { NtUserDeleteMenu(hMenu, ID_CNTX_DISPLAYCTRL, MF_BYCOMMAND); NtUserDeleteMenu(hMenu, ID_CNTX_RTL, MF_BYCOMMAND); NtUserDeleteMenu(hMenu, ID_CNTX_INSERTCTRL, MF_BYCOMMAND); if (state.fIME) { // One separator is left in the menu, // no need to add the one before IME menus state.fNeedSeparatorBeforeImeMenu = FALSE; } else { // Extra separator is left. Remove it. HMENU hmenuSub = GetSubMenu(hMenu, 0); int nItems = GetMenuItemCount(hmenuSub) - 1; UserAssert(nItems >= 0); UserAssert(GetMenuState(hmenuSub, nItems, MF_BYPOSITION) & MF_SEPARATOR); // remove needless separator UserVerify(NtUserDeleteMenu(hmenuSub, nItems, MF_BYPOSITION)); } } // IME specific menu if (state.fIME) { ECSetIMEMenu(hMenu, hwnd, state); } // BOGUS // We position the menu below & to the right of the point clicked on. // Is this cool? I think so. Excel 4.0 does the same thing. It // seems like it would be neat if we could avoid obscuring the // selection. But in actuality, it seems even more awkward to move // the menu out of the way of the selection. The user can't click // and drag that way, and they have to move the mouse a ton. // We need to use TPM_NONOTIFY because VBRUN100 and VBRUN200 GP-fault // on unexpected menu messages. /* * if message came via the keyboard then center on the control * We use -1 && -1 here not 0xFFFFFFFF like Win95 becuase we * previously converted the lParam to a point with sign extending. */ if (pt->x == -1 && pt->y == -1) { RECT rc; GetWindowRect(hwnd, &rc); x = rc.left + (rc.right - rc.left) / 2; y = rc.top + (rc.bottom - rc.top) / 2; } else { x = pt->x; y = pt->y; } cmd = NtUserTrackPopupMenuEx(GetSubMenu(hMenu, 0), TPM_NONOTIFY | TPM_LEFTALIGN | TPM_TOPALIGN | TPM_RETURNCMD | TPM_RIGHTBUTTON, x, y, hwnd, NULL); // Free our menu NtUserDestroyMenu(hMenu); if (cmd && (cmd != -1)) { if (ped->pLpkEditCallout && cmd) { ped->pLpkEditCallout->EditProcessMenu(ped, cmd); } if (!state.fIME || !ECDoIMEMenuCommand(ped, cmd, hwnd)) { // if cmd is not IME specific menu, send it. SendMessage(hwnd, cmd, 0, (cmd == EM_SETSEL) ? 0xFFFFFFFF : 0L ); } } } /* * ECClearText() * Clears selected text. Does NOT _send_ a fake char backspace. */ void ECClearText(PED ped) { if (!ped->fReadOnly && (ped->ichMinSel < ped->ichMaxSel)) { if (ped->fSingle) SLEditWndProc(ped->hwnd, ped, WM_CHAR, VK_BACK, 0L ); else MLEditWndProc(ped->hwnd, ped, WM_CHAR, VK_BACK, 0L ); } } /* * ECCutText() - * Cuts selected text. This removes and copies the selection to the clip, * or if nothing is selected we delete (clear) the left character. */ void ECCutText(PED ped) { // Cut selection--IE, remove and copy to clipboard, or if no selection, // delete (clear) character left. if (!ped->fReadOnly && (ped->ichMinSel < ped->ichMaxSel) && SendMessage(ped->hwnd, WM_COPY, 0, 0L)) { // If copy was successful, delete the copied text by sending a // backspace message which will redraw the text and take care of // notifying the parent of changes. ECClearText(ped); } } /* * ECGetModKeys() * Gets modifier key states. Currently, we only check for VK_CONTROL and * VK_SHIFT. */ int ECGetModKeys(int keyMods) { int scState; scState = 0; if (!keyMods) { if (GetKeyState(VK_CONTROL) < 0) scState |= CTRLDOWN; if (GetKeyState(VK_SHIFT) < 0) scState |= SHFTDOWN; } else if (keyMods != NOMODIFY) scState = keyMods; return scState; } /* * ECTabTheTextOut() AorW * If fDrawText == FALSE, then this function returns the text extent of * of the given strip of text. It does not worry about the Negative widths. * If fDrawText == TRUE, this draws the given strip of Text expanding the * tabs to proper lengths, calculates and fills up the NegCInfoForStrip with * details required to draw the portions of this strip that goes beyond the * xClipEndPos due to Negative C widths. * Returns the max width AS A DWORD. We don't care about the height * at all. No one uses it. We keep a DWORD because that way we avoid * overflow. * NOTE: If the language pack is loaded EcTabTheTextOut is not used - the * language pack must take care of all tab expansion and selection * highlighting with full support for bidi layout and complex script * glyph reordering. */ UINT ECTabTheTextOut( HDC hdc, int xClipStPos, int xClipEndPos, int xStart, int y, LPSTR lpstring, int nCount, ICH ichString, PED ped, int iTabOrigin, BOOL fDraw, LPSTRIPINFO NegCInfoForStrip) { int nTabPositions; // Count of tabstops in tabstop array. LPINT lpintTabStopPositions; // Tab stop positions in pixels. int cch; UINT textextent; int xEnd; int pixeltabstop = 0; int i; int cxCharWidth; RECT rc; BOOL fOpaque; BOOL fFirstPass = TRUE; PINT charWidthBuff; int iTabLength; int nConsecutiveTabs; int xStripStPos; int xStripEndPos; int xEndOfStrip; STRIPINFO RedrawStripInfo; STRIPINFO NegAInfo; LPSTR lpTab; LPWSTR lpwTab; UINT wNegCwidth, wNegAwidth; int xRightmostPoint = xClipStPos; int xTabStartPos; int iSavedBkMode = 0; WCHAR wchar; SIZE size; ABC abc ; // Algorithm: Draw the strip opaquely first. If a tab length is so // small that the portions of text on either side of a tab overlap with // the other, then this will result in some clipping. So, such portion // of the strip is remembered in "RedrawStripInfo" and redrawn // transparently later to compensate the clippings. // NOTE: "RedrawStripInfo" can hold info about just one portion. So, if // more than one portion of the strip needs to be redrawn transparently, // then we "merge" all such portions into a single strip and redraw that // strip at the end. if (fDraw) { // To begin with, let us assume that there is no Negative C for this // strip and initialize the Negative Width Info structure. NegCInfoForStrip->nCount = 0; NegCInfoForStrip->XStartPos = xClipEndPos; // We may not have to redraw any portion of this strip. RedrawStripInfo.nCount = 0; fOpaque = (GetBkMode(hdc) == OPAQUE) || (fDraw == ECT_SELECTED); } #if DBG else { // Both MLGetLineWidth() and ECCchInWidth() should be clipping // nCount to avoid overflow. if (nCount > MAXLINELENGTH) RIPMSG0(RIP_WARNING, "ECTabTheTextOut: nCount > MAXLINELENGTH"); } #endif // Let us define the Clip rectangle. rc.left = xClipStPos; rc.right = xClipEndPos; rc.top = y; rc.bottom = y + ped->lineHeight; // Check if anything needs to be drawn. if (!lpstring || !nCount) { if (fDraw) ExtTextOutW(hdc, xClipStPos, y, (fOpaque ? ETO_OPAQUE | ETO_CLIPPED : ETO_CLIPPED), &rc, L"", 0, 0L); return(0L); } // Starting position xEnd = xStart; cxCharWidth = ped->aveCharWidth; nTabPositions = (ped->pTabStops ? *(ped->pTabStops) : 0); if (ped->pTabStops) { lpintTabStopPositions = (LPINT)(ped->pTabStops+1); if (nTabPositions == 1) { pixeltabstop = lpintTabStopPositions[0]; if (!pixeltabstop) pixeltabstop = 1; } } else { lpintTabStopPositions = NULL; pixeltabstop = 8*cxCharWidth; } // The first time we will draw the strip Opaquely. If some portions need // to be redrawn , then we will set the mode to TRANSPARENT and // jump to this location to redraw those portions. RedrawStrip: while (nCount) { wNegCwidth = ped->wMaxNegC; // Search for the first TAB in this strip; also compute the extent // of the the strip upto and not including the tab character. // Note - If the langpack is loaded, there will be no charWidthBuffer. if (ped->charWidthBuffer) { // Do we have a character width buffer? textextent = 0; cch = nCount; if (ped->fTrueType) { // If so, does it have ABC widths? UINT iRightmostPoint = 0; UINT wCharIndex; PABC pABCwidthBuff; pABCwidthBuff = (PABC) ped->charWidthBuffer; if ( ped->fAnsi ) { for (i = 0; i < nCount; i++) { if (lpstring[i] == VK_TAB) { cch = i; break; } wCharIndex = (UINT)(((unsigned char *)lpstring)[i]); if (wCharIndex < CHAR_WIDTH_BUFFER_LENGTH) { textextent += (UINT)(pABCwidthBuff[wCharIndex].abcA + pABCwidthBuff[wCharIndex].abcB); } else { // not in cache, will ask driver GetCharABCWidthsA(hdc, wCharIndex, wCharIndex, &abc); textextent += abc.abcA + abc.abcB ; } if (textextent > iRightmostPoint) iRightmostPoint = textextent; if (wCharIndex < CHAR_WIDTH_BUFFER_LENGTH) { textextent += pABCwidthBuff[wCharIndex].abcC; } else { // not in cache textextent += abc.abcC; } if (textextent > iRightmostPoint) iRightmostPoint = textextent; } } else { // Unicode for (i = 0; i < nCount; i++) { WCHAR UNALIGNED * lpwstring = (WCHAR UNALIGNED *)lpstring; if (lpwstring[i] == VK_TAB) { cch = i; break; } wCharIndex = lpwstring[i] ; if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) textextent += pABCwidthBuff[wCharIndex].abcA + pABCwidthBuff[wCharIndex].abcB; else { GetCharABCWidthsW(hdc, wCharIndex, wCharIndex, &abc) ; textextent += abc.abcA + abc.abcB ; } /* * Note that abcC could be negative so we need this * statement here *and* below */ if (textextent > iRightmostPoint) iRightmostPoint = textextent; if ( wCharIndex < CHAR_WIDTH_BUFFER_LENGTH ) textextent += pABCwidthBuff[wCharIndex].abcC; else textextent += abc.abcC ; if (textextent > iRightmostPoint) iRightmostPoint = textextent; } } wNegCwidth = (int)(iRightmostPoint - textextent); } else { // !ped->fTrueType // No! This is not a TrueType font; So, we have only character // width info in this buffer. charWidthBuff = ped->charWidthBuffer; if ( ped->fAnsi ) { // Initially assume no tabs exist in the text so cch=nCount. for (i = 0; i < nCount; i++) { if (lpstring[i] == VK_TAB) { cch = i; break; } // Call GetTextExtentPoint for dbcs/hankaku characters if (ped->fDBCS && (i+1 < nCount) && ECIsDBCSLeadByte(ped,lpstring[i])) { GetTextExtentPointA(hdc, &lpstring[i], 2, &size); textextent += size.cx; i++; } else if ((UCHAR)lpstring[i] >= CHAR_WIDTH_BUFFER_LENGTH) { // Skip this GetExtentPoint call for non hankaku code points // Or if the character is in the width cache. GetTextExtentPointA(hdc, &lpstring[i], 1, &size); textextent += size.cx; } else { textextent += (UINT)(charWidthBuff[(UINT)(((unsigned char *)lpstring)[i])]); } } } else { LPWSTR lpwstring = (LPWSTR) lpstring ; INT cchUStart; // start of unicode character count for (i = 0; i < nCount; i++) { if (lpwstring[i] == VK_TAB) { cch = i; break; } wchar = lpwstring[i]; if (wchar >= CHAR_WIDTH_BUFFER_LENGTH) { /* * We have a Unicode character that is not in our * cache, get all the characters outside the cache * before getting the text extent on this part of the * string. */ cchUStart = i; while (wchar >= CHAR_WIDTH_BUFFER_LENGTH && wchar != VK_TAB && i < nCount) { wchar = lpwstring[++i]; } GetTextExtentPointW(hdc, (LPWSTR)lpwstring + cchUStart, i-cchUStart, &size); textextent += size.cx; if (wchar == VK_TAB || i >= nCount) { cch = i; break; } /* * We have a char that is in the cache, fall through. */ } /* * The width of this character is in the cache buffer. */ textextent += ped->charWidthBuffer[wchar]; } } } // fTrueType else. nCount -= cch; } else { // If we don't have a buffer that contains the width info. /* * Gotta call the driver to do our text extent. */ if ( ped->fAnsi ) { cch = (int)ECFindTabA(lpstring, nCount); GetTextExtentPointA(hdc, lpstring, cch, &size) ; } else { cch = (int)ECFindTabW((LPWSTR) lpstring, nCount); GetTextExtentPointW(hdc, (LPWSTR)lpstring, cch, &size); } nCount -= cch; // Subtruct Overhang for Italic fonts. textextent = (size.cx - ped->charOverhang); } // textextent is computed. xStripStPos = xEnd; xEnd += (int)textextent; xStripEndPos = xEnd; // We will consider the negative widths only if when we draw opaquely. if (fFirstPass && fDraw) { xRightmostPoint = max(xStripEndPos + (int)wNegCwidth, xRightmostPoint); // Check if this strip peeps beyond the clip region. if (xRightmostPoint > xClipEndPos) { if (!NegCInfoForStrip->nCount) { NegCInfoForStrip->lpString = lpstring; NegCInfoForStrip->ichString = ichString; NegCInfoForStrip->nCount = nCount+cch; NegCInfoForStrip->XStartPos = xStripStPos; } } } /* if (fFirstPass && fDraw) */ if ( ped->fAnsi ) lpTab = lpstring + cch; // Possibly Points to a tab character. else lpwTab = ((LPWSTR)lpstring) + cch ; // we must consider all the consecutive tabs and calculate the // the begining of next strip. nConsecutiveTabs = 0; while (nCount && (ped->fAnsi ? (*lpTab == VK_TAB) : (*lpwTab == VK_TAB))) { // Find the next tab position and update the x value. xTabStartPos = xEnd; if (pixeltabstop) xEnd = (((xEnd-iTabOrigin)/pixeltabstop)*pixeltabstop) + pixeltabstop + iTabOrigin; else { for (i = 0; i < nTabPositions; i++) { if (xEnd < (lpintTabStopPositions[i] + iTabOrigin)) { xEnd = (lpintTabStopPositions[i] + iTabOrigin); break; } } // Check if all the tabstops set are exhausted; Then start using // default tab stop positions. if (i == nTabPositions) { pixeltabstop = 8*cxCharWidth; xEnd = ((xEnd - iTabOrigin)/pixeltabstop)*pixeltabstop + pixeltabstop + iTabOrigin; } } if (fFirstPass && fDraw) { xRightmostPoint = max(xEnd, xRightmostPoint); /* Check if this strip peeps beyond the clip region */ if (xRightmostPoint > xClipEndPos) { if (!NegCInfoForStrip->nCount) { NegCInfoForStrip->ichString = ichString + cch + nConsecutiveTabs; NegCInfoForStrip->nCount = nCount; NegCInfoForStrip->lpString = (ped->fAnsi ? lpTab : (LPSTR) lpwTab); NegCInfoForStrip->XStartPos = xTabStartPos; } } } /* if(fFirstPass) */ nConsecutiveTabs++; nCount--; ped->fAnsi ? lpTab++ : (LPSTR) (lpwTab++) ; // Move to the next character. } // while(*lpTab == TAB) // if (fDraw) { if (fFirstPass) { // Is anything remaining to be drawn in this strip? if (!nCount) rc.right = xEnd; // No! We are done. else { // "x" is the effective starting position of next strip. iTabLength = xEnd - xStripEndPos; // Check if there is a possibility of this tab length being too small // compared to the negative A and C widths if any. if ((wNegCwidth + (wNegAwidth = ped->wMaxNegA)) > (UINT)iTabLength) { // Unfortunately, there is a possiblity of an overlap. // Let us find out the actual NegA for the next strip. wNegAwidth = GetActualNegA( hdc, ped, xEnd, lpstring + (cch + nConsecutiveTabs)*ped->cbChar, ichString + cch + nConsecutiveTabs, nCount, &NegAInfo); } // Check if they actually overlap // if ((wNegCwidth + wNegAwidth) <= (UINT)iTabLength) { // No overlap between the strips. This is the ideal situation. rc.right = xEnd - wNegAwidth; } else { // Yes! They overlap. rc.right = xEnd; // See if negative C width is too large compared to tab length. if (wNegCwidth > (UINT)iTabLength) { // Must redraw transparently a part of the current strip later. if (RedrawStripInfo.nCount) { // A previous strip also needs to be redrawn; So, merge this // strip to that strip. RedrawStripInfo.nCount = (ichString - RedrawStripInfo.ichString) + cch; } else { RedrawStripInfo.nCount = cch; RedrawStripInfo.lpString = lpstring; RedrawStripInfo.ichString = ichString; RedrawStripInfo.XStartPos = xStripStPos; } } if (wNegAwidth) { // Must redraw transparently the first part of the next strip later. if (RedrawStripInfo.nCount) { // A previous strip also needs to be redrawn; So, merge this // strip to that strip. RedrawStripInfo.nCount = (NegAInfo.ichString - RedrawStripInfo.ichString) + NegAInfo.nCount; } else RedrawStripInfo = NegAInfo; } } } // else (!nCount) // } // if (fFirstPass) // if (rc.left < xClipEndPos) { if (fFirstPass) { // If this is the end of the strip, then complete the rectangle. if ((!nCount) && (xClipEndPos == MAXCLIPENDPOS)) rc.right = max(rc.right, xClipEndPos); else rc.right = min(rc.right, xClipEndPos); } // Draw the current strip. if (rc.left < rc.right) if ( ped->fAnsi ) ExtTextOutA(hdc, xStripStPos, y, (fFirstPass && fOpaque ? (ETO_OPAQUE | ETO_CLIPPED) : ETO_CLIPPED), (LPRECT)&rc, lpstring, cch, 0L); else ExtTextOutW(hdc, xStripStPos, y, (fFirstPass && fOpaque ? (ETO_OPAQUE | ETO_CLIPPED) : ETO_CLIPPED), (LPRECT)&rc, (LPWSTR)lpstring, cch, 0L); } if (fFirstPass) rc.left = max(rc.right, xClipStPos); ichString += (cch+nConsecutiveTabs); } // if (fDraw) // // Skip over the tab and the characters we just drew. lpstring += (cch + nConsecutiveTabs) * ped->cbChar; } // while (nCount) // xEndOfStrip = xEnd; // check if we need to draw some portions transparently. if (fFirstPass && fDraw && RedrawStripInfo.nCount) { iSavedBkMode = SetBkMode(hdc, TRANSPARENT); fFirstPass = FALSE; nCount = RedrawStripInfo.nCount; rc.left = xClipStPos; rc.right = xClipEndPos; lpstring = RedrawStripInfo.lpString; ichString = RedrawStripInfo.ichString; xEnd = RedrawStripInfo.XStartPos; goto RedrawStrip; // Redraw Transparently. } if (iSavedBkMode) // Did we change the Bk mode? SetBkMode(hdc, iSavedBkMode); // Then, let us set it back! return((UINT)(xEndOfStrip - xStart)); } /* * ECCchInWidth AorW * Returns maximum count of characters (up to cch) from the given * string (starting either at the beginning and moving forward or at the * end and moving backwards based on the setting of the fForward flag) * which will fit in the given width. ie. Will tell you how much of * lpstring will fit in the given width even when using proportional * characters. WARNING: If we use kerning, then this loses... * History: * NOTE: ECCchInWidth is not called if the language pack is loaded. */ ICH ECCchInWidth( PED ped, HDC hdc, LPSTR lpText, ICH cch, int width, BOOL fForward) { int stringExtent; int cchhigh; int cchnew = 0; int cchlow = 0; SIZE size; LPSTR lpStart; if ((width <= 0) || !cch) return (0); /* * Optimize nonproportional fonts for single line ec since they don't have * tabs. */ // Change optimize condition for fixed pitch font if (ped->fNonPropFont && ped->fSingle && !ped->fDBCS) { return (ECAdjustIch( ped, lpText, umin(width/ped->aveCharWidth,(int)cch))); } /* * Check if password hidden chars are being used. */ if (ped->charPasswordChar) { return (umin(width / ped->cPasswordCharWidth, (int)cch)); } /* * ALWAYS RESTRICT TO AT MOST MAXLINELENGTH to avoid overflow... */ cch = umin(MAXLINELENGTH, cch); cchhigh = cch + 1; while (cchlow < cchhigh - 1) { cchnew = umax((cchhigh - cchlow) / 2, 1) + cchlow; lpStart = lpText; /* * If we want to figure out how many fit starting at the end and moving * backwards, make sure we move to the appropriate position in the * string before calculating the text extent. */ if (!fForward) lpStart += (cch - cchnew)*ped->cbChar; if (ped->fSingle) { if (ped->fAnsi) GetTextExtentPointA(hdc, (LPSTR)lpStart, cchnew, &size); else GetTextExtentPointW(hdc, (LPWSTR)lpStart, cchnew, &size); stringExtent = size.cx; } else { stringExtent = ECTabTheTextOut(hdc, 0, 0, 0, 0, lpStart, cchnew, 0, ped, 0, ECT_CALC, NULL ); } if (stringExtent > width) { cchhigh = cchnew; } else { cchlow = cchnew; } } // Call ECAdjustIch ( generic case ) cchlow = ECAdjustIch( ped, lpText, cchlow ); return (cchlow); } /* * ECFindTab * Scans lpstr and return s the number of CHARs till the first TAB. * Scans at most cch chars of lpstr. * History: */ ICH ECFindTabA( LPSTR lpstr, ICH cch) { LPSTR copylpstr = lpstr; if (!cch) return 0; while (*lpstr != VK_TAB) { lpstr++; if (--cch == 0) break; } return ((ICH)(lpstr - copylpstr)); } ICH ECFindTabW( LPWSTR lpstr, ICH cch) { LPWSTR copylpstr = lpstr; if (!cch) return 0; while (*lpstr != VK_TAB) { lpstr++; if (--cch == 0) break; } return ((ICH)(lpstr - copylpstr)); } /* * ECGetBrush() * Gets appropriate background brush to erase with. */ HBRUSH ECGetBrush(PED ped, HDC hdc) { HBRUSH hbr; BOOL f40Compat; f40Compat = (GETAPPVER() >= VER40); // Get background brush if ((ped->fReadOnly || ped->fDisabled) && f40Compat) { hbr = ECGetControlBrush(ped, hdc, WM_CTLCOLORSTATIC); } else hbr = ECGetControlBrush(ped, hdc, WM_CTLCOLOREDIT); if (ped->fDisabled && (ped->fSingle || f40Compat)) { DWORD rgb; // Change text color rgb = GetSysColor(COLOR_GRAYTEXT); if (rgb != GetBkColor(hdc)) SetTextColor(hdc, rgb); } return(hbr); } /* * NextWordCallBack * History: * 02-19-92 JimA Ported from Win31 sources. */ void NextWordCallBack( PED ped, ICH ichStart, BOOL fLeft, ICH *pichMin, ICH *pichMax ) { ICH ichMinSel; ICH ichMaxSel; LPSTR pText; pText = ECLock(ped); if (fLeft || (!(BOOL)CALLWORDBREAKPROC(ped->lpfnNextWord, (LPSTR)pText, ichStart, ped->cch, WB_ISDELIMITER) && (ped->fAnsi ? (*(pText + ichStart) != VK_RETURN) : (*((LPWSTR)pText + ichStart) != VK_RETURN)) )) ichMinSel = CALLWORDBREAKPROC(*ped->lpfnNextWord, (LPSTR)pText, ichStart, ped->cch, WB_LEFT); else ichMinSel = CALLWORDBREAKPROC(*ped->lpfnNextWord, (LPSTR)pText, ichStart, ped->cch, WB_RIGHT); ichMaxSel = min(ichMinSel + 1, ped->cch); if (ped->fAnsi) { if (*(pText + ichMinSel) == VK_RETURN) { if (ichMinSel > 0 && *(pText + ichMinSel - 1) == VK_RETURN) { /* * So that we can treat CRCRLF as one word also. */ ichMinSel--; } else if (*(pText+ichMinSel + 1) == VK_RETURN) { /* * Move MaxSel on to the LF */ ichMaxSel++; } } } else { if (*((LPWSTR)pText + ichMinSel) == VK_RETURN) { if (ichMinSel > 0 && *((LPWSTR)pText + ichMinSel - 1) == VK_RETURN) { /* * So that we can treat CRCRLF as one word also. */ ichMinSel--; } else if (*((LPWSTR)pText+ichMinSel + 1) == VK_RETURN) { /* * Move MaxSel on to the LF */ ichMaxSel++; } } } ichMaxSel = CALLWORDBREAKPROC(ped->lpfnNextWord, (LPSTR)pText, ichMaxSel, ped->cch, WB_RIGHT); ECUnlock(ped); if (pichMin) *pichMin = ichMinSel; if (pichMax) *pichMax = ichMaxSel; } /* * NextWordLpkCallback * Identifies next/prev word position for complex scripts * History: * 04-22-97 DBrown */ void NextWordLpkCallBack( PED ped, ICH ichStart, BOOL fLeft, ICH *pichMin, ICH *pichMax) { PSTR pText = ECLock(ped); HDC hdc = ECGetEditDC(ped, TRUE); ped->pLpkEditCallout->EditNextWord(ped, hdc, pText, ichStart, fLeft, pichMin, pichMax); ECReleaseEditDC(ped, hdc, TRUE); ECUnlock(ped); } /* * ECWordAorW * if fLeft, Returns the ichMinSel and ichMaxSel of the word to the * left of ichStart. ichMinSel contains the starting letter of the word, * ichmaxsel contains all spaces up to the first character of the next word. * if !fLeft, Returns the ichMinSel and ichMaxSel of the word to the right of * ichStart. ichMinSel contains the starting letter of the word, ichmaxsel * contains the first letter of the next word. If ichStart is in the middle * of a word, that word is considered the left or right word. * A CR LF pair or CRCRLF triple is considered a single word in * multiline edit controls. * History: */ void ECWord( PED ped, ICH ichStart, BOOL fLeft, ICH *pichMin, ICH *pichMax ) { BOOL charLocated = FALSE; BOOL spaceLocated = FALSE; if ((!ichStart && fLeft) || (ichStart == ped->cch && !fLeft)) { /* * We are at the beginning of the text (looking left) or we are at end * of text (looking right), no word here */ if (pichMin) *pichMin=0; if (pichMax) *pichMax=0; return; } /* * Don't give out hints about word breaks if password chars are being used, */ if (ped->charPasswordChar) { if (pichMin) *pichMin=0; if (pichMax) *pichMax=ped->cch; return; } if (ped->fAnsi) { PSTR pText; PSTR pWordMinSel; PSTR pWordMaxSel; PSTR pPrevChar; UserAssert(ped->cbChar == sizeof(CHAR)); if (ped->lpfnNextWord) { NextWordCallBack(ped, ichStart, fLeft, pichMin, pichMax); return; } if (ped->pLpkEditCallout) { NextWordLpkCallBack(ped, ichStart, fLeft, pichMin, pichMax); return; } pText = ECLock(ped); pWordMinSel = pWordMaxSel = pText + ichStart; /* * if fLeft: Move pWordMinSel to the left looking for the start of a word. * If we start at a space, we will include spaces in the selection as we * move left untill we find a nonspace character. At that point, we continue * looking left until we find a space. Thus, the selection will consist of * a word with its trailing spaces or, it will consist of any leading at the * beginning of a line of text. */ /* * if !fLeft: (ie. right word) Move pWordMinSel looking for the start of a * word. If the pWordMinSel points to a character, then we move left * looking for a space which will signify the start of the word. If * pWordMinSel points to a space, we look right till we come upon a * character. pMaxWord will look right starting at pMinWord looking for the * end of the word and its trailing spaces. */ if (fLeft || !ISDELIMETERA(*pWordMinSel) && *pWordMinSel != 0x0D) { /* * If we are moving left or if we are moving right and we are not on a * space or a CR (the start of a word), then we was look left for the * start of a word which is either a CR or a character. We do this by * looking left till we find a character (or if CR we stop), then we * continue looking left till we find a space or LF. */ while (pWordMinSel > pText && ((!ISDELIMETERA(*(pWordMinSel - 1)) && *(pWordMinSel - 1) != 0x0A) || !charLocated)) { /* * Treat double byte character as a word ( in ansi pWordMinSel loop ) */ pPrevChar = ECAnsiPrev( ped, pText, pWordMinSel ); /* ** we are looking right ( !fLeft ). ** if current character is a double byte chararacter or ** previous character is a double byte character, we ** are on the beggining of a word. */ if ( !fLeft && ( ISDELIMETERA( *pPrevChar ) || *pPrevChar == 0x0A || ECIsDBCSLeadByte(ped, *pWordMinSel) || pWordMinSel - pPrevChar == 2 ) ) { /* * If we are looking for the start of the word right, then we * stop when we have found it. (needed in case charLocated is * still FALSE) */ break; } if ( pWordMinSel - pPrevChar == 2 ) { /* ** previous character is a double byte character. ** if we are in a word ( charLocated == TRUE ) ** current position is the beginning of the word ** if we are not in a word ( charLocated == FALSE ) ** the previous character is what we looking for. */ if ( ! charLocated ) { pWordMinSel = pPrevChar; } break; } pWordMinSel = pPrevChar; if (!ISDELIMETERA(*pWordMinSel) && *pWordMinSel != 0x0A) { /* * We have found the last char in the word. Continue looking * backwards till we find the first char of the word */ charLocated = TRUE; /* * We will consider a CR the start of a word */ if (*pWordMinSel == 0x0D) break; } } } else { while ((ISDELIMETERA(*pWordMinSel) || *pWordMinSel == 0x0A) && pWordMinSel < pText + ped->cch) pWordMinSel++; } /* * Adjust the initial position of pWordMaxSel ( in ansi ) */ pWordMaxSel = ECAnsiNext(ped, pWordMinSel); pWordMaxSel = min(pWordMaxSel, pText + ped->cch); /* ** If pWordMinSel points a double byte character AND ** pWordMaxSel points non space ** then ** pWordMaxSel points the beggining of next word. */ if ( ( pWordMaxSel - pWordMinSel == 2 ) && ! ISDELIMETERA(*pWordMaxSel) ) goto FastReturnA; if (*pWordMinSel == 0x0D) { if (pWordMinSel > pText && *(pWordMinSel - 1) == 0x0D) /* So that we can treat CRCRLF as one word also. */ pWordMinSel--; else if (*(pWordMinSel + 1) == 0x0D) /* Move MaxSel on to the LF */ pWordMaxSel++; } /* * Check if we have a one character word */ if (ISDELIMETERA(*pWordMaxSel)) spaceLocated = TRUE; /* * Move pWordMaxSel to the right looking for the end of a word and its * trailing spaces. WordMaxSel stops on the first character of the next * word. Thus, we break either at a CR or at the first nonspace char after * a run of spaces or LFs. */ while ((pWordMaxSel < pText + ped->cch) && (!spaceLocated || (ISDELIMETERA(*pWordMaxSel)))) { if (*pWordMaxSel == 0x0D) break; /* * Treat double byte character as a word ( in ansi pWordMaxSel loop ) */ /* ** if it's a double byte character then ** we are at the beginning of next word ** which is a double byte character. */ if (ECIsDBCSLeadByte( ped, *pWordMaxSel)) break; pWordMaxSel++; if (ISDELIMETERA(*pWordMaxSel)) spaceLocated = TRUE; if (*(pWordMaxSel - 1) == 0x0A) break; } /* * label for fast return ( for Ansi ) */ FastReturnA: ECUnlock(ped); if (pichMin) *pichMin = (ICH)(pWordMinSel - pText); if (pichMax) *pichMax = (ICH)(pWordMaxSel - pText); return; } else { // !fAnsi LPWSTR pwText; LPWSTR pwWordMinSel; LPWSTR pwWordMaxSel; BOOL charLocated = FALSE; BOOL spaceLocated = FALSE; PWSTR pwPrevChar; UserAssert(ped->cbChar == sizeof(WCHAR)); if (ped->lpfnNextWord) { NextWordCallBack(ped, ichStart, fLeft, pichMin, pichMax); return; } if (ped->pLpkEditCallout) { NextWordLpkCallBack(ped, ichStart, fLeft, pichMin, pichMax); return; } pwText = (LPWSTR)ECLock(ped); pwWordMinSel = pwWordMaxSel = pwText + ichStart; /* * if fLeft: Move pWordMinSel to the left looking for the start of a word. * If we start at a space, we will include spaces in the selection as we * move left untill we find a nonspace character. At that point, we continue * looking left until we find a space. Thus, the selection will consist of * a word with its trailing spaces or, it will consist of any leading at the * beginning of a line of text. */ /* * if !fLeft: (ie. right word) Move pWordMinSel looking for the start of a * word. If the pWordMinSel points to a character, then we move left * looking for a space which will signify the start of the word. If * pWordMinSel points to a space, we look right till we come upon a * character. pMaxWord will look right starting at pMinWord looking for the * end of the word and its trailing spaces. */ if (fLeft || (!ISDELIMETERW(*pwWordMinSel) && *pwWordMinSel != 0x0D)) /* If we are moving left or if we are moving right and we are not on a * space or a CR (the start of a word), then we was look left for the * start of a word which is either a CR or a character. We do this by * looking left till we find a character (or if CR we stop), then we * continue looking left till we find a space or LF. */ { while (pwWordMinSel > pwText && ((!ISDELIMETERW(*(pwWordMinSel - 1)) && *(pwWordMinSel - 1) != 0x0A) || !charLocated)) { /* * Treat double byte character as a word ( in unicode pwWordMinSel loop ) */ pwPrevChar = pwWordMinSel - 1; /* ** we are looking right ( !fLeft ). ** if current character is a double width chararacter ** or previous character is a double width character, ** we are on the beggining of a word. */ if (!fLeft && (ISDELIMETERW( *pwPrevChar) || *pwPrevChar == 0x0A || UserIsFullWidth(CP_ACP,*pwWordMinSel) || UserIsFullWidth(CP_ACP,*pwPrevChar))) { /* * If we are looking for the start of the word right, then we * stop when we have found it. (needed in case charLocated is * still FALSE) */ break; } if (UserIsFullWidth(CP_ACP,*pwPrevChar)) { /* ** Previous character is a double width character. ** if we are in a word ( charLocated == TRUE ) ** current position is the beginning of the word ** if we are not in a word ( charLocated == FALSE ) ** the previous character is what we looking for. */ if ( ! charLocated ) { pwWordMinSel = pwPrevChar; } break; } pwWordMinSel = pwPrevChar; if (!ISDELIMETERW(*pwWordMinSel) && *pwWordMinSel != 0x0A) /* * We have found the last char in the word. Continue looking * backwards till we find the first char of the word */ { charLocated = TRUE; /* * We will consider a CR the start of a word */ if (*pwWordMinSel == 0x0D) break; } } } else { /* * We are moving right and we are in between words so we need to move * right till we find the start of a word (either a CR or a character. */ while ((ISDELIMETERW(*pwWordMinSel) || *pwWordMinSel == 0x0A) && pwWordMinSel < pwText + ped->cch) pwWordMinSel++; } pwWordMaxSel = min((pwWordMinSel + 1), (pwText + ped->cch)); /* ** If pwWordMinSel points a double width character AND ** pwWordMaxSel points non space ** then ** pwWordMaxSel points the beggining of next word. */ if (UserIsFullWidth(CP_ACP,*pwWordMinSel) && ! ISDELIMETERW(*pwWordMaxSel)) goto FastReturnW; if (*pwWordMinSel == 0x0D) { if (pwWordMinSel > pwText && *(pwWordMinSel - 1) == 0x0D) /* So that we can treat CRCRLF as one word also. */ pwWordMinSel--; else if (*(pwWordMinSel + 1) == 0x0D) /* Move MaxSel on to the LF */ pwWordMaxSel++; } /* * Check if we have a one character word */ if (ISDELIMETERW(*pwWordMaxSel)) spaceLocated = TRUE; /* * Move pwWordMaxSel to the right looking for the end of a word and its * trailing spaces. WordMaxSel stops on the first character of the next * word. Thus, we break either at a CR or at the first nonspace char after * a run of spaces or LFs. */ while ((pwWordMaxSel < pwText + ped->cch) && (!spaceLocated || (ISDELIMETERW(*pwWordMaxSel)))) { if (*pwWordMaxSel == 0x0D) break; /* * treat double byte character as a word ( in unicode pwWordMaxSel loop ) */ /* ** if it's a double width character ** then we are at the beginning of ** the next word which is a double ** width character. */ if (UserIsFullWidth(CP_ACP,*pwWordMaxSel)) break; pwWordMaxSel++; if (ISDELIMETERW(*pwWordMaxSel)) spaceLocated = TRUE; if (*(pwWordMaxSel - 1) == 0x0A) break; } /* * label for fast return ( for Unicode ) */ FastReturnW: ECUnlock(ped); if (pichMin) *pichMin = (ICH)(pwWordMinSel - pwText); if (pichMax) *pichMax = (ICH)(pwWordMaxSel - pwText); return; } } /* * ECSaveUndo() - * Saves old undo information into given buffer, and clears out info in * passed in undo buffer. If we're restoring, pundoFrom and pundoTo are * reversed. */ void ECSaveUndo(PUNDO pundoFrom, PUNDO pundoTo, BOOL fClear) { /* * Save undo data */ RtlCopyMemory(pundoTo, pundoFrom, sizeof(UNDO)); /* * Clear passed in undo buffer */ if (fClear) RtlZeroMemory(pundoFrom, sizeof(UNDO) ); } /* * ECEmptyUndo AorW * empties the undo buffer. * History: */ void ECEmptyUndo( PUNDO pundo ) { if (pundo->hDeletedText) UserGlobalFree(pundo->hDeletedText); RtlZeroMemory(pundo, sizeof(UNDO) ); } /* * ECMergeUndoInsertInfo() - * When an insert takes place, this function is called with the info about * the new insertion (the insertion point and the count of chars inserted); * This looks at the existing Undo info and merges the new new insert info * with it. */ void ECMergeUndoInsertInfo(PUNDO pundo, ICH ichInsert, ICH cchInsert) \ { // If undo buffer is empty, just insert the new info as UNDO_INSERT if (pundo->undoType == UNDO_NONE) { pundo->undoType = UNDO_INSERT; pundo->ichInsStart = ichInsert; pundo->ichInsEnd = ichInsert+cchInsert; } else if (pundo->undoType & UNDO_INSERT) { // If there's already some undo insert info, // try to merge the two. if (pundo->ichInsEnd == ichInsert) // Check they are adjacent. pundo->ichInsEnd += cchInsert; // if so, just concatenate. else { // The new insert is not contiguous with the old one. UNDOINSERT: // If there is some UNDO_DELETE info already here, check to see // if the new insert takes place at a point different from where // that deletion occurred. if ((pundo->undoType & UNDO_DELETE) && (pundo->ichDeleted != ichInsert)) { // User is inserting into a different point; So, let us // forget any UNDO_DELETE info; if (pundo->hDeletedText) UserGlobalFree(pundo->hDeletedText); pundo->hDeletedText = NULL; pundo->ichDeleted = 0xFFFFFFFF; pundo->undoType &= ~UNDO_DELETE; } // Since the old insert and new insert are not adjacent, let us // forget everything about the old insert and keep just the new // insert info as the UNDO_INSERT. pundo->ichInsStart = ichInsert; pundo->ichInsEnd = ichInsert + cchInsert; pundo->undoType |= UNDO_INSERT; } } else if (pundo->undoType == UNDO_DELETE) { // If there is some Delete Info already present go and handle it. goto UNDOINSERT; } } /* * ECInsertText AorW * Adds cch characters from lpText into the ped->hText starting at * ped->ichCaret. Returns TRUE if successful else FALSE. Updates * ped->cchAlloc and ped->cch properly if additional memory was allocated or * if characters were actually added. Updates ped->ichCaret to be at the end * of the inserted text. min and maxsel are equal to ichcaret. * History: */ BOOL ECInsertText( PED ped, LPSTR lpText, ICH* pcchInsert) { PSTR pedText; PSTR pTextBuff; LONG style; HANDLE hTextCopy; DWORD allocamt; // If the last byte (lpText[cchInsert - 1]) is a DBCS leading byte // we need to adjust it. *pcchInsert = ECAdjustIch(ped, lpText, *pcchInsert); if (!*pcchInsert) return TRUE; /* * Do we already have enough memory?? */ if (*pcchInsert >= (ped->cchAlloc - ped->cch)) { /* * Allocate what we need plus a little extra. Return FALSE if we are * unsuccessful. */ allocamt = (ped->cch + *pcchInsert) * ped->cbChar; allocamt += CCHALLOCEXTRA; // if (!ped->fSingle) { hTextCopy = LOCALREALLOC(ped->hText, allocamt, LHND, ped->hInstance, &lpText); if (hTextCopy) { ped->hText = hTextCopy; } else { return FALSE; } // } else { // if (!LocalReallocSafe(ped->hText, allocamt, LHND, pped)) // return FALSE; // } ped->cchAlloc = LOCALSIZE(ped->hText, ped->hInstance) / ped->cbChar; } /* * Ok, we got the memory. Now copy the text into the structure */ pedText = ECLock(ped); if (ped->pLpkEditCallout) { HDC hdc; INT iResult; hdc = ECGetEditDC (ped, TRUE); iResult = ped->pLpkEditCallout->EditVerifyText (ped, hdc, pedText, ped->ichCaret, lpText, *pcchInsert); ECReleaseEditDC (ped, hdc, TRUE); if (iResult == 0) { ECUnlock (ped); return TRUE; } } /* * Get a pointer to the place where text is to be inserted */ pTextBuff = pedText + ped->ichCaret * ped->cbChar; if (ped->ichCaret != ped->cch) { /* * We are inserting text into the middle. We have to shift text to the * right before inserting new text. */ memmove(pTextBuff + *pcchInsert * ped->cbChar, pTextBuff, (ped->cch-ped->ichCaret) * ped->cbChar); } /* * Make a copy of the text being inserted in the edit buffer. * Use this copy for doing UPPERCASE/LOWERCASE ANSI/OEM conversions * Fix for Bug #3406 -- 01/29/91 -- SANKAR -- */ memmove(pTextBuff, lpText, *pcchInsert * ped->cbChar); ped->cch += *pcchInsert; /* * Get the control's style */ style = ped->pwnd->style; /* * Do the Upper/Lower conversion */ if (style & ES_LOWERCASE) { if (ped->fAnsi) CharLowerBuffA((LPSTR)pTextBuff, *pcchInsert); else CharLowerBuffW((LPWSTR)pTextBuff, *pcchInsert); } else { if (style & ES_UPPERCASE) { if (ped->fAnsi) { CharUpperBuffA(pTextBuff, *pcchInsert); } else { CharUpperBuffW((LPWSTR)pTextBuff, *pcchInsert); } } } /* * Do the OEM conversion */ if ((style & ES_OEMCONVERT) && // For backward compatibility with NT4, we don't perform OEM conversion // for older apps if the system locale is FarEast. (!IS_DBCS_ENABLED() || GETAPPVER() >= VER50 || GetOEMCP() != GetACP())) { ICH i; if (ped->fAnsi) { for (i = 0; i < *pcchInsert; i++) { // We don't need to call CharToOemBuff etc. if the character // is a double byte character. And, calling ECIsDBCSLeadByte is // faster and less complicated because we don't have to deal // with the 2 byte dbcs cases. if (IS_DBCS_ENABLED() && ECIsDBCSLeadByte(ped, *(lpText+i))) { i++; continue; } // Windows Bug (Whistler) 35289 // greek has funny rules for casing, so we need to check for it. // for nashville we should be doing something more appropriate // but for now, leave as Win95 golden if (ped->charSet != GREEK_CHARSET && IsCharLowerA(*(pTextBuff + i))) { CharUpperBuffA(pTextBuff + i, 1); CharToOemBuffA(pTextBuff + i, pTextBuff + i, 1); OemToCharBuffA(pTextBuff + i, pTextBuff + i, 1); CharLowerBuffA(pTextBuff + i, 1); } else { CharToOemBuffA(pTextBuff + i, pTextBuff + i, 1); OemToCharBuffA(pTextBuff + i, pTextBuff + i, 1); } } } else { // Because 'ch' may become DBCS, and have a space for NULL. UCHAR ch[4]; LPWSTR lpTextW = (LPWSTR)pTextBuff; for (i = 0; i < *pcchInsert; i++) { if (*(lpTextW + i) == UNICODE_CARRIAGERETURN || *(lpTextW + i) == UNICODE_LINEFEED || *(lpTextW + i) == UNICODE_TAB) { continue; } // Windows Bug (Whistler) 35289 // greek has funny rules for casing, so we need to check for it. // for nashville we should be doing something more appropriate // but for now, leave as Win95 golden if (ped->charSet != GREEK_CHARSET && IsCharLowerW(*(lpTextW + i))) { CharUpperBuffW(lpTextW + i, 1); *(LPDWORD)ch = 0; // make sure the null-terminate. CharToOemBuffW(lpTextW + i, ch, 1); // We assume any SBCS/DBCS character will converted // to 1 Unicode char, Otherwise, we may overwrite // next character... OemToCharBuffW(ch, lpTextW + i, strlen(ch)); CharLowerBuffW(lpTextW + i, 1); } else { *(LPDWORD)ch = 0; // make sure the null-terminate. CharToOemBuffW(lpTextW + i, ch, 1); // We assume any SBCS/DBCS character will converted // to 1 Unicode char, Otherwise, we may overwrite // next character... OemToCharBuffW(ch, lpTextW + i, strlen(ch)); } } } } /* Adjust UNDO fields so that we can undo this insert... */ ECMergeUndoInsertInfo(Pundo(ped), ped->ichCaret, *pcchInsert); ped->ichCaret += *pcchInsert; if (ped->pLpkEditCallout) { HDC hdc; hdc = ECGetEditDC (ped, TRUE); ped->ichCaret = ped->pLpkEditCallout->EditAdjustCaret (ped, hdc, pedText, ped->ichCaret); ECReleaseEditDC (ped, hdc, TRUE); } ped->ichMinSel = ped->ichMaxSel = ped->ichCaret; ECUnlock(ped); /* * Set dirty bit */ ped->fDirty = TRUE; return TRUE; } /* * ECDeleteText AorW * Deletes the text between ped->ichMinSel and ped->ichMaxSel. The * character at ichMaxSel is not deleted. But the character at ichMinSel is * deleted. ped->cch is updated properly and memory is deallocated if enough * text is removed. ped->ichMinSel, ped->ichMaxSel, and ped->ichCaret are set * to point to the original ped->ichMinSel. Returns the number of characters * deleted. * History: */ ICH ECDeleteText( PED ped) { PSTR pedText; ICH cchDelete; LPSTR lpDeleteSaveBuffer; HANDLE hDeletedText; DWORD bufferOffset; cchDelete = ped->ichMaxSel - ped->ichMinSel; if (!cchDelete) return (0); /* * Ok, now lets delete the text. */ pedText = ECLock(ped); /* * Adjust UNDO fields so that we can undo this delete... */ if (ped->undoType == UNDO_NONE) { UNDODELETEFROMSCRATCH: if (ped->hDeletedText = UserGlobalAlloc(GPTR, (LONG)((cchDelete+1)*ped->cbChar))) { ped->undoType = UNDO_DELETE; ped->ichDeleted = ped->ichMinSel; ped->cchDeleted = cchDelete; lpDeleteSaveBuffer = ped->hDeletedText; RtlCopyMemory(lpDeleteSaveBuffer, pedText + ped->ichMinSel*ped->cbChar, cchDelete*ped->cbChar); lpDeleteSaveBuffer[cchDelete*ped->cbChar] = 0; } } else if (ped->undoType & UNDO_INSERT) { UNDODELETE: ECEmptyUndo(Pundo(ped)); ped->ichInsStart = ped->ichInsEnd = 0xFFFFFFFF; ped->ichDeleted = 0xFFFFFFFF; ped->cchDeleted = 0; goto UNDODELETEFROMSCRATCH; } else if (ped->undoType == UNDO_DELETE) { if (ped->ichDeleted == ped->ichMaxSel) { /* * Copy deleted text to front of undo buffer */ hDeletedText = UserGlobalReAlloc(ped->hDeletedText, (LONG)(cchDelete + ped->cchDeleted + 1)*ped->cbChar, GHND); if (!hDeletedText) goto UNDODELETE; bufferOffset = 0; ped->ichDeleted = ped->ichMinSel; } else if (ped->ichDeleted == ped->ichMinSel) { /* * Copy deleted text to end of undo buffer */ hDeletedText = UserGlobalReAlloc(ped->hDeletedText, (LONG)(cchDelete + ped->cchDeleted + 1)*ped->cbChar, GHND); if (!hDeletedText) goto UNDODELETE; bufferOffset = ped->cchDeleted*ped->cbChar; } else { /* * Clear the current UNDO delete and add the new one since the deletes aren't contiguous. */ goto UNDODELETE; } ped->hDeletedText = hDeletedText; lpDeleteSaveBuffer = (LPSTR)hDeletedText; if (!bufferOffset) { /* * Move text in delete buffer up so that we can insert the next * text at the head of the buffer. */ RtlMoveMemory(lpDeleteSaveBuffer + cchDelete*ped->cbChar, lpDeleteSaveBuffer, ped->cchDeleted*ped->cbChar); } RtlCopyMemory(lpDeleteSaveBuffer + bufferOffset, pedText + ped->ichMinSel*ped->cbChar, cchDelete*ped->cbChar); lpDeleteSaveBuffer[(ped->cchDeleted + cchDelete)*ped->cbChar] = 0; ped->cchDeleted += cchDelete; } if (ped->ichMaxSel != ped->cch) { /* * We are deleting text from the middle of the buffer so we have to shift text to the left. */ RtlMoveMemory(pedText + ped->ichMinSel*ped->cbChar, pedText + ped->ichMaxSel*ped->cbChar, (ped->cch - ped->ichMaxSel)*ped->cbChar); } if (ped->cchAlloc - ped->cch > CCHALLOCEXTRA) { /* * Free some memory since we deleted a lot */ LOCALREALLOC(ped->hText, (DWORD)(ped->cch + (CCHALLOCEXTRA / 2))*ped->cbChar, LHND, ped->hInstance, NULL); ped->cchAlloc = LOCALSIZE(ped->hText, ped->hInstance) / ped->cbChar; } ped->cch -= cchDelete; if (ped->pLpkEditCallout) { HDC hdc; hdc = ECGetEditDC (ped, TRUE); ped->ichMinSel = ped->pLpkEditCallout->EditAdjustCaret (ped, hdc, pedText, ped->ichMinSel); ECReleaseEditDC (ped, hdc, TRUE); } ped->ichCaret = ped->ichMaxSel = ped->ichMinSel; ECUnlock(ped); /* * Set dirty bit */ ped->fDirty = TRUE; return (cchDelete); } /* * ECNotifyParent AorW * Sends the notification code to the parent of the edit control * History: */ void ECNotifyParent( PED ped, int notificationCode) { /* * wParam is NotificationCode (hiword) and WindowID (loword) * lParam is HWND of control sending the message * Windows 95 checks for hwndParent != NULL before sending the message, but * this is surely rare, and SendMessage NULL hwnd does nowt anyway (IanJa) */ SendMessage(ped->hwndParent, WM_COMMAND, (DWORD)MAKELONG(PTR_TO_ID(ped->pwnd->spmenu), notificationCode), (LPARAM)ped->hwnd); } /* * ECSetEditClip() AorW * Sets the clip rect for the hdc to the formatting rectangle intersected * with the client area. */ void ECSetEditClip(PED ped, HDC hdc, BOOL fLeftMargin) { RECT rcClient; RECT rcClip; CopyRect(&rcClip, &ped->rcFmt); if (ped->pLpkEditCallout) { // Complex script handling chooses whether to write margins later rcClip.left -= ped->wLeftMargin; rcClip.right += ped->wRightMargin; } else { if (fLeftMargin) /* Should we consider the left margin? */ rcClip.left -= ped->wLeftMargin; if (ped->fWrap) /* Should we consider the right margin? */ rcClip.right += ped->wRightMargin; } /* Set clip rectangle to rectClient intersect rectClip */ /* We must clip for single line edits also. -- B#1360 */ _GetClientRect(ped->pwnd, &rcClient); if (ped->fFlatBorder) InflateRect(&rcClient, -SYSMET(CXBORDER), -SYSMET(CYBORDER)); IntersectRect(&rcClient, &rcClient, &rcClip); IntersectClipRect(hdc,rcClient.left, rcClient.top, rcClient.right, rcClient.bottom); } /* * ECGetEditDC AorW * Hides the caret, gets the DC for the edit control, and clips to * the rcFmt rectangle specified for the edit control and sets the proper * font. If fFastDC, just select the proper font but don't bother about clip * regions or hiding the caret. * History: */ HDC ECGetEditDC( PED ped, BOOL fFastDC ) { HDC hdc; if (!fFastDC) NtUserHideCaret(ped->hwnd); if ( hdc = NtUserGetDC(ped->hwnd) ) { ECSetEditClip(ped, hdc, (BOOL)(ped->xOffset == 0)); /* * Select the proper font for this edit control's dc. */ if (ped->hFont) SelectObject(hdc, ped->hFont); } return hdc; } /* * ECReleaseEditDC AorW * Releases the DC (hdc) for the edit control and shows the caret. * If fFastDC, just select the proper font but don't bother about showing the * caret. * History: */ void ECReleaseEditDC( PED ped, HDC hdc, BOOL fFastDC) { /* * Restoring font not necessary */ ReleaseDC(ped->hwnd, hdc); if (!fFastDC) NtUserShowCaret(ped->hwnd); } /* * ECResetTextInfo() AorW * Handles a global change to the text by resetting text offsets, emptying * the undo buffer, and rebuilding the lines */ void ECResetTextInfo(PED ped) { // Reset caret, selections, scrolling, and dirty information. ped->iCaretLine = ped->ichCaret = 0; ped->ichMinSel = ped->ichMaxSel = 0; ped->xOffset = ped->ichScreenStart = 0; ped->fDirty = FALSE; ECEmptyUndo(Pundo(ped)); if (ped->fSingle) { if (!ped->listboxHwnd) ECNotifyParent(ped, EN_UPDATE); } else { #ifdef BOGUS // B#14640 // We don't want to strip soft breaks or anything else from text // that was passed in by the caller. - karlst. MLStripCrCrLf(ped); #endif MLBuildchLines(ped, 0, 0, FALSE, NULL, NULL); } if (_IsWindowVisible(ped->pwnd)) { BOOL fErase; if (ped->fSingle) fErase = FALSE; else fErase = ((ped->ichLinesOnScreen + ped->ichScreenStart) >= ped->cLines); // Always redraw whether or not the insert was successful. We might // have NULL text. Paint() will check the redraw flag for us. ECInvalidateClient(ped, fErase); // BACKWARD COMPAT HACK: RAID expects the text to have been updated, // so we have to do an UpdateWindow here. It moves an edit control // around with fRedraw == FALSE, so it'll never get the paint message // with the control in the right place. if (!ped->fWin31Compat) UpdateWindow(ped->hwnd); } if (ped->fSingle && !ped->listboxHwnd) ECNotifyParent(ped, EN_CHANGE); if (FWINABLE()) { NotifyWinEvent(EVENT_OBJECT_VALUECHANGE, HW(ped->pwnd), OBJID_CLIENT, INDEXID_CONTAINER); } } /* * ECSetText AorW * Copies the null terminated text in lpstr to the ped. Notifies the * parent if there isn't enough memory. Sets the minsel, maxsel, and caret to * the beginning of the inserted text. Returns TRUE if successful else FALSE * if no memory (and notifies the parent). * History: */ BOOL ECSetText( PED ped, LPSTR lpstr) { ICH cchLength; ICH cchSave = ped->cch; ICH ichCaretSave = ped->ichCaret; HWND hwndSave = ped->hwnd; HANDLE hText; ped->cch = ped->ichCaret = 0; ped->cchAlloc = LOCALSIZE(ped->hText, ped->hInstance) / ped->cbChar; if (!lpstr) { hText = LOCALREALLOC(ped->hText, CCHALLOCEXTRA*ped->cbChar, LHND, ped->hInstance, &lpstr); if (hText != NULL) { ped->hText = hText; } else { return FALSE; } } else { cchLength = StringLength(lpstr, ped->fAnsi); #ifdef NEVER // win3.1 does limit single line edit controls to 32K (minus 3) but NT doesn't if (ped->fSingle) { /* * Limit single line edit controls to 32K */ cchLength = min(cchLength, (ICH)(0x7FFD/ped->cbChar)); } #endif /* * Add the text */ if (cchLength && !ECInsertText(ped, lpstr, &cchLength)) { /* * Restore original state and notify parent we ran out of memory. */ ped->cch = cchSave; ped->ichCaret = ichCaretSave; ECNotifyParent(ped, EN_ERRSPACE); return FALSE; } } ped->cchAlloc = LOCALSIZE(ped->hText, ped->hInstance) / ped->cbChar; if (IsWindow(hwndSave)) ECResetTextInfo(ped); return TRUE; } /* * ECInvalidateClient() * Invalidates client of edit field. For old 3.x guys with borders, * we draw it ourself (compatibility). So we don't want to invalidate * the border or we'll get flicker. */ void ECInvalidateClient(PED ped, BOOL fErase) { if (ped->fFlatBorder) { RECT rcT; _GetClientRect(ped->pwnd, &rcT); InflateRect(&rcT, -SYSMET(CXBORDER), -SYSMET(CYBORDER)); NtUserInvalidateRect(ped->hwnd, &rcT, fErase); } else { NtUserInvalidateRect(ped->hwnd, NULL, fErase); } } /* * ECCopy AorW * Copies the text between ichMinSel and ichMaxSel to the clipboard. * Returns the number of characters copied. * History: */ ICH ECCopy( PED ped) { HANDLE hData; char *pchSel; char FAR *lpchClip; ICH cbData; /* * Don't allow copies from password style controls */ if (ped->charPasswordChar) { NtUserMessageBeep(0); return 0; } cbData = (ped->ichMaxSel - ped->ichMinSel) * ped->cbChar; if (!cbData) return 0; if (!OpenClipboard(ped->hwnd)) return 0; NtUserEmptyClipboard(); /* * If we just called EmptyClipboard in the context of a 16 bit * app then we also have to tell WOW to nix its 16 handle copy of * clipboard data. WOW does its own clipboard caching because * some 16 bit apps use clipboard data even after the clipboard * has been emptied. See the note in the server code. * Note: this is the only place where EmptyClipboard is called * for a 16 bit app not going through WOW. If we added others * we might want to move this into EmptyClipboard and have two * versions. */ if (GetClientInfo()->CI_flags & CI_16BIT) { pfnWowEmptyClipBoard(); } /* * +1 for the terminating NULL */ if (!(hData = UserGlobalAlloc(LHND, (LONG)(cbData + ped->cbChar)))) { NtUserCloseClipboard(); return (0); } USERGLOBALLOCK(hData, lpchClip); UserAssert(lpchClip); pchSel = ECLock(ped); pchSel = pchSel + (ped->ichMinSel * ped->cbChar); RtlCopyMemory(lpchClip, pchSel, cbData); if (ped->fAnsi) *(lpchClip + cbData) = 0; else *(LPWSTR)(lpchClip + cbData) = (WCHAR)0; ECUnlock(ped); USERGLOBALUNLOCK(hData); SetClipboardData( ped->fAnsi ? CF_TEXT : CF_UNICODETEXT, hData); NtUserCloseClipboard(); return (cbData); } /* * EditWndProcA * Always receives Ansi messages and translates them if appropriate to unicode * depending on the PED type */ LRESULT EditWndProcA( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { PWND pwnd; if ((pwnd = ValidateHwnd(hwnd)) == NULL) return 0; /* * If the control is not interested in this message, * pass it to DefWindowProc. */ if (!FWINDOWMSG(message, FNID_EDIT)) return DefWindowProcWorker(pwnd, message, wParam, lParam, TRUE); return EditWndProcWorker(pwnd, message, wParam, lParam, TRUE); } LRESULT EditWndProcW( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { PWND pwnd; if ((pwnd = ValidateHwnd(hwnd)) == NULL) return 0; /* * If the control is not interested in this message, * pass it to DefWindowProc. */ if (!FWINDOWMSG(message, FNID_EDIT)) { return DefWindowProcWorker(pwnd, message, wParam, lParam, FALSE); } return EditWndProcWorker(pwnd, message, wParam, lParam, FALSE); } LRESULT EditWndProcWorker( PWND pwnd, UINT message, WPARAM wParam, LPARAM lParam, DWORD fAnsi) { PED ped; HWND hwnd = HWq(pwnd); static BOOL fInit = TRUE; VALIDATECLASSANDSIZE(pwnd, FNID_EDIT); INITCONTROLLOOKASIDE(&EditLookaside, ED, pwnd, 4); /* * Get the ped for the given window now since we will use it a lot in * various handlers. This was stored using SetWindowLong(hwnd,0,hped) when * we initially created the edit control. */ ped = ((PEDITWND)pwnd)->ped; /* * Make sure the ANSI flag is set correctly. */ if (!ped->fInitialized) { ped->fInitialized = TRUE; ped->fAnsi = TestWF(pwnd, WFANSICREATOR) ? TRUE : FALSE; } /* * We just call the regular EditWndProc if the ped is not created, the * incoming message type already matches the PED type or the message * does not need any translation. */ if (ped->fAnsi == fAnsi || (message >= WM_USER) || !MessageTable[message].bThunkMessage) { return EditWndProc(pwnd, message, wParam, lParam); } return CsSendMessage(hwnd, message, wParam, lParam, fAnsi ? (ULONG_PTR)EditWndProcW : (ULONG_PTR)EditWndProcA, FNID_CALLWINDOWPROC, fAnsi); } /* * EditWndProc * Class procedure for all edit controls. * Dispatches all messages to the appropriate handlers which are named * as follows: * SL (single line) prefixes all single line edit control procedures while * ML (multi line) prefixes all multi- line edit controls. * EC (edit control) prefixes all common handlers. * The EditWndProc only handles messages common to both single and multi * line edit controls. Messages which are handled differently between * single and multi are sent to SLEditWndProc or MLEditWndProc. * Top level procedures are EditWndPoc, SLEditWndProc, and MLEditWndProc. * SL*Handler or ML*Handler or EC*Handler procs are called to handle * the various messages. Support procedures are prefixed with SL ML or * EC depending on which code they support. They are never called * directly and most assumptions/effects are documented in the effects * clause. * WARNING: If you add a message here, add it to gawEditWndProc[] in * kernel\server.c too, otherwise EditWndProcA/W will send it straight to * DefWindowProcWorker * History: */ LRESULT EditWndProc( PWND pwnd, UINT message, WPARAM wParam, LPARAM lParam) { HWND hwnd = HWq(pwnd); LRESULT lreturn; PED ped; /* * Get the ped for the given window now since we will use it a lot in * various handlers. This was stored using SetWindowLong(hwnd,0,hped) when * we initially created the edit control. */ ped = ((PEDITWND)pwnd)->ped; /* * Dispatch the various messages we can receive */ lreturn = 1L; switch (message) { /* * Messages which are handled the same way for both single and multi line * edit controls. */ case WM_KEYDOWN: // LPK handling of Ctrl/LShift, Ctrl/RShift if (ped && ped->pLpkEditCallout && ped->fAllowRTL) { ped->fSwapRoOnUp = FALSE; // Any keydown cancels a ctrl/shift reading order change switch (wParam) { case VK_SHIFT: if ((GetKeyState(VK_CONTROL) & 0x8000) && !(GetKeyState(VK_MENU) & 0x8000)) { // Left shift or right shift pressed while control held down // Check that alt (VK_MENU) isn't down to avoid false firing on AltGr which equals Ctrl+Alt. if (MapVirtualKey((LONG)lParam>>16&0xff, 3) == VK_LSHIFT) { // User wants left to right reading order ped->fSwapRoOnUp = (ped->fRtoLReading) || (ped->format & ES_RIGHT) ; ped->fLShift = TRUE; } else { // User wants right to left reading order ped->fSwapRoOnUp = (!ped->fRtoLReading) || (ped->format & ES_RIGHT); ped->fLShift = FALSE; } } break; case VK_LEFT: if (ped->fRtoLReading) { wParam = VK_RIGHT; } break; case VK_RIGHT: if (ped->fRtoLReading) { wParam = VK_LEFT; } break; } } goto HandleEditMsg; case WM_KEYUP: if (ped && ped->pLpkEditCallout && ped->fAllowRTL && ped->fSwapRoOnUp) { BOOL fReadingOrder; // Complete reading order change detected earlier during keydown ped->fSwapRoOnUp = FALSE; fReadingOrder = ped->fRtoLReading; // Remove any overriding ES_CENTRE or ES_RIGHT format from dwStyle SetWindowLong(hwnd, GWL_STYLE, GetWindowLong(hwnd, GWL_STYLE) & ~ES_FMTMASK); if (ped->fLShift) { // Set Left to Right reading order and right scrollbar in EX_STYLE SetWindowLong(hwnd, GWL_EXSTYLE, GetWindowLong(hwnd, GWL_EXSTYLE) & ~(WS_EX_RTLREADING | WS_EX_RIGHT | WS_EX_LEFTSCROLLBAR)); // Edit control is LTR now, then notify the parent. ECNotifyParent(ped, EN_ALIGN_LTR_EC); // ? Select a keyboard layout appropriate to LTR operation } else { // Set Right to Left reading order, right alignment and left scrollbar SetWindowLong(hwnd, GWL_EXSTYLE, GetWindowLong(hwnd, GWL_EXSTYLE) | WS_EX_RTLREADING | WS_EX_RIGHT | WS_EX_LEFTSCROLLBAR); // Edit control is RTL now, then notify the parent. ECNotifyParent(ped, EN_ALIGN_RTL_EC); // ? Select a keyboard layout appropriate to RTL operation } // If reading order didn't change, so we are sure the alignment changed and the edit window didn't invalidate yet. if (fReadingOrder == (BOOL) ped->fRtoLReading) { ECInvalidateClient(ped, TRUE); } } goto HandleEditMsg; case WM_INPUTLANGCHANGE: if (ped) { // EC_INSERT_COMPOSITION_CHAR : WM_INPUTLANGCHANGE - call ECInitInsert() HKL hkl = THREAD_HKL(); ECInitInsert(ped, hkl); if (ped->fInReconversion) { ECInOutReconversionMode(ped, FALSE); } // Font and caret position might be changed while // another keyboard layout is active. Set those // if the edit control has the focus. if (ped->fFocus && fpImmIsIME(hkl)) { POINT pt; ECImmSetCompositionFont(ped); NtUserGetCaretPos(&pt); ECImmSetCompositionWindow(ped, pt.x, pt.y); } } goto HandleEditMsg; case WM_COPY: /* * wParam - not used * lParam - not used */ lreturn = (LONG)ECCopy(ped); break; case WM_CUT: /* * wParamLo -- unused * lParam -- unused */ ECCutText(ped); return 0; case WM_CLEAR: /* * wParamLo -- unused * lParam -- unused */ ECClearText(ped); return 0; case WM_ENABLE: /* * wParam - nonzero if window is enabled else disable window if 0. * lParam - not used */ lreturn = (LONG)(ped->fDisabled = !((BOOL)wParam)); ECInvalidateClient(ped, TRUE); break; case WM_SYSCHAR: // wParamLo -- key value // lParam -- unused // If this is a WM_SYSCHAR message generated by the UNDO // keystroke we want to EAT IT if ((lParam & SYS_ALTERNATE) && ((WORD)wParam == VK_BACK)) return TRUE; else { return DefWindowProcWorker(pwnd, message, wParam, lParam, ped->fAnsi); } break; case EM_GETLINECOUNT: /* * wParam - not used lParam - not used */ lreturn = (LONG)ped->cLines; break; case EM_GETMODIFY: /* * wParam - not used lParam - not used */ /* * Gets the state of the modify flag for this edit control. */ lreturn = (LONG)ped->fDirty; break; case EM_SETMODIFY: /* * wParam - specifies the new value for the modify flag lParam - not used */ /* * Sets the state of the modify flag for this edit control. */ ped->fDirty = (wParam != 0); break; case EM_GETRECT: /* * wParam - not used lParam - pointer to a RECT data structure that gets the dimensions. */ /* * Copies the rcFmt rect to *lpRect. */ CopyRect((LPRECT)lParam, (LPRECT)&ped->rcFmt); lreturn = (LONG)TRUE; break; case WM_GETFONT: /* * wParam - not used lParam - not used */ lreturn = (LRESULT)ped->hFont; break; case WM_SETFONT: /* * wParam - handle to the font lParam - redraw if true else don't */ ECSetFont(ped, (HANDLE)wParam, (BOOL)LOWORD(lParam)); break; case WM_GETTEXT: /* * wParam - max number of _bytes_ (not characters) to copy * lParam - buffer to copy text to. Text is 0 terminated. */ lreturn = (LRESULT)ECGetText(ped, (ICH)wParam, (LPSTR)lParam, TRUE); break; case WM_SETTEXT: // wParamLo -- unused // lParam -- LPSTR, null-terminated, with new text. lreturn = (LRESULT)ECSetText(ped, (LPSTR)lParam); break; case WM_GETTEXTLENGTH: /* * Return count of CHARs!!! */ lreturn = (LONG)ped->cch; break; case WM_NCDESTROY: case WM_FINALDESTROY: /* * wParam - not used lParam - not used */ ECNcDestroyHandler(pwnd, ped); return 0; /* * Most apps (i.e. everyone but Quicken) don't pass on the rbutton * messages when they do something with 'em inside of subclassed * edit fields. As such, we keep track of whether we saw the * down before the up. If we don't see the up, then DefWindowProc * won't generate the context menu message, so no big deal. If * we didn't see the down, then don't let WM_CONTEXTMENU do * anything. * We also might want to not generate WM_CONTEXTMENUs for old * apps when the mouse is captured. */ case WM_RBUTTONDOWN: ped->fSawRButtonDown = TRUE; goto HandleEditMsg; case WM_RBUTTONUP: if (ped->fSawRButtonDown) { ped->fSawRButtonDown = FALSE; if (!ped->fInReconversion) { goto HandleEditMsg; } } // Don't pass this on to DWP so WM_CONTEXTMENU isn't generated. return 0; case WM_CONTEXTMENU: { POINT pt ; int nHit = FindNCHit(pwnd, (LONG)lParam); if ((nHit == HTVSCROLL) || (nHit == HTHSCROLL)) { return DefWindowProcWorker(pwnd, message, wParam, lParam, ped->fAnsi); } POINTSTOPOINT(pt, lParam); if (!TestWF(pwnd, WFOLDUI) && ECIsAncestorActive(hwnd)) ECMenu(hwnd, ped, &pt); } return 0; case EM_CANUNDO: /* * wParam - not used lParam - not used */ lreturn = (LONG)(ped->undoType != UNDO_NONE); break; case EM_EMPTYUNDOBUFFER: /* * wParam - not used lParam - not used */ ECEmptyUndo(Pundo(ped)); break; case EM_GETMARGINS: // wParam -- unused // lParam -- unused return(MAKELONG(ped->wLeftMargin, ped->wRightMargin)); case EM_SETMARGINS: // wParam -- EC_ margin flags // lParam -- LOWORD is left, HIWORD is right margin ECSetMargin(ped, (UINT)wParam, (DWORD)lParam, TRUE); return 0; case EM_GETSEL: /* * Gets the selection range for the given edit control. The * starting position is in the low order word. It contains the position * of the first nonselected character after the end of the selection in * the high order word. */ if ((PDWORD)wParam != NULL) { *((PDWORD)wParam) = ped->ichMinSel; } if ((PDWORD)lParam != NULL) { *((PDWORD)lParam) = ped->ichMaxSel; } lreturn = MAKELONG(ped->ichMinSel,ped->ichMaxSel); break; case EM_GETLIMITTEXT: // wParamLo -- unused // lParam -- unused return(ped->cchTextMax); case EM_SETLIMITTEXT: /* Renamed from EM_LIMITTEXT in Chicago */ /* * wParam - max number of CHARACTERS that can be entered * lParam - not used */ /* * Specifies the maximum number of characters of text the user may * enter. If maxLength is 0, we may enter MAXINT number of CHARACTERS. */ if (ped->fSingle) { if (wParam) { wParam = min(0x7FFFFFFEu, wParam); } else { wParam = 0x7FFFFFFEu; } } if (wParam) { ped->cchTextMax = (ICH)wParam; } else { ped->cchTextMax = 0xFFFFFFFFu; } break; case EM_POSFROMCHAR: // Validate that char index is within text range if (wParam >= ped->cch) { return(-1L); } goto HandleEditMsg; case EM_CHARFROMPOS: { // Validate that point is within client of edit field RECT rc; POINT pt; POINTSTOPOINT(pt, lParam); _GetClientRect(pwnd, &rc); if (!PtInRect(&rc, pt)) { return(-1L); } goto HandleEditMsg; } case EM_SETPASSWORDCHAR: /* * wParam - sepecifies the new char to display instead of the * real text. if null, display the real text. */ ECSetPasswordChar(ped, (UINT)wParam); break; case EM_GETPASSWORDCHAR: lreturn = (DWORD)ped->charPasswordChar; break; case EM_SETREADONLY: /* * wParam - state to set read only flag to */ ped->fReadOnly = (wParam != 0); if (wParam) SetWindowState(pwnd, EFREADONLY); else ClearWindowState(pwnd, EFREADONLY); lreturn = 1L; ECEnableDisableIME( ped ); // We need to redraw the edit field so that the background color // changes. Read-only edits are drawn in CTLCOLOR_STATIC while // others are drawn with CTLCOLOR_EDIT. ECInvalidateClient(ped, TRUE); break; case EM_SETWORDBREAKPROC: /* * wParam - unused * lParam - FARPROC address of an app supplied call back function */ ped->lpfnNextWord = (EDITWORDBREAKPROCA)lParam; break; case EM_GETWORDBREAKPROC: lreturn = (LRESULT)ped->lpfnNextWord; break; // IME case EM_GETIMESTATUS: // wParam == sub command switch (wParam) { case EMSIS_COMPOSITIONSTRING: return ped->wImeStatus; #if 0 // memphis case EMSIS_GETLBBIT: return (DWORD)ped->bLBBit; #endif } break; case EM_SETIMESTATUS: // wParam == sub command switch (wParam) { case EMSIS_COMPOSITIONSTRING: ped->wImeStatus = (WORD)lParam; } break; case WM_NCCREATE: lreturn = ECNcCreate(ped, pwnd, (LPCREATESTRUCT)lParam); break; case WM_LBUTTONDOWN: // B#3623 // Don't set focus to edit field if it is within an inactive, // captioned child. // We might want to version switch this... I haven't found // any problems by not, but you never know... if (ECIsAncestorActive(hwnd)) { /* * Reconversion support: quit reconversion if left button is clicked. * Otherwise, if the current KL is Korean, finailize the composition string. */ if (ped->fInReconversion || ped->fKorea) { BOOLEAN fReconversion = (BOOLEAN)ped->fInReconversion; DWORD dwIndex = fReconversion ? CPS_CANCEL : CPS_COMPLETE; HIMC hImc; ped->fReplaceCompChr = FALSE; hImc = fpImmGetContext(ped->hwnd); if (hImc) { fpImmNotifyIME(hImc, NI_COMPOSITIONSTR, dwIndex, 0); fpImmReleaseContext(ped->hwnd, hImc); } if (fReconversion) { ECInOutReconversionMode(ped, FALSE); } ECSetCaretHandler(ped); } goto HandleEditMsg; } break; case WM_MOUSEMOVE: // We only care about mouse messages when mouse is down. if (ped->fMouseDown) goto HandleEditMsg; break; case WM_IME_SETCONTEXT: // If ped->fInsertCompChr is TRUE, that means we will do // all the composition character drawing by ourself. if ( ped->fInsertCompChr ) { lParam &= ~ISC_SHOWUICOMPOSITIONWINDOW; } if ( wParam ) { PINPUTCONTEXT pInputContext; HIMC hImc; hImc = fpImmGetContext( hwnd ); if ( (pInputContext = fpImmLockIMC( hImc )) != NULL ) { pInputContext->fdw31Compat &= ~F31COMPAT_ECSETCFS; fpImmUnlockIMC( hImc ); } if (GetClientInfo()->CI_flags & CI_16BIT) { fpImmNotifyIME(hImc, NI_COMPOSITIONSTR, CPS_CANCEL, 0L); } fpImmReleaseContext( hwnd, hImc ); } return DefWindowProcWorker(pwnd, message, wParam, lParam, ped->fAnsi); case WM_IME_ENDCOMPOSITION: ECInOutReconversionMode(ped, FALSE); if (ped->fReplaceCompChr) { ICH ich; HDC hdc; // we have a DBCS character to be replaced. // let's delete it before inserting the new one. ich = (ped->fAnsi) ? 2 : 1; ped->fReplaceCompChr = FALSE; ped->ichMaxSel = min(ped->ichCaret + ich, ped->cch); ped->ichMinSel = ped->ichCaret; if (ped->fSingle) { if (ECDeleteText( ped ) > 0) { // Update the display ECNotifyParent(ped, EN_UPDATE); hdc = ECGetEditDC(ped, FALSE); SLDrawText(ped, hdc, 0); ECReleaseEditDC(ped, hdc, FALSE); // Tell parent our text contents changed. ECNotifyParent(ped, EN_CHANGE); } } else { MLDeleteText(ped); } ECSetCaretHandler( ped ); } return DefWindowProcWorker(pwnd, message, wParam, lParam, ped->fAnsi); case WM_IME_STARTCOMPOSITION: if ( ped->fInsertCompChr ) { // BUG BUG // sending WM_IME_xxxCOMPOSITION will let // IME draw composition window. IME should // not do that since we cleared // ISC_SHOWUICOMPOSITIONWINDOW bit when // we got WM_IME_SETCONTEXT message. // Korean IME should be fixed in the future. break; } else { return DefWindowProcWorker(pwnd, message, wParam, lParam, ped->fAnsi); } // simple composition character support for FE IME. case WM_IME_COMPOSITION: return ECImeComposition(ped, wParam, lParam); case WM_KILLFOCUS: // when focus is removed from the window, // composition character should be finalized if (ped && fpImmIsIME(THREAD_HKL())) { HIMC hImc = fpImmGetContext(hwnd); if (hImc != NULL_HIMC) { if (ped->fReplaceCompChr || (ped->wImeStatus & EIMES_COMPLETECOMPSTRKILLFOCUS)) { // If the composition string to be determined upon kill focus, // do it now. fpImmNotifyIME(hImc, NI_COMPOSITIONSTR, CPS_COMPLETE, 0); } else if (ped->fInReconversion) { // If the composition string it not to be determined, // and if we're in reconversion mode, cancel reconversion now. fpImmNotifyIME(hImc, NI_COMPOSITIONSTR, CPS_CANCEL, 0); } // Get out from reconversion mode if (ped->fInReconversion) { ECInOutReconversionMode(ped, FALSE); } fpImmReleaseContext(hwnd, hImc); } } goto HandleEditMsg; break; case WM_SETFOCUS: if (ped && !ped->fFocus) { HKL hkl = THREAD_HKL(); if (fpImmIsIME(hkl)) { HIMC hImc; hImc = fpImmGetContext(hwnd); if (hImc) { LPINPUTCONTEXT lpImc; if (ped->wImeStatus & EIMES_CANCELCOMPSTRINFOCUS) { // cancel when in-focus fpImmNotifyIME(hImc, NI_COMPOSITIONSTR, CPS_CANCEL, 0); } ECImmSetCompositionFont(ped); if ((lpImc = fpImmLockIMC(hImc)) != NULL) { // We presume the CompForm will reset to CFS_DEFAULT, // when the edit control loses Focus. // IMEWndProc32 will call ImmSetCompositionWindow with // CFS_DEFAULT, when it receive WM_IME_SETCONTEXT. lpImc->fdw31Compat |= F31COMPAT_ECSETCFS; fpImmUnlockIMC(hImc); } fpImmReleaseContext(hwnd, hImc); } // force to set IME composition window when // first getting focus. ped->ptScreenBounding.x = -1; ped->ptScreenBounding.y = -1; } ECInitInsert(ped, hkl); } goto HandleEditMsg; break; case WM_IME_REQUEST: // simple ImeRequest Handler return EcImeRequestHandler(ped, wParam, lParam); case WM_CREATE: if (ped) ECEnableDisableIME(ped); goto HandleEditMsg; break; default: HandleEditMsg: /* (picked up from NT40FE SP3) * HACK ALERT: We may receive messages before the PED has been * allocated (eg: WM_GETMINMAXINFO is sent before WM_NCCREATE) * so we must test ped before dreferencing. */ if (ped != NULL) { if (ped->fSingle) { lreturn = SLEditWndProc(hwnd, ped, message, wParam, lParam); } else { lreturn = MLEditWndProc(hwnd, ped, message, wParam, lParam); } } } return lreturn; } /* * ECFindXORblks * This finds the XOR of lpOldBlk and lpNewBlk and return s resulting blocks * through the lpBlk1 and lpBlk2; This could result in a single block or * at the maximum two blocks; * If a resulting block is empty, then it's StPos field has -1. * NOTE: * When called from MultiLine edit control, StPos and EndPos fields of * these blocks have the Starting line and Ending line of the block; * When called from SingleLine edit control, StPos and EndPos fields * of these blocks have the character index of starting position and * ending position of the block. * History: */ void ECFindXORblks( LPBLOCK lpOldBlk, LPBLOCK lpNewBlk, LPBLOCK lpBlk1, LPBLOCK lpBlk2) { if (lpOldBlk->StPos >= lpNewBlk->StPos) { lpBlk1->StPos = lpNewBlk->StPos; lpBlk1->EndPos = min(lpOldBlk->StPos, lpNewBlk->EndPos); } else { lpBlk1->StPos = lpOldBlk->StPos; lpBlk1->EndPos = min(lpNewBlk->StPos, lpOldBlk->EndPos); } if (lpOldBlk->EndPos <= lpNewBlk->EndPos) { lpBlk2->StPos = max(lpOldBlk->EndPos, lpNewBlk->StPos); lpBlk2->EndPos = lpNewBlk->EndPos; } else { lpBlk2->StPos = max(lpNewBlk->EndPos, lpOldBlk->StPos); lpBlk2->EndPos = lpOldBlk->EndPos; } } /* * ECCalcChangeSelection * This function finds the XOR between two selection blocks(OldBlk and NewBlk) * and return s the resulting areas thro the same parameters; If the XOR of * both the blocks is empty, then this return s FALSE; Otherwise TRUE. * NOTE: * When called from MultiLine edit control, StPos and EndPos fields of * these blocks have the Starting line and Ending line of the block; * When called from SingleLine edit control, StPos and EndPos fields * of these blocks have the character index of starting position and * ending position of the block. * History: */ BOOL ECCalcChangeSelection( PED ped, ICH ichOldMinSel, ICH ichOldMaxSel, LPBLOCK OldBlk, LPBLOCK NewBlk) { BLOCK Blk[2]; int iBlkCount = 0; Blk[0].StPos = Blk[0].EndPos = Blk[1].StPos = Blk[1].EndPos = 0xFFFFFFFF; /* * Check if the Old selection block existed */ if (ichOldMinSel != ichOldMaxSel) { /* * Yes! Old block existed. */ Blk[0].StPos = OldBlk->StPos; Blk[0].EndPos = OldBlk->EndPos; iBlkCount++; } /* * Check if the new Selection block exists */ if (ped->ichMinSel != ped->ichMaxSel) { /* * Yes! New block exists */ Blk[1].StPos = NewBlk->StPos; Blk[1].EndPos = NewBlk->EndPos; iBlkCount++; } /* * If both the blocks exist find the XOR of them */ if (iBlkCount == 2) { /* * Check if both blocks start at the same character position */ if (ichOldMinSel == ped->ichMinSel) { /* * Check if they end at the same character position */ if (ichOldMaxSel == ped->ichMaxSel) return FALSE; /* Nothing changes */ Blk[0].StPos = min(NewBlk -> EndPos, OldBlk -> EndPos); Blk[0].EndPos = max(NewBlk -> EndPos, OldBlk -> EndPos); Blk[1].StPos = 0xFFFFFFFF; } else { if (ichOldMaxSel == ped->ichMaxSel) { Blk[0].StPos = min(NewBlk->StPos, OldBlk->StPos); Blk[0].EndPos = max(NewBlk->StPos, OldBlk->StPos); Blk[1].StPos = 0xFFFFFFFF; } else { ECFindXORblks(OldBlk, NewBlk, &Blk[0], &Blk[1]); } } } RtlCopyMemory(OldBlk, &Blk[0], sizeof(BLOCK)); RtlCopyMemory(NewBlk, &Blk[1], sizeof(BLOCK)); return TRUE; /* Yup , There is something to paint */ } /* * ECGetControlBrush * Client side optimization replacement for NtUserGetControlBrush * message is one of the WM_CTLCOLOR* messages. */ HBRUSH ECGetControlBrush( PED ped, HDC hdc, LONG message) { PWND pwndSend; PWND pwndEdit; pwndEdit = ValidateHwnd(ped->hwnd); if (pwndEdit == (PWND)NULL) return (HBRUSH)0; if ((pwndSend = (TestwndPopup(pwndEdit) ? pwndEdit->spwndOwner : pwndEdit->spwndParent)) == NULL) pwndSend = pwndEdit; else pwndSend = REBASEPTR(pwndEdit, pwndSend); UserAssert(pwndSend); if (PtiCurrent() != GETPTI(pwndSend)) { return (HBRUSH)DefWindowProcWorker(pwndSend, message, (WPARAM)hdc, (LPARAM)pwndEdit, ped->fAnsi); } /* * By using the correct A/W call we avoid a c/s transition * on this SendMessage(). */ return (HBRUSH)SendMessageWorker(pwndSend, message, (WPARAM)hdc, (LPARAM)ped->hwnd, ped->fAnsi); } UINT WINAPI QueryFontAssocStatus(void); UINT fFontAssocStatus = 0xffff; /* * ECGetDBCSVector( PED ped, BYTE CharSet ) * This function sets DBCS Vector for specified character set and sets * ped->fDBCS flag if needed. * History: 18-Jun-1996 Hideyuki Nagase */ int ECGetDBCSVector(PED ped, HDC hdc, BYTE CharSet) { BOOL bDBCSCodePage = FALSE; /* * if DEFAUT_CHARSET was passed, we will convert that to Shell charset.. */ if (CharSet == DEFAULT_CHARSET) { CharSet = (BYTE)GetTextCharset(hdc); /* * if CharSet is still DEFAULT_CHARSET, it means gdi has some problem.. * then just return default.. we get charset from CP_ACP.. */ if (CharSet == DEFAULT_CHARSET) { CharSet = (BYTE)GetACPCharSet(); } } switch (CharSet) { case SHIFTJIS_CHARSET: case HANGEUL_CHARSET: case CHINESEBIG5_CHARSET: case GB2312_CHARSET: bDBCSCodePage = TRUE; break; case ANSI_CHARSET: // 0 case SYMBOL_CHARSET: // 2 case OEM_CHARSET: // 255 if (fFontAssocStatus == 0xffff) fFontAssocStatus = QueryFontAssocStatus(); if ((((CharSet + 2) & 0xf) & fFontAssocStatus)) { bDBCSCodePage = TRUE; /* * Bug 117558, etc. * Try to get a meaningful character set for associated font. */ CharSet = (BYTE)GetACPCharSet(); } else { bDBCSCodePage = FALSE; } break; default: bDBCSCodePage = FALSE; } if (bDBCSCodePage) { CHARSETINFO CharsetInfo; DWORD CodePage; CPINFO CPInfo; int lbIX; if (TranslateCharsetInfo((DWORD *)CharSet, &CharsetInfo, TCI_SRCCHARSET)) { CodePage = CharsetInfo.ciACP; } else { CodePage = CP_ACP; } GetCPInfo(CodePage, &CPInfo); for (lbIX=0 ; CPInfo.LeadByte[lbIX] != 0 ; lbIX+=2) { ped->DBCSVector[lbIX ] = CPInfo.LeadByte[lbIX]; ped->DBCSVector[lbIX+1] = CPInfo.LeadByte[lbIX+1]; } ped->DBCSVector[lbIX ] = 0x0; ped->DBCSVector[lbIX+1] = 0x0; } else { ped->DBCSVector[0] = 0x0; ped->DBCSVector[1] = 0x0; } // Final check: if the font supports DBCS glyphs // If we've got a font with DBCS glyphs, let's mark PED so. // But since the font's primary charset is the one other than FE, // we can only support UNICODE Edit control. // a) GDI performs A/W conversion for ANSI apps based on the primary // character set in hDC, so it will break anyway. // b) ANSI applications are only supported on their native system locales: // GetACPCharSet() is expected to return a FE code page. // c) ANSI Edit control requires DBCSVector, which cannot be // initialized without a FE code page. if (!ped->fAnsi) { FONTSIGNATURE fontSig; GetTextCharsetInfo(hdc, &fontSig, 0); if (fontSig.fsCsb[0] & FAREAST_CHARSET_BITS) { bDBCSCodePage = TRUE; // Since this is UNICODE, we're not } } return bDBCSCodePage; } /* * LPSTR ECAnsiNext( ped, lpCurrent ) * This function advances string pointer for Edit Control use only. * History: */ LPSTR ECAnsiNext(PED ped, LPSTR lpCurrent) { return lpCurrent+((ECIsDBCSLeadByte(ped,*lpCurrent)==TRUE) ? 2 : 1); } /* * LPSTR ECAnsiPrev( ped, lpBase, lpStr ) * This function decrements string pointer for Edit Control use only. * History: */ LPSTR ECAnsiPrev(PED ped, LPSTR lpBase, LPSTR lpStr ) { LPSTR lpCurrent = lpStr -1; if (!ped->fDBCS) return lpCurrent; // just return ( lpStr - 1 ) if (lpBase >= lpCurrent) return lpBase; if (ECIsDBCSLeadByte(ped, *lpCurrent)) // this check makes things faster return (lpCurrent - 1); // 92/04/04 takaok do { lpCurrent--; if (!ECIsDBCSLeadByte(ped, *lpCurrent)) { lpCurrent++; break; } } while(lpCurrent != lpBase); return lpStr - (((lpStr - lpCurrent) & 1) ? 1 : 2); } /* * ICH ECNextIch( ped, pText, ichCurrent ) * This function advances string pointer for Edit Control use only. * History: */ ICH ECNextIch( PED ped, LPSTR pStart, ICH ichCurrent ) { if (!ped->fDBCS || !ped->fAnsi) { return (ichCurrent + 1); } else { ICH ichRet; LPSTR pText; if (pStart) pText = pStart + ichCurrent; else pText = (LPSTR)ECLock(ped) + ichCurrent; ichRet = ichCurrent + ( ECIsDBCSLeadByte(ped, *pText) ? 2 : 1 ); if (!pStart) ECUnlock(ped); return (ichRet); } } /* * ICH ECPrevIch( ped, LPSTR pStart, ICH ichCurrent ) * This function decrements string pointer for Edit Control use only. * History: */ ICH ECPrevIch( PED ped, LPSTR pStart, ICH ichCurrent ) { LPSTR lpCurrent; LPSTR lpStr; LPSTR lpBase; #ifdef SURROGATE // Handle Unicode surrogates pairs when CSLPK is loaded if (ped->fAnsi || !ped->pLpkEditCallout) // if no surrogate processing required #endif if (!ped->fDBCS || !ped->fAnsi) if ( ichCurrent ) return (ichCurrent - 1); else return (ichCurrent); if (ichCurrent <= 1) return 0; if (pStart) lpBase = pStart; else lpBase = ECLock(ped); #ifdef SURROGATE // Handle characters represented by multiple codepoints if (ped->fAnsi) { // ANSI PrevIch with DBCS support #endif lpStr = lpBase + ichCurrent; lpCurrent = lpStr - 1; if (ECIsDBCSLeadByte(ped,*lpCurrent)) { if (!pStart) ECUnlock(ped); return (ichCurrent - 2); } do { lpCurrent--; if (!ECIsDBCSLeadByte(ped, *lpCurrent)) { lpCurrent++; break; } } while(lpCurrent != lpBase); if (!pStart) ECUnlock(ped); return (ichCurrent - (((lpStr - lpCurrent) & 1) ? 1 : 2)); #ifdef SURROGATE } else { // Unicode PrevIch with surrogate pair support ichCurrent--; if ( (((WCHAR*)lpBase)[ichCurrent] & 0xFC00) == 0xDC00 && (((WCHAR*)lpBase)[ichCurrent-1] & 0xFC00) == 0xD800) { ichCurrent--; } if (!pStart) ECUnlock(ped); return ichCurrent; } #endif } /* * BOOL ECIsDBCSLeadByte( PED ped, BYTE cch ) * IsDBCSLeadByte for Edit Control use only. * History: 18-Jun-1996 Hideyuki Nagase */ BOOL ECIsDBCSLeadByte(PED ped, BYTE cch) { int i; if (!ped->fDBCS || !ped->fAnsi) return (FALSE); for (i = 0; ped->DBCSVector[i]; i += 2) { if ((ped->DBCSVector[i] <= cch) && (ped->DBCSVector[i+1] >= cch)) return (TRUE); } return (FALSE); } /* * int DBCSCombine(HWND hwnd, int ch) * Assemble two WM_CHAR messages to single DBCS character. * If program detects first byte of DBCS character in WM_CHAR message, * it calls this function to obtain second WM_CHAR message from queue. * finally this routine assembles first byte and second byte into single * DBCS character. * History: */ WORD DbcsCombine(HWND hwnd, WORD ch) { MSG msg; int i = 10; /* loop counter to avoid the infinite loop */ while (!PeekMessageA(&msg, hwnd, WM_CHAR, WM_CHAR, PM_REMOVE)) { if (--i == 0) return 0; Sleep(1); } return (WORD)ch | ((WORD)(msg.wParam) << 8); } /* * ICH ECAdjustIch( PED ped, LPSTR lpstr, ICH ch ) * This function adjusts a current pointer correctly. If a current * pointer is lying between DBCS first byte and second byte, this * function adjusts a current pointer to a first byte of DBCS position * by decrement once. * History: */ ICH ECAdjustIch( PED ped, LPSTR lpstr, ICH ch ) { ICH newch = ch; if (!ped->fAnsi || !ped->fDBCS || newch == 0) return ( ch ); if (!ECIsDBCSLeadByte(ped,lpstr[--newch])) return ( ch ); // previous char is SBCS while(1) { if (!ECIsDBCSLeadByte(ped,lpstr[newch])) { newch++; break; } if (newch) newch--; else break; } return ((ch - newch) & 1) ? ch-1 : ch; } /* * ICH ECAdjustIchNext( PED ped, LPSTR lpstr, ICH ch ) * History: * 19.Jun.1996 Hideyuki Nagase [hideyukn] - Port from Win95-FarEast version */ ICH FAR PASCAL ECAdjustIchNext(PED ped, LPSTR lpstr, ICH ch) { ICH ichNew = ECAdjustIch(ped,lpstr,ch); LPSTR lpnew = lpstr+ichNew; // if ch > ichNew then ECAdjustIch adjusted ich. if (ch > ichNew) lpnew = ECAnsiNext(ped, lpnew); return (ICH)(lpnew-lpstr); } /* * ECUpdateFormat * Computes ped->format and ped->fRtoLReading from dwStyle and dwExStyle. * Refreshes the display if either are changed. * History: * May 12, 1997 [samera] wrote it * May 12, 1997 [dbrown] rewrote it */ void ECUpdateFormat( PED ped, DWORD dwStyle, DWORD dwExStyle) { UINT fNewRtoLReading; UINT uiNewFormat; // Extract new format and reading order from style fNewRtoLReading = dwExStyle & WS_EX_RTLREADING ? 1 : 0; uiNewFormat = dwStyle & ES_FMTMASK; // WS_EX_RIGHT is ignored unless dwStyle is ES_LEFT if (uiNewFormat == ES_LEFT && dwExStyle & WS_EX_RIGHT) { uiNewFormat = ES_RIGHT; } // Internally ES_LEFT and ES_RIGHT are swapped for RtoLReading order // (Think of them as ES_LEADING and ES_TRAILING) if (fNewRtoLReading) { switch (uiNewFormat) { case ES_LEFT: uiNewFormat = ES_RIGHT; break; case ES_RIGHT: uiNewFormat = ES_LEFT; break; } } // Format change does not cause redisplay by itself ped->format = uiNewFormat; // Refresh display on change of reading order if (fNewRtoLReading != ped->fRtoLReading) { ped->fRtoLReading = fNewRtoLReading; if (ped->fWrap) { // Redo wordwrap MLBuildchLines(ped, 0, 0, FALSE, NULL, NULL); MLUpdateiCaretLine(ped); } else { // Refresh horizontal scrollbar display MLScroll(ped, FALSE, 0xffffffff, 0, TRUE); } ECInvalidateClient(ped, TRUE); } }