// Berkeley Open Infrastructure for Network Computing // http://boinc.berkeley.edu // Copyright (C) 2005 University of California // // This 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 2.1 of the License, or (at your option) any later version. // // This software 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. // // To view the GNU Lesser General Public License visit // http://www.gnu.org/copyleft/lesser.html // or write to the Free Software Foundation, Inc., // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA #if defined(__GNUG__) && !defined(__APPLE__) #pragma implementation "ViewWorkGrid.h" #endif #include "stdwx.h" #include "util.h" #include "BOINCGUIApp.h" #include "BOINCBaseFrame.h" #include "MainDocument.h" #include "AdvancedFrame.h" #include "BOINCTaskCtrl.h" #include "ViewWorkGrid.h" #include "Events.h" #include "error_numbers.h" #include "res/result.xpm" using std::string; #define COLUMN_PROJECT 0 #define COLUMN_APPLICATION 1 #define COLUMN_NAME 2 #define COLUMN_CPUTIME 3 #define COLUMN_PROGRESS 4 #define COLUMN_TOCOMPLETION 5 #define COLUMN_REPORTDEADLINE 6 #define COLUMN_STATUS 7 #define COLUMN_HIDDEN_URL 8 #define COLUMN_RESULTS_INDEX 9 #define NUM_COLUMNS (COLUMN_RESULTS_INDEX+1) // groups that contain buttons #define GRP_TASKS 0 #define GRP_WEBSITES 1 // buttons in the "tasks" area #define BTN_GRAPHICS 0 #define BTN_SUSPEND 1 #define BTN_ABORT 2 IMPLEMENT_DYNAMIC_CLASS(CViewWorkGrid, CBOINCBaseView) BEGIN_EVENT_TABLE (CViewWorkGrid, CBOINCBaseView) EVT_BUTTON(ID_TASK_WORK_SUSPEND, CViewWorkGrid::OnWorkSuspend) EVT_BUTTON(ID_TASK_WORK_SHOWGRAPHICS, CViewWorkGrid::OnWorkShowGraphics) EVT_BUTTON(ID_TASK_WORK_ABORT, CViewWorkGrid::OnWorkAbort) EVT_CUSTOM_RANGE(wxEVT_COMMAND_BUTTON_CLICKED, ID_TASK_PROJECT_WEB_PROJDEF_MIN, ID_TASK_PROJECT_WEB_PROJDEF_MAX, CViewWorkGrid::OnProjectWebsiteClicked) EVT_GRID_SELECT_CELL(CViewWorkGrid::OnGridSelectCell) EVT_GRID_RANGE_SELECT(CViewWorkGrid::OnGridSelectRange) #if PREVENT_MULTIPLE_TASK_SELECTIONS EVT_GRID_CELL_LEFT_CLICK(CViewWorkGrid::OnCellLeftClick) #endif END_EVENT_TABLE () CViewWorkGrid::CViewWorkGrid() { } CViewWorkGrid::CViewWorkGrid(wxNotebook* pNotebook) : CBOINCBaseView(pNotebook) { // // Setup View // wxFlexGridSizer* itemFlexGridSizer = new wxFlexGridSizer(2, 0, 0); wxASSERT(itemFlexGridSizer); itemFlexGridSizer->AddGrowableRow(0); itemFlexGridSizer->AddGrowableCol(1); m_pTaskPane = new CBOINCTaskCtrl(this, ID_TASK_WORKGRIDVIEW, DEFAULT_TASK_FLAGS); wxASSERT(m_pTaskPane); m_pGridPane = new CBOINCGridCtrl(this, ID_LIST_WORKGRIDVIEW); wxASSERT(m_pGridPane); itemFlexGridSizer->Add(m_pTaskPane, 1, wxGROW|wxALL, 1); itemFlexGridSizer->Add(m_pGridPane, 1, wxGROW|wxALL, 1); SetSizer(itemFlexGridSizer); Layout(); // Setup TaskPane CTaskItemGroup* pGroup = NULL; CTaskItem* pItem = NULL; pGroup = new CTaskItemGroup( _("Commands") ); m_TaskGroups.push_back( pGroup ); pItem = new CTaskItem( _("Show graphics"), _("Show application graphics in a window."), ID_TASK_WORK_SHOWGRAPHICS ); pGroup->m_Tasks.push_back( pItem ); pItem = new CTaskItem( _("Suspend"), _("Suspend work for this result."), ID_TASK_WORK_SUSPEND ); pGroup->m_Tasks.push_back( pItem ); pItem = new CTaskItem( _("Abort"), _("Abandon work on the result. " "You will get no credit for it."), ID_TASK_WORK_ABORT ); pGroup->m_Tasks.push_back( pItem ); // Create Task Pane Items m_pTaskPane->UpdateControls(); // Create Grid (one dummy row is needed) m_pGridPane->Setup(); m_pGridPane->SetTable(new CBOINCGridTable(1,NUM_COLUMNS)); //don't use wxGrid->Create() here !!!! m_pGridPane->SetSelectionMode(wxGrid::wxGridSelectRows); // init grid columns wxInt32 colSizes[] = {125,95,285,80,60,100,150,135,0,0}; wxString colTitles[] = {_("Project"),_("Application"),_("Name"),_("CPU time"), _("Progress"),_("To completion"),_("Report deadline"), _("Status"),wxEmptyString,wxEmptyString }; for(int i=0; i< NUM_COLUMNS;i++){ m_pGridPane->SetColLabelValue(i,colTitles[i]); m_pGridPane->SetColSize(i,colSizes[i]); } // set alignment for cpu time column to right m_pGridPane->SetColAlignment(COLUMN_CPUTIME,wxALIGN_RIGHT,wxALIGN_CENTER); // set alignment for completion column to right m_pGridPane->SetColAlignment(COLUMN_TOCOMPLETION,wxALIGN_RIGHT,wxALIGN_CENTER); //change the default cell renderer m_pGridPane->SetDefaultRenderer(new CBOINCGridCellProgressRenderer(COLUMN_PROGRESS)); //set column sort types m_pGridPane->SetColumnSortType(COLUMN_PROGRESS,CST_FLOAT); m_pGridPane->SetColumnSortType(COLUMN_CPUTIME,CST_TIME); m_pGridPane->SetColumnSortType(COLUMN_TOCOMPLETION,CST_TIME); m_pGridPane->SetColumnSortType(COLUMN_REPORTDEADLINE,CST_DATETIME); m_pGridPane->SetColumnSortType(COLUMN_RESULTS_INDEX,CST_LONG); //set primary key column index m_pGridPane->SetPrimaryKeyColumns(COLUMN_NAME,COLUMN_HIDDEN_URL); // Hide the Index and URL columns int min_width = m_pGridPane->GetColMinimalAcceptableWidth(); m_pGridPane->SetColMinimalAcceptableWidth(0); m_pGridPane->SetColSize(COLUMN_HIDDEN_URL,0); m_pGridPane->SetColSize(COLUMN_RESULTS_INDEX,0); m_pGridPane->SetColMinimalAcceptableWidth(min_width); UpdateSelection(); } CViewWorkGrid::~CViewWorkGrid() { EmptyTasks(); } wxString& CViewWorkGrid::GetViewName() { static wxString strViewName(_("TasksGrid")); return strViewName; } wxString& CViewWorkGrid::GetViewDisplayName() { static wxString strViewName(_("Tasks")); return strViewName; } const char** CViewWorkGrid::GetViewIcon() { return result_xpm; } void CViewWorkGrid::OnWorkSuspend( wxCommandEvent& WXUNUSED(event) ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkSuspend - Function Begin")); CMainDocument* pDoc = wxGetApp().GetDocument(); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); int i, n; wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pGridPane); wxArrayInt arrSelRows = m_pGridPane->GetSelectedRows2(); n = (int)arrSelRows.GetCount(); for(i=0; iGetCellValue(arrSelRows[i],COLUMN_NAME).Trim(false); wxString projectURL = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_HIDDEN_URL).Trim(false); RESULT* result = pDoc->result(resultName, projectURL); if (result->suspended_via_gui) { pFrame->UpdateStatusText(_("Resuming task...")); pDoc->WorkResume(result->project_url, result->name); } else { pFrame->UpdateStatusText(_("Suspending task...")); pDoc->WorkSuspend(result->project_url, result->name); } } pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkSuspend - Function End")); } void CViewWorkGrid::OnWorkShowGraphics( wxCommandEvent& WXUNUSED(event) ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkShowGraphics - Function Begin")); wxInt32 iAnswer = 0; wxString strMachineName = wxEmptyString; CMainDocument* pDoc = wxGetApp().GetDocument(); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); int i, n; wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pGridPane); pFrame->UpdateStatusText(_("Showing graphics for task...")); // TODO: implement hide as well as show #if (defined(_WIN32) || defined(__WXMAC__)) pDoc->GetConnectedComputerName(strMachineName); if (!pDoc->IsComputerNameLocal(strMachineName)) { iAnswer = ::wxMessageBox( _("Are you sure you want to display graphics on a remote machine?"), _("Show graphics"), wxYES_NO | wxICON_QUESTION, this ); } else { iAnswer = wxYES; } #else iAnswer = wxYES; #endif if (wxYES == iAnswer) { wxArrayInt arrSelRows = m_pGridPane->GetSelectedRows2(); n = (int)arrSelRows.GetCount(); for(i=0; iGetCellValue(arrSelRows[i],COLUMN_NAME).Trim(false); wxString projectURL = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_HIDDEN_URL).Trim(false); RESULT* result = pDoc->result(resultName, projectURL); pDoc->WorkShowGraphics(result); } } pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkShowGraphics - Function End")); } void CViewWorkGrid::OnWorkAbort( wxCommandEvent& WXUNUSED(event) ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkAbort - Function Begin")); wxInt32 iAnswer = 0; //wxInt32 iResult = 0; wxString strMessage = wxEmptyString; wxString strName = wxEmptyString; wxString strProgress = wxEmptyString; wxString strStatus = wxEmptyString; CMainDocument* pDoc = wxGetApp().GetDocument(); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); int i, n; wxASSERT(pDoc); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); if (!pDoc->IsUserAuthorized()) return; pFrame->UpdateStatusText(_("Aborting result...")); wxArrayInt arrSelRows = m_pGridPane->GetSelectedRows2(); n = (int)arrSelRows.GetCount(); for(i=0; iGetCellValue(arrSelRows[i],COLUMN_NAME).Trim(false); strProgress = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_PROGRESS).Trim(false); strStatus = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_STATUS).Trim(false); //FormatName(iResult, strName); //FormatProgress(iResult, strProgress); //FormatStatus(iResult, strStatus); strMessage.Printf( _("Are you sure you want to abort this task '%s'?\n" "(Progress: %s %%, Status: %s)"), strName.c_str(), strProgress.c_str(), strStatus.c_str() ); iAnswer = ::wxMessageBox( strMessage, _("Abort task"), wxYES_NO | wxICON_QUESTION, this ); if (wxYES == iAnswer) { wxString projectURL = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_HIDDEN_URL).Trim(false); std::string resultNameStr = (const char*)strName.mb_str(); std::string projectURLStr = (const char*)projectURL.mb_str(); pDoc->WorkAbort(projectURLStr, resultNameStr); } } pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnWorkAbort - Function End")); } void CViewWorkGrid::OnProjectWebsiteClicked( wxEvent& event ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnProjectWebsiteClicked - Function Begin")); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); pFrame->UpdateStatusText(_("Launching browser...")); int website_task_index = event.GetId() - ID_TASK_PROJECT_WEB_PROJDEF_MIN; pFrame->ExecuteBrowserLink( m_TaskGroups[1]->m_Tasks[website_task_index]->m_strWebSiteLink ); pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewWorkGrid::OnProjectWebsiteClicked - Function End")); } #if PREVENT_MULTIPLE_TASK_SELECTIONS void CViewWorkGrid::OnGridSelectRange( wxGridRangeSelectEvent& event ) { // Disallow multiple selections if (m_pGridPane->GetSelectedRows2().size() > 1) { int theRow = event.GetBottomRow(); m_pGridPane->ClearSelection(); m_pGridPane->SelectRow(theRow); } CBOINCBaseView::OnGridSelectRange(event); } void CViewWorkGrid::OnCellLeftClick( wxGridEvent& event ) { // Disallow multiple selections int theRow = event.GetRow(); m_pGridPane->ClearSelection(); m_pGridPane->SelectRow(theRow); } #endif void CViewWorkGrid::UpdateSelection() { CTaskItemGroup* pGroup = NULL; RESULT* result = NULL; PROJECT* project = NULL; CC_STATUS status; CMainDocument* pDoc = wxGetApp().GetDocument(); int i, n; bool wasSuspended=false, all_same_project=false; string first_project_url; wxASSERT(NULL != pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(NULL != m_pTaskPane); CBOINCBaseView::PreUpdateSelection(); pGroup = m_TaskGroups[0]; wxArrayInt arrSelRows = m_pGridPane->GetSelectedRows2(); n = (int)arrSelRows.GetCount(); if (n > 0) { m_pTaskPane->EnableTaskGroupTasks(pGroup); pDoc->GetCoreClientStatus(status); if (status.task_suspend_reason & ~(SUSPEND_REASON_CPU_USAGE_LIMIT)) { m_pTaskPane->DisableTask(pGroup->m_Tasks[BTN_GRAPHICS]); } } else { m_pTaskPane->DisableTaskGroupTasks(pGroup); } for(i=0; iGetCellValue(arrSelRows[i],COLUMN_NAME).Trim(false); wxString projectURL = m_pGridPane->GetCellValue(arrSelRows[i],COLUMN_HIDDEN_URL).Trim(false); result = pDoc->result(resultName, projectURL); if (result) { if (i == 0) { wasSuspended = result->suspended_via_gui; if (result->suspended_via_gui) { m_pTaskPane->UpdateTask( pGroup->m_Tasks[BTN_SUSPEND], _("Resume"), _("Resume work for this task.") ); } else { m_pTaskPane->UpdateTask( pGroup->m_Tasks[BTN_SUSPEND], _("Suspend"), _("Suspend work for this task.") ); } } else { if (wasSuspended != result->suspended_via_gui) { // Disable Suspend / Resume button if the multiple selection // has a mix of suspended and not suspended tasks m_pTaskPane->DisableTask(pGroup->m_Tasks[BTN_SUSPEND]); } } // Disable Show Graphics button if any selected task can't display graphics if (((!result->supports_graphics) || pDoc->GetState()->executing_as_daemon) && result->graphics_exec_path.empty() ) { m_pTaskPane->DisableTask(pGroup->m_Tasks[BTN_GRAPHICS]); } if (result->suspended_via_gui || result->project_suspended_via_gui) { m_pTaskPane->DisableTask(pGroup->m_Tasks[BTN_GRAPHICS]); } // Disable Abort button if any selected task already aborted if ( result->active_task_state == PROCESS_ABORT_PENDING || result->active_task_state == PROCESS_ABORTED || result->state == RESULT_ABORTED ) { m_pTaskPane->DisableTask(pGroup->m_Tasks[BTN_ABORT]); } if (i == 0) { first_project_url = result->project_url; all_same_project = true; } else { if (first_project_url != result->project_url) { all_same_project = false; } } } } if (all_same_project) { project = pDoc->state.lookup_project(result->project_url); UpdateWebsiteSelection(GRP_WEBSITES, project); if(m_TaskGroups.size()>1) { m_pTaskPane->EnableTaskGroupTasks(m_TaskGroups[1]); } } else { UpdateWebsiteSelection(GRP_WEBSITES, NULL); if(m_TaskGroups.size()>1) { m_pTaskPane->DisableTaskGroupTasks(m_TaskGroups[1]); } } CBOINCBaseView::PostUpdateSelection(); } void CViewWorkGrid::UpdateWebsiteSelection(long lControlGroup, PROJECT* project){ unsigned int i; CTaskItemGroup* pGroup = NULL; CTaskItem* pItem = NULL; wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); // Update the websites list // if (m_bForceUpdateSelection) { if (m_TaskGroups.size() > 1) { // Delete task group, objects, and controls. pGroup = m_TaskGroups[lControlGroup]; m_pTaskPane->DeleteTaskGroupAndTasks(pGroup); for (i=0; im_Tasks.size(); i++) { delete pGroup->m_Tasks[i]; } pGroup->m_Tasks.clear(); delete pGroup; pGroup = NULL; m_TaskGroups.erase( m_TaskGroups.begin() + 1 ); } // If something is selected create the tasks and controls if (m_pGridPane->GetFirstSelectedRow()>=0) { if (project) { // Create the web sites task group pGroup = new CTaskItemGroup( _("Web sites") ); m_TaskGroups.push_back( pGroup ); // Default project url pItem = new CTaskItem( wxString(project->project_name.c_str(), wxConvUTF8), wxT(""), wxString(project->master_url.c_str(), wxConvUTF8), ID_TASK_PROJECT_WEB_PROJDEF_MIN ); pGroup->m_Tasks.push_back(pItem); // Project defined urls for (i=0;(igui_urls.size())&&(i<=ID_TASK_PROJECT_WEB_PROJDEF_MAX);i++) { pItem = new CTaskItem( wxGetTranslation(wxString(project->gui_urls[i].name.c_str(), wxConvUTF8)), wxGetTranslation(wxString(project->gui_urls[i].description.c_str(), wxConvUTF8)), wxString(project->gui_urls[i].url.c_str(), wxConvUTF8), ID_TASK_PROJECT_WEB_PROJDEF_MIN + 1 + i ); pGroup->m_Tasks.push_back(pItem); } } } m_bForceUpdateSelection = false; } } wxInt32 CViewWorkGrid::FormatProjectName(wxInt32 item, wxString& strBuffer) const { CMainDocument* doc = wxGetApp().GetDocument(); RESULT* result = wxGetApp().GetDocument()->result(item); PROJECT* state_project = NULL; std::string project_name; wxASSERT(doc); wxASSERT(wxDynamicCast(doc, CMainDocument)); if (result) { state_project = doc->state.lookup_project(result->project_url); if (state_project) { state_project->get_name(project_name); strBuffer = wxT(" ") + HtmlEntityDecode(wxString(project_name.c_str(), wxConvUTF8)); } else { doc->ForceCacheUpdate(); } } return 0; } wxInt32 CViewWorkGrid::FormatApplicationName(wxInt32 item, wxString& strBuffer) const { CMainDocument* pDoc = wxGetApp().GetDocument(); RESULT* result = wxGetApp().GetDocument()->result(item); RESULT* state_result = NULL; wxString strLocalBuffer; wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); if (result) { state_result = pDoc->state.lookup_result(result->project_url, result->name); if (!state_result) { pDoc->ForceCacheUpdate(); state_result = pDoc->state.lookup_result(result->project_url, result->name); } wxASSERT(state_result); wxString strLocale = wxString(setlocale(LC_NUMERIC, NULL), wxConvUTF8); setlocale(LC_NUMERIC, "C"); if (state_result->wup->app->user_friendly_name.size()) { strLocalBuffer = HtmlEntityDecode(wxString(state_result->app->user_friendly_name.c_str(), wxConvUTF8)); } else { strLocalBuffer = HtmlEntityDecode(wxString(state_result->wup->avp->app_name.c_str(), wxConvUTF8)); } char buf[256]; if (state_result->wup->avp->plan_class.size()) { sprintf(buf, " (%s)", state_result->wup->avp->plan_class.c_str()); } else { strcpy(buf, ""); } strBuffer.Printf( wxT(" %s %.2f%s"), strLocalBuffer.c_str(), state_result->wup->avp->version_num/100.0, buf ); setlocale(LC_NUMERIC, (const char*)strLocale.mb_str()); } return 0; } wxInt32 CViewWorkGrid::FormatName(wxInt32 item, wxString& strBuffer) const { RESULT* result = wxGetApp().GetDocument()->result(item); wxASSERT(result); if (result) { strBuffer = _T(" ") + wxString(result->name.c_str(), wxConvUTF8); } return 0; } wxInt32 CViewWorkGrid::FormatCPUTime(wxInt32 item, wxString& strBuffer) const { float fBuffer = 0; wxInt32 iHour = 0; wxInt32 iMin = 0; wxInt32 iSec = 0; wxTimeSpan ts; RESULT* result = wxGetApp().GetDocument()->result(item); if (result) { if (result->active_task) { fBuffer = result->current_cpu_time; } else { if(result->state < RESULT_COMPUTE_ERROR) { fBuffer = 0; } else { fBuffer = result->final_cpu_time; } } } if (0 == fBuffer) { strBuffer = wxT("---"); } else { iHour = (wxInt32)(fBuffer / (60 * 60)); iMin = (wxInt32)(fBuffer / 60) % 60; iSec = (wxInt32)(fBuffer) % 60; ts = wxTimeSpan(iHour, iMin, iSec); strBuffer = wxT(" ") + ts.Format(); } return 0; } wxInt32 CViewWorkGrid::FormatProgress(wxInt32 item, wxString& strBuffer) const { float fBuffer = 0; RESULT* result = wxGetApp().GetDocument()->result(item); if (result) { if (result->active_task) { fBuffer = floor(result->fraction_done * 100000)/1000; } else { if(result->state < RESULT_COMPUTE_ERROR) { fBuffer = 0.0; } else { fBuffer = 100.0; } } } strBuffer.Printf(wxT("%.3f"), fBuffer); return 0; } wxInt32 CViewWorkGrid::FormatTimeToCompletion(wxInt32 item, wxString& strBuffer) const { float fBuffer = 0; wxInt32 iHour = 0; wxInt32 iMin = 0; wxInt32 iSec = 0; wxTimeSpan ts; RESULT* result = wxGetApp().GetDocument()->result(item); if (result) { fBuffer = result->estimated_cpu_time_remaining; } if (0 >= fBuffer) { strBuffer = wxT("---"); } else { iHour = (wxInt32)(fBuffer / (60 * 60)); iMin = (wxInt32)(fBuffer / 60) % 60; iSec = (wxInt32)(fBuffer) % 60; ts = wxTimeSpan(iHour, iMin, iSec); strBuffer = wxT(" ") + ts.Format(); } return 0; } wxInt32 CViewWorkGrid::FormatReportDeadline(wxInt32 item, wxString& strBuffer) const { wxDateTime dtTemp; RESULT* result = wxGetApp().GetDocument()->result(item); if (result) { dtTemp.Set((time_t)result->report_deadline); //don't trust default date string representation, this could prevent correct sorting strBuffer = dtTemp.Format(wxT(" %x %X")); } return 0; } wxInt32 CViewWorkGrid::FormatStatus(wxInt32 item, wxString& strBuffer) const { CMainDocument* doc = wxGetApp().GetDocument(); RESULT* result = wxGetApp().GetDocument()->result(item); CC_STATUS status; wxASSERT(doc); wxASSERT(wxDynamicCast(doc, CMainDocument)); doc->GetCoreClientStatus(status); if (!result) return 0; int throttled = status.task_suspend_reason & SUSPEND_REASON_CPU_USAGE_LIMIT; switch(result->state) { case RESULT_NEW: strBuffer = _("New"); break; case RESULT_FILES_DOWNLOADING: if (result->ready_to_report) { strBuffer = _("Download failed"); } else { strBuffer = _("Downloading"); } break; case RESULT_FILES_DOWNLOADED: if (result->project_suspended_via_gui) { strBuffer = _("Project suspended by user"); } else if (result->suspended_via_gui) { strBuffer = _("Task suspended by user"); } else if (status.task_suspend_reason && !throttled) { strBuffer = _("Suspended"); if (status.task_suspend_reason & SUSPEND_REASON_BATTERIES) { strBuffer += _(" - on batteries"); } if (status.task_suspend_reason & SUSPEND_REASON_USER_ACTIVE) { strBuffer += _(" - user active"); } if (status.task_suspend_reason & SUSPEND_REASON_USER_REQ) { strBuffer += _(" - computation suspended"); } if (status.task_suspend_reason & SUSPEND_REASON_TIME_OF_DAY) { strBuffer += _(" - time of day"); } if (status.task_suspend_reason & SUSPEND_REASON_BENCHMARKS) { strBuffer += _(" - CPU benchmarks"); } if (status.task_suspend_reason & SUSPEND_REASON_DISK_SIZE) { strBuffer += _(" - need disk space"); } } else if (result->active_task) { if (result->too_large) { strBuffer = _("Waiting for memory"); } else if (result->needs_shmem) { strBuffer = _("Waiting for shared memory"); } else if (result->scheduler_state == CPU_SCHED_SCHEDULED) { if (result->edf_scheduled) { strBuffer = _("Running, high priority"); } else { strBuffer = _("Running"); } #if 0 // doesn't work - result pointer not there if (result->project->non_cpu_intensive) { strBuffer += _(" (non-CPU-intensive)"); } #endif } else if (result->scheduler_state == CPU_SCHED_PREEMPTED) { strBuffer = _("Waiting to run"); } else if (result->scheduler_state == CPU_SCHED_UNINITIALIZED) { strBuffer = _("Ready to start"); } } else { strBuffer = _("Ready to start"); } break; case RESULT_COMPUTE_ERROR: strBuffer = _("Computation error"); break; case RESULT_FILES_UPLOADING: if (result->ready_to_report) { strBuffer = _("Upload failed"); } else { strBuffer = _("Uploading"); } break; case RESULT_ABORTED: switch(result->exit_status) { case ERR_ABORTED_VIA_GUI: strBuffer = _("Aborted by user"); break; case ERR_ABORTED_BY_PROJECT: strBuffer = _("Aborted by project"); break; default: strBuffer = _("Aborted"); } break; default: if (result->got_server_ack) { strBuffer = _("Acknowledged"); } else if (result->ready_to_report) { strBuffer = _("Ready to report"); } else { strBuffer.Format(_("Error: invalid state '%d'"), result->state); } break; } strBuffer = wxT(" ") + strBuffer; return 0; } wxInt32 CViewWorkGrid::FormatProjectURL(wxInt32 item, wxString& strBuffer) const { RESULT* result = wxGetApp().GetDocument()->result(item); wxASSERT(result); if (result) { strBuffer = wxString(result->project_url.c_str(), wxConvUTF8); } return 0; } bool CViewWorkGrid::OnSaveState(wxConfigBase* pConfig) { bool bReturnValue = true; wxASSERT(pConfig); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); if (!m_pTaskPane->OnSaveState(pConfig)) { bReturnValue = false; } if (!m_pGridPane->OnSaveState(pConfig)) { bReturnValue = false; } return bReturnValue; } bool CViewWorkGrid::OnRestoreState(wxConfigBase* pConfig) { wxASSERT(pConfig); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); if (!m_pTaskPane->OnRestoreState(pConfig)) { return false; } if (!m_pGridPane->OnRestoreState(pConfig)) { return false; } return true; } wxInt32 CViewWorkGrid::GetDocCount() { return wxGetApp().GetDocument()->GetWorkCount(); } void CViewWorkGrid::OnListRender( wxTimerEvent& WXUNUSED(event) ) { wxInt32 docCount = GetDocCount(); // We haven't connected up to the CC yet, there is nothing to display, make sure // everything is deleted. if ( docCount <= 0 ) { if ( m_pGridPane->GetNumberRows() ) { m_pGridPane->DeleteRows(0, m_pGridPane->GetNumberRows()); m_bForceUpdateSelection = true; UpdateSelection(); UpdateWebsiteSelection(GRP_WEBSITES, NULL); } return; } // flag for row count changes bool rowCountChanged=false; // Right-size the grid so that the number of rows matches // the document state. if(docCount != m_pGridPane->GetNumberRows()) { if (docCount > m_pGridPane->GetNumberRows()) { m_pGridPane->AppendRows(docCount - m_pGridPane->GetNumberRows()); rowCountChanged=true; } else { m_pGridPane->DeleteRows(0, m_pGridPane->GetNumberRows() - docCount); rowCountChanged=true; m_bForceUpdateSelection = true; } wxASSERT(docCount == m_pGridPane->GetNumberRows()); } //init array to detect cell value changes wxArrayInt arrColumnDataChanged; for(int i=0; i< NUM_COLUMNS;i++) { arrColumnDataChanged.Add(0); } //update cell values wxString strBuffer; int iMax = m_pGridPane->GetNumberRows(); if (rowCountChanged) { for(int iRow = 0; iRow < iMax; iRow++) { m_pGridPane->SetCellValue(iRow, COLUMN_RESULTS_INDEX, strBuffer.Format(wxT("%d"), iRow)); } } for(int iRow = 0; iRow < iMax; iRow++) { long results_index; (m_pGridPane->GetCellValue(iRow, COLUMN_RESULTS_INDEX)).ToLong(&results_index); FormatProjectName(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_PROJECT) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_PROJECT, strBuffer); arrColumnDataChanged[COLUMN_PROJECT]=1; } FormatApplicationName(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_APPLICATION) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_APPLICATION, strBuffer); arrColumnDataChanged[COLUMN_APPLICATION]=1; } FormatName(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_NAME) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_NAME, strBuffer); arrColumnDataChanged[COLUMN_NAME]=1; } FormatCPUTime(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_CPUTIME) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_CPUTIME, strBuffer); arrColumnDataChanged[COLUMN_CPUTIME]=1; } FormatProgress(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_PROGRESS) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_PROGRESS, strBuffer); m_pGridPane->SetCellAlignment(iRow, COLUMN_PROGRESS, wxALIGN_CENTRE, wxALIGN_CENTRE); arrColumnDataChanged[COLUMN_PROGRESS]=1; } FormatTimeToCompletion(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_TOCOMPLETION) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_TOCOMPLETION, strBuffer); arrColumnDataChanged[COLUMN_TOCOMPLETION]=1; } FormatReportDeadline(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_REPORTDEADLINE) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_REPORTDEADLINE, strBuffer); arrColumnDataChanged[COLUMN_REPORTDEADLINE]=1; } strBuffer = wxEmptyString; FormatStatus(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_STATUS) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_STATUS, strBuffer); arrColumnDataChanged[COLUMN_STATUS]=1; } FormatProjectURL(results_index, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_HIDDEN_URL) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_HIDDEN_URL, strBuffer); arrColumnDataChanged[COLUMN_HIDDEN_URL]=1; } } //sort only //1. if row count changed //2. if sorting column has cell value changes //3. if sort is enforced by user through label click if( rowCountChanged || (arrColumnDataChanged[m_pGridPane->sortColumn]==1) || m_pGridPane->sortNeededByLabelClick) { wxArrayString ordered_indexes; m_bIgnoreUIEvents = true; m_pGridPane->SaveSelection(); m_bIgnoreUIEvents = false; for(int iRow = 0; iRow < iMax; iRow++) { ordered_indexes.Add(m_pGridPane->GetCellValue(iRow, COLUMN_RESULTS_INDEX)); } m_pGridPane->SortData(); for(int iRow = 0; iRow < iMax; iRow++) { if (ordered_indexes[iRow] != m_pGridPane->GetCellValue(iRow, COLUMN_RESULTS_INDEX)) { // Refresh entire grid if sort order has changed m_bIgnoreUIEvents = true; m_pGridPane->RestoreSelection(); m_bIgnoreUIEvents = false; m_pGridPane->ForceRefresh(); break; } } } UpdateSelection(); }