]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/motif/utils.cpp
Fix build with wxUSE_FFILE=0.
[wxWidgets.git] / src / motif / utils.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/motif/utils.cpp
3// Purpose: Various utilities
4// Author: Julian Smart
5// Modified by:
6// Created: 17/09/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#include "wx/utils.h"
24
25#ifndef WX_PRECOMP
26 #include "wx/app.h"
27 #include "wx/dcmemory.h"
28 #include "wx/bitmap.h"
29#endif
30
31#include "wx/apptrait.h"
32#include "wx/evtloop.h"
33#include "wx/private/eventloopsourcesmanager.h"
34#include "wx/motif/private/timer.h"
35
36#include <string.h>
37
38#if (defined(__SUNCC__) || defined(__CLCC__))
39 #include <sysent.h>
40#endif
41
42#ifdef __VMS__
43#pragma message disable nosimpint
44#endif
45
46#include <Xm/Xm.h>
47#include <Xm/Frame.h>
48
49#include "wx/motif/private.h"
50
51#include "X11/Xutil.h"
52
53#ifdef __VMS__
54#pragma message enable nosimpint
55#endif
56
57
58// ============================================================================
59// implementation
60// ============================================================================
61
62// ----------------------------------------------------------------------------
63// async event processing
64// ----------------------------------------------------------------------------
65
66// Consume all events until no more left
67void wxFlushEvents(WXDisplay* wxdisplay)
68{
69 Display *display = (Display*)wxdisplay;
70 wxEventLoop evtLoop;
71
72 XSync (display, False);
73
74 while (evtLoop.Pending())
75 {
76 XFlush (display);
77 evtLoop.Dispatch();
78 }
79}
80
81#if wxUSE_EVENTLOOP_SOURCE
82
83extern "C"
84{
85
86static
87void
88wxMotifInputHandler(XtPointer data,
89 int* WXUNUSED(fd),
90 XtInputId* WXUNUSED(inputId))
91{
92 wxEventLoopSourceHandler * const
93 handler = static_cast<wxEventLoopSourceHandler *>(data);
94
95 handler->OnReadWaiting();
96}
97
98}
99
100// This class exists just to call XtRemoveInput() in its dtor, the real work of
101// dispatching events on the file descriptor to the handler is done by
102// wxMotifInputHandler callback above.
103class wxMotifEventLoopSource : public wxEventLoopSource
104{
105public:
106 wxMotifEventLoopSource(XtInputId inputId,
107 wxEventLoopSourceHandler *handler,
108 int flags)
109 : wxEventLoopSource(handler, flags),
110 m_inputId(inputId)
111 {
112 }
113
114 virtual ~wxMotifEventLoopSource()
115 {
116 XtRemoveInput(m_inputId);
117 }
118
119private:
120 const XtInputId m_inputId;
121
122 wxDECLARE_NO_COPY_CLASS(wxMotifEventLoopSource);
123};
124
125class wxMotifEventLoopSourcesManager : public wxEventLoopSourcesManagerBase
126{
127public:
128 wxEventLoopSource *
129 AddSourceForFD(int fd, wxEventLoopSourceHandler* handler, int flags)
130 {
131 wxCHECK_MSG( wxTheApp, NULL, "Must create wxTheApp first" );
132
133 // The XtInputXXXMask values cannot be combined (hence "Mask" is a
134 // complete misnomer), and supporting those would make the code more
135 // complicated and we don't need them for now.
136 wxCHECK_MSG( !(flags & (wxEVENT_SOURCE_OUTPUT |
137 wxEVENT_SOURCE_EXCEPTION)),
138 NULL,
139 "Monitoring FDs for output/errors not supported" );
140
141 wxCHECK_MSG( flags & wxEVENT_SOURCE_INPUT,
142 NULL,
143 "Should be monitoring for input" );
144
145 XtInputId inputId = XtAppAddInput
146 (
147 (XtAppContext) wxTheApp->GetAppContext(),
148 fd,
149 (XtPointer) XtInputReadMask,
150 wxMotifInputHandler,
151 handler
152 );
153 if ( inputId < 0 )
154 return 0;
155
156 return new wxMotifEventLoopSource(inputId, handler, flags);
157 }
158};
159
160wxEventLoopSourcesManagerBase* wxGUIAppTraits::GetEventLoopSourcesManager()
161{
162 static wxMotifEventLoopSourcesManager s_eventLoopSourcesManager;
163
164 return &s_eventLoopSourcesManager;
165}
166
167#endif // wxUSE_EVENTLOOP_SOURCE
168
169// ----------------------------------------------------------------------------
170// misc
171// ----------------------------------------------------------------------------
172
173// Emit a beeeeeep
174void wxBell()
175{
176 // Use current setting for the bell
177 XBell (wxGlobalDisplay(), 0);
178}
179
180wxPortId wxGUIAppTraits::GetToolkitVersion(int *verMaj, int *verMin) const
181{
182 // XmVERSION and XmREVISION are defined in Xm/Xm.h
183 if ( verMaj )
184 *verMaj = XmVERSION;
185 if ( verMin )
186 *verMin = XmREVISION;
187
188 return wxPORT_MOTIF;
189}
190
191wxEventLoopBase* wxGUIAppTraits::CreateEventLoop()
192{
193 return new wxEventLoop;
194}
195
196wxTimerImpl* wxGUIAppTraits::CreateTimerImpl(wxTimer* timer)
197{
198 return new wxMotifTimerImpl(timer);
199}
200
201// ----------------------------------------------------------------------------
202// display info
203// ----------------------------------------------------------------------------
204
205void wxGetMousePosition( int* x, int* y )
206{
207#if wxUSE_NANOX
208 // TODO
209 *x = 0;
210 *y = 0;
211#else
212 XMotionEvent xev;
213 Window root, child;
214 XQueryPointer(wxGlobalDisplay(),
215 DefaultRootWindow(wxGlobalDisplay()),
216 &root, &child,
217 &(xev.x_root), &(xev.y_root),
218 &(xev.x), &(xev.y),
219 &(xev.state));
220 *x = xev.x_root;
221 *y = xev.y_root;
222#endif
223}
224
225// Return true if we have a colour display
226bool wxColourDisplay()
227{
228 return wxDisplayDepth() > 1;
229}
230
231// Returns depth of screen
232int wxDisplayDepth()
233{
234 Display *dpy = wxGlobalDisplay();
235
236 return DefaultDepth (dpy, DefaultScreen (dpy));
237}
238
239// Get size of display
240void wxDisplaySize(int *width, int *height)
241{
242 Display *dpy = wxGlobalDisplay();
243
244 if ( width )
245 *width = DisplayWidth (dpy, DefaultScreen (dpy));
246 if ( height )
247 *height = DisplayHeight (dpy, DefaultScreen (dpy));
248}
249
250void wxDisplaySizeMM(int *width, int *height)
251{
252 Display *dpy = wxGlobalDisplay();
253
254 if ( width )
255 *width = DisplayWidthMM(dpy, DefaultScreen (dpy));
256 if ( height )
257 *height = DisplayHeightMM(dpy, DefaultScreen (dpy));
258}
259
260// Configurable display in wxX11 and wxMotif
261static WXDisplay *gs_currentDisplay = NULL;
262static wxString gs_displayName;
263
264WXDisplay *wxGetDisplay()
265{
266 if (gs_currentDisplay)
267 return gs_currentDisplay;
268 else if (wxTheApp)
269 return wxTheApp->GetInitialDisplay();
270 return NULL;
271}
272
273bool wxSetDisplay(const wxString& display_name)
274{
275 gs_displayName = display_name;
276
277 if ( display_name.empty() )
278 {
279 gs_currentDisplay = NULL;
280
281 return true;
282 }
283 else
284 {
285 Cardinal argc = 0;
286
287 Display *display = XtOpenDisplay((XtAppContext) wxTheApp->GetAppContext(),
288 display_name.c_str(),
289 wxTheApp->GetAppName().c_str(),
290 wxTheApp->GetClassName().c_str(),
291 NULL,
292#if XtSpecificationRelease < 5
293 0, &argc,
294#else
295 0, (int *)&argc,
296#endif
297 NULL);
298
299 if (display)
300 {
301 gs_currentDisplay = (WXDisplay*) display;
302 return true;
303 }
304 else
305 return false;
306 }
307}
308
309wxString wxGetDisplayName()
310{
311 return gs_displayName;
312}
313
314wxWindow* wxFindWindowAtPoint(const wxPoint& pt)
315{
316 return wxGenericFindWindowAtPoint(pt);
317}
318
319// ----------------------------------------------------------------------------
320// Some colour manipulation routines
321// ----------------------------------------------------------------------------
322
323void wxHSVToXColor(wxHSV *hsv,XColor *rgb)
324{
325 int h = hsv->h;
326 int s = hsv->s;
327 int v = hsv->v;
328 int r = 0, g = 0, b = 0;
329 int i, f;
330 int p, q, t;
331 s = (s * wxMAX_RGB) / wxMAX_SV;
332 v = (v * wxMAX_RGB) / wxMAX_SV;
333 if (h == 360) h = 0;
334 if (s == 0) { h = 0; r = g = b = v; }
335 i = h / 60;
336 f = h % 60;
337 p = v * (wxMAX_RGB - s) / wxMAX_RGB;
338 q = v * (wxMAX_RGB - s * f / 60) / wxMAX_RGB;
339 t = v * (wxMAX_RGB - s * (60 - f) / 60) / wxMAX_RGB;
340 switch (i)
341 {
342 case 0: r = v, g = t, b = p; break;
343 case 1: r = q, g = v, b = p; break;
344 case 2: r = p, g = v, b = t; break;
345 case 3: r = p, g = q, b = v; break;
346 case 4: r = t, g = p, b = v; break;
347 case 5: r = v, g = p, b = q; break;
348 }
349 rgb->red = (unsigned short)(r << 8);
350 rgb->green = (unsigned short)(g << 8);
351 rgb->blue = (unsigned short)(b << 8);
352}
353
354void wxXColorToHSV(wxHSV *hsv,XColor *rgb)
355{
356 int r = rgb->red >> 8;
357 int g = rgb->green >> 8;
358 int b = rgb->blue >> 8;
359 int maxv = wxMax3(r, g, b);
360 int minv = wxMin3(r, g, b);
361 int h = 0, s, v;
362 v = maxv;
363 if (maxv) s = (maxv - minv) * wxMAX_RGB / maxv;
364 else s = 0;
365 if (s == 0) h = 0;
366 else
367 {
368 int rc, gc, bc, hex = 0;
369 rc = (maxv - r) * wxMAX_RGB / (maxv - minv);
370 gc = (maxv - g) * wxMAX_RGB / (maxv - minv);
371 bc = (maxv - b) * wxMAX_RGB / (maxv - minv);
372 if (r == maxv) { h = bc - gc, hex = 0; }
373 else if (g == maxv) { h = rc - bc, hex = 2; }
374 else if (b == maxv) { h = gc - rc, hex = 4; }
375 h = hex * 60 + (h * 60 / wxMAX_RGB);
376 if (h < 0) h += 360;
377 }
378 hsv->h = h;
379 hsv->s = (s * wxMAX_SV) / wxMAX_RGB;
380 hsv->v = (v * wxMAX_SV) / wxMAX_RGB;
381}
382
383void wxAllocNearestColor(Display *d,Colormap cmp,XColor *xc)
384{
385#if !wxUSE_NANOX
386 int llp;
387
388 int screen = DefaultScreen(d);
389 int num_colors = DisplayCells(d,screen);
390
391 XColor *color_defs = new XColor[num_colors];
392 for(llp = 0;llp < num_colors;llp++) color_defs[llp].pixel = llp;
393 XQueryColors(d,cmp,color_defs,num_colors);
394
395 wxHSV hsv_defs, hsv;
396 wxXColorToHSV(&hsv,xc);
397
398 int diff, min_diff = 0, pixel = 0;
399
400 for(llp = 0;llp < num_colors;llp++)
401 {
402 wxXColorToHSV(&hsv_defs,&color_defs[llp]);
403 diff = wxSIGN(wxH_WEIGHT * (hsv.h - hsv_defs.h)) +
404 wxSIGN(wxS_WEIGHT * (hsv.s - hsv_defs.s)) +
405 wxSIGN(wxV_WEIGHT * (hsv.v - hsv_defs.v));
406 if (llp == 0) min_diff = diff;
407 if (min_diff > diff) { min_diff = diff; pixel = llp; }
408 if (min_diff == 0) break;
409 }
410
411 xc -> red = color_defs[pixel].red;
412 xc -> green = color_defs[pixel].green;
413 xc -> blue = color_defs[pixel].blue;
414 xc -> flags = DoRed | DoGreen | DoBlue;
415
416/* FIXME, TODO
417 if (!XAllocColor(d,cmp,xc))
418 cout << "wxAllocNearestColor : Warning : Cannot find nearest color !\n";
419*/
420
421 delete[] color_defs;
422#endif
423}
424
425void wxAllocColor(Display *d,Colormap cmp,XColor *xc)
426{
427 if (!XAllocColor(d,cmp,xc))
428 {
429 // cout << "wxAllocColor : Warning : cannot allocate color, attempt find nearest !\n";
430 wxAllocNearestColor(d,cmp,xc);
431 }
432}
433
434wxString wxGetXEventName(XEvent& event)
435{
436#if wxUSE_NANOX
437 wxString str(wxT("(some event)"));
438 return str;
439#else
440 int type = event.xany.type;
441 static char* event_name[] = {
442 wxMOTIF_STR(""), wxMOTIF_STR("unknown(-)"), // 0-1
443 wxMOTIF_STR("KeyPress"), wxMOTIF_STR("KeyRelease"), wxMOTIF_STR("ButtonPress"), wxMOTIF_STR("ButtonRelease"), // 2-5
444 wxMOTIF_STR("MotionNotify"), wxMOTIF_STR("EnterNotify"), wxMOTIF_STR("LeaveNotify"), wxMOTIF_STR("FocusIn"), // 6-9
445 wxMOTIF_STR("FocusOut"), wxMOTIF_STR("KeymapNotify"), wxMOTIF_STR("Expose"), wxMOTIF_STR("GraphicsExpose"), // 10-13
446 wxMOTIF_STR("NoExpose"), wxMOTIF_STR("VisibilityNotify"), wxMOTIF_STR("CreateNotify"), // 14-16
447 wxMOTIF_STR("DestroyNotify"), wxMOTIF_STR("UnmapNotify"), wxMOTIF_STR("MapNotify"), wxMOTIF_STR("MapRequest"),// 17-20
448 wxMOTIF_STR("ReparentNotify"), wxMOTIF_STR("ConfigureNotify"), wxMOTIF_STR("ConfigureRequest"), // 21-23
449 wxMOTIF_STR("GravityNotify"), wxMOTIF_STR("ResizeRequest"), wxMOTIF_STR("CirculateNotify"), // 24-26
450 wxMOTIF_STR("CirculateRequest"), wxMOTIF_STR("PropertyNotify"), wxMOTIF_STR("SelectionClear"), // 27-29
451 wxMOTIF_STR("SelectionRequest"), wxMOTIF_STR("SelectionNotify"), wxMOTIF_STR("ColormapNotify"), // 30-32
452 wxMOTIF_STR("ClientMessage"), wxMOTIF_STR("MappingNotify"), // 33-34
453 wxMOTIF_STR("unknown(+)")}; // 35
454 type = wxMin(35, type); type = wxMax(1, type);
455 wxString str(event_name[type]);
456 return str;
457#endif
458}
459
460// ----------------------------------------------------------------------------
461// accelerators
462// ----------------------------------------------------------------------------
463
464// Find the letter corresponding to the mnemonic, for Motif
465char wxFindMnemonic (const char *s)
466{
467 char mnem = 0;
468 int len = strlen (s);
469 int i;
470
471 for (i = 0; i < len; i++)
472 {
473 if (s[i] == '&')
474 {
475 // Carefully handle &&
476 if ((i + 1) <= len && s[i + 1] == '&')
477 i++;
478 else
479 {
480 mnem = s[i + 1];
481 break;
482 }
483 }
484 }
485 return mnem;
486}
487
488char* wxFindAccelerator( const char *s )
489{
490#if 1
491 wxUnusedVar(s);
492 // VZ: this function returns incorrect keysym which completely breaks kbd
493 // handling
494 return NULL;
495#else
496 // The accelerator text is after the \t char.
497 s = strchr( s, '\t' );
498
499 if( !s ) return NULL;
500
501 /*
502 Now we need to format it as X standard:
503
504 input output
505
506 F7 --> <Key>F7
507 Ctrl+N --> Ctrl<Key>N
508 Alt+k --> Meta<Key>k
509 Ctrl+Shift+A --> Ctrl Shift<Key>A
510
511 and handle Ctrl-N & similia
512 */
513
514 static char buf[256];
515
516 buf[0] = '\0';
517 wxString tmp = s + 1; // skip TAB
518 size_t index = 0;
519
520 while( index < tmp.length() )
521 {
522 size_t plus = tmp.find( '+', index );
523 size_t minus = tmp.find( '-', index );
524
525 // neither '+' nor '-', add <Key>
526 if( plus == wxString::npos && minus == wxString::npos )
527 {
528 strcat( buf, "<Key>" );
529 strcat( buf, tmp.c_str() + index );
530
531 return buf;
532 }
533
534 // OK: npos is big and positive
535 size_t sep = wxMin( plus, minus );
536 wxString mod = tmp.substr( index, sep - index );
537
538 // Ctrl -> Ctrl
539 // Shift -> Shift
540 // Alt -> Meta
541 if( mod == "Alt" )
542 mod = "Meta";
543
544 if( buf[0] )
545 strcat( buf, " " );
546
547 strcat( buf, mod.c_str() );
548
549 index = sep + 1;
550 }
551
552 return NULL;
553#endif
554}
555
556XmString wxFindAcceleratorText (const char *s)
557{
558#if 1
559 wxUnusedVar(s);
560 // VZ: this function returns incorrect keysym which completely breaks kbd
561 // handling
562 return NULL;
563#else
564 // The accelerator text is after the \t char.
565 s = strchr( s, '\t' );
566
567 if( !s ) return NULL;
568
569 return wxStringToXmString( s + 1 ); // skip TAB!
570#endif
571}
572
573// Change a widget's foreground and background colours.
574void wxDoChangeForegroundColour(WXWidget widget, wxColour& foregroundColour)
575{
576 if (!foregroundColour.IsOk())
577 return;
578
579 // When should we specify the foreground, if it's calculated
580 // by wxComputeColours?
581 // Solution: say we start with the default (computed) foreground colour.
582 // If we call SetForegroundColour explicitly for a control or window,
583 // then the foreground is changed.
584 // Therefore SetBackgroundColour computes the foreground colour, and
585 // SetForegroundColour changes the foreground colour. The ordering is
586 // important.
587
588 XtVaSetValues ((Widget) widget,
589 XmNforeground, foregroundColour.AllocColour(XtDisplay((Widget) widget)),
590 NULL);
591}
592
593void wxDoChangeBackgroundColour(WXWidget widget, const wxColour& backgroundColour, bool changeArmColour)
594{
595 if (!backgroundColour.IsOk())
596 return;
597
598 wxComputeColours (XtDisplay((Widget) widget), & backgroundColour,
599 NULL);
600
601 XtVaSetValues ((Widget) widget,
602 XmNbackground, g_itemColors[wxBACK_INDEX].pixel,
603 XmNtopShadowColor, g_itemColors[wxTOPS_INDEX].pixel,
604 XmNbottomShadowColor, g_itemColors[wxBOTS_INDEX].pixel,
605 XmNforeground, g_itemColors[wxFORE_INDEX].pixel,
606 NULL);
607
608 if (changeArmColour)
609 XtVaSetValues ((Widget) widget,
610 XmNarmColor, g_itemColors[wxSELE_INDEX].pixel,
611 NULL);
612}
613
614extern void wxDoChangeFont(WXWidget widget, const wxFont& font)
615{
616 // Lesstif 0.87 hangs here, but 0.93 does not; MBN: sometimes it does
617#if !wxCHECK_LESSTIF() // || wxCHECK_LESSTIF_VERSION( 0, 93 )
618 Widget w = (Widget)widget;
619 XtVaSetValues( w,
620 wxFont::GetFontTag(), font.GetFontTypeC( XtDisplay(w) ),
621 NULL );
622#else
623 wxUnusedVar(widget);
624 wxUnusedVar(font);
625#endif
626
627}
628
629wxString wxXmStringToString( const XmString& xmString )
630{
631 char *txt;
632 if( XmStringGetLtoR( xmString, XmSTRING_DEFAULT_CHARSET, &txt ) )
633 {
634 wxString str(txt);
635 XtFree (txt);
636 return str;
637 }
638
639 return wxEmptyString;
640}
641
642XmString wxStringToXmString( const char* str )
643{
644 return XmStringCreateLtoR((char *)str, XmSTRING_DEFAULT_CHARSET);
645}
646
647// ----------------------------------------------------------------------------
648// wxBitmap utility functions
649// ----------------------------------------------------------------------------
650
651// Creates a bitmap with transparent areas drawn in
652// the given colour.
653wxBitmap wxCreateMaskedBitmap(const wxBitmap& bitmap, const wxColour& colour)
654{
655 wxBitmap newBitmap(bitmap.GetWidth(),
656 bitmap.GetHeight(),
657 bitmap.GetDepth());
658 wxMemoryDC destDC;
659 wxMemoryDC srcDC;
660
661 srcDC.SelectObjectAsSource(bitmap);
662 destDC.SelectObject(newBitmap);
663
664 wxBrush brush(colour, wxSOLID);
665 destDC.SetBackground(brush);
666 destDC.Clear();
667 destDC.Blit(0, 0, bitmap.GetWidth(), bitmap.GetHeight(),
668 &srcDC, 0, 0, wxCOPY, true);
669
670 return newBitmap;
671}
672
673// ----------------------------------------------------------------------------
674// Miscellaneous functions
675// ----------------------------------------------------------------------------
676
677WXWidget wxCreateBorderWidget( WXWidget parent, long style )
678{
679 Widget borderWidget = (Widget)NULL, parentWidget = (Widget)parent;
680
681 if (style & wxSIMPLE_BORDER)
682 {
683 borderWidget = XtVaCreateManagedWidget
684 (
685 "simpleBorder",
686 xmFrameWidgetClass, parentWidget,
687 XmNshadowType, XmSHADOW_ETCHED_IN,
688 XmNshadowThickness, 1,
689 NULL
690 );
691 }
692 else if ((style & wxSUNKEN_BORDER) || (style & wxBORDER_THEME))
693 {
694 borderWidget = XtVaCreateManagedWidget
695 (
696 "sunkenBorder",
697 xmFrameWidgetClass, parentWidget,
698 XmNshadowType, XmSHADOW_IN,
699 NULL
700 );
701 }
702 else if (style & wxRAISED_BORDER)
703 {
704 borderWidget = XtVaCreateManagedWidget
705 (
706 "raisedBorder",
707 xmFrameWidgetClass, parentWidget,
708 XmNshadowType, XmSHADOW_OUT,
709 NULL
710 );
711 }
712
713 return borderWidget;
714}