// The contents of this file are subject to the BOINC Public License // Version 1.0 (the "License"); you may not use this file except in // compliance with the License. You may obtain a copy of the License at // http://boinc.berkeley.edu/license_1.0.txt // // Software distributed under the License is distributed on an "AS IS" // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the // License for the specific language governing rights and limitations // under the License. // // The Original Code is the Berkeley Open Infrastructure for Network Computing. // // The Initial Developer of the Original Code is the SETI@home project. // Portions created by the SETI@home project are Copyright (C) 2002 // University of California at Berkeley. All Rights Reserved. // // Contributor(s): // #include "wingui_mainwindow.h" #include "Stackwalker.h" CMyApp g_myApp; CMainWindow* g_myWnd = NULL; /*static struct _test { _test() { InitAllocCheck(); } ~_test() { DeInitAllocCheck(); } } _myLeakFinder;*/ ///////////////////////////////////////////////////////////////////////// // CMyApp member functions ////////// // CMyApp::InitInstance // arguments: void // returns: true if initialization is successful, otherwise false // function: creates and shows the main window if boinc is not running, // otherwise shows the currently running window BOOL CMyApp::InitInstance() { if(CreateMutex(NULL, false, RUN_MUTEX) == 0 || GetLastError() == ERROR_ALREADY_EXISTS) { UINT nShowMsg = RegisterWindowMessage(SHOW_WIN_MSG); PostMessage(HWND_BROADCAST, nShowMsg, 0, 0); return FALSE; } m_pMainWnd = new CMainWindow(); if(gstate.projects.size() == 0) { ((CMainWindow*)m_pMainWnd)->SendMessage(WM_COMMAND, ID_SETTINGS_LOGIN); } if(gstate.start_saver) { UINT nStartSaver = RegisterWindowMessage(START_SS_MSG); ((CMainWindow*)m_pMainWnd)->SendMessage(nStartSaver, 0); } return TRUE; } int CMyApp::ExitInstance() { if (m_pMainWnd) m_pMainWnd->DestroyWindow(); //gstate.free_mem(); return CWinApp::ExitInstance(); } //////// void GetLanguageFilename(CString& path) { GetCurrentDirectory(256, path.GetBuffer(256)); path.ReleaseBuffer(); path += '\\'; path += LANGUAGE_FILE_NAME; } inline bool UpdateLanguageString(LPCSTR szSection, CString& strItem, LPCSTR szPath) { CString strItemNoAmp = strItem; strItemNoAmp.Remove('&'); if(strItemNoAmp.IsEmpty()) return false; char szValue[256]; GetPrivateProfileString(szSection, strItemNoAmp, strItem, szValue, 256, szPath); strItem = szValue; return true; } void UpdateLanguageStrings(CWnd* wnd, char const * windowname, int const* pnIDs, CString * const * pStrs) { CString strPath, strKey; char szVal[256]; GetLanguageFilename(strPath); // title wnd->GetWindowText(strKey); GetPrivateProfileString(windowname, "Title", strKey, szVal, 256, strPath); wnd->SetWindowText(szVal); while (*pnIDs) { int nID = *pnIDs++; wnd->GetDlgItemText(nID, strKey); GetPrivateProfileString(windowname, strKey, strKey, szVal, 256, strPath); wnd->SetDlgItemText(nID, szVal); } if (pStrs) { while (*pStrs) { CString& str = **pStrs++; GetPrivateProfileString(windowname, str, str, szVal, 256, strPath); str = szVal; } } } inline CString GetStrTime(CTime timeObj = CTime::GetCurrentTime()) { CString strTime; strTime = timeObj.Format("%c"); // TODO: use windows time locale return strTime; } ///////////////////////////////////////////////////////////////////////// // CMainWindow message map and member functions BEGIN_MESSAGE_MAP(CMainWindow, CWnd) ON_WM_CLOSE() ON_WM_DESTROY() //ON_COMMAND(ID_FILE_CLEARINACTIVE, OnCommandFileClearInactive) //ON_COMMAND(ID_FILE_CLEARMESSAGES, OnCommandFileClearMessages) ON_COMMAND(ID_FILE_RUN_REQUEST_ALWAYS, OnCommandRunRequestAlways) ON_COMMAND(ID_FILE_RUN_REQUEST_AUTO, OnCommandRunRequestAuto) ON_COMMAND(ID_FILE_RUN_REQUEST_NEVER, OnCommandRunRequestNever) ON_COMMAND(ID_FILE_RUN_BENCHMARKS, OnCommandRunBenchmarks) ON_COMMAND(ID_FILE_HIDE, OnCommandHide) ON_COMMAND(ID_FILE_EXIT, OnCommandExit) ON_COMMAND(ID_SETTINGS_LOGIN, OnCommandSettingsLogin) ON_COMMAND(ID_SETTINGS_PROXYSERVER, OnCommandSettingsProxyServer) ON_COMMAND(ID_HELP_ABOUT, OnCommandHelpAbout) ON_COMMAND(ID_PROJECT_WEB_SITE, OnCommandProjectWebSite) ON_COMMAND(ID_PROJECT_GET_PREFS, OnCommandProjectGetPrefs) ON_COMMAND(ID_PROJECT_DETACH, OnCommandProjectDetach) ON_COMMAND(ID_PROJECT_RESET, OnCommandProjectReset) ON_COMMAND(ID_STATUSICON_SHOW, OnCommandShow) ON_COMMAND(ID_STATUSICON_HIDE, OnCommandHide) ON_COMMAND(ID_STATUSICON_EXIT, OnCommandExit) ON_COMMAND(ID_WORK_SHOWGRAPHICS, OnCommandWorkShowGraphics) ON_COMMAND(ID_TRANSFERS_RETRYNOW, OnCommandTransfersRetryNow) ON_COMMAND(ID_MESSAGE_COPY_TO_CLIP, OnCommandMessageCopyToClip) // Added by JBK. ON_WM_CREATE() ON_WM_RBUTTONDOWN() ON_WM_SIZE() ON_WM_SETFOCUS() ON_WM_TIMER() ON_MESSAGE(STATUS_ICON_ID, OnStatusIcon) END_MESSAGE_MAP() ////////// // CMainWindow::CMainWindow // arguments: void // returns: void // function: registers window class, creates and poisitions window. CMainWindow::CMainWindow() { WNDCLASS wndcls; memset(&wndcls, 0, sizeof(WNDCLASS)); // start with NULL defaults wndcls.style = 0; //wndcls.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW; wndcls.lpfnWndProc = ::DefWindowProc; wndcls.hInstance = AfxGetInstanceHandle(); wndcls.hIcon = g_myApp.LoadIcon(IDI_ICON); wndcls.hCursor = g_myApp.LoadStandardCursor(IDC_ARROW); wndcls.hbrBackground = (HBRUSH)(COLOR_3DFACE+1); wndcls.lpszMenuName = NULL; // Specify class name for using FindWindow in the installer wndcls.lpszClassName = _T("BOINCWindowClass"); // Register the new class and exit if it fails if(!AfxRegisterClass(&wndcls)) return; // register window class //CString strWndClass = AfxRegisterWndClass (0, g_myApp.LoadStandardCursor(IDC_ARROW), // (HBRUSH)(COLOR_3DFACE+1), g_myApp.LoadIcon(IDI_ICON)); m_MenuLabelRetryNow = "&Retry now"; m_MenuLabelGetPreferences = "Get p&references"; m_DialogResetQuery = "Are you sure you want to reset the project %1?"; m_DialogDetachQuery = "Are you sure you want to detach from the project %1?"; // create and position window CreateEx(0, wndcls.lpszClassName, WND_TITLE, WS_OVERLAPPEDWINDOW|WS_EX_OVERLAPPEDWINDOW|WS_CLIPCHILDREN, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, NULL); m_nShowMsg = RegisterWindowMessage(SHOW_WIN_MSG); m_nNetActivityMsg = RegisterWindowMessage(NET_ACTIVITY_MSG); m_uScreenSaverMsg = RegisterWindowMessage(START_SS_MSG); m_uEndSSMsg = RegisterWindowMessage(END_SS_MSG); } ////////// // CMainWindow::GetPieColor // arguments: nPiece: index of pie piece // returns: the color that piece should be // function: detemines colors for pie pieces in usage control COLORREF CMainWindow::GetPieColor(int nPiece) { return RGB( (64*(nPiece%4))%256, 0, (256-64*(nPiece%4)-1)%256 ); } ////////// // CMainWindow::ClearProjectItems // arguments: proj_url: master url of the project // returns: void // function: removes all active and inactive projects, transfers, // and workunits associated with the project void CMainWindow::ClearProjectItems(char *proj_url) { int i; CString ItemURL; // TODO: find out how proj_url can be null if (!proj_url) { fprintf(stderr, "CMainWindow::ClearProjectItems failed: proj_url is null"); return; } for(i = 0; i < m_ProjectListCtrl.GetItemCount();) { ItemURL = m_ProjectListCtrl.GetProjectURL(i); if(!ItemURL.Compare(proj_url)) { m_ProjectListCtrl.DeleteItem(i); } else { i ++; } } for(i = 0; i < m_ResultListCtrl.GetItemCount();) { ItemURL = m_ResultListCtrl.GetProjectURL(i); if(!ItemURL.Compare(proj_url)) { m_ResultListCtrl.DeleteItem(i); } else { i ++; } } for(i = 0; i < m_XferListCtrl.GetItemCount();) { ItemURL = m_XferListCtrl.GetProjectURL(i); if(!ItemURL.Compare(proj_url)) { m_XferListCtrl.DeleteItem(i); } else { i ++; } } } ////////// // CMainWindow::UpdateGUI // arguments: pcs: pointer to the client state for the gui to display // returns: void // function: syncronizes list controls with vectors in client state // and displays them. void CMainWindow::UpdateGUI(CLIENT_STATE* pcs) { CString strBuf; float totalres; int i, n; // If we failed to set the taskbar icon before, keep trying! if (m_nDesiredIconState != m_nIconState) SetStatusIcon(m_nDesiredIconState); switch (m_TabCtrl.GetCurFocus()) { case PROJECT_ID: // display projects m_ProjectListCtrl.SetRedraw(FALSE); totalres = 0; Syncronize(&m_ProjectListCtrl, (vector*)(&pcs->projects)); for(i = 0; i < pcs->projects.size(); i ++) { totalres += pcs->projects[i]->resource_share; } for(i = 0; i < m_ProjectListCtrl.GetItemCount(); i ++) { PROJECT* pr = (PROJECT*)m_ProjectListCtrl.GetItemData(i); if(!pr) { m_ProjectListCtrl.DeleteItem(i--); continue; } // Set the master URL for this object if (m_ProjectListCtrl.GetProjectURL(i) != pr->master_url) m_ProjectListCtrl.SetProjectURL(i, pr->master_url); // project if (m_ProjectListCtrl.GetItemText(i, 0) != pr->get_project_name()) m_ProjectListCtrl.SetItemText(i, 0, pr->get_project_name()); // account if (m_ProjectListCtrl.GetItemText(i, 1) != pr->user_name) m_ProjectListCtrl.SetItemText(i, 1, pr->user_name); // total credit strBuf.Format("%0.2f", pr->user_total_credit); if (m_ProjectListCtrl.GetItemText(i, 2) != strBuf) m_ProjectListCtrl.SetItemText(i, 2, strBuf); // avg credit strBuf.Format("%0.2f", pr->user_expavg_credit); if (m_ProjectListCtrl.GetItemText(i, 3) != strBuf) m_ProjectListCtrl.SetItemText(i, 3, strBuf); // resource share if(totalres <= 0) { if (m_ProjectListCtrl.GetItemProgress(i, 4) != 100) m_ProjectListCtrl.SetItemProgress(i, 4, 100); } else { if (m_ProjectListCtrl.GetItemProgress(i, 4) != (100 * pr->resource_share) / totalres) m_ProjectListCtrl.SetItemProgress(i, 4, (100 * pr->resource_share) / totalres); } } m_ProjectListCtrl.SetRedraw(TRUE); break; case RESULT_ID: // update results m_ResultListCtrl.SetRedraw(FALSE); Syncronize(&m_ResultListCtrl, (vector*)(&pcs->results)); for(i = 0; i < m_ResultListCtrl.GetItemCount(); i ++) { RESULT* re = (RESULT*)m_ResultListCtrl.GetItemData(i); if(!re) { m_ResultListCtrl.DeleteItem(i--); continue; } // Set the master URL for this object if (m_ResultListCtrl.GetProjectURL(i) != re->project->master_url) m_ResultListCtrl.SetProjectURL(i, re->project->master_url); // project if (m_ResultListCtrl.GetItemText(i, 0) != re->project->project_name) m_ResultListCtrl.SetItemText(i, 0, re->project->project_name); // application if (m_ResultListCtrl.GetItemText(i, 1) != re->app->name) m_ResultListCtrl.SetItemText(i, 1, re->app->name); // name if (m_ResultListCtrl.GetItemText(i, 2) != re->name) m_ResultListCtrl.SetItemText(i, 2, re->name); // cpu time ACTIVE_TASK* at = gstate.lookup_active_task_by_result(re); double cur_cpu; if (at) { cur_cpu = at->current_cpu_time; } else { if(re->state < RESULT_COMPUTE_DONE) cur_cpu = 0; else cur_cpu = re->final_cpu_time; } int cpuhour = (int)(cur_cpu / (60 * 60)); int cpumin = (int)(cur_cpu / 60) % 60; int cpusec = (int)(cur_cpu) % 60; if (cur_cpu == 0) strBuf.Format("---"); else strBuf.Format("%0.2d:%0.2d:%0.2d", cpuhour, cpumin, cpusec); if (m_ResultListCtrl.GetItemText(i, 3) != strBuf) m_ResultListCtrl.SetItemText(i, 3, strBuf); // progress if(!at) { m_ResultListCtrl.SetItemProgress(i, 4, 0); } else { m_ResultListCtrl.SetItemProgress(i, 4, at->fraction_done * 100); } // to completion double tocomp; if(!at || (tocomp = at->est_time_to_completion()) <= 0) { tocomp = gstate.estimate_cpu_time(*re->wup); } if (tocomp > 0) { cpuhour = (int)(tocomp / (60 * 60)); cpumin = (int)(tocomp / 60) % 60; cpusec = (int)(tocomp) % 60; strBuf.Format("%0.2d:%0.2d:%0.2d", cpuhour, cpumin, cpusec); //strBuf.Format("%s [%.f]", strBuf, tocomp); //if (at) { // strBuf.Format("%s frac_done=%.7f frac_r=%.9f rec_ch=%.9f", strBuf, at->fraction_done, at->frac_rate_of_change, at->recent_change); //} } else { strBuf = "---"; } if (m_ResultListCtrl.GetItemText(i, 5) != strBuf) m_ResultListCtrl.SetItemText(i, 5, strBuf); if (re->report_deadline) { strBuf = GetStrTime(re->report_deadline); } else { strBuf = "---"; } if (m_ResultListCtrl.GetItemText(i,6) != strBuf) { m_ResultListCtrl.SetItemText(i, 6, strBuf); } // status switch(re->state) { case RESULT_NEW: strBuf.Format(g_szMiscItems[0]); break; case RESULT_FILES_DOWNLOADING: strBuf.Format(g_szMiscItems[9]); break; case RESULT_FILES_DOWNLOADED: if (at) strBuf.Format(g_szMiscItems[1]); else strBuf.Format(g_szMiscItems[2]); break; case RESULT_COMPUTE_DONE: strBuf.Format(g_szMiscItems[3]); break; break; case RESULT_FILES_UPLOADING: strBuf.Format(g_szMiscItems[8]); break; default: if (re->server_ack) strBuf.Format(g_szMiscItems[5]); else if (re->ready_to_ack) strBuf.Format(g_szMiscItems[4]); else strBuf.Format(g_szMiscItems[6]); break; } if (gstate.is_suspended()) { strBuf = CString(g_szMiscItems[13]) + " (" + strBuf + ")"; } if (m_ResultListCtrl.GetItemText(i, 7) != strBuf) m_ResultListCtrl.SetItemText(i, 7, strBuf); } m_ResultListCtrl.SetRedraw(TRUE); break; case XFER_ID: // update xfers m_XferListCtrl.SetRedraw(FALSE); Syncronize(&m_XferListCtrl, (vector*)(&pcs->pers_file_xfers->pers_file_xfers)); for(i = 0; i < m_XferListCtrl.GetItemCount(); i ++) { PERS_FILE_XFER* pfx = (PERS_FILE_XFER*)m_XferListCtrl.GetItemData(i); if(!pfx) { m_XferListCtrl.DeleteItem(i--); continue; } // Set the master URL for this object if (m_XferListCtrl.GetProjectURL(i) != pfx->fip->project->master_url) m_XferListCtrl.SetProjectURL(i, pfx->fip->project->master_url); // project if (m_XferListCtrl.GetItemText(i, 0) != pfx->fip->project->project_name) m_XferListCtrl.SetItemText(i, 0, pfx->fip->project->project_name); // file if (m_XferListCtrl.GetItemText(i, 1) != pfx->fip->name) m_XferListCtrl.SetItemText(i, 1, pfx->fip->name); // progress double xSent = 0, f_size; char pathnm[256]; if (pfx->fxp) { xSent = pfx->fxp->bytes_xferred; } else { get_pathname(pfx->fip, pathnm); if (file_size(pathnm, f_size)) { f_size = 0; } xSent = f_size; } // if (m_XferListCtrl.GetItemProgress(i, 2) != 100 * xSent / pfx->fip->nbytes) m_XferListCtrl.SetItemProgress(i, 2, 100 * xSent / pfx->fip->nbytes); // size char size_buf[256]; get_byte_string(xSent, pfx->fip->nbytes, size_buf, 256); if (strcmp(m_XferListCtrl.GetItemText(i, 3).GetBuffer(0), size_buf)) m_XferListCtrl.SetItemText(i, 3, size_buf); // time double xtime = 0; xtime = pfx->time_so_far; int xhour = (int)(xtime / (60 * 60)); int xmin = (int)(xtime / 60) % 60; int xsec = (int)(xtime) % 60; strBuf.Format("%0.2d:%0.2d:%0.2d", xhour, xmin, xsec); if (m_XferListCtrl.GetItemText(i, 4) != strBuf.GetBuffer(0)) m_XferListCtrl.SetItemText(i, 4, strBuf.GetBuffer(0)); // speed strBuf.Format("0.00 KBps"); if(pfx->fxp) { strBuf.Format("%0.2f KBps", pfx->fxp->xfer_speed/1024); } if (m_XferListCtrl.GetItemText(i, 5) != strBuf.GetBuffer(0)) m_XferListCtrl.SetItemText(i, 5, strBuf.GetBuffer(0)); // status if (pfx->next_request_time > time(0)) { double xtime = pfx->next_request_time-time(0); int xhour = (int)(xtime / (60 * 60)); int xmin = (int)(xtime / 60) % 60; int xsec = (int)(xtime) % 60; strBuf.Format("%s %0.2d:%0.2d:%0.2d", g_szMiscItems[10], xhour, xmin, xsec); } else if (pfx->fip->status == ERR_GIVEUP_DOWNLOAD) { strBuf.Format(g_szMiscItems[11]); } else if (pfx->fip->status == ERR_GIVEUP_UPLOAD) { strBuf.Format(g_szMiscItems[12]); } else { strBuf.Format(pfx->fip->generated_locally?g_szMiscItems[8]:g_szMiscItems[9]); } if (gstate.is_suspended()) { strBuf = CString(g_szMiscItems[13]) + " (" + strBuf + ")"; } if (m_XferListCtrl.GetItemText(i, 6) != strBuf) m_XferListCtrl.SetItemText(i, 6, strBuf); } m_XferListCtrl.SetRedraw(TRUE); break; case MESSAGE_ID: // check message pane size n = m_MessageListCtrl.GetItemCount(); if (n > 2000) { m_MessageListCtrl.SetRedraw(FALSE); for (i = 0 ; i < n - 2000 ; i++) m_MessageListCtrl.DeleteItem(0); m_MessageListCtrl.SetRedraw(TRUE); } break; case USAGE_ID: // update usage double xDiskTotal; double xDiskFree; get_filesystem_info(xDiskTotal, xDiskFree); double xDiskUsed = xDiskTotal - xDiskFree; double xDiskAllow; gstate.allowed_disk_usage(xDiskAllow); xDiskAllow = xDiskFree - xDiskAllow; double xDiskUsage; gstate.current_disk_usage(xDiskUsage); m_UsagePieCtrl.SetTotal(xDiskTotal); m_UsagePieCtrl.SetPiece(0, xDiskFree); // Free space m_UsagePieCtrl.SetPiece(1, xDiskUsed - xDiskUsage); // Used space m_UsagePieCtrl.SetPiece(2, xDiskUsage); // Used space: BOINC m_UsagePieCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_NOERASE|RDW_FRAME); while(m_UsageBOINCPieCtrl.GetItemCount() - 1 < gstate.projects.size()) { m_UsageBOINCPieCtrl.AddPiece("", GetPieColor(m_UsageBOINCPieCtrl.GetItemCount()), 0); } while(m_UsageBOINCPieCtrl.GetItemCount() - 1 > gstate.projects.size()) { m_UsageBOINCPieCtrl.RemovePiece(m_UsageBOINCPieCtrl.GetItemCount() - 1); } m_UsageBOINCPieCtrl.SetTotal(xDiskUsage); m_UsageBOINCPieCtrl.SetPiece(0, 1); // BOINC: core application for(i = 0; i < gstate.projects.size(); i ++) { double xUsage; CString strLabel; strLabel.Format("%s", gstate.projects[i]->project_name); gstate.project_disk_usage(gstate.projects[i], xUsage); m_UsageBOINCPieCtrl.SetPieceLabel(i + 1, strLabel.GetBuffer(0)); m_UsageBOINCPieCtrl.SetPiece(i + 1, xUsage); xDiskUsage -= xUsage; } m_UsageBOINCPieCtrl.SetPiece(0, xDiskUsage); // BOINC: core application m_UsageBOINCPieCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_NOERASE|RDW_FRAME); break; } // make icon flash if needed if(m_bMessage || m_bRequest) { if(m_nIconState == ICON_NORMAL) { SetStatusIcon(ICON_HIGHLIGHT); } else if(m_nIconState == ICON_HIGHLIGHT) { SetStatusIcon(ICON_NORMAL); } } } ////////// // CMainWindow::MessageUser // arguments: message: message string to display // priority: string with priority of message // returns: void // function: if message is MSG_ERROR priority, flashes the status icon, // then adds to message edit control. void CMainWindow::MessageUser(char* szProject, char* szMessage, int szPriority) { if(!m_MessageListCtrl.GetSafeHwnd()) return; int nNewPos = m_MessageListCtrl.GetItemCount(); m_MessageListCtrl.InsertItem(nNewPos, szProject); m_MessageListCtrl.Scroll(m_MessageListCtrl.ApproximateViewRect()); m_MessageListCtrl.SetItemText(nNewPos, 1, GetStrTime()); m_MessageListCtrl.SetItemText(nNewPos, 2, szMessage); // set status icon to flash if((szPriority == MSG_ERROR) && (m_TabCtrl.GetCurSel() != MESSAGE_ID || GetForegroundWindow() != this)) { m_bMessage = true; } } ////////// // CMainWindow::RequestNetConnect // arguments: void // returns: true if the user can connect, false otherwise // function: asks the user for permission to connect to the network BOOL CMainWindow::RequestNetConnect() { if(GetForegroundWindow() != this || !IsWindowVisible()) { m_bRequest = true; return FALSE; } CConnectDialog dlg(IDD_CONNECT); int retval = dlg.DoModal(); m_bRequest = false; if(retval == IDOK) { return TRUE; } return FALSE; } ////////// // CMainWindow::ShowTab // arguments: nTab: tab number to show // returns: void // function: handles everything necessary to switch to a new tab void CMainWindow::ShowTab(int nTab) { m_TabCtrl.SetCurSel(nTab); // make the selected control visible, all the rest invisible m_ProjectListCtrl.ModifyStyle(WS_VISIBLE, 0); m_ResultListCtrl.ModifyStyle(WS_VISIBLE, 0); m_XferListCtrl.ModifyStyle(WS_VISIBLE, 0); m_MessageListCtrl.ModifyStyle(WS_VISIBLE, 0); m_UsagePieCtrl.ModifyStyle(WS_VISIBLE, 0); m_UsageBOINCPieCtrl.ModifyStyle(WS_VISIBLE, 0); if(nTab == PROJECT_ID) { m_ProjectListCtrl.ModifyStyle(0, WS_VISIBLE); m_ProjectListCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } else if(nTab == RESULT_ID) { m_ResultListCtrl.ModifyStyle(0, WS_VISIBLE); m_ResultListCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } else if(nTab == XFER_ID) { m_XferListCtrl.ModifyStyle(0, WS_VISIBLE); m_XferListCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } else if(nTab == MESSAGE_ID) { m_MessageListCtrl.ModifyStyle(0, WS_VISIBLE); if(m_bMessage) { m_bMessage = false; SetStatusIcon(ICON_NORMAL); } m_MessageListCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } else if(nTab == USAGE_ID) { m_UsagePieCtrl.ModifyStyle(0, WS_VISIBLE); m_UsagePieCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); m_UsageBOINCPieCtrl.ModifyStyle(0, WS_VISIBLE); m_UsageBOINCPieCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } m_TabCtrl.RedrawWindow(NULL, NULL, RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); RedrawWindow(); UpdateGUI(&gstate); } ////////// // CMainWindow::SetStatusIcon // arguments: dwMessage: hide or show the icon // returns: void // function: controls the status icon in the taskbar bool CMainWindow::SetStatusIcon(DWORD dwMessage) { BOOL success = false; NOTIFYICONDATA icon_data; if(dwMessage != ICON_OFF && dwMessage != ICON_NORMAL && dwMessage != ICON_HIGHLIGHT) { return false; } // if icon is in that state already, there is nothing to do if(dwMessage == m_nIconState) return true; m_nDesiredIconState = dwMessage; memset( &icon_data, 0, sizeof(NOTIFYICONDATA) ); icon_data.cbSize = sizeof(NOTIFYICONDATA); icon_data.uFlags = NIF_MESSAGE | NIF_ICON | NIF_TIP; icon_data.hWnd = GetSafeHwnd(); icon_data.uID = STATUS_ICON_ID; safe_strncpy(icon_data.szTip, WND_TITLE, min(sizeof(icon_data.szTip), sizeof(WND_TITLE))); icon_data.uCallbackMessage = STATUS_ICON_ID; if(dwMessage == ICON_OFF) { icon_data.hIcon = NULL; success = Shell_NotifyIcon(NIM_DELETE, &icon_data); } else if(dwMessage == ICON_NORMAL) { icon_data.hIcon = g_myApp.LoadIcon(IDI_ICONSMALL); if(m_nIconState == ICON_OFF) { success = Shell_NotifyIcon(NIM_ADD, &icon_data); } else { success = Shell_NotifyIcon(NIM_MODIFY, &icon_data); } } else if(dwMessage == ICON_HIGHLIGHT) { icon_data.hIcon = g_myApp.LoadIcon(IDI_ICONHIGHLIGHT); if(m_nIconState == ICON_OFF) { success = Shell_NotifyIcon(NIM_ADD, &icon_data); } else { success = Shell_NotifyIcon(NIM_MODIFY, &icon_data); } } if (!success) return false; m_nIconState = dwMessage; return true; } ////////// // CMainWindow::SaveListControls // arguments: void // returns: void // function: saves relevant elements of list controls void CMainWindow::SaveListControls() { char szPath[256]; CString strKey, strVal; GetCurrentDirectory(256, szPath); strcat(szPath, "\\"); strcat(szPath, LIST_STATE_FILE_NAME); file_delete(szPath); m_ProjectListCtrl.SaveInactive(szPath, "PROJECTS"); m_ResultListCtrl.SaveInactive(szPath, "WORK"); m_XferListCtrl.SaveInactive(szPath, "TRANSFERS"); m_MessageListCtrl.SaveInactive(szPath, "MESSAGES"); } ////////// // CMainWindow::LoadListControls // arguments: void // returns: void // function: loads relevant elements of list controls void CMainWindow::LoadListControls() { char szPath[256]; CString strKey, strVal; GetCurrentDirectory(256, szPath); strcat(szPath, "\\"); strcat(szPath, LIST_STATE_FILE_NAME); m_ProjectListCtrl.LoadInactive(szPath, "PROJECTS"); m_ResultListCtrl.LoadInactive(szPath, "WORK"); m_XferListCtrl.LoadInactive(szPath, "TRANSFERS"); m_MessageListCtrl.LoadInactive(szPath, "MESSAGES"); } inline void SaveColumns(CProgressListCtrl& listCtrl, const char* header_name, const char* szPath ) { int colOrder[MAX_COLS]; int itemCount = listCtrl.GetHeaderCtrl()->GetItemCount(); listCtrl.GetColumnOrderArray(colOrder, itemCount); CString strKey, strVal; strKey.Format("%s-order", header_name); WritePrivateProfileStruct("HEADERS", strKey, colOrder, sizeof(int)*itemCount, szPath); for(int i = 0; i < itemCount; i ++) { strKey.Format("%s-%d", header_name, i); strVal.Format("%d", listCtrl.GetColumnWidth(i)); WritePrivateProfileString("HEADERS", strKey, strVal, szPath); } } inline void LoadColumns(CProgressListCtrl& listCtrl, const char* header_name, const char* szPath, int const* default_widths = NULL ) { int colOrder[MAX_COLS]; int itemCount = listCtrl.GetHeaderCtrl()->GetItemCount(); CString strKey; strKey.Format("%s-order", header_name); memset(colOrder, 0, sizeof(colOrder)); if(GetPrivateProfileStruct("HEADERS", strKey, colOrder, sizeof(int)*itemCount, szPath)) { listCtrl.SetColumnOrderArray(itemCount, colOrder); } for(int i = 0; i < itemCount; i ++) { strKey.Format("%s-%d", header_name, i); int width = default_widths ? default_widths[i] : DEF_COL_WIDTH; width = GetPrivateProfileInt("HEADERS", strKey, width, szPath); listCtrl.SetColumnWidth(i, width); } } ////////// // CMainWindow::SaveUserSettings // arguments: void // returns: void // function: saves relevant user settings to boinc ini file void CMainWindow::SaveUserSettings() { char szPath[256]; CString strKey, strVal; GetCurrentDirectory(256, szPath); strcat(szPath, "\\"); strcat(szPath, INI_FILE_NAME); // get rid of old lists file_delete(szPath); // save window size/position CRect rt; GetWindowRect(&rt); strVal.Format("%d", rt.left); WritePrivateProfileString("WINDOW", "xposition", strVal, szPath); strVal.Format("%d", rt.top); WritePrivateProfileString("WINDOW", "yposition", strVal, szPath); strVal.Format("%d", rt.Width()); WritePrivateProfileString("WINDOW", "width", strVal, szPath); strVal.Format("%d", rt.Height()); WritePrivateProfileString("WINDOW", "height", strVal, szPath); // save selected tab strVal.Format("%d", m_TabCtrl.GetCurSel()); WritePrivateProfileString("WINDOW", "selection", strVal, szPath); SaveColumns(m_ProjectListCtrl, "projects", szPath); SaveColumns(m_ResultListCtrl, "results", szPath); SaveColumns(m_XferListCtrl, "xfers", szPath); SaveColumns(m_MessageListCtrl, "messages", szPath); } static const int message_header_widths[] = { DEF_COL_WIDTH, DEF_COL_WIDTH*1.5, DEF_COL_WIDTH*4 }; ////////// // CMainWindow::LoadUserSettings // arguments: void // returns: void // function: loads relevant user settings from boinc ini file void CMainWindow::LoadUserSettings() { char szPath[256]; CString strKey; GetCurrentDirectory(256, szPath); strcat(szPath, "\\"); strcat(szPath, INI_FILE_NAME); int nBuf; // load window size/position CRect rt; nBuf = GetPrivateProfileInt("WINDOW", "xposition", 100, szPath); rt.left = nBuf; nBuf = GetPrivateProfileInt("WINDOW", "yposition", 100, szPath); rt.top = nBuf; nBuf = GetPrivateProfileInt("WINDOW", "width", 600, szPath); rt.right = nBuf + rt.left; nBuf = GetPrivateProfileInt("WINDOW", "height", 400, szPath); rt.bottom = nBuf + rt.top; CRect rtScreen, rtInt; HDC screenDC=::GetDC(NULL); rtScreen.left = rtScreen.top = 0; rtScreen.right=GetDeviceCaps(screenDC, HORZRES); rtScreen.bottom=GetDeviceCaps(screenDC, VERTRES); ::ReleaseDC(NULL, screenDC); rtInt.IntersectRect(&rt, &rtScreen); if(rtInt.IsRectEmpty()) { rt.SetRect(100, 100, 600, 400); } SetWindowPos(&wndNoTopMost, rt.left, rt.top, rt.Width(), rt.Height(), 0); // load selected tab nBuf = GetPrivateProfileInt("WINDOW", "selection", 0, szPath); ShowTab(nBuf); LoadColumns(m_ProjectListCtrl, "projects", szPath); LoadColumns(m_ResultListCtrl, "results", szPath); LoadColumns(m_XferListCtrl, "xfers", szPath); LoadColumns(m_MessageListCtrl, "messages", szPath, message_header_widths); } ////////// // CMainWindow::LoadLanguage // arguments: void // returns: void // function: loads new captions from language file void CMainWindow::LoadLanguage() { CString strPath; GetLanguageFilename(strPath); int col; CString strSection; // load column headers strSection.Format("HEADER-%s", g_szTabItems[PROJECT_ID]); for(col = 0; col < PROJECT_COLS; col ++) { GetPrivateProfileString(strSection, g_szColumnTitles[PROJECT_ID][col], g_szColumnTitles[PROJECT_ID][col], g_szColumnTitles[PROJECT_ID][col], 256, strPath); } GetPrivateProfileString(strSection, "Title", g_szTabItems[PROJECT_ID], g_szTabItems[PROJECT_ID], 16, strPath); strSection.Format("HEADER-%s", g_szTabItems[RESULT_ID]); for(col = 0; col < RESULT_COLS; col ++) { GetPrivateProfileString(strSection, g_szColumnTitles[RESULT_ID][col], g_szColumnTitles[RESULT_ID][col], g_szColumnTitles[RESULT_ID][col], 256, strPath); } GetPrivateProfileString(strSection, "Title", g_szTabItems[RESULT_ID], g_szTabItems[RESULT_ID], 16, strPath); strSection.Format("HEADER-%s", g_szTabItems[XFER_ID]); for(col = 0; col < XFER_COLS; col ++) { GetPrivateProfileString(strSection, g_szColumnTitles[XFER_ID][col], g_szColumnTitles[XFER_ID][col], g_szColumnTitles[XFER_ID][col], 256, strPath); } GetPrivateProfileString(strSection, "Title", g_szTabItems[XFER_ID], g_szTabItems[XFER_ID], 16, strPath); strSection.Format("HEADER-%s", g_szTabItems[MESSAGE_ID]); for(col = 0; col < MESSAGE_COLS; col ++) { GetPrivateProfileString(strSection, g_szColumnTitles[MESSAGE_ID][col], g_szColumnTitles[MESSAGE_ID][col], g_szColumnTitles[MESSAGE_ID][col], 256, strPath); } GetPrivateProfileString(strSection, "Title", g_szTabItems[MESSAGE_ID], g_szTabItems[MESSAGE_ID], 16, strPath); // load usage labels strSection.Format("HEADER-%s", g_szTabItems[USAGE_ID]); for(col = 0; col < MAX_USAGE_STR; col ++) { GetPrivateProfileString(strSection, g_szUsageItems[col], g_szUsageItems[col], g_szUsageItems[col], 256, strPath); } GetPrivateProfileString(strSection, "Title", g_szTabItems[USAGE_ID], g_szTabItems[USAGE_ID], 16, strPath); // load miscellaneous text strSection.Format("HEADER-MISC"); for(col = 0; col < MAX_MISC_STR; col ++) { GetPrivateProfileString(strSection, g_szMiscItems[col], g_szMiscItems[col], g_szMiscItems[col], 256, strPath); } // load menu items CString strItem, strItemNoAmp; char szItem[256]; int i, is; for(i = 0; i < m_MainMenu.GetMenuItemCount(); i ++) { m_MainMenu.GetMenuString(i, strItem, MF_BYPOSITION); strItemNoAmp = strItem; strItemNoAmp.Remove('&'); strSection.Format("MENU-%s", strItemNoAmp); GetPrivateProfileString(strSection, "Title", strItem, szItem, 256, strPath); m_MainMenu.ModifyMenu(i, MF_BYPOSITION|MF_STRING, 0, szItem); CMenu* pSubMenu = m_MainMenu.GetSubMenu(i); if(!pSubMenu) continue; for(is = 0; is < pSubMenu->GetMenuItemCount(); is ++) { pSubMenu->GetMenuString(is, strItem, MF_BYPOSITION); if (UpdateLanguageString(strSection, strItem, strPath)) { pSubMenu->ModifyMenu(is, MF_BYPOSITION|MF_STRING, pSubMenu->GetMenuItemID(is), strItem); } } } for(i = 0; i < m_ContextMenu.GetMenuItemCount(); i ++) { m_ContextMenu.GetMenuString(i, strItem, MF_BYPOSITION); strItemNoAmp = strItem; strItemNoAmp.Remove('&'); strSection.Format("MENU-%s", strItemNoAmp); GetPrivateProfileString(strSection, "Title", strItem, szItem, 256, strPath); m_ContextMenu.ModifyMenu(i, MF_BYPOSITION|MF_STRING, 0, szItem); CMenu* pSubMenu = m_ContextMenu.GetSubMenu(i); if(!pSubMenu) continue; for(is = 0; is < pSubMenu->GetMenuItemCount(); is ++) { pSubMenu->GetMenuString(is, strItem, MF_BYPOSITION); if (UpdateLanguageString(strSection, strItem, strPath)) { pSubMenu->ModifyMenu(is, MF_BYPOSITION|MF_STRING, pSubMenu->GetMenuItemID(is), strItem); } } } UpdateLanguageString("MENU-Project", m_MenuLabelGetPreferences, strPath); UpdateLanguageString("MENU-Project", m_MenuLabelRetryNow, strPath); UpdateLanguageString("DIALOG-RESET", m_DialogResetQuery, strPath); UpdateLanguageString("DIALOG-DETACH", m_DialogDetachQuery, strPath); } ////////// // CMainWindow::GetUserIdleTime // arguments: void // returns: time the user has been idle in milliseconds // function: calls a dll function to determine the the user's idle time DWORD CMainWindow::GetUserIdleTime() { if(m_hIdleDll) { typedef DWORD (CALLBACK* GetFn)(); GetFn fn; fn = (GetFn)GetProcAddress(m_hIdleDll, "IdleTrackerGetLastTickCount"); if(fn) { return GetTickCount() - fn(); } else { typedef void (CALLBACK* TermFn)(); TermFn tfn; tfn = (TermFn)GetProcAddress(m_hIdleDll, "IdleTrackerTerm"); if(tfn) { tfn(); } FreeLibrary(m_hIdleDll); m_hIdleDll = NULL; } } return 0; } ////////// // CMainWindow::Syncronize // arguments: pProg: pointer to a progress list control // pVect: pointer to a vector of pointers // returns: void // function: first, goes through the vector and adds items to the list // control for any pointers it does not already contain, then // goes through the list control and removes any pointers the // vector does not contain. void CMainWindow::Syncronize(CProgressListCtrl* pProg, vector* pVect) { int i, j; // add items to list that are not already in it for(i = 0; i < pVect->size(); i ++) { void* item = (*pVect)[i]; BOOL contained = false; for(j = 0; j < pProg->GetItemCount(); j ++) { if((DWORD)item == pProg->GetItemData(j)) { contained = true; break; } } if(!contained) { pProg->InsertItem(i, ""); pProg->SetItemData(i, (DWORD)item); } } // remove items from list that are not in vector // now just set the pointer to NULL but leave the item in the list for(i = 0; i < pProg->GetItemCount(); i ++) { DWORD item = pProg->GetItemData(i); BOOL contained = false; for(j = 0; j < pVect->size(); j ++) { if(item == (DWORD)(*pVect)[j]) { contained = true; break; } } if(!contained) { pProg->SetItemData(i, (DWORD)NULL); } } } ////////// // CMainWindow::PostNcDestroy // arguments: void // returns: void // function: takes care of window being destroyed void CMainWindow::PostNcDestroy() { delete this; } ////////// // CMainWindow::DefWindowProc // arguments: message: message received // wParam: message's wparam // lParam: message's lparam // returns: dependent on message // function: handles any messages not handled by the window previously LRESULT CMainWindow::DefWindowProc(UINT message, WPARAM wParam, LPARAM lParam) { unsigned long time_until_blank, blank_screen, x; if(m_nShowMsg == message) { ShowWindow(SW_SHOW); SetForegroundWindow(); return 0; } else if(m_nNetActivityMsg == message) { gstate.net_sleep(0); return 0; } else if(m_uEndSSMsg == message) { gstate.ss_logic.stop_ss(); return 0; } else if(m_uScreenSaverMsg == message) { // Get the current screen blanking information UtilGetRegKey(REG_BLANK_NAME, blank_screen); UtilGetRegKey(REG_BLANK_TIME, time_until_blank); if (blank_screen) { x = time(0) + time_until_blank*60; } else { x = 0; } gstate.ss_logic.start_ss(x); return 0; } return CWnd::DefWindowProc(message, wParam, lParam); } ////////// // CMainWindow::OnClose // arguments: void // returns: void // function: hides the window, keeps status icon void CMainWindow::OnClose() { ShowWindow(SW_HIDE); } ////////// // CMainWindow::OnDestroy // arguments: void // returns: void // function: quits the program void CMainWindow::OnDestroy() { OnCommandExit(); } ////////// // CMainWindow::OnCommandSettingsLogin // arguments: void // returns: void // function: shows the account login dialog box void CMainWindow::OnCommandSettingsLogin() { if(GetForegroundWindow() != this || !IsWindowVisible()) { return; } CLoginDialog dlg(IDD_LOGIN, "", ""); int nResult = dlg.DoModal(); if(nResult == IDOK) { gstate.add_project( dlg.m_strUrl.GetBuffer(dlg.m_strUrl.GetLength()+10), dlg.m_strAuth.GetBuffer(0)); dlg.m_strUrl.ReleaseBuffer(); // might have been changed by canonicalize_url() } } ////////// // CMainWindow::OnCommandSettingsProxyServer // arguments: void // returns: void // function: shows the proxy dialog box void CMainWindow::OnCommandSettingsProxyServer() { CProxyDialog dlg(IDD_PROXY); int nResult = dlg.DoModal(); } ////////// // CMainWindow::OnCommandHelpAbout // arguments: void // returns: void // function: shows the about dialog box void CMainWindow::OnCommandHelpAbout() { CAboutDialog dlg(IDD_ABOUTBOX); int nResult = dlg.DoModal(); } /* ////////// // CMainWindow::OnCommandFileClearInactive // arguments: void // returns: void // function: clears inactive items from lists void CMainWindow::OnCommandFileClearInactive() { int i; for(i = 0; i < m_ProjectListCtrl.GetItemCount();) { if(!m_ProjectListCtrl.GetItemData(i)) { m_ProjectListCtrl.DeleteItem(i); } else { i ++; } } for(i = 0; i < m_ResultListCtrl.GetItemCount();) { if(!m_ResultListCtrl.GetItemData(i)) { m_ResultListCtrl.DeleteItem(i); } else { i ++; } } for(i = 0; i < m_XferListCtrl.GetItemCount();) { if(!m_XferListCtrl.GetItemData(i)) { m_XferListCtrl.DeleteItem(i); } else { i ++; } } } ////////// // CMainWindow::OnCommandFileClearMessages // arguments: void // returns: void // function: clears messages void CMainWindow::OnCommandFileClearMessages() { m_MessageListCtrl.DeleteAllItems(); } */ ////////// // CMainWindow::GetProjectFromContextMenu // arguments: void // returns: PROJECT * // function: returns the project associated with the most // recently selected context menu item PROJECT* CMainWindow::GetProjectFromContextMenu() { if(m_nContextItem < 0 || m_nContextItem > m_ProjectListCtrl.GetItemCount()) return NULL; PROJECT* proj = (PROJECT*)m_ProjectListCtrl.GetItemData(m_nContextItem); m_nContextItem = -1; return proj; } ////////// // CMainWindow::OnCommandProjectWebSite // arguments: void // returns: void // function: lets the user quit a project void CMainWindow::OnCommandProjectWebSite() { PROJECT *proj; char url[263] = ""; proj = GetProjectFromContextMenu(); if (proj) { if (strncmp(proj->master_url, "http://", 7) != 0) // ***WARNING*** ASSUMING LOWER-CASE URL sprintf(url, "http://"); ShellExecute(GetSafeHwnd(), "open", strcat(url, proj->master_url), "", "", SW_SHOWNORMAL); } } ////////// // CMainWindow::OnCommandProjectGetPrefs // arguments: void // returns: void // function: reset rpc time to get prefences now void CMainWindow::OnCommandProjectGetPrefs() { PROJECT *proj; proj = GetProjectFromContextMenu(); if (proj) { proj->sched_rpc_pending = true; proj->min_rpc_time = 0; } } ////////// // CMainWindow::OnCommandProjectDetach // arguments: void // returns: void // function: lets the user quit a project void CMainWindow::OnCommandProjectDetach() { PROJECT *proj; proj = GetProjectFromContextMenu(); if (!proj) return; CString strBuf = m_DialogDetachQuery; strBuf.Replace("%1", proj->get_project_name()); if(AfxMessageBox(strBuf, MB_YESNO, 0) == IDYES) DetachProject(proj); } ////////// // CMainWindow::DetachProject // arguments: void // returns: void // function: detaches from a project and clears its items void CMainWindow::DetachProject(PROJECT *proj) { ClearProjectItems(proj->master_url); gstate.detach_project(proj); } ////////// // CMainWindow::OnCommandProjectReset // arguments: void // returns: void // function: lets the user quit a project void CMainWindow::OnCommandProjectReset() { PROJECT *proj; proj = GetProjectFromContextMenu(); if (!proj) return; CString strBuf = m_DialogResetQuery; strBuf.Replace("%1", proj->get_project_name()); if(AfxMessageBox(strBuf, MB_YESNO, 0) == IDYES) { gstate.reset_project(proj); } } ////////// // CMainWindow::OnCommandWorkShowGraphics // arguments: void // returns: void // function: brings up the graphics window for the selescted app void CMainWindow::OnCommandWorkShowGraphics() { RESULT* resToShow = (RESULT*)m_ResultListCtrl.GetItemData(m_nContextItem); if(resToShow) { ACTIVE_TASK* at = gstate.lookup_active_task_by_result(resToShow); if(at) { at->request_graphics_mode(MODE_WINDOW); } } } void CMainWindow::OnCommandTransfersRetryNow() { PERS_FILE_XFER* pfx = (PERS_FILE_XFER*)m_XferListCtrl.GetItemData(m_nContextItem); if (pfx) { pfx->next_request_time = 0; } } ////////// // CMainWindow::OnCommandShow // arguments: void // returns: void // function: shows the window void CMainWindow::OnCommandShow() { CMenu* pMainMenu; pMainMenu = GetMenu(); ShowWindow(SW_SHOW); pMainMenu->GetSubMenu(0)->EnableMenuItem(ID_STATUSICON_HIDE, MF_ENABLED); pMainMenu->GetSubMenu(0)->EnableMenuItem(ID_STATUSICON_SHOW, MF_GRAYED); } ////////// // CMainWindow::OnCommandHide // arguments: void // returns: void // function: hides the window void CMainWindow::OnCommandHide() { CMenu* pMainMenu; pMainMenu = GetMenu(); ShowWindow(SW_HIDE); pMainMenu->GetSubMenu(0)->EnableMenuItem(ID_STATUSICON_HIDE, MF_GRAYED); pMainMenu->GetSubMenu(0)->EnableMenuItem(ID_STATUSICON_SHOW, MF_ENABLED); } void CMainWindow::UpdateRunRequestMenu(CMenu* pMenu) { if (!pMenu) return; /*switch(gstate.user_run_request) { case USER_RUN_REQUEST_ALWAYS: pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_ALWAYS, MF_GRAYED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_AUTO, MF_ENABLED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_NEVER, MF_ENABLED); break; case USER_RUN_REQUEST_AUTO: pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_ALWAYS, MF_ENABLED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_AUTO, MF_GRAYED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_NEVER, MF_ENABLED); break; case USER_RUN_REQUEST_NEVER: pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_ALWAYS, MF_ENABLED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_AUTO, MF_ENABLED); pMenu->EnableMenuItem(ID_FILE_RUN_REQUEST_NEVER, MF_GRAYED); break; }*/ // NOTE: // ID_FILE_RUN_REQUEST_ALWAYS, ID_FILE_RUN_REQUEST_AUTO, ID_FILE_RUN_REQUEST_NEVER // and // USER_RUN_REQUEST_ALWAYS, USER_RUN_REQUEST_AUTO, USER_RUN_REQUEST_NEVER // must be #defined sequentially pMenu->CheckMenuRadioItem(ID_FILE_RUN_REQUEST_ALWAYS, ID_FILE_RUN_REQUEST_NEVER, ID_FILE_RUN_REQUEST_ALWAYS+gstate.user_run_request-USER_RUN_REQUEST_ALWAYS, MF_BYCOMMAND); } void CMainWindow::UpdateRunRequestFileMenu() { UpdateRunRequestMenu(GetMenu()->GetSubMenu(0)); } void CMainWindow::OnCommandRunRequestAlways() { gstate.user_run_request = USER_RUN_REQUEST_ALWAYS; UpdateRunRequestFileMenu(); } void CMainWindow::OnCommandRunRequestAuto() { gstate.user_run_request = USER_RUN_REQUEST_AUTO; UpdateRunRequestFileMenu(); } void CMainWindow::OnCommandRunRequestNever() { gstate.user_run_request = USER_RUN_REQUEST_NEVER; UpdateRunRequestFileMenu(); } void CMainWindow::OnCommandRunBenchmarks() { gstate.fork_run_cpu_benchmarks(); } void CMainWindow::OnBenchmarksBegin() { CMenu* pMainMenu; CMenu* pFileMenu; pMainMenu = GetMenu(); if(pMainMenu) { pFileMenu = pMainMenu->GetSubMenu(0); } if(pFileMenu) { pFileMenu->EnableMenuItem(ID_FILE_RUN_BENCHMARKS, MF_GRAYED); } } void CMainWindow::OnBenchmarksEnd() { CMenu* pMainMenu; CMenu* pFileMenu; pMainMenu = GetMenu(); if(pMainMenu) { pFileMenu = pMainMenu->GetSubMenu(0); } if(pFileMenu) { pFileMenu->EnableMenuItem(ID_FILE_RUN_BENCHMARKS, MF_ENABLED); } } ////////// // CMainWindow::OnCommandExit // arguments: void // returns: void // function: cleans up, closes and quits everything void CMainWindow::OnCommandExit() { static bool already_exited = false; if (already_exited) return; already_exited = true; // quit gstate.cleanup_and_exit(); PostQuitMessage(0); KillTimer(m_nGuiTimerID); // status icon in taskbar SetStatusIcon(ICON_OFF); // clean up and delete objects m_Font.DeleteObject(); m_TabBMP[0].DeleteObject(); m_TabBMP[1].DeleteObject(); m_TabBMP[2].DeleteObject(); m_TabBMP[3].DeleteObject(); m_TabBMP[4].DeleteObject(); m_TabIL.DeleteImageList(); m_MainMenu.DestroyMenu(); m_ContextMenu.DestroyMenu(); // free dll and idle detection if(m_hIdleDll) { typedef void (CALLBACK* TermFn)(); TermFn fn; fn = (TermFn)GetProcAddress(m_hIdleDll, "IdleTrackerTerm"); if(!fn) { show_message(NULL, "Error in DLL \"boinc.dll\"", MSG_INFO); } else { fn(); } FreeLibrary(m_hIdleDll); m_hIdleDll = NULL; } SaveUserSettings(); //SaveListControls(); // This was causing trouble in the installer because the SS window // would recieve the WM_DESTROY message first, thereby causing // this delete to access already freed memory //delete m_pSSWnd; CWnd::OnClose(); } ////////// // CMainWindow::OnCommandMessageCopyToClip // arguments: void // returns: 0 if successful, otherwise -1 // function: copies the selected message(s) to the clipboard as // formatted text int CMainWindow::OnCommandMessageCopyToClip() { HGLOBAL hClipboardData; POSITION pos; char * pchData; const int NUM_COLS = 3; CString strData = ""; if (!OpenClipboard()) return -1; EmptyClipboard(); /* Get strings from selected items: - If GetFirstSelectedItem() returns null, the user right-clicked a single item without selecting it first. Use click location to determine the chosen item. - Otherwise, iterate through the selected items. */ pos = m_MessageListCtrl.GetFirstSelectedItemPosition(); if (pos == NULL) { if(m_nContextItem < 0 || m_nContextItem > m_MessageListCtrl.GetItemCount()) return -1; // TODO: Allow arbitrary # of columns instead of hardcoding 3, if possible. for (int i = 0; i < NUM_COLS-1; i++) { strData += m_MessageListCtrl.GetItemTextOrPos(m_nContextItem, i); strData += " - "; } // Add last column here to avoid concatenating an extra delimiter to the end. // NOTE: Carriage-return/Newline combination is required to separate lines // in the CF_TEXT clipboard format. strData += m_MessageListCtrl.GetItemTextOrPos(m_nContextItem, i) + "\r\n"; } else { while (pos) { int nItem = m_MessageListCtrl.GetNextSelectedItem(pos); for (int i = 0; i < NUM_COLS-1; i++) { strData += m_MessageListCtrl.GetItemTextOrPos(nItem, i); strData += " - "; } strData += m_MessageListCtrl.GetItemTextOrPos(nItem, i) + "\r\n"; } } // allocate a global memory block to hold the copied text // (size is +1 to account for the null terminator). hClipboardData = GlobalAlloc(GMEM_DDESHARE, strData.GetLength()+1); pchData = (char*)GlobalLock(hClipboardData); strcpy(pchData, LPCSTR(strData)); GlobalUnlock(hClipboardData); SetClipboardData(CF_TEXT,hClipboardData); CloseClipboard(); return 0; } ////////// // CMainWindow::OnCreate // arguments: lpcs: a pointer to the create structure // returns: 0 if successful, otherwise -1 // function: sets window's global variable, loads resource, creates child // windows, and initializes client state and timer int CMainWindow::OnCreate(LPCREATESTRUCT lpcs) { char curDir[512]; char* szTitles[MAX_COLS]; int i; if (CWnd::OnCreate(lpcs) == -1) { return -1; } // Determine the OS version UtilInitOSVersion(); g_myWnd = this; m_nIconState = ICON_OFF; m_bMessage = false; m_bRequest = false; m_nContextItem = -1; m_pSSWnd = new CSSWindow(); // load menus m_ContextMenu.LoadMenu(IDR_CONTEXT); m_MainMenu.LoadMenu(IDR_MAINFRAME); SetMenu(&m_MainMenu); LoadLanguage(); // create project list control m_ProjectListCtrl.Create(LVS_REPORT|WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, PROJECT_ID); m_ProjectListCtrl.SetExtendedStyle(m_ProjectListCtrl.GetExtendedStyle()|LVS_EX_HEADERDRAGDROP|LVS_EX_FULLROWSELECT); for(i = 0; i < MAX_COLS; i ++) szTitles[i] = g_szColumnTitles[PROJECT_ID][i]; m_ProjectListCtrl.SetMenuItems(szTitles, PROJECT_COLS); for(i = 0; i < PROJECT_COLS; i ++) { m_ProjectListCtrl.InsertColumn(i, g_szColumnTitles[PROJECT_ID][i], LVCFMT_LEFT, DEF_COL_WIDTH, -1); } // create result list control m_ResultListCtrl.Create(LVS_REPORT|WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, RESULT_ID); m_ResultListCtrl.SetExtendedStyle(m_ResultListCtrl.GetExtendedStyle()|LVS_EX_HEADERDRAGDROP|LVS_EX_FULLROWSELECT); m_ResultListCtrl.ModifyStyle(WS_VISIBLE, 0); for(i = 0; i < MAX_COLS; i ++) szTitles[i] = g_szColumnTitles[RESULT_ID][i]; m_ResultListCtrl.SetMenuItems(szTitles, RESULT_COLS); for(i = 0; i < RESULT_COLS; i ++) { m_ResultListCtrl.InsertColumn(i, g_szColumnTitles[RESULT_ID][i], LVCFMT_LEFT, DEF_COL_WIDTH, -1); } // create xfer list control m_XferListCtrl.Create(LVS_REPORT|WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, XFER_ID); m_XferListCtrl.SetExtendedStyle(m_XferListCtrl.GetExtendedStyle()|LVS_EX_HEADERDRAGDROP|LVS_EX_FULLROWSELECT); m_XferListCtrl.ModifyStyle(WS_VISIBLE, 0); for(i = 0; i < MAX_COLS; i ++) szTitles[i] = g_szColumnTitles[XFER_ID][i]; m_XferListCtrl.SetMenuItems(szTitles, XFER_COLS); for(i = 0; i < XFER_COLS; i ++) { m_XferListCtrl.InsertColumn(i, g_szColumnTitles[XFER_ID][i], LVCFMT_LEFT, DEF_COL_WIDTH, -1); } // create message edit control m_MessageListCtrl.Create(LVS_REPORT|WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, MESSAGE_ID); m_MessageListCtrl.SetExtendedStyle(m_MessageListCtrl.GetExtendedStyle()|LVS_EX_HEADERDRAGDROP|LVS_EX_FULLROWSELECT); m_MessageListCtrl.ModifyStyle(WS_VISIBLE, 0); for(i = 0; i < MAX_COLS; i ++) szTitles[i] = g_szColumnTitles[MESSAGE_ID][i]; m_MessageListCtrl.SetMenuItems(szTitles, MESSAGE_COLS); for(i = 0; i < MESSAGE_COLS; i ++) { int width = DEF_COL_WIDTH; if(i == 1) width *= 1.5; if(i == 2) width *= 4; m_MessageListCtrl.InsertColumn(i, g_szColumnTitles[MESSAGE_ID][i], LVCFMT_LEFT, width, -1); } // create usage pie control m_UsagePieCtrl.Create(WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, USAGE_ID); m_UsagePieCtrl.ModifyStyle(WS_VISIBLE, 0); m_UsagePieCtrl.AddPiece(g_szUsageItems[0], RGB(255, 0, 255), 0); m_UsagePieCtrl.AddPiece(g_szUsageItems[1], RGB(192, 64, 192), 0); m_UsagePieCtrl.AddPiece(g_szUsageItems[2], RGB(0, 0, 255), 0); m_UsageBOINCPieCtrl.Create(WS_CHILD|WS_BORDER|WS_VISIBLE, CRect(0,0,0,0), this, USAGE_ID); m_UsageBOINCPieCtrl.ModifyStyle(WS_VISIBLE, 0); m_UsageBOINCPieCtrl.AddPiece(g_szUsageItems[3], GetPieColor(0), 0); // set up image list for tab control m_TabIL.Create(16, 16, ILC_COLOR8|ILC_MASK, MAX_TABS, 1); m_TabBMP[0].LoadBitmap(IDB_PROJ); m_TabIL.Add(&m_TabBMP[0], RGB(255, 0, 255)); m_TabBMP[1].LoadBitmap(IDB_RESULT); m_TabIL.Add(&m_TabBMP[1], RGB(255, 0, 255)); m_TabBMP[2].LoadBitmap(IDB_XFER); m_TabIL.Add(&m_TabBMP[2], RGB(255, 0, 255)); m_TabBMP[3].LoadBitmap(IDB_MESS); m_TabIL.Add(&m_TabBMP[3], RGB(255, 0, 255)); m_TabBMP[4].LoadBitmap(IDB_USAGE); m_TabIL.Add(&m_TabBMP[4], RGB(255, 0, 255)); // create tab control //m_TabCtrl.Create(TCS_FIXEDWIDTH|TCS_BUTTONS|TCS_FLATBUTTONS|TCS_FOCUSNEVER|WS_CHILD|WS_VISIBLE, CRect(0,0,0,0), this, TAB_ID); m_TabCtrl.Create(TCS_FIXEDWIDTH|TCS_FOCUSNEVER|WS_CHILD|WS_VISIBLE, CRect(0,0,0,0), this, TAB_ID); m_TabCtrl.SetImageList(&m_TabIL); m_TabCtrl.InsertItem(1, g_szTabItems[0], 0); m_TabCtrl.InsertItem(2, g_szTabItems[1], 1); m_TabCtrl.InsertItem(3, g_szTabItems[2], 2); m_TabCtrl.InsertItem(4, g_szTabItems[3], 3); m_TabCtrl.InsertItem(5, g_szTabItems[4], 4); // make all fonts the same nice font CFont* pFont; pFont = m_ProjectListCtrl.GetFont(); LOGFONT lf; ZeroMemory(&lf, sizeof(LOGFONT)); pFont->GetLogFont(&lf); m_Font.CreateFontIndirect(&lf); m_TabCtrl.SetFont(&m_Font); m_UsagePieCtrl.SetFont(&m_Font); m_UsageBOINCPieCtrl.SetFont(&m_Font); // Set the current directory to the default UtilGetRegStr("ClientDir", curDir); if (strlen(curDir)) SetCurrentDirectory(curDir); // add status icon to taskbar SetStatusIcon(ICON_NORMAL); // take care of other things // // Redirect stdout and stderr to files freopen(STDOUT_FILE_NAME, "w", stdout); freopen(STDERR_FILE_NAME, "w", stderr); // Check what (if any) activities should be logged read_log_flags(); LoadUserSettings(); //LoadListControls(); LPSTR command_line; char* argv[100]; int argc; int retval = gstate.init(); if (retval) { OnCommandExit(); return 0; } command_line = GetCommandLine(); argc = parse_command_line( command_line, argv ); gstate.parse_cmdline(argc, argv); m_nGuiTimerID = SetTimer(GUI_TIMER, GUI_WAIT, (TIMERPROC) NULL); // load dll and start idle detection m_hIdleDll = LoadLibrary("boinc.dll"); if(!m_hIdleDll) { show_message(NULL,"Can't load \"boinc.dll\", will not be able to determine idle time", MSG_ERROR); } else { typedef BOOL (CALLBACK* InitFn)(); InitFn fn; fn = (InitFn)GetProcAddress(m_hIdleDll, "IdleTrackerInit"); if(!fn) { show_message(NULL,"Error in DLL \"boinc.dll\", will not be able to determine idle time", MSG_INFO); FreeLibrary(m_hIdleDll); m_hIdleDll = NULL; } else { if(!fn()) { show_message(NULL,"Error in DLL \"boinc.dll\", will not be able to determine idle time", MSG_INFO); FreeLibrary(m_hIdleDll); m_hIdleDll = NULL; } } } UpdateGUI(&gstate); // see if we need to add this to startup if(gstate.global_prefs.run_on_startup) { UtilGetRegStr("ClientPath", curDir); if(strlen(curDir)) { strcat(curDir, " -min"); UtilSetRegStartupStr("boincclient", curDir); } } else { UtilSetRegStartupStr("boincclient", ""); } // see if we need to hide the window if(gstate.global_prefs.run_minimized || gstate.start_saver) { ShowWindow(SW_HIDE); } else { ShowWindow(SW_SHOW); } UpdateRunRequestFileMenu(); return 0; } ////////// // CMainWindow::OnNotify // arguments: wParam: notification's wparam // lParam: notification's lparam // pResult: pointer to result of notification // returns: true if the notification is processed, otherwise false // function: handles notifications from children, including: // user selecting a new tab sets display to selected tab's control BOOL CMainWindow::OnNotify(WPARAM wParam, LPARAM lParam, LRESULT* pResult) { HD_NOTIFY* phdn = (HD_NOTIFY*)lParam; // notification from tab control, user is changing the selection if(phdn->hdr.code == TCN_SELCHANGE) { int newTab = m_TabCtrl.GetCurSel(); ShowTab(newTab); } return CWnd::OnNotify(wParam, lParam, pResult); } ////////// // CMainWindow::OnRButtonDown // arguments: nFlags: message flags (keys down) // point: mouse's point // returns: void // function: shows context menu for list items void CMainWindow::OnRButtonDown(UINT nFlags, CPoint point) { CMenu* pContextMenu = NULL; CRect rt; CListCtrl* pMenuCtrl = NULL; int nMenuId = -1; GetCursorPos(&point); if(m_ProjectListCtrl.IsWindowVisible()) { pMenuCtrl = &m_ProjectListCtrl; nMenuId = PROJECT_MENU; } else if(m_ResultListCtrl.IsWindowVisible()) { pMenuCtrl = &m_ResultListCtrl; nMenuId = RESULT_MENU; } else if(m_XferListCtrl.IsWindowVisible()) { pMenuCtrl = &m_XferListCtrl; nMenuId = XFER_MENU; } else if(m_MessageListCtrl.IsWindowVisible()) { pMenuCtrl = &m_MessageListCtrl; nMenuId = MESSAGE_MENU; } // " pMenuCtrl->OnLButtonDown(nFlags, point) " /*{ _AFX_THREAD_STATE* pThreadState = _afxThreadState.GetData(); pMenuCtrl->SendMessage(WM_LBUTTONUP, pThreadState->m_lastSentMsg.wParam, pThreadState->m_lastSentMsg.lParam); pMenuCtrl->SendMessage(WM_LBUTTONDOWN, pThreadState->m_lastSentMsg.wParam, pThreadState->m_lastSentMsg.lParam); pMenuCtrl->SendMessage(WM_LBUTTONUP, pThreadState->m_lastSentMsg.wParam, pThreadState->m_lastSentMsg.lParam); pMenuCtrl->SendMessage(WM_LBUTTONDOWN, nFlags, MAKELPARAM(point.x,point.y)); }*/ if(pMenuCtrl) { pMenuCtrl->SetFocus(); // clear all other items - this must be done first int indexSelected = -1; for(int i = 0; i < pMenuCtrl->GetItemCount(); i ++) { pMenuCtrl->GetItemRect(i, &rt, LVIR_BOUNDS); pMenuCtrl->ClientToScreen(&rt); if(rt.PtInRect(point)) { indexSelected = i; break; } } if (indexSelected == -1) return; bool shouldSetNewSelection = nMenuId != MESSAGE_MENU || !pMenuCtrl->GetItemState(indexSelected, LVIS_SELECTED); if(shouldSetNewSelection) { for(int i = 0; i < pMenuCtrl->GetItemCount(); i ++) { pMenuCtrl->SetItemState(i, 0, LVIS_SELECTED); } pMenuCtrl->SetItemState(indexSelected, LVIS_SELECTED, LVIS_SELECTED); } pContextMenu = m_ContextMenu.GetSubMenu(nMenuId); if(pContextMenu) { switch(nMenuId) { case PROJECT_MENU: { // if we are backing off, show "retry now", else "get preferences" PROJECT *proj = (PROJECT *)pMenuCtrl->GetItemData(indexSelected); pContextMenu->ModifyMenu(ID_PROJECT_GET_PREFS, 0, ID_PROJECT_GET_PREFS, ((proj && proj->min_rpc_time > time(0)) ? m_MenuLabelRetryNow : m_MenuLabelGetPreferences)); break; } case RESULT_MENU: { pContextMenu->EnableMenuItem(ID_WORK_SHOWGRAPHICS,MF_GRAYED); //disable RESULT *rp = (RESULT *)pMenuCtrl->GetItemData(indexSelected); if (rp) { ACTIVE_TASK *atp = gstate.lookup_active_task_by_result(rp); if (atp && atp->supports_graphics()) pContextMenu->EnableMenuItem(ID_WORK_SHOWGRAPHICS,MF_ENABLED); // enable } break; } case XFER_MENU: { // enable "retry now" only if currently waiting to retry PERS_FILE_XFER* pfx = (PERS_FILE_XFER*)m_XferListCtrl.GetItemData(indexSelected); pContextMenu->EnableMenuItem(ID_TRANSFERS_RETRYNOW, pfx && (pfx->next_request_time > time(0) ? MF_ENABLED : MF_GRAYED)); break; } } pContextMenu->TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON, point.x, point.y, this); m_nContextItem = indexSelected; } } //CWnd::OnRButtonDown(nFlags, point); } ////////// // CMainWindow::OnFocus // arguments: pOldWnd: pointer to previous window that had focus // returns: void // function: if there is a message for the user when this window // gets the focus, selects the message tab void CMainWindow::OnSetFocus(CWnd* pOldWnd) { if(m_TabCtrl.GetSafeHwnd() && m_bMessage) { ShowTab(MESSAGE_ID); m_bMessage = false; SetStatusIcon(ICON_NORMAL); } // TODO: review this if(m_bRequest) { m_bRequest = false; // if(RequestNetConnect()) OnCommandConnectionConnectNow(); } UpdateGUI(&gstate); } ////////// // CMainWindow::OnSize // arguments: nType: type of resizing // cx: new width of window // cy: new height of window // returns: void // function: calculates new rectangles for child windows and resizes // them appropriately void CMainWindow::OnSize(UINT nType, int cx, int cy) { CWnd::OnSize(nType, cx, cy); // calculate the main rect for the tab control RECT rt = {EDGE_BUFFER, TOP_BUFFER, cx-EDGE_BUFFER, cy-EDGE_BUFFER*2}; RECT irt = {0, 0, 0, 0}; if(m_TabCtrl.GetSafeHwnd()) { m_TabCtrl.MoveWindow(&rt, false); m_TabCtrl.GetItemRect(0, &irt); // calculate the rects for other controls inside the tab control RECT srt = {rt.left+EDGE_BUFFER, irt.bottom+EDGE_BUFFER*2+TOP_BUFFER, rt.right-EDGE_BUFFER, rt.bottom-EDGE_BUFFER}; if(m_ProjectListCtrl.GetSafeHwnd()) { m_ProjectListCtrl.MoveWindow(&srt, false); m_ProjectListCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } if(m_ResultListCtrl.GetSafeHwnd()) { m_ResultListCtrl.MoveWindow(&srt, false); m_ResultListCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } if(m_XferListCtrl.GetSafeHwnd()) { m_XferListCtrl.MoveWindow(&srt, false); m_XferListCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } if(m_MessageListCtrl.GetSafeHwnd()) { m_MessageListCtrl.MoveWindow(&srt, false); m_MessageListCtrl.Scroll(CSize(0, m_MessageListCtrl.ApproximateViewRect().cy)); m_MessageListCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } if(m_UsagePieCtrl.GetSafeHwnd()) { RECT mrt = {rt.left+EDGE_BUFFER, irt.bottom+EDGE_BUFFER*2+TOP_BUFFER, rt.right/2, rt.bottom-EDGE_BUFFER}; m_UsagePieCtrl.MoveWindow(&mrt, false); m_UsagePieCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_NOERASE|RDW_FRAME); } if(m_UsageBOINCPieCtrl.GetSafeHwnd()) { RECT brt = {rt.right/2-1, irt.bottom+EDGE_BUFFER*2+TOP_BUFFER, rt.right-EDGE_BUFFER, rt.bottom-EDGE_BUFFER}; m_UsageBOINCPieCtrl.MoveWindow(&brt, false); m_UsageBOINCPieCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_NOERASE|RDW_FRAME); } m_TabCtrl.RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_NOERASE|RDW_FRAME); RedrawWindow(NULL,NULL,RDW_INVALIDATE|RDW_UPDATENOW|RDW_ERASE|RDW_FRAME); } } ////////// // CMainWindow::OnStatusIcon // arguments: wParam: id of icon clicked // lParam: message from icon // returns: true if the menu is shown, false otherwise // function: handles messages from status icon, including: // right click: shows popup menu // double click: alternates visibility of window LRESULT CMainWindow::OnStatusIcon(WPARAM wParam, LPARAM lParam) { if(lParam == WM_RBUTTONDOWN) { CPoint point; SetForegroundWindow(); GetCursorPos(&point); CMenu* pSubmenu; pSubmenu = m_ContextMenu.GetSubMenu(STATUS_MENU); UpdateRunRequestMenu(pSubmenu); if (IsWindowVisible()) { pSubmenu->EnableMenuItem(ID_STATUSICON_SHOW, MF_GRAYED); pSubmenu->EnableMenuItem(ID_STATUSICON_HIDE, MF_ENABLED); } else { pSubmenu->EnableMenuItem(ID_STATUSICON_SHOW, MF_ENABLED); pSubmenu->EnableMenuItem(ID_STATUSICON_HIDE, MF_GRAYED); } pSubmenu->TrackPopupMenu(TPM_LEFTALIGN|TPM_RIGHTBUTTON, point.x, point.y, this); } else if(lParam == WM_LBUTTONDOWN) { if(IsWindowVisible()) { SetForegroundWindow(); } } else if(lParam == WM_LBUTTONDBLCLK) { if(IsWindowVisible()) { ShowWindow(SW_HIDE); } else { ShowWindow(SW_SHOWNORMAL); } } return TRUE; } ////////// // CMainWindow::CheckIdle // arguments: void // returns: void // function: check user's idle time for suspension of apps void CMainWindow::CheckIdle() { if (gstate.global_prefs.idle_time_to_run > 0) { //msg_printf(NULL, "idle seconds: %d\n", GetUserIdleTime() / 1000); if (GetUserIdleTime() / 1000 > 60 * gstate.global_prefs.idle_time_to_run) { gstate.user_idle = true; } else { gstate.user_idle = false; } } else { gstate.user_idle = true; } } ////////// // CMainWindow::OnTimer // arguments: uEventID: timer's id // returns: void // function: checks idle time, updates client state, flushed output streams, // and updates gui display. void CMainWindow::OnTimer(UINT uEventID) { if(uEventID == m_nGuiTimerID) { // stop the timer while we do processing KillTimer(m_nGuiTimerID); // update state and gui while(gstate.do_something()); NetCheck(); // check if network connection can be terminated if (gstate.user_run_request == USER_RUN_REQUEST_NEVER) { // user suspended - don't bother checking idle } else if (gstate.is_suspended()) { // otherwise suspended, possibly due to not being idle CheckIdle(); } else { // active CheckIdle(); gstate.trunc_stderr_stdout(); } if (!gstate.is_suspended() || !gstate.was_previously_suspended()) { UpdateGUI(&gstate); } // Start the timer again m_nGuiTimerID = SetTimer(GUI_TIMER, GUI_WAIT, (TIMERPROC) NULL); } } void create_curtain() { g_myWnd->m_pSSWnd->ShowSSWindow(true); } void delete_curtain() { g_myWnd->m_pSSWnd->ShowSSWindow(false); } void project_add_failed(PROJECT* project) { char buf[512]; if (project->scheduler_urls.size()) { sprintf( buf, "BOINC failed to log in to %s.\n" "Please check your account ID and try again.", project->master_url ); } else { sprintf( buf, "BOINC couldn't get main page for %s.\n" "Please check the URL and try again.", project->master_url ); } AfxMessageBox(buf); g_myWnd->DetachProject(project); // TODO: To be filled in } void guiOnBenchmarksBegin() { g_myWnd->OnBenchmarksBegin(); } void guiOnBenchmarksEnd() { g_myWnd->OnBenchmarksEnd(); }