| /* |
| * Theming - Combo box control |
| * |
| * Copyright (c) 2005 by Frank Richter |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| * |
| */ |
| |
| #include <stdarg.h> |
| #include <string.h> |
| #include <stdlib.h> |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "wingdi.h" |
| #include "winuser.h" |
| #include "uxtheme.h" |
| #include "tmschema.h" |
| #include "comctl32.h" |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(themingcombo); |
| |
| /* Subclass-private state flags */ |
| #define STATE_NOREDRAW 1 |
| #define STATE_HOT 2 |
| |
| /* some constants for metrics, same as in user32 */ |
| #define COMBO_XBORDERSIZE 2 |
| #define COMBO_YBORDERSIZE 2 |
| #define COMBO_EDITBUTTONSPACE 0 |
| #define EDIT_CONTROL_PADDING 1 |
| |
| /* paint text of combobox, needed for read-only drop downs. */ |
| static void paint_text (HWND hwnd, HDC hdc, DWORD dwStyle, COMBOBOXINFO* cbi) |
| { |
| INT id, size = 0; |
| LPWSTR pText = NULL; |
| UINT itemState = ODS_COMBOBOXEDIT; |
| HFONT font = (HFONT)SendMessageW (hwnd, WM_GETFONT, 0, 0); |
| HFONT hPrevFont = (font) ? SelectObject(hdc, font) : 0; |
| RECT rectEdit; |
| BOOL focused = GetFocus () == hwnd; |
| BOOL dropped = cbi->stateButton == STATE_SYSTEM_PRESSED; |
| |
| TRACE("\n"); |
| |
| /* follow Windows combobox that sends a bunch of text |
| * inquiries to its listbox while processing WM_PAINT. */ |
| |
| if( (id = SendMessageW (cbi->hwndList, LB_GETCURSEL, 0, 0) ) != LB_ERR ) |
| { |
| size = SendMessageW (cbi->hwndList, LB_GETTEXTLEN, id, 0); |
| if (size == LB_ERR) |
| FIXME("LB_ERR probably not handled yet\n"); |
| if( (pText = HeapAlloc( GetProcessHeap(), 0, (size + 1) * sizeof(WCHAR))) ) |
| { |
| /* size from LB_GETTEXTLEN may be too large, from LB_GETTEXT is accurate */ |
| size=SendMessageW (cbi->hwndList, LB_GETTEXT, (WPARAM)id, (LPARAM)pText); |
| pText[size] = '\0'; /* just in case */ |
| } else return; |
| } |
| else |
| if( !(dwStyle & (CBS_OWNERDRAWFIXED | CBS_OWNERDRAWVARIABLE)) ) |
| return; |
| |
| /* |
| * Give ourselves some space. |
| */ |
| CopyRect (&rectEdit, &cbi->rcItem); |
| InflateRect( &rectEdit, -1, -1 ); |
| |
| if(dwStyle & (CBS_OWNERDRAWFIXED | CBS_OWNERDRAWVARIABLE)) |
| { |
| DRAWITEMSTRUCT dis; |
| HRGN clipRegion; |
| UINT ctlid = (UINT)GetWindowLongPtrW( hwnd, GWLP_ID ); |
| |
| /* setup state for DRAWITEM message. Owner will highlight */ |
| if ( focused && !dropped ) |
| itemState |= ODS_SELECTED | ODS_FOCUS; |
| |
| /* |
| * Save the current clip region. |
| * To retrieve the clip region, we need to create one "dummy" |
| * clip region. |
| */ |
| clipRegion = CreateRectRgnIndirect(&rectEdit); |
| |
| if (GetClipRgn(hdc, clipRegion)!=1) |
| { |
| DeleteObject(clipRegion); |
| clipRegion=NULL; |
| } |
| |
| if (!IsWindowEnabled(hwnd) & WS_DISABLED) itemState |= ODS_DISABLED; |
| |
| dis.CtlType = ODT_COMBOBOX; |
| dis.CtlID = ctlid; |
| dis.hwndItem = hwnd; |
| dis.itemAction = ODA_DRAWENTIRE; |
| dis.itemID = id; |
| dis.itemState = itemState; |
| dis.hDC = hdc; |
| dis.rcItem = rectEdit; |
| dis.itemData = SendMessageW(cbi->hwndList, LB_GETITEMDATA, |
| (WPARAM)id, 0 ); |
| |
| /* |
| * Clip the DC and have the parent draw the item. |
| */ |
| IntersectClipRect(hdc, |
| rectEdit.left, rectEdit.top, |
| rectEdit.right, rectEdit.bottom); |
| |
| SendMessageW(GetParent (hwnd), WM_DRAWITEM, ctlid, (LPARAM)&dis ); |
| |
| /* |
| * Reset the clipping region. |
| */ |
| SelectClipRgn(hdc, clipRegion); |
| } |
| else |
| { |
| static const WCHAR empty_stringW[] = { 0 }; |
| |
| if ( focused && !dropped ) { |
| |
| /* highlight */ |
| FillRect( hdc, &rectEdit, GetSysColorBrush(COLOR_HIGHLIGHT) ); |
| SetBkColor( hdc, GetSysColor( COLOR_HIGHLIGHT ) ); |
| SetTextColor( hdc, GetSysColor( COLOR_HIGHLIGHTTEXT ) ); |
| } |
| |
| ExtTextOutW( hdc, |
| rectEdit.left + 1, |
| rectEdit.top + 1, |
| ETO_OPAQUE | ETO_CLIPPED, |
| &rectEdit, |
| pText ? pText : empty_stringW , size, NULL ); |
| |
| if ( focused && !dropped ) |
| DrawFocusRect( hdc, &rectEdit ); |
| } |
| |
| if( hPrevFont ) |
| SelectObject(hdc, hPrevFont ); |
| |
| HeapFree( GetProcessHeap(), 0, pText ); |
| } |
| |
| /* paint the combobox */ |
| static LRESULT paint (HTHEME theme, HWND hwnd, HDC hParamDC, ULONG state) |
| { |
| PAINTSTRUCT ps; |
| HDC hDC; |
| COMBOBOXINFO cbi; |
| DWORD dwStyle = GetWindowLongW (hwnd, GWL_STYLE); |
| |
| hDC = (hParamDC) ? hParamDC |
| : BeginPaint( hwnd, &ps); |
| |
| TRACE("hdc=%p\n", hDC); |
| |
| if( hDC && !(state & STATE_NOREDRAW) ) |
| { |
| RECT frameRect; |
| int buttonState; |
| |
| cbi.cbSize = sizeof (cbi); |
| SendMessageW (hwnd, CB_GETCOMBOBOXINFO, 0, (LPARAM)&cbi); |
| |
| /* paint border */ |
| if ((dwStyle & CBS_DROPDOWNLIST) != CBS_SIMPLE) |
| GetClientRect (hwnd, &frameRect); |
| else |
| { |
| CopyRect (&frameRect, &cbi.rcItem); |
| |
| InflateRect(&frameRect, |
| EDIT_CONTROL_PADDING + COMBO_XBORDERSIZE, |
| EDIT_CONTROL_PADDING + COMBO_YBORDERSIZE); |
| } |
| |
| DrawThemeBackground (theme, hDC, 0, |
| IsWindowEnabled (hwnd) ? CBXS_NORMAL : CBXS_DISABLED, &frameRect, NULL); |
| |
| /* paint button */ |
| if (cbi.stateButton != STATE_SYSTEM_INVISIBLE) |
| { |
| if (!IsWindowEnabled (hwnd)) |
| buttonState = CBXS_DISABLED; |
| else if (cbi.stateButton == STATE_SYSTEM_PRESSED) |
| buttonState = CBXS_PRESSED; |
| else if (state & STATE_HOT) |
| buttonState = CBXS_HOT; |
| else |
| buttonState = CBXS_NORMAL; |
| DrawThemeBackground (theme, hDC, CP_DROPDOWNBUTTON, buttonState, |
| &cbi.rcButton, NULL); |
| } |
| |
| /* paint text, if we need to */ |
| if ((dwStyle & CBS_DROPDOWNLIST) == CBS_DROPDOWNLIST) |
| paint_text (hwnd, hDC, dwStyle, &cbi); |
| } |
| |
| if( !hParamDC ) |
| EndPaint(hwnd, &ps); |
| |
| return 0; |
| } |
| |
| |
| /********************************************************************** |
| * The combo control subclass window proc. |
| */ |
| LRESULT CALLBACK THEMING_ComboSubclassProc (HWND hwnd, UINT msg, |
| WPARAM wParam, LPARAM lParam, |
| ULONG_PTR dwRefData) |
| { |
| const WCHAR* themeClass = WC_COMBOBOXW; |
| HTHEME theme; |
| LRESULT result; |
| |
| switch (msg) |
| { |
| case WM_CREATE: |
| result = THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| OpenThemeData( hwnd, themeClass ); |
| return result; |
| |
| case WM_DESTROY: |
| theme = GetWindowTheme( hwnd ); |
| CloseThemeData ( theme ); |
| return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| |
| case WM_THEMECHANGED: |
| theme = GetWindowTheme( hwnd ); |
| CloseThemeData ( theme ); |
| OpenThemeData( hwnd, themeClass ); |
| break; |
| |
| case WM_SYSCOLORCHANGE: |
| theme = GetWindowTheme( hwnd ); |
| if (!theme) return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| /* Do nothing. When themed, a WM_THEMECHANGED will be received, too, |
| * which will do the repaint. */ |
| break; |
| |
| case WM_PAINT: |
| theme = GetWindowTheme( hwnd ); |
| if (!theme) return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| return paint (theme, hwnd, (HDC)wParam, dwRefData); |
| |
| case WM_SETREDRAW: |
| /* Since there doesn't seem to be WM_GETREDRAW, do redraw tracking in |
| * the subclass as well. */ |
| if( wParam ) |
| dwRefData &= ~STATE_NOREDRAW; |
| else |
| dwRefData |= STATE_NOREDRAW; |
| THEMING_SetSubclassData (hwnd, dwRefData); |
| return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| |
| case WM_MOUSEMOVE: |
| { |
| /* Dropdown button hot-tracking */ |
| COMBOBOXINFO cbi; |
| POINT pt = {LOWORD(lParam), HIWORD(lParam)}; |
| |
| cbi.cbSize = sizeof (cbi); |
| SendMessageW (hwnd, CB_GETCOMBOBOXINFO, 0, (LPARAM)&cbi); |
| |
| if (cbi.stateButton != STATE_SYSTEM_INVISIBLE) |
| { |
| if (PtInRect (&cbi.rcButton, pt)) |
| { |
| if (!(dwRefData & STATE_HOT)) |
| { |
| dwRefData |= STATE_HOT; |
| THEMING_SetSubclassData (hwnd, dwRefData); |
| RedrawWindow (hwnd, &cbi.rcButton, 0, |
| RDW_INVALIDATE | RDW_UPDATENOW); |
| } |
| } |
| else |
| { |
| if (dwRefData & STATE_HOT) |
| { |
| dwRefData &= ~STATE_HOT; |
| THEMING_SetSubclassData (hwnd, dwRefData); |
| RedrawWindow (hwnd, &cbi.rcButton, 0, |
| RDW_INVALIDATE | RDW_UPDATENOW); |
| } |
| } |
| } |
| } |
| return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| |
| default: |
| /* Call old proc */ |
| return THEMING_CallOriginalClass (hwnd, msg, wParam, lParam); |
| } |
| return 0; |
| } |