/* * Caret functions * * Copyright 1993 David Metcalfe * Copyright 1996 Frans van Dorsselaer */ #include "winuser.h" #include "module.h" #include "debug.h" DEFAULT_DEBUG_CHANNEL(caret) typedef struct { HWND hwnd; UINT hidden; BOOL on; INT x; INT y; INT width; INT height; HBRUSH16 hBrush; UINT timeout; UINT timerid; } CARET; typedef enum { CARET_OFF = 0, CARET_ON, CARET_TOGGLE } DISPLAY_CARET; static CARET Caret = { 0, 0, FALSE, 0, 0, 2, 12, 0, 500, 0 }; /***************************************************************** * CARET_GetHwnd */ HWND CARET_GetHwnd(void) { return Caret.hwnd; } /***************************************************************** * CARET_GetRect */ void CARET_GetRect(LPRECT lprc) { lprc->right = (lprc->left = Caret.x) + Caret.width - 1; lprc->bottom = (lprc->top = Caret.y) + Caret.height - 1; } /***************************************************************** * CARET_DisplayCaret */ static void CARET_DisplayCaret( DISPLAY_CARET status ) { HDC hdc; HBRUSH16 hPrevBrush; if (Caret.on && (status == CARET_ON)) return; if (!Caret.on && (status == CARET_OFF)) return; /* So now it's always a toggle */ Caret.on = !Caret.on; /* do not use DCX_CACHE here, for x,y,width,height are in logical units */ if (!(hdc = GetDCEx( Caret.hwnd, 0, DCX_USESTYLE /*| DCX_CACHE*/ ))) return; hPrevBrush = SelectObject( hdc, Caret.hBrush ); PatBlt( hdc, Caret.x, Caret.y, Caret.width, Caret.height, PATINVERT ); SelectObject( hdc, hPrevBrush ); ReleaseDC( Caret.hwnd, hdc ); } /***************************************************************** * CARET_Callback */ static VOID CALLBACK CARET_Callback( HWND hwnd, UINT msg, UINT id, DWORD ctime) { TRACE(caret,"hwnd=%04x, timerid=%d, caret=%d\n", hwnd, id, Caret.on); CARET_DisplayCaret(CARET_TOGGLE); } /***************************************************************** * CARET_SetTimer */ static void CARET_SetTimer(void) { if (Caret.timerid) KillSystemTimer( (HWND)0, Caret.timerid ); Caret.timerid = SetSystemTimer( (HWND)0, 0, Caret.timeout, CARET_Callback ); } /***************************************************************** * CARET_ResetTimer */ static void CARET_ResetTimer(void) { if (Caret.timerid) { KillSystemTimer( (HWND)0, Caret.timerid ); Caret.timerid = SetSystemTimer( (HWND)0, 0, Caret.timeout, CARET_Callback ); } } /***************************************************************** * CARET_KillTimer */ static void CARET_KillTimer(void) { if (Caret.timerid) { KillSystemTimer( (HWND)0, Caret.timerid ); Caret.timerid = 0; } } /***************************************************************** * CreateCaret16 (USER.163) */ void WINAPI CreateCaret16( HWND16 hwnd, HBITMAP16 bitmap, INT16 width, INT16 height ) { CreateCaret( hwnd, bitmap, width, height ); } /***************************************************************** * CreateCaret32 (USER32.66) */ BOOL WINAPI CreateCaret( HWND hwnd, HBITMAP bitmap, INT width, INT height ) { TRACE(caret,"hwnd=%04x\n", hwnd); if (!hwnd) return FALSE; /* if cursor already exists, destroy it */ if (Caret.hwnd) DestroyCaret(); if (bitmap && (bitmap != 1)) { BITMAP16 bmp; if (!GetObject16( bitmap, sizeof(bmp), &bmp )) return FALSE; Caret.width = bmp.bmWidth; Caret.height = bmp.bmHeight; /* FIXME: we should make a copy of the bitmap instead of a brush */ Caret.hBrush = CreatePatternBrush( bitmap ); } else { Caret.width = width ? width : GetSystemMetrics(SM_CXBORDER); Caret.height = height ? height : GetSystemMetrics(SM_CYBORDER); Caret.hBrush = CreateSolidBrush(bitmap ? GetSysColor(COLOR_GRAYTEXT) : GetSysColor(COLOR_WINDOW) ); } Caret.hwnd = hwnd; Caret.hidden = 1; Caret.on = FALSE; Caret.x = 0; Caret.y = 0; Caret.timeout = GetProfileIntA( "windows", "CursorBlinkRate", 500 ); return TRUE; } /***************************************************************** * DestroyCaret16 (USER.164) */ void WINAPI DestroyCaret16(void) { DestroyCaret(); } /***************************************************************** * DestroyCaret32 (USER32.131) */ BOOL WINAPI DestroyCaret(void) { if (!Caret.hwnd) return FALSE; TRACE(caret,"hwnd=%04x, timerid=%d\n", Caret.hwnd, Caret.timerid); CARET_KillTimer(); CARET_DisplayCaret(CARET_OFF); DeleteObject( Caret.hBrush ); Caret.hwnd = 0; return TRUE; } /***************************************************************** * SetCaretPos16 (USER.165) */ void WINAPI SetCaretPos16( INT16 x, INT16 y ) { SetCaretPos( x, y ); } /***************************************************************** * SetCaretPos32 (USER32.466) */ BOOL WINAPI SetCaretPos( INT x, INT y) { if (!Caret.hwnd) return FALSE; if ((x == Caret.x) && (y == Caret.y)) return TRUE; TRACE(caret,"x=%d, y=%d\n", x, y); CARET_KillTimer(); CARET_DisplayCaret(CARET_OFF); Caret.x = x; Caret.y = y; if (!Caret.hidden) { CARET_DisplayCaret(CARET_ON); CARET_SetTimer(); } return TRUE; } /***************************************************************** * HideCaret16 (USER.166) */ void WINAPI HideCaret16( HWND16 hwnd ) { HideCaret( hwnd ); } /***************************************************************** * HideCaret32 (USER32.317) */ BOOL WINAPI HideCaret( HWND hwnd ) { if (!Caret.hwnd) return FALSE; if (hwnd && (Caret.hwnd != hwnd)) return FALSE; TRACE(caret,"hwnd=%04x, hidden=%d\n", hwnd, Caret.hidden); CARET_KillTimer(); CARET_DisplayCaret(CARET_OFF); Caret.hidden++; return TRUE; } /***************************************************************** * ShowCaret16 (USER.167) */ void WINAPI ShowCaret16( HWND16 hwnd ) { ShowCaret( hwnd ); } /***************************************************************** * ShowCaret32 (USER32.529) */ BOOL WINAPI ShowCaret( HWND hwnd ) { if (!Caret.hwnd) return FALSE; if (hwnd && (Caret.hwnd != hwnd)) return FALSE; TRACE(caret,"hwnd=%04x, hidden=%d\n", hwnd, Caret.hidden); if (Caret.hidden) { Caret.hidden--; if (!Caret.hidden) { CARET_DisplayCaret(CARET_ON); CARET_SetTimer(); } } return TRUE; } /***************************************************************** * SetCaretBlinkTime16 (USER.168) */ void WINAPI SetCaretBlinkTime16( UINT16 msecs ) { SetCaretBlinkTime( msecs ); } /***************************************************************** * SetCaretBlinkTime32 (USER32.465) */ BOOL WINAPI SetCaretBlinkTime( UINT msecs ) { if (!Caret.hwnd) return FALSE; TRACE(caret,"hwnd=%04x, msecs=%d\n", Caret.hwnd, msecs); Caret.timeout = msecs; CARET_ResetTimer(); return TRUE; } /***************************************************************** * GetCaretBlinkTime16 (USER.169) */ UINT16 WINAPI GetCaretBlinkTime16(void) { return (UINT16)GetCaretBlinkTime(); } /***************************************************************** * GetCaretBlinkTime32 (USER32.209) */ UINT WINAPI GetCaretBlinkTime(void) { return Caret.timeout; } /***************************************************************** * GetCaretPos16 (USER.183) */ VOID WINAPI GetCaretPos16( LPPOINT16 pt ) { if (!Caret.hwnd || !pt) return; TRACE(caret,"hwnd=%04x, pt=%p, x=%d, y=%d\n", Caret.hwnd, pt, Caret.x, Caret.y); pt->x = (INT16)Caret.x; pt->y = (INT16)Caret.y; } /***************************************************************** * GetCaretPos32 (USER32.210) */ BOOL WINAPI GetCaretPos( LPPOINT pt ) { if (!Caret.hwnd || !pt) return FALSE; pt->x = Caret.x; pt->y = Caret.y; return TRUE; }