1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: Emulator wxWidgets sample
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
17 #pragma implementation "emulator.h"
20 // ----------------------------------------------------------------------------
22 // ----------------------------------------------------------------------------
24 // For compilers that support precompilation, includes "wx/wx.h".
25 #include "wx/wxprec.h"
31 // for all others, include the necessary headers (this file is usually all you
32 // need because it includes almost all "standard" wxWidgets headers)
37 #include "wx/confbase.h"
38 #include "wx/fileconf.h"
39 #include "wx/cmdline.h"
44 #include "wx/x11/reparent.h"
49 // ----------------------------------------------------------------------------
51 // ----------------------------------------------------------------------------
53 // the application icon (under Windows and OS/2 it is in resources)
54 #if defined(__WXGTK__) || defined(__WXMOTIF__) || defined(__WXMAC__) || defined(__WXMGL__) || defined(__WXX11__)
55 #include "emulator.xpm"
58 // ----------------------------------------------------------------------------
59 // event tables and other macros for wxWidgets
60 // ----------------------------------------------------------------------------
62 // the event tables connect the wxWidgets events with the functions (event
63 // handlers) which process them. It can be also done at run-time, but for the
64 // simple menu events like this the static method is much simpler.
65 BEGIN_EVENT_TABLE(wxEmulatorFrame
, wxFrame
)
66 EVT_MENU(Emulator_Quit
, wxEmulatorFrame::OnQuit
)
67 EVT_MENU(Emulator_About
, wxEmulatorFrame::OnAbout
)
68 EVT_CLOSE(wxEmulatorFrame::OnCloseWindow
)
71 // Create a new application object: this macro will allow wxWidgets to create
72 // the application object during program execution (it's better than using a
73 // static object for many reasons) and also declares the accessor function
74 // wxGetApp() which will return the reference of the right type (i.e. wxEmulatorApp and
76 IMPLEMENT_APP(wxEmulatorApp
)
78 static const wxCmdLineEntryDesc sg_cmdLineDesc
[] =
80 { wxCMD_LINE_OPTION
, _T("u"), _T("use-display"), _T("display number to use (default 100)") },
82 { wxCMD_LINE_SWITCH
, _T("h"), _T("help"), _T("displays help on the command line parameters") },
83 { wxCMD_LINE_SWITCH
, _T("v"), _T("version"), _T("print version") },
85 { wxCMD_LINE_PARAM
, NULL
, NULL
, _T("config file 1"), wxCMD_LINE_VAL_STRING
, wxCMD_LINE_PARAM_OPTIONAL
},
91 // ============================================================================
93 // ============================================================================
95 // ----------------------------------------------------------------------------
96 // the application class
97 // ----------------------------------------------------------------------------
99 wxEmulatorApp::wxEmulatorApp()
101 m_xnestWindow
= NULL
;
102 m_containerWindow
= NULL
;
103 m_displayNumber
= wxT("100");
108 // 'Main program' equivalent: the program execution "starts" here
109 bool wxEmulatorApp::OnInit()
111 wxLog::SetTimestamp(NULL
);
112 wxInitAllImageHandlers();
114 wxString currentDir
= wxGetCwd();
116 // Use argv to get current app directory
117 m_appDir
= wxFindAppPath(argv
[0], currentDir
, wxT("WXEMUDIR"));
119 // If the development version, go up a directory.
121 if ((m_appDir
.Right(5).CmpNoCase(_T("DEBUG")) == 0) ||
122 (m_appDir
.Right(11).CmpNoCase(_T("DEBUGSTABLE")) == 0) ||
123 (m_appDir
.Right(7).CmpNoCase(_T("RELEASE")) == 0) ||
124 (m_appDir
.Right(13).CmpNoCase(_T("RELEASESTABLE")) == 0)
126 m_appDir
= wxPathOnly(m_appDir
);
129 // Parse the command-line parameters and options
130 wxCmdLineParser
parser(sg_cmdLineDesc
, argc
, argv
);
134 res
= parser
.Parse();
136 if (res
== -1 || res
> 0 || parser
.Found(wxT("h")))
139 wxLog::SetActiveTarget(new wxLogStderr
);
144 if (parser
.Found(wxT("v")))
147 wxLog::SetActiveTarget(new wxLogStderr
);
150 msg
.Printf(wxT("wxWidgets PDA Emulator (c) Julian Smart, 2002 Version %.2f, %s"), wxEMULATOR_VERSION
, __DATE__
);
154 if (parser
.Found(wxT("u"), & m_displayNumber
))
156 // Should only be number, so strip out anything before
157 // and including a : character
158 if (m_displayNumber
.Find(wxT(':')) != -1)
160 m_displayNumber
= m_displayNumber
.AfterFirst(wxT(':'));
163 if (parser
.GetParamCount() == 0)
165 m_emulatorInfo
.m_emulatorFilename
= wxT("default.wxe");
167 else if (parser
.GetParamCount() > 0)
169 m_emulatorInfo
.m_emulatorFilename
= parser
.GetParam(0);
172 // Load the emulation info
173 if (!LoadEmulator(m_appDir
))
175 //wxMessageBox(wxT("Sorry, could not load this emulator. Please check bitmaps are valid."));
179 // create the main application window
180 wxEmulatorFrame
*frame
= new wxEmulatorFrame(_T("wxEmulator"),
181 wxPoint(50, 50), wxSize(450, 340));
183 frame
->SetStatusText(m_emulatorInfo
.m_emulatorTitle
, 0);
186 sizeStr
.Printf(wxT("Screen: %dx%d"), (int) m_emulatorInfo
.m_emulatorScreenSize
.x
,
187 (int) m_emulatorInfo
.m_emulatorScreenSize
.y
);
188 frame
->SetStatusText(sizeStr
, 1);
190 m_containerWindow
= new wxEmulatorContainer(frame
, wxID_ANY
);
192 frame
->SetClientSize(m_emulatorInfo
.m_emulatorDeviceSize
.x
,
193 m_emulatorInfo
.m_emulatorDeviceSize
.y
);
195 // and show it (the frames, unlike simple controls, are not shown when
196 // created initially)
200 m_xnestWindow
= new wxAdoptedWindow
;
203 cmd
.Printf(wxT("Xnest :%s -geometry %dx%d"),
204 m_displayNumber
.c_str(),
205 (int) m_emulatorInfo
.m_emulatorScreenSize
.x
,
206 (int) m_emulatorInfo
.m_emulatorScreenSize
.y
);
208 // Asynchronously executes Xnest
209 m_xnestPID
= wxExecute(cmd
);
213 wxMessageBox(wxT("Sorry, could not run Xnest. Please check your PATH."));
217 wxReparenter reparenter
;
218 if (!reparenter
.WaitAndReparent(m_containerWindow
, m_xnestWindow
, wxT("Xnest")))
220 wxMessageBox(wxT("Sorry, could not reparent Xnest.."));
226 m_containerWindow
->DoResize();
228 // success: wxApp::OnRun() will be called which will enter the main message
229 // loop and the application will run. If we returned false here, the
230 // application would exit immediately.
234 // Prepend the current program directory to the name
235 wxString
wxEmulatorApp::GetFullAppPath(const wxString
& filename
) const
237 wxString
path(m_appDir
);
238 if (path
.Last() != '\\' && path
.Last() != '/' && filename
[0] != '\\' && filename
[0] != '/')
250 // Load the specified emulator.
251 // For now, hard-wired. TODO: make this configurable
252 bool wxEmulatorApp::LoadEmulator(const wxString
& appDir
)
254 // Load config file and bitmaps
255 return m_emulatorInfo
.Load(appDir
);
258 // ----------------------------------------------------------------------------
260 // ----------------------------------------------------------------------------
263 wxEmulatorFrame::wxEmulatorFrame(const wxString
& title
,
264 const wxPoint
& pos
, const wxSize
& size
)
265 : wxFrame(NULL
, wxID_ANY
, title
, pos
, size
)
267 // set the frame icon
268 SetIcon(wxICON(emulator
));
272 wxMenu
*menuFile
= new wxMenu
;
274 // the "About" item should be in the help menu
275 wxMenu
*helpMenu
= new wxMenu
;
276 helpMenu
->Append(Emulator_About
, _T("&About...\tF1"), _T("Show about dialog"));
278 menuFile
->Append(Emulator_Quit
, _T("E&xit\tAlt-X"), _T("Quit this program"));
280 // now append the freshly created menu to the menu bar...
281 wxMenuBar
*menuBar
= new wxMenuBar();
282 menuBar
->Append(menuFile
, _T("&File"));
283 menuBar
->Append(helpMenu
, _T("&Help"));
285 // ... and attach this menu bar to the frame
287 #endif // wxUSE_MENUS
290 // create a status bar just for fun (by default with 1 pane only)
292 #endif // wxUSE_STATUSBAR
298 void wxEmulatorFrame::OnQuit(wxCommandEvent
& WXUNUSED(event
))
300 // true is to force the frame to close
304 void wxEmulatorFrame::OnAbout(wxCommandEvent
& WXUNUSED(event
))
307 msg
.Printf( _T("wxEmulator is an environment for testing embedded X11 apps.\n"));
309 wxMessageBox(msg
, _T("About wxEmulator"), wxOK
| wxICON_INFORMATION
, this);
312 void wxEmulatorFrame::OnCloseWindow(wxCloseEvent
& WXUNUSED(event
))
315 if (wxGetApp().m_xnestWindow
)
317 wxGetApp().m_xnestWindow
->SetHandle((WXWindow
) NULL
);
321 if (wxGetApp().m_xnestPID
> 0)
323 wxKill(wxGetApp().m_xnestPID
);
324 wxGetApp().m_xnestPID
= 0;
328 IMPLEMENT_CLASS(wxEmulatorContainer
, wxWindow
)
330 BEGIN_EVENT_TABLE(wxEmulatorContainer
, wxWindow
)
331 EVT_SIZE(wxEmulatorContainer::OnSize
)
332 EVT_PAINT(wxEmulatorContainer::OnPaint
)
333 EVT_ERASE_BACKGROUND(wxEmulatorContainer::OnEraseBackground
)
336 wxEmulatorContainer::wxEmulatorContainer(wxWindow
* parent
, wxWindowID id
):
337 wxWindow(parent
, id
, wxDefaultPosition
, wxDefaultSize
)
341 void wxEmulatorContainer::OnSize(wxSizeEvent
& WXUNUSED(event
))
346 void wxEmulatorContainer::DoResize()
348 wxSize sz
= GetClientSize();
349 if (wxGetApp().m_xnestWindow
351 && wxGetApp().m_xnestWindow
->GetMainWindow()
355 int deviceWidth
= wxGetApp().m_emulatorInfo
.m_emulatorDeviceSize
.x
;
356 int deviceHeight
= wxGetApp().m_emulatorInfo
.m_emulatorDeviceSize
.y
;
358 int x
= wxMax(0, (int) ((sz
.x
- deviceWidth
)/2.0));
359 int y
= wxMax(0, (int) ((sz
.y
- deviceHeight
)/2.0));
361 x
+= wxGetApp().m_emulatorInfo
.m_emulatorScreenPosition
.x
;
362 y
+= wxGetApp().m_emulatorInfo
.m_emulatorScreenPosition
.y
;
364 wxGetApp().m_xnestWindow
->Move(x
, y
);
369 void wxEmulatorContainer::OnPaint(wxPaintEvent
& WXUNUSED(event
))
373 wxSize sz
= GetClientSize();
374 if (wxGetApp().m_emulatorInfo
.m_emulatorBackgroundBitmap
.Ok())
376 int deviceWidth
= wxGetApp().m_emulatorInfo
.m_emulatorDeviceSize
.x
;
377 int deviceHeight
= wxGetApp().m_emulatorInfo
.m_emulatorDeviceSize
.y
;
379 int x
= wxMax(0, (int) ((sz
.x
- deviceWidth
)/2.0));
380 int y
= wxMax(0, (int) ((sz
.y
- deviceHeight
)/2.0));
382 dc
.DrawBitmap(wxGetApp().m_emulatorInfo
.m_emulatorBackgroundBitmap
, x
, y
);
386 void wxEmulatorContainer::OnEraseBackground(wxEraseEvent
& event
)
388 wxDC
* dc
wxDUMMY_INITIALIZE(NULL
);
396 dc
= new wxClientDC(this);
399 dc
->SetBackground(wxBrush(wxGetApp().m_emulatorInfo
.m_emulatorBackgroundColour
, wxSOLID
));
406 // Information about the emulator decorations
408 void wxEmulatorInfo::Copy(const wxEmulatorInfo
& info
)
410 m_emulatorFilename
= info
.m_emulatorFilename
;
411 m_emulatorTitle
= info
.m_emulatorTitle
;
412 m_emulatorDescription
= info
.m_emulatorDescription
;
413 m_emulatorScreenPosition
= info
.m_emulatorScreenPosition
;
414 m_emulatorScreenSize
= info
.m_emulatorScreenSize
;
415 m_emulatorBackgroundBitmap
= info
.m_emulatorBackgroundBitmap
;
416 m_emulatorBackgroundBitmapName
= info
.m_emulatorBackgroundBitmapName
;
417 m_emulatorBackgroundColour
= info
.m_emulatorBackgroundColour
;
418 m_emulatorDeviceSize
= info
.m_emulatorDeviceSize
;
422 void wxEmulatorInfo::Init()
424 m_emulatorDeviceSize
= wxSize(260, 340);
425 m_emulatorScreenSize
= wxSize(240, 320);
429 bool wxEmulatorInfo::Load(const wxString
& appDir
)
431 // Try to find absolute path
432 wxString absoluteConfigPath
= m_emulatorFilename
;
433 if ( !::wxIsAbsolutePath(absoluteConfigPath
) )
435 wxString currDir
= wxGetCwd();
436 absoluteConfigPath
= currDir
+ wxString(wxFILE_SEP_PATH
) + m_emulatorFilename
;
437 if ( !wxFile::Exists(absoluteConfigPath
) )
439 absoluteConfigPath
= appDir
+ wxString(wxFILE_SEP_PATH
)
440 + m_emulatorFilename
;
444 if ( !wxFile::Exists(absoluteConfigPath
) )
447 str
.Printf( wxT("Could not find config file %s"),
448 absoluteConfigPath
.c_str() );
454 wxString rootPath
= wxPathOnly(absoluteConfigPath
);
457 wxFileConfig
config(wxT("wxEmulator"), wxT("wxWidgets"),
458 absoluteConfigPath
, wxEmptyString
, wxCONFIG_USE_LOCAL_FILE
);
460 config
.Read(wxT("/General/title"), & m_emulatorTitle
);
461 config
.Read(wxT("/General/description"), & m_emulatorDescription
);
462 config
.Read(wxT("/General/backgroundBitmap"), & m_emulatorBackgroundBitmapName
);
465 if (config
.Read(wxT("/General/backgroundColour"), & colString
) ||
466 config
.Read(wxT("/General/backgroundColor"), & colString
)
469 m_emulatorBackgroundColour
= wxHexStringToColour(colString
);
472 int x
= 0, y
= 0, w
= 0, h
= 0, dw
= 0, dh
= 0;
473 config
.Read(wxT("/General/screenX"), & x
);
474 config
.Read(wxT("/General/screenY"), & y
);
475 config
.Read(wxT("/General/screenWidth"), & w
);
476 config
.Read(wxT("/General/screenHeight"), & h
);
477 if (config
.Read(wxT("/General/deviceWidth"), & dw
) && config
.Read(wxT("/General/deviceHeight"), & dh
))
479 m_emulatorDeviceSize
= wxSize(dw
, dh
);
482 m_emulatorScreenPosition
= wxPoint(x
, y
);
483 m_emulatorScreenSize
= wxSize(w
, h
);
486 if (!m_emulatorBackgroundBitmapName
.IsEmpty())
488 wxString absoluteBackgroundBitmapName
= rootPath
+ wxString(wxFILE_SEP_PATH
) + m_emulatorBackgroundBitmapName
;
489 if ( !wxFile::Exists(absoluteBackgroundBitmapName
) )
492 str
.Printf( wxT("Could not find bitmap %s"),
493 absoluteBackgroundBitmapName
.c_str() );
498 wxBitmapType type
= wxDetermineImageType(m_emulatorBackgroundBitmapName
);
499 if (type
== wxBITMAP_TYPE_INVALID
)
502 if (!m_emulatorBackgroundBitmap
.LoadFile(m_emulatorBackgroundBitmapName
, type
))
505 str
.Printf( wxT("Could not load bitmap file %s"),
506 m_emulatorBackgroundBitmapName
.c_str() );
511 m_emulatorDeviceSize
= wxSize(m_emulatorBackgroundBitmap
.GetWidth(),
512 m_emulatorBackgroundBitmap
.GetHeight());
517 // Returns the image type, or -1, determined from the extension.
518 wxBitmapType
wxDetermineImageType(const wxString
& filename
)
520 wxString path
, name
, ext
;
522 wxSplitPath(filename
, & path
, & name
, & ext
);
525 if (ext
== _T("jpg") || ext
== _T("jpeg"))
526 return wxBITMAP_TYPE_JPEG
;
527 if (ext
== _T("gif"))
528 return wxBITMAP_TYPE_GIF
;
529 if (ext
== _T("bmp"))
530 return wxBITMAP_TYPE_BMP
;
531 if (ext
== _T("png"))
532 return wxBITMAP_TYPE_PNG
;
533 if (ext
== _T("pcx"))
534 return wxBITMAP_TYPE_PCX
;
535 if (ext
== _T("tif") || ext
== _T("tiff"))
536 return wxBITMAP_TYPE_TIF
;
538 return wxBITMAP_TYPE_INVALID
;
541 // Convert a colour to a 6-digit hex string
542 wxString
wxColourToHexString(const wxColour
& col
)
546 hex
+= wxDecToHex(col
.Red());
547 hex
+= wxDecToHex(col
.Green());
548 hex
+= wxDecToHex(col
.Blue());
553 // Convert 6-digit hex string to a colour
554 wxColour
wxHexStringToColour(const wxString
& hex
)
556 unsigned int r
= wxHexToDec(hex
.Mid(0, 2));
557 unsigned int g
= wxHexToDec(hex
.Mid(2, 2));
558 unsigned int b
= wxHexToDec(hex
.Mid(4, 2));
560 return wxColour(r
, g
, b
);
563 // Find the absolute path where this application has been run from.
564 // argv0 is wxTheApp->argv[0]
565 // cwd is the current working directory (at startup)
566 // appVariableName is the name of a variable containing the directory for this app, e.g.
567 // MYAPPDIR. This is checked first.
569 wxString
wxFindAppPath(const wxString
& argv0
, const wxString
& cwd
, const wxString
& appVariableName
)
573 // Try appVariableName
574 if (!appVariableName
.IsEmpty())
576 str
= wxGetenv(appVariableName
);
581 if (wxIsAbsolutePath(argv0
))
582 return wxPathOnly(argv0
);
585 // Is it a relative path?
586 wxString
currentDir(cwd
);
587 if (!wxEndsWithPathSeparator(currentDir
))
588 currentDir
+= wxFILE_SEP_PATH
;
590 str
= currentDir
+ argv0
;
591 if ( wxFile::Exists(str
) )
592 return wxPathOnly(str
);
595 // OK, it's neither an absolute path nor a relative path.
599 pathList
.AddEnvList(wxT("PATH"));
600 str
= pathList
.FindAbsoluteValidPath(argv0
);
602 return wxPathOnly(str
);
605 return wxEmptyString
;