1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     exec sample demonstrates wxExecute and related functions 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) Vadim Zeitlin 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  20 // For compilers that support precompilation, includes "wx/wx.h". 
  21 #include "wx/wxprec.h" 
  27 // for all others, include the necessary headers (this file is usually all you 
  28 // need because it includes almost all "standard" wxWidgets headers 
  40     #include "wx/msgdlg.h" 
  41     #include "wx/textdlg.h" 
  42     #include "wx/filedlg.h" 
  43     #include "wx/choicdlg.h" 
  45     #include "wx/button.h" 
  46     #include "wx/textctrl.h" 
  47     #include "wx/listbox.h" 
  52 #include "wx/txtstrm.h" 
  53 #include "wx/numdlg.h" 
  54 #include "wx/textdlg.h" 
  56 #include "wx/stopwatch.h" 
  58 #include "wx/process.h" 
  60 #include "wx/mimetype.h" 
  66 // ---------------------------------------------------------------------------- 
  67 // the usual application and main frame classes 
  68 // ---------------------------------------------------------------------------- 
  70 // Define a new application type, each program should derive a class from wxApp 
  71 class MyApp 
: public wxApp
 
  74     // override base class virtuals 
  75     // ---------------------------- 
  77     // this one is called on application startup and is a good place for the app 
  78     // initialization (doing it here and not in the ctor allows to have an error 
  79     // return: if OnInit() returns false, the application terminates) 
  80     virtual bool OnInit(); 
  83 // Define an array of process pointers used by MyFrame 
  85 WX_DEFINE_ARRAY_PTR(MyPipedProcess 
*, MyPipedProcessesArray
); 
  88 WX_DEFINE_ARRAY_PTR(MyProcess 
*, MyProcessesArray
); 
  90 // Define a new frame type: this is going to be our main frame 
  91 class MyFrame 
: public wxFrame
 
  95     MyFrame(const wxString
& title
, const wxPoint
& pos
, const wxSize
& size
); 
  98     // event handlers (these functions should _not_ be virtual) 
  99     void OnQuit(wxCommandEvent
& event
); 
 101     void OnKill(wxCommandEvent
& event
); 
 103     void OnClear(wxCommandEvent
& event
); 
 105     void OnBeginBusyCursor(wxCommandEvent
& event
); 
 106     void OnEndBusyCursor(wxCommandEvent
& event
); 
 108     void OnSyncExec(wxCommandEvent
& event
); 
 109     void OnSyncNoEventsExec(wxCommandEvent
& event
); 
 110     void OnAsyncExec(wxCommandEvent
& event
); 
 111     void OnShell(wxCommandEvent
& event
); 
 112     void OnExecWithRedirect(wxCommandEvent
& event
); 
 113     void OnExecWithPipe(wxCommandEvent
& event
); 
 115     void OnPOpen(wxCommandEvent
& event
); 
 117     void OnFileExec(wxCommandEvent
& event
); 
 118     void OnOpenURL(wxCommandEvent
& event
); 
 120     void OnAbout(wxCommandEvent
& event
); 
 122     // polling output of async processes 
 123     void OnTimer(wxTimerEvent
& event
); 
 124     void OnIdle(wxIdleEvent
& event
); 
 126     // for MyPipedProcess 
 127     void OnProcessTerminated(MyPipedProcess 
*process
); 
 128     wxListBox 
*GetLogListBox() const { return m_lbox
; } 
 131     void OnAsyncTermination(MyProcess 
*process
); 
 134     void ShowOutput(const wxString
& cmd
, 
 135                     const wxArrayString
& output
, 
 136                     const wxString
& title
); 
 138     void DoAsyncExec(const wxString
& cmd
); 
 140     void AddAsyncProcess(MyProcess 
*process
) { m_allAsync
.push_back(process
); } 
 142     void AddPipedProcess(MyPipedProcess 
*process
); 
 143     void RemovePipedProcess(MyPipedProcess 
*process
); 
 146     // the PID of the last process we launched asynchronously 
 149     // last command we executed 
 153     void OnDDEExec(wxCommandEvent
& event
); 
 154     void OnDDERequest(wxCommandEvent
& event
); 
 158     // last params of a DDE transaction 
 162 #endif // __WINDOWS__ 
 166     // array of running processes with redirected IO 
 167     MyPipedProcessesArray m_running
; 
 169     // array of all asynchrously running processes 
 170     MyProcessesArray m_allAsync
; 
 172     // the idle event wake up timer 
 173     wxTimer m_timerIdleWakeUp
; 
 175     // any class wishing to process wxWidgets events must use this macro 
 176     DECLARE_EVENT_TABLE() 
 179 // ---------------------------------------------------------------------------- 
 180 // MyPipeFrame: allows the user to communicate with the child process 
 181 // ---------------------------------------------------------------------------- 
 183 class MyPipeFrame 
: public wxFrame
 
 186     MyPipeFrame(wxFrame 
*parent
, 
 191     void OnTextEnter(wxCommandEvent
& WXUNUSED(event
)) { DoSend(); } 
 192     void OnBtnSend(wxCommandEvent
& WXUNUSED(event
)) { DoSend(); } 
 193     void OnBtnSendFile(wxCommandEvent
& WXUNUSED(event
)); 
 194     void OnBtnGet(wxCommandEvent
& WXUNUSED(event
)) { DoGet(); } 
 195     void OnBtnClose(wxCommandEvent
& WXUNUSED(event
)) { DoClose(); } 
 197     void OnClose(wxCloseEvent
& event
); 
 199     void OnProcessTerm(wxProcessEvent
& event
); 
 203         wxString 
s(m_textOut
->GetValue()); 
 205         m_out
.Write(s
.c_str(), s
.length()); 
 215     void DoGetFromStream(wxTextCtrl 
*text
, wxInputStream
& in
); 
 217     void DisableOutput(); 
 220     wxProcess 
*m_process
; 
 222     wxOutputStream 
&m_out
; 
 226     wxTextCtrl 
*m_textOut
, 
 230     DECLARE_EVENT_TABLE() 
 233 // ---------------------------------------------------------------------------- 
 234 // wxProcess-derived classes 
 235 // ---------------------------------------------------------------------------- 
 237 // This is the handler for process termination events 
 238 class MyProcess 
: public wxProcess
 
 241     MyProcess(MyFrame 
*parent
, const wxString
& cmd
) 
 242         : wxProcess(parent
), m_cmd(cmd
) 
 247     // instead of overriding this virtual function we might as well process the 
 248     // event from it in the frame class - this might be more convenient in some 
 250     virtual void OnTerminate(int pid
, int status
); 
 257 // A specialization of MyProcess for redirecting the output 
 258 class MyPipedProcess 
: public MyProcess
 
 261     MyPipedProcess(MyFrame 
*parent
, const wxString
& cmd
) 
 262         : MyProcess(parent
, cmd
) 
 267     virtual void OnTerminate(int pid
, int status
); 
 269     virtual bool HasInput(); 
 272 // A version of MyPipedProcess which also sends input to the stdin of the 
 274 class MyPipedProcess2 
: public MyPipedProcess
 
 277     MyPipedProcess2(MyFrame 
*parent
, const wxString
& cmd
, const wxString
& input
) 
 278         : MyPipedProcess(parent
, cmd
), m_input(input
) 
 282     virtual bool HasInput(); 
 288 // ---------------------------------------------------------------------------- 
 290 // ---------------------------------------------------------------------------- 
 292 // IDs for the controls and the menu commands 
 299     Exec_BeginBusyCursor
, 
 302     Exec_SyncNoEventsExec
, 
 315     Exec_Btn_Send 
= 1000, 
 321 static const wxChar 
*DIALOG_TITLE 
= _T("Exec sample"); 
 323 // ---------------------------------------------------------------------------- 
 324 // event tables and other macros for wxWidgets 
 325 // ---------------------------------------------------------------------------- 
 327 // the event tables connect the wxWidgets events with the functions (event 
 328 // handlers) which process them. It can be also done at run-time, but for the 
 329 // simple menu events like this the static method is much simpler. 
 330 BEGIN_EVENT_TABLE(MyFrame
, wxFrame
) 
 331     EVT_MENU(Exec_Quit
,  MyFrame::OnQuit
) 
 332     EVT_MENU(Exec_Kill
,  MyFrame::OnKill
) 
 333     EVT_MENU(Exec_ClearLog
,  MyFrame::OnClear
) 
 334     EVT_MENU(Exec_BeginBusyCursor
,  MyFrame::OnBeginBusyCursor
) 
 335     EVT_MENU(Exec_EndBusyCursor
,  MyFrame::OnEndBusyCursor
) 
 337     EVT_MENU(Exec_SyncExec
, MyFrame::OnSyncExec
) 
 338     EVT_MENU(Exec_SyncNoEventsExec
, MyFrame::OnSyncNoEventsExec
) 
 339     EVT_MENU(Exec_AsyncExec
, MyFrame::OnAsyncExec
) 
 340     EVT_MENU(Exec_Shell
, MyFrame::OnShell
) 
 341     EVT_MENU(Exec_Redirect
, MyFrame::OnExecWithRedirect
) 
 342     EVT_MENU(Exec_Pipe
, MyFrame::OnExecWithPipe
) 
 344     EVT_MENU(Exec_POpen
, MyFrame::OnPOpen
) 
 346     EVT_MENU(Exec_OpenFile
, MyFrame::OnFileExec
) 
 347     EVT_MENU(Exec_OpenURL
, MyFrame::OnOpenURL
) 
 350     EVT_MENU(Exec_DDEExec
, MyFrame::OnDDEExec
) 
 351     EVT_MENU(Exec_DDERequest
, MyFrame::OnDDERequest
) 
 352 #endif // __WINDOWS__ 
 354     EVT_MENU(Exec_About
, MyFrame::OnAbout
) 
 356     EVT_IDLE(MyFrame::OnIdle
) 
 358     EVT_TIMER(wxID_ANY
, MyFrame::OnTimer
) 
 361 BEGIN_EVENT_TABLE(MyPipeFrame
, wxFrame
) 
 362     EVT_BUTTON(Exec_Btn_Send
, MyPipeFrame::OnBtnSend
) 
 363     EVT_BUTTON(Exec_Btn_SendFile
, MyPipeFrame::OnBtnSendFile
) 
 364     EVT_BUTTON(Exec_Btn_Get
, MyPipeFrame::OnBtnGet
) 
 365     EVT_BUTTON(Exec_Btn_Close
, MyPipeFrame::OnBtnClose
) 
 367     EVT_TEXT_ENTER(wxID_ANY
, MyPipeFrame::OnTextEnter
) 
 369     EVT_CLOSE(MyPipeFrame::OnClose
) 
 371     EVT_END_PROCESS(wxID_ANY
, MyPipeFrame::OnProcessTerm
) 
 374 // Create a new application object: this macro will allow wxWidgets to create 
 375 // the application object during program execution (it's better than using a 
 376 // static object for many reasons) and also declares the accessor function 
 377 // wxGetApp() which will return the reference of the right type (i.e. MyApp and 
 381 // ============================================================================ 
 383 // ============================================================================ 
 385 // ---------------------------------------------------------------------------- 
 386 // the application class 
 387 // ---------------------------------------------------------------------------- 
 389 // `Main program' equivalent: the program execution "starts" here 
 392     if ( !wxApp::OnInit() ) 
 395     // Create the main application window 
 396     MyFrame 
*frame 
= new MyFrame(_T("Exec wxWidgets sample"), 
 397                                  wxDefaultPosition
, wxSize(500, 140)); 
 399     // Show it and tell the application that it's our main window 
 403     // success: wxApp::OnRun() will be called which will enter the main message 
 404     // loop and the application will run. If we returned false here, the 
 405     // application would exit immediately. 
 409 // ---------------------------------------------------------------------------- 
 411 // ---------------------------------------------------------------------------- 
 414 #pragma warning(disable: 4355) // this used in base member initializer list 
 418 MyFrame::MyFrame(const wxString
& title
, const wxPoint
& pos
, const wxSize
& size
) 
 419        : wxFrame((wxFrame 
*)NULL
, wxID_ANY
, title
, pos
, size
), 
 420          m_timerIdleWakeUp(this) 
 425     // we need this in order to allow the about menu relocation, since ABOUT is 
 426     // not the default id of the about menu 
 427     wxApp::s_macAboutMenuItemId 
= Exec_About
; 
 431     wxMenu 
*menuFile 
= new wxMenu(wxEmptyString
, wxMENU_TEAROFF
); 
 432     menuFile
->Append(Exec_Kill
, _T("&Kill process...\tCtrl-K"), 
 433                      _T("Kill a process by PID")); 
 434     menuFile
->AppendSeparator(); 
 435     menuFile
->Append(Exec_ClearLog
, _T("&Clear log\tCtrl-L"), 
 436                      _T("Clear the log window")); 
 437     menuFile
->AppendSeparator(); 
 438     menuFile
->Append(Exec_BeginBusyCursor
, _T("Show &busy cursor\tCtrl-C")); 
 439     menuFile
->Append(Exec_EndBusyCursor
, _T("Show &normal cursor\tShift-Ctrl-C")); 
 440     menuFile
->AppendSeparator(); 
 441     menuFile
->Append(Exec_Quit
, _T("E&xit\tAlt-X"), _T("Quit this program")); 
 443     wxMenu 
*execMenu 
= new wxMenu
; 
 444     execMenu
->Append(Exec_SyncExec
, _T("Sync &execution...\tCtrl-E"), 
 445                      _T("Launch a program and return when it terminates")); 
 446     execMenu
->Append(Exec_SyncNoEventsExec
, _T("Sync execution and &block...\tCtrl-B"), 
 447                      _T("Launch a program and block until it terminates")); 
 448     execMenu
->Append(Exec_AsyncExec
, _T("&Async execution...\tCtrl-A"), 
 449                      _T("Launch a program and return immediately")); 
 450     execMenu
->Append(Exec_Shell
, _T("Execute &shell command...\tCtrl-S"), 
 451                      _T("Launch a shell and execute a command in it")); 
 452     execMenu
->AppendSeparator(); 
 453     execMenu
->Append(Exec_Redirect
, _T("Capture command &output...\tCtrl-O"), 
 454                      _T("Launch a program and capture its output")); 
 455     execMenu
->Append(Exec_Pipe
, _T("&Pipe through command..."), 
 456                      _T("Pipe a string through a filter")); 
 457     execMenu
->Append(Exec_POpen
, _T("&Open a pipe to a command...\tCtrl-P"), 
 458                      _T("Open a pipe to and from another program")); 
 460     execMenu
->AppendSeparator(); 
 461     execMenu
->Append(Exec_OpenFile
, _T("Open &file...\tCtrl-F"), 
 462                      _T("Launch the command to open this kind of files")); 
 463     execMenu
->Append(Exec_OpenURL
, _T("Open &URL...\tCtrl-U"), 
 464                      _T("Launch the default browser with the given URL")); 
 466     execMenu
->AppendSeparator(); 
 467     execMenu
->Append(Exec_DDEExec
, _T("Execute command via &DDE...\tCtrl-D")); 
 468     execMenu
->Append(Exec_DDERequest
, _T("Send DDE &request...\tCtrl-R")); 
 471     wxMenu 
*helpMenu 
= new wxMenu(wxEmptyString
, wxMENU_TEAROFF
); 
 472     helpMenu
->Append(Exec_About
, _T("&About...\tF1"), _T("Show about dialog")); 
 474     // now append the freshly created menu to the menu bar... 
 475     wxMenuBar 
*menuBar 
= new wxMenuBar(); 
 476     menuBar
->Append(menuFile
, _T("&File")); 
 477     menuBar
->Append(execMenu
, _T("&Exec")); 
 478     menuBar
->Append(helpMenu
, _T("&Help")); 
 480     // ... and attach this menu bar to the frame 
 483     // create the listbox in which we will show misc messages as they come 
 484     m_lbox 
= new wxListBox(this, wxID_ANY
); 
 485     wxFont 
font(12, wxFONTFAMILY_TELETYPE
, wxFONTSTYLE_NORMAL
, 
 486                 wxFONTWEIGHT_NORMAL
); 
 488         m_lbox
->SetFont(font
); 
 491     // create a status bar just for fun (by default with 1 pane only) 
 493     SetStatusText(_T("Welcome to wxWidgets exec sample!")); 
 494 #endif // wxUSE_STATUSBAR 
 499     // any processes left until now must be deleted manually: normally this is 
 500     // done when the associated process terminates but it must be still running 
 501     // if this didn't happen until now 
 502     for ( size_t n 
= 0; n 
< m_allAsync
.size(); n
++ ) 
 504         delete m_allAsync
[n
]; 
 508 // ---------------------------------------------------------------------------- 
 509 // event handlers: file and help menu 
 510 // ---------------------------------------------------------------------------- 
 512 void MyFrame::OnQuit(wxCommandEvent
& WXUNUSED(event
)) 
 514     // true is to force the frame to close 
 518 void MyFrame::OnClear(wxCommandEvent
& WXUNUSED(event
)) 
 523 void MyFrame::OnBeginBusyCursor(wxCommandEvent
& WXUNUSED(event
)) 
 528 void MyFrame::OnEndBusyCursor(wxCommandEvent
& WXUNUSED(event
)) 
 533 void MyFrame::OnAbout(wxCommandEvent
& WXUNUSED(event
)) 
 535     wxMessageBox(_T("Exec wxWidgets Sample\n(c) 2000-2002 Vadim Zeitlin"), 
 536                  _T("About Exec"), wxOK 
| wxICON_INFORMATION
, this); 
 539 void MyFrame::OnKill(wxCommandEvent
& WXUNUSED(event
)) 
 541     long pid 
= wxGetNumberFromUser(_T("Please specify the process to kill"), 
 545                                    // we need the full unsigned int range 
 554     static const wxString signalNames
[] = 
 556         _T("Just test (SIGNONE)"), 
 557         _T("Hangup (SIGHUP)"), 
 558         _T("Interrupt (SIGINT)"), 
 559         _T("Quit (SIGQUIT)"), 
 560         _T("Illegal instruction (SIGILL)"), 
 561         _T("Trap (SIGTRAP)"), 
 562         _T("Abort (SIGABRT)"), 
 563         _T("Emulated trap (SIGEMT)"), 
 564         _T("FP exception (SIGFPE)"), 
 565         _T("Kill (SIGKILL)"), 
 567         _T("Segment violation (SIGSEGV)"), 
 568         _T("System (SIGSYS)"), 
 569         _T("Broken pipe (SIGPIPE)"), 
 570         _T("Alarm (SIGALRM)"), 
 571         _T("Terminate (SIGTERM)"), 
 574     int sig 
= wxGetSingleChoiceIndex(_T("How to kill the process?"), 
 576                                      WXSIZEOF(signalNames
), signalNames
, 
 581             wxFAIL_MSG( _T("unexpected return value") ); 
 609         if ( wxProcess::Exists(pid
) ) 
 610             wxLogStatus(_T("Process %ld is running."), pid
); 
 612             wxLogStatus(_T("No process with pid = %ld."), pid
); 
 616         wxKillError rc 
= wxProcess::Kill(pid
, (wxSignal
)sig
); 
 617         if ( rc 
== wxKILL_OK 
) 
 619             wxLogStatus(_T("Process %ld killed with signal %d."), pid
, sig
); 
 623             static const wxChar 
*errorText
[] = 
 626                 _T("signal not supported"), 
 627                 _T("permission denied"), 
 628                 _T("no such process"), 
 629                 _T("unspecified error"), 
 632             wxLogStatus(_T("Failed to kill process %ld with signal %d: %s"), 
 633                         pid
, sig
, errorText
[rc
]); 
 638 // ---------------------------------------------------------------------------- 
 639 // event handlers: exec menu 
 640 // ---------------------------------------------------------------------------- 
 642 void MyFrame::DoAsyncExec(const wxString
& cmd
) 
 644     MyProcess 
* const process 
= new MyProcess(this, cmd
); 
 645     m_pidLast 
= wxExecute(cmd
, wxEXEC_ASYNC
, process
); 
 648         wxLogError(_T("Execution of '%s' failed."), cmd
.c_str()); 
 654         wxLogStatus(_T("Process %ld (%s) launched."), m_pidLast
, cmd
.c_str()); 
 658         // the parent frame keeps track of all async processes as it needs to 
 659         // free them if we exit before the child process terminates 
 660         AddAsyncProcess(process
); 
 664 void MyFrame::OnSyncExec(wxCommandEvent
& WXUNUSED(event
)) 
 666     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 673     wxLogStatus( _T("'%s' is running please wait..."), cmd
.c_str() ); 
 675     int code 
= wxExecute(cmd
, wxEXEC_SYNC
); 
 677     wxLogStatus(_T("Process '%s' terminated with exit code %d."), 
 683 void MyFrame::OnSyncNoEventsExec(wxCommandEvent
& WXUNUSED(event
)) 
 685     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 692     wxLogStatus( _T("'%s' is running please wait..."), cmd
.c_str() ); 
 694     int code 
= wxExecute(cmd
, wxEXEC_BLOCK
); 
 696     wxLogStatus(_T("Process '%s' terminated with exit code %d."), 
 702 void MyFrame::OnAsyncExec(wxCommandEvent
& WXUNUSED(event
)) 
 704     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 714 void MyFrame::OnShell(wxCommandEvent
& WXUNUSED(event
)) 
 716     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 723     int code 
= wxShell(cmd
); 
 724     wxLogStatus(_T("Shell command '%s' terminated with exit code %d."), 
 729 void MyFrame::OnExecWithRedirect(wxCommandEvent
& WXUNUSED(event
)) 
 734         m_cmdLast 
= "type Makefile.in"; 
 736         m_cmdLast 
= "cat -n Makefile"; 
 740     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 748     switch ( wxMessageBox(_T("Execute it synchronously?"), 
 750                           wxYES_NO 
| wxCANCEL 
| wxICON_QUESTION
, this) ) 
 766         wxLogStatus("\"%s\" is running please wait...", cmd
); 
 770         wxArrayString output
, errors
; 
 771         int code 
= wxExecute(cmd
, output
, errors
); 
 773         wxLogStatus("Command \"%s\" terminated after %ldms; exit code %d.", 
 774                     cmd
, sw
.Time(), code
); 
 778             ShowOutput(cmd
, output
, _T("Output")); 
 779             ShowOutput(cmd
, errors
, _T("Errors")); 
 784         MyPipedProcess 
*process 
= new MyPipedProcess(this, cmd
); 
 785         if ( !wxExecute(cmd
, wxEXEC_ASYNC
, process
) ) 
 787             wxLogError(_T("Execution of '%s' failed."), cmd
.c_str()); 
 793             AddPipedProcess(process
); 
 800 void MyFrame::OnExecWithPipe(wxCommandEvent
& WXUNUSED(event
)) 
 803         m_cmdLast 
= _T("tr [a-z] [A-Z]"); 
 805     wxString cmd 
= wxGetTextFromUser(_T("Enter the command: "), 
 812     wxString input 
= wxGetTextFromUser(_T("Enter the string to send to it: "), 
 817     // always execute the filter asynchronously 
 818     MyPipedProcess2 
*process 
= new MyPipedProcess2(this, cmd
, input
); 
 819     long pid 
= wxExecute(cmd
, wxEXEC_ASYNC
, process
); 
 822         wxLogStatus(_T("Process %ld (%s) launched."), pid
, cmd
.c_str()); 
 824         AddPipedProcess(process
); 
 828         wxLogError(_T("Execution of '%s' failed."), cmd
.c_str()); 
 836 void MyFrame::OnPOpen(wxCommandEvent
& WXUNUSED(event
)) 
 838     wxString cmd 
= wxGetTextFromUser(_T("Enter the command to launch: "), 
 844     wxProcess 
*process 
= wxProcess::Open(cmd
); 
 847         wxLogError(_T("Failed to launch the command.")); 
 851     wxLogVerbose(_T("PID of the new process: %ld"), process
->GetPid()); 
 853     wxOutputStream 
*out 
= process
->GetOutputStream(); 
 856         wxLogError(_T("Failed to connect to child stdin")); 
 860     wxInputStream 
*in 
= process
->GetInputStream(); 
 863         wxLogError(_T("Failed to connect to child stdout")); 
 867     new MyPipeFrame(this, cmd
, process
); 
 870 void MyFrame::OnFileExec(wxCommandEvent
& WXUNUSED(event
)) 
 872     static wxString s_filename
; 
 877     filename 
= wxLoadFileSelector(_T("any file"), wxEmptyString
, s_filename
, this); 
 878 #else // !wxUSE_FILEDLG 
 879     filename 
= wxGetTextFromUser(_T("Enter the file name"), _T("exec sample"), 
 881 #endif // wxUSE_FILEDLG/!wxUSE_FILEDLG 
 883     if ( filename
.empty() ) 
 886     s_filename 
= filename
; 
 888     wxString ext 
= filename
.AfterFirst(_T('.')); 
 889     wxFileType 
*ft 
= wxTheMimeTypesManager
->GetFileTypeFromExtension(ext
); 
 892         wxLogError(_T("Impossible to determine the file type for extension '%s'"), 
 898     bool ok 
= ft
->GetOpenCommand(&cmd
, 
 899                                  wxFileType::MessageParameters(filename
)); 
 903         wxLogError(_T("Impossible to find out how to open files of extension '%s'"), 
 911 void MyFrame::OnOpenURL(wxCommandEvent
& WXUNUSED(event
)) 
 913     static wxString 
s_filename(_T("http://www.wxwidgets.org/")); 
 915     wxString filename 
= wxGetTextFromUser
 
 923     if ( filename
.empty() ) 
 926     s_filename 
= filename
; 
 928     if ( !wxLaunchDefaultBrowser(s_filename
) ) 
 929         wxLogError(_T("Failed to open URL \"%s\""), s_filename
.c_str()); 
 932 // ---------------------------------------------------------------------------- 
 934 // ---------------------------------------------------------------------------- 
 938 bool MyFrame::GetDDEServer() 
 940     wxString server 
= wxGetTextFromUser(_T("Server to connect to:"), 
 941                                         DIALOG_TITLE
, m_server
); 
 947     wxString topic 
= wxGetTextFromUser(_T("DDE topic:"), DIALOG_TITLE
, m_topic
); 
 953     wxString cmd 
= wxGetTextFromUser(_T("DDE command:"), DIALOG_TITLE
, m_cmdDde
); 
 962 void MyFrame::OnDDEExec(wxCommandEvent
& WXUNUSED(event
)) 
 964     if ( !GetDDEServer() ) 
 968     wxConnectionBase 
*conn 
= client
.MakeConnection(wxEmptyString
, m_server
, m_topic
); 
 971         wxLogError(_T("Failed to connect to the DDE server '%s'."), 
 976         if ( !conn
->Execute(m_cmdDde
) ) 
 978             wxLogError(_T("Failed to execute command '%s' via DDE."), 
 983             wxLogStatus(_T("Successfully executed DDE command")); 
 988 void MyFrame::OnDDERequest(wxCommandEvent
& WXUNUSED(event
)) 
 990     if ( !GetDDEServer() ) 
 994     wxConnectionBase 
*conn 
= client
.MakeConnection(wxEmptyString
, m_server
, m_topic
); 
 997         wxLogError(_T("Failed to connect to the DDE server '%s'."), 
1002         if ( !conn
->Request(m_cmdDde
) ) 
1004             wxLogError(_T("Failed to  send request '%s' via DDE."), 
1009             wxLogStatus(_T("Successfully sent DDE request.")); 
1014 #endif // __WINDOWS__ 
1016 // ---------------------------------------------------------------------------- 
1018 // ---------------------------------------------------------------------------- 
1021 void MyFrame::OnIdle(wxIdleEvent
& event
) 
1023     size_t count 
= m_running
.GetCount(); 
1024     for ( size_t n 
= 0; n 
< count
; n
++ ) 
1026         if ( m_running
[n
]->HasInput() ) 
1028             event
.RequestMore(); 
1033 void MyFrame::OnTimer(wxTimerEvent
& WXUNUSED(event
)) 
1038 void MyFrame::OnProcessTerminated(MyPipedProcess 
*process
) 
1040     RemovePipedProcess(process
); 
1043 void MyFrame::OnAsyncTermination(MyProcess 
*process
) 
1045     m_allAsync
.Remove(process
); 
1050 void MyFrame::AddPipedProcess(MyPipedProcess 
*process
) 
1052     if ( m_running
.IsEmpty() ) 
1054         // we want to start getting the timer events to ensure that a 
1055         // steady stream of idle events comes in -- otherwise we 
1056         // wouldn't be able to poll the child process input 
1057         m_timerIdleWakeUp
.Start(100); 
1059     //else: the timer is already running 
1061     m_running
.Add(process
); 
1062     m_allAsync
.Add(process
); 
1065 void MyFrame::RemovePipedProcess(MyPipedProcess 
*process
) 
1067     m_running
.Remove(process
); 
1069     if ( m_running
.IsEmpty() ) 
1071         // we don't need to get idle events all the time any more 
1072         m_timerIdleWakeUp
.Stop(); 
1076 void MyFrame::ShowOutput(const wxString
& cmd
, 
1077                          const wxArrayString
& output
, 
1078                          const wxString
& title
) 
1080     size_t count 
= output
.GetCount(); 
1084     m_lbox
->Append(wxString::Format(_T("--- %s of '%s' ---"), 
1085                                     title
.c_str(), cmd
.c_str())); 
1087     for ( size_t n 
= 0; n 
< count
; n
++ ) 
1089         m_lbox
->Append(output
[n
]); 
1092     m_lbox
->Append(wxString::Format(_T("--- End of %s ---"), 
1093                                     title
.Lower().c_str())); 
1096 // ---------------------------------------------------------------------------- 
1098 // ---------------------------------------------------------------------------- 
1100 void MyProcess::OnTerminate(int pid
, int status
) 
1102     wxLogStatus(m_parent
, _T("Process %u ('%s') terminated with exit code %d."), 
1103                 pid
, m_cmd
.c_str(), status
); 
1105     m_parent
->OnAsyncTermination(this); 
1108 // ---------------------------------------------------------------------------- 
1110 // ---------------------------------------------------------------------------- 
1112 bool MyPipedProcess::HasInput() 
1114     bool hasInput 
= false; 
1116     if ( IsInputAvailable() ) 
1118         wxTextInputStream 
tis(*GetInputStream()); 
1120         // this assumes that the output is always line buffered 
1122         msg 
<< m_cmd 
<< _T(" (stdout): ") << tis
.ReadLine(); 
1124         m_parent
->GetLogListBox()->Append(msg
); 
1129     if ( IsErrorAvailable() ) 
1131         wxTextInputStream 
tis(*GetErrorStream()); 
1133         // this assumes that the output is always line buffered 
1135         msg 
<< m_cmd 
<< _T(" (stderr): ") << tis
.ReadLine(); 
1137         m_parent
->GetLogListBox()->Append(msg
); 
1145 void MyPipedProcess::OnTerminate(int pid
, int status
) 
1147     // show the rest of the output 
1148     while ( HasInput() ) 
1151     m_parent
->OnProcessTerminated(this); 
1153     MyProcess::OnTerminate(pid
, status
); 
1156 // ---------------------------------------------------------------------------- 
1158 // ---------------------------------------------------------------------------- 
1160 bool MyPipedProcess2::HasInput() 
1162     if ( !m_input
.empty() ) 
1164         wxTextOutputStream 
os(*GetOutputStream()); 
1165         os
.WriteString(m_input
); 
1170         // call us once again - may be we'll have output 
1174     return MyPipedProcess::HasInput(); 
1177 // ============================================================================ 
1178 // MyPipeFrame implementation 
1179 // ============================================================================ 
1181 MyPipeFrame::MyPipeFrame(wxFrame 
*parent
, 
1182                          const wxString
& cmd
, 
1184            : wxFrame(parent
, wxID_ANY
, cmd
), 
1186              // in a real program we'd check that the streams are !NULL here 
1187              m_out(*process
->GetOutputStream()), 
1188              m_in(*process
->GetInputStream()), 
1189              m_err(*process
->GetErrorStream()) 
1191     m_process
->SetNextHandler(this); 
1193     wxPanel 
*panel 
= new wxPanel(this, wxID_ANY
); 
1195     m_textOut 
= new wxTextCtrl(panel
, wxID_ANY
, wxEmptyString
, 
1196                               wxDefaultPosition
, wxDefaultSize
, 
1197                               wxTE_PROCESS_ENTER
); 
1198     m_textIn 
= new wxTextCtrl(panel
, wxID_ANY
, wxEmptyString
, 
1199                                wxDefaultPosition
, wxDefaultSize
, 
1200                                wxTE_MULTILINE 
| wxTE_RICH
); 
1201     m_textIn
->SetEditable(false); 
1202     m_textErr 
= new wxTextCtrl(panel
, wxID_ANY
, wxEmptyString
, 
1203                                wxDefaultPosition
, wxDefaultSize
, 
1204                                wxTE_MULTILINE 
| wxTE_RICH
); 
1205     m_textErr
->SetEditable(false); 
1207     wxSizer 
*sizerTop 
= new wxBoxSizer(wxVERTICAL
); 
1208     sizerTop
->Add(m_textOut
, 0, wxGROW 
| wxALL
, 5); 
1210     wxSizer 
*sizerBtns 
= new wxBoxSizer(wxHORIZONTAL
); 
1212         Add(new wxButton(panel
, Exec_Btn_Send
, _T("&Send")), 0, wxALL
, 5); 
1214         Add(new wxButton(panel
, Exec_Btn_SendFile
, _T("&File...")), 0, wxALL
, 5); 
1216         Add(new wxButton(panel
, Exec_Btn_Get
, _T("&Get")), 0, wxALL
, 5); 
1218         Add(new wxButton(panel
, Exec_Btn_Close
, _T("&Close")), 0, wxALL
, 5); 
1220     sizerTop
->Add(sizerBtns
, 0, wxCENTRE 
| wxALL
, 5); 
1221     sizerTop
->Add(m_textIn
, 1, wxGROW 
| wxALL
, 5); 
1222     sizerTop
->Add(m_textErr
, 1, wxGROW 
| wxALL
, 5); 
1224     panel
->SetSizer(sizerTop
); 
1225     sizerTop
->Fit(this); 
1230 void MyPipeFrame::OnBtnSendFile(wxCommandEvent
& WXUNUSED(event
)) 
1233     wxFileDialog 
filedlg(this, _T("Select file to send")); 
1234     if ( filedlg
.ShowModal() != wxID_OK 
) 
1237     wxFFile 
file(filedlg
.GetFilename(), _T("r")); 
1239     if ( !file
.IsOpened() || !file
.ReadAll(&data
) ) 
1242     // can't write the entire string at once, this risk overflowing the pipe 
1243     // and we would dead lock 
1244     size_t len 
= data
.length(); 
1245     const wxChar 
*pc 
= data
.c_str(); 
1248         const size_t CHUNK_SIZE 
= 4096; 
1249         m_out
.Write(pc
, len 
> CHUNK_SIZE 
? CHUNK_SIZE 
: len
); 
1251         // note that not all data could have been written as we don't block on 
1252         // the write end of the pipe 
1253         const size_t lenChunk 
= m_out
.LastWrite(); 
1260 #endif // wxUSE_FILEDLG 
1263 void MyPipeFrame::DoGet() 
1265     // we don't have any way to be notified when any input appears on the 
1266     // stream so we have to poll it :-( 
1267     DoGetFromStream(m_textIn
, m_in
); 
1268     DoGetFromStream(m_textErr
, m_err
); 
1271 void MyPipeFrame::DoGetFromStream(wxTextCtrl 
*text
, wxInputStream
& in
) 
1273     while ( in
.CanRead() ) 
1275         wxChar buffer
[4096]; 
1276         buffer
[in
.Read(buffer
, WXSIZEOF(buffer
) - 1).LastRead()] = _T('\0'); 
1278         text
->AppendText(buffer
); 
1282 void MyPipeFrame::DoClose() 
1284     m_process
->CloseOutput(); 
1289 void MyPipeFrame::DisableInput() 
1291     m_textOut
->SetEditable(false); 
1292     FindWindow(Exec_Btn_Send
)->Disable(); 
1293     FindWindow(Exec_Btn_SendFile
)->Disable(); 
1294     FindWindow(Exec_Btn_Close
)->Disable(); 
1297 void MyPipeFrame::DisableOutput() 
1299     FindWindow(Exec_Btn_Get
)->Disable(); 
1302 void MyPipeFrame::OnClose(wxCloseEvent
& event
) 
1306         // we're not interested in getting the process termination notification 
1307         // if we are closing it ourselves 
1308         wxProcess 
*process 
= m_process
; 
1310         process
->SetNextHandler(NULL
); 
1312         process
->CloseOutput(); 
1318 void MyPipeFrame::OnProcessTerm(wxProcessEvent
& WXUNUSED(event
)) 
1325     wxLogWarning(_T("The other process has terminated, closing"));