]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: bmpbuttn.cpp | |
3 | // Purpose: wxBitmapButton | |
4 | // Author: Julian Smart | |
5 | // Modified by: | |
6 | // Created: 04/01/98 | |
7 | // RCS-ID: $Id$ | |
8 | // Copyright: (c) Julian Smart and Markus Holzem | |
9 | // Licence: wxWindows license | |
10 | ///////////////////////////////////////////////////////////////////////////// | |
11 | ||
12 | #ifdef __GNUG__ | |
13 | #pragma implementation "bmpbuttn.h" | |
14 | #endif | |
15 | ||
16 | // For compilers that support precompilation, includes "wx.h". | |
17 | #include "wx/wxprec.h" | |
18 | ||
19 | #ifdef __BORLANDC__ | |
20 | #pragma hdrstop | |
21 | #endif | |
22 | ||
23 | #ifndef WX_PRECOMP | |
24 | #include "wx/bmpbuttn.h" | |
25 | #include "wx/log.h" | |
26 | #endif | |
27 | ||
28 | #include "wx/msw/private.h" | |
29 | ||
30 | IMPLEMENT_DYNAMIC_CLASS(wxBitmapButton, wxButton) | |
31 | ||
32 | #define BUTTON_HEIGHT_FACTOR (EDIT_CONTROL_FACTOR * 1.1) | |
33 | ||
34 | bool wxBitmapButton::Create(wxWindow *parent, wxWindowID id, const wxBitmap& bitmap, | |
35 | const wxPoint& pos, | |
36 | const wxSize& size, long style, | |
37 | const wxValidator& validator, | |
38 | const wxString& name) | |
39 | { | |
40 | m_buttonBitmap = bitmap; | |
41 | SetName(name); | |
42 | ||
43 | #if wxUSE_VALIDATORS | |
44 | SetValidator(validator); | |
45 | #endif // wxUSE_VALIDATORS | |
46 | ||
47 | parent->AddChild(this); | |
48 | ||
49 | m_backgroundColour = parent->GetBackgroundColour() ; | |
50 | m_foregroundColour = parent->GetForegroundColour() ; | |
51 | m_windowStyle = style; | |
52 | m_marginX = 0; | |
53 | m_marginY = 0; | |
54 | ||
55 | if ( style & wxBU_AUTODRAW ) | |
56 | { | |
57 | m_marginX = wxDEFAULT_BUTTON_MARGIN; | |
58 | m_marginY = wxDEFAULT_BUTTON_MARGIN; | |
59 | } | |
60 | ||
61 | int x = pos.x; | |
62 | int y = pos.y; | |
63 | int width = size.x; | |
64 | int height = size.y; | |
65 | ||
66 | if (id == -1) | |
67 | m_windowId = NewControlId(); | |
68 | else | |
69 | m_windowId = id; | |
70 | ||
71 | if ( width == -1 && bitmap.Ok()) | |
72 | width = bitmap.GetWidth() + 2*m_marginX; | |
73 | ||
74 | if ( height == -1 && bitmap.Ok()) | |
75 | height = bitmap.GetHeight() + 2*m_marginY; | |
76 | ||
77 | m_hWnd = (WXHWND)CreateWindowEx | |
78 | ( | |
79 | 0, | |
80 | wxT("BUTTON"), | |
81 | wxT(""), | |
82 | WS_VISIBLE | WS_TABSTOP | WS_CHILD | BS_OWNERDRAW , | |
83 | 0, 0, 0, 0, | |
84 | GetWinHwnd(parent), | |
85 | (HMENU)m_windowId, | |
86 | wxGetInstance(), | |
87 | NULL | |
88 | ); | |
89 | ||
90 | // Subclass again for purposes of dialog editing mode | |
91 | SubclassWin(m_hWnd); | |
92 | ||
93 | SetFont(parent->GetFont()) ; | |
94 | ||
95 | SetSize(x, y, width, height); | |
96 | ||
97 | return TRUE; | |
98 | } | |
99 | ||
100 | void wxBitmapButton::SetBitmapLabel(const wxBitmap& bitmap) | |
101 | { | |
102 | m_buttonBitmap = bitmap; | |
103 | } | |
104 | ||
105 | bool wxBitmapButton::MSWOnDraw(WXDRAWITEMSTRUCT *item) | |
106 | { | |
107 | #if defined(__WIN95__) | |
108 | long style = GetWindowLong((HWND) GetHWND(), GWL_STYLE); | |
109 | if (style & BS_BITMAP) | |
110 | { | |
111 | // Let default procedure draw the bitmap, which is defined | |
112 | // in the Windows resource. | |
113 | return FALSE; | |
114 | } | |
115 | #endif | |
116 | ||
117 | LPDRAWITEMSTRUCT lpDIS = (LPDRAWITEMSTRUCT) item; | |
118 | ||
119 | // choose the bitmap to use depending on the buttons state | |
120 | wxBitmap* bitmap; | |
121 | ||
122 | UINT state = lpDIS->itemState; | |
123 | bool isSelected = (state & ODS_SELECTED) != 0; | |
124 | if ( isSelected && m_buttonBitmapSelected.Ok() ) | |
125 | bitmap = &m_buttonBitmapSelected; | |
126 | else if ((state & ODS_FOCUS) && m_buttonBitmapFocus.Ok()) | |
127 | bitmap = &m_buttonBitmapFocus; | |
128 | else if ((state & ODS_DISABLED) && m_buttonBitmapDisabled.Ok()) | |
129 | bitmap = &m_buttonBitmapDisabled; | |
130 | else | |
131 | bitmap = &m_buttonBitmap; | |
132 | ||
133 | if ( !bitmap->Ok() ) | |
134 | return FALSE; | |
135 | ||
136 | // draw it on the memory DC | |
137 | HDC hDC = lpDIS->hDC; | |
138 | HDC memDC = ::CreateCompatibleDC(hDC); | |
139 | ||
140 | HBITMAP old = (HBITMAP) ::SelectObject(memDC, (HBITMAP) bitmap->GetHBITMAP()); | |
141 | ||
142 | if (!old) | |
143 | { | |
144 | wxLogLastError(_T("SelectObject")); | |
145 | ||
146 | return FALSE; | |
147 | } | |
148 | ||
149 | int x = lpDIS->rcItem.left; | |
150 | int y = lpDIS->rcItem.top; | |
151 | int width = lpDIS->rcItem.right - x; | |
152 | int height = lpDIS->rcItem.bottom - y; | |
153 | ||
154 | int wBmp = bitmap->GetWidth(), | |
155 | hBmp = bitmap->GetHeight(); | |
156 | ||
157 | // Draw the face, if auto-drawing | |
158 | bool autoDraw = (GetWindowStyleFlag() & wxBU_AUTODRAW) != 0; | |
159 | if ( autoDraw ) | |
160 | { | |
161 | DrawFace((WXHDC) hDC, | |
162 | lpDIS->rcItem.left, lpDIS->rcItem.top, | |
163 | lpDIS->rcItem.right, lpDIS->rcItem.bottom, | |
164 | isSelected); | |
165 | } | |
166 | ||
167 | // Centre the bitmap in the control area | |
168 | int x1 = x + (width - wBmp) / 2; | |
169 | int y1 = y + (height - hBmp) / 2; | |
170 | ||
171 | if ( isSelected && autoDraw ) | |
172 | { | |
173 | x1++; | |
174 | y1++; | |
175 | } | |
176 | ||
177 | BOOL ok; | |
178 | ||
179 | // no MaskBlt() under Win16 | |
180 | #ifdef __WIN32__ | |
181 | wxMask *mask = bitmap->GetMask(); | |
182 | if ( mask ) | |
183 | { | |
184 | // the fg ROP is applied for the pixels of the mask bitmap which are 1 | |
185 | // (for a wxMask this means that this is a non transparent pixel), the | |
186 | // bg ROP is applied for all the others | |
187 | ||
188 | HBRUSH hbrBackground = | |
189 | ::CreateSolidBrush(wxColourToRGB(GetBackgroundColour())); | |
190 | HBRUSH hbrOld = (HBRUSH)::SelectObject(hDC, hbrBackground); | |
191 | ||
192 | ok = ::MaskBlt( | |
193 | hDC, x1, y1, wBmp, hBmp, // dst | |
194 | memDC, 0, 0, // src | |
195 | (HBITMAP)mask->GetMaskBitmap(), 0, 0, // mask | |
196 | MAKEROP4(SRCCOPY, // fg ROP | |
197 | PATCOPY) // bg ROP | |
198 | ); | |
199 | ||
200 | ::SelectObject(hDC, hbrOld); | |
201 | ::DeleteObject(hbrBackground); | |
202 | } | |
203 | else | |
204 | { | |
205 | // this will make the check below fail and BitBlt() will be used if | |
206 | // MaskBlt() is not supported (for example, under Win95) | |
207 | ok = FALSE; | |
208 | } | |
209 | ||
210 | if ( !ok ) | |
211 | #endif // Win32 | |
212 | { | |
213 | ok = ::BitBlt(hDC, x1, y1, wBmp, hBmp, // dst | |
214 | memDC, 0, 0, // src | |
215 | SRCCOPY); // ROP | |
216 | } | |
217 | ||
218 | if ( !ok ) | |
219 | { | |
220 | wxLogLastError(_T("Mask/BitBlt()")); | |
221 | } | |
222 | ||
223 | if ( (state & ODS_DISABLED) && autoDraw ) | |
224 | { | |
225 | DrawButtonDisable((WXHDC) hDC, | |
226 | lpDIS->rcItem.left, lpDIS->rcItem.top, | |
227 | lpDIS->rcItem.right, lpDIS->rcItem.bottom, | |
228 | TRUE); | |
229 | } | |
230 | else if ( (state & ODS_FOCUS) && autoDraw ) | |
231 | { | |
232 | DrawButtonFocus((WXHDC) hDC, | |
233 | lpDIS->rcItem.left, | |
234 | lpDIS->rcItem.top, | |
235 | lpDIS->rcItem.right, | |
236 | lpDIS->rcItem.bottom, | |
237 | isSelected); | |
238 | } | |
239 | ||
240 | ::SelectObject(memDC, old); | |
241 | ||
242 | ::DeleteDC(memDC); | |
243 | ||
244 | return TRUE; | |
245 | } | |
246 | ||
247 | void wxBitmapButton::DrawFace( WXHDC dc, int left, int top, int right, int bottom, bool sel ) | |
248 | { | |
249 | HPEN oldp; | |
250 | HBRUSH oldb ; | |
251 | ||
252 | HPEN penBorder; | |
253 | HPEN penLight; | |
254 | HPEN penShadow; | |
255 | HBRUSH brushFace; | |
256 | COLORREF ms_color; | |
257 | ||
258 | ms_color = GetSysColor(COLOR_WINDOWFRAME) ; | |
259 | penBorder = CreatePen(PS_SOLID,0,ms_color) ; | |
260 | ||
261 | ms_color = GetSysColor(COLOR_BTNSHADOW) ; | |
262 | penShadow = CreatePen(PS_SOLID,0,ms_color) ; | |
263 | ||
264 | ms_color = GetSysColor(COLOR_BTNHIGHLIGHT) ; | |
265 | penLight = CreatePen(PS_SOLID,0,ms_color) ; | |
266 | ||
267 | ms_color = GetSysColor(COLOR_BTNFACE) ; | |
268 | brushFace = CreateSolidBrush(ms_color) ; | |
269 | ||
270 | oldp = (HPEN) SelectObject( (HDC) dc, GetStockObject( NULL_PEN ) ) ; | |
271 | oldb = (HBRUSH) SelectObject( (HDC) dc, brushFace ) ; | |
272 | Rectangle( (HDC) dc, left, top, right, bottom ) ; | |
273 | SelectObject( (HDC) dc, penBorder) ; | |
274 | MoveToEx((HDC) dc,left+1,top,NULL);LineTo((HDC) dc,right-1,top); | |
275 | MoveToEx((HDC) dc,left,top+1,NULL);LineTo((HDC) dc,left,bottom-1); | |
276 | MoveToEx((HDC) dc,left+1,bottom-1,NULL);LineTo((HDC) dc,right-1,bottom-1); | |
277 | MoveToEx((HDC) dc,right-1,top+1,NULL);LineTo((HDC) dc,right-1,bottom-1); | |
278 | ||
279 | SelectObject( (HDC) dc, penShadow) ; | |
280 | if (sel) | |
281 | { | |
282 | MoveToEx((HDC) dc,left+1 ,bottom-2 ,NULL) ; | |
283 | LineTo((HDC) dc, left+1 ,top+1) ; | |
284 | LineTo((HDC) dc, right-2 ,top+1) ; | |
285 | } | |
286 | else | |
287 | { | |
288 | MoveToEx((HDC) dc,left+1 ,bottom-2 ,NULL) ; | |
289 | LineTo((HDC) dc, right-2 ,bottom-2) ; | |
290 | LineTo((HDC) dc, right-2 ,top) ; | |
291 | MoveToEx((HDC) dc,left+2 ,bottom-3 ,NULL) ; | |
292 | LineTo((HDC) dc, right-3 ,bottom-3) ; | |
293 | LineTo((HDC) dc, right-3 ,top+1) ; | |
294 | ||
295 | SelectObject( (HDC) dc, penLight) ; | |
296 | ||
297 | MoveToEx((HDC) dc,left+1 ,bottom-2 ,NULL) ; | |
298 | LineTo((HDC) dc, left+1 ,top+1) ; | |
299 | LineTo((HDC) dc, right-2 ,top+1) ; | |
300 | } | |
301 | SelectObject((HDC) dc,oldp) ; | |
302 | SelectObject((HDC) dc,oldb) ; | |
303 | ||
304 | DeleteObject(penBorder); | |
305 | DeleteObject(penLight); | |
306 | DeleteObject(penShadow); | |
307 | DeleteObject(brushFace); | |
308 | } | |
309 | ||
310 | // VZ: should be at the very least less than wxDEFAULT_BUTTON_MARGIN | |
311 | #define FOCUS_MARGIN 3 | |
312 | ||
313 | void wxBitmapButton::DrawButtonFocus( WXHDC dc, int left, int top, int right, int bottom, bool sel ) | |
314 | { | |
315 | RECT rect; | |
316 | rect.left = left; | |
317 | rect.top = top; | |
318 | rect.right = right; | |
319 | rect.bottom = bottom; | |
320 | InflateRect( &rect, - FOCUS_MARGIN, - FOCUS_MARGIN ) ; | |
321 | if ( sel ) | |
322 | OffsetRect( &rect, 1, 1 ) ; | |
323 | DrawFocusRect( (HDC) dc, &rect ) ; | |
324 | } | |
325 | ||
326 | extern HBRUSH wxDisableButtonBrush; | |
327 | void wxBitmapButton::DrawButtonDisable( WXHDC dc, int left, int top, int right, int bottom, bool with_marg ) | |
328 | { | |
329 | HBRUSH old = (HBRUSH) SelectObject( (HDC) dc, wxDisableButtonBrush ) ; | |
330 | ||
331 | // VZ: what's this?? there is no such ROP AFAIK | |
332 | #ifdef __SALFORDC__ | |
333 | DWORD dwRop = 0xFA0089L; | |
334 | #else | |
335 | DWORD dwRop = 0xFA0089UL; | |
336 | #endif | |
337 | ||
338 | if ( with_marg ) | |
339 | { | |
340 | left += m_marginX; | |
341 | top += m_marginY; | |
342 | right -= 2 * m_marginX; | |
343 | bottom -= 2 * m_marginY; | |
344 | } | |
345 | ||
346 | ::PatBlt( (HDC) dc, left, top, right, bottom, dwRop); | |
347 | ||
348 | ::SelectObject( (HDC) dc, old ) ; | |
349 | } | |
350 | ||
351 | void wxBitmapButton::SetDefault() | |
352 | { | |
353 | wxButton::SetDefault(); | |
354 | } |