1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/minifram.cpp
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
15 #include "wx/minifram.h"
18 #include "wx/settings.h"
19 #include "wx/dcclient.h"
25 //-----------------------------------------------------------------------------
27 //-----------------------------------------------------------------------------
29 extern bool g_blockEventsOnDrag
;
30 extern bool g_blockEventsOnScroll
;
32 //-----------------------------------------------------------------------------
33 // "expose_event" of m_mainWidget
34 //-----------------------------------------------------------------------------
36 // StepColour() it a utility function that simply darkens
37 // or lightens a color, based on the specified percentage
38 static wxColor
StepColour(const wxColor
& c
, int percent
)
40 int r
= c
.Red(), g
= c
.Green(), b
= c
.Blue();
41 return wxColour((unsigned char)wxMin((r
*percent
)/100,255),
42 (unsigned char)wxMin((g
*percent
)/100,255),
43 (unsigned char)wxMin((b
*percent
)/100,255));
46 static wxColor
LightContrastColour(const wxColour
& c
)
50 // if the color is especially dark, then
51 // make the contrast even lighter
52 if (c
.Red() < 128 && c
.Green() < 128 && c
.Blue() < 128)
55 return StepColour(c
, amount
);
59 static gboolean
gtk_window_own_expose_callback(GtkWidget
* widget
, GdkEventExpose
* gdk_event
, wxMiniFrame
* win
)
61 if (!win
->m_hasVMT
|| gdk_event
->count
> 0)
64 gtk_paint_shadow (widget
->style
,
68 NULL
, NULL
, NULL
, // FIXME: No clipping?
70 win
->m_width
, win
->m_height
);
72 int style
= win
->GetWindowStyle();
77 wxImplDC
*impl
= dc
.GetImpl();
78 wxGTKClientImplDC
*client_impl
= wxDynamicCast( impl
, wxGTKClientImplDC
);
80 client_impl
->m_window
= widget
->window
;
83 dc
.m_window
= widget
->window
;
86 if (style
& wxRESIZE_BORDER
)
88 dc
.SetBrush( *wxGREY_BRUSH
);
89 dc
.SetPen( *wxTRANSPARENT_PEN
);
90 dc
.DrawRectangle( win
->m_width
- 14, win
->m_height
-14, 14, 14 );
93 if (!win
->GetTitle().empty() &&
94 ((style
& wxCAPTION
) ||
95 (style
& wxTINY_CAPTION_HORIZ
) ||
96 (style
& wxTINY_CAPTION_VERT
)))
98 dc
.SetFont( *wxSMALL_FONT
);
100 wxBrush
brush( LightContrastColour( wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT
) ) );
101 dc
.SetBrush( brush
);
102 dc
.SetPen( *wxTRANSPARENT_PEN
);
103 dc
.DrawRectangle( win
->m_miniEdge
-1,
105 win
->m_width
- (2*(win
->m_miniEdge
-1)),
108 dc
.SetTextForeground( *wxWHITE
);
109 dc
.DrawText( win
->GetTitle(), 6, 4 );
111 if (style
& wxCLOSE_BOX
)
112 dc
.DrawBitmap( win
->m_closeButton
, win
->m_width
-18, 3, true );
119 //-----------------------------------------------------------------------------
120 // "button_press_event" of m_mainWidget
121 //-----------------------------------------------------------------------------
124 static gint
gtk_window_button_press_callback( GtkWidget
*widget
, GdkEventButton
*gdk_event
, wxMiniFrame
*win
)
126 if (!win
->m_hasVMT
) return FALSE
;
127 if (g_blockEventsOnDrag
) return TRUE
;
128 if (g_blockEventsOnScroll
) return TRUE
;
130 if (win
->m_isDragging
) return TRUE
;
132 int style
= win
->GetWindowStyle();
134 int y
= (int)gdk_event
->y
;
135 int x
= (int)gdk_event
->x
;
137 if ((style
& wxRESIZE_BORDER
) &&
138 (x
> win
->m_width
-14) && (y
> win
->m_height
-14))
140 GtkWidget
*ancestor
= gtk_widget_get_toplevel( widget
);
142 GdkWindow
*source
= widget
->window
;
146 gdk_window_get_origin( source
, &org_x
, &org_y
);
148 gtk_window_begin_resize_drag (GTK_WINDOW (ancestor
),
149 GDK_WINDOW_EDGE_SOUTH_EAST
,
158 if ((style
& wxCLOSE_BOX
) &&
159 ((style
& wxCAPTION
) || (style
& wxTINY_CAPTION_HORIZ
) || (style
& wxTINY_CAPTION_VERT
)))
161 if ((y
> 3) && (y
< 19) && (x
> win
->m_width
-19) && (x
< win
->m_width
-3))
168 if (y
> win
->m_miniEdge
-1 + 15) return TRUE
;
170 gdk_window_raise( win
->m_widget
->window
);
172 gdk_pointer_grab( widget
->window
, FALSE
,
174 (GDK_BUTTON_PRESS_MASK
|
175 GDK_BUTTON_RELEASE_MASK
|
176 GDK_POINTER_MOTION_MASK
|
177 GDK_POINTER_MOTION_HINT_MASK
|
178 GDK_BUTTON_MOTION_MASK
|
179 GDK_BUTTON1_MOTION_MASK
),
182 (unsigned int) GDK_CURRENT_TIME
);
189 win
->m_isDragging
= true;
195 //-----------------------------------------------------------------------------
196 // "button_release_event" of m_mainWidget
197 //-----------------------------------------------------------------------------
200 static gint
gtk_window_button_release_callback( GtkWidget
*widget
, GdkEventButton
*gdk_event
, wxMiniFrame
*win
)
202 if (!win
->m_hasVMT
) return FALSE
;
203 if (g_blockEventsOnDrag
) return TRUE
;
204 if (g_blockEventsOnScroll
) return TRUE
;
206 if (!win
->m_isDragging
) return TRUE
;
208 win
->m_isDragging
= false;
210 int x
= (int)gdk_event
->x
;
211 int y
= (int)gdk_event
->y
;
213 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
216 gdk_window_get_origin( widget
->window
, &org_x
, &org_y
);
217 x
+= org_x
- win
->m_diffX
;
218 y
+= org_y
- win
->m_diffY
;
221 gtk_window_move( GTK_WINDOW(win
->m_widget
), x
, y
);
227 //-----------------------------------------------------------------------------
228 // "leave_notify_event" of m_mainWidget
229 //-----------------------------------------------------------------------------
233 gtk_window_leave_callback(GtkWidget
*widget
,
234 GdkEventCrossing
* WXUNUSED(gdk_event
),
237 if (!win
->m_hasVMT
) return FALSE
;
238 if (g_blockEventsOnDrag
) return FALSE
;
240 gdk_window_set_cursor( widget
->window
, NULL
);
246 //-----------------------------------------------------------------------------
247 // "motion_notify_event" of m_mainWidget
248 //-----------------------------------------------------------------------------
252 gtk_window_motion_notify_callback( GtkWidget
*widget
, GdkEventMotion
*gdk_event
, wxMiniFrame
*win
)
254 if (!win
->m_hasVMT
) return FALSE
;
255 if (g_blockEventsOnDrag
) return TRUE
;
256 if (g_blockEventsOnScroll
) return TRUE
;
258 if (gdk_event
->is_hint
)
262 GdkModifierType state
;
263 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
266 gdk_event
->state
= state
;
269 int style
= win
->GetWindowStyle();
271 int x
= (int)gdk_event
->x
;
272 int y
= (int)gdk_event
->y
;
274 if (!win
->m_isDragging
)
276 if (style
& wxRESIZE_BORDER
)
278 if ((x
> win
->m_width
-14) && (y
> win
->m_height
-14))
279 gdk_window_set_cursor( widget
->window
, gdk_cursor_new( GDK_BOTTOM_RIGHT_CORNER
) );
281 gdk_window_set_cursor( widget
->window
, NULL
);
286 win
->m_oldX
= x
- win
->m_diffX
;
287 win
->m_oldY
= y
- win
->m_diffY
;
291 gdk_window_get_origin( widget
->window
, &org_x
, &org_y
);
292 x
+= org_x
- win
->m_diffX
;
293 y
+= org_y
- win
->m_diffY
;
296 gtk_window_move( GTK_WINDOW(win
->m_widget
), x
, y
);
302 //-----------------------------------------------------------------------------
304 //-----------------------------------------------------------------------------
306 static unsigned char close_bits
[]={
307 0xff, 0xff, 0xff, 0xff, 0x07, 0xf0, 0xfb, 0xef, 0xdb, 0xed, 0x8b, 0xe8,
308 0x1b, 0xec, 0x3b, 0xee, 0x1b, 0xec, 0x8b, 0xe8, 0xdb, 0xed, 0xfb, 0xef,
309 0x07, 0xf0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
312 IMPLEMENT_DYNAMIC_CLASS(wxMiniFrame
,wxFrame
)
314 bool wxMiniFrame::Create( wxWindow
*parent
, wxWindowID id
, const wxString
&title
,
315 const wxPoint
&pos
, const wxSize
&size
,
316 long style
, const wxString
&name
)
318 if ((style
& wxCAPTION
) || (style
& wxTINY_CAPTION_HORIZ
) || (style
& wxTINY_CAPTION_VERT
))
321 if (style
& wxRESIZE_BORDER
)
325 m_isDragging
= false;
331 wxFrame::Create( parent
, id
, title
, pos
, size
, style
, name
);
333 // Use a GtkEventBox for the title and borders. Using m_widget for this
334 // almost works, except that setting the resize cursor has no effect.
335 GtkWidget
* eventbox
= gtk_event_box_new();
336 gtk_widget_add_events(eventbox
,
337 GDK_POINTER_MOTION_MASK
|
338 GDK_POINTER_MOTION_HINT_MASK
);
339 gtk_widget_show(eventbox
);
340 // Use a GtkAlignment to position m_mainWidget inside the decorations
341 GtkWidget
* alignment
= gtk_alignment_new(0, 0, 1, 1);
342 gtk_alignment_set_padding(GTK_ALIGNMENT(alignment
),
343 m_miniTitle
+ m_miniEdge
, m_miniEdge
, m_miniEdge
, m_miniEdge
);
344 gtk_widget_show(alignment
);
345 // The GtkEventBox and GtkAlignment go between m_widget and m_mainWidget
346 gtk_widget_reparent(m_mainWidget
, alignment
);
347 gtk_container_add(GTK_CONTAINER(eventbox
), alignment
);
348 gtk_container_add(GTK_CONTAINER(m_widget
), eventbox
);
352 if (style
& wxRESIZE_BORDER
)
353 m_gdkFunc
= GDK_FUNC_RESIZE
;
355 // need to reset default size after changing m_gdkDecor
356 gtk_window_set_default_size(GTK_WINDOW(m_widget
), m_width
, m_height
);
357 m_decorSize
.Set(0, 0);
359 // don't allow sizing smaller than decorations
361 geom
.min_width
= 2 * m_miniEdge
;
362 geom
.min_height
= 2 * m_miniEdge
+ m_miniTitle
;
363 gtk_window_set_geometry_hints(GTK_WINDOW(m_widget
), NULL
, &geom
, GDK_HINT_MIN_SIZE
);
365 if (m_parent
&& (GTK_IS_WINDOW(m_parent
->m_widget
)))
367 gtk_window_set_transient_for( GTK_WINDOW(m_widget
), GTK_WINDOW(m_parent
->m_widget
) );
370 if ((style
& wxCLOSE_BOX
) &&
371 ((style
& wxCAPTION
) || (style
& wxTINY_CAPTION_HORIZ
) || (style
& wxTINY_CAPTION_VERT
)))
373 wxImage img
= wxBitmap((const char*)close_bits
, 16, 16).ConvertToImage();
374 img
.Replace(0,0,0,123,123,123);
375 img
.SetMaskColour(123,123,123);
376 m_closeButton
= wxBitmap( img
);
379 /* these are called when the borders are drawn */
380 g_signal_connect_after(eventbox
, "expose_event",
381 G_CALLBACK (gtk_window_own_expose_callback
), this );
383 /* these are required for dragging the mini frame around */
384 g_signal_connect (eventbox
, "button_press_event",
385 G_CALLBACK (gtk_window_button_press_callback
), this);
386 g_signal_connect (eventbox
, "button_release_event",
387 G_CALLBACK (gtk_window_button_release_callback
), this);
388 g_signal_connect (eventbox
, "motion_notify_event",
389 G_CALLBACK (gtk_window_motion_notify_callback
), this);
390 g_signal_connect (eventbox
, "leave_notify_event",
391 G_CALLBACK (gtk_window_leave_callback
), this);
395 void wxMiniFrame::DoGetClientSize(int* width
, int* height
) const
397 wxFrame::DoGetClientSize(width
, height
);
400 *width
-= 2 * m_miniEdge
;
401 if (*width
< 0) *width
= 0;
405 *height
-= m_miniTitle
+ 2 * m_miniEdge
;
406 if (*height
< 0) *height
= 0;
410 // Keep min size at least as large as decorations
411 void wxMiniFrame::DoSetSizeHints(int minW
, int minH
, int maxW
, int maxH
, int incW
, int incH
)
413 const int w
= 2 * m_miniEdge
;
414 const int h
= 2 * m_miniEdge
+ m_miniTitle
;
415 if (minW
< w
) minW
= w
;
416 if (minH
< h
) minH
= h
;
417 wxFrame::DoSetSizeHints(minW
, minH
, maxW
, maxH
, incW
, incH
);
420 void wxMiniFrame::SetTitle( const wxString
&title
)
422 wxFrame::SetTitle( title
);
424 GtkWidget
* widget
= GTK_BIN(m_widget
)->child
;
426 gdk_window_invalidate_rect(widget
->window
, NULL
, false);
429 #endif // wxUSE_MINIFRAME