]>
git.saurik.com Git - wxWidgets.git/blob - src/common/utilscmn.cpp
52b8878187060f46b40e249b37de2a950de351bb
1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: Miscellaneous utility functions and classes
4 // Author: Julian Smart
8 // Copyright: (c) 1998 Julian Smart
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "utils.h"
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
26 #include "wx/window.h"
29 #include "wx/msgdlg.h"
30 #include "wx/textdlg.h"
33 #include "wx/ioswrap.h"
45 #if !defined(__WATCOMC__)
46 #if !(defined(_MSC_VER) && (_MSC_VER > 800))
52 #include <sys/types.h>
60 // Pattern matching code.
61 // Yes, this path is deliberate (for Borland compilation)
62 #ifdef wx_mac /* MATTHEW: [5] Mac doesn't like paths with "/" */
65 #include "../common/glob.inc"
72 #define _MAXPATHLEN 500
74 extern char *wxBuffer
;
76 // ----------------------------------------------------------------------------
78 // ----------------------------------------------------------------------------
80 static wxWindow
*wxFindWindowByLabel1(const wxString
& title
, wxWindow
* parent
);
81 static wxWindow
*wxFindWindowByName1 (const wxString
& title
, wxWindow
* parent
);
84 int strcasecmp(const char *str_1
, const char *str_2
)
88 c1
= tolower(*str_1
++);
89 c2
= tolower(*str_2
++);
90 } while ( c1
&& (c1
== c2
) );
95 int strncasecmp(const char *str_1
, const char *str_2
, size_t maxchar
)
101 c1
= tolower(*str_1
++);
102 c2
= tolower(*str_2
++);
114 // we have no strI functions under VMS, therefore I have implemented
115 // an inefficient but portable version: convert copies of strings to lowercase
116 // and then use the normal comparison
117 static void myLowerString(char *s
)
120 if(isalpha(*s
)) *s
= (char)tolower(*s
);
125 int strcasecmp(const char *str_1
, const char *str_2
)
127 char *temp1
= new char[strlen(str_1
)+1];
128 char *temp2
= new char[strlen(str_2
)+1];
131 myLowerString(temp1
);
132 myLowerString(temp2
);
134 int result
= strcmp(temp1
,temp2
);
141 int strncasecmp(const char *str_1
, const char *str_2
, size_t maxchar
)
143 char *temp1
= new char[strlen(str_1
)+1];
144 char *temp2
= new char[strlen(str_2
)+1];
147 myLowerString(temp1
);
148 myLowerString(temp2
);
150 int result
= strncmp(temp1
,temp2
,maxchar
);
162 #define strcasecmp stricmp
163 #define strncasecmp strnicmp
165 #define strcasecmp _stricmp
166 #define strncasecmp _strnicmp
171 // This declaration is missing in SunOS!
172 // (Yes, I know it is NOT ANSI-C but its in BSD libc)
173 #if defined(__xlC) || defined(__AIX__) || defined(__GNUG__)
176 int strcasecmp (const char *, const char *);
177 int strncasecmp (const char *, const char *, size_t);
180 #endif /* __WXMSW__ */
184 copystring (const char *s
)
186 if (s
== NULL
) s
= "";
187 size_t len
= strlen (s
) + 1;
189 char *news
= new char[len
];
190 memcpy (news
, s
, len
); // Should be the fastest
196 static long wxCurrentId
= 100;
201 return wxCurrentId
++;
205 wxGetCurrentId(void) { return wxCurrentId
; }
208 wxRegisterId (long id
)
210 if (id
>= wxCurrentId
)
211 wxCurrentId
= id
+ 1;
215 StringToFloat (char *s
, float *number
)
217 if (s
&& *s
&& number
)
218 *number
= (float) strtod (s
, (char **) NULL
);
222 StringToDouble (char *s
, double *number
)
224 if (s
&& *s
&& number
)
225 *number
= strtod (s
, (char **) NULL
);
229 FloatToString (float number
, const char *fmt
)
231 static char buf
[256];
233 // sprintf (buf, "%.2f", number);
234 sprintf (buf
, fmt
, number
);
239 DoubleToString (double number
, const char *fmt
)
241 static char buf
[256];
243 sprintf (buf
, fmt
, number
);
248 StringToInt (char *s
, int *number
)
250 if (s
&& *s
&& number
)
251 *number
= (int) strtol (s
, (char **) NULL
, 10);
255 StringToLong (char *s
, long *number
)
257 if (s
&& *s
&& number
)
258 *number
= strtol (s
, (char **) NULL
, 10);
262 IntToString (int number
)
266 sprintf (buf
, "%d", number
);
271 LongToString (long number
)
275 sprintf (buf
, "%ld", number
);
279 // Array used in DecToHex conversion routine.
280 static char hexArray
[] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B',
281 'C', 'D', 'E', 'F' };
283 // Convert 2-digit hex number to decimal
284 int wxHexToDec(const wxString
& buf
)
286 int firstDigit
, secondDigit
;
288 if (buf
.GetChar(0) >= 'A')
289 firstDigit
= buf
.GetChar(0) - 'A' + 10;
291 firstDigit
= buf
.GetChar(0) - '0';
293 if (buf
.GetChar(1) >= 'A')
294 secondDigit
= buf
.GetChar(1) - 'A' + 10;
296 secondDigit
= buf
.GetChar(1) - '0';
298 return firstDigit
* 16 + secondDigit
;
301 // Convert decimal integer to 2-character hex string
302 void wxDecToHex(int dec
, char *buf
)
304 int firstDigit
= (int)(dec
/16.0);
305 int secondDigit
= (int)(dec
- (firstDigit
*16.0));
306 buf
[0] = hexArray
[firstDigit
];
307 buf
[1] = hexArray
[secondDigit
];
311 // Convert decimal integer to 2-character hex string
312 wxString
wxDecToHex(int dec
)
315 wxDecToHex(dec
, buf
);
316 return wxString(buf
);
319 // Match a string INDEPENDENT OF CASE
321 StringMatch (char *str1
, char *str2
, bool subString
, bool exact
)
323 if (str1
== NULL
|| str2
== NULL
)
330 int len1
= strlen (str1
);
331 int len2
= strlen (str2
);
334 // Search for str1 in str2
335 // Slow .... but acceptable for short strings
336 for (i
= 0; i
<= len2
- len1
; i
++)
338 if (strncasecmp (str1
, str2
+ i
, len1
) == 0)
344 if (strcasecmp (str1
, str2
) == 0)
349 int len1
= strlen (str1
);
350 int len2
= strlen (str2
);
352 if (strncasecmp (str1
, str2
, wxMin (len1
, len2
)) == 0)
359 // Return the current date/time
361 wxString
wxNow( void )
363 time_t now
= time((time_t *) NULL
);
364 char *date
= ctime(&now
);
366 return wxString(date
);
370 * Strip out any menu codes
373 char *wxStripMenuCodes (char *in
, char *out
)
376 return (char *) NULL
;
379 out
= copystring(in
);
387 // Check && -> &, &x -> x
391 else if (*in
== '\t')
393 // Remove all stuff after \t in X mode, and let the stuff as is
395 // Accelerators are handled in wx_item.cc for Motif, and are not
396 // YET supported in XView
408 wxString
wxStripMenuCodes(const wxString
& str
)
410 char *buf
= new char[str
.Length() + 1];
411 wxStripMenuCodes((char*) (const char*) str
, buf
);
418 * Window search functions
423 * If parent is non-NULL, look through children for a label or title
424 * matching the specified string. If NULL, look through all top-level windows.
429 wxFindWindowByLabel (const wxString
& title
, wxWindow
* parent
)
433 return wxFindWindowByLabel1(title
, parent
);
437 for ( wxWindowList::Node
* node
= wxTopLevelWindows
.GetFirst();
439 node
= node
->GetNext() )
441 wxWindow
*win
= node
->GetData();
442 wxWindow
*retwin
= wxFindWindowByLabel1 (title
, win
);
448 return (wxWindow
*) NULL
;
453 wxFindWindowByLabel1 (const wxString
& title
, wxWindow
* parent
)
457 if (parent
->GetLabel() == title
)
463 for ( wxNode
* node
= parent
->GetChildren().GetFirst();
465 node
= node
->GetNext() )
467 wxWindow
*win
= (wxWindow
*)node
->GetData();
468 wxWindow
*retwin
= wxFindWindowByLabel1 (title
, win
);
475 return (wxWindow
*) NULL
; // Not found
479 * If parent is non-NULL, look through children for a name
480 * matching the specified string. If NULL, look through all top-level windows.
485 wxFindWindowByName (const wxString
& title
, wxWindow
* parent
)
489 return wxFindWindowByName1 (title
, parent
);
493 for ( wxWindowList::Node
* node
= wxTopLevelWindows
.GetFirst();
495 node
= node
->GetNext() )
497 wxWindow
*win
= node
->GetData();
498 wxWindow
*retwin
= wxFindWindowByName1 (title
, win
);
505 // Failed? Try by label instead.
506 return wxFindWindowByLabel(title
, parent
);
511 wxFindWindowByName1 (const wxString
& title
, wxWindow
* parent
)
515 if ( parent
->GetName() == title
)
521 for (wxNode
* node
= parent
->GetChildren().First (); node
; node
= node
->Next ())
523 wxWindow
*win
= (wxWindow
*) node
->Data ();
524 wxWindow
*retwin
= wxFindWindowByName1 (title
, win
);
531 return (wxWindow
*) NULL
; // Not found
535 // Returns menu item id or -1 if none.
537 wxFindMenuItemId (wxFrame
* frame
, const wxString
& menuString
, const wxString
& itemString
)
539 wxMenuBar
*menuBar
= frame
->GetMenuBar ();
542 return menuBar
->FindMenuItem (menuString
, itemString
);
546 On Fri, 21 Jul 1995, Paul Craven wrote:
548 > Is there a way to find the path of running program's executable? I can get
549 > my home directory, and the current directory, but I don't know how to get the
550 > executable directory.
553 The code below (warty as it is), does what you want on most Unix,
554 DOS, and Mac platforms (it's from the ALS Prolog main).
556 || Ken Bowen Applied Logic Systems, Inc. PO Box 180,
557 ||==== Voice: +1 (617)965-9191 Newton Centre,
558 || FAX: +1 (617)965-1636 MA 02159 USA
559 Email: ken@als.com WWW: http://www.als.com
560 ------------------------------------------------------------------------
563 // This code is commented out but it may be integrated with wxWin at
564 // a later date, after testing. Thanks Ken!
567 /*--------------------------------------------------------------------*
568 | whereami is given a filename f in the form: whereami(argv[0])
569 | It returns the directory in which the executable file (containing
570 | this code [main.c] ) may be found. A dot will be returned to indicate
571 | the current directory.
572 *--------------------------------------------------------------------*/
578 register char *cutoff
= NULL
; /* stifle -Wall */
585 * See if the file is accessible either through the current directory
586 * or through an absolute path.
589 if (access(name
, R_OK
) == 0) {
591 /*-------------------------------------------------------------*
592 * The file was accessible without any other work. But the current
593 * working directory might change on us, so if it was accessible
594 * through the cwd, then we should get it for later accesses.
595 *-------------------------------------------------------------*/
598 if (!absolute_pathname(name
)) {
599 #if defined(DOS) || defined(__WIN32__)
605 if (*(name
+ 1) == ':') {
606 if (*name
>= 'a' && *name
<= 'z')
607 drive
= (int) (*name
- 'a' + 1);
609 drive
= (int) (*name
- 'A' + 1);
611 *newrbuf
++ = *(name
+ 1);
612 *newrbuf
++ = DIR_SEPARATOR
;
616 *newrbuf
++ = DIR_SEPARATOR
;
618 if (getcwd(newrbuf
, drive
) == 0) { /* } */
620 if (getcwd(newrbuf
, 1024) == 0) { /* } */
624 if (getwd(imagedir
) == 0) { /* } */
625 #else /* !HAVE_GETWD */
626 if (getcwd(imagedir
, 1024) == 0) {
627 #endif /* !HAVE_GETWD */
629 fatal_error(FE_GETCWD
, 0);
631 for (; *t
; t
++) /* Set t to end of buffer */
633 if (*(t
- 1) == DIR_SEPARATOR
) /* leave slash if already
638 cutoff
= t
; /* otherwise put one in */
639 *t
++ = DIR_SEPARATOR
;
642 #if (!defined(__MAC__) && !defined(__DJGPP__) && !defined(__GO32__) && !defined(__WIN32__))
644 (*t
++ = DIR_SEPARATOR
);
647 /*-------------------------------------------------------------*
648 * Copy the rest of the string and set the cutoff if it was not
649 * already set. If the first character of name is a slash, cutoff
650 * is not presently set but will be on the first iteration of the
652 *-------------------------------------------------------------*/
654 for ((*name
== DIR_SEPARATOR
? (s
= name
+1) : (s
= name
));;) {
655 if (*s
== DIR_SEPARATOR
)
664 /*-------------------------------------------------------------*
665 * Get the path list from the environment. If the path list is
666 * inaccessible for any reason, leave with fatal error.
667 *-------------------------------------------------------------*/
670 if ((s
= getenv("Commands")) == (char *) 0)
672 if ((s
= getenv("PATH")) == (char *) 0)
674 fatal_error(FE_PATH
, 0);
677 * Copy path list into ebuf and set the source pointer to the
678 * beginning of this buffer.
686 while (*s
&& *s
!= PATH_SEPARATOR
)
688 if (t
> imagedir
&& *(t
- 1) == DIR_SEPARATOR
)
689 ; /* do nothing -- slash already is in place */
691 *t
++ = DIR_SEPARATOR
; /* put in the slash */
692 cutoff
= t
- 1; /* set cutoff */
694 if (access(imagedir
, R_OK
) == 0)
698 s
++; /* advance source pointer */
700 fatal_error(FE_INFND
, 0);
705 /*-------------------------------------------------------------*
706 | At this point the full pathname should exist in imagedir and
707 | cutoff should be set to the final slash. We must now determine
708 | whether the file name is a symbolic link or not and chase it down
709 | if it is. Note that we reuse ebuf for getting the link.
710 *-------------------------------------------------------------*/
713 while ((cc
= readlink(imagedir
, ebuf
, 512)) != -1) {
716 if (*s
== DIR_SEPARATOR
) {
723 if (*s
== DIR_SEPARATOR
)
724 cutoff
= t
; /* mark the last slash seen */
725 if (!(*t
++ = *s
++)) /* copy the character */
730 #endif /* HAVE_SYMLINK */
732 strcpy(imagename
, cutoff
+ 1); /* keep the image name */
733 *(cutoff
+ 1) = 0; /* chop off the filename part */
739 // Yield to other apps/messages and disable user input
740 bool wxSafeYield(wxWindow
*win
)
742 wxWindowList::Node
*node
;
743 for ( node
= wxTopLevelWindows
.GetFirst(); node
; node
= node
->GetNext() )
745 node
->GetData()->Enable(FALSE
);
748 // always enable ourselves
753 for ( node
= wxTopLevelWindows
.GetFirst(); node
; node
= node
->GetNext() )
755 node
->GetData()->Enable(TRUE
);
762 * N.B. these convenience functions must be separate from msgdlgg.cpp, textdlgg.cpp
763 * since otherwise the generic code may be pulled in unnecessarily.
766 int wxMessageBox(const wxString
& message
, const wxString
& caption
, long style
,
767 wxWindow
*parent
, int WXUNUSED(x
), int WXUNUSED(y
) )
769 wxMessageDialog
dialog(parent
, message
, caption
, style
);
771 int ans
= dialog
.ShowModal();
791 wxString
wxGetTextFromUser(const wxString
& message
, const wxString
& caption
,
792 const wxString
& defaultValue
, wxWindow
*parent
,
793 int x
, int y
, bool WXUNUSED(centre
) )
795 wxTextEntryDialog
dialog(parent
, message
, caption
, defaultValue
, wxOK
|wxCANCEL
, wxPoint(x
, y
));
796 if (dialog
.ShowModal() == wxID_OK
)
797 return dialog
.GetValue();
803 char *strdup(const char *s
)
805 return strcpy( (char*) malloc( strlen( s
) + 1 ) , s
) ;
810 return ( c
>= 0 && c
< 128 ) ;
814 // ----------------------------------------------------------------------------
815 // network and user id functions
816 // ----------------------------------------------------------------------------
818 // Get Full RFC822 style email address
819 bool wxGetEmailAddress(char *address
, int maxSize
)
821 wxString email
= wxGetEmailAddress();
825 strncpy(address
, email
, maxSize
- 1);
826 address
[maxSize
- 1] = '\0';
831 wxString
wxGetEmailAddress()
835 wxString host
= wxGetHostName();
838 wxString user
= wxGetUserId();
841 wxString
email(user
);
842 email
<< '@' << host
;
849 wxString
wxGetUserId()
851 static const int maxLoginLen
= 256; // FIXME arbitrary number
854 bool ok
= wxGetUserId(buf
.GetWriteBuf(maxLoginLen
), maxLoginLen
);
863 wxString
wxGetUserName()
865 static const int maxUserNameLen
= 1024; // FIXME arbitrary number
868 bool ok
= wxGetUserName(buf
.GetWriteBuf(maxUserNameLen
), maxUserNameLen
);
877 wxString
wxGetHostName()
879 static const size_t hostnameSize
= 257;
882 bool ok
= wxGetHostName(buf
.GetWriteBuf(hostnameSize
), hostnameSize
);