]> git.saurik.com Git - wxWidgets.git/blame - src/x11/toplevel.cpp
fix another memory leak in SetCommand() (coverity checker CID 52)
[wxWidgets.git] / src / x11 / toplevel.cpp
CommitLineData
1b0fb34b 1///////////////////////////////////////////////////////////////////////////////
ffd84c94 2// Name: src/x11/toplevel.cpp
1b0fb34b 3// Purpose: implements wxTopLevelWindow for X11
83df96d6
JS
4// Author: Julian Smart
5// Modified by:
1b0fb34b 6// Created: 24.09.01
83df96d6 7// RCS-ID: $Id$
1b0fb34b 8// Copyright: (c) 2002 Julian Smart
65571936 9// License: wxWindows licence
1b0fb34b 10///////////////////////////////////////////////////////////////////////////////
83df96d6
JS
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
1b0fb34b
JS
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
83df96d6 22
1b0fb34b
JS
23#ifdef __BORLANDC__
24 #pragma hdrstop
83df96d6
JS
25#endif
26
1b0fb34b
JS
27#ifndef WX_PRECOMP
28 #include "wx/app.h"
29 #include "wx/toplevel.h"
30 #include "wx/string.h"
31 #include "wx/log.h"
32 #include "wx/intl.h"
33 #include "wx/frame.h"
e5053ade
JS
34 #include "wx/menu.h"
35 #include "wx/statusbr.h"
1b0fb34b 36#endif //WX_PRECOMP
83df96d6 37
2034b748 38#include "wx/settings.h"
1b0fb34b 39#include "wx/x11/private.h"
2034b748 40#include "X11/Xutil.h"
b513212d 41
f618020a
MB
42#include "wx/unix/utilsx11.h"
43
1b0b798d 44bool wxMWMIsRunning(Window w);
b513212d 45
83df96d6 46// ----------------------------------------------------------------------------
1b0fb34b 47// wxTopLevelWindowX11 creation
83df96d6
JS
48// ----------------------------------------------------------------------------
49
1b0fb34b
JS
50void wxTopLevelWindowX11::Init()
51{
52 m_iconized =
ffd84c94 53 m_maximizeOnShow = false;
83df96d6 54
1b0fb34b 55 // unlike (almost?) all other windows, frames are created hidden
ffd84c94 56 m_isShown = false;
83df96d6 57
1b0fb34b
JS
58 // Data to save/restore when calling ShowFullScreen
59 m_fsStyle = 0;
ffd84c94
WS
60 m_fsIsMaximized = false;
61 m_fsIsShowing = false;
62
63 m_needResizeInIdle = false;
64
65 m_x = wxDefaultCoord;
66 m_y = wxDefaultCoord;
c2c0dabf
RR
67 m_width = 20;
68 m_height = 20;
1b0fb34b 69}
83df96d6 70
1b0fb34b
JS
71bool wxTopLevelWindowX11::Create(wxWindow *parent,
72 wxWindowID id,
73 const wxString& title,
74 const wxPoint& pos,
75 const wxSize& size,
76 long style,
77 const wxString& name)
78{
79 // init our fields
80 Init();
83df96d6
JS
81
82 m_windowStyle = style;
b28d3abf 83 m_parent = parent;
83df96d6 84
1b0fb34b 85 SetName(name);
83df96d6 86
ffd84c94 87 m_windowId = id == wxID_ANY ? NewControlId() : id;
83df96d6 88
b28d3abf
JS
89 if (parent)
90 parent->AddChild(this);
91
1b0fb34b 92 wxTopLevelWindows.Append(this);
ffd84c94 93
952ebeba
RR
94 Display *xdisplay = wxGlobalDisplay();
95 int xscreen = DefaultScreen( xdisplay );
96 Visual *xvisual = DefaultVisual( xdisplay, xscreen );
97 Window xparent = RootWindow( xdisplay, xscreen );
2034b748 98 Colormap cm = DefaultColormap( xdisplay, xscreen );
ffd84c94 99
7e085304
RR
100 if (GetExtraStyle() & wxTOPLEVEL_EX_DIALOG)
101 m_backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE);
102 else
103 m_backgroundColour = wxSystemSettings::GetColour(wxSYS_COLOUR_APPWORKSPACE);
2034b748 104 m_backgroundColour.CalcPixel( (WXColormap) cm );
ffd84c94
WS
105 m_hasBgCol = true;
106
c2c0dabf
RR
107 m_x = pos.x;
108 if (m_x < -1)
109 m_x = 10;
ffd84c94 110
c2c0dabf
RR
111 m_y = pos.y;
112 if (m_y < 0)
113 m_y = 10;
ffd84c94 114
c2c0dabf
RR
115 m_width = size.x;
116 if (m_width < 0)
117 m_width = 500;
ffd84c94 118
c2c0dabf
RR
119 m_height = size.y;
120 if (m_height < 0)
121 m_height = 380;
ffd84c94 122
461e93f9 123#if !wxUSE_NANOX
952ebeba 124 XSetWindowAttributes xattributes;
ffd84c94 125
952ebeba 126 long xattributes_mask =
952ebeba 127 CWBorderPixel | CWBackPixel;
461e93f9 128
2034b748 129 xattributes.background_pixel = m_backgroundColour.GetPixel();
952ebeba 130 xattributes.border_pixel = BlackPixel( xdisplay, xscreen );
5e29f97a 131
ab6b6b15
RR
132 if (HasFlag( wxNO_BORDER ))
133 {
134 xattributes_mask |= CWOverrideRedirect;
135 xattributes.override_redirect = True;
136 }
ffd84c94 137
e441e1f4 138 if (!HasFlag( wxFULL_REPAINT_ON_RESIZE ))
2f12683e
RR
139 {
140 xattributes_mask |= CWBitGravity;
065722d7 141 xattributes.bit_gravity = NorthWestGravity;
2f12683e 142 }
ffd84c94 143
2f12683e 144 xattributes_mask |= CWEventMask;
ffd84c94 145 xattributes.event_mask =
2f12683e
RR
146 ExposureMask | KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
147 ButtonMotionMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask |
148 KeymapStateMask | FocusChangeMask | ColormapChangeMask | StructureNotifyMask |
149 PropertyChangeMask;
ffd84c94 150
c2c0dabf 151 Window xwindow = XCreateWindow( xdisplay, xparent, m_x, m_y, m_width, m_height,
2f12683e
RR
152 0, DefaultDepth(xdisplay,xscreen), InputOutput, xvisual, xattributes_mask, &xattributes );
153#else
461e93f9
JS
154 long backColor, foreColor;
155 backColor = GR_RGB(m_backgroundColour.Red(), m_backgroundColour.Green(), m_backgroundColour.Blue());
156 foreColor = GR_RGB(m_foregroundColour.Red(), m_foregroundColour.Green(), m_foregroundColour.Blue());
ffd84c94 157
c2c0dabf 158 Window xwindow = XCreateWindowWithColor( xdisplay, xparent, m_x, m_y, m_width, m_height,
461e93f9 159 0, 0, InputOutput, xvisual, backColor, foreColor);
461e93f9 160#endif
2f12683e 161
ab6b6b15
RR
162 m_mainWindow = (WXWindow) xwindow;
163 m_clientWindow = (WXWindow) xwindow;
164 wxAddWindowToTable( xwindow, (wxWindow*) this );
307be31a 165
256d631a 166#if wxUSE_NANOX
461e93f9 167 XSelectInput( xdisplay, xwindow,
2f12683e 168 GR_EVENT_MASK_CLOSE_REQ |
70b8ab77
JS
169 ExposureMask |
170 KeyPressMask |
171 KeyReleaseMask |
172 ButtonPressMask |
173 ButtonReleaseMask |
174 ButtonMotionMask |
175 EnterWindowMask |
176 LeaveWindowMask |
177 PointerMotionMask |
178 KeymapStateMask |
179 FocusChangeMask |
180 ColormapChangeMask |
181 StructureNotifyMask |
182 PropertyChangeMask
183 );
2f12683e 184#endif
307be31a 185
ba696cfa
RR
186 // Set background to None which will prevent X11 from clearing the
187 // background completely.
188 XSetWindowBackgroundPixmap( xdisplay, xwindow, None );
189
461e93f9 190#if !wxUSE_NANOX
ab6b6b15 191 if (HasFlag( wxSTAY_ON_TOP ))
7e4501ee 192 {
ab6b6b15
RR
193 Window xroot = RootWindow( xdisplay, xscreen );
194 XSetTransientForHint( xdisplay, xwindow, xroot );
195 }
196 else
197 {
198 if (GetExtraStyle() & wxTOPLEVEL_EX_DIALOG)
199 {
200 if (GetParent() && GetParent()->GetMainWindow())
201 {
202 Window xparentwindow = (Window) GetParent()->GetMainWindow();
203 XSetTransientForHint( xdisplay, xwindow, xparentwindow );
204 }
7e4501ee
RR
205 }
206 }
207
c2c0dabf 208 XSizeHints size_hints;
065722d7 209 size_hints.flags = PSize | PPosition | PWinGravity;
c2c0dabf
RR
210 size_hints.x = m_x;
211 size_hints.y = m_y;
212 size_hints.width = m_width;
213 size_hints.height = m_height;
065722d7 214 size_hints.win_gravity = NorthWestGravity;
952ebeba 215 XSetWMNormalHints( xdisplay, xwindow, &size_hints);
ffd84c94 216
461e93f9 217 XWMHints wm_hints;
ab6b6b15
RR
218 wm_hints.flags = InputHint | StateHint;
219 if (GetParent())
ffd84c94 220 {
ab6b6b15
RR
221 wm_hints.flags |= WindowGroupHint;
222 wm_hints.window_group = (Window) GetParent()->GetMainWindow();
223 }
952ebeba
RR
224 wm_hints.input = True;
225 wm_hints.initial_state = NormalState;
226 XSetWMHints( xdisplay, xwindow, &wm_hints);
461e93f9 227
7e4501ee
RR
228 Atom wm_protocols[2];
229 wm_protocols[0] = XInternAtom( xdisplay, "WM_DELETE_WINDOW", False );
230 wm_protocols[1] = XInternAtom( xdisplay, "WM_TAKE_FOCUS", False );
231 XSetWMProtocols( xdisplay, xwindow, wm_protocols, 2);
1016f0de 232
461e93f9 233#endif
ffd84c94 234
7e4501ee 235 wxSetWMDecorations( xwindow, style);
83df96d6 236
b513212d 237 SetTitle(title);
ffd84c94
WS
238
239 return true;
83df96d6
JS
240}
241
1b0fb34b 242wxTopLevelWindowX11::~wxTopLevelWindowX11()
83df96d6 243{
83df96d6 244 wxTopLevelWindows.DeleteObject(this);
83df96d6 245
1b0fb34b 246 // If this is the last top-level window, exit.
09a1dffa 247 if ( wxTheApp && (wxTopLevelWindows.GetCount() == 0) )
83df96d6
JS
248 {
249 wxTheApp->SetTopWindow(NULL);
250
251 if (wxTheApp->GetExitOnFrameDelete())
252 {
253 // Signal to the app that we're going to close
254 wxTheApp->ExitMainLoop();
255 }
256 }
257}
258
77df2fbc
RR
259void wxTopLevelWindowX11::OnInternalIdle()
260{
261 wxWindow::OnInternalIdle();
ffd84c94 262
193e19cf
RR
263 // Do this only after the last idle event so that
264 // all windows have been updated before a new
265 // round of size events is sent
266 if (m_needResizeInIdle && !wxTheApp->Pending())
77df2fbc
RR
267 {
268 wxSizeEvent event( GetClientSize(), GetId() );
269 event.SetEventObject( this );
270 GetEventHandler()->ProcessEvent( event );
ffd84c94
WS
271
272 m_needResizeInIdle = false;
77df2fbc
RR
273 }
274}
275
1b0fb34b
JS
276// ----------------------------------------------------------------------------
277// wxTopLevelWindowX11 showing
278// ----------------------------------------------------------------------------
83df96d6 279
1b0fb34b 280bool wxTopLevelWindowX11::Show(bool show)
83df96d6 281{
e25f954b
JS
282 if (show)
283 {
284 wxSizeEvent event(GetSize(), GetId());
ffd84c94 285
e25f954b
JS
286 event.SetEventObject(this);
287 GetEventHandler()->ProcessEvent(event);
ffd84c94
WS
288
289 m_needResizeInIdle = false;
e25f954b 290 }
7ded318b 291
df0e1b64 292 bool ret = wxWindowX11::Show(show);
ffd84c94 293
df0e1b64 294 return ret;
83df96d6
JS
295}
296
1b0fb34b
JS
297// ----------------------------------------------------------------------------
298// wxTopLevelWindowX11 maximize/minimize
299// ----------------------------------------------------------------------------
83df96d6 300
1b0fb34b 301void wxTopLevelWindowX11::Maximize(bool maximize)
83df96d6 302{
1b0fb34b 303 // TODO
83df96d6
JS
304}
305
1b0fb34b 306bool wxTopLevelWindowX11::IsMaximized() const
83df96d6 307{
1b0fb34b 308 // TODO
ffd84c94 309 return true;
83df96d6
JS
310}
311
1b0fb34b 312void wxTopLevelWindowX11::Iconize(bool iconize)
83df96d6 313{
b513212d
JS
314 if (!m_iconized && GetMainWindow())
315 {
316 if (XIconifyWindow(wxGlobalDisplay(),
317 (Window) GetMainWindow(), DefaultScreen(wxGlobalDisplay())) != 0)
ffd84c94 318 m_iconized = true;
b513212d 319 }
83df96d6
JS
320}
321
1b0fb34b 322bool wxTopLevelWindowX11::IsIconized() const
83df96d6 323{
1b0fb34b 324 return m_iconized;
83df96d6
JS
325}
326
1b0fb34b 327void wxTopLevelWindowX11::Restore()
83df96d6 328{
b513212d
JS
329 // This is the way to deiconify the window, according to the X FAQ
330 if (m_iconized && GetMainWindow())
331 {
332 XMapWindow(wxGlobalDisplay(), (Window) GetMainWindow());
ffd84c94 333 m_iconized = false;
b513212d 334 }
83df96d6
JS
335}
336
1b0fb34b
JS
337// ----------------------------------------------------------------------------
338// wxTopLevelWindowX11 fullscreen
339// ----------------------------------------------------------------------------
83df96d6 340
1b0fb34b 341bool wxTopLevelWindowX11::ShowFullScreen(bool show, long style)
83df96d6 342{
1b0fb34b 343 if (show)
83df96d6 344 {
1b0fb34b 345 if (IsFullScreen())
ffd84c94 346 return false;
83df96d6 347
ffd84c94 348 m_fsIsShowing = true;
1b0fb34b 349 m_fsStyle = style;
83df96d6 350
1b0fb34b 351 // TODO
83df96d6 352
ffd84c94 353 return true;
83df96d6 354 }
1b0fb34b 355 else
83df96d6 356 {
1b0fb34b 357 if (!IsFullScreen())
ffd84c94 358 return false;
83df96d6 359
ffd84c94 360 m_fsIsShowing = false;
83df96d6 361
1b0fb34b 362 // TODO
ffd84c94 363 return true;
83df96d6
JS
364 }
365}
366
1b0fb34b
JS
367// ----------------------------------------------------------------------------
368// wxTopLevelWindowX11 misc
369// ----------------------------------------------------------------------------
83df96d6 370
f618020a 371void wxTopLevelWindowX11::DoSetIcon(const wxIcon& icon)
83df96d6 372{
b513212d
JS
373 if (icon.Ok() && GetMainWindow())
374 {
461e93f9
JS
375#if wxUSE_NANOX
376#else
b513212d 377 XWMHints *wmHints = XAllocWMHints();
6a44bffd 378 wmHints->icon_pixmap = (Pixmap) icon.GetPixmap();
b513212d 379
6a44bffd 380 wmHints->flags = IconPixmapHint;
b513212d
JS
381
382 if (icon.GetMask())
383 {
6a44bffd 384 wmHints->flags |= IconMaskHint;
a11672a4 385 wmHints->icon_mask = (Pixmap) icon.GetMask()->GetBitmap();
b513212d
JS
386 }
387
1b0b798d 388 XSetWMHints(wxGlobalDisplay(), (Window) GetMainWindow(), wmHints);
b513212d 389 XFree(wmHints);
461e93f9 390#endif
b513212d
JS
391 }
392}
393
f618020a
MB
394void wxTopLevelWindowX11::SetIcons(const wxIconBundle& icons )
395{
396 // this sets m_icon
397 wxTopLevelWindowBase::SetIcons( icons );
398
399 DoSetIcon( icons.GetIcon( -1 ) );
85cb693c 400 wxSetIconsX11( wxGlobalDisplay(), GetMainWindow(), icons );
f618020a
MB
401}
402
f7f78039
MB
403bool wxTopLevelWindowX11::SetShape(const wxRegion& region)
404{
405 return wxDoSetShape( wxGlobalDisplay(),
406 (Window)GetMainWindow(),
407 region );
408}
409
b513212d
JS
410void wxTopLevelWindowX11::SetTitle(const wxString& title)
411{
412 m_title = title;
ffd84c94 413
b513212d
JS
414 if (GetMainWindow())
415 {
2b5f62a0
VZ
416#if wxUSE_UNICODE
417 // I wonder of e.g. Metacity takes UTF-8 here
418 XStoreName(wxGlobalDisplay(), (Window) GetMainWindow(),
419 (const char*) title.ToAscii() );
420 XSetIconName(wxGlobalDisplay(), (Window) GetMainWindow(),
421 (const char*) title.ToAscii() );
422#else
b513212d
JS
423 XStoreName(wxGlobalDisplay(), (Window) GetMainWindow(),
424 (const char*) title);
425 XSetIconName(wxGlobalDisplay(), (Window) GetMainWindow(),
426 (const char*) title);
2b5f62a0 427#endif
c2c0dabf
RR
428 }
429}
430
431wxString wxTopLevelWindowX11::GetTitle() const
432{
433 return m_title;
434}
b28d3abf 435
c2c0dabf
RR
436// For implementation purposes - sometimes decorations make the client area
437// smaller
438wxPoint wxTopLevelWindowX11::GetClientAreaOrigin() const
439{
440 // wxFrame::GetClientAreaOrigin
441 // does the required calculation already.
442 return wxPoint(0, 0);
443}
444
445void wxTopLevelWindowX11::DoGetClientSize( int *width, int *height ) const
446{
447 if (width)
448 *width = m_width;
449 if (height)
450 *height = m_height;
451}
452
e97d2576
RR
453void wxTopLevelWindowX11::DoGetSize( int *width, int *height ) const
454{
455 // TODO add non-client size
456
457 if (width)
458 *width = m_width;
459 if (height)
460 *height = m_height;
461}
462
c2c0dabf
RR
463void wxTopLevelWindowX11::DoSetClientSize(int width, int height)
464{
e97d2576
RR
465 int old_width = m_width;
466 int old_height = m_height;
467
c2c0dabf
RR
468 m_width = width;
469 m_height = height;
ffd84c94 470
e97d2576
RR
471 if (m_width == old_width && m_height == old_height)
472 return;
ffd84c94 473
e97d2576 474 // wxLogDebug("DoSetClientSize: %s (%ld) %dx%d", GetClassInfo()->GetClassName(), GetId(), width, height);
ffd84c94 475
c2c0dabf
RR
476#if !wxUSE_NANOX
477 XSizeHints size_hints;
478 size_hints.flags = PSize;
479 size_hints.width = width;
480 size_hints.height = height;
481 XSetWMNormalHints( wxGlobalDisplay(), (Window) GetMainWindow(), &size_hints );
482#endif
ffd84c94 483
c2c0dabf
RR
484 wxWindowX11::DoSetClientSize(width, height);
485}
486
487void wxTopLevelWindowX11::DoSetSize(int x, int y, int width, int height, int sizeFlags)
488{
e97d2576
RR
489 int old_x = m_x;
490 int old_y = m_y;
491 int old_width = m_width;
492 int old_height = m_height;
c2c0dabf 493
ffd84c94 494 if (x != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf 495 m_x = x;
ffd84c94
WS
496
497 if (y != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf 498 m_y = y;
ffd84c94
WS
499
500 if (width != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf 501 m_width = width;
ffd84c94
WS
502
503 if (height != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf 504 m_height = height;
ffd84c94 505
e97d2576
RR
506 if (m_x == old_x && m_y == old_y && m_width == old_width && m_height == old_height)
507 return;
ffd84c94 508
e97d2576
RR
509 // wxLogDebug("DoSetSize: %s (%ld) %d, %d %dx%d", GetClassInfo()->GetClassName(), GetId(), x, y, width, height);
510
ffd84c94 511#if !wxUSE_NANOX
c2c0dabf
RR
512 XSizeHints size_hints;
513 size_hints.flags = 0;
514 size_hints.flags |= PPosition;
515 size_hints.flags |= PSize;
516 size_hints.x = m_x;
517 size_hints.y = m_y;
518 size_hints.width = m_width;
519 size_hints.height = m_height;
520 XSetWMNormalHints( wxGlobalDisplay(), (Window) GetMainWindow(), &size_hints);
521#endif
522
523 wxWindowX11::DoSetSize(x, y, width, height, sizeFlags);
ffd84c94 524
b513212d 525#if 0
c2c0dabf
RR
526 Display *display = wxGlobalDisplay();
527 Window root = RootWindowOfScreen(DefaultScreenOfDisplay(display));
528 Window parent_window = window,
529 next_parent = window;
530
531 // search for the parent that is child of ROOT, because the WM may
532 // reparent twice and notify only the next parent (like FVWM)
533 while (next_parent != root) {
534 Window *theChildren;
535#if wxUSE_NANOX
536 GR_COUNT n;
537#else
538 unsigned int n;
b513212d 539#endif
c2c0dabf
RR
540 parent_window = next_parent;
541 XQueryTree(display, parent_window, &root,
542 &next_parent, &theChildren, &n);
543 XFree(theChildren); // not needed
544 }
545
546 XWindowChanges windowChanges;
547 windowChanges.x = x;
548 windowChanges.y = y;
549 windowChanges.width = width;
550 windowChanges.height = height;
551 windowChanges.stack_mode = 0;
552 int valueMask = CWX | CWY | CWWidth | CWHeight;
553
ffd84c94 554 if (x != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf
RR
555 {
556 valueMask |= CWX;
557 }
ffd84c94 558 if (y != wxDefaultCoord || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
c2c0dabf
RR
559 {
560 valueMask |= CWY;
b513212d 561 }
ffd84c94 562 if (width != wxDefaultCoord)
c2c0dabf
RR
563 {
564 windowChanges.width = wxMax(1, width);
565 valueMask |= CWWidth;
566 }
ffd84c94 567 if (height != wxDefaultCoord)
c2c0dabf
RR
568 {
569 windowChanges.height = wxMax(1, height);
570 valueMask |= CWHeight;
571 }
572
573 XConfigureWindow( display, parent_window, valueMask, &windowChanges );
574#endif
b513212d
JS
575}
576
c2c0dabf 577void wxTopLevelWindowX11::DoGetPosition(int *x, int *y) const
b513212d 578{
c2c0dabf
RR
579 XSync(wxGlobalDisplay(), False);
580 Window window = (Window) m_mainWindow;
581 if (!window)
582 return ;
583
584 Display *display = wxGlobalDisplay();
585 Window root = RootWindowOfScreen(DefaultScreenOfDisplay(display));
586 Window parent_window = window,
587 next_parent = window;
588
589 // search for the parent that is child of ROOT, because the WM may
590 // reparent twice and notify only the next parent (like FVWM)
591 while (next_parent != root) {
592 Window *theChildren;
593#if wxUSE_NANOX
594 GR_COUNT n;
595#else
596 unsigned int n;
597#endif
598 parent_window = next_parent;
599 XQueryTree(display, parent_window, &root,
600 &next_parent, &theChildren, &n);
601 XFree(theChildren); // not needed
602 }
603#if 0
604 int xx, yy; unsigned int dummy;
605 XGetGeometry(display, parent_window, &root,
606 &xx, &yy, &dummy, &dummy, &dummy, &dummy);
607 if (x) *x = xx;
608 if (y) *y = yy;
609#else
610 XWindowAttributes attr;
611 Status status = XGetWindowAttributes( wxGlobalDisplay(), parent_window, & attr);
612 if (status)
613 {
614 if (x) *x = attr.x;
615 if (y) *y = attr.y;
616 }
617 else
618 {
619 if (x) *x = 0;
620 if (y) *y = 0;
621 }
622#endif
b513212d
JS
623}
624
c2c0dabf 625
b513212d 626#ifndef MWM_DECOR_BORDER
5268c5a3
RR
627
628#define MWM_HINTS_FUNCTIONS (1L << 0)
629#define MWM_HINTS_DECORATIONS (1L << 1)
630#define MWM_HINTS_INPUT_MODE (1L << 2)
631#define MWM_HINTS_STATUS (1L << 3)
b513212d
JS
632
633#define MWM_DECOR_ALL (1L << 0)
634#define MWM_DECOR_BORDER (1L << 1)
635#define MWM_DECOR_RESIZEH (1L << 2)
636#define MWM_DECOR_TITLE (1L << 3)
637#define MWM_DECOR_MENU (1L << 4)
638#define MWM_DECOR_MINIMIZE (1L << 5)
639#define MWM_DECOR_MAXIMIZE (1L << 6)
5268c5a3
RR
640
641#define MWM_FUNC_ALL (1L << 0)
642#define MWM_FUNC_RESIZE (1L << 1)
643#define MWM_FUNC_MOVE (1L << 2)
644#define MWM_FUNC_MINIMIZE (1L << 3)
645#define MWM_FUNC_MAXIMIZE (1L << 4)
646#define MWM_FUNC_CLOSE (1L << 5)
647
648#define MWM_INPUT_MODELESS 0
649#define MWM_INPUT_PRIMARY_APPLICATION_MODAL 1
650#define MWM_INPUT_SYSTEM_MODAL 2
651#define MWM_INPUT_FULL_APPLICATION_MODAL 3
652#define MWM_INPUT_APPLICATION_MODAL MWM_INPUT_PRIMARY_APPLICATION_MODAL
653
ffd84c94 654#define MWM_TEAROFF_WINDOW (1L<<0)
5268c5a3 655
b513212d
JS
656#endif
657
658struct MwmHints {
659 long flags;
660 long functions;
661 long decorations;
662 long input_mode;
663};
664
665#define PROP_MOTIF_WM_HINTS_ELEMENTS 5
666
667// Set the window manager decorations according to the
77ffb593 668// given wxWidgets style
b28d3abf 669bool wxSetWMDecorations(Window w, long style)
b513212d 670{
461e93f9
JS
671#if wxUSE_NANOX
672 GR_WM_PROPERTIES wmProp;
673
674 wmProp.flags = 0;
0b5c0e1a 675 wmProp.props = 0;
461e93f9
JS
676
677 if (style & wxRESIZE_BORDER)
678 {
679 wmProp.props |= GR_WM_PROPS_APPFRAME ;
680 wmProp.flags |= GR_WM_FLAGS_PROPS ;
681 }
682
850c6ed4 683 if (style & wxCLOSE_BOX)
461e93f9
JS
684 {
685 wmProp.props |= GR_WM_PROPS_CLOSEBOX ;
686 wmProp.flags |= GR_WM_FLAGS_PROPS ;
687 }
688
689 if ((style & wxCAPTION) ||
690 (style & wxTINY_CAPTION_HORIZ) ||
691 (style & wxTINY_CAPTION_VERT))
692 {
693 wmProp.props |= GR_WM_PROPS_CAPTION ;
694 wmProp.flags |= GR_WM_FLAGS_PROPS ;
bc55104d
JS
695
696 // The default dialog style doesn't include any kind
697 // of border, which is a bit odd. Anyway, inclusion
698 // of a caption surely implies a border.
699 style |= wxTHICK_FRAME;
461e93f9
JS
700 }
701
702 if (style & wxTHICK_FRAME)
703 {
704 wmProp.props |= GR_WM_PROPS_APPFRAME ;
705 wmProp.flags |= GR_WM_FLAGS_PROPS ;
706 }
707
708 if (style & wxSIMPLE_BORDER)
709 {
710 wmProp.props |= GR_WM_PROPS_BORDER ;
711 wmProp.flags |= GR_WM_FLAGS_PROPS ;
712 }
713
714 if (style & wxMINIMIZE_BOX)
715 {
716 }
717
718 if (style & wxMAXIMIZE_BOX)
719 {
720 wmProp.props |= GR_WM_PROPS_MAXIMIZE ;
721 wmProp.flags |= GR_WM_FLAGS_PROPS ;
722 }
723
724 if (((style & wxBORDER) != wxBORDER) && ((style & wxTHICK_FRAME) != wxTHICK_FRAME)
725 && ((style & wxRESIZE_BORDER) != wxRESIZE_BORDER))
726 {
727 wmProp.props |= GR_WM_PROPS_NODECORATE ;
728 wmProp.flags |= GR_WM_FLAGS_PROPS ;
729 }
730
731 GrSetWMProperties(w, & wmProp);
ffd84c94 732
461e93f9 733#else
b513212d
JS
734
735 Atom mwm_wm_hints = XInternAtom(wxGlobalDisplay(),"_MOTIF_WM_HINTS", False);
5268c5a3 736 if (mwm_wm_hints == 0)
ffd84c94
WS
737 return false;
738
b513212d 739 MwmHints hints;
5268c5a3 740 hints.flags = MWM_HINTS_DECORATIONS | MWM_HINTS_FUNCTIONS;
b513212d 741 hints.decorations = 0;
5268c5a3 742 hints.functions = 0;
ffd84c94 743
5268c5a3 744 if ((style & wxSIMPLE_BORDER) || (style & wxNO_BORDER))
b513212d 745 {
5268c5a3 746 // leave zeros
b513212d 747 }
5268c5a3 748 else
b513212d 749 {
5268c5a3
RR
750 hints.decorations = MWM_DECOR_BORDER;
751 hints.functions = MWM_FUNC_MOVE;
b513212d 752
5268c5a3
RR
753 if ((style & wxCAPTION) != 0)
754 hints.decorations |= MWM_DECOR_TITLE;
ffd84c94 755
5268c5a3 756 if ((style & wxSYSTEM_MENU) != 0)
5268c5a3 757 hints.decorations |= MWM_DECOR_MENU;
ffd84c94 758
850c6ed4 759 if ((style & wxCLOSE_BOX) != 0)
c3d8ee42 760 hints.functions |= MWM_FUNC_CLOSE;
ffd84c94 761
5268c5a3
RR
762 if ((style & wxMINIMIZE_BOX) != 0)
763 {
764 hints.functions |= MWM_FUNC_MINIMIZE;
765 hints.decorations |= MWM_DECOR_MINIMIZE;
766 }
ffd84c94 767
5268c5a3
RR
768 if ((style & wxMAXIMIZE_BOX) != 0)
769 {
770 hints.functions |= MWM_FUNC_MAXIMIZE;
771 hints.decorations |= MWM_DECOR_MAXIMIZE;
772 }
ffd84c94 773
5268c5a3
RR
774 if ((style & wxRESIZE_BORDER) != 0)
775 {
776 hints.functions |= MWM_FUNC_RESIZE;
777 hints.decorations |= MWM_DECOR_RESIZEH;
778 }
b513212d
JS
779 }
780
781 XChangeProperty(wxGlobalDisplay(),
ffd84c94
WS
782 w,
783 mwm_wm_hints, mwm_wm_hints,
784 32, PropModeReplace,
785 (unsigned char *) &hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
b513212d 786
461e93f9 787#endif
ffd84c94 788 return true;
b513212d 789}