git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@22561 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
1024 lines
25 KiB
C++
1024 lines
25 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: utils.cpp
|
|
// Purpose: Various utilities
|
|
// Author: Stefan Csomor
|
|
// Modified by:
|
|
// Created: 1998-01-01
|
|
// RCS-ID: $Id$
|
|
// Copyright: (c) Stefan Csomor
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
#ifdef __GNUG__
|
|
// Note: this is done in utilscmn.cpp now.
|
|
// #pragma implementation "utils.h"
|
|
#endif
|
|
|
|
#include "wx/setup.h"
|
|
#include "wx/utils.h"
|
|
#include "wx/app.h"
|
|
#include "wx/apptrait.h"
|
|
|
|
#if wxUSE_GUI
|
|
#include "wx/mac/uma.h"
|
|
#endif
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdarg.h>
|
|
|
|
#ifdef __DARWIN__
|
|
# include "MoreFilesX.h"
|
|
#else
|
|
# include "MoreFiles.h"
|
|
# include "MoreFilesExtras.h"
|
|
#endif
|
|
|
|
#ifndef __DARWIN__
|
|
#include <Threads.h>
|
|
#include <Sound.h>
|
|
#endif
|
|
|
|
#include "ATSUnicode.h"
|
|
#include "TextCommon.h"
|
|
#include "TextEncodingConverter.h"
|
|
|
|
#if defined(__WXMAC__)
|
|
#include "wx/mac/private.h" // includes mac headers
|
|
#endif
|
|
|
|
#if defined(__MWERKS__) && wxUSE_UNICODE
|
|
#include <wtime.h>
|
|
#endif
|
|
|
|
#if wxUSE_BASE
|
|
|
|
#ifndef __DARWIN__
|
|
// defined in unix/utilsunx.cpp for Mac OS X
|
|
|
|
// get full hostname (with domain name if possible)
|
|
bool wxGetFullHostName(wxChar *buf, int maxSize)
|
|
{
|
|
return wxGetHostName(buf, maxSize);
|
|
}
|
|
|
|
// Get hostname only (without domain name)
|
|
bool wxGetHostName(wxChar *buf, int maxSize)
|
|
{
|
|
// Gets Chooser name of user by examining a System resource.
|
|
|
|
const short kComputerNameID = -16413;
|
|
|
|
short oldResFile = CurResFile() ;
|
|
UseResFile(0);
|
|
StringHandle chooserName = (StringHandle)::GetString(kComputerNameID);
|
|
UseResFile(oldResFile);
|
|
|
|
if (chooserName && *chooserName)
|
|
{
|
|
HLock( (Handle) chooserName ) ;
|
|
wxString name = wxMacMakeStringFromPascal( *chooserName ) ;
|
|
HUnlock( (Handle) chooserName ) ;
|
|
ReleaseResource( (Handle) chooserName ) ;
|
|
wxStrncpy( buf , name , maxSize - 1 ) ;
|
|
}
|
|
else
|
|
buf[0] = 0 ;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Get user ID e.g. jacs
|
|
bool wxGetUserId(wxChar *buf, int maxSize)
|
|
{
|
|
return wxGetUserName( buf , maxSize ) ;
|
|
}
|
|
|
|
const wxChar* wxGetHomeDir(wxString *pstr)
|
|
{
|
|
*pstr = wxMacFindFolder( (short) kOnSystemDisk, kPreferencesFolderType, kDontCreateFolder ) ;
|
|
return pstr->c_str() ;
|
|
}
|
|
|
|
// Get user name e.g. Stefan Csomor
|
|
bool wxGetUserName(wxChar *buf, int maxSize)
|
|
{
|
|
// Gets Chooser name of user by examining a System resource.
|
|
|
|
const short kChooserNameID = -16096;
|
|
|
|
short oldResFile = CurResFile() ;
|
|
UseResFile(0);
|
|
StringHandle chooserName = (StringHandle)::GetString(kChooserNameID);
|
|
UseResFile(oldResFile);
|
|
|
|
if (chooserName && *chooserName)
|
|
{
|
|
HLock( (Handle) chooserName ) ;
|
|
wxString name = wxMacMakeStringFromPascal( *chooserName ) ;
|
|
HUnlock( (Handle) chooserName ) ;
|
|
ReleaseResource( (Handle) chooserName ) ;
|
|
wxStrncpy( buf , name , maxSize - 1 ) ;
|
|
}
|
|
else
|
|
buf[0] = 0 ;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
int wxKill(long pid, wxSignal sig , wxKillError *rc )
|
|
{
|
|
// TODO
|
|
return 0;
|
|
}
|
|
|
|
WXDLLEXPORT bool wxGetEnv(const wxString& var, wxString *value)
|
|
{
|
|
// TODO : under classic there is no environement support, under X yes
|
|
return false ;
|
|
}
|
|
|
|
// set the env var name to the given value, return TRUE on success
|
|
WXDLLEXPORT bool wxSetEnv(const wxString& var, const wxChar *value)
|
|
{
|
|
// TODO : under classic there is no environement support, under X yes
|
|
return false ;
|
|
}
|
|
|
|
//
|
|
// Execute a program in an Interactive Shell
|
|
//
|
|
bool wxShell(const wxString& command)
|
|
{
|
|
// TODO
|
|
return FALSE;
|
|
}
|
|
|
|
// Shutdown or reboot the PC
|
|
bool wxShutdown(wxShutdownFlags wFlags)
|
|
{
|
|
// TODO
|
|
return FALSE;
|
|
}
|
|
|
|
// Get free memory in bytes, or -1 if cannot determine amount (e.g. on UNIX)
|
|
long wxGetFreeMemory()
|
|
{
|
|
return FreeMem() ;
|
|
}
|
|
|
|
void wxUsleep(unsigned long milliseconds)
|
|
{
|
|
clock_t start = clock() ;
|
|
do
|
|
{
|
|
YieldToAnyThread() ;
|
|
} while( clock() - start < milliseconds / 1000.0 * CLOCKS_PER_SEC ) ;
|
|
}
|
|
|
|
void wxSleep(int nSecs)
|
|
{
|
|
wxUsleep(1000*nSecs);
|
|
}
|
|
|
|
// Consume all events until no more left
|
|
void wxFlushEvents()
|
|
{
|
|
}
|
|
|
|
#endif // !__DARWIN__
|
|
|
|
// Emit a beeeeeep
|
|
void wxBell()
|
|
{
|
|
SysBeep(30);
|
|
}
|
|
|
|
// our OS version is the same in non GUI and GUI cases
|
|
static int DoGetOSVersion(int *majorVsn, int *minorVsn)
|
|
{
|
|
long theSystem ;
|
|
|
|
// are there x-platform conventions ?
|
|
|
|
Gestalt(gestaltSystemVersion, &theSystem) ;
|
|
if (minorVsn != NULL) {
|
|
*minorVsn = (theSystem & 0xFF ) ;
|
|
}
|
|
if (majorVsn != NULL) {
|
|
*majorVsn = (theSystem >> 8 ) ;
|
|
}
|
|
#ifdef __DARWIN__
|
|
return wxMAC_DARWIN;
|
|
#else
|
|
return wxMAC;
|
|
#endif
|
|
}
|
|
|
|
wxToolkitInfo& wxConsoleAppTraits::GetToolkitInfo()
|
|
{
|
|
static wxToolkitInfo info;
|
|
info.os = DoGetOSVersion(&info.versionMajor, &info.versionMinor);
|
|
info.name = _T("wxBase");
|
|
return info;
|
|
}
|
|
|
|
#endif // wxUSE_BASE
|
|
|
|
#if wxUSE_GUI
|
|
|
|
wxToolkitInfo& wxGUIAppTraits::GetToolkitInfo()
|
|
{
|
|
static wxToolkitInfo info;
|
|
info.os = DoGetOSVersion(&info.versionMajor, &info.versionMinor);
|
|
info.shortName = _T("mac");
|
|
info.name = _T("wxMac");
|
|
#ifdef __WXUNIVERSAL__
|
|
info.shortName << _T("univ");
|
|
info.name << _T("/wxUniversal");
|
|
#endif
|
|
return info;
|
|
}
|
|
|
|
// Reading and writing resources (eg WIN.INI, .Xdefaults)
|
|
#if wxUSE_RESOURCES
|
|
bool wxWriteResource(const wxString& section, const wxString& entry, const wxString& value, const wxString& file)
|
|
{
|
|
// TODO
|
|
return FALSE;
|
|
}
|
|
|
|
bool wxWriteResource(const wxString& section, const wxString& entry, float value, const wxString& file)
|
|
{
|
|
wxString buf;
|
|
buf.Printf(wxT("%.4f"), value);
|
|
|
|
return wxWriteResource(section, entry, buf, file);
|
|
}
|
|
|
|
bool wxWriteResource(const wxString& section, const wxString& entry, long value, const wxString& file)
|
|
{
|
|
wxString buf;
|
|
buf.Printf(wxT("%ld"), value);
|
|
|
|
return wxWriteResource(section, entry, buf, file);
|
|
}
|
|
|
|
bool wxWriteResource(const wxString& section, const wxString& entry, int value, const wxString& file)
|
|
{
|
|
wxString buf;
|
|
buf.Printf(wxT("%d"), value);
|
|
|
|
return wxWriteResource(section, entry, buf, file);
|
|
}
|
|
|
|
bool wxGetResource(const wxString& section, const wxString& entry, char **value, const wxString& file)
|
|
{
|
|
// TODO
|
|
return FALSE;
|
|
}
|
|
|
|
bool wxGetResource(const wxString& section, const wxString& entry, float *value, const wxString& file)
|
|
{
|
|
char *s = NULL;
|
|
bool succ = wxGetResource(section, entry, (char **)&s, file);
|
|
if (succ)
|
|
{
|
|
*value = (float)strtod(s, NULL);
|
|
delete[] s;
|
|
return TRUE;
|
|
}
|
|
else return FALSE;
|
|
}
|
|
|
|
bool wxGetResource(const wxString& section, const wxString& entry, long *value, const wxString& file)
|
|
{
|
|
char *s = NULL;
|
|
bool succ = wxGetResource(section, entry, (char **)&s, file);
|
|
if (succ)
|
|
{
|
|
*value = strtol(s, NULL, 10);
|
|
delete[] s;
|
|
return TRUE;
|
|
}
|
|
else return FALSE;
|
|
}
|
|
|
|
bool wxGetResource(const wxString& section, const wxString& entry, int *value, const wxString& file)
|
|
{
|
|
char *s = NULL;
|
|
bool succ = wxGetResource(section, entry, (char **)&s, file);
|
|
if (succ)
|
|
{
|
|
*value = (int)strtol(s, NULL, 10);
|
|
delete[] s;
|
|
return TRUE;
|
|
}
|
|
else return FALSE;
|
|
}
|
|
#endif // wxUSE_RESOURCES
|
|
|
|
int gs_wxBusyCursorCount = 0;
|
|
extern wxCursor gMacCurrentCursor ;
|
|
wxCursor gMacStoredActiveCursor ;
|
|
|
|
// Set the cursor to the busy cursor for all windows
|
|
void wxBeginBusyCursor(wxCursor *cursor)
|
|
{
|
|
if (gs_wxBusyCursorCount++ == 0)
|
|
{
|
|
gMacStoredActiveCursor = gMacCurrentCursor ;
|
|
cursor->MacInstall() ;
|
|
}
|
|
//else: nothing to do, already set
|
|
}
|
|
|
|
// Restore cursor to normal
|
|
void wxEndBusyCursor()
|
|
{
|
|
wxCHECK_RET( gs_wxBusyCursorCount > 0,
|
|
wxT("no matching wxBeginBusyCursor() for wxEndBusyCursor()") );
|
|
|
|
if (--gs_wxBusyCursorCount == 0)
|
|
{
|
|
gMacStoredActiveCursor.MacInstall() ;
|
|
gMacStoredActiveCursor = wxNullCursor ;
|
|
}
|
|
}
|
|
|
|
// TRUE if we're between the above two calls
|
|
bool wxIsBusy()
|
|
{
|
|
return (gs_wxBusyCursorCount > 0);
|
|
}
|
|
|
|
wxString wxMacFindFolder( short vol,
|
|
OSType folderType,
|
|
Boolean createFolder)
|
|
{
|
|
short vRefNum ;
|
|
long dirID ;
|
|
wxString strDir ;
|
|
|
|
if ( FindFolder( vol, folderType, createFolder, &vRefNum, &dirID) == noErr)
|
|
{
|
|
FSSpec file ;
|
|
if ( FSMakeFSSpec( vRefNum , dirID , "\p" , &file ) == noErr )
|
|
{
|
|
strDir = wxMacFSSpec2MacFilename( &file ) + wxFILE_SEP_PATH ;
|
|
}
|
|
}
|
|
return strDir ;
|
|
}
|
|
|
|
// Check whether this window wants to process messages, e.g. Stop button
|
|
// in long calculations.
|
|
bool wxCheckForInterrupt(wxWindow *wnd)
|
|
{
|
|
// TODO
|
|
return FALSE;
|
|
}
|
|
|
|
void wxGetMousePosition( int* x, int* y )
|
|
{
|
|
Point pt ;
|
|
|
|
GetMouse( &pt ) ;
|
|
LocalToGlobal( &pt ) ;
|
|
*x = pt.h ;
|
|
*y = pt.v ;
|
|
};
|
|
|
|
// Return TRUE if we have a colour display
|
|
bool wxColourDisplay()
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
// Returns depth of screen
|
|
int wxDisplayDepth()
|
|
{
|
|
Rect globRect ;
|
|
SetRect(&globRect, -32760, -32760, 32760, 32760);
|
|
GDHandle theMaxDevice;
|
|
|
|
int theDepth = 8;
|
|
theMaxDevice = GetMaxDevice(&globRect);
|
|
if (theMaxDevice != nil)
|
|
theDepth = (**(**theMaxDevice).gdPMap).pixelSize;
|
|
|
|
return theDepth ;
|
|
}
|
|
|
|
// Get size of display
|
|
void wxDisplaySize(int *width, int *height)
|
|
{
|
|
BitMap screenBits;
|
|
GetQDGlobalsScreenBits( &screenBits );
|
|
|
|
if (width != NULL) {
|
|
*width = screenBits.bounds.right - screenBits.bounds.left ;
|
|
}
|
|
if (height != NULL) {
|
|
*height = screenBits.bounds.bottom - screenBits.bounds.top ;
|
|
}
|
|
}
|
|
|
|
void wxDisplaySizeMM(int *width, int *height)
|
|
{
|
|
wxDisplaySize(width, height);
|
|
// on mac 72 is fixed (at least now ;-)
|
|
float cvPt2Mm = 25.4 / 72;
|
|
|
|
if (width != NULL) {
|
|
*width = int( *width * cvPt2Mm );
|
|
}
|
|
if (height != NULL) {
|
|
*height = int( *height * cvPt2Mm );
|
|
}
|
|
}
|
|
|
|
void wxClientDisplayRect(int *x, int *y, int *width, int *height)
|
|
{
|
|
BitMap screenBits;
|
|
GetQDGlobalsScreenBits( &screenBits );
|
|
|
|
if (x) *x = 0;
|
|
if (y) *y = 0;
|
|
|
|
if (width != NULL) {
|
|
*width = screenBits.bounds.right - screenBits.bounds.left ;
|
|
}
|
|
if (height != NULL) {
|
|
*height = screenBits.bounds.bottom - screenBits.bounds.top ;
|
|
}
|
|
|
|
SInt16 mheight ;
|
|
#if TARGET_CARBON
|
|
GetThemeMenuBarHeight( &mheight ) ;
|
|
#else
|
|
mheight = LMGetMBarHeight() ;
|
|
#endif
|
|
if (height != NULL) {
|
|
*height -= mheight ;
|
|
}
|
|
if (y)
|
|
*y = mheight ;
|
|
}
|
|
|
|
wxWindow* wxFindWindowAtPoint(const wxPoint& pt)
|
|
{
|
|
return wxGenericFindWindowAtPoint(pt);
|
|
}
|
|
|
|
#endif // wxUSE_GUI
|
|
|
|
#if wxUSE_BASE
|
|
|
|
wxString wxGetOsDescription()
|
|
{
|
|
#ifdef WXWIN_OS_DESCRIPTION
|
|
// use configure generated description if available
|
|
return wxString("MacOS (") + WXWIN_OS_DESCRIPTION + wxString(")");
|
|
#else
|
|
return wxT("MacOS") ; //TODO:define further
|
|
#endif
|
|
}
|
|
|
|
#ifndef __DARWIN__
|
|
wxChar *wxGetUserHome (const wxString& user)
|
|
{
|
|
// TODO
|
|
return NULL;
|
|
}
|
|
|
|
bool wxGetDiskSpace(const wxString& path, wxLongLong *pTotal, wxLongLong *pFree)
|
|
{
|
|
if ( path.empty() )
|
|
return FALSE;
|
|
|
|
wxString p = path ;
|
|
if (p[0] == ':' ) {
|
|
p = wxGetCwd() + p ;
|
|
}
|
|
|
|
int pos = p.Find(':') ;
|
|
if ( pos != wxNOT_FOUND ) {
|
|
p = p.Mid(1,pos) ;
|
|
}
|
|
|
|
p = p + wxT(":") ;
|
|
|
|
Str255 volumeName ;
|
|
XVolumeParam pb ;
|
|
|
|
wxMacStringToPascal( p , volumeName ) ;
|
|
OSErr err = XGetVolumeInfoNoName( volumeName , 0 , &pb ) ;
|
|
if ( err == noErr ) {
|
|
if ( pTotal ) {
|
|
(*pTotal) = wxLongLong( pb.ioVTotalBytes ) ;
|
|
}
|
|
if ( pFree ) {
|
|
(*pFree) = wxLongLong( pb.ioVFreeBytes ) ;
|
|
}
|
|
}
|
|
|
|
return err == noErr ;
|
|
}
|
|
#endif // !__DARWIN__
|
|
|
|
#endif // wxUSE_BASE
|
|
|
|
#if wxUSE_GUI
|
|
|
|
//---------------------------------------------------------------------------
|
|
// wxMac Specific utility functions
|
|
//---------------------------------------------------------------------------
|
|
|
|
char StringMac[] = "\x0d\x80\x81\x82\x83\x84\x85\x86\x87\x88\x89\x8a\x8b\x8c\x8d\x8e\x8f"
|
|
"\x90\x91\x92\x93\x94\x95\x96\x97\x98\x99\x9a\x9b\x9c\x9d\x9e\x9f"
|
|
"\xa0\xa1\xa2\xa3\xa4\xa5\xa6\xa7\xa8\xa9\xaa\xab\xac\xae\xaf"
|
|
"\xb1\xb4\xb5\xb6\xbb\xbc\xbe\xbf"
|
|
"\xc0\xc1\xc2\xc4\xc7\xc8\xc9\xcb\xcc\xcd\xce\xcf"
|
|
"\xd0\xd1\xd2\xd3\xd4\xd5\xd6\xd8\xca\xdb" ;
|
|
|
|
char StringANSI[] = "\x0a\xC4\xC5\xC7\xC9\xD1\xD6\xDC\xE1\xE0\xE2\xE4\xE3\xE5\xE7\xE9\xE8"
|
|
"\xEA\xEB\xED\xEC\xEE\xEF\xF1\xF3\xF2\xF4\xF6\xF5\xFA\xF9\xFB\xFC"
|
|
"\x86\xBA\xA2\xA3\xA7\x95\xB6\xDF\xAE\xA9\x99\xB4\xA8\xC6\xD8"
|
|
"\xB1\xA5\xB5\xF0\xAA\xBA\xE6\xF8"
|
|
"\xBF\xA1\xAC\x83\xAB\xBB\x85\xC0\xC3\xD5\x8C\x9C"
|
|
"\x96\x97\x93\x94\x91\x92\xF7\xFF\xA0\x80" ;
|
|
|
|
void wxMacConvertFromPC( const char *from , char *to , int len )
|
|
{
|
|
char *c ;
|
|
if ( from == to )
|
|
{
|
|
for( int i = 0 ; i < len ; ++ i )
|
|
{
|
|
c = strchr( StringANSI , *from ) ;
|
|
if ( c != NULL )
|
|
{
|
|
*to = StringMac[ c - StringANSI] ;
|
|
}
|
|
++to ;
|
|
++from ;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for( int i = 0 ; i < len ; ++ i )
|
|
{
|
|
c = strchr( StringANSI , *from ) ;
|
|
if ( c != NULL )
|
|
{
|
|
*to = StringMac[ c - StringANSI] ;
|
|
}
|
|
else
|
|
{
|
|
*to = *from ;
|
|
}
|
|
++to ;
|
|
++from ;
|
|
}
|
|
}
|
|
}
|
|
|
|
void wxMacConvertToPC( const char *from , char *to , int len )
|
|
{
|
|
char *c ;
|
|
if ( from == to )
|
|
{
|
|
for( int i = 0 ; i < len ; ++ i )
|
|
{
|
|
c = strchr( StringMac , *from ) ;
|
|
if ( c != NULL )
|
|
{
|
|
*to = StringANSI[ c - StringMac] ;
|
|
}
|
|
++to ;
|
|
++from ;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for( int i = 0 ; i < len ; ++ i )
|
|
{
|
|
c = strchr( StringMac , *from ) ;
|
|
if ( c != NULL )
|
|
{
|
|
*to = StringANSI[ c - StringMac] ;
|
|
}
|
|
else
|
|
{
|
|
*to = *from ;
|
|
}
|
|
++to ;
|
|
++from ;
|
|
}
|
|
}
|
|
}
|
|
|
|
TECObjectRef s_TECNativeCToUnicode = NULL ;
|
|
TECObjectRef s_TECUnicodeToNativeC = NULL ;
|
|
TECObjectRef s_TECPlatformToNativeC = NULL ;
|
|
TECObjectRef s_TECNativeCToPlatform = NULL ;
|
|
void wxMacSetupConverters()
|
|
{
|
|
// if we assume errors are happening here we need low level debugging
|
|
// since the high level assert will use the encoders that are not yet
|
|
// setup...
|
|
const int kEncoding = wxApp::s_macDefaultEncodingIsPC
|
|
? (int)kTextEncodingWindowsLatin1
|
|
: (int)kTextEncodingMacRoman;
|
|
|
|
OSStatus status = noErr ;
|
|
status = TECCreateConverter(&s_TECNativeCToUnicode,
|
|
kEncoding,
|
|
kTextEncodingUnicodeDefault);
|
|
|
|
|
|
status = TECCreateConverter(&s_TECUnicodeToNativeC,
|
|
kTextEncodingUnicodeDefault,
|
|
kEncoding);
|
|
|
|
if ( wxApp::s_macDefaultEncodingIsPC )
|
|
{
|
|
status = TECCreateConverter(&s_TECPlatformToNativeC,
|
|
kTextEncodingMacRoman,
|
|
kTextEncodingWindowsLatin1);
|
|
|
|
|
|
status = TECCreateConverter(&s_TECNativeCToPlatform,
|
|
kTextEncodingWindowsLatin1,
|
|
kTextEncodingMacRoman);
|
|
}
|
|
}
|
|
|
|
void wxMacCleanupConverters()
|
|
{
|
|
OSStatus status = noErr ;
|
|
status = TECDisposeConverter(s_TECNativeCToUnicode);
|
|
|
|
status = TECDisposeConverter(s_TECUnicodeToNativeC);
|
|
|
|
status = TECDisposeConverter(s_TECPlatformToNativeC);
|
|
|
|
status = TECDisposeConverter(s_TECNativeCToPlatform);
|
|
}
|
|
|
|
wxWCharBuffer wxMacStringToWString( const wxString &from )
|
|
{
|
|
#if wxUSE_UNICODE
|
|
wxWCharBuffer result( from.wc_str() ) ;
|
|
#else
|
|
OSStatus status = noErr ;
|
|
ByteCount byteOutLen ;
|
|
ByteCount byteInLen = from.Length() ;
|
|
ByteCount byteBufferLen = byteInLen *2 ;
|
|
wxWCharBuffer result( from.Length() ) ;
|
|
status = TECConvertText(s_TECNativeCToUnicode, (ConstTextPtr)from.c_str() , byteInLen, &byteInLen,
|
|
(TextPtr)result.data(), byteBufferLen, &byteOutLen);
|
|
result.data()[byteOutLen/2] = 0 ;
|
|
#endif
|
|
return result ;
|
|
}
|
|
|
|
wxString wxMacMakeStringFromCString( const char * from , int len )
|
|
{
|
|
OSStatus status = noErr ;
|
|
wxString result ;
|
|
wxChar* buf = result.GetWriteBuf( len ) ;
|
|
#if wxUSE_UNICODE
|
|
ByteCount byteOutLen ;
|
|
ByteCount byteInLen = len ;
|
|
ByteCount byteBufferLen = len *2 ;
|
|
|
|
status = TECConvertText(s_TECNativeCToUnicode, (ConstTextPtr)from , byteInLen, &byteInLen,
|
|
(TextPtr)buf, byteBufferLen, &byteOutLen);
|
|
#else
|
|
if ( !wxApp::s_macDefaultEncodingIsPC )
|
|
memcpy( buf , from , len ) ;
|
|
else
|
|
{
|
|
ByteCount byteOutLen ;
|
|
ByteCount byteInLen = len ;
|
|
ByteCount byteBufferLen = byteInLen ;
|
|
|
|
status = TECConvertText(s_TECPlatformToNativeC, (ConstTextPtr)from , byteInLen, &byteInLen,
|
|
(TextPtr)buf, byteBufferLen, &byteOutLen);
|
|
}
|
|
#endif
|
|
buf[len] = 0 ;
|
|
result.UngetWriteBuf() ;
|
|
return result ;
|
|
}
|
|
|
|
wxString wxMacMakeStringFromCString( const char * from )
|
|
{
|
|
return wxMacMakeStringFromCString( from , strlen(from) ) ;
|
|
}
|
|
|
|
wxCharBuffer wxMacStringToCString( const wxString &from )
|
|
{
|
|
#if wxUSE_UNICODE
|
|
OSStatus status = noErr ;
|
|
ByteCount byteOutLen ;
|
|
ByteCount byteInLen = from.Length() * 2 ;
|
|
ByteCount byteBufferLen = from.Length() ;
|
|
wxCharBuffer result( from.Length() ) ;
|
|
status = TECConvertText(s_TECUnicodeToNativeC , (ConstTextPtr)from.wc_str() , byteInLen, &byteInLen,
|
|
(TextPtr)result.data(), byteBufferLen, &byteOutLen);
|
|
return result ;
|
|
#else
|
|
if ( !wxApp::s_macDefaultEncodingIsPC )
|
|
return wxCharBuffer( from.c_str() ) ;
|
|
else
|
|
{
|
|
wxCharBuffer result( from.Length() ) ;
|
|
OSStatus status = noErr ;
|
|
ByteCount byteOutLen ;
|
|
ByteCount byteInLen = from.Length() ;
|
|
ByteCount byteBufferLen = byteInLen ;
|
|
|
|
status = TECConvertText(s_TECNativeCToPlatform, (ConstTextPtr)from.c_str() , byteInLen, &byteInLen,
|
|
(TextPtr)result.data(), byteBufferLen, &byteOutLen);
|
|
return result ;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void wxMacStringToPascal( const wxString&from , StringPtr to )
|
|
{
|
|
wxCharBuffer buf = wxMacStringToCString( from ) ;
|
|
int len = strlen(buf) ;
|
|
|
|
if ( len > 255 )
|
|
len = 255 ;
|
|
to[0] = len ;
|
|
memcpy( (char*) &to[1] , buf , len ) ;
|
|
}
|
|
|
|
wxString wxMacMakeStringFromPascal( ConstStringPtr from )
|
|
{
|
|
return wxMacMakeStringFromCString( (char*) &from[1] , from[0] ) ;
|
|
}
|
|
|
|
//
|
|
// CFStringRefs (Carbon only)
|
|
//
|
|
|
|
#if TARGET_CARBON
|
|
// converts this string into a carbon foundation string with optional pc 2 mac encoding
|
|
void wxMacCFStringHolder::Assign( const wxString &str )
|
|
{
|
|
#if wxUSE_UNICODE
|
|
m_cfs = CFStringCreateWithCharacters( kCFAllocatorDefault,
|
|
(const unsigned short*)str.wc_str(), str.Len() );
|
|
#else
|
|
m_cfs = CFStringCreateWithCString( kCFAllocatorSystemDefault , str.c_str() ,
|
|
wxApp::s_macDefaultEncodingIsPC ?
|
|
kCFStringEncodingWindowsLatin1 : CFStringGetSystemEncoding() ) ;
|
|
#endif
|
|
m_release = true ;
|
|
}
|
|
|
|
wxString wxMacCFStringHolder::AsString()
|
|
{
|
|
wxString result ;
|
|
Size len = CFStringGetLength( m_cfs ) ;
|
|
wxChar* buf = result.GetWriteBuf( len ) ;
|
|
#if wxUSE_UNICODE
|
|
CFStringGetCharacters( m_cfs , CFRangeMake( 0 , len ) , (UniChar*) buf ) ;
|
|
#else
|
|
CFStringGetCString( m_cfs , buf , len+1 , wxApp::s_macDefaultEncodingIsPC ?
|
|
kCFStringEncodingWindowsLatin1 : CFStringGetSystemEncoding() ) ;
|
|
#endif
|
|
buf[len] = 0 ;
|
|
result.UngetWriteBuf() ;
|
|
return result ;
|
|
}
|
|
|
|
#if 0
|
|
|
|
wxString wxMacMakeMacStringFromPC( const wxChar * p )
|
|
{
|
|
wxString result ;
|
|
int len = wxStrlen ( p ) ;
|
|
if ( len > 0 )
|
|
{
|
|
wxChar* ptr = result.GetWriteBuf(len) ;
|
|
wxMacConvertFromPC( p , ptr , len ) ;
|
|
ptr[len] = 0 ;
|
|
result.UngetWriteBuf( len ) ;
|
|
}
|
|
return result ;
|
|
}
|
|
|
|
wxString wxMacMakePCStringFromMac( const wxChar * p )
|
|
{
|
|
wxString result ;
|
|
int len = wxStrlen ( p ) ;
|
|
if ( len > 0 )
|
|
{
|
|
wxChar* ptr = result.GetWriteBuf(len) ;
|
|
wxMacConvertToPC( p , ptr , len ) ;
|
|
ptr[len] = 0 ;
|
|
result.UngetWriteBuf( len ) ;
|
|
}
|
|
return result ;
|
|
}
|
|
|
|
wxString wxMacMakeStringFromMacString( const wxChar* from , bool mac2pcEncoding )
|
|
{
|
|
if (mac2pcEncoding)
|
|
{
|
|
return wxMacMakePCStringFromMac( from ) ;
|
|
}
|
|
else
|
|
{
|
|
return wxString( from ) ;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Pascal Strings
|
|
//
|
|
|
|
wxString wxMacMakeStringFromPascal( ConstStringPtr from , bool mac2pcEncoding )
|
|
{
|
|
// this is safe since a pascal string can never be larger than 256 bytes
|
|
char s[256] ;
|
|
CopyPascalStringToC( from , s ) ;
|
|
if (mac2pcEncoding)
|
|
{
|
|
return wxMacMakePCStringFromMac( s ) ;
|
|
}
|
|
else
|
|
{
|
|
return wxString( s ) ;
|
|
}
|
|
}
|
|
|
|
void wxMacStringToPascal( const wxChar * from , StringPtr to , bool pc2macEncoding )
|
|
{
|
|
if (pc2macEncoding)
|
|
{
|
|
CopyCStringToPascal( wxMacMakeMacStringFromPC( from ) , to ) ;
|
|
}
|
|
else
|
|
{
|
|
CopyCStringToPascal( from , to ) ;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
#endif //TARGET_CARBON
|
|
|
|
// ----------------------------------------------------------------------------
|
|
// debugging support
|
|
// ----------------------------------------------------------------------------
|
|
|
|
#if defined(__WXMAC__) && !defined(__DARWIN__) && defined(__MWERKS__) && (__MWERKS__ >= 0x2400)
|
|
|
|
// MetroNub stuff doesn't seem to work in CodeWarrior 5.3 Carbon builds...
|
|
|
|
#ifndef __MetroNubUtils__
|
|
#include "MetroNubUtils.h"
|
|
#endif
|
|
|
|
#ifndef __GESTALT__
|
|
#include <Gestalt.h>
|
|
#endif
|
|
|
|
#if TARGET_API_MAC_CARBON
|
|
|
|
#include <CodeFragments.h>
|
|
|
|
extern "C" long CallUniversalProc(UniversalProcPtr theProcPtr, ProcInfoType procInfo, ...);
|
|
|
|
ProcPtr gCallUniversalProc_Proc = NULL;
|
|
|
|
#endif
|
|
|
|
static MetroNubUserEntryBlock* gMetroNubEntry = NULL;
|
|
|
|
static long fRunOnce = false;
|
|
|
|
/* ---------------------------------------------------------------------------
|
|
IsMetroNubInstalled
|
|
--------------------------------------------------------------------------- */
|
|
|
|
Boolean IsMetroNubInstalled()
|
|
{
|
|
if (!fRunOnce)
|
|
{
|
|
long result, value;
|
|
|
|
fRunOnce = true;
|
|
gMetroNubEntry = NULL;
|
|
|
|
if (Gestalt(gestaltSystemVersion, &value) == noErr && value < 0x1000)
|
|
{
|
|
/* look for MetroNub's Gestalt selector */
|
|
if (Gestalt(kMetroNubUserSignature, &result) == noErr)
|
|
{
|
|
|
|
#if TARGET_API_MAC_CARBON
|
|
if (gCallUniversalProc_Proc == NULL)
|
|
{
|
|
CFragConnectionID connectionID;
|
|
Ptr mainAddress;
|
|
Str255 errorString;
|
|
ProcPtr symbolAddress;
|
|
OSErr err;
|
|
CFragSymbolClass symbolClass;
|
|
|
|
symbolAddress = NULL;
|
|
err = GetSharedLibrary("\pInterfaceLib", kPowerPCCFragArch, kFindCFrag,
|
|
&connectionID, &mainAddress, errorString);
|
|
|
|
if (err != noErr)
|
|
{
|
|
gCallUniversalProc_Proc = NULL;
|
|
goto end;
|
|
}
|
|
|
|
err = FindSymbol(connectionID, "\pCallUniversalProc",
|
|
(Ptr *) &gCallUniversalProc_Proc, &symbolClass);
|
|
|
|
if (err != noErr)
|
|
{
|
|
gCallUniversalProc_Proc = NULL;
|
|
goto end;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
{
|
|
MetroNubUserEntryBlock* block = (MetroNubUserEntryBlock *)result;
|
|
|
|
/* make sure the version of the API is compatible */
|
|
if (block->apiLowVersion <= kMetroNubUserAPIVersion &&
|
|
kMetroNubUserAPIVersion <= block->apiHiVersion)
|
|
gMetroNubEntry = block; /* success! */
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
end:
|
|
|
|
#if TARGET_API_MAC_CARBON
|
|
return (gMetroNubEntry != NULL && gCallUniversalProc_Proc != NULL);
|
|
#else
|
|
return (gMetroNubEntry != NULL);
|
|
#endif
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------
|
|
IsMWDebuggerRunning [v1 API]
|
|
--------------------------------------------------------------------------- */
|
|
|
|
Boolean IsMWDebuggerRunning()
|
|
{
|
|
if (IsMetroNubInstalled())
|
|
return CallIsDebuggerRunningProc(gMetroNubEntry->isDebuggerRunning);
|
|
else
|
|
return false;
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------
|
|
AmIBeingMWDebugged [v1 API]
|
|
--------------------------------------------------------------------------- */
|
|
|
|
Boolean AmIBeingMWDebugged()
|
|
{
|
|
if (IsMetroNubInstalled())
|
|
return CallAmIBeingDebuggedProc(gMetroNubEntry->amIBeingDebugged);
|
|
else
|
|
return false;
|
|
}
|
|
|
|
extern bool WXDLLEXPORT wxIsDebuggerRunning()
|
|
{
|
|
return IsMWDebuggerRunning() && AmIBeingMWDebugged();
|
|
}
|
|
|
|
#else
|
|
|
|
extern bool WXDLLEXPORT wxIsDebuggerRunning()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#endif // defined(__WXMAC__) && !defined(__DARWIN__) && (__MWERKS__ >= 0x2400)
|
|
|
|
#endif // wxUSE_GUI
|
|
|