1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/utilscmn.cpp
3 // Purpose: Miscellaneous utility functions and classes
4 // Author: Julian Smart
8 // Copyright: (c) 1998 Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
29 #include "wx/string.h"
35 #include "wx/window.h"
38 #include "wx/msgdlg.h"
39 #include "wx/textdlg.h"
40 #include "wx/textctrl.h" // for wxTE_PASSWORD
42 #include "wx/menuitem.h"
48 #include "wx/apptrait.h"
50 #include "wx/process.h"
51 #include "wx/txtstrm.h"
53 #include "wx/mimetype.h"
54 #include "wx/config.h"
56 #if defined(__WXWINCE__) && wxUSE_DATETIME
57 #include "wx/datetime.h"
65 #if !wxONLY_WATCOM_EARLIER_THAN(1,4)
66 #if !(defined(_MSC_VER) && (_MSC_VER > 800))
72 #include "wx/colordlg.h"
73 #include "wx/fontdlg.h"
74 #include "wx/notebook.h"
75 #include "wx/statusbr.h"
82 #include "wx/msw/wince/time.h"
84 #endif // ! __WXPALMOS5__
87 #include "wx/osx/private.h"
91 #if !defined(__MWERKS__) && !defined(__WXWINCE__)
92 #include <sys/types.h>
95 #endif // ! __WXPALMOS5__
97 #if defined(__WXMSW__)
98 #include "wx/msw/private.h"
99 #include "wx/msw/registry.h"
100 #include <shellapi.h> // needed for SHELLEXECUTEINFO
103 #if wxUSE_GUI && defined(__WXGTK__)
104 #include <gtk/gtk.h> // for GTK_XXX_VERSION constants
109 // ----------------------------------------------------------------------------
111 // ----------------------------------------------------------------------------
113 // ============================================================================
115 // ============================================================================
117 // Array used in DecToHex conversion routine.
118 static wxChar hexArray
[] = wxT("0123456789ABCDEF");
120 // Convert 2-digit hex number to decimal
121 int wxHexToDec(const wxString
& buf
)
123 int firstDigit
, secondDigit
;
125 if (buf
.GetChar(0) >= wxT('A'))
126 firstDigit
= buf
.GetChar(0) - wxT('A') + 10;
128 firstDigit
= buf
.GetChar(0) - wxT('0');
130 if (buf
.GetChar(1) >= wxT('A'))
131 secondDigit
= buf
.GetChar(1) - wxT('A') + 10;
133 secondDigit
= buf
.GetChar(1) - wxT('0');
135 return (firstDigit
& 0xF) * 16 + (secondDigit
& 0xF );
138 // Convert decimal integer to 2-character hex string
139 void wxDecToHex(int dec
, wxChar
*buf
)
141 int firstDigit
= (int)(dec
/16.0);
142 int secondDigit
= (int)(dec
- (firstDigit
*16.0));
143 buf
[0] = hexArray
[firstDigit
];
144 buf
[1] = hexArray
[secondDigit
];
148 // Convert decimal integer to 2 characters
149 void wxDecToHex(int dec
, char* ch1
, char* ch2
)
151 int firstDigit
= (int)(dec
/16.0);
152 int secondDigit
= (int)(dec
- (firstDigit
*16.0));
153 (*ch1
) = (char) hexArray
[firstDigit
];
154 (*ch2
) = (char) hexArray
[secondDigit
];
157 // Convert decimal integer to 2-character hex string
158 wxString
wxDecToHex(int dec
)
161 wxDecToHex(dec
, buf
);
162 return wxString(buf
);
165 // ----------------------------------------------------------------------------
167 // ----------------------------------------------------------------------------
169 // Return the current date/time
174 wxDateTime now
= wxDateTime::Now();
177 return wxEmptyString
;
180 time_t now
= time((time_t *) NULL
);
181 char *date
= ctime(&now
);
183 return wxString::FromAscii(date
);
187 void wxUsleep(unsigned long milliseconds
)
189 wxMilliSleep(milliseconds
);
192 const wxChar
*wxGetInstallPrefix()
196 if ( wxGetEnv(wxT("WXPREFIX"), &prefix
) )
197 return prefix
.c_str();
199 #ifdef wxINSTALL_PREFIX
200 return wxT(wxINSTALL_PREFIX
);
202 return wxEmptyString
;
206 wxString
wxGetDataDir()
208 wxString dir
= wxGetInstallPrefix();
209 dir
<< wxFILE_SEP_PATH
<< wxT("share") << wxFILE_SEP_PATH
<< wxT("wx");
213 bool wxIsPlatformLittleEndian()
215 // Are we little or big endian? This method is from Harbison & Steele.
219 char c
[sizeof(long)];
228 * Class to make it easier to specify platform-dependent values
231 wxArrayInt
* wxPlatform::sm_customPlatforms
= NULL
;
233 void wxPlatform::Copy(const wxPlatform
& platform
)
235 m_longValue
= platform
.m_longValue
;
236 m_doubleValue
= platform
.m_doubleValue
;
237 m_stringValue
= platform
.m_stringValue
;
240 wxPlatform
wxPlatform::If(int platform
, long value
)
243 return wxPlatform(value
);
248 wxPlatform
wxPlatform::IfNot(int platform
, long value
)
251 return wxPlatform(value
);
256 wxPlatform
& wxPlatform::ElseIf(int platform
, long value
)
263 wxPlatform
& wxPlatform::ElseIfNot(int platform
, long value
)
270 wxPlatform
wxPlatform::If(int platform
, double value
)
273 return wxPlatform(value
);
278 wxPlatform
wxPlatform::IfNot(int platform
, double value
)
281 return wxPlatform(value
);
286 wxPlatform
& wxPlatform::ElseIf(int platform
, double value
)
289 m_doubleValue
= value
;
293 wxPlatform
& wxPlatform::ElseIfNot(int platform
, double value
)
296 m_doubleValue
= value
;
300 wxPlatform
wxPlatform::If(int platform
, const wxString
& value
)
303 return wxPlatform(value
);
308 wxPlatform
wxPlatform::IfNot(int platform
, const wxString
& value
)
311 return wxPlatform(value
);
316 wxPlatform
& wxPlatform::ElseIf(int platform
, const wxString
& value
)
319 m_stringValue
= value
;
323 wxPlatform
& wxPlatform::ElseIfNot(int platform
, const wxString
& value
)
326 m_stringValue
= value
;
330 wxPlatform
& wxPlatform::Else(long value
)
336 wxPlatform
& wxPlatform::Else(double value
)
338 m_doubleValue
= value
;
342 wxPlatform
& wxPlatform::Else(const wxString
& value
)
344 m_stringValue
= value
;
348 void wxPlatform::AddPlatform(int platform
)
350 if (!sm_customPlatforms
)
351 sm_customPlatforms
= new wxArrayInt
;
352 sm_customPlatforms
->Add(platform
);
355 void wxPlatform::ClearPlatforms()
357 delete sm_customPlatforms
;
358 sm_customPlatforms
= NULL
;
361 /// Function for testing current platform
363 bool wxPlatform::Is(int platform
)
366 if (platform
== wxOS_WINDOWS
)
370 if (platform
== wxOS_WINDOWS_CE
)
376 // FIXME: wxWinPocketPC and wxWinSmartPhone are unknown symbols
378 #if defined(__WXWINCE__) && defined(__POCKETPC__)
379 if (platform
== wxWinPocketPC
)
382 #if defined(__WXWINCE__) && defined(__SMARTPHONE__)
383 if (platform
== wxWinSmartPhone
)
390 if (platform
== wxPORT_GTK
)
394 if (platform
== wxPORT_MAC
)
398 if (platform
== wxPORT_X11
)
402 if (platform
== wxOS_UNIX
)
406 if (platform
== wxPORT_MGL
)
410 if (platform
== wxOS_OS2
)
414 if (platform
== wxPORT_PM
)
418 if (platform
== wxPORT_MAC
)
422 if (sm_customPlatforms
&& sm_customPlatforms
->Index(platform
) != wxNOT_FOUND
)
428 // ----------------------------------------------------------------------------
429 // network and user id functions
430 // ----------------------------------------------------------------------------
432 // Get Full RFC822 style email address
433 bool wxGetEmailAddress(wxChar
*address
, int maxSize
)
435 wxString email
= wxGetEmailAddress();
439 wxStrlcpy(address
, email
.t_str(), maxSize
);
444 wxString
wxGetEmailAddress()
448 wxString host
= wxGetFullHostName();
451 wxString user
= wxGetUserId();
454 email
<< user
<< wxT('@') << host
;
461 wxString
wxGetUserId()
463 static const int maxLoginLen
= 256; // FIXME arbitrary number
466 bool ok
= wxGetUserId(wxStringBuffer(buf
, maxLoginLen
), maxLoginLen
);
474 wxString
wxGetUserName()
476 static const int maxUserNameLen
= 1024; // FIXME arbitrary number
479 bool ok
= wxGetUserName(wxStringBuffer(buf
, maxUserNameLen
), maxUserNameLen
);
487 wxString
wxGetHostName()
489 static const size_t hostnameSize
= 257;
492 bool ok
= wxGetHostName(wxStringBuffer(buf
, hostnameSize
), hostnameSize
);
500 wxString
wxGetFullHostName()
502 static const size_t hostnameSize
= 257;
505 bool ok
= wxGetFullHostName(wxStringBuffer(buf
, hostnameSize
), hostnameSize
);
513 wxString
wxGetHomeDir()
523 wxString
wxGetCurrentDir()
530 ok
= getcwd(dir
.GetWriteBuf(len
+ 1), len
) != NULL
;
535 if ( errno
!= ERANGE
)
537 wxLogSysError(_T("Failed to get current directory"));
539 return wxEmptyString
;
543 // buffer was too small, retry with a larger one
555 // ----------------------------------------------------------------------------
557 // ----------------------------------------------------------------------------
559 // wxDoExecuteWithCapture() helper: reads an entire stream into one array
561 // returns true if ok, false if error
563 static bool ReadAll(wxInputStream
*is
, wxArrayString
& output
)
565 wxCHECK_MSG( is
, false, _T("NULL stream in wxExecute()?") );
567 // the stream could be already at EOF or in wxSTREAM_BROKEN_PIPE state
570 wxTextInputStream
tis(*is
);
574 wxString line
= tis
.ReadLine();
576 // check for EOF before other errors as it's not really an error
579 // add the last, possibly incomplete, line
585 // any other error is fatal
594 #endif // wxUSE_STREAMS
596 // this is a private function because it hasn't a clean interface: the first
597 // array is passed by reference, the second by pointer - instead we have 2
598 // public versions of wxExecute() below
599 static long wxDoExecuteWithCapture(const wxString
& command
,
600 wxArrayString
& output
,
601 wxArrayString
* error
,
604 // create a wxProcess which will capture the output
605 wxProcess
*process
= new wxProcess
;
608 long rc
= wxExecute(command
, wxEXEC_SYNC
| flags
, process
);
613 if ( !ReadAll(process
->GetInputStream(), output
) )
618 if ( !ReadAll(process
->GetErrorStream(), *error
) )
626 #endif // wxUSE_STREAMS/!wxUSE_STREAMS
633 long wxExecute(const wxString
& command
, wxArrayString
& output
, int flags
)
635 return wxDoExecuteWithCapture(command
, output
, NULL
, flags
);
638 long wxExecute(const wxString
& command
,
639 wxArrayString
& output
,
640 wxArrayString
& error
,
643 return wxDoExecuteWithCapture(command
, output
, &error
, flags
);
646 // ----------------------------------------------------------------------------
647 // wxApp::Yield() wrappers for backwards compatibility
648 // ----------------------------------------------------------------------------
652 return wxTheApp
&& wxTheApp
->Yield();
655 bool wxYieldIfNeeded()
657 return wxTheApp
&& wxTheApp
->Yield(true);
661 static long wxCurrentId
= 100;
665 // skip the part of IDs space that contains hard-coded values:
666 if (wxCurrentId
== wxID_LOWEST
)
667 wxCurrentId
= wxID_HIGHEST
+ 1;
669 return wxCurrentId
++;
673 wxGetCurrentId(void) { return wxCurrentId
; }
676 wxRegisterId (long id
)
678 if (id
>= wxCurrentId
)
679 wxCurrentId
= id
+ 1;
682 // ----------------------------------------------------------------------------
683 // wxQsort, adapted by RR to allow user_data
684 // ----------------------------------------------------------------------------
686 /* This file is part of the GNU C Library.
687 Written by Douglas C. Schmidt (schmidt@ics.uci.edu).
689 Douglas Schmidt kindly gave permission to relicence the
690 code under the wxWindows licence:
692 From: "Douglas C. Schmidt" <schmidt@dre.vanderbilt.edu>
693 To: Robert Roebling <robert.roebling@uni-ulm.de>
694 Subject: Re: qsort licence
695 Date: Mon, 23 Jul 2007 03:44:25 -0500
696 Sender: schmidt@dre.vanderbilt.edu
697 Message-Id: <20070723084426.64F511000A8@tango.dre.vanderbilt.edu>
701 > [...] I'm asking if you'd be willing to relicence your code
702 > under the wxWindows licence. [...]
704 That's fine with me [...]
711 /* Byte-wise swap two items of size SIZE. */
712 #define SWAP(a, b, size) \
715 register size_t __size = (size); \
716 register char *__a = (a), *__b = (b); \
722 } while (--__size > 0); \
725 /* Discontinue quicksort algorithm when partition gets below this size.
726 This particular magic number was chosen to work best on a Sun 4/260. */
729 /* Stack node declarations used to store unfulfilled partition obligations. */
736 /* The next 4 #defines implement a very fast in-line stack abstraction. */
737 #define STACK_SIZE (8 * sizeof(unsigned long int))
738 #define PUSH(low, high) ((void) ((top->lo = (low)), (top->hi = (high)), ++top))
739 #define POP(low, high) ((void) (--top, (low = top->lo), (high = top->hi)))
740 #define STACK_NOT_EMPTY (stack < top)
743 /* Order size using quicksort. This implementation incorporates
744 four optimizations discussed in Sedgewick:
746 1. Non-recursive, using an explicit stack of pointer that store the
747 next array partition to sort. To save time, this maximum amount
748 of space required to store an array of MAX_INT is allocated on the
749 stack. Assuming a 32-bit integer, this needs only 32 *
750 sizeof(stack_node) == 136 bits. Pretty cheap, actually.
752 2. Chose the pivot element using a median-of-three decision tree.
753 This reduces the probability of selecting a bad pivot value and
754 eliminates certain extraneous comparisons.
756 3. Only quicksorts TOTAL_ELEMS / MAX_THRESH partitions, leaving
757 insertion sort to order the MAX_THRESH items within each partition.
758 This is a big win, since insertion sort is faster for small, mostly
759 sorted array segments.
761 4. The larger of the two sub-partitions is always pushed onto the
762 stack first, with the algorithm then concentrating on the
763 smaller partition. This *guarantees* no more than log (n)
764 stack size is needed (actually O(1) in this case)! */
766 void wxQsort(void *const pbase
, size_t total_elems
,
767 size_t size
, CMPFUNCDATA cmp
, const void* user_data
)
769 register char *base_ptr
= (char *) pbase
;
770 const size_t max_thresh
= MAX_THRESH
* size
;
772 if (total_elems
== 0)
773 /* Avoid lossage with unsigned arithmetic below. */
776 if (total_elems
> MAX_THRESH
)
779 char *hi
= &lo
[size
* (total_elems
- 1)];
780 stack_node stack
[STACK_SIZE
];
781 stack_node
*top
= stack
;
785 while (STACK_NOT_EMPTY
)
790 /* Select median value from among LO, MID, and HI. Rearrange
791 LO and HI so the three values are sorted. This lowers the
792 probability of picking a pathological pivot value and
793 skips a comparison for both the LEFT_PTR and RIGHT_PTR. */
795 char *mid
= lo
+ size
* ((hi
- lo
) / size
>> 1);
797 if ((*cmp
) ((void *) mid
, (void *) lo
, user_data
) < 0)
798 SWAP (mid
, lo
, size
);
799 if ((*cmp
) ((void *) hi
, (void *) mid
, user_data
) < 0)
800 SWAP (mid
, hi
, size
);
803 if ((*cmp
) ((void *) mid
, (void *) lo
, user_data
) < 0)
804 SWAP (mid
, lo
, size
);
806 left_ptr
= lo
+ size
;
807 right_ptr
= hi
- size
;
809 /* Here's the famous ``collapse the walls'' section of quicksort.
810 Gotta like those tight inner loops! They are the main reason
811 that this algorithm runs much faster than others. */
814 while ((*cmp
) ((void *) left_ptr
, (void *) mid
, user_data
) < 0)
817 while ((*cmp
) ((void *) mid
, (void *) right_ptr
, user_data
) < 0)
820 if (left_ptr
< right_ptr
)
822 SWAP (left_ptr
, right_ptr
, size
);
825 else if (mid
== right_ptr
)
830 else if (left_ptr
== right_ptr
)
837 while (left_ptr
<= right_ptr
);
839 /* Set up pointers for next iteration. First determine whether
840 left and right partitions are below the threshold size. If so,
841 ignore one or both. Otherwise, push the larger partition's
842 bounds on the stack and continue sorting the smaller one. */
844 if ((size_t) (right_ptr
- lo
) <= max_thresh
)
846 if ((size_t) (hi
- left_ptr
) <= max_thresh
)
847 /* Ignore both small partitions. */
850 /* Ignore small left partition. */
853 else if ((size_t) (hi
- left_ptr
) <= max_thresh
)
854 /* Ignore small right partition. */
856 else if ((right_ptr
- lo
) > (hi
- left_ptr
))
858 /* Push larger left partition indices. */
859 PUSH (lo
, right_ptr
);
864 /* Push larger right partition indices. */
871 /* Once the BASE_PTR array is partially sorted by quicksort the rest
872 is completely sorted using insertion sort, since this is efficient
873 for partitions below MAX_THRESH size. BASE_PTR points to the beginning
874 of the array to sort, and END_PTR points at the very last element in
875 the array (*not* one beyond it!). */
878 char *const end_ptr
= &base_ptr
[size
* (total_elems
- 1)];
879 char *tmp_ptr
= base_ptr
;
880 char *thresh
= base_ptr
+ max_thresh
;
881 if ( thresh
> end_ptr
)
883 register char *run_ptr
;
885 /* Find smallest element in first threshold and place it at the
886 array's beginning. This is the smallest array element,
887 and the operation speeds up insertion sort's inner loop. */
889 for (run_ptr
= tmp_ptr
+ size
; run_ptr
<= thresh
; run_ptr
+= size
)
890 if ((*cmp
) ((void *) run_ptr
, (void *) tmp_ptr
, user_data
) < 0)
893 if (tmp_ptr
!= base_ptr
)
894 SWAP (tmp_ptr
, base_ptr
, size
);
896 /* Insertion sort, running from left-hand-side up to right-hand-side. */
898 run_ptr
= base_ptr
+ size
;
899 while ((run_ptr
+= size
) <= end_ptr
)
901 tmp_ptr
= run_ptr
- size
;
902 while ((*cmp
) ((void *) run_ptr
, (void *) tmp_ptr
, user_data
) < 0)
906 if (tmp_ptr
!= run_ptr
)
910 trav
= run_ptr
+ size
;
911 while (--trav
>= run_ptr
)
916 for (hi
= lo
= trav
; (lo
-= size
) >= tmp_ptr
; hi
= lo
)
929 // ============================================================================
930 // GUI-only functions from now on
931 // ============================================================================
935 // ----------------------------------------------------------------------------
936 // Launch document with default app
937 // ----------------------------------------------------------------------------
939 bool wxLaunchDefaultApplication(const wxString
& document
, int flags
)
944 static const char * const OPEN_CMD
= "/usr/bin/open";
945 if ( wxFileExists(OPEN_CMD
) &&
946 wxExecute(wxString(OPEN_CMD
) + " " + document
) )
948 #elif defined(__UNIX__)
949 // Our best best is to use xdg-open from freedesktop.org cross-desktop
950 // compatibility suite xdg-utils
951 // (see http://portland.freedesktop.org/wiki/) -- this is installed on
952 // most modern distributions and may be tweaked by them to handle
953 // distribution specifics.
954 wxString path
, xdg_open
;
955 if ( wxGetEnv("PATH", &path
) &&
956 wxFindFileInPath(&xdg_open
, path
, "xdg-open") )
958 if ( wxExecute(xdg_open
+ " " + document
) )
961 #elif defined(__WXMSW__)
962 WinStruct
<SHELLEXECUTEINFO
> sei
;
963 sei
.lpFile
= document
.wx_str();
964 sei
.lpVerb
= _T("open");
965 sei
.nShow
= SW_SHOWDEFAULT
;
967 // avoid Windows message box in case of error for consistency with
968 // wxLaunchDefaultBrowser() even if don't show the error ourselves in this
970 sei
.fMask
= SEE_MASK_FLAG_NO_UI
;
972 if ( ::ShellExecuteEx(&sei
) )
979 // ----------------------------------------------------------------------------
980 // Launch default browser
981 // ----------------------------------------------------------------------------
984 // Private method in Objective-C++ source file.
985 bool wxCocoaLaunchDefaultBrowser(const wxString
& url
, int flags
);
988 static bool DoLaunchDefaultBrowser(const wxString
& urlOrig
, int flags
)
992 // set the scheme of url to http if it does not have one
993 // RR: This doesn't work if the url is just a local path
994 wxString
url(urlOrig
);
996 if ( !uri
.HasScheme() )
998 if (wxFileExists(urlOrig
))
999 url
.Prepend( wxT("file://") );
1001 url
.Prepend(wxT("http://"));
1005 #if defined(__WXMSW__)
1008 if ( flags
& wxBROWSER_NEW_WINDOW
)
1010 // ShellExecuteEx() opens the URL in an existing window by default so
1011 // we can't use it if we need a new window
1012 wxRegKey
key(wxRegKey::HKCR
, uri
.GetScheme() + _T("\\shell\\open"));
1013 if ( !key
.Exists() )
1015 // try default browser, it must be registered at least for http URLs
1016 key
.SetName(wxRegKey::HKCR
, _T("http\\shell\\open"));
1021 wxRegKey
keyDDE(key
, wxT("DDEExec"));
1022 if ( keyDDE
.Exists() )
1024 // we only know the syntax of WWW_OpenURL DDE request for IE,
1025 // optimistically assume that all other browsers are compatible
1027 static const wxChar
*TOPIC_OPEN_URL
= wxT("WWW_OpenURL");
1029 wxRegKey
keyTopic(keyDDE
, wxT("topic"));
1030 bool ok
= keyTopic
.Exists() &&
1031 keyTopic
.QueryDefaultValue() == TOPIC_OPEN_URL
;
1034 ddeCmd
= keyDDE
.QueryDefaultValue();
1035 ok
= !ddeCmd
.empty();
1040 // for WWW_OpenURL, the index of the window to open the URL
1041 // in is -1 (meaning "current") by default, replace it with
1042 // 0 which means "new" (see KB article 160957)
1043 ok
= ddeCmd
.Replace(wxT("-1"), wxT("0"),
1044 false /* only first occurrence */) == 1;
1049 // and also replace the parameters: the topic should
1050 // contain a placeholder for the URL
1051 ok
= ddeCmd
.Replace(wxT("%1"), url
, false) == 1;
1056 // try to send it the DDE request now but ignore the errors
1059 const wxString ddeServer
= wxRegKey(keyDDE
, wxT("application"));
1060 if ( wxExecuteDDE(ddeServer
, TOPIC_OPEN_URL
, ddeCmd
) )
1063 // this is not necessarily an error: maybe browser is
1064 // simply not running, but no matter, in any case we're
1065 // going to launch it using ShellExecuteEx() below now and
1066 // we shouldn't try to open a new window if we open a new
1074 WinStruct
<SHELLEXECUTEINFO
> sei
;
1075 sei
.lpFile
= url
.c_str();
1076 sei
.lpVerb
= _T("open");
1077 sei
.nShow
= SW_SHOWNORMAL
;
1078 sei
.fMask
= SEE_MASK_FLAG_NO_UI
; // we give error message ourselves
1080 if ( ::ShellExecuteEx(&sei
) )
1082 #elif defined(__WXCOCOA__)
1083 // NOTE: We need to call the real implementation from src/cocoa/utils.mm
1084 // because the code must use Objective-C features.
1085 return wxCocoaLaunchDefaultBrowser(url
, flags
);
1086 #elif defined(__WXMAC__) && !defined(__WXOSX_IPHONE__)
1087 wxCFRef
< CFURLRef
> curl( CFURLCreateWithString( kCFAllocatorDefault
,
1088 wxCFStringRef( url
), NULL
) );
1089 OSStatus err
= LSOpenCFURLRef( curl
, NULL
);
1097 wxLogDebug(wxT("Browser Launch error %d"), (int) err
);
1101 // (non-Mac, non-MSW)
1105 // Our best best is to use xdg-open from freedesktop.org cross-desktop
1106 // compatibility suite xdg-utils
1107 // (see http://portland.freedesktop.org/wiki/) -- this is installed on
1108 // most modern distributions and may be tweaked by them to handle
1109 // distribution specifics. Only if that fails, try to find the right
1110 // browser ourselves.
1111 wxString path
, xdg_open
;
1112 if ( wxGetEnv("PATH", &path
) &&
1113 wxFindFileInPath(&xdg_open
, path
, "xdg-open") )
1115 if ( wxExecute(xdg_open
+ " " + url
) )
1119 wxString desktop
= wxTheApp
->GetTraits()->GetDesktopEnvironment();
1121 // GNOME and KDE desktops have some applications which should be always installed
1122 // together with their main parts, which give us the
1123 if (desktop
== wxT("GNOME"))
1125 wxArrayString errors
;
1126 wxArrayString output
;
1128 // gconf will tell us the path of the application to use as browser
1129 long res
= wxExecute( wxT("gconftool-2 --get /desktop/gnome/applications/browser/exec"),
1130 output
, errors
, wxEXEC_NODISABLE
);
1131 if (res
>= 0 && errors
.GetCount() == 0)
1133 wxString cmd
= output
[0];
1134 cmd
<< _T(' ') << url
;
1139 else if (desktop
== wxT("KDE"))
1141 // kfmclient directly opens the given URL
1142 if (wxExecute(wxT("kfmclient openURL ") + url
))
1151 wxFileType
*ft
= wxTheMimeTypesManager
->GetFileTypeFromExtension(_T("html"));
1155 ft
->GetMimeType(&mt
);
1157 ok
= ft
->GetOpenCommand(&cmd
, wxFileType::MessageParameters(url
));
1160 #endif // wxUSE_MIMETYPE
1162 if ( !ok
|| cmd
.empty() )
1164 // fallback to checking for the BROWSER environment variable
1165 cmd
= wxGetenv(wxT("BROWSER"));
1167 cmd
<< _T(' ') << url
;
1170 ok
= ( !cmd
.empty() && wxExecute(cmd
) );
1174 // no file type for HTML extension
1175 wxLogError(_("No default application configured for HTML files."));
1177 #endif // !wxUSE_MIMETYPE && !__WXMSW__
1179 wxLogSysError(_("Failed to open URL \"%s\" in default browser."),
1185 bool wxLaunchDefaultBrowser(const wxString
& url
, int flags
)
1187 if ( flags
& wxBROWSER_NOBUSYCURSOR
)
1188 return DoLaunchDefaultBrowser(url
, flags
);
1191 return DoLaunchDefaultBrowser(url
, flags
);
1194 // ----------------------------------------------------------------------------
1195 // Menu accelerators related functions
1196 // ----------------------------------------------------------------------------
1198 wxChar
*wxStripMenuCodes(const wxChar
*in
, wxChar
*out
)
1201 wxString s
= wxMenuItem::GetLabelText(in
);
1204 wxString s
= wxStripMenuCodes(str
);
1205 #endif // wxUSE_MENUS
1208 // go smash their buffer if it's not big enough - I love char * params
1209 memcpy(out
, s
.c_str(), s
.length() * sizeof(wxChar
));
1213 out
= new wxChar
[s
.length() + 1];
1214 wxStrcpy(out
, s
.c_str());
1220 wxString
wxStripMenuCodes(const wxString
& in
, int flags
)
1222 wxASSERT_MSG( flags
, _T("this is useless to call without any flags") );
1226 size_t len
= in
.length();
1229 for ( size_t n
= 0; n
< len
; n
++ )
1232 if ( (flags
& wxStrip_Mnemonics
) && ch
== _T('&') )
1234 // skip it, it is used to introduce the accel char (or to quote
1235 // itself in which case it should still be skipped): note that it
1236 // can't be the last character of the string
1239 wxLogDebug(_T("Invalid menu string '%s'"), in
.c_str());
1243 // use the next char instead
1247 else if ( (flags
& wxStrip_Accel
) && ch
== _T('\t') )
1249 // everything after TAB is accel string, exit the loop
1259 // ----------------------------------------------------------------------------
1260 // Window search functions
1261 // ----------------------------------------------------------------------------
1264 * If parent is non-NULL, look through children for a label or title
1265 * matching the specified string. If NULL, look through all top-level windows.
1270 wxFindWindowByLabel (const wxString
& title
, wxWindow
* parent
)
1272 return wxWindow::FindWindowByLabel( title
, parent
);
1277 * If parent is non-NULL, look through children for a name
1278 * matching the specified string. If NULL, look through all top-level windows.
1283 wxFindWindowByName (const wxString
& name
, wxWindow
* parent
)
1285 return wxWindow::FindWindowByName( name
, parent
);
1288 // Returns menu item id or wxNOT_FOUND if none.
1290 wxFindMenuItemId(wxFrame
*frame
,
1291 const wxString
& menuString
,
1292 const wxString
& itemString
)
1295 wxMenuBar
*menuBar
= frame
->GetMenuBar ();
1297 return menuBar
->FindMenuItem (menuString
, itemString
);
1298 #else // !wxUSE_MENUS
1300 wxUnusedVar(menuString
);
1301 wxUnusedVar(itemString
);
1302 #endif // wxUSE_MENUS/!wxUSE_MENUS
1307 // Try to find the deepest child that contains 'pt'.
1308 // We go backwards, to try to allow for controls that are spacially
1309 // within other controls, but are still siblings (e.g. buttons within
1310 // static boxes). Static boxes are likely to be created _before_ controls
1311 // that sit inside them.
1312 wxWindow
* wxFindWindowAtPoint(wxWindow
* win
, const wxPoint
& pt
)
1314 if (!win
->IsShown())
1317 // Hack for wxNotebook case: at least in wxGTK, all pages
1318 // claim to be shown, so we must only deal with the selected one.
1320 if (win
->IsKindOf(CLASSINFO(wxNotebook
)))
1322 wxNotebook
* nb
= (wxNotebook
*) win
;
1323 int sel
= nb
->GetSelection();
1326 wxWindow
* child
= nb
->GetPage(sel
);
1327 wxWindow
* foundWin
= wxFindWindowAtPoint(child
, pt
);
1334 wxWindowList::compatibility_iterator node
= win
->GetChildren().GetLast();
1337 wxWindow
* child
= node
->GetData();
1338 wxWindow
* foundWin
= wxFindWindowAtPoint(child
, pt
);
1341 node
= node
->GetPrevious();
1344 wxPoint pos
= win
->GetPosition();
1345 wxSize sz
= win
->GetSize();
1346 if ( !win
->IsTopLevel() && win
->GetParent() )
1348 pos
= win
->GetParent()->ClientToScreen(pos
);
1351 wxRect
rect(pos
, sz
);
1352 if (rect
.Contains(pt
))
1358 wxWindow
* wxGenericFindWindowAtPoint(const wxPoint
& pt
)
1360 // Go backwards through the list since windows
1361 // on top are likely to have been appended most
1363 wxWindowList::compatibility_iterator node
= wxTopLevelWindows
.GetLast();
1366 wxWindow
* win
= node
->GetData();
1367 wxWindow
* found
= wxFindWindowAtPoint(win
, pt
);
1370 node
= node
->GetPrevious();
1375 // ----------------------------------------------------------------------------
1377 // ----------------------------------------------------------------------------
1380 * N.B. these convenience functions must be separate from msgdlgg.cpp, textdlgg.cpp
1381 * since otherwise the generic code may be pulled in unnecessarily.
1386 int wxMessageBox(const wxString
& message
, const wxString
& caption
, long style
,
1387 wxWindow
*parent
, int WXUNUSED(x
), int WXUNUSED(y
) )
1389 long decorated_style
= style
;
1391 if ( ( style
& ( wxICON_EXCLAMATION
| wxICON_HAND
| wxICON_INFORMATION
| wxICON_QUESTION
) ) == 0 )
1393 decorated_style
|= ( style
& wxYES
) ? wxICON_QUESTION
: wxICON_INFORMATION
;
1396 wxMessageDialog
dialog(parent
, message
, caption
, decorated_style
);
1398 int ans
= dialog
.ShowModal();
1411 wxFAIL_MSG( _T("unexpected return code from wxMessageDialog") );
1416 void wxInfoMessageBox(wxWindow
* parent
)
1418 // don't translate these strings, they're for diagnostics purposes only
1420 msg
.Printf(_T("wxWidgets Library (%s port)\n")
1421 _T("Version %d.%d.%d%s%s, compiled at %s %s\n")
1422 _T("Runtime version of toolkit used is %d.%d.%s\n")
1423 _T("Copyright (c) 1995-2008 wxWidgets team"),
1424 wxPlatformInfo::Get().GetPortIdName().c_str(),
1440 wxPlatformInfo::Get().GetToolkitMajorVersion(),
1441 wxPlatformInfo::Get().GetToolkitMinorVersion(),
1443 wxString::Format("\nThe compile-time GTK+ version is %d.%d.%d.",
1446 GTK_MICRO_VERSION
).c_str()
1451 wxMessageBox(msg
, _T("wxWidgets information"),
1452 wxICON_INFORMATION
| wxOK
,
1456 #endif // wxUSE_MSGDLG
1460 wxString
wxGetTextFromUser(const wxString
& message
, const wxString
& caption
,
1461 const wxString
& defaultValue
, wxWindow
*parent
,
1462 wxCoord x
, wxCoord y
, bool centre
)
1465 long style
= wxTextEntryDialogStyle
;
1472 wxTextEntryDialog
dialog(parent
, message
, caption
, defaultValue
, style
, wxPoint(x
, y
));
1474 if (dialog
.ShowModal() == wxID_OK
)
1476 str
= dialog
.GetValue();
1482 wxString
wxGetPasswordFromUser(const wxString
& message
,
1483 const wxString
& caption
,
1484 const wxString
& defaultValue
,
1486 wxCoord x
, wxCoord y
, bool centre
)
1489 long style
= wxTextEntryDialogStyle
;
1496 wxPasswordEntryDialog
dialog(parent
, message
, caption
, defaultValue
,
1497 style
, wxPoint(x
, y
));
1498 if ( dialog
.ShowModal() == wxID_OK
)
1500 str
= dialog
.GetValue();
1506 #endif // wxUSE_TEXTDLG
1510 wxColour
wxGetColourFromUser(wxWindow
*parent
,
1511 const wxColour
& colInit
,
1512 const wxString
& caption
,
1513 wxColourData
*ptrData
)
1515 // contains serialized representation of wxColourData used the last time
1516 // the dialog was shown: we want to reuse it the next time in order to show
1517 // the same custom colours to the user (and we can't just have static
1518 // wxColourData itself because it's a GUI object and so should be destroyed
1519 // before GUI shutdown and doing it during static cleanup is too late)
1520 static wxString s_strColourData
;
1526 if ( !s_strColourData
.empty() )
1528 if ( !data
.FromString(s_strColourData
) )
1530 wxFAIL_MSG( "bug in wxColourData::FromString()?" );
1534 // we don't get back the "choose full" flag value from the native
1535 // dialog and so we can't preserve it between runs, so we decide to
1536 // always use it as it seems better than not using it (user can
1537 // just ignore the extra controls in the dialog but having to click
1538 // a button each time to show them would be very annoying
1539 data
.SetChooseFull(true);
1544 if ( colInit
.IsOk() )
1546 ptrData
->SetColour(colInit
);
1550 wxColourDialog
dialog(parent
, ptrData
);
1551 if (!caption
.empty())
1552 dialog
.SetTitle(caption
);
1553 if ( dialog
.ShowModal() == wxID_OK
)
1555 *ptrData
= dialog
.GetColourData();
1556 colRet
= ptrData
->GetColour();
1557 s_strColourData
= ptrData
->ToString();
1559 //else: leave colRet invalid
1564 #endif // wxUSE_COLOURDLG
1568 wxFont
wxGetFontFromUser(wxWindow
*parent
, const wxFont
& fontInit
, const wxString
& caption
)
1571 if ( fontInit
.Ok() )
1573 data
.SetInitialFont(fontInit
);
1577 wxFontDialog
dialog(parent
, data
);
1578 if (!caption
.empty())
1579 dialog
.SetTitle(caption
);
1580 if ( dialog
.ShowModal() == wxID_OK
)
1582 fontRet
= dialog
.GetFontData().GetChosenFont();
1584 //else: leave it invalid
1589 #endif // wxUSE_FONTDLG
1591 // ----------------------------------------------------------------------------
1592 // wxSafeYield and supporting functions
1593 // ----------------------------------------------------------------------------
1595 void wxEnableTopLevelWindows(bool enable
)
1597 wxWindowList::compatibility_iterator node
;
1598 for ( node
= wxTopLevelWindows
.GetFirst(); node
; node
= node
->GetNext() )
1599 node
->GetData()->Enable(enable
);
1602 wxWindowDisabler::wxWindowDisabler(bool disable
)
1604 m_disabled
= disable
;
1609 wxWindowDisabler::wxWindowDisabler(wxWindow
*winToSkip
)
1612 DoDisable(winToSkip
);
1615 void wxWindowDisabler::DoDisable(wxWindow
*winToSkip
)
1617 // remember the top level windows which were already disabled, so that we
1618 // don't reenable them later
1619 m_winDisabled
= NULL
;
1621 wxWindowList::compatibility_iterator node
;
1622 for ( node
= wxTopLevelWindows
.GetFirst(); node
; node
= node
->GetNext() )
1624 wxWindow
*winTop
= node
->GetData();
1625 if ( winTop
== winToSkip
)
1628 // we don't need to disable the hidden or already disabled windows
1629 if ( winTop
->IsEnabled() && winTop
->IsShown() )
1635 if ( !m_winDisabled
)
1637 m_winDisabled
= new wxWindowList
;
1640 m_winDisabled
->Append(winTop
);
1645 wxWindowDisabler::~wxWindowDisabler()
1650 wxWindowList::compatibility_iterator node
;
1651 for ( node
= wxTopLevelWindows
.GetFirst(); node
; node
= node
->GetNext() )
1653 wxWindow
*winTop
= node
->GetData();
1654 if ( !m_winDisabled
|| !m_winDisabled
->Find(winTop
) )
1658 //else: had been already disabled, don't reenable
1661 delete m_winDisabled
;
1664 // Yield to other apps/messages and disable user input to all windows except
1666 bool wxSafeYield(wxWindow
*win
, bool onlyIfNeeded
)
1668 wxWindowDisabler
wd(win
);
1672 rc
= wxYieldIfNeeded();
1679 // Don't synthesize KeyUp events holding down a key and producing KeyDown
1680 // events with autorepeat. On by default and always on in wxMSW. wxGTK version
1683 bool wxSetDetectableAutoRepeat( bool WXUNUSED(flag
) )
1685 return true; // detectable auto-repeat is the only mode MSW supports