]>
Commit | Line | Data |
---|---|---|
2bda0e17 KB |
1 | ///////////////////////////////////////////////////////////////////////////// |
2 | // Name: statbox.cpp | |
3 | // Purpose: wxStaticBox | |
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 "statbox.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/dcclient.h" | |
25 | #include "wx/app.h" | |
26 | #endif | |
27 | ||
28 | #include "wx/statbox.h" | |
29 | #include "wx/msw/private.h" | |
30 | ||
31 | #if !USE_SHARED_LIBRARY | |
32 | IMPLEMENT_DYNAMIC_CLASS(wxStaticBox, wxControl) | |
33 | ||
34 | BEGIN_EVENT_TABLE(wxStaticBox, wxControl) | |
35 | EVT_ERASE_BACKGROUND(wxStaticBox::OnEraseBackground) | |
36 | END_EVENT_TABLE() | |
37 | ||
38 | #endif | |
39 | ||
40 | /* | |
41 | * Group box | |
42 | */ | |
43 | ||
debe6624 | 44 | bool wxStaticBox::Create(wxWindow *parent, wxWindowID id, |
2bda0e17 KB |
45 | const wxString& label, |
46 | const wxPoint& pos, | |
47 | const wxSize& size, | |
debe6624 | 48 | long style, |
2bda0e17 KB |
49 | const wxString& name) |
50 | { | |
51 | SetName(name); | |
52 | ||
53 | if (parent) parent->AddChild(this); | |
54 | ||
55 | SetBackgroundColour(parent->GetDefaultBackgroundColour()) ; | |
56 | SetForegroundColour(parent->GetDefaultForegroundColour()) ; | |
57 | ||
58 | if ( id == -1 ) | |
59 | m_windowId = (int)NewControlId(); | |
60 | else | |
61 | m_windowId = id; | |
62 | ||
63 | int x = pos.x; | |
64 | int y = pos.y; | |
65 | int width = size.x; | |
66 | int height = size.y; | |
67 | ||
68 | m_windowStyle = style; | |
69 | ||
70 | long msStyle = BS_GROUPBOX | WS_CHILD | WS_VISIBLE ; // GROUP_FLAGS; | |
71 | ||
72 | bool want3D; | |
73 | WXDWORD exStyle = Determine3DEffects(0, &want3D) ; | |
74 | ||
75 | HWND wx_button = | |
76 | CreateWindowEx(exStyle, "BUTTON", (const char *)label, msStyle, | |
77 | 0, 0, 0, 0, (HWND) parent->GetHWND(), (HMENU)m_windowId, | |
78 | wxGetInstance(), NULL); | |
79 | #if CTL3D | |
80 | if (want3D) | |
81 | { | |
82 | Ctl3dSubclassCtl(wx_button); | |
83 | m_useCtl3D = TRUE; | |
84 | } | |
85 | #endif | |
86 | ||
87 | m_hWnd = (WXHWND)wx_button; | |
88 | ||
89 | // Subclass again for purposes of dialog editing mode | |
90 | SubclassWin(GetHWND()); | |
91 | ||
92 | SetFont(* parent->GetFont()); | |
93 | ||
94 | SetSize(x, y, width, height); | |
95 | ShowWindow(wx_button, SW_SHOW); | |
96 | ||
97 | return TRUE; | |
98 | } | |
99 | ||
100 | void wxStaticBox::SetLabel(const wxString& label) | |
101 | { | |
102 | SetWindowText((HWND)m_hWnd, (const char *)label); | |
103 | } | |
104 | ||
debe6624 | 105 | void wxStaticBox::SetSize(int x, int y, int width, int height, int sizeFlags) |
2bda0e17 KB |
106 | { |
107 | int currentX, currentY; | |
108 | GetPosition(¤tX, ¤tY); | |
109 | ||
110 | int x1 = x; | |
111 | int y1 = y; | |
112 | int w1 = width; | |
113 | int h1 = height; | |
114 | ||
115 | if (x == -1 || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE)) | |
116 | x1 = currentX; | |
117 | if (y == -1 || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE)) | |
118 | y1 = currentY; | |
119 | ||
81d66cf3 JS |
120 | AdjustForParentClientOrigin(x1, y1, sizeFlags); |
121 | ||
2bda0e17 KB |
122 | // If we're prepared to use the existing size, then... |
123 | if (width == -1 && height == -1 && ((sizeFlags & wxSIZE_AUTO) != wxSIZE_AUTO)) | |
124 | { | |
125 | GetSize(&w1, &h1); | |
126 | } | |
127 | ||
128 | char buf[300]; | |
129 | ||
debe6624 | 130 | int current_width; |
2bda0e17 KB |
131 | |
132 | int cx; | |
133 | int cy; | |
debe6624 | 134 | int cyf; |
2bda0e17 KB |
135 | |
136 | HWND button = (HWND)m_hWnd; | |
137 | wxGetCharSize(GetHWND(), &cx, &cy,GetFont()); | |
138 | ||
139 | GetWindowText(button, buf, 300); | |
140 | GetTextExtent(buf, ¤t_width, &cyf,NULL,NULL,GetFont()); | |
141 | if (w1 < 0) | |
142 | w1 = (int)(current_width + 3*cx) ; | |
143 | if (h1<0) | |
144 | h1 = (int)(cyf*EDIT_CONTROL_FACTOR) ; | |
145 | MoveWindow(button, x1, y1, w1, h1, TRUE); | |
2bda0e17 KB |
146 | } |
147 | ||
debe6624 | 148 | WXHBRUSH wxStaticBox::OnCtlColor(WXHDC pDC, WXHWND pWnd, WXUINT nCtlColor, |
2bda0e17 KB |
149 | WXUINT message, WXWPARAM wParam, WXLPARAM lParam) |
150 | { | |
151 | #if CTL3D | |
152 | if ( m_useCtl3D ) | |
153 | { | |
154 | HBRUSH hbrush = Ctl3dCtlColorEx(message, wParam, lParam); | |
155 | return (WXHBRUSH) hbrush; | |
156 | } | |
157 | #endif | |
158 | ||
159 | if (GetParent()->GetTransparentBackground()) | |
160 | SetBkMode((HDC) pDC, TRANSPARENT); | |
161 | else | |
162 | SetBkMode((HDC) pDC, OPAQUE); | |
163 | ||
164 | ::SetBkColor((HDC) pDC, RGB(GetBackgroundColour().Red(), GetBackgroundColour().Green(), GetBackgroundColour().Blue())); | |
165 | ::SetTextColor((HDC) pDC, RGB(GetForegroundColour().Red(), GetForegroundColour().Green(), GetForegroundColour().Blue())); | |
166 | ||
167 | wxBrush *backgroundBrush = wxTheBrushList->FindOrCreateBrush(GetBackgroundColour(), wxSOLID); | |
168 | ||
169 | // Note that this will be cleaned up in wxApp::OnIdle, if backgroundBrush | |
170 | // has a zero usage count. | |
171 | // backgroundBrush->RealizeResource(); | |
172 | return (WXHBRUSH) backgroundBrush->GetResourceHandle(); | |
173 | } | |
174 | ||
175 | // Shouldn't erase the whole window, since the static box must only paint its | |
176 | // outline. | |
177 | void wxStaticBox::OnEraseBackground(wxEraseEvent& event) | |
178 | { | |
1c089c47 JS |
179 | // If we don't have this (call Default()), we don't paint the background properly. |
180 | // If we do have this, we seem to overwrite enclosed controls. | |
181 | // Is it the WS_CLIPCHILDREN style that's causing the problems? | |
182 | // Probably - without this style, the background of the window will show through, | |
183 | // so the control doesn't have to paint it. The window background will always be | |
184 | // painted before all other controls, therefore there are no problems with | |
185 | // controls being hidden by the static box. | |
186 | // So, if we could specify wxCLIP_CHILDREN in window, or not, we could optimise painting better. | |
187 | // We would assume wxCLIP_CHILDREN in a frame and a scrolled window, but not in a panel. | |
188 | // Is this too platform-specific?? What else can we do? Not a lot, since we have to pass | |
189 | // this information from arbitrary wxWindow derivatives, and it depends on what you wish to | |
190 | // do with the windows. | |
191 | // Alternatively, just make sure that wxStaticBox is always at the back! There are probably | |
192 | // few other circumstances where it matters about child clipping. But what about painting onto | |
193 | // to panel, inside a groupbox? Doesn't appear, because the box wipes it out. | |
2bda0e17 KB |
194 | wxWindow *parent = GetParent(); |
195 | if ( parent && parent->GetHWND() && (::GetWindowLong((HWND) parent->GetHWND(), GWL_STYLE) & WS_CLIPCHILDREN) ) | |
196 | { | |
197 | // TODO: May in fact need to generate a paint event for inside this | |
198 | // control's rectangle, otherwise all controls are going to be clipped - | |
199 | // ugh. | |
200 | HBRUSH hBrush = ::CreateSolidBrush(PALETTERGB(GetBackgroundColour().Red(), GetBackgroundColour().Green(), GetBackgroundColour().Blue())); | |
201 | int mode = ::SetMapMode((HDC) event.GetDC()->GetHDC(), MM_TEXT); | |
202 | ||
203 | RECT rect; | |
204 | ||
205 | ::GetClientRect((HWND) GetHWND(), &rect); | |
206 | ::FillRect ((HDC) event.GetDC()->GetHDC(), &rect, hBrush); | |
207 | ::DeleteObject(hBrush); | |
208 | ::SetMapMode((HDC) event.GetDC()->GetHDC(), mode); | |
209 | } | |
210 | else | |
211 | Default(); | |
212 | } | |
213 | ||
214 | long wxStaticBox::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam) | |
215 | { | |
1c089c47 | 216 | // TODO: somehow, this has to accept mouse clicks in user interface edit mode, |
2bda0e17 KB |
217 | // but not otherwise. Only there is no longer a UI edit mode... |
218 | ||
219 | // It worked before because the message could be processed if not in UI | |
220 | // edit mode. We have to find some way of distinguishing this. | |
221 | // Maybe this class can have an AcceptMouseEvents(bool) function; a sort of | |
222 | // kludge... or, we can search for an active event table entry that will | |
223 | // intercept mouse events, and if one exists (that isn't the default), | |
224 | // skip the code below. Too time consuming though. | |
225 | // Perhaps it's ok to do the default thing *anyway* because the title or edge | |
226 | // of the window may still be active! | |
1c089c47 JS |
227 | // if (nMsg == WM_NCHITTEST) |
228 | // return Default(); | |
229 | ||
2bda0e17 | 230 | if (nMsg == WM_NCHITTEST) |
1c089c47 JS |
231 | { |
232 | int xPos = LOWORD(lParam); // horizontal position of cursor | |
233 | int yPos = HIWORD(lParam); // vertical position of cursor | |
234 | ||
235 | ScreenToClient(&xPos, &yPos); | |
236 | ||
237 | // Make sure you can drag by the top of the groupbox, but let | |
238 | // other (enclosed) controls get mouse events also | |
239 | if (yPos < 10) | |
240 | return (long)HTCLIENT; | |
241 | } | |
2bda0e17 KB |
242 | |
243 | return wxControl::MSWWindowProc(nMsg, wParam, lParam); | |
244 | } | |
245 |