// 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., // 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA #if defined(__GNUG__) && !defined(__APPLE__) #pragma implementation "ViewTransfersGrid.h" #endif #include "stdwx.h" #include "BOINCGUIApp.h" #include "BOINCBaseFrame.h" #include "MainDocument.h" #include "AdvancedFrame.h" #include "BOINCTaskCtrl.h" #include "ViewTransfersGrid.h" #include "Events.h" #include "error_numbers.h" #include "res/xfer.xpm" #define COLUMN_PROJECT 0 #define COLUMN_FILE 1 #define COLUMN_PROGRESS 2 #define COLUMN_SIZE 3 #define COLUMN_TIME 4 #define COLUMN_SPEED 5 #define COLUMN_STATUS 6 // buttons in the "tasks" area #define BTN_RETRY 0 #define BTN_ABORT 1 IMPLEMENT_DYNAMIC_CLASS(CViewTransfersGrid, CBOINCBaseView) BEGIN_EVENT_TABLE (CViewTransfersGrid, CBOINCBaseView) EVT_BUTTON(ID_TASK_TRANSFERS_RETRYNOW, CViewTransfersGrid::OnTransfersRetryNow) EVT_BUTTON(ID_TASK_TRANSFERS_ABORT, CViewTransfersGrid::OnTransfersAbort) EVT_GRID_SELECT_CELL(CViewTransfersGrid::OnGridSelectCell) EVT_GRID_RANGE_SELECT(CViewTransfersGrid::OnGridSelectRange) END_EVENT_TABLE () CViewTransfersGrid::CViewTransfersGrid() {} CViewTransfersGrid::CViewTransfersGrid(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_TRANSFERSGRIDVIEW, DEFAULT_TASK_FLAGS); wxASSERT(m_pTaskPane); m_pGridPane = new CBOINCGridCtrl(this, ID_LIST_TRANSFERSGRIDVIEW); 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; wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); // // Setup View // pGroup = new CTaskItemGroup( _("Commands") ); m_TaskGroups.push_back( pGroup ); pItem = new CTaskItem( _("Retry Now"), _("Click 'Retry now' to transfer the file now"), ID_TASK_TRANSFERS_RETRYNOW ); pGroup->m_Tasks.push_back( pItem ); pItem = new CTaskItem( _("Abort Transfer"), _("Click 'Abort transfer' to delete the file from the transfer queue. " "This will prevent you from being granted credit for this result."), ID_TASK_TRANSFERS_ABORT ); pGroup->m_Tasks.push_back( pItem ); // Create Task Pane Items m_pTaskPane->UpdateControls(); // Create Grid m_pGridPane->Setup(); m_pGridPane->SetTable(new CBOINCGridTable(1,7)); m_pGridPane->SetSelectionMode(wxGrid::wxGridSelectRows); // init grid columns wxInt32 colSizes[] = {125,205,60,80,80,80,150}; wxString colTitles[] = {_("Project"),_("File"),_("Progress"),_("Size"),_("Elapsed Time"),_("Speed"),_("Status")}; for(int i=0; i<= COLUMN_STATUS;i++){ m_pGridPane->SetColLabelValue(i,colTitles[i]); m_pGridPane->SetColSize(i,colSizes[i]); } //change the default cell renderer m_pGridPane->SetDefaultRenderer(new CBOINCGridCellProgressRenderer(COLUMN_PROGRESS,false)); //set column sort types m_pGridPane->SetColumnSortType(COLUMN_PROGRESS,CST_FLOAT); m_pGridPane->SetColumnSortType(COLUMN_TIME,CST_TIME); //m_pGridPane->SetColumnSortType(COLUMN_SIZE,CST_FLOAT); m_pGridPane->SetColumnSortType(COLUMN_SPEED,CST_FLOAT); //set primary key column index m_pGridPane->SetPrimaryKeyColumn(COLUMN_FILE); UpdateSelection(); } CViewTransfersGrid::~CViewTransfersGrid() { EmptyTasks(); } wxString& CViewTransfersGrid::GetViewName() { static wxString strViewName(_("Transfers")); return strViewName; } const char** CViewTransfersGrid::GetViewIcon() { return xfer_xpm; } void CViewTransfersGrid::OnTransfersRetryNow( wxCommandEvent& WXUNUSED(event) ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewTransfersGrid::OnTransfersRetryNow - Function Begin")); CMainDocument* pDoc = wxGetApp().GetDocument(); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); pFrame->UpdateStatusText(_("Retrying transfer now...")); wxArrayInt aRows = m_pGridPane->GetSelectedRows2(); for(unsigned int i=0; i< aRows.Count();i++) { int row = aRows.Item(i); wxString searchName = m_pGridPane->GetCellValue(row,COLUMN_FILE).Trim(false); pDoc->TransferRetryNow(searchName); } pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->ResetReminderTimers(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewTransfersGrid::OnTransfersRetryNow - Function End")); } void CViewTransfersGrid::OnTransfersAbort( wxCommandEvent& WXUNUSED(event) ) { wxLogTrace(wxT("Function Start/End"), wxT("CViewTransfersGrid::OnTransfersAbort - Function Begin")); wxInt32 iAnswer = 0; wxString strName = wxEmptyString; wxString strMessage = wxEmptyString; CMainDocument* pDoc = wxGetApp().GetDocument(); CAdvancedFrame* pFrame = wxDynamicCast(GetParent()->GetParent()->GetParent(), CAdvancedFrame); wxASSERT(pDoc); wxASSERT(wxDynamicCast(pDoc, CMainDocument)); wxASSERT(pFrame); wxASSERT(wxDynamicCast(pFrame, CAdvancedFrame)); wxASSERT(m_pTaskPane); wxASSERT(m_pGridPane); if (!pDoc->IsUserAuthorized()) return; pFrame->UpdateStatusText(_("Aborting transfer(s)...")); strMessage.Printf( _("Are you sure you want to abort this file(s) transfer ?\n" "NOTE: Aborting a transfer will invalidate a task and you\n" "will not receive credit for it.")); iAnswer = ::wxMessageBox(strMessage,_("Abort File Transfer(s)"),wxYES_NO | wxICON_QUESTION,this); if (wxYES == iAnswer) { wxArrayInt aRows = m_pGridPane->GetSelectedRows2(); for(unsigned int i=0; i< aRows.Count();i++) { int row = aRows.Item(i); wxString searchName = m_pGridPane->GetCellValue(row,COLUMN_FILE).Trim(false); pDoc->TransferAbort(searchName); } } pFrame->UpdateStatusText(wxT("")); UpdateSelection(); pFrame->FireRefreshView(); wxLogTrace(wxT("Function Start/End"), wxT("CViewTransfersGrid::OnTransfersAbort - Function End")); } wxInt32 CViewTransfersGrid::GetDocCount() { return wxGetApp().GetDocument()->GetTransferCount(); } void CViewTransfersGrid::UpdateSelection() { CTaskItemGroup* pGroup = m_TaskGroups[0]; CBOINCBaseView::PreUpdateSelection(); if (m_pGridPane->GetSelectedRows2().size() > 0) { m_pTaskPane->EnableTaskGroupTasks(pGroup); } else { m_pTaskPane->DisableTaskGroupTasks(pGroup); } CBOINCBaseView::PostUpdateSelection(); } wxInt32 CViewTransfersGrid::FormatProjectName(wxInt32 item, wxString& strBuffer) const { FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { strBuffer = wxT(" ") + wxString(transfer->project_name.c_str(), wxConvUTF8); } return 0; } wxInt32 CViewTransfersGrid::FormatFileName(wxInt32 item, wxString& strBuffer) const { FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { strBuffer = wxT(" ") + wxString(transfer->name.c_str(), wxConvUTF8); } return 0; } wxInt32 CViewTransfersGrid::FormatProgress(wxInt32 item, wxString& strBuffer) const { float fBytesSent = 0; float fFileSize = 0; FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { fBytesSent = transfer->bytes_xferred; fFileSize = transfer->nbytes; } // Curl apparently counts the HTTP header in byte count. // Prevent this from causing > 100% display // if (fBytesSent > fFileSize) { fBytesSent = fFileSize; } if ( 0.0 == fFileSize ) { strBuffer = wxT("0%"); } else { strBuffer.Printf(wxT("%.2f%%"), floor((fBytesSent / fFileSize) * 10000)/100); } return 0; } wxInt32 CViewTransfersGrid::FormatSize(wxInt32 item, wxString& strBuffer) const { float fBytesSent = 0; float fFileSize = 0; double xTera = 1099511627776.0; double xGiga = 1073741824.0; double xMega = 1048576.0; double xKilo = 1024.0; FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { fBytesSent = transfer->bytes_xferred; fFileSize = transfer->nbytes; } if (fFileSize != 0) { if (fFileSize >= xTera) { strBuffer.Printf(wxT("%0.2f/%0.2f TB"), fBytesSent/xTera, fFileSize/xTera); } else if (fFileSize >= xGiga) { strBuffer.Printf(wxT("%0.2f/%0.2f GB"), fBytesSent/xGiga, fFileSize/xGiga); } else if (fFileSize >= xMega) { strBuffer.Printf(wxT("%0.2f/%0.2f MB"), fBytesSent/xMega, fFileSize/xMega); } else if (fFileSize >= xKilo) { strBuffer.Printf(wxT("%0.2f/%0.2f KB"), fBytesSent/xKilo, fFileSize/xKilo); } else { strBuffer.Printf(wxT("%0.0f/%0.0f bytes"), fBytesSent, fFileSize); } } else { if (fBytesSent >= xTera) { strBuffer.Printf(wxT("%0.2f TB"), fBytesSent/xTera); } else if (fBytesSent >= xGiga) { strBuffer.Printf(wxT("%0.2f GB"), fBytesSent/xGiga); } else if (fBytesSent >= xMega) { strBuffer.Printf(wxT("%0.2f MB"), fBytesSent/xMega); } else if (fBytesSent >= xKilo) { strBuffer.Printf(wxT("%0.2f KB"), fBytesSent/xKilo); } else { strBuffer.Printf(wxT("%0.0f bytes"), fBytesSent); } } strBuffer = wxT(" ") + strBuffer; return 0; } wxInt32 CViewTransfersGrid::FormatTime(wxInt32 item, wxString& strBuffer) const { float fBuffer = 0; wxInt32 iHour = 0; wxInt32 iMin = 0; wxInt32 iSec = 0; wxTimeSpan ts; FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { fBuffer = transfer->time_so_far; } 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 CViewTransfersGrid::FormatSpeed(wxInt32 item, wxString& strBuffer) const { float fTransferSpeed = 0; FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); if (transfer) { if (transfer->xfer_active) fTransferSpeed = transfer->xfer_speed / 1024; else fTransferSpeed = 0.0; } strBuffer.Printf(wxT(" %.2f KBps"), fTransferSpeed); return 0; } wxInt32 CViewTransfersGrid::FormatStatus(wxInt32 item, wxString& strBuffer) const { CMainDocument* doc = wxGetApp().GetDocument(); FILE_TRANSFER* transfer = wxGetApp().GetDocument()->file_transfer(item); CC_STATUS status; wxASSERT(doc); wxASSERT(wxDynamicCast(doc, CMainDocument)); doc->GetCoreClientStatus(status); wxDateTime dtNextRequest((time_t)transfer->next_request_time); wxDateTime dtNow(wxDateTime::Now()); if (transfer) { if (dtNextRequest > dtNow) { wxTimeSpan tsNextRequest(dtNextRequest - dtNow); strBuffer = _("Retry in ") + tsNextRequest.Format(); } else if (ERR_GIVEUP_DOWNLOAD == transfer->status) { strBuffer = _("Download failed"); } else if (ERR_GIVEUP_UPLOAD == transfer->status) { strBuffer = _("Upload failed"); } else { if (status.network_suspend_reason) { strBuffer = _("Suspended"); } else { if (transfer->xfer_active) { strBuffer = transfer->generated_locally? _("Uploading") : _("Downloading"); } else { strBuffer = transfer->generated_locally? _("Upload pending") : _("Download pending"); } } } } strBuffer = wxT(" ") + strBuffer; return 0; } bool CViewTransfersGrid::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 CViewTransfersGrid::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; } void CViewTransfersGrid::OnListRender( wxTimerEvent& WXUNUSED(event) ) { // We haven't connected up to the CC yet, there is nothing to display, make sure // everything is deleted. if ( GetDocCount() <= 0 ) { if ( m_pGridPane->GetNumberRows() ) { m_pGridPane->DeleteRows(0, m_pGridPane->GetNumberRows()); } return; } // Right-size the grid so that the number of rows matches // the document state. if(GetDocCount() != m_pGridPane->GetNumberRows()) { if (GetDocCount() > m_pGridPane->GetNumberRows()) { m_pGridPane->AppendRows(GetDocCount() - m_pGridPane->GetNumberRows()); } else { m_pGridPane->DeleteRows(0, m_pGridPane->GetNumberRows() - GetDocCount()); } wxASSERT(GetDocCount() == m_pGridPane->GetNumberRows()); } //update cell values wxString strBuffer; int iMax = m_pGridPane->GetNumberRows(); for(int iRow = 0; iRow < iMax; iRow++) { FormatProjectName(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_PROJECT) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_PROJECT, strBuffer); } FormatFileName(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_FILE) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_FILE, strBuffer); } FormatProgress(iRow, 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); } FormatSize(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_SIZE) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_SIZE, strBuffer); } FormatTime(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_TIME) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_TIME, strBuffer); } FormatSpeed(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_SPEED) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_SPEED, strBuffer); } strBuffer = wxEmptyString; FormatStatus(iRow, strBuffer); if (m_pGridPane->GetCellValue(iRow, COLUMN_STATUS) != strBuffer) { m_pGridPane->SetCellValue(iRow, COLUMN_STATUS, strBuffer); } } m_pGridPane->SortData(); UpdateSelection(); }