// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2008 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . #if defined(__GNUG__) && !defined(__APPLE__) #pragma implementation "BOINCBaseFrame.h" #endif #include "stdwx.h" #include "diagnostics.h" #include "util.h" #include "mfile.h" #include "miofile.h" #include "parse.h" #include "error_numbers.h" #include "BOINCGUIApp.h" #include "SkinManager.h" #include "MainDocument.h" #include "BOINCClientManager.h" #include "BOINCTaskBar.h" #include "BOINCBaseFrame.h" #include "BOINCDialupManager.h" #include "Events.h" #include "DlgEventLog.h" #include "DlgSelectComputer.h" #include "BOINCInternetFSHandler.h" DEFINE_EVENT_TYPE(wxEVT_FRAME_ALERT) DEFINE_EVENT_TYPE(wxEVT_FRAME_CONNECT) DEFINE_EVENT_TYPE(wxEVT_FRAME_INITIALIZED) DEFINE_EVENT_TYPE(wxEVT_FRAME_REFRESHVIEW) DEFINE_EVENT_TYPE(wxEVT_FRAME_UPDATESTATUS) DEFINE_EVENT_TYPE(wxEVT_FRAME_RELOADSKIN) DEFINE_EVENT_TYPE(wxEVT_FRAME_NOTIFICATION) IMPLEMENT_DYNAMIC_CLASS(CBOINCBaseFrame, wxFrame) BEGIN_EVENT_TABLE (CBOINCBaseFrame, wxFrame) EVT_TIMER(ID_DOCUMENTPOLLTIMER, CBOINCBaseFrame::OnDocumentPoll) EVT_TIMER(ID_ALERTPOLLTIMER, CBOINCBaseFrame::OnAlertPoll) EVT_TIMER(ID_PERIODICRPCTIMER, CBOINCBaseFrame::OnPeriodicRPC) EVT_FRAME_INITIALIZED(CBOINCBaseFrame::OnInitialized) EVT_FRAME_ALERT(CBOINCBaseFrame::OnAlert) EVT_FRAME_REFRESH(CBOINCBaseFrame::OnRefreshView) EVT_CLOSE(CBOINCBaseFrame::OnClose) EVT_MENU(ID_CLOSEWINDOW, CBOINCBaseFrame::OnCloseWindow) EVT_MENU(wxID_EXIT, CBOINCBaseFrame::OnExit) END_EVENT_TABLE () CBOINCBaseFrame::CBOINCBaseFrame() { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::CBOINCBaseFrame - Default Constructor Function Begin")); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::CBOINCBaseFrame - Default Constructor Function End")); } CBOINCBaseFrame::CBOINCBaseFrame(wxWindow* parent, const wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, const long style) : wxFrame(parent, id, title, pos, size, style) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::CBOINCBaseFrame - Function Begin")); // Configuration Settings m_iSelectedLanguage = 0; m_iReminderFrequency = 0; m_strNetworkDialupConnectionName = wxEmptyString; m_aSelectedComputerMRU.Clear(); m_bShowConnectionFailedAlert = false; m_pDialupManager = new CBOINCDialUpManager(); wxASSERT(m_pDialupManager->IsOk()); m_pDocumentPollTimer = new wxTimer(this, ID_DOCUMENTPOLLTIMER); wxASSERT(m_pDocumentPollTimer); m_pDocumentPollTimer->Start(250); // Send event every 250 milliseconds m_pAlertPollTimer = new wxTimer(this, ID_ALERTPOLLTIMER); wxASSERT(m_pAlertPollTimer); m_pAlertPollTimer->Start(1000); // Send event every 1000 milliseconds m_pPeriodicRPCTimer = new wxTimer(this, ID_PERIODICRPCTIMER); wxASSERT(m_pPeriodicRPCTimer); m_pPeriodicRPCTimer->Start(1000); // Send event every 1000 milliseconds m_iFrameRefreshRate = 1000; // Refresh frame every 1000 milliseconds // Limit the number of times the UI can update itself to two times a second // NOTE: Linux and Mac were updating several times a second and eating // CPU time wxUpdateUIEvent::SetUpdateInterval(500); // The second half of the initialization process picks up in the OnFrameRender() // routine since the menus' and status bars' are drawn in the frameworks // on idle routines, on idle events are sent in between the end of the // constructor and the first call to OnFrameRender // // Look for the 'if (!bAlreadyRunOnce) {' statement wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::CBOINCBaseFrame - Function End")); } CBOINCBaseFrame::~CBOINCBaseFrame() { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::~CBOINCBaseFrame - Function Begin")); wxASSERT(m_pPeriodicRPCTimer); wxASSERT(m_pAlertPollTimer); wxASSERT(m_pDocumentPollTimer); if (m_pPeriodicRPCTimer) { m_pPeriodicRPCTimer->Stop(); delete m_pPeriodicRPCTimer; } if (m_pAlertPollTimer) { m_pAlertPollTimer->Stop(); delete m_pAlertPollTimer; } if (m_pDocumentPollTimer) { m_pDocumentPollTimer->Stop(); delete m_pDocumentPollTimer; } if (m_pDialupManager) { delete m_pDialupManager; } wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::~CBOINCBaseFrame - Function End")); } void CBOINCBaseFrame::OnPeriodicRPC(wxTimerEvent& WXUNUSED(event)) { static bool bAlreadyRunningLoop = false; CMainDocument* pDoc = wxGetApp().GetDocument(); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); if (!bAlreadyRunningLoop && m_pPeriodicRPCTimer->IsRunning()) { bAlreadyRunningLoop = true; pDoc->RunPeriodicRPCs(m_iFrameRefreshRate); bAlreadyRunningLoop = false; } } void CBOINCBaseFrame::OnDocumentPoll(wxTimerEvent& WXUNUSED(event)) { static bool bAlreadyRunOnce = false; CMainDocument* pDoc = wxGetApp().GetDocument(); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); // Timer events are handled while the RPC Wait dialog is shown // which may cause unintended recursion and repeatedly posting // the same RPC requests from timer routines. if (pDoc->WaitingForRPC()) return; if (!bAlreadyRunOnce && m_pDocumentPollTimer->IsRunning()) { // Complete any remaining initialization that has to happen after we are up // and running FireInitialize(); bAlreadyRunOnce = true; } pDoc->OnPoll(); } void CBOINCBaseFrame::OnAlertPoll(wxTimerEvent& WXUNUSED(event)) { static bool bAlreadyRunningLoop = false; CMainDocument* pDoc = wxGetApp().GetDocument(); if (!bAlreadyRunningLoop && m_pAlertPollTimer->IsRunning()) { bAlreadyRunningLoop = true; // Update idle detection if needed. wxGetApp().UpdateSystemIdleDetection(); // Check to see if there is anything that we need to do from the // dial up user perspective. if (pDoc && m_pDialupManager) { // Timer events are handled while the RPC Wait dialog is shown // which may cause unintended recursion and repeatedly posting // the same RPC requests from timer routines. if (pDoc->IsConnected() && !pDoc->WaitingForRPC()) { m_pDialupManager->OnPoll(); } } if (m_bShowConnectionFailedAlert && IsShown()) { m_bShowConnectionFailedAlert = false; ShowConnectionFailedAlert(); } bAlreadyRunningLoop = false; } } void CBOINCBaseFrame::OnInitialized(CFrameEvent& WXUNUSED(event)) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnInitialized - Function Begin")); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnInitialized - Function End")); } void CBOINCBaseFrame::OnRefreshView(CFrameEvent& ) { } void CBOINCBaseFrame::OnAlert(CFrameAlertEvent& event) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnAlert - Function Begin")); static bool bAlreadyRunningLoop = false; if (!bAlreadyRunningLoop) { bAlreadyRunningLoop = true; #ifdef __WXMSW__ CTaskBarIcon* pTaskbar = wxGetApp().GetTaskBarIcon(); wxASSERT(pTaskbar); if ((IsShown() && !event.m_notification_only) || (IsShown() && !pTaskbar->IsBalloonsSupported())) { if (!event.m_notification_only) { int retval = 0; if (!IsShown()) { Show(); } retval = wxGetApp().SafeMessageBox(event.m_message, event.m_title, event.m_style, this); if (event.m_alert_event_type == AlertProcessResponse) { event.ProcessResponse(retval); } } } else { // If the main window is hidden or minimzed use the system tray ballon // to notify the user instead. This keeps dialogs from interfering // with people typing email messages or any other activity where they // do not want keyboard focus changed to another window while typing. unsigned int icon_type; if (wxICON_ERROR & event.m_style) { icon_type = NIIF_ERROR; } else if (wxICON_WARNING & event.m_style) { icon_type = NIIF_WARNING; } else if (wxICON_INFORMATION & event.m_style) { icon_type = NIIF_INFO; } else { icon_type = NIIF_NONE; } pTaskbar->SetBalloon( pTaskbar->m_iconTaskBarNormal, event.m_title, event.m_message, icon_type ); } #elif defined (__WXMAC__) // SafeMessageBox() / ProcessResponse() hangs the Manager if hidden. // Currently, the only non-notification-only alert is Connection Failed, // which is now has logic to be displayed when Manager is maximized. // Notification only events on platforms other than Windows are // currently discarded. Otherwise the application would be restored // and input focus set on the notification which interrupts whatever // the user was doing. if (IsShown() && !event.m_notification_only) { int retval = 0; retval = wxGetApp().SafeMessageBox(event.m_message, event.m_title, event.m_style, this); if (event.m_alert_event_type == AlertProcessResponse) { event.ProcessResponse(retval); } } #endif bAlreadyRunningLoop = false; } wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnAlert - Function End")); } void CBOINCBaseFrame::OnClose(wxCloseEvent& event) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnClose - Function Begin")); if (!event.CanVeto() || IsIconized()) { wxGetApp().FrameClosed(); Destroy(); } else { #ifdef __WXGTK__ // Apparently aborting a close event just causes the main window to be displayed // again. Just minimize the window instead. Iconize(); #else Hide(); #endif } wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnClose - Function End")); } void CBOINCBaseFrame::OnCloseWindow(wxCommandEvent& WXUNUSED(event)) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnCloseWindow - Function Begin")); #ifdef __WXMAC__ CFStringRef frontWindowTitle, eventLogTitle; CDlgEventLog* eventLog = wxGetApp().GetEventLog(); if (eventLog) { WindowRef win = FrontNonFloatingWindow(); if (win) { CopyWindowTitleAsCFString(win, &frontWindowTitle); eventLogTitle = CFStringCreateWithCString(NULL, eventLog->GetTitle().char_str(), kCFStringEncodingUTF8); CFComparisonResult res = CFStringCompare(eventLogTitle, frontWindowTitle, 0); CFRelease(eventLogTitle); CFRelease(frontWindowTitle); if (res == kCFCompareEqualTo) { wxCloseEvent eventClose; eventLog->OnClose(eventClose); return; } } } #endif Close(); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::OnCloseWindow - Function End")); } void CBOINCBaseFrame::OnExit(wxCommandEvent& WXUNUSED(event)) { wxLogTrace(wxT("Function Start/End"), wxT("CAdvancedFrame::OnExit - Function Begin")); if (wxGetApp().ConfirmExit()) { wxFileSystemHandler *internetFSHandler = wxGetApp().GetInternetFSHandler(); if (internetFSHandler) { ((CBOINCInternetFSHandler*)internetFSHandler)->SetAbortInternetIO(); } // Save state before exiting SaveState(); // Under wxWidgets 2.8.0, the task bar icons must be deleted for app to exit its main loop #ifdef __WXMAC__ wxGetApp().DeleteMacSystemMenu(); #endif wxGetApp().DeleteTaskBarIcon(); CDlgEventLog* eventLog = wxGetApp().GetEventLog(); if (eventLog) { eventLog->Destroy(); } Close(true); } wxLogTrace(wxT("Function Start/End"), wxT("CAdvancedFrame::OnExit - Function End")); } int CBOINCBaseFrame::GetCurrentViewPage() { return _GetCurrentViewPage(); } void CBOINCBaseFrame::FireInitialize() { CFrameEvent event(wxEVT_FRAME_INITIALIZED, this); AddPendingEvent(event); } void CBOINCBaseFrame::FireRefreshView() { CMainDocument* pDoc = wxGetApp().GetDocument(); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); pDoc->RefreshRPCs(); pDoc->RunPeriodicRPCs(0); } void CBOINCBaseFrame::FireConnect() { CFrameEvent event(wxEVT_FRAME_CONNECT, this); AddPendingEvent(event); } void CBOINCBaseFrame::FireReloadSkin() { CFrameEvent event(wxEVT_FRAME_RELOADSKIN, this); AddPendingEvent(event); } void CBOINCBaseFrame::FireNotification() { CFrameEvent event(wxEVT_FRAME_NOTIFICATION, this); AddPendingEvent(event); } bool CBOINCBaseFrame::SelectComputer(wxString& hostName, int& portNum, wxString& password, bool required) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::SelectComputer - Function Begin")); CDlgSelectComputer dlg(this, required); size_t lIndex = 0; wxArrayString aComputerNames; bool bResult = false; // Lets copy the template store in the system state aComputerNames = m_aSelectedComputerMRU; // Lets populate the combo control with the MRU list dlg.m_ComputerNameCtrl->Clear(); for (lIndex = 0; lIndex < aComputerNames.Count(); lIndex++) { dlg.m_ComputerNameCtrl->Append(aComputerNames.Item(lIndex)); } if (wxID_OK == dlg.ShowModal()) { hostName = dlg.m_ComputerNameCtrl->GetValue(); // Make a null hostname be the same thing as localhost if (wxEmptyString == hostName) { hostName = wxT("localhost"); portNum = GUI_RPC_PORT; password = wxEmptyString; } else { // Parse the remote machine info wxString sHost = dlg.m_ComputerNameCtrl->GetValue(); long lPort = GUI_RPC_PORT; int iPos = sHost.Find(wxT(":")); if (iPos != wxNOT_FOUND) { wxString sPort = sHost.substr(iPos + 1); if (!sPort.ToLong(&lPort)) lPort = GUI_RPC_PORT; sHost.erase(iPos); } hostName = sHost; portNum = (int)lPort; password = dlg.m_ComputerPasswordCtrl->GetValue(); } // Insert a copy of the current combo box value to the head of the // computer names string array if (wxEmptyString != dlg.m_ComputerNameCtrl->GetValue()) { aComputerNames.Insert(dlg.m_ComputerNameCtrl->GetValue(), 0); } // Loops through the computer names and remove any duplicates that // might exist with the new head value for (lIndex = 1; lIndex < aComputerNames.Count(); lIndex++) { if (aComputerNames.Item(lIndex) == aComputerNames.Item(0)) aComputerNames.RemoveAt(lIndex); } // Store the modified computer name MRU list back to the system state m_aSelectedComputerMRU = aComputerNames; bResult = true; } else { bResult = false; // User cancelled } wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::SelectComputer - Function End")); return bResult; } void CBOINCBaseFrame::ShowConnectionBadPasswordAlert( bool bUsedDefaultPassword, int iReadGUIRPCAuthFailure ) { CSkinAdvanced* pSkinAdvanced = wxGetApp().GetSkinManager()->GetAdvanced(); wxString strDialogTitle = wxEmptyString; wxASSERT(pSkinAdvanced); wxASSERT(wxDynamicCast(pSkinAdvanced, CSkinAdvanced)); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowConnectionBadPasswordAlert - Function Begin")); // %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' strDialogTitle.Printf( _("%s - Connection Error"), pSkinAdvanced->GetApplicationName().c_str() ); if ( bUsedDefaultPassword ) { #ifdef __WXMSW__ if ( EACCES == iReadGUIRPCAuthFailure || ENOENT == iReadGUIRPCAuthFailure ) { ShowAlert( strDialogTitle, _("You currently are not authorized to manage the client.\nPlease contact your administrator to add you to the 'boinc_users' local user group."), wxOK | wxICON_ERROR ); } else #endif { ShowAlert( strDialogTitle, #ifndef __WXMAC__ _("Authorization failed connecting to running client.\nMake sure you start this program in the same directory as the client."), #else _("Authorization failed connecting to running client."), #endif wxOK | wxICON_ERROR ); } } else { ShowAlert( strDialogTitle, _("The password you have provided is incorrect, please try again."), wxOK | wxICON_ERROR ); } wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowConnectionBadPasswordAlert - Function End")); } void CBOINCBaseFrame::ShowConnectionFailedAlert() { CSkinAdvanced* pSkinAdvanced = wxGetApp().GetSkinManager()->GetAdvanced(); CMainDocument* pDoc = wxGetApp().GetDocument(); wxString strConnectedCompter = wxEmptyString; wxString strDialogTitle = wxEmptyString; wxString strDialogMessage = wxEmptyString; wxASSERT(pSkinAdvanced); wxASSERT(wxDynamicCast(pSkinAdvanced, CSkinAdvanced)); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowConnectionFailedAlert - Function Begin")); // Did BOINC crash on local computer? If so restart it and reconnect. pDoc->GetConnectedComputerName(strConnectedCompter); if (pDoc->IsComputerNameLocal(strConnectedCompter)) { if (pDoc->m_pClientManager->AutoRestart()) { boinc_sleep(0.5); // Allow time for Client to restart if (pDoc->m_pClientManager->IsBOINCCoreRunning()) { pDoc->Reconnect(); return; } } else { // Don't ask whether to reconnect to local client if it is not running if (!pDoc->m_pClientManager->IsBOINCCoreRunning()) { return; } } } // %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' strDialogTitle.Printf( _("%s - Connection Failed"), pSkinAdvanced->GetApplicationName().c_str() ); // 1st %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' // 2st %s is the project name // i.e. 'BOINC', 'GridRepublic' strDialogMessage.Printf( _("%s is not able to connect to a %s client.\nWould you like to try to connect again?"), pSkinAdvanced->GetApplicationName().c_str(), pSkinAdvanced->GetApplicationShortName().c_str() ); ShowAlert( strDialogTitle, strDialogMessage, wxYES_NO | wxICON_QUESTION, false, AlertProcessResponse ); // If we are minimized, set flag to show alert when maximized m_bShowConnectionFailedAlert = !IsShown(); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowConnectionFailedAlert - Function End")); } void CBOINCBaseFrame::ShowDaemonStartFailedAlert() { CSkinAdvanced* pSkinAdvanced = wxGetApp().GetSkinManager()->GetAdvanced(); wxString strDialogTitle = wxEmptyString; wxString strDialogMessage = wxEmptyString; wxASSERT(pSkinAdvanced); wxASSERT(wxDynamicCast(pSkinAdvanced, CSkinAdvanced)); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowDaemonStartFailedAlert - Function Begin")); // %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' strDialogTitle.Printf( _("%s - Daemon Start Failed"), pSkinAdvanced->GetApplicationName().c_str() ); // 1st %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' // 2st %s is the project name // i.e. 'BOINC', 'GridRepublic' #ifdef __WXMSW__ strDialogMessage.Printf( _("%s is not able to start a %s client.\nPlease launch the Control Panel->Administative Tools->Services applet and start the BOINC service."), pSkinAdvanced->GetApplicationName().c_str(), pSkinAdvanced->GetApplicationShortName().c_str() ); #else strDialogMessage.Printf( _("%s is not able to start a %s client.\nPlease start the daemon and try again."), pSkinAdvanced->GetApplicationName().c_str(), pSkinAdvanced->GetApplicationShortName().c_str() ); #endif ShowAlert( strDialogTitle, strDialogMessage, wxOK | wxICON_ERROR ); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowDaemonStartFailedAlert - Function End")); } void CBOINCBaseFrame::ShowNotCurrentlyConnectedAlert() { CSkinAdvanced* pSkinAdvanced = wxGetApp().GetSkinManager()->GetAdvanced(); CMainDocument* pDoc = wxGetApp().GetDocument(); wxString strConnectedCompter = wxEmptyString; wxString strDialogTitle = wxEmptyString; wxString strDialogMessage = wxEmptyString; wxASSERT(pSkinAdvanced); wxASSERT(wxDynamicCast(pSkinAdvanced, CSkinAdvanced)); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowNotCurrentlyConnectedAlert - Function Begin")); // Did BOINC crash on local computer? If so restart it and reconnect. pDoc->GetConnectedComputerName(strConnectedCompter); if (pDoc->IsComputerNameLocal(strConnectedCompter)) { if (pDoc->m_pClientManager->AutoRestart()) { boinc_sleep(0.5); // Allow time for Client to restart if (pDoc->m_pClientManager->IsBOINCCoreRunning()) { pDoc->Reconnect(); return; } } else { // Don't ask whether to reconnect to local client if it is not running if (!pDoc->m_pClientManager->IsBOINCCoreRunning()) { return; } } } // %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' strDialogTitle.Printf( _("%s - Connection Status"), pSkinAdvanced->GetApplicationName().c_str() ); // 1st %s is the application name // i.e. 'BOINC Manager', 'GridRepublic Manager' // 2nd %s is the project name // i.e. 'BOINC', 'GridRepublic' // 3nd %s is the project name // i.e. 'BOINC', 'GridRepublic' strDialogMessage.Printf( _("%s is not currently connected to a %s client.\nPlease use the 'Advanced\\Select Computer...' menu option to connect up to a %s client.\nTo connect up to your local computer please use 'localhost' as the host name."), pSkinAdvanced->GetApplicationName().c_str(), pSkinAdvanced->GetApplicationShortName().c_str(), pSkinAdvanced->GetApplicationShortName().c_str() ); ShowAlert( strDialogTitle, strDialogMessage, wxOK | wxICON_ERROR ); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::ShowNotCurrentlyConnectedAlert - Function End")); } void CBOINCBaseFrame::StartTimers() { wxASSERT(m_pAlertPollTimer); wxASSERT(m_pPeriodicRPCTimer); wxASSERT(m_pDocumentPollTimer); m_pAlertPollTimer->Start(); m_pPeriodicRPCTimer->Start(); m_pDocumentPollTimer->Start(); } void CBOINCBaseFrame::StopTimers() { wxASSERT(m_pAlertPollTimer); wxASSERT(m_pPeriodicRPCTimer); wxASSERT(m_pDocumentPollTimer); m_pAlertPollTimer->Stop(); m_pPeriodicRPCTimer->Stop(); m_pDocumentPollTimer->Stop(); } void CBOINCBaseFrame::UpdateRefreshTimerInterval() { } #if 0 void CBOINCBaseFrame::UpdateStatusText(const wxChar* szStatus) { CFrameEvent event(wxEVT_FRAME_UPDATESTATUS, this, szStatus); ProcessEvent(event); } #endif void CBOINCBaseFrame::ShowAlert( const wxString title, const wxString message, const int style, const bool notification_only, const FrameAlertEventType alert_event_type ) { CFrameAlertEvent event(wxEVT_FRAME_ALERT, this, title, message, style, notification_only, alert_event_type); AddPendingEvent(event); } bool CBOINCBaseFrame::SaveState() { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::SaveState - Function Begin")); wxString strBaseConfigLocation = wxString(wxT("/")); wxConfigBase* pConfig = wxConfigBase::Get(FALSE); wxString strConfigLocation; wxString strPreviousLocation; wxString strBuffer; int iIndex; int iItemCount; wxASSERT(pConfig); // An odd case happens every once and awhile where wxWidgets looses // the pointer to the config object, or it is cleaned up before // the window has finished it's cleanup duty. If we detect a NULL // pointer, return false. if (!pConfig) return false; // // Save Frame State // pConfig->SetPath(strBaseConfigLocation); pConfig->Write(wxT("Language"), m_iSelectedLanguage); pConfig->Write(wxT("ReminderFrequencyV3"), m_iReminderFrequency); pConfig->Write(wxT("NetworkDialupConnectionName"), m_strNetworkDialupConnectionName); // // Save Computer MRU list // strPreviousLocation = pConfig->GetPath(); strConfigLocation = strPreviousLocation + wxT("ComputerMRU"); pConfig->SetPath(strConfigLocation); iItemCount = (int)m_aSelectedComputerMRU.GetCount() - 1; for (iIndex = 0; iIndex <= iItemCount; iIndex++) { strBuffer.Printf(wxT("%d"), iIndex); pConfig->Write( strBuffer, m_aSelectedComputerMRU.Item(iIndex) ); } pConfig->SetPath(strPreviousLocation); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::SaveState - Function End")); return true; } bool CBOINCBaseFrame::RestoreState() { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::RestoreState - Function Begin")); wxString strBaseConfigLocation = wxString(wxT("/")); wxConfigBase* pConfig = wxConfigBase::Get(FALSE); wxString strConfigLocation; wxString strPreviousLocation; wxString strBuffer; wxString strValue; long iIndex; bool bKeepEnumerating = false; wxASSERT(pConfig); // An odd case happens every once and awhile where wxWidgets looses // the pointer to the config object, or it is cleaned up before // the window has finished it's cleanup duty. If we detect a NULL // pointer, return false. if (!pConfig) return false; // // Restore Frame State // pConfig->SetPath(strBaseConfigLocation); pConfig->Read(wxT("Language"), &m_iSelectedLanguage, 0L); pConfig->Read(wxT("ReminderFrequencyV3"), &m_iReminderFrequency, 360L); pConfig->Read(wxT("NetworkDialupConnectionName"), &m_strNetworkDialupConnectionName, wxEmptyString); // // Restore Computer MRU list // strPreviousLocation = pConfig->GetPath(); strConfigLocation = strPreviousLocation + wxT("ComputerMRU"); pConfig->SetPath(strConfigLocation); m_aSelectedComputerMRU.Clear(); bKeepEnumerating = pConfig->GetFirstEntry(strBuffer, iIndex); while (bKeepEnumerating) { pConfig->Read(strBuffer, &strValue); m_aSelectedComputerMRU.Add(strValue); bKeepEnumerating = pConfig->GetNextEntry(strBuffer, iIndex); } pConfig->SetPath(strPreviousLocation); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::RestoreState - Function End")); return true; } bool CBOINCBaseFrame::Show(bool bShow) { wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::Show - Function Begin")); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::Show - Show: '%d'"), (int)bShow); bool retval; if (bShow) { wxGetApp().ShowApplication(true); } else { if ( this == wxGetApp().GetFrame() ) { if (wxGetApp().IsApplicationVisible()) { wxGetApp().ShowApplication(false); } } } CDlgEventLog* pEventLog = wxGetApp().GetEventLog(); if (pEventLog) { #ifdef __WXMAC__ if (bShow) { pEventLog->Show(bShow); } #else pEventLog->Show(bShow); #endif } retval = wxFrame::Show(bShow); wxFrame::Raise(); wxLogTrace(wxT("Function Start/End"), wxT("CBOINCBaseFrame::Show - Function End")); return retval; } int CBOINCBaseFrame::_GetCurrentViewPage() { wxASSERT(false); return 0; } void CFrameAlertEvent::ProcessResponse(const int response) const { CMainDocument* pDoc = wxGetApp().GetDocument(); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); if ((AlertProcessResponse == m_alert_event_type) && (wxYES == response)) { pDoc->Reconnect(); } }