// ToDoListWnd.cpp : implementation file
//
#include "stdafx.h"
#include "ToDoList.h"
#include "ToDoListWnd.h"
#include "ToolsCmdlineParser.h"
#include "ToolsUserInputDlg.h"
#include "Toolshelper.h"
#include "tdlexportDlg.h"
#include "tasklisthtmlexporter.h"
#include "tasklisttxtexporter.h"
#include "tdcmsg.h"
#include "tdlschemadef.h"
#include "tdlprintdialog.h"
#include "tdltransformdialog.h"
#include "tdstringres.h"
#include "tdlcolumnselectiondlg.h"
#include "tdlfilterdlg.h"
#include "OffsetDatesDlg.h"
#include "KeyboardShortcutDisplayDlg.h"
#include "tdlimportdialog.h"
#include "tdlsetreminderdlg.h"
#include "tdlshowreminderdlg.h"
#include "tdlmultisortdlg.h"
#include "tdladdloggedtimedlg.h"
#include "multitaskfile.h"
#include "tdcstatic.h"
#include "tdlcustomattributedlg.h"
#include "tdccustomattributehelper.h"
#include "welcomewizard.h"
#include "..\shared\aboutdlg.h"
#include "..\shared\holdredraw.h"
#include "..\shared\autoflag.h"
#include "..\shared\enbitmap.h"
#include "..\shared\spellcheckdlg.h"
#include "..\shared\encolordialog.h"
#include "..\shared\winclasses.h"
#include "..\shared\wclassdefines.h"
#include "..\shared\datehelper.h"
#include "..\shared\osversion.h"
#include "..\shared\enfiledialog.h"
#include "..\shared\misc.h"
#include "..\shared\graphicsmisc.h"
#include "..\shared\filemisc.h"
#include "..\shared\themed.h"
#include "..\shared\enstring.h"
#include "..\shared\fileregister.h"
#include "..\shared\mousewheelMgr.h"
#include "..\shared\editshortcutMgr.h"
#include "..\shared\dlgunits.h"
#include "..\shared\passworddialog.h"
#include "..\shared\sysimagelist.h"
#include "..\shared\regkey.h"
#include "..\shared\uiextensionuihelper.h"
#include "..\shared\lightbox.h"
#include "..\shared\remotefile.h"
#include "..\shared\serverdlg.h"
#include "..\shared\focuswatcher.h"
#include "..\shared\localizer.h"
#include "..\shared\outlookhelper.h"
#include "..\3rdparty\gui.h"
#include "..\3rdparty\sendfileto.h"
#include
#include
#include
#include
#include // for WM_KICKIDLE
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/
// CToDoListWnd dialog
// popup menus
enum { TRAYICON, TASKCONTEXT, TABCTRLCONTEXT, HEADERCONTEXT };
enum { TB_TOOLBARHIDDEN, TB_DUMMY, TB_TOOLBARANDMENU };
enum { FILEALL, NEWTASK, EDITTASK, VIEW, MOVE, SORTTASK, TOOLS, HELP };
const int TD_VERSION = 31000;
const int BEVEL = 2; // pixels
const int BORDER = 3; // pixels
const int TB_VOFFSET = 4;
const int ONE_MINUTE = 60000; // milliseconds
const LPCTSTR UPDATE_SCRIPT_PATH = _T("http://www.abstractspoon.com/todolist_update.txt");
const LPCTSTR UPDATE_SCRIPT_PATH_MANUAL = _T("http://www.abstractspoon.com/todolist_update_manual.txt");
const LPCTSTR UPDATE_SCRIPT_PATH_STAGING = _T("http://www.abstractspoon.com/todolist_update_manual_staging.txt");
const LPCTSTR PREF_KEY = _T("Preferences");
#define DOPROGRESS(stringID) \
CWaitCursor cursor; \
CStatusBarProgressProxy prog(&m_sbProgress, m_statusBar, CEnString(stringID));
enum
{
WM_POSTONCREATE = (WM_APP+1),
WM_WEBUPDATEWIZARD,
WM_ADDTOOLBARTOOLS,
WM_APPRESTOREFOCUS,
};
enum
{
TIMER_READONLYSTATUS = 1,
TIMER_TIMESTAMPCHANGE,
TIMER_AUTOSAVE,
TIMER_CHECKOUTSTATUS,
TIMER_DUEITEMS,
TIMER_TIMETRACKING,
TIMER_AUTOMINIMIZE,
};
enum
{
INTERVAL_READONLYSTATUS = 1000,
INTERVAL_TIMESTAMPCHANGE = 10000,
// INTERVAL_AUTOSAVE, // dynamically determined
INTERVAL_CHECKOUTSTATUS = 5000,
INTERVAL_DUEITEMS = ONE_MINUTE,
INTERVAL_TIMETRACKING = 5000,
// INTERVAL_AUTOMINIMIZE, // dynamically determined
};
/
CToDoListWnd::CToDoListWnd() : CFrameWnd(),
m_bVisible(-1),
m_mruList(0, _T("MRU"), _T("TaskList%d"), 16, AFX_ABBREV_FILENAME_LEN, CEnString(IDS_RECENTFILES)),
m_nLastSelItem(-1),
m_nMaxState(TDCMS_NORMAL),
m_nPrevMaxState(TDCMS_NORMAL),
m_bShowFilterBar(TRUE),
m_bShowStatusBar(TRUE),
m_bInNewTask(FALSE),
m_bSaving(FALSE),
m_mgrShortcuts(FALSE),
m_pPrefs(NULL),
m_bClosing(FALSE),
m_mgrToDoCtrls(m_tabCtrl),
m_bFindShowing(FALSE),
m_bShowProjectName(TRUE),
m_bQueryOpenAllow(FALSE),
m_bPasswordPrompting(TRUE),
m_bShowToolbar(TRUE),
m_bReloading(FALSE),
m_hIcon(NULL),
m_hwndLastFocus(NULL),
m_bStartHidden(FALSE),
m_cbQuickFind(ACBS_ALLOWDELETE | ACBS_ADDTOSTART),
m_bShowTasklistBar(TRUE),
m_bShowTreeListBar(TRUE),
m_bEndingSession(FALSE),
m_nContextColumnID(TDCC_NONE)
{
// must do this before initializing any controls
SetupUIStrings();
// delete temporary web-update-wizard
CString sWuwPathTemp = FileMisc::GetAppFolder() + _T("\\WebUpdateSvc2.exe");
::DeleteFile(sWuwPathTemp);
// init preferences
ResetPrefs();
CFilteredToDoCtrl::EnableExtendedSelection(FALSE, TRUE);
m_bAutoMenuEnable = FALSE;
}
CToDoListWnd::~CToDoListWnd()
{
delete m_pPrefs;
if (m_hIcon)
DestroyIcon(m_hIcon);
// cleanup temp files
// Note: Due task notifications are removed by CToDoCtrlMgr
::DeleteFile(FileMisc::GetTempFileName(_T("ToDoList.print"), _T("html")));
::DeleteFile(FileMisc::GetTempFileName(_T("ToDoList.import"), _T("txt")));
::DeleteFile(FileMisc::GetTempFileName(_T("tdt")));
}
BEGIN_MESSAGE_MAP(CToDoListWnd, CFrameWnd)
//{{AFX_MSG_MAP(CToDoListWnd)
ON_WM_QUERYOPEN()
ON_COMMAND(ID_ADDTIMETOLOGFILE, OnAddtimetologfile)
ON_COMMAND(ID_ARCHIVE_SELECTEDTASKS, OnArchiveSelectedTasks)
ON_COMMAND(ID_CLOSEALLBUTTHIS, OnCloseallbutthis)
ON_COMMAND(ID_EDIT_COPYAS_DEPEND, OnCopyTaskasDependency)
ON_COMMAND(ID_EDIT_COPYAS_DEPENDFULL, OnCopyTaskasDependencyFull)
ON_COMMAND(ID_EDIT_COPYAS_PATH, OnCopyTaskasPath)
ON_COMMAND(ID_EDIT_COPYAS_LINK, OnCopyTaskasLink)
ON_COMMAND(ID_EDIT_COPYAS_LINKFULL, OnCopyTaskasLinkFull)
ON_COMMAND(ID_EDIT_CLEARREMINDER, OnEditClearReminder)
ON_COMMAND(ID_EDIT_CLEARTASKCOLOR, OnEditCleartaskcolor)
ON_COMMAND(ID_EDIT_CLEARTASKICON, OnEditCleartaskicon)
ON_COMMAND(ID_EDIT_DECTASKPERCENTDONE, OnEditDectaskpercentdone)
ON_COMMAND(ID_EDIT_DECTASKPRIORITY, OnEditDectaskpriority)
ON_COMMAND(ID_EDIT_FLAGTASK, OnEditFlagtask)
ON_COMMAND(ID_EDIT_INCTASKPERCENTDONE, OnEditInctaskpercentdone)
ON_COMMAND(ID_EDIT_INCTASKPRIORITY, OnEditInctaskpriority)
ON_COMMAND(ID_EDIT_INSERTDATE, OnEditInsertdate)
ON_COMMAND(ID_EDIT_INSERTDATETIME, OnEditInsertdatetime)
ON_COMMAND(ID_EDIT_INSERTTIME, OnEditInserttime)
ON_COMMAND(ID_EDIT_OFFSETDATES, OnEditOffsetdates)
ON_COMMAND(ID_EDIT_REDO, OnEditRedo)
ON_COMMAND(ID_EDIT_SELECTALL, OnEditSelectall)
ON_COMMAND(ID_EDIT_SETREMINDER, OnEditSetReminder)
ON_COMMAND(ID_EDIT_SETTASKICON, OnEditSettaskicon)
ON_COMMAND(ID_EDIT_UNDO, OnEditUndo)
ON_WM_ENABLE()
ON_COMMAND(ID_FILE_CHANGEPASSWORD, OnFileChangePassword)
ON_COMMAND(ID_FILE_OPENARCHIVE, OnFileOpenarchive)
ON_COMMAND(ID_NEXTTASK, OnGotoNexttask)
ON_COMMAND(ID_PREVTASK, OnGotoPrevtask)
ON_WM_MEASUREITEM()
ON_COMMAND(ID_NEWSUBTASK, OnNewsubtask)
ON_COMMAND(ID_NEWTASK, OnNewtask)
ON_COMMAND(ID_PRINTPREVIEW, OnPrintpreview)
ON_COMMAND(ID_EDIT_QUICKFIND, OnQuickFind)
ON_COMMAND(ID_EDIT_QUICKFINDNEXT, OnQuickFindNext)
ON_COMMAND(ID_EDIT_QUICKFINDPREV, OnQuickFindPrev)
ON_COMMAND(ID_SENDTASKS, OnSendTasks)
ON_COMMAND(ID_SEND_SELTASKS, OnSendSelectedTasks)
ON_COMMAND(ID_HELP_KEYBOARDSHORTCUTS, OnShowKeyboardshortcuts)
ON_COMMAND(ID_SHOWTIMELOGFILE, OnShowTimelogfile)
ON_COMMAND(ID_SORT_MULTI, OnSortMulti)
ON_WM_SYSCOLORCHANGE()
ON_COMMAND(ID_TABCTRL_PREFERENCES, OnTabctrlPreferences)
ON_COMMAND(ID_TASKLIST_SELECTCOLUMNS, OnTasklistSelectColumns)
ON_COMMAND(ID_TOOLS_CHECKFORUPDATES, OnToolsCheckforupdates)
ON_COMMAND(ID_TOOLS_TRANSFORM, OnToolsTransformactivetasklist)
ON_UPDATE_COMMAND_UI(ID_ADDTIMETOLOGFILE, OnUpdateAddtimetologfile)
ON_UPDATE_COMMAND_UI(ID_ARCHIVE_SELECTEDTASKS, OnUpdateArchiveSelectedCompletedTasks)
ON_UPDATE_COMMAND_UI(ID_CLOSEALLBUTTHIS, OnUpdateCloseallbutthis)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_DEPEND, OnUpdateCopyTaskasDependency)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_DEPENDFULL, OnUpdateCopyTaskasDependencyFull)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_PATH, OnUpdateCopyTaskasPath)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_LINK, OnUpdateCopyTaskasLink)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_LINKFULL, OnUpdateCopyTaskasLinkFull)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLEARREMINDER, OnUpdateEditClearReminder)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLEARTASKCOLOR, OnUpdateEditCleartaskcolor)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLEARTASKICON, OnUpdateEditCleartaskicon)
ON_UPDATE_COMMAND_UI(ID_EDIT_DECTASKPERCENTDONE, OnUpdateEditDectaskpercentdone)
ON_UPDATE_COMMAND_UI(ID_EDIT_DECTASKPRIORITY, OnUpdateEditDectaskpriority)
ON_UPDATE_COMMAND_UI(ID_EDIT_FLAGTASK, OnUpdateEditFlagtask)
ON_UPDATE_COMMAND_UI(ID_EDIT_INCTASKPERCENTDONE, OnUpdateEditInctaskpercentdone)
ON_UPDATE_COMMAND_UI(ID_EDIT_INCTASKPRIORITY, OnUpdateEditInctaskpriority)
ON_UPDATE_COMMAND_UI(ID_EDIT_INSERTDATE, OnUpdateEditInsertdate)
ON_UPDATE_COMMAND_UI(ID_EDIT_INSERTDATETIME, OnUpdateEditInsertdatetime)
ON_UPDATE_COMMAND_UI(ID_EDIT_INSERTTIME, OnUpdateEditInserttime)
ON_UPDATE_COMMAND_UI(ID_EDIT_OFFSETDATES, OnUpdateEditOffsetdates)
ON_UPDATE_COMMAND_UI(ID_EDIT_REDO, OnUpdateEditRedo)
ON_UPDATE_COMMAND_UI(ID_EDIT_SELECTALL, OnUpdateEditSelectall)
ON_UPDATE_COMMAND_UI(ID_EDIT_SETREMINDER, OnUpdateEditSetReminder)
ON_UPDATE_COMMAND_UI(ID_EDIT_SETTASKICON, OnUpdateEditSettaskicon)
ON_UPDATE_COMMAND_UI(ID_EDIT_UNDO, OnUpdateEditUndo)
ON_UPDATE_COMMAND_UI(ID_FILE_CHANGEPASSWORD, OnUpdateFileChangePassword)
ON_UPDATE_COMMAND_UI(ID_FILE_OPENARCHIVE, OnUpdateFileOpenarchive)
ON_UPDATE_COMMAND_UI(ID_NEXTTASK, OnUpdateGotoNexttask)
ON_UPDATE_COMMAND_UI(ID_PREVTASK, OnUpdateGotoPrevtask)
ON_UPDATE_COMMAND_UI(ID_NEWSUBTASK, OnUpdateNewsubtask)
ON_UPDATE_COMMAND_UI(ID_EDIT_QUICKFIND, OnUpdateQuickFind)
ON_UPDATE_COMMAND_UI(ID_EDIT_QUICKFINDNEXT, OnUpdateQuickFindNext)
ON_UPDATE_COMMAND_UI(ID_EDIT_QUICKFINDPREV, OnUpdateQuickFindPrev)
ON_UPDATE_COMMAND_UI(ID_SHOWTIMELOGFILE, OnUpdateShowTimelogfile)
ON_UPDATE_COMMAND_UI(ID_SENDTASKS, OnUpdateSendTasks)
ON_UPDATE_COMMAND_UI(ID_SEND_SELTASKS, OnUpdateSendSelectedTasks)
ON_UPDATE_COMMAND_UI(ID_SORT_MULTI, OnUpdateSortMulti)
ON_UPDATE_COMMAND_UI(ID_VIEW_CLEARFILTER, OnUpdateViewClearfilter)
ON_UPDATE_COMMAND_UI(ID_VIEW_COLLAPSEDUE, OnUpdateViewCollapseDuetasks)
ON_UPDATE_COMMAND_UI(ID_VIEW_COLLAPSESTARTED, OnUpdateViewCollapseStartedtasks)
ON_UPDATE_COMMAND_UI(ID_VIEW_COLLAPSEALL, OnUpdateViewCollapseall)
ON_UPDATE_COMMAND_UI(ID_VIEW_COLLAPSETASK, OnUpdateViewCollapsetask)
ON_UPDATE_COMMAND_UI(ID_VIEW_EXPANDDUE, OnUpdateViewExpandDuetasks)
ON_UPDATE_COMMAND_UI(ID_VIEW_EXPANDSTARTED, OnUpdateViewExpandStartedtasks)
ON_UPDATE_COMMAND_UI(ID_VIEW_EXPANDALL, OnUpdateViewExpandall)
ON_UPDATE_COMMAND_UI(ID_VIEW_EXPANDTASK, OnUpdateViewExpandtask)
ON_UPDATE_COMMAND_UI(ID_VIEW_FILTER, OnUpdateViewFilter)
ON_UPDATE_COMMAND_UI(ID_VIEW_PROJECTNAME, OnUpdateViewProjectname)
ON_UPDATE_COMMAND_UI(ID_VIEW_SHOWTASKLISTTABBAR, OnUpdateViewShowTasklistTabbar)
ON_UPDATE_COMMAND_UI(ID_VIEW_SHOWTREELISTTABBAR, OnUpdateViewShowTreeListTabbar)
ON_UPDATE_COMMAND_UI(ID_VIEW_SHOWFILTERBAR, OnUpdateViewShowfilterbar)
ON_UPDATE_COMMAND_UI(ID_VIEW_SORTTASKLISTTABS, OnUpdateViewSorttasklisttabs)
ON_UPDATE_COMMAND_UI(ID_VIEW_STATUS_BAR, OnUpdateViewStatusBar)
ON_UPDATE_COMMAND_UI(ID_VIEW_CYCLETASKVIEWS, OnUpdateViewCycleTaskViews)
ON_UPDATE_COMMAND_UI(ID_VIEW_TOGGLETREEANDLIST, OnUpdateViewToggleTreeandList)
ON_UPDATE_COMMAND_UI(ID_VIEW_TOGGLEFILTER, OnUpdateViewTogglefilter)
ON_UPDATE_COMMAND_UI(ID_VIEW_TOGGLETASKEXPANDED, OnUpdateViewToggletaskexpanded)
ON_UPDATE_COMMAND_UI(ID_VIEW_TOGGLETASKSANDCOMMENTS, OnUpdateViewToggletasksandcomments)
ON_UPDATE_COMMAND_UI(ID_WINDOW1, OnUpdateWindow)
ON_COMMAND(ID_VIEW_CLEARFILTER, OnViewClearfilter)
ON_COMMAND(ID_VIEW_COLLAPSEDUE, OnViewCollapseDuetasks)
ON_COMMAND(ID_VIEW_COLLAPSESTARTED, OnViewCollapseStartedtasks)
ON_COMMAND(ID_VIEW_COLLAPSEALL, OnViewCollapseall)
ON_COMMAND(ID_VIEW_COLLAPSETASK, OnViewCollapsetask)
ON_COMMAND(ID_VIEW_EXPANDDUE, OnViewExpandDuetasks)
ON_COMMAND(ID_VIEW_EXPANDSTARTED, OnViewExpandStartedtasks)
ON_COMMAND(ID_VIEW_EXPANDALL, OnViewExpandall)
ON_COMMAND(ID_VIEW_EXPANDTASK, OnViewExpandtask)
ON_COMMAND(ID_VIEW_FILTER, OnViewFilter)
ON_COMMAND(ID_VIEW_PROJECTNAME, OnViewProjectname)
ON_COMMAND(ID_VIEW_SHOWTASKLISTTABBAR, OnViewShowTasklistTabbar)
ON_COMMAND(ID_VIEW_SHOWTREELISTTABBAR, OnViewShowTreeListTabbar)
ON_COMMAND(ID_VIEW_SHOWFILTERBAR, OnViewShowfilterbar)
ON_COMMAND(ID_VIEW_SORTTASKLISTTABS, OnViewSorttasklisttabs)
ON_COMMAND(ID_VIEW_STATUS_BAR, OnViewStatusBar)
ON_COMMAND(ID_VIEW_CYCLETASKVIEWS, OnViewCycleTaskViews)
ON_COMMAND(ID_VIEW_TOGGLETREEANDLIST, OnViewToggleTreeandList)
ON_COMMAND(ID_VIEW_TOGGLEFILTER, OnViewTogglefilter)
ON_COMMAND(ID_VIEW_TOGGLETASKEXPANDED, OnViewToggletaskexpanded)
ON_COMMAND(ID_VIEW_TOGGLETASKSANDCOMMENTS, OnViewToggletasksandcomments)
ON_WM_WINDOWPOSCHANGED()
ON_WM_WINDOWPOSCHANGING()
ON_COMMAND(ID_TASKLIST_CUSTOMCOLUMNS, OnTasklistCustomColumns)
ON_COMMAND(ID_EDIT_GOTODEPENDENCY, OnEditGotoDependency)
ON_UPDATE_COMMAND_UI(ID_EDIT_GOTODEPENDENCY, OnUpdateEditGotoDependency)
ON_COMMAND(ID_EDIT_RECURRENCE, OnEditRecurrence)
ON_UPDATE_COMMAND_UI(ID_EDIT_RECURRENCE, OnUpdateEditRecurrence)
ON_WM_QUERYENDSESSION()
ON_COMMAND(ID_EDIT_CLEARFIELD, OnEditClearAttribute)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLEARFIELD, OnUpdateEditClearAttribute)
ON_COMMAND(ID_EDIT_CLEARFOCUSEDFIELD, OnEditClearFocusedAttribute)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLEARFOCUSEDFIELD, OnUpdateEditClearFocusedAttribute)
ON_UPDATE_COMMAND_UI(ID_TASKLIST_CUSTOMCOLUMNS, OnUpdateTasklistCustomcolumns)
ON_COMMAND(ID_ADDTIMETOLOGFILE, OnAddtimetologfile)
ON_UPDATE_COMMAND_UI(ID_ADDTIMETOLOGFILE, OnUpdateAddtimetologfile)
ON_WM_ACTIVATEAPP()
ON_WM_ERASEBKGND()
ON_WM_INITMENUPOPUP()
ON_WM_MOVE()
ON_WM_SYSCOMMAND()
ON_COMMAND(ID_EDIT_PASTEASREF, OnEditPasteAsRef)
ON_UPDATE_COMMAND_UI(ID_EDIT_PASTEASREF, OnUpdateEditPasteAsRef)
//}}AFX_MSG_MAP
ON_CBN_EDITCHANGE(IDC_QUICKFIND, OnEditChangeQuickFind)
ON_CBN_SELCHANGE(IDC_QUICKFIND, OnSelChangeQuickFind)
ON_COMMAND(ID_ABOUT, OnAbout)
ON_COMMAND(ID_ARCHIVE_COMPLETEDTASKS, OnArchiveCompletedtasks)
ON_COMMAND(ID_CLOSE, OnCloseTasklist)
ON_COMMAND(ID_CLOSEALL, OnCloseall)
ON_COMMAND(ID_DELETEALLTASKS, OnDeleteAllTasks)
ON_COMMAND(ID_DELETETASK, OnDeleteTask)
ON_COMMAND(ID_EDIT_CLOCK_TASK, OnEditTimeTrackTask)
ON_COMMAND(ID_EDIT_COPY, OnEditCopy)
ON_COMMAND(ID_EDIT_COPYAS_HTML, OnEditCopyashtml)
ON_COMMAND(ID_EDIT_COPYAS_TEXT, OnEditCopyastext)
ON_COMMAND(ID_EDIT_CUT, OnEditCut)
ON_COMMAND(ID_EDIT_FINDTASKS, OnFindTasks)
ON_COMMAND(ID_EDIT_OPENFILEREF, OnEditOpenfileref)
ON_COMMAND(ID_EDIT_PASTEAFTER, OnEditPasteAfter)
ON_COMMAND(ID_EDIT_PASTESUB, OnEditPasteSub)
ON_COMMAND(ID_EDIT_SETFILEREF, OnEditSetfileref)
ON_COMMAND(ID_EDIT_SPELLCHECKCOMMENTS, OnSpellcheckcomments)
ON_COMMAND(ID_EDIT_SPELLCHECKTITLE, OnSpellchecktitle)
ON_COMMAND(ID_EDIT_TASKCOLOR, OnEditTaskcolor)
ON_COMMAND(ID_EDIT_TASKDONE, OnEditTaskdone)
ON_COMMAND(ID_EDIT_TASKTEXT, OnEditTasktext)
ON_COMMAND(ID_EXIT, OnExit)
ON_COMMAND(ID_FILE_ENCRYPT, OnFileEncrypt)
ON_COMMAND(ID_FILE_RESETVERSION, OnFileResetversion)
ON_COMMAND(ID_LOAD_NORMAL, OnLoad)
ON_COMMAND(ID_MAXCOMMENTS, OnMaximizeComments)
ON_COMMAND(ID_MAXTASKLIST, OnMaximizeTasklist)
ON_COMMAND(ID_MINIMIZETOTRAY, OnMinimizeToTray)
ON_COMMAND(ID_MOVETASKDOWN, OnMovetaskdown)
ON_COMMAND(ID_MOVETASKLEFT, OnMovetaskleft)
ON_COMMAND(ID_MOVETASKRIGHT, OnMovetaskright)
ON_COMMAND(ID_MOVETASKUP, OnMovetaskup)
ON_COMMAND(ID_NEW, OnNew)
ON_COMMAND(ID_NEWSUBTASK_ATBOTTOM, OnNewsubtaskAtbottom)
ON_COMMAND(ID_NEWSUBTASK_ATTOP, OnNewsubtaskAttop)
ON_COMMAND(ID_NEWTASK_AFTERSELECTEDTASK, OnNewtaskAfterselectedtask)
ON_COMMAND(ID_NEWTASK_ATBOTTOM, OnNewtaskAtbottom)
ON_COMMAND(ID_NEWTASK_ATBOTTOMSELECTED, OnNewtaskAtbottomSelected)
ON_COMMAND(ID_NEWTASK_ATTOP, OnNewtaskAttop)
ON_COMMAND(ID_NEWTASK_ATTOPSELECTED, OnNewtaskAttopSelected)
ON_COMMAND(ID_NEWTASK_BEFORESELECTEDTASK, OnNewtaskBeforeselectedtask)
ON_COMMAND(ID_NEXTTOPLEVELTASK, OnNexttopleveltask)
ON_COMMAND(ID_OPEN_RELOAD, OnReload)
ON_COMMAND(ID_PREFERENCES, OnPreferences)
ON_COMMAND(ID_PREVTOPLEVELTASK, OnPrevtopleveltask)
ON_COMMAND(ID_PRINT, OnPrint)
ON_COMMAND(ID_SAVEALL, OnSaveall)
ON_COMMAND(ID_SAVEAS, OnSaveas)
ON_COMMAND(ID_SAVE_NORMAL, OnSave)
ON_COMMAND(ID_SORT, OnSort)
ON_COMMAND(ID_TOOLS_CHECKIN, OnToolsCheckin)
ON_COMMAND(ID_TOOLS_CHECKOUT, OnToolsCheckout)
ON_COMMAND(ID_TOOLS_EXPORT, OnExport)
ON_COMMAND(ID_TOOLS_IMPORT, OnImportTasklist)
ON_COMMAND(ID_TOOLS_SPELLCHECKTASKLIST, OnSpellcheckTasklist)
ON_COMMAND(ID_TOOLS_REMOVEFROMSOURCECONTROL, OnToolsRemovefromsourcecontrol)
ON_COMMAND(ID_TOOLS_TOGGLECHECKIN, OnToolsToggleCheckin)
ON_COMMAND(ID_TRAYICON_CLOSE, OnTrayiconClose)
ON_COMMAND(ID_TRAYICON_CREATETASK, OnTrayiconCreatetask)
ON_COMMAND(ID_TRAYICON_SHOW, OnTrayiconShow)
ON_COMMAND(ID_VIEW_MOVETASKLISTLEFT, OnViewMovetasklistleft)
ON_COMMAND(ID_VIEW_MOVETASKLISTRIGHT, OnViewMovetasklistright)
ON_COMMAND(ID_VIEW_NEXT, OnViewNext)
ON_COMMAND(ID_VIEW_NEXT_SEL, OnViewNextSel)
ON_COMMAND(ID_VIEW_PREV, OnViewPrev)
ON_COMMAND(ID_VIEW_PREV_SEL, OnViewPrevSel)
ON_COMMAND(ID_VIEW_REFRESHFILTER, OnViewRefreshfilter)
ON_COMMAND(ID_VIEW_TOOLBAR, OnViewToolbar)
ON_COMMAND_EX_RANGE(ID_FILE_MRU_FILE1, ID_FILE_MRU_FILE16, OnOpenRecentFile)
ON_COMMAND_RANGE(ID_EDIT_SETPRIORITYNONE, ID_EDIT_SETPRIORITY10, OnSetPriority)
ON_COMMAND_RANGE(ID_NEWTASK_SPLITTASKINTO_TWO, ID_NEWTASK_SPLITTASKINTO_FIVE, OnSplitTaskIntoPieces)
ON_COMMAND_RANGE(ID_SORT_BYFIRST, ID_SORT_BYLAST, OnSortBy)
ON_COMMAND_RANGE(ID_TOOLS_SHOWTASKS_DUETODAY, ID_TOOLS_SHOWTASKS_DUEENDNEXTMONTH, OnToolsShowtasksDue)
// ON_COMMAND_RANGE(ID_TOOLS_USEREXT1, ID_TOOLS_USEREXT16, OnUserUIExtension)
ON_COMMAND_RANGE(ID_TOOLS_USERTOOL1, ID_TOOLS_USERTOOL16, OnUserTool)
ON_COMMAND_RANGE(ID_FILE_OPEN_USERSTORAGE1, ID_FILE_OPEN_USERSTORAGE16, OnFileOpenFromUserStorage)
ON_COMMAND_RANGE(ID_FILE_SAVE_USERSTORAGE1, ID_FILE_SAVE_USERSTORAGE16, OnFileSaveToUserStorage)
ON_COMMAND_RANGE(ID_TRAYICON_SHOWDUETASKS1, ID_TRAYICON_SHOWDUETASKS20, OnTrayiconShowDueTasks)
ON_COMMAND_RANGE(ID_WINDOW1, ID_WINDOW16, OnWindow)
ON_MESSAGE(WM_ADDTOOLBARTOOLS, OnAddToolbarTools)
ON_MESSAGE(WM_APPRESTOREFOCUS, OnAppRestoreFocus)
ON_MESSAGE(WM_GETFONT, OnGetFont)
ON_MESSAGE(WM_GETICON, OnGetIcon)
ON_MESSAGE(WM_HOTKEY, OnHotkey)
ON_MESSAGE(WM_POSTONCREATE, OnPostOnCreate)
ON_MESSAGE(WM_POWERBROADCAST, OnPowerBroadcast)
ON_MESSAGE(WM_WEBUPDATEWIZARD, OnWebUpdateWizard)
ON_NOTIFY(NM_CLICK, IDC_TRAYICON, OnTrayIconClick)
ON_NOTIFY(NM_DBLCLK, IDC_TRAYICON, OnTrayIconDblClk)
ON_NOTIFY(NM_MCLICK, IDC_TABCONTROL, OnMBtnClickTabcontrol)
ON_NOTIFY(NM_RCLICK, IDC_TRAYICON, OnTrayIconRClick)
ON_NOTIFY(TCN_SELCHANGE, IDC_TABCONTROL, OnSelchangeTabcontrol)
ON_NOTIFY(TCN_SELCHANGING, IDC_TABCONTROL, OnSelchangingTabcontrol)
// ON_NOTIFY(TTN_NEEDTEXTA, 0, OnNeedTooltipText)
ON_NOTIFY(TTN_NEEDTEXTW, 0, OnNeedTooltipText)
ON_REGISTERED_MESSAGE(WM_ACB_ITEMADDED, OnQuickFindItemAdded)
ON_REGISTERED_MESSAGE(WM_FBN_FILTERCHNG, OnSelchangeFilter)
ON_REGISTERED_MESSAGE(WM_FTD_APPLYASFILTER, OnFindApplyAsFilter)
ON_REGISTERED_MESSAGE(WM_FTD_CLOSE, OnFindDlgClose)
ON_REGISTERED_MESSAGE(WM_FTD_ADDSEARCH, OnFindAddSearch)
ON_REGISTERED_MESSAGE(WM_FTD_DELETESEARCH, OnFindDeleteSearch)
ON_REGISTERED_MESSAGE(WM_FTD_FIND, OnFindDlgFind)
ON_REGISTERED_MESSAGE(WM_FTD_SELECTALL, OnFindSelectAll)
ON_REGISTERED_MESSAGE(WM_FTD_SELECTRESULT, OnFindSelectResult)
ON_REGISTERED_MESSAGE(WM_FW_FOCUSCHANGE, OnFocusChange)
ON_REGISTERED_MESSAGE(WM_PGP_CLEARMRU, OnPreferencesClearMRU)
ON_REGISTERED_MESSAGE(WM_PGP_CLEANUPDICTIONARY, OnPreferencesCleanupDictionary)
ON_REGISTERED_MESSAGE(WM_PTDP_LISTCHANGE, OnPreferencesDefaultListChange)
ON_REGISTERED_MESSAGE(WM_PTP_TESTTOOL, OnPreferencesTestTool)
ON_REGISTERED_MESSAGE(WM_TDCM_TASKHASREMINDER, OnToDoCtrlTaskHasReminder)
ON_REGISTERED_MESSAGE(WM_TDCM_TASKISDONE, OnToDoCtrlTaskIsDone)
ON_REGISTERED_MESSAGE(WM_TDCM_LENGTHYOPERATION, OnToDoCtrlDoLengthyOperation)
ON_REGISTERED_MESSAGE(WM_TDCM_TASKLINK, OnToDoCtrlDoTaskLink)
ON_REGISTERED_MESSAGE(WM_TDCM_FAILEDLINK, OnTodoCtrlFailedLink)
ON_REGISTERED_MESSAGE(WM_TDCN_DOUBLECLKREMINDERCOL, OnDoubleClkReminderCol)
ON_REGISTERED_MESSAGE(WM_TDCN_LISTCHANGE, OnToDoCtrlNotifyListChange)
ON_REGISTERED_MESSAGE(WM_TDCN_MODIFY, OnToDoCtrlNotifyMod)
ON_REGISTERED_MESSAGE(WM_TDCN_RECREATERECURRINGTASK, OnToDoCtrlNotifyRecreateRecurringTask)
ON_REGISTERED_MESSAGE(WM_TDCN_TIMETRACK, OnToDoCtrlNotifyTimeTrack)
ON_REGISTERED_MESSAGE(WM_TDCN_VIEWPOSTCHANGE, OnToDoCtrlNotifyViewChange)
ON_REGISTERED_MESSAGE(WM_TDL_GETVERSION , OnToDoListGetVersion)
ON_REGISTERED_MESSAGE(WM_TDL_ISCLOSING , OnToDoListIsClosing)
ON_REGISTERED_MESSAGE(WM_TDL_REFRESHPREFS , OnToDoListRefreshPrefs)
ON_REGISTERED_MESSAGE(WM_TDL_RESTORE , OnToDoListRestore)
ON_REGISTERED_MESSAGE(WM_TDL_SHOWWINDOW , OnToDoListShowWindow)
ON_REGISTERED_MESSAGE(WM_TD_REMINDER, OnToDoCtrlReminder)
ON_REGISTERED_MESSAGE(WM_TLDT_DROP, OnDropFile)
ON_UPDATE_COMMAND_UI(ID_ARCHIVE_COMPLETEDTASKS, OnUpdateArchiveCompletedtasks)
ON_UPDATE_COMMAND_UI(ID_CLOSEALL, OnUpdateCloseall)
ON_UPDATE_COMMAND_UI(ID_DELETEALLTASKS, OnUpdateDeletealltasks)
ON_UPDATE_COMMAND_UI(ID_DELETETASK, OnUpdateDeletetask)
ON_UPDATE_COMMAND_UI(ID_EDIT_CLOCK_TASK, OnUpdateEditTimeTrackTask)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPY, OnUpdateEditCopy)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_HTML, OnUpdateEditCopy)
ON_UPDATE_COMMAND_UI(ID_EDIT_COPYAS_TEXT, OnUpdateEditCopy)
ON_UPDATE_COMMAND_UI(ID_EDIT_CUT, OnUpdateEditCut)
ON_UPDATE_COMMAND_UI(ID_EDIT_OPENFILEREF, OnUpdateEditOpenfileref)
ON_UPDATE_COMMAND_UI(ID_EDIT_PASTEAFTER, OnUpdateEditPasteAfter)
ON_UPDATE_COMMAND_UI(ID_EDIT_PASTESUB, OnUpdateEditPasteSub)
ON_UPDATE_COMMAND_UI(ID_EDIT_SETFILEREF, OnUpdateEditSetfileref)
ON_UPDATE_COMMAND_UI(ID_EDIT_SPELLCHECKCOMMENTS, OnUpdateSpellcheckcomments)
ON_UPDATE_COMMAND_UI(ID_EDIT_SPELLCHECKTITLE, OnUpdateSpellchecktitle)
ON_UPDATE_COMMAND_UI(ID_EDIT_TASKCOLOR, OnUpdateTaskcolor)
ON_UPDATE_COMMAND_UI(ID_EDIT_TASKDONE, OnUpdateTaskdone)
ON_UPDATE_COMMAND_UI(ID_EDIT_TASKTEXT, OnUpdateEditTasktext)
ON_UPDATE_COMMAND_UI(ID_FILE_ENCRYPT, OnUpdateFileEncrypt)
ON_UPDATE_COMMAND_UI(ID_FILE_MRU_FILE1, OnUpdateRecentFileMenu)
ON_UPDATE_COMMAND_UI(ID_FILE_RESETVERSION, OnUpdateFileResetversion)
ON_UPDATE_COMMAND_UI(ID_MAXCOMMENTS, OnUpdateMaximizeComments)
ON_UPDATE_COMMAND_UI(ID_MAXTASKLIST, OnUpdateMaximizeTasklist)
ON_UPDATE_COMMAND_UI(ID_MOVETASKDOWN, OnUpdateMovetaskdown)
ON_UPDATE_COMMAND_UI(ID_MOVETASKLEFT, OnUpdateMovetaskleft)
ON_UPDATE_COMMAND_UI(ID_MOVETASKRIGHT, OnUpdateMovetaskright)
ON_UPDATE_COMMAND_UI(ID_MOVETASKUP, OnUpdateMovetaskup)
ON_UPDATE_COMMAND_UI(ID_NEW, OnUpdateNew)
ON_UPDATE_COMMAND_UI(ID_NEWSUBTASK_ATBOTTOM, OnUpdateNewsubtaskAtBottom)
ON_UPDATE_COMMAND_UI(ID_NEWSUBTASK_ATTOP, OnUpdateNewsubtaskAttop)
ON_UPDATE_COMMAND_UI(ID_NEWTASK, OnUpdateNewtask)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_AFTERSELECTEDTASK, OnUpdateNewtaskAfterselectedtask)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_ATBOTTOM, OnUpdateNewtaskAtbottom)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_ATBOTTOMSELECTED, OnUpdateNewtaskAtbottomSelected)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_ATTOP, OnUpdateNewtaskAttop)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_ATTOPSELECTED, OnUpdateNewtaskAttopSelected)
ON_UPDATE_COMMAND_UI(ID_NEWTASK_BEFORESELECTEDTASK, OnUpdateNewtaskBeforeselectedtask)
ON_UPDATE_COMMAND_UI(ID_NEXTTOPLEVELTASK, OnUpdateNexttopleveltask)
ON_UPDATE_COMMAND_UI(ID_OPEN_RELOAD, OnUpdateReload)
ON_UPDATE_COMMAND_UI(ID_PREVTOPLEVELTASK, OnUpdatePrevtopleveltask)
ON_UPDATE_COMMAND_UI(ID_PRINT, OnUpdatePrint)
ON_UPDATE_COMMAND_UI(ID_SAVEALL, OnUpdateSaveall)
ON_UPDATE_COMMAND_UI(ID_SAVEAS, OnUpdateSaveas)
ON_UPDATE_COMMAND_UI(ID_SAVE_NORMAL, OnUpdateSave)
ON_UPDATE_COMMAND_UI(ID_SB_SELCOUNT, OnUpdateSBSelectionCount)
ON_UPDATE_COMMAND_UI(ID_SB_TASKCOUNT, OnUpdateSBTaskCount)
ON_UPDATE_COMMAND_UI(ID_SORT, OnUpdateSort)
ON_UPDATE_COMMAND_UI(ID_TOOLS_CHECKIN, OnUpdateToolsCheckin)
ON_UPDATE_COMMAND_UI(ID_TOOLS_CHECKOUT, OnUpdateToolsCheckout)
ON_UPDATE_COMMAND_UI(ID_TOOLS_EXPORT, OnUpdateExport)
ON_UPDATE_COMMAND_UI(ID_TOOLS_IMPORT, OnUpdateImport)
ON_UPDATE_COMMAND_UI(ID_TOOLS_REMOVEFROMSOURCECONTROL, OnUpdateToolsRemovefromsourcecontrol)
ON_UPDATE_COMMAND_UI(ID_TOOLS_SPELLCHECKTASKLIST, OnUpdateSpellcheckTasklist)
ON_UPDATE_COMMAND_UI(ID_TOOLS_TOGGLECHECKIN, OnUpdateToolsToggleCheckin)
ON_UPDATE_COMMAND_UI(ID_TOOLS_TRANSFORM, OnUpdateExport) // use same text as export
ON_UPDATE_COMMAND_UI(ID_VIEW_MOVETASKLISTLEFT, OnUpdateViewMovetasklistleft)
ON_UPDATE_COMMAND_UI(ID_VIEW_MOVETASKLISTRIGHT, OnUpdateViewMovetasklistright)
ON_UPDATE_COMMAND_UI(ID_VIEW_NEXT, OnUpdateViewNext)
ON_UPDATE_COMMAND_UI(ID_VIEW_NEXT_SEL, OnUpdateViewNextSel)
ON_UPDATE_COMMAND_UI(ID_VIEW_PREV, OnUpdateViewPrev)
ON_UPDATE_COMMAND_UI(ID_VIEW_PREV_SEL, OnUpdateViewPrevSel)
ON_UPDATE_COMMAND_UI(ID_VIEW_REFRESHFILTER, OnUpdateViewRefreshfilter)
ON_UPDATE_COMMAND_UI(ID_VIEW_TOOLBAR, OnUpdateViewToolbar)
ON_UPDATE_COMMAND_UI_RANGE(ID_EDIT_SETPRIORITYNONE, ID_EDIT_SETPRIORITY10, OnUpdateSetPriority)
ON_UPDATE_COMMAND_UI_RANGE(ID_NEWTASK_SPLITTASKINTO_TWO, ID_NEWTASK_SPLITTASKINTO_FIVE, OnUpdateSplitTaskIntoPieces)
ON_UPDATE_COMMAND_UI_RANGE(ID_SORT_BYFIRST, ID_SORT_BYLAST, OnUpdateSortBy)
// ON_UPDATE_COMMAND_UI_RANGE(ID_TOOLS_USEREXT1, ID_TOOLS_USEREXT16, OnUpdateUserUIExtension)
ON_UPDATE_COMMAND_UI_RANGE(ID_TOOLS_USERTOOL1, ID_TOOLS_USERTOOL16, OnUpdateUserTool)
ON_WM_CLOSE()
ON_WM_CONTEXTMENU()
ON_WM_COPYDATA()
ON_WM_CREATE()
ON_WM_DRAWITEM()
ON_WM_ERASEBKGND()
ON_WM_HELPINFO()
ON_WM_INITMENUPOPUP()
ON_WM_ENDSESSION()
ON_WM_SIZE()
ON_WM_SYSCOMMAND()
ON_WM_TIMER()
#ifdef _DEBUG
ON_COMMAND(ID_DEBUGENDSESSION, OnDebugEndSession)
ON_COMMAND(ID_DEBUGSHOWSETUPDLG, OnDebugShowSetupDlg)
#endif
END_MESSAGE_MAP()
/
// CToDoListWnd message handlers
void CToDoListWnd::SetupUIStrings()
{
// set up UI strings for helper classes
CTimeEdit::SetUnits(THU_MINS, CEnString(IDS_TE_MINS), CEnString(IDS_MINS_ABBREV));
CTimeEdit::SetUnits(THU_HOURS, CEnString(IDS_TE_HOURS), CEnString(IDS_HOURS_ABBREV));
CTimeEdit::SetUnits(THU_DAYS, CEnString(IDS_TE_DAYS), CEnString(IDS_DAYS_ABBREV));
CTimeEdit::SetUnits(THU_WEEKS, CEnString(IDS_TE_WEEKS), CEnString(IDS_WEEKS_ABBREV));
CTimeEdit::SetUnits(THU_MONTHS, CEnString(IDS_TE_MONTHS), CEnString(IDS_MONTHS_ABBREV));
CTimeEdit::SetUnits(THU_YEARS, CEnString(IDS_TE_YEARS), CEnString(IDS_YEARS_ABBREV));
CTimeEdit::SetDefaultButtonTip(CEnString(IDS_TIMEUNITS));
CFileEdit::SetDefaultButtonTips(CEnString(IDS_BROWSE), CEnString(IDS_VIEW));
CFileEdit::SetDefaultBrowseTitles(CEnString(IDS_BROWSEFILE_TITLE), CEnString(IDS_BROWSEFOLDER_TITLE));
CTDLRecurringTaskEdit::SetDefaultButtonTip(CEnString(IDS_OPTIONS));
CXmlFileEx::SetUIStrings(CEnString(IDS_ENCRYPTEDFILE), CEnString(IDS_DECRYPTFAILED));
/*
CServerDlg::SetItemText(SD_TITLE, IDS_SCD_TITLE);
CServerDlg::SetItemText(IDC_SD_SERVERLABEL, IDS_SD_SERVERLABEL);
CServerDlg::SetItemText(IDC_SD_USERNAMELABEL, IDS_SD_USERNAMELABEL);
CServerDlg::SetItemText(IDC_SD_PASSWORDLABEL, IDS_SD_PASSWORDLABEL);
CServerDlg::SetItemText(IDC_SD_ANONLOGIN, IDS_SD_ANONLOGIN);
CServerDlg::SetItemText(IDOK, IDS_OK);
CServerDlg::SetItemText(IDCANCEL, IDS_CANCEL);
CPasswordDialog::SetItemText(PD_TITLE, IDS_PD_TITLE);
CPasswordDialog::SetItemText(IDC_PD_PASSWORDLABEL, IDS_PD_PASSWORDLABEL);
CPasswordDialog::SetItemText(IDC_PD_CONFIRMLABEL, IDS_PD_CONFIRMLABEL);
CPasswordDialog::SetItemText(DLG_PD_CONFIRMFAILED, IDS_PD_CONFIRMFAILED);
CPasswordDialog::SetItemText(IDOK, IDS_OK);
CPasswordDialog::SetItemText(IDCANCEL, IDS_CANCEL);
CSpellCheckDlg::SetItemText(IDC_SCD_DICTLABEL, IDS_SCD_DICTLABEL);
CSpellCheckDlg::SetItemText(IDC_SCD_BROWSE, IDS_SCD_BROWSE);
CSpellCheckDlg::SetItemText(IDC_SCD_URL, IDS_SCD_URL);
CSpellCheckDlg::SetItemText(IDC_SCD_CHECKINGLABEL, IDS_SCD_CHECKINGLABEL);
CSpellCheckDlg::SetItemText(IDC_SCD_RESTART, IDS_SCD_RESTART);
CSpellCheckDlg::SetItemText(IDC_SCD_REPLACELABEL, IDS_SCD_REPLACELABEL);
CSpellCheckDlg::SetItemText(IDC_SCD_WITHLABEL, IDS_SCD_WITHLABEL);
CSpellCheckDlg::SetItemText(IDC_SCD_REPLACE, IDS_SCD_REPLACE);
CSpellCheckDlg::SetItemText(IDC_SCD_NEXT, IDS_SCD_NEXT);
CSpellCheckDlg::SetItemText(IDOK, IDS_OK);
CSpellCheckDlg::SetItemText(IDCANCEL, IDS_CANCEL);
CSpellCheckDlg::SetItemText(SCD_TITLE, IDS_SCD_TITLE);
CSpellCheckDlg::SetItemText(DLG_SCD_SETUPMSG, IDS_SCD_SETUPMSG);
CSpellCheckDlg::SetItemText(DLG_SCD_DICTFILTER, IDS_SCD_DICTFILTER);
CSpellCheckDlg::SetItemText(DLG_SCD_ENGINEFILTER, IDS_SCD_ENGINEFILTER);
CSpellCheckDlg::SetItemText(DLG_SCD_ENGINETITLE, IDS_SCD_ENGINETITLE);
*/
CSpellCheckDlg::SetItemText(DLG_SCD_BROWSETITLE, IDS_SCD_BROWSETITLE);
}
BOOL CToDoListWnd::OnHelpInfo(HELPINFO* /*pHelpInfo*/)
{
// always eat this because we handle the F1 ourselves
return FALSE;
}
void CToDoListWnd::SetUITheme(const CString& sThemeFile)
{
if (COSVersion() < OSV_XP)
return;
// cache existing theme
CUIThemeFile themeCur = m_theme;
if (CThemed::IsThemeActive() && m_theme.LoadThemeFile(sThemeFile))
m_sThemeFile = sThemeFile;
else
{
m_sThemeFile.Empty();
m_theme.Reset();
}
// update the UI
if (themeCur != m_theme)
{
m_cbQuickFind.DestroyWindow();
m_toolbar.DestroyWindow();
m_tbHelper.Release();
InitToolbar();
// reinitialize the menu icon manager
m_mgrMenuIcons.Release();
InitMenuIconManager();
}
else
m_toolbar.SetBackgroundColors(m_theme.crToolbarLight,
m_theme.crToolbarDark,
m_theme.HasGradient(),
m_theme.HasGlass());
m_statusBar.SetUIColors(m_theme.crStatusBarLight,
m_theme.crStatusBarDark,
m_theme.crStatusBarText,
m_theme.HasGradient(),
m_theme.HasGlass());
m_menubar.SetBackgroundColor(m_theme.crMenuBack);
m_filterBar.SetUIColors(m_theme.crAppBackLight, m_theme.crAppText);
m_tabCtrl.SetBackgroundColor(m_theme.crAppBackDark);
for (int nCtl = 0; nCtl < GetTDCCount(); nCtl++)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtl);
tdc.SetUITheme(m_theme);
}
if (m_findDlg.GetSafeHwnd())
m_findDlg.SetUITheme(m_theme);
DrawMenuBar();
Invalidate();
}
BOOL CToDoListWnd::Create(const TDCSTARTUP& startup)
{
m_startupOptions = startup;
m_bVisible = startup.HasFlag(TLD_FORCEVISIBLE) ? 1 : -1;
m_bUseStagingScript = startup.HasFlag(TLD_STAGING);
#ifdef _DEBUG
m_bPasswordPrompting = FALSE;
#else
m_bPasswordPrompting = startup.HasFlag(TLD_PASSWORDPROMPTING);
#endif
FileMisc::EnableLogging(startup.HasFlag(TLD_LOGGING), TRUE, FALSE);
return CFrameWnd::LoadFrame(IDR_MAINFRAME, WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN, NULL, NULL);
}
int CToDoListWnd::GetVersion()
{
return TD_VERSION;
}
int CToDoListWnd::MessageBox(UINT nIDText, UINT nIDCaption, UINT nType, LPCTSTR szData)
{
if (szData && *szData)
return MessageBox(CEnString(nIDText, szData), nIDCaption, nType);
else
return MessageBox(CEnString(nIDText), nIDCaption, nType);
}
int CToDoListWnd::MessageBox(const CString& sText, UINT nIDCaption, UINT nType)
{
return MessageBox(sText, CEnString(nIDCaption), nType);
}
int CToDoListWnd::MessageBox(const CString& sText, const CString& sCaption, UINT nType)
{
CString sMessage;
if (!sCaption.IsEmpty())
sMessage.Format(_T("%s|%s"), sCaption, sText);
else
sMessage = sText;
return AfxMessageBox(sMessage, nType);
}
int CToDoListWnd::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
return -1;
// set frame icon
HICON hIcon = GraphicsMisc::LoadIcon(IDR_MAINFRAME);
SetIcon(hIcon, FALSE);
// set taskbar icon
hIcon = GraphicsMisc::LoadIcon(IDR_MAINFRAME, 32);
SetIcon(hIcon, TRUE);
SetWindowPos(NULL, 0, 0, 0, 0, SWP_DRAWFRAME | SWP_NOSIZE | SWP_NOZORDER | SWP_NOMOVE);
// menu
if (!LoadMenubar())
return -1;
// drop target
if (!m_dropTarget.Register(this, this))
return -1;
// trayicon
// we always create the trayicon (for simplicity) but we only
// show it if required
BOOL bUseSysTray = Prefs().GetUseSysTray();
m_trayIcon.Create(WS_CHILD | (bUseSysTray ? WS_VISIBLE : 0),
this,
IDC_TRAYICON,
IDI_TRAY_STD,
CString(IDS_COPYRIGHT));
// toolbar
if (!InitToolbar())
return -1;
// statusbar
if (!InitStatusbar())
return -1;
// filterbar
if (!InitFilterbar())
return -1;
// tabctrl
if (!m_tabCtrl.Create(WS_CHILD | WS_VISIBLE | TCS_HOTTRACK | TCS_TABS | TCS_SINGLELINE | TCS_RIGHTJUSTIFY | TCS_TOOLTIPS,
CRect(0, 0, 10, 10), this, IDC_TABCONTROL))
return -1;
m_tabCtrl.GetToolTips()->ModifyStyle(0, TTS_ALWAYSTIP);
CLocalizer::EnableTranslation(m_tabCtrl, FALSE);
BOOL bStackTabbar = Prefs().GetStackTabbarItems();
m_tabCtrl.ModifyStyle(bStackTabbar ? 0 : TCS_MULTILINE, bStackTabbar ? TCS_MULTILINE : 0);
UpdateTabSwitchTooltip();
if (m_ilTabCtrl.Create(16, 16, ILC_COLOR32 | ILC_MASK, 4, 1))
{
CBitmap bm;
bm.LoadBitmap(IDB_SOURCECONTROL_STD);
m_ilTabCtrl.Add(&bm, RGB(255, 0, 255));
m_tabCtrl.SetImageList(&m_ilTabCtrl);
}
else
return -1;
// UI Font
InitUIFont();
LoadSettings();
// add a barebones tasklist while we're still hidden
if (!CreateNewTaskList(FALSE))
return -1;
// timers
SetTimer(TIMER_DUEITEMS, TRUE);
SetTimer(TIMER_TIMETRACKING, TRUE);
// notify users about dodgy content plugins
if (m_mgrContent.SomePluginsHaveBadversions())
{
if (MessageBox(IDS_BADPLUGINVERSIONS, IDS_BADPLUGINTITLE, MB_OKCANCEL) == IDCANCEL)
return -1;
}
// inherited parent task attributes for new tasks
CTDCAttributeArray aParentAttrib;
BOOL bUpdateAttrib;
Prefs().GetParentAttribsUsed(aParentAttrib, bUpdateAttrib);
CFilteredToDoCtrl::SetInheritedParentAttributes(aParentAttrib, bUpdateAttrib);
// theme
SetUITheme(Prefs().GetUITheme());
// late initialization
PostMessage(WM_POSTONCREATE);
return 0; // success
}
void CToDoListWnd::InitUIFont()
{
GraphicsMisc::VerifyDeleteObject(m_fontMain);
HFONT hFontUI = GraphicsMisc::CreateFont(_T("Tahoma"), 8);
if (m_fontMain.Attach(hFontUI))
CDialogHelper::SetFont(this, m_fontMain); // will update all child controls
}
void CToDoListWnd::InitShortcutManager()
{
// setup defaults first
m_mgrShortcuts.AddShortcut(ID_LOAD_NORMAL, 'O', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_OPEN_RELOAD, 'R', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_SAVE_NORMAL, 'S', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_NEWTASK_BEFORESELECTEDTASK, 'N', HOTKEYF_CONTROL | HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_NEWTASK_AFTERSELECTEDTASK, 'N', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_NEWSUBTASK_ATBOTTOM, 'N', HOTKEYF_CONTROL | HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_MAXTASKLIST, 'M', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_MAXCOMMENTS, 'M', HOTKEYF_CONTROL | HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_PRINT, 'P', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_VIEW_NEXT, VK_TAB, HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_VIEW_PREV, VK_TAB, HOTKEYF_CONTROL | HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_EDIT_CUT, 'X', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_COPY, 'C', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_PASTEAFTER, 'V', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_PASTESUB, 'V', HOTKEYF_CONTROL | HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_EDIT_INSERTDATETIME, 'D', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_OPENFILEREF, 'G', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EXIT, VK_F4, HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_CLOSE, VK_F4, HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_MOVETASKDOWN, VK_DOWN, HOTKEYF_CONTROL | HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_MOVETASKUP, VK_UP, HOTKEYF_CONTROL | HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_MOVETASKLEFT, VK_LEFT, HOTKEYF_CONTROL | HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_MOVETASKRIGHT, VK_RIGHT, HOTKEYF_CONTROL | HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_DELETETASK, VK_DELETE, HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_EDIT_TASKTEXT, VK_F2, 0);
m_mgrShortcuts.AddShortcut(ID_EDIT_TASKDONE, VK_SPACE, HOTKEYF_CONTROL | HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_TOOLS_IMPORT, 'I', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_TOOLS_EXPORT, 'E', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_HELP, VK_F1, 0);
m_mgrShortcuts.AddShortcut(ID_WINDOW1, '1', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW2, '2', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW3, '3', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW4, '4', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW5, '5', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW6, '6', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW7, '7', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW8, '8', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_WINDOW9, '9', HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_TOOLS_TRANSFORM, 'T', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_FINDTASKS, 'F', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_QUICKFIND, 'Q', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_QUICKFINDNEXT, VK_F3, 0);
m_mgrShortcuts.AddShortcut(ID_EDIT_QUICKFINDPREV, VK_F3, HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_VIEW_REFRESHFILTER, VK_F5, HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_VIEW_TOGGLEFILTER, VK_F12, 0);
m_mgrShortcuts.AddShortcut(ID_EDIT_SELECTALL, 'A', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_PREVTASK, VK_F7, 0);
m_mgrShortcuts.AddShortcut(ID_NEXTTASK, VK_F8, 0);
m_mgrShortcuts.AddShortcut(ID_EDIT_UNDO, 'Z', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_REDO, 'Y', HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_VIEW_TOGGLETREEANDLIST, VK_F10, 0);
m_mgrShortcuts.AddShortcut(ID_VIEW_CYCLETASKVIEWS, VK_F10, HOTKEYF_SHIFT);
m_mgrShortcuts.AddShortcut(ID_VIEW_TOGGLETASKSANDCOMMENTS, VK_F11, 0);
m_mgrShortcuts.AddShortcut(ID_VIEW_TOGGLETASKEXPANDED, VK_SPACE, HOTKEYF_CONTROL | HOTKEYF_ALT);
m_mgrShortcuts.AddShortcut(ID_EDIT_INCTASKPERCENTDONE, VK_ADD, HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_EDIT_DECTASKPERCENTDONE, VK_SUBTRACT, HOTKEYF_CONTROL);
m_mgrShortcuts.AddShortcut(ID_VIEW_PREV_SEL, VK_LEFT, HOTKEYF_ALT | HOTKEYF_EXT);
m_mgrShortcuts.AddShortcut(ID_VIEW_NEXT_SEL, VK_RIGHT, HOTKEYF_ALT | HOTKEYF_EXT);
// now init shortcut mgr which will override the defaults
// with the users actual settings
CPreferences prefs;
if (m_mgrShortcuts.Initialize(this, &prefs))
{
// fix for previously adding escape key as a shortcut for IDCLOSE
// (big mistake)
if (m_mgrShortcuts.GetShortcut(IDCLOSE) == VK_ESCAPE)
m_mgrShortcuts.DeleteShortcut(IDCLOSE);
// fix for paste being wrongly set up
if (m_mgrShortcuts.GetShortcut(ID_EDIT_PASTE))
{
// delete existing
m_mgrShortcuts.DeleteShortcut(ID_EDIT_PASTE);
// if nothing already assigned use Ctrl+V
if (!m_mgrShortcuts.GetShortcut(ID_EDIT_PASTESUB))
m_mgrShortcuts.AddShortcut(ID_EDIT_PASTESUB, 'V', HOTKEYF_CONTROL);
}
}
}
void CToDoListWnd::InitMenuIconManager()
{
if (!m_mgrMenuIcons.Initialize(this))
return;
m_mgrMenuIcons.ClearImages();
// images
UINT nToolbarImageID = IDB_APP_TOOLBAR_STD;
UINT nExtraImageID = IDB_APP_EXTRA_STD;
CUIntArray aCmdIDs;
// toolbar
aCmdIDs.Add(ID_LOAD_NORMAL);
aCmdIDs.Add(ID_SAVE_NORMAL);
aCmdIDs.Add(ID_SAVEALL);
// new tasks
aCmdIDs.Add(GetNewTaskCmdID());
aCmdIDs.Add(GetNewSubtaskCmdID());
aCmdIDs.Add(ID_EDIT_TASKTEXT);
aCmdIDs.Add(ID_EDIT_SETTASKICON);
aCmdIDs.Add(ID_EDIT_SETREMINDER);
aCmdIDs.Add(ID_EDIT_UNDO);
aCmdIDs.Add(ID_EDIT_REDO);
aCmdIDs.Add(ID_MAXTASKLIST);
aCmdIDs.Add(ID_VIEW_EXPANDTASK);
aCmdIDs.Add(ID_VIEW_COLLAPSETASK);
aCmdIDs.Add(ID_VIEW_PREV_SEL);
aCmdIDs.Add(ID_VIEW_NEXT_SEL);
aCmdIDs.Add(ID_EDIT_FINDTASKS);
aCmdIDs.Add(ID_SORT);
aCmdIDs.Add(ID_DELETETASK);
// source control
if (GetTDCCount() && m_mgrToDoCtrls.PathSupportsSourceControl(GetSelToDoCtrl()))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (tdc.IsCheckedOut())
aCmdIDs.Add(ID_TOOLS_CHECKIN);
else
aCmdIDs.Add(ID_TOOLS_CHECKOUT);
}
else
aCmdIDs.Add(ID_TOOLS_TOGGLECHECKIN);
aCmdIDs.Add(ID_PREFERENCES);
if (m_theme.HasToolbarImageFile(_T("TODOLIST")))
{
COLORREF crMask = CLR_NONE;
CString sImagePath = m_theme.GetToolbarImageFile(_T("TODOLIST"), crMask);
VERIFY(m_mgrMenuIcons.AddImages(aCmdIDs, sImagePath, 16, crMask));
}
else
m_mgrMenuIcons.AddImages(aCmdIDs, nToolbarImageID, 16, RGB(255, 0, 255));
// extra
aCmdIDs.RemoveAll();
aCmdIDs.Add(ID_HELP_WIKI);
aCmdIDs.Add(ID_HELP_DONATE);
aCmdIDs.Add(ID_HELP);
if (m_theme.HasToolbarImageFile(_T("TODOLIST_EXTRA")))
{
COLORREF crMask = CLR_NONE;
CString sImagePath = m_theme.GetToolbarImageFile(_T("TODOLIST_EXTRA"), crMask);
VERIFY(m_mgrMenuIcons.AddImages(aCmdIDs, sImagePath, 16, crMask));
}
else
m_mgrMenuIcons.AddImages(aCmdIDs, nExtraImageID, 16, RGB(255, 0, 255));
}
void CToDoListWnd::OnShowKeyboardshortcuts()
{
CStringArray aMapping;
if (m_mgrShortcuts.BuildMapping(IDR_MAINFRAME, aMapping, '|'))
{
// add a few misc items that don't appear in the menus
CString sMisc;
for (int nItem = 0; nItem < NUM_MISCSHORTCUTS; nItem++)
{
if (MISC_SHORTCUTS[nItem].dwShortcut)
sMisc.Format(_T("%s|%s"), m_mgrShortcuts.GetShortcutText(MISC_SHORTCUTS[nItem].dwShortcut),
CEnString(MISC_SHORTCUTS[nItem].nIDShortcut));
else
sMisc.Empty();
aMapping.Add(sMisc);
}
CKeyboardShortcutDisplayDlg dialog(aMapping, '|');
dialog.DoModal();
}
}
LRESULT CToDoListWnd::OnFocusChange(WPARAM wp, LPARAM /*lp*/)
{
if (m_statusBar.GetSafeHwnd() && IsWindowEnabled() && GetTDCCount() && wp)
{
// grab the previous window in the z-order and if its
// static text then use that as the focus hint
CWnd* pFocus = CWnd::FromHandle((HWND)wp);
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
m_sCurrentFocus.Empty();
if (CDialogHelper::IsChildOrSame(tdc.GetSafeHwnd(), (HWND)wp))
{
m_sCurrentFocus.LoadString(IDS_FOCUS_TASKS);
m_sCurrentFocus += ": ";
m_sCurrentFocus += tdc.GetControlDescription(pFocus);
}
else if (pFocus == m_cbQuickFind.GetWindow(GW_CHILD))
{
m_sCurrentFocus.LoadString(IDS_QUICKFIND);
}
else
{
if (m_findDlg.GetSafeHwnd() && m_findDlg.IsChild(pFocus))
{
m_sCurrentFocus.LoadString(IDS_FINDTASKS);
}
else if (m_filterBar.GetSafeHwnd() && m_filterBar.IsChild(pFocus))
{
m_sCurrentFocus.LoadString(IDS_FOCUS_FILTERBAR);
}
if (!m_sCurrentFocus.IsEmpty())
m_sCurrentFocus += ": ";
m_sCurrentFocus += GetControlLabel(pFocus);
}
// limit length of string
if (m_sCurrentFocus.GetLength() > 22)
m_sCurrentFocus = m_sCurrentFocus.Left(20) + _T("...");
m_statusBar.SetPaneText(m_statusBar.CommandToIndex(ID_SB_FOCUS), m_sCurrentFocus);
// if the status bar is hidden then add text to title bar
if (!m_bShowStatusBar)
UpdateCaption();
}
return 0L;
}
LRESULT CToDoListWnd::OnGetIcon(WPARAM bLargeIcon, LPARAM /*not used*/)
{
if (!bLargeIcon)
{
// cache small icon for reuse
if (!m_hIcon)
m_hIcon = CSysImageList(FALSE).ExtractAppIcon();
return (LRESULT)m_hIcon;
}
else
return Default();
}
BOOL CToDoListWnd::InitStatusbar()
{
static SBACTPANEINFO SB_PANES[] =
{
{ ID_SB_FILEPATH, MAKEINTRESOURCE(IDS_SB_FILEPATH_TIP), SBACTF_STRETCHY | SBACTF_RESOURCETIP },
{ ID_SB_FILEVERSION, MAKEINTRESOURCE(IDS_SB_FILEVERSION_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
{ ID_SB_TASKCOUNT, MAKEINTRESOURCE(IDS_SB_TASKCOUNT_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
//{ ID_SB_SPACER },
{ ID_SB_SELCOUNT, MAKEINTRESOURCE(0), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
{ ID_SB_SELTIMEEST, MAKEINTRESOURCE(IDS_SB_SELTIMEEST_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
{ ID_SB_SELTIMESPENT, MAKEINTRESOURCE(IDS_SB_SELTIMESPENT_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
{ ID_SB_SELCOST, MAKEINTRESOURCE(IDS_SB_SELCOST_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
//{ ID_SB_SPACER },
{ ID_SB_FOCUS, MAKEINTRESOURCE(IDS_SB_FOCUS_TIP), SBACTF_AUTOFIT | SBACTF_RESOURCETIP },
};
static int SB_PANECOUNT = sizeof(SB_PANES) / sizeof(SBACTPANEINFO);
if (!m_statusBar.Create(this, WS_CHILD | WS_VISIBLE | CBRS_BOTTOM, IDC_FILENAME))
return FALSE;
// prevent translation because we handle it manually
CLocalizer::EnableTranslation(m_statusBar, FALSE);
if (!m_statusBar.SetPanes(SB_PANES, SB_PANECOUNT))
return FALSE;
return TRUE;
}
BOOL CToDoListWnd::InitFilterbar()
{
if (!m_filterBar.Create(this))
return FALSE;
m_filterBar.EnableMultiSelection(Prefs().GetMultiSelFilters());
m_filterBar.ShowDefaultFilters(Prefs().GetShowDefaultFilters());
RefreshFilterBarCustomFilters();
return TRUE;
}
BOOL CToDoListWnd::InitToolbar()
{
if (m_toolbar.GetSafeHwnd())
return TRUE;
UINT nStyle = WS_CHILD | CBRS_ALIGN_TOP | WS_CLIPCHILDREN | CBRS_TOOLTIPS;
if (m_bShowToolbar)
nStyle |= WS_VISIBLE;
if (!m_toolbar.CreateEx(this, TBSTYLE_FLAT | TBSTYLE_WRAPABLE, nStyle))
return FALSE;
if (!m_toolbar.LoadToolBar(IDR_APP_TOOLBAR))
return FALSE;
// colors
if (CThemed::IsThemeActive())
{
m_toolbar.SetBackgroundColors(m_theme.crToolbarLight,
m_theme.crToolbarDark,
m_theme.HasGradient(),
m_theme.HasGlass());
}
// toolbar images
if (m_theme.HasToolbarImageFile(_T("TODOLIST")))
{
COLORREF crMask = CLR_NONE;
CString sImagePath = m_theme.GetToolbarImageFile(_T("TODOLIST"), crMask);
VERIFY(m_toolbar.SetImage(sImagePath, crMask));
}
else
{
const COLORREF MAGENTA = RGB(255, 0, 255);
m_toolbar.SetImage(IDB_APP_TOOLBAR_STD, MAGENTA);
}
// resize the toolbar in one row so that our subsequent calculations work
m_toolbar.GetToolBarCtrl().HideButton(ID_TOOLS_TOGGLECHECKIN, !Prefs().GetEnableSourceControl());
m_toolbar.MoveWindow(0, 0, 1000, 32);
// insert combobox for quick Find after Find Tasks button
int nPos = m_toolbar.CommandToIndex(ID_EDIT_FINDTASKS) + 1;
TBBUTTON tbbQuickFind = { 0, nPos, 0, TBSTYLE_SEP, 0, NULL };
TBBUTTON tbbSep = { 0, nPos + 1, 0, TBSTYLE_SEP, 0, NULL };
m_toolbar.GetToolBarCtrl().InsertButton(nPos, &tbbQuickFind);
m_toolbar.GetToolBarCtrl().InsertButton(nPos + 1, &tbbSep);
TBBUTTONINFO tbi;
tbi.cbSize = sizeof( TBBUTTONINFO );
tbi.cx = 150;
tbi.dwMask = TBIF_SIZE; // By index
m_toolbar.GetToolBarCtrl().SetButtonInfo(nPos + 1, &tbi);
CRect rect;
m_toolbar.GetToolBarCtrl().GetItemRect(nPos + 1, &rect);
rect.bottom += 200;
if (!m_cbQuickFind.Create(WS_CHILD | WS_VSCROLL | WS_VISIBLE | CBS_AUTOHSCROLL |
CBS_DROPDOWN, rect, &m_toolbar, IDC_QUICKFIND))
return FALSE;
m_cbQuickFind.SetFont(&m_fontMain);
m_mgrPrompts.SetComboEditPrompt(m_cbQuickFind, IDS_QUICKFIND);
m_tbHelper.Initialize(&m_toolbar, this);
return TRUE;
}
void CToDoListWnd::OnEditChangeQuickFind()
{
m_cbQuickFind.GetWindowText(m_sQuickFind);
if (!GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTNEXTINCLCURRENT))
GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTFIRST);
}
void CToDoListWnd::OnSelChangeQuickFind()
{
int nSel = m_cbQuickFind.GetCurSel();
if (nSel != CB_ERR)
{
m_sQuickFind = CDialogHelper::GetSelectedItem(m_cbQuickFind);
if (!GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTNEXT))
GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTFIRST);
}
}
BOOL CToDoListWnd::PreTranslateMessage(MSG* pMsg)
{
// the only way we get a WM_CLOSE here is if it was sent from an external app
// so we shut down as gracefully as possible
if (pMsg->message == WM_CLOSE && IsWindowEnabled())
{
DoExit();
return TRUE;
}
if (ProcessDialogControlShortcut(pMsg))
return TRUE;
if (IsDroppedComboBox(pMsg->hwnd))
return FALSE;
// process for app level shortcuts first so we can handle
// reserved shortcuts
DWORD dwShortcut = 0;
UINT nCmdID = m_mgrShortcuts.ProcessMessage(pMsg, &dwShortcut);
// if it's a reserved shortcut let's notify the user to change it
if (CFilteredToDoCtrl::IsReservedShortcut(dwShortcut))
{
int nRet = MessageBox(IDS_RESERVEDSHORTCUT_MSG, IDS_RESERVEDSHORTCUT_TITLE, MB_YESNOCANCEL);
if (nRet == IDYES)
DoPreferences(PREFPAGE_SHORTCUT);
// and keep eating it until the user changes it
return TRUE;
}
// also we handle undo/redo
if (nCmdID != ID_EDIT_UNDO && nCmdID != ID_EDIT_REDO)
{
// now try active task list
if (GetTDCCount() && GetToDoCtrl().PreTranslateMessage(pMsg))
return TRUE;
}
if (nCmdID)
{
BOOL bSendMessage = TRUE; // default
// focus checks
switch (nCmdID)
{
case ID_EDIT_CUT:
case ID_EDIT_COPY:
// tree must have the focus
if (!GetToDoCtrl().TasksHaveFocus())
{
bSendMessage = FALSE;
GetToDoCtrl().ClearCopiedItem();
}
break;
// tree must have the focus
case ID_EDIT_SELECT_ALL:
case ID_EDIT_PASTE:
case ID_DELETEALLTASKS:
case ID_DELETETASK:
bSendMessage = GetToDoCtrl().TasksHaveFocus();
break;
}
// send off
if (bSendMessage)
{
SendMessage(WM_COMMAND, nCmdID);
return TRUE;
}
}
// we need to check for , and
switch (pMsg->message)
{
case WM_KEYDOWN:
{
switch (pMsg->wParam)
{
case VK_ESCAPE:
if (Prefs().GetEscapeMinimizes() && GetCapture() == NULL)
{
// if the window with the target is either a combobox or
// the child edit of a combobox and the combo is
// dropped down then let it thru else if the target is
// a child of ours then treat as a cancel
BOOL bHandle = TRUE;
if (CWinClasses::IsClass(pMsg->hwnd, WC_COMBOBOX))
bHandle = !ComboBox_GetDroppedState(pMsg->hwnd);
else if (CWinClasses::IsClass(::GetParent(pMsg->hwnd), WC_COMBOBOX))
bHandle = !ComboBox_GetDroppedState(::GetParent(pMsg->hwnd));
else if (GetTDCCount() && GetToDoCtrl().IsTaskLabelEditing())
bHandle = FALSE;
if (bHandle && ::IsChild(*this, pMsg->hwnd))
{
OnCancel();
return TRUE;
}
}
break;
case VK_TAB: // tabbing away from Quick Find -> tasks
if (::IsChild(m_cbQuickFind, pMsg->hwnd))
{
GetToDoCtrl().SetFocusToTasks();
return TRUE;
}
break;
case VK_RETURN: // hitting return in filter bar and quick find
if (Prefs().GetFocusTreeOnEnter())
{
CWnd* pFocus = GetFocus();
if (pFocus && (m_filterBar.IsChild(pFocus) || m_cbQuickFind.IsChild(pFocus)))
{
if (!ControlWantsEnter(*pFocus))
GetToDoCtrl().SetFocusToTasks();
return FALSE; // continue routing
}
}
break;
}
}
break;
}
return CFrameWnd::PreTranslateMessage(pMsg);
}
void CToDoListWnd::OnCancel()
{
ASSERT (Prefs().GetEscapeMinimizes());
// if the close button has been configured to Minimize to tray
// then do that here else normal minimize
int nOption = Prefs().GetSysTrayOption();
if (nOption == STO_ONMINCLOSE || nOption == STO_ONCLOSE)
MinimizeToTray();
else
SendMessage(WM_SYSCOMMAND, SC_MINIMIZE, 0);
}
void CToDoListWnd::OnDeleteTask()
{
if (GetToDoCtrl().GetSelectedItem())
GetToDoCtrl().DeleteSelectedTask();
}
void CToDoListWnd::OnDeleteAllTasks()
{
if (GetToDoCtrl().DeleteAllTasks())
{
// m_mgrToDoCtrls.ClearFilePath(GetSelToDoCtrl()); // this will ensure that the user must explicitly overwrite the original file
UpdateStatusbar();
}
}
void CToDoListWnd::OnSave()
{
if (SaveTaskList(GetSelToDoCtrl()) == TDCO_SUCCESS)
UpdateCaption();
}
BOOL CToDoListWnd::DoBackup(int nIndex)
{
if (!Prefs().GetBackupOnSave())
return TRUE;
CString sTDLPath = m_mgrToDoCtrls.GetFilePath(nIndex);
if (sTDLPath.IsEmpty())
return TRUE; // not yet saved
// get backup path
CString sBackupFolder = Prefs().GetBackupLocation(sTDLPath);
sBackupFolder.TrimRight();
// cull old backups
int nKeepBackups = Prefs().GetKeepBackupCount();
if (nKeepBackups)
{
CStringArray aFiles;
CString sPath = CFileBackup::BuildBackupPath(sTDLPath, sBackupFolder, FBS_APPVERSION, _T(""));
CString sDrive, sFolder, sFName, sExt, sPattern;
FileMisc::SplitPath(sPath, &sDrive, &sFolder, &sFName, &sExt);
FileMisc::MakePath(sPath, sDrive, sFolder);
int nFiles = FileMisc::FindFiles(sPath, aFiles, FALSE, sFName + _T("*") + sExt);
if (nFiles >= nKeepBackups)
{
Misc::SortArray(aFiles); // sorts oldest backups first
// cull as required
while (aFiles.GetSize() >= nKeepBackups)
{
DeleteFile(aFiles[0]);
aFiles.RemoveAt(0);
}
}
}
CFileBackup backup;
return backup.MakeBackup(sTDLPath, sBackupFolder, FBS_APPVERSION | FBS_TIMESTAMP, _T(""));
}
TDC_FILE CToDoListWnd::SaveTaskList(int nIndex, LPCTSTR szFilePath, BOOL bAuto)
{
CAutoFlag af(m_bSaving, TRUE);
CString sFilePath = szFilePath ? szFilePath : m_mgrToDoCtrls.GetFilePath(nIndex);
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
tdc.Flush();
// build dialog title, incorporating tasklist name
CString sName = m_mgrToDoCtrls.GetFriendlyProjectName(nIndex);
CEnString sTitle(IDS_SAVETASKLIST_TITLE, sName);
// conditions for saving
// 1. Save As... ie szFilePath != NULL and not empty
// 2. tasklist has been modified
if ((szFilePath && !sFilePath.IsEmpty()) || tdc.IsModified())
{
CPreferences prefs;
// do this in a loop in case the save fails for _any_ reason
while (TRUE)
{
if (sFilePath.IsEmpty()) // means first time save
{
// activate tasklist
if (!SelectToDoCtrl(nIndex, (nIndex != GetSelToDoCtrl())))
return TDCO_CANCELLED;
// use tab text as hint to user
sFilePath = m_mgrToDoCtrls.GetFilePath(nIndex, FALSE);
CFileSaveDialog dialog(sTitle,
GetDefaultFileExt(),
sFilePath,
EOFN_DEFAULTSAVE,
GetFileFilter(FALSE),
this);
// get the user's last choice of saving new tasklists
// as the best hint for this time
BOOL bUnicode = prefs.GetProfileInt(PREF_KEY, _T("UnicodeNewTasklists"), TRUE);
dialog.m_ofn.nFilterIndex = (bUnicode ? 2 : 1);
if (dialog.DoModal(&prefs) != IDOK)
return TDCO_CANCELLED; // user elected not to proceed
// else make sure the file is not readonly
sFilePath = dialog.GetPathName();
// check for format change
bUnicode = (dialog.m_ofn.nFilterIndex == 2);
tdc.SetUnicode(bUnicode);
// save this choice as the best hint for the next new tasklist
prefs.WriteProfileInt(PREF_KEY, _T("UnicodeNewTasklists"), bUnicode);
// else make sure the file is not readonly
if (CDriveInfo::IsReadonlyPath(sFilePath) > 0)
{
CEnString sMessage(IDS_SAVEREADONLY, sFilePath);
if (MessageBox(sMessage, sTitle, MB_OKCANCEL) == IDCANCEL)
return TDCO_CANCELLED; // user elected not to proceed
else
{
sFilePath.Empty(); // try again
continue;
}
}
}
// back file up
DoBackup(nIndex);
// update source control status
const CPreferencesDlg& userPrefs = Prefs();
BOOL bSrcControl = m_mgrToDoCtrls.PathSupportsSourceControl(sFilePath);
tdc.SetStyle(TDCS_ENABLESOURCECONTROL, bSrcControl);
tdc.SetStyle(TDCS_CHECKOUTONLOAD, bSrcControl ? userPrefs.GetAutoCheckOut() : FALSE);
TDC_FILE nSave = TDCO_SUCCESS;
CTaskFile tasks;
// scoped to end status bar progress
// before calling UpdateStatusbar
{
DOPROGRESS(IDS_SAVINGPROGRESS)
nSave = tdc.Save(tasks, sFilePath);
// send to storage as appropriate
TSM_TASKLISTINFO storageInfo;
if (nSave == TDCO_SUCCESS && m_mgrToDoCtrls.GetStorageDetails(nIndex, storageInfo))
{
m_mgrStorage.StoreTasklist(&storageInfo, &tasks, -1, &prefs);
}
}
if (nSave == TDCO_SUCCESS)
{
m_mgrToDoCtrls.SetModifiedStatus(nIndex, FALSE);
m_mgrToDoCtrls.RefreshLastModified(nIndex);
m_mgrToDoCtrls.RefreshReadOnlyStatus(nIndex);
m_mgrToDoCtrls.RefreshPathType(nIndex);
if (userPrefs.GetAddFilesToMRU() && !m_mgrToDoCtrls.UsesStorage(nIndex))
m_mruList.Add(sFilePath);
UpdateCaption();
UpdateStatusbar();
// auto-export after saving
CString sExt;
if (userPrefs.GetAutoExport() && GetAutoExportExtension(sExt))
{
DOPROGRESS(IDS_EXPORTPROGRESS)
// construct output path
CString sExportPath = userPrefs.GetAutoExportFolderPath();
CString sDrive, sFolder, sFName;
FileMisc::SplitPath(sFilePath, &sDrive, &sFolder, &sFName);
if (!sExportPath.IsEmpty() && FileMisc::CreateFolder(sExportPath))
FileMisc::MakePath(sFilePath, NULL, sExportPath, sFName, sExt);
else
FileMisc::MakePath(sFilePath, sDrive, sFolder, sFName, sExt);
// The current contents of 'tasks' is 'All Tasks' and
// 'All Columns' but NOT 'Html Comments'.
// So if user either wants 'Filtered Tasks' or 'Html Comments' or
// only 'Visible Columns' we need to grab the tasks again.
BOOL bFiltered = (userPrefs.GetExportFilteredOnly() && (tdc.HasCustomFilter() || tdc.HasFilter()));
if (bFiltered || userPrefs.GetExportToHTML() || !userPrefs.GetExportAllAttributes())
{
TSD_TASKS nWhatTasks = bFiltered ? TSDT_FILTERED : TSDT_ALL;
TDCGETTASKS filter;
if (!userPrefs.GetExportAllAttributes())
{
CTDCColumnIDArray aCols;
tdc.GetVisibleColumns(aCols);
MapColumnsToAttributes(aCols, filter.aAttribs);
// add comments always
filter.aAttribs.Add(TDCA_COMMENTS);
}
BOOL bHtmlComments = userPrefs.GetExportToHTML();
BOOL bTransform = FileMisc::FileExists(userPrefs.GetSaveExportStylesheet());
// set the html image folder to be the output path with
// an different extension
CString sImgFolder;
if (bHtmlComments)
{
sImgFolder = sFilePath;
FileMisc::ReplaceExtension(sImgFolder, _T("html_images"));
}
tasks.Reset();
GetTasks(tdc, bHtmlComments, bTransform, nWhatTasks, filter, 0, tasks, sImgFolder);
}
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
// want HTML
if (userPrefs.GetExportToHTML())
{
Export2Html(tasks, sFilePath, userPrefs.GetSaveExportStylesheet());
}
else if (userPrefs.GetOtherExporter() != -1)
{
int nExporter = userPrefs.GetOtherExporter();
m_mgrImportExport.ExportTaskList(&tasks, sFilePath, nExporter, TRUE);
}
}
// we're done
break;
}
else if (!bAuto)
{
// error handling if this is not an auto-save
if (nSave == TDCO_NOTALLOWED)
{
CEnString sMessage(IDS_SAVEACCESSDENIED, sFilePath);
if (IDYES == MessageBox(sMessage, sTitle, MB_YESNOCANCEL | MB_ICONEXCLAMATION))
{
sFilePath.Empty(); // try again
continue;
}
else // clear modified status
{
tdc.SetModified(FALSE);
m_mgrToDoCtrls.SetModifiedStatus(nIndex, FALSE);
break;
}
}
else
{
CString sMessage;
switch (nSave)
{
case TDCO_CANCELLED:
break;
case TDCO_BADMSXML:
sMessage.Format(IDS_SAVEBADXML, sFilePath);
break;
case TDCO_INUSE:
sMessage.Format(IDS_SAVESHARINGVIOLATION, sFilePath);
break;
case TDCO_NOTCHECKEDOUT:
sMessage.Format(IDS_SAVENOTCHECKEDOUT, sFilePath);
break;
default:
sMessage.Format(IDS_UNKNOWNSAVEERROR2, sFilePath, (nSave - (int)TDCO_OTHER));
break;
}
if (!sMessage.IsEmpty())
MessageBox(sMessage, sTitle, MB_OK);
}
break; // we're done
}
else // bAuto == TRUE
{
break; // we're done
}
}
}
return TDCO_SUCCESS;
}
BOOL CToDoListWnd::GetAutoExportExtension(CString& sExt) const
{
sExt.Empty();
if (Prefs().GetExportToHTML())
sExt = ".html";
else
{
int nExporter = Prefs().GetOtherExporter();
if (nExporter != -1)
sExt = m_mgrImportExport.GetExporterFileExtension(nExporter);
}
return !sExt.IsEmpty();
}
LPCTSTR CToDoListWnd::GetFileFilter(BOOL bOpen)
{
static CEnString TDLFILEOPENFILTER(IDS_TDLFILEOPENFILTER);
static CEnString XMLFILEOPENFILTER(IDS_XMLFILEOPENFILTER);
static CEnString TDLFILESAVEFILTER(IDS_TDLFILESAVEFILTER);
static CEnString XMLFILESAVEFILTER(IDS_XMLFILESAVEFILTER);
if (Prefs().GetEnableTDLExtension())
return bOpen ? TDLFILEOPENFILTER : TDLFILESAVEFILTER;
// else
return bOpen ? XMLFILEOPENFILTER : XMLFILESAVEFILTER;
}
LPCTSTR CToDoListWnd::GetDefaultFileExt()
{
static LPCTSTR TDLEXT = _T("tdl");
static LPCTSTR XMLEXT = _T("xml");
if (Prefs().GetEnableTDLExtension())
return TDLEXT;
else
return XMLEXT;
}
void CToDoListWnd::UpdateStatusbar()
{
if (!m_sbProgress.IsActive() && GetTDCCount())
{
// get display path
int nTasklist = GetSelToDoCtrl();
const CFilteredToDoCtrl& tdc = GetToDoCtrl(nTasklist);
CEnString sText = m_mgrToDoCtrls.GetDisplayPath(nTasklist);
if (sText.IsEmpty())
sText.LoadString(ID_SB_FILEPATH);
else if (tdc.IsUnicode())
sText += _T(" (Unicode)");
m_statusBar.SetPaneText(m_statusBar.CommandToIndex(ID_SB_FILEPATH), sText);
// get file version
sText.Format(ID_SB_FILEVERSION, tdc.GetFileVersion());
m_statusBar.SetPaneText(m_statusBar.CommandToIndex(ID_SB_FILEVERSION), sText);
}
}
void CToDoListWnd::OnLoad()
{
CPreferences prefs;
CFileOpenDialog dialog(IDS_OPENTASKLIST_TITLE,
GetDefaultFileExt(),
NULL,
EOFN_DEFAULTOPEN | OFN_ALLOWMULTISELECT,
GetFileFilter(TRUE),
this);
const UINT BUFSIZE = 1024 * 5;
static TCHAR FILEBUF[BUFSIZE] = { 0 };
dialog.m_ofn.lpstrFile = FILEBUF;
dialog.m_ofn.nMaxFile = BUFSIZE;
if (dialog.DoModal(&prefs) == IDOK)
{
CWaitCursor cursor;
POSITION pos = dialog.GetStartPosition();
while (pos)
{
CString sTaskList = dialog.GetNextPathName(pos);
TDC_FILE nOpen = OpenTaskList(sTaskList);
if (nOpen == TDCO_SUCCESS)
{
Resize();
UpdateWindow();
}
else
HandleLoadTasklistError(nOpen, sTaskList);
}
RefreshTabOrder();
}
}
void CToDoListWnd::HandleLoadTasklistError(TDC_FILE nErr, LPCTSTR szTaskList)
{
CEnString sMessage, sTitle(IDS_OPENTASKLIST_TITLE);
switch (nErr)
{
case TDCO_SUCCESS:
break; // not an error!
case TDCO_CANCELLED:
break;
case TDCO_NOTEXIST:
sMessage.Format(IDS_TASKLISTNOTFOUND, szTaskList);
break;
case TDCO_NOTTASKLIST:
sMessage.Format(IDS_INVALIDTASKLIST, szTaskList);
break;
case TDCO_NOTALLOWED:
sMessage.Format(IDS_OPENACCESSDENIED, szTaskList);
break;
case TDCO_INUSE:
sMessage.Format(IDS_OPENSHARINGVIOLATION, szTaskList);
break;
case TDCO_BADMSXML:
sMessage.Format(IDS_BADXML, szTaskList);
break;
case TDCO_NOENCRYPTIONDLL:
sMessage.Format(IDS_NOENCRYPTIONDLL, szTaskList);
break;
case TDCO_UNKNOWNENCRYPTION:
sMessage.Format(IDS_UNKNOWNENCRYPTION, szTaskList);
break;
default: // all the other errors
sMessage.Format(IDS_UNKNOWNOPENERROR, szTaskList, nErr - (int)TDCO_OTHER);
break;
}
if (!sMessage.IsEmpty())
MessageBox(sMessage, sTitle, MB_OK);
}
void CToDoListWnd::SaveSettings()
{
CPreferences prefs;
// pos
WINDOWPLACEMENT wp;
GetWindowPlacement(&wp);
prefs.WriteProfileInt(_T("Pos"), _T("Left"), wp.rcNormalPosition.left);
prefs.WriteProfileInt(_T("Pos"), _T("Top"), wp.rcNormalPosition.top);
prefs.WriteProfileInt(_T("Pos"), _T("Right"), wp.rcNormalPosition.right);
prefs.WriteProfileInt(_T("Pos"), _T("Bottom"), wp.rcNormalPosition.bottom);
// FileMisc::LogText(_T("SavePosition: TopLeft=(%d,%d) BotRight=(%d,%d) MinPos=(%d,%d) MaxPos=(%d,%d)"),
// wp.rcNormalPosition.left, wp.rcNormalPosition.top,
// wp.rcNormalPosition.right, wp.rcNormalPosition.bottom,
// wp.ptMaxPosition.x, wp.ptMaxPosition.y,
// wp.ptMinPosition.x, wp.ptMinPosition.y);
prefs.WriteProfileInt(_T("Pos"), _T("Hidden"), !m_bVisible);
prefs.WriteProfileInt(_T("Pos"), _T("Maximized"), IsZoomed());
// version
prefs.WriteProfileInt(_T("Version"), _T("Version"), GetVersion());
// last open files
int nCount = GetTDCCount();
int nSel = GetSelToDoCtrl(); // and last active file
if (nCount) // but don't overwrite files saved in OnQueryEndSession() or OnClose()
{
for (int nTDC = 0, nItem = 0; nTDC < nCount; nTDC++)
{
CString sFilePath = m_mgrToDoCtrls.GetFilePath(nTDC);
TSM_TASKLISTINFO storageInfo;
if (m_mgrToDoCtrls.GetStorageDetails(nTDC, storageInfo))
{
sFilePath = storageInfo.EncodeInfo(Prefs().GetSaveStoragePasswords());
#ifdef _DEBUG
ASSERT(storageInfo.DecodeInfo(sFilePath));
ASSERT(storageInfo.EncodeInfo(TRUE) == sFilePath);
#endif
}
else // make file paths relative
{
FileMisc::MakeRelativePath(sFilePath, FileMisc::GetAppFolder(), FALSE);
}
CString sKey = Misc::MakeKey(_T("LastFile%d"), nItem);
prefs.WriteProfileString(_T("Settings"), sKey, sFilePath);
if (nSel == nTDC)
prefs.WriteProfileString(_T("Settings"), _T("LastActiveFile"), sFilePath);
nItem++;
}
prefs.WriteProfileInt(_T("Settings"), _T("NumLastFiles"), nCount);
}
// other settings
prefs.WriteProfileInt(_T("Settings"), _T("ViewState"), m_nMaxState);
prefs.WriteProfileInt(_T("Settings"), _T("ShowFilterBar"), m_bShowFilterBar);
prefs.WriteProfileInt(_T("Settings"), _T("ToolbarOption"), m_bShowToolbar ? TB_TOOLBARANDMENU : TB_TOOLBARHIDDEN);
prefs.WriteProfileInt(_T("Settings"), _T("ShowProjectName"), m_bShowProjectName);
prefs.WriteProfileInt(_T("Settings"), _T("ShowStatusBar"), m_bShowStatusBar);
prefs.WriteProfileInt(_T("Settings"), _T("ShowTasklistBar"), m_bShowTasklistBar);
prefs.WriteProfileInt(_T("Settings"), _T("ShowTreeListBar"), m_bShowTreeListBar);
if (m_findDlg.GetSafeHwnd())
prefs.WriteProfileInt(_T("Settings"), _T("FindTasksVisible"), m_bFindShowing && m_findDlg.IsWindowVisible());
if (Prefs().GetAddFilesToMRU())
m_mruList.WriteList(prefs, TRUE);
// quick find items
nCount = m_cbQuickFind.GetCount();
prefs.WriteProfileInt(_T("QuickFind"), _T("Count"), nCount);
for (int nItem = 0; nItem < nCount; nItem++)
{
CString sItem, sKey = Misc::MakeKey(_T("Item%d"), nItem);
m_cbQuickFind.GetLBText(nItem, sItem);
prefs.WriteProfileString(_T("QuickFind"), sKey, sItem);
}
// save to permanent storage
prefs.Save();
}
LRESULT CToDoListWnd::OnWebUpdateWizard(WPARAM /*wp*/, LPARAM /*lp*/)
{
ASSERT (Prefs().GetAutoCheckForUpdates());
CheckForUpdates(FALSE);
return 0L;
}
LRESULT CToDoListWnd::OnAddToolbarTools(WPARAM /*wp*/, LPARAM /*lp*/)
{
Misc::ProcessMsgLoop();
AppendTools2Toolbar();
return 0L;
}
TDC_PREPAREPATH CToDoListWnd::PrepareFilePath(CString& sFilePath, TSM_TASKLISTINFO* pInfo)
{
TDC_PREPAREPATH nType = TDCPP_NONE;
TSM_TASKLISTINFO temp;
if (pInfo == NULL)
pInfo = &temp;
// first test for storage
if (pInfo->DecodeInfo(sFilePath, Prefs().GetSaveStoragePasswords()))
{
sFilePath = pInfo->szLocalFileName;
// check for overflow and non-existence
if (FileMisc::FileExists(sFilePath))
nType = TDCPP_STORAGE;
else
sFilePath.Empty();
}
// else it's a file path.
// if it starts with a colon then we need to find the removable drive it's stored on
else if (!sFilePath.IsEmpty())
{
if (sFilePath[0] == ':')
{
for (int nDrive = 4; nDrive <= 26; nDrive++) // from D: upwards
{
if (CDriveInfo::GetType(nDrive) == DRIVE_REMOVABLE)
{
CString sTryPath = CDriveInfo::GetLetter(nDrive) + sFilePath;
if (FileMisc::FileExists(sTryPath))
{
sFilePath = sTryPath;
break; // finished
}
}
}
}
else
FileMisc::MakeFullPath(sFilePath, FileMisc::GetAppFolder()); // handle relative paths
// check for existence
if (FileMisc::FileExists(sFilePath))
nType = TDCPP_FILE;
else
sFilePath.Empty();
}
return nType;
}
LRESULT CToDoListWnd::OnPostOnCreate(WPARAM /*wp*/, LPARAM /*lp*/)
{
// late initialization
CMouseWheelMgr::Initialize();
CEditShortcutMgr::Initialize();
CFocusWatcher::Initialize(this);
InitShortcutManager();
InitMenuIconManager();
// reminders
m_reminders.Initialize(this);
// with or without Stickies Support
const CPreferencesDlg& userPrefs = Prefs();
CString sStickiesPath;
if (userPrefs.GetUseStickies(sStickiesPath))
VERIFY(m_reminders.UseStickies(TRUE, sStickiesPath));
// light boxing
if (Prefs().GetEnableLightboxMgr())
CLightBoxMgr::Initialize(this, m_theme.crAppBackDark);
// add outstanding translated items to dictionary
if (CLocalizer::GetTranslationOption() == ITTTO_ADD2DICTIONARY)
{
CUIntArray aDictVersion, aAppVersion;
VERIFY (FileMisc::GetAppVersion(aAppVersion));
BOOL bUpdateDict = !CLocalizer::GetDictionaryVersion(aDictVersion) ||
aDictVersion.GetSize() < 2;
if (!bUpdateDict)
{
// check if the major or minor version has increased
bUpdateDict = (FileMisc::CompareVersions(aAppVersion, aDictVersion, 2) > 0);
// check for pre-release build then update
if (!bUpdateDict && aAppVersion[2] >= 297)
{
// compare entire version string
bUpdateDict = (FileMisc::CompareVersions(aAppVersion, aDictVersion) > 0);
}
}
if (bUpdateDict)
TranslateUIElements();
}
RestoreVisibility();
// load last open tasklists
CAutoFlag af(m_bReloading, TRUE);
CPreferences prefs;
// initialize Progress first time
m_sbProgress.BeginProgress(m_statusBar, CEnString(IDS_STARTUPPROGRESS));
// open cmdline tasklist
int nTDCCount = prefs.GetProfileInt(_T("Settings"), _T("NumLastFiles"), 0);
if (!m_startupOptions.HasFilePath() || nTDCCount)
{
// if we have a file on the commandline or any previous tasklists
// set the prompt of the initial tasklist to something appropriate
// TODO
}
// theme
SetUITheme(userPrefs.GetUITheme());
// cache empty flag for later
BOOL bStartupEmpty = m_startupOptions.HasFlag(TLD_STARTEMPTY);
// what to (re)load?
BOOL bReloadTasklists = (!bStartupEmpty && userPrefs.GetReloadTasklists());
// filepath overrides
if (m_startupOptions.HasFilePath())
{
ProcessStartupOptions(m_startupOptions);
// don't reload previous if a tasklist was actually loaded
if (!m_mgrToDoCtrls.IsPristine(0))
bReloadTasklists = FALSE;
}
m_startupOptions.Reset(); // always
// load last files
if (bReloadTasklists)
{
// get the last active tasklist
CString sLastActiveFile = prefs.GetProfileString(_T("Settings"), _T("LastActiveFile")), sOrgLastActiveFile;
BOOL bCanDelayLoad = userPrefs.GetEnableDelayedLoading();
for (int nTDC = 0; nTDC < nTDCCount; nTDC++)
{
CString sKey = Misc::MakeKey(_T("LastFile%d"), nTDC);
CString sLastFile = prefs.GetProfileString(_T("Settings"), sKey);
if (!sLastFile.IsEmpty())
{
// delay-open all but the non-active tasklist
// unless the tasklist has reminders
BOOL bActiveTDC = (sLastFile == sLastActiveFile);
if (!bActiveTDC && bCanDelayLoad && !m_reminders.ToDoCtrlHasReminders(sLastFile))
{
DelayOpenTaskList(sLastFile);
}
else
{
TDC_FILE nResult = OpenTaskList(sLastFile, FALSE);
// if the last active tasklist was cancelled then
// delay load it and mark the last active todoctrl as not found
if (bActiveTDC && nResult != TDCO_SUCCESS)
{
sOrgLastActiveFile = sLastActiveFile;
sLastActiveFile.Empty();
if (nResult == TDCO_CANCELLED && bCanDelayLoad)
DelayOpenTaskList(sLastFile);
}
}
}
}
// process all pending messages
Misc::ProcessMsgLoop();
// if the last active tasklist could not be loaded then we need to find another
if (GetTDCCount())
{
// make Last Active Files actual filepaths
PrepareFilePath(sLastActiveFile);
PrepareFilePath(sOrgLastActiveFile);
if (sLastActiveFile.IsEmpty())
{
for (int nTDC = 0; nTDC < GetTDCCount() && sLastActiveFile.IsEmpty(); nTDC++)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nTDC);
// ignore original active tasklist
if (tdc.GetFilePath() != sOrgLastActiveFile)
{
if (VerifyTaskListOpen(nTDC, FALSE))
sLastActiveFile = tdc.GetFilePath();
}
}
}
// if nothing suitable found then create an empty tasklist
if (sLastActiveFile.IsEmpty())
{
if (GetTDCCount() == 0)
CreateNewTaskList(FALSE);
}
else if (!SelectToDoCtrl(sLastActiveFile, FALSE))
SelectToDoCtrl(0, FALSE); // the first one
Resize();
}
}
// if there's only one tasklist open and it's pristine then
// it's the original one so add a sample task unless
// 'empty' flag is set
if (GetTDCCount() == 1 && m_mgrToDoCtrls.IsPristine(0))
{
if (!bStartupEmpty)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
ASSERT (tdc.GetTaskCount() == 0);
tdc.CreateNewTask(CEnString(IDS_SAMPLETASK), TDC_INSERTATTOP, FALSE);
tdc.SetModified(FALSE);
m_mgrToDoCtrls.SetModifiedStatus(0, FALSE);
UpdateCaption();
}
}
else // due task notifications
{
int nDueBy = userPrefs.GetNotifyDueByOnLoad();
if (nDueBy != PFP_DONTNOTIFY)
{
UpdateWindow();
m_tabCtrl.UpdateWindow();
int nCtrls = GetTDCCount();
for (int nCtrl = 0; nCtrl < nCtrls; nCtrl++)
{
if (m_mgrToDoCtrls.IsLoaded(nCtrl))
DoDueTaskNotification(nCtrl, nDueBy);
}
}
}
// refresh toolbar 'tools' buttons unless minimized because
// we must handle it when we're first shown
if (m_bShowToolbar && AfxGetApp()->m_nCmdShow != SW_SHOWMINIMIZED)
AppendTools2Toolbar();
// web update
if (Prefs().GetAutoCheckForUpdates())
PostMessage(WM_WEBUPDATEWIZARD);
// current focus
PostMessage(WM_FW_FOCUSCHANGE, (WPARAM)::GetFocus(), 0L);
RefreshTabOrder();
// end progress before refreshing statusbar
m_sbProgress.EndProgress();
UpdateStatusbar();
// find tasks dialog
InitFindDialog();
if (prefs.GetProfileInt(_T("Settings"), _T("FindTasksVisible"), 0))
{
OnFindTasks();
if (userPrefs.GetRefreshFindOnLoad())
m_findDlg.RefreshSearch();
}
// log the app and its dlls for debugging
FileMisc::LogAppModuleState(FBM_SORTBY_HMODULE);
return 0L;
}
void CToDoListWnd::CheckForUpdates(BOOL bManual)
{
CPreferences prefs;
// only check once a day
int nLastUpdate = prefs.GetProfileInt(_T("Updates"), _T("LastUpdate"), 0);
int nToday = (int)CDateHelper::GetDate(DHD_TODAY);
if (!bManual && nLastUpdate >= nToday)
return;
prefs.WriteProfileInt(_T("Updates"), _T("LastUpdate"), nToday);
// get the app wuw path
CString sFolder, sDrive;
CString sWuwPath = FileMisc::GetAppFolder() + _T("\\WebUpdateSvc.exe");
// check for existence if manual
if (bManual && !FileMisc::FileExists(sWuwPath))
{
LPCTSTR DOWNLOAD_WUW_PATH = _T("http://www.abstractspoon.com/todolist_wuw.zip");
if (MessageBox(IDS_NOWUW, 0, MB_YESNO) == IDYES)
::ShellExecute(NULL, _T("open"), DOWNLOAD_WUW_PATH, NULL, NULL, SW_HIDE);
else
return;
}
// because the download may include the WuW we copy it to a temp name
// so that the original can be overwritten.
CString sWuwPathTemp = FileMisc::GetAppFolder() + _T("\\WebUpdateSvc2.exe");
if (::CopyFile(sWuwPath, sWuwPathTemp, FALSE))
sWuwPath = sWuwPathTemp;
if (bManual)
{
if (m_bUseStagingScript)
::ShellExecute(NULL, _T("open"), sWuwPath, UPDATE_SCRIPT_PATH_STAGING, NULL, SW_HIDE);
else
::ShellExecute(NULL, _T("open"), sWuwPath, UPDATE_SCRIPT_PATH_MANUAL, NULL, SW_HIDE);
}
else
::ShellExecute(NULL, _T("open"), sWuwPath, UPDATE_SCRIPT_PATH, NULL, SW_HIDE);
}
void CToDoListWnd::LoadSettings()
{
// settings
CPreferences prefs;
BOOL bMaxTasklists = prefs.GetProfileInt(_T("Settings"), _T("SimpleMode"), FALSE); // backward compatibility
m_nMaxState = (TDC_MAXSTATE)prefs.GetProfileInt(_T("Settings"), _T("ViewState"), bMaxTasklists ? TDCMS_MAXTASKLIST : TDCMS_NORMAL);
m_bShowFilterBar = prefs.GetProfileInt(_T("Settings"), _T("ShowFilterBar"), m_bShowFilterBar);
m_bShowProjectName = prefs.GetProfileInt(_T("Settings"), _T("ShowProjectName"), m_bShowProjectName);
m_bShowStatusBar = prefs.GetProfileInt(_T("Settings"), _T("ShowStatusBar"), m_bShowStatusBar);
m_statusBar.ShowWindow(m_bShowStatusBar ? SW_SHOW : SW_HIDE);
// toolbar
m_bShowToolbar = (prefs.GetProfileInt(_T("Settings"), _T("ToolbarOption"), TB_TOOLBARANDMENU) != TB_TOOLBARHIDDEN);
m_toolbar.ShowWindow(m_bShowToolbar ? SW_SHOW : SW_HIDE);
m_toolbar.EnableWindow(m_bShowToolbar);
// tabbars
m_bShowTasklistBar = prefs.GetProfileInt(_T("Settings"), _T("ShowTasklistBar"), TRUE);
m_bShowTreeListBar = prefs.GetProfileInt(_T("Settings"), _T("ShowTreeListBar"), TRUE);
// pos
RestorePosition();
// user preferences
const CPreferencesDlg& userPrefs = Prefs();
// MRU
if (userPrefs.GetAddFilesToMRU())
m_mruList.ReadList(prefs);
// note: we do not restore visibility until OnPostOnCreate
// default attributes
UpdateDefaultTaskAttributes(userPrefs);
// hotkey
UpdateGlobalHotkey();
// time periods
CTimeHelper::SetHoursInOneDay(userPrefs.GetHoursInOneDay());
CTimeHelper::SetDaysInOneWeek(userPrefs.GetDaysInOneWeek());
// support for .tdl
CFileRegister filereg(_T("tdl"), _T("tdl_Tasklist"));
if (userPrefs.GetEnableTDLExtension())
filereg.RegisterFileType(_T("Tasklist"), 0);
else
filereg.UnRegisterFileType();
// support for tdl protocol
EnableTDLProtocol(userPrefs.GetEnableTDLProtocol());
// previous quick find items
int nCount = prefs.GetProfileInt(_T("QuickFind"), _T("Count"), 0);
for (int nItem = 0; nItem < nCount; nItem++)
{
CString sKey = Misc::MakeKey(_T("Item%d"), nItem);
m_cbQuickFind.AddUniqueItem(prefs.GetProfileString(_T("QuickFind"), sKey));
}
// Recently modified period
CFilteredToDoCtrl::SetRecentlyModifiedPeriod(userPrefs.GetRecentlyModifiedPeriod());
}
void CToDoListWnd::EnableTDLProtocol(BOOL bEnable)
{
if (bEnable)
{
CRegKey reg;
if (reg.Open(HKEY_CLASSES_ROOT, _T("tdl")) == ERROR_SUCCESS)
{
reg.Write(_T(""), _T("URL: ToDoList protocol"));
reg.Write(_T("URL Protocol"), _T(""));
// write exe name out
CString sAppPath = FileMisc::GetAppFileName() + _T(" -l \"%1\"");
reg.Close();
if (reg.Open(HKEY_CLASSES_ROOT, _T("tdl\\shell\\open\\command")) == ERROR_SUCCESS)
reg.Write(_T(""), sAppPath);
}
}
else
CRegKey::DeleteKey(HKEY_CLASSES_ROOT, _T("tdl"));
}
void CToDoListWnd::RestoreVisibility()
{
const CPreferencesDlg& userPrefs = Prefs();
CPreferences prefs;
int nDefShowState = AfxGetApp()->m_nCmdShow;
BOOL bShowOnStartup = userPrefs.GetShowOnStartup();
BOOL bMaximized = prefs.GetProfileInt(_T("Pos"), _T("Maximized"), FALSE) || (nDefShowState == SW_SHOWMAXIMIZED);
BOOL bMinimized = !bShowOnStartup && (nDefShowState == SW_SHOWMINIMIZED || nDefShowState == SW_SHOWMINNOACTIVE);
if (bMinimized)
{
bMaximized = FALSE; // can't be max-ed and min-ed
m_bStartHidden = TRUE;
}
if (m_bVisible == -1) // not yet set
{
m_bVisible = TRUE;
// the only reason it can be hidden is if it uses the systray
// and the user has elected to not have it show at startup
// and it was hidden the last time it closed or its set to run
// minimized and that is the trigger to hide it
if (!bShowOnStartup && userPrefs.GetUseSysTray())
{
if (prefs.GetProfileInt(_T("Pos"), _T("Hidden"), FALSE))
m_bVisible = FALSE;
// also if wp.showCmd == minimized and we would hide to sys
// tray when minimized then hide here too
else if (nDefShowState == SW_SHOWMINIMIZED || nDefShowState == SW_SHOWMINNOACTIVE)
{
int nSysTrayOption = Prefs().GetSysTrayOption();
if (nSysTrayOption == STO_ONMINIMIZE || nSysTrayOption == STO_ONMINCLOSE)
m_bVisible = FALSE;
}
}
}
if (m_bVisible)
{
int nShowCmd = (bMaximized ? SW_SHOWMAXIMIZED :
(bMinimized ? SW_SHOWMINIMIZED : SW_SHOW));
ShowWindow(nShowCmd);
Invalidate();
UpdateWindow();
}
else
m_bStartHidden = TRUE;
// don't set topmost if maximized
if (userPrefs.GetAlwaysOnTop() && !bMaximized)
SetWindowPos(&wndTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
}
void CToDoListWnd::RestorePosition()
{
CPreferences prefs;
int nLeft = prefs.GetProfileInt(_T("Pos"), _T("Left"), -1);
int nTop = prefs.GetProfileInt(_T("Pos"), _T("Top"), -1);
int nRight = prefs.GetProfileInt(_T("Pos"), _T("Right"), -1);
int nBottom = prefs.GetProfileInt(_T("Pos"), _T("Bottom"), -1);
CRect rect(nLeft, nTop, nRight, nBottom);
if (rect.Width() > 0 && rect.Height() > 0)
{
// ensure this intersects with the desktop by a decent amount
int BORDER = 200;
rect.DeflateRect(BORDER, BORDER);
CRect rScreen;
if (GraphicsMisc::GetAvailableScreenSpace(rect, rScreen))
{
rect.InflateRect(BORDER, BORDER);
// because the position was saved using the results of
// GetWindowPlacement we must use SetWindowPlacement
// to restore the window
WINDOWPLACEMENT wp = { 0 };
wp.rcNormalPosition = rect;
wp.ptMaxPosition.x = -1;
wp.ptMaxPosition.y = -1;
wp.ptMinPosition.x = -1;
wp.ptMinPosition.y = -1;
// FileMisc::LogText(_T("RestorePosition: TopLeft=(%d,%d) BotRight=(%d,%d) MinPos=(%d,%d) MaxPos=(%d,%d)"),
// wp.rcNormalPosition.left, wp.rcNormalPosition.top,
// wp.rcNormalPosition.right, wp.rcNormalPosition.bottom,
// wp.ptMaxPosition.x, wp.ptMaxPosition.y,
// wp.ptMinPosition.x, wp.ptMinPosition.y);
SetWindowPlacement(&wp);
}
else
rect.SetRectEmpty();
}
// first time or monitors changed?
if (rect.IsRectEmpty())
{
rect.SetRect(0, 0, 1024, 730); // default
// make sure it fits the screen
CRect rScreen;
GraphicsMisc::GetAvailableScreenSpace(rScreen);
if (rect.Height() > rScreen.Height())
rect.bottom = rScreen.Height();
MoveWindow(rect);
CenterWindow();
}
}
void CToDoListWnd::OnNew()
{
CreateNewTaskList(FALSE);
RefreshTabOrder();
}
BOOL CToDoListWnd::CreateNewTaskList(BOOL bAddDefTask)
{
CFilteredToDoCtrl* pNew = NewToDoCtrl();
if (pNew)
{
int nNew = AddToDoCtrl(pNew);
// insert a default task
if (bAddDefTask)
{
if (pNew->GetTaskCount() == 0)
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATTOP, FALSE));
}
else // ensure it is empty
pNew->DeleteAllTasks();
// clear modified flag
pNew->SetModified(FALSE);
m_mgrToDoCtrls.SetModifiedStatus(nNew, FALSE);
}
return (pNew != NULL);
}
void CToDoListWnd::OnUpdateDeletetask(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.HasSelection());
}
void CToDoListWnd::OnUpdateEditTasktext(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount == 1 && !tdc.IsReadOnly());
}
void CToDoListWnd::OnUpdateTaskcolor(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.HasSelection() &&
Prefs().GetTextColorOption() == COLOROPT_DEFAULT);
}
void CToDoListWnd::OnUpdateTaskdone(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
if (nSelCount == 1)
pCmdUI->SetCheck(tdc.IsSelectedTaskDone() ? 1 : 0);
pCmdUI->Enable(!tdc.IsReadOnly() && GetToDoCtrl().GetSelectedItem());
}
void CToDoListWnd::OnUpdateDeletealltasks(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && GetToDoCtrl().GetTaskCount());
}
void CToDoListWnd::OnUpdateSave(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(GetTDCCount() && tdc.IsModified() && !tdc.IsReadOnly());
}
void CToDoListWnd::OnUpdateNew(CCmdUI* pCmdUI)
{
pCmdUI->Enable(TRUE);
}
BOOL CToDoListWnd::OnEraseBkgnd(CDC* pDC)
{
if (!GetTDCCount())
return CFrameWnd::OnEraseBkgnd(pDC);
CDialogHelper::ExcludeChild(this, pDC, &m_toolbar);
CDialogHelper::ExcludeChild(this, pDC, &m_statusBar);
CDialogHelper::ExcludeChild(this, pDC, &m_tabCtrl);
CDialogHelper::ExcludeChild(this, pDC, &m_filterBar);
CDialogHelper::ExcludeChild(this, pDC, &GetToDoCtrl());
CRect rClient;
GetClientRect(rClient);
if (CThemed::IsThemeActive())
{
// paint between the window top and the top of the toolbar
// in toolbar color
if (m_bShowToolbar)
{
CRect rToolbar = CDialogHelper::OffsetCtrl(this, &m_toolbar);
pDC->FillSolidRect(rClient.left, rClient.top, rClient.Width(), rToolbar.top, m_theme.crToolbarLight);
rClient.top += rToolbar.bottom;// + 2;
}
// we need to paint a smidgen between the base of the toolbar
// and the top of the tab bar in crAppBackDark
if (WantTasklistTabbarVisible())
{
pDC->FillSolidRect(rClient.left, rClient.top, rClient.Width(), 5, m_theme.crAppBackDark);
rClient.top += 5;
}
// and then the rest in crAppBackLight
pDC->FillSolidRect(rClient, m_theme.crAppBackLight);
}
else
pDC->FillSolidRect(rClient, GetSysColor(COLOR_3DFACE));
// we must draw out own bevel below the toolbar (or menu if the toolbar is not visible)
int nVPos = 0;
if (m_bShowToolbar)
{
if (COSVersion() <= OSV_XP)
pDC->FillSolidRect(rClient.left, nVPos, rClient.Width(), 1, m_theme.crAppLinesDark);
nVPos = m_toolbar.GetHeight() + TB_VOFFSET;
}
pDC->FillSolidRect(rClient.left, nVPos, rClient.Width(), 1, m_theme.crAppLinesDark);
pDC->FillSolidRect(rClient.left, nVPos + 1, rClient.Width(), 1, m_theme.crAppLinesLight);
// bevel below filter bar
if (m_bShowFilterBar)
{
CRect rFilter;
m_filterBar.GetWindowRect(rFilter);
ScreenToClient(rFilter);
int nVPos = rFilter.bottom;
pDC->FillSolidRect(rClient.left, nVPos, rClient.Width(), 1, m_theme.crAppLinesDark);
pDC->FillSolidRect(rClient.left, nVPos + 1, rClient.Width(), 1, m_theme.crAppLinesLight);
}
// bevel above the statusbar if themed
if (m_bShowStatusBar && CThemed::IsThemeActive())
{
CRect rStatBar;
m_statusBar.GetWindowRect(rStatBar);
ScreenToClient(rStatBar);
pDC->FillSolidRect(0, rStatBar.top - 2, rClient.Width(), 1, m_theme.crAppLinesDark);
pDC->FillSolidRect(0, rStatBar.top - 1, rClient.Width(), 1, m_theme.crAppLinesLight);
}
// this is definitely amongst the worst hacks I've ever had to implement.
// It occurs because the CSysImageList class seems to not initialize
// properly unless the main window is visible. so in the case of starting hidden
// or starting minimized we must wait until we become visible before
// adding the tools to the toolbar.
if (m_bStartHidden)
{
m_bStartHidden = FALSE;
PostMessage(WM_ADDTOOLBARTOOLS);
}
return TRUE;
}
void CToDoListWnd::OnSortBy(UINT nCmdID)
{
if (nCmdID == ID_SORT_MULTI)
return;
TDC_COLUMN nSortBy = GetSortBy(nCmdID);
// update todoctrl
GetToDoCtrl().Sort(nSortBy);
}
void CToDoListWnd::OnUpdateSortBy(CCmdUI* pCmdUI)
{
if (pCmdUI->m_nID == ID_SORT_MULTI)
return;
CFilteredToDoCtrl& tdc = GetToDoCtrl();
// disable if column not visible
TDC_COLUMN nSortBy = GetSortBy(pCmdUI->m_nID);
pCmdUI->Enable(tdc.IsColumnShowing(nSortBy));
// only set the radio button if we're not multisorting
BOOL bChecked = !tdc.IsMultiSorting() && (tdc.GetSortBy() == nSortBy);
pCmdUI->SetRadio(bChecked);
// let menu icon manager handle setting selected state
switch (pCmdUI->m_nID)
{
case ID_SORT_NONE:
pCmdUI->Enable(TRUE);
break;
case ID_SORT_BYCOLOR:
pCmdUI->Enable(Prefs().GetTextColorOption() == COLOROPT_DEFAULT);
break;
case ID_SORT_BYPATH:
pCmdUI->Enable(tdc.GetView() != FTCV_TASKTREE);
break;
}
}
TDC_COLUMN CToDoListWnd::GetSortBy(UINT nSortID)
{
switch (nSortID)
{
case ID_SORT_BYNAME: return TDCC_CLIENT;
case ID_SORT_BYID: return TDCC_ID;
case ID_SORT_BYALLOCTO: return TDCC_ALLOCTO;
case ID_SORT_BYALLOCBY: return TDCC_ALLOCBY;
case ID_SORT_BYSTATUS: return TDCC_STATUS;
case ID_SORT_BYCATEGORY: return TDCC_CATEGORY;
case ID_SORT_BYTAG: return TDCC_TAGS;
case ID_SORT_BYPERCENT: return TDCC_PERCENT;
case ID_SORT_BYTIMEEST: return TDCC_TIMEEST;
case ID_SORT_BYTIMESPENT: return TDCC_TIMESPENT;
case ID_SORT_BYSTARTDATE: return TDCC_STARTDATE;
case ID_SORT_BYDUEDATE: return TDCC_DUEDATE;
case ID_SORT_BYDONEDATE: return TDCC_DONEDATE;
case ID_SORT_BYDONE: return TDCC_DONE;
case ID_SORT_BYPRIORITY: return TDCC_PRIORITY;
case ID_SORT_BYCREATEDBY: return TDCC_CREATEDBY;
case ID_SORT_BYCREATIONDATE: return TDCC_CREATIONDATE;
case ID_SORT_BYMODIFYDATE: return TDCC_LASTMOD;
case ID_SORT_BYRISK: return TDCC_RISK;
case ID_SORT_BYEXTERNALID: return TDCC_EXTERNALID;
case ID_SORT_BYCOST: return TDCC_COST;
case ID_SORT_BYVERSION: return TDCC_VERSION;
case ID_SORT_BYRECURRENCE: return TDCC_RECURRENCE;
case ID_SORT_NONE: return TDCC_NONE;
case ID_SORT_BYFLAG: return TDCC_FLAG;
case ID_SORT_BYREMAINING: return TDCC_REMAINING;
case ID_SORT_BYRECENTEDIT: return TDCC_RECENTEDIT;
case ID_SORT_BYICON: return TDCC_ICON;
case ID_SORT_BYFILEREF: return TDCC_FILEREF;
case ID_SORT_BYTIMETRACKING: return TDCC_TRACKTIME;
case ID_SORT_BYPATH: return TDCC_PATH;
case ID_SORT_BYCOLOR: return TDCC_COLOR;
case ID_SORT_BYDEPENDENCY: return TDCC_DEPENDENCY;
case ID_SORT_BYPOSITION: return TDCC_POSITION;
}
// handle custom columns
if (nSortID >= ID_SORT_BYCUSTOMCOLUMN_FIRST && nSortID <= ID_SORT_BYCUSTOMCOLUMN_LAST)
return (TDC_COLUMN)(TDCC_CUSTOMCOLUMN_FIRST + (nSortID - ID_SORT_BYCUSTOMCOLUMN_FIRST));
// all else
ASSERT (0);
return TDCC_NONE;
}
UINT CToDoListWnd::GetSortID(TDC_COLUMN nSortBy)
{
switch (nSortBy)
{
case TDCC_CLIENT: return ID_SORT_BYNAME;
case TDCC_ID: return ID_SORT_BYID;
case TDCC_ALLOCTO: return ID_SORT_BYALLOCTO;
case TDCC_ALLOCBY: return ID_SORT_BYALLOCBY;
case TDCC_STATUS: return ID_SORT_BYSTATUS;
case TDCC_CATEGORY: return ID_SORT_BYCATEGORY;
case TDCC_TAGS: return ID_SORT_BYTAG;
case TDCC_PERCENT: return ID_SORT_BYPERCENT;
case TDCC_TIMEEST: return ID_SORT_BYTIMEEST;
case TDCC_TIMESPENT: return ID_SORT_BYTIMESPENT;
case TDCC_STARTDATE: return ID_SORT_BYSTARTDATE;
case TDCC_DUEDATE: return ID_SORT_BYDUEDATE;
case TDCC_DONEDATE: return ID_SORT_BYDONEDATE;
case TDCC_DONE: return ID_SORT_BYDONE;
case TDCC_PRIORITY: return ID_SORT_BYPRIORITY;
case TDCC_FLAG: return ID_SORT_BYFLAG;
case TDCC_CREATEDBY: return ID_SORT_BYCREATEDBY;
case TDCC_CREATIONDATE: return ID_SORT_BYCREATIONDATE;
case TDCC_LASTMOD: return ID_SORT_BYMODIFYDATE;
case TDCC_RISK: return ID_SORT_BYRISK;
case TDCC_EXTERNALID: return ID_SORT_BYEXTERNALID;
case TDCC_COST: return ID_SORT_BYCOST;
case TDCC_VERSION: return ID_SORT_BYVERSION;
case TDCC_RECURRENCE: return ID_SORT_BYRECURRENCE;
case TDCC_REMAINING: return ID_SORT_BYREMAINING;
case TDCC_RECENTEDIT: return ID_SORT_BYRECENTEDIT;
case TDCC_NONE: return ID_SORT_NONE;
case TDCC_ICON: return ID_SORT_BYICON;
case TDCC_FILEREF: return ID_SORT_BYFILEREF;
case TDCC_TRACKTIME: return ID_SORT_BYTIMETRACKING;
case TDCC_PATH: return ID_SORT_BYPATH;
case TDCC_COLOR: return ID_SORT_BYCOLOR;
case TDCC_POSITION: return ID_SORT_BYPOSITION;
}
// handle custom columns
if (nSortBy >= TDCC_CUSTOMCOLUMN_FIRST && nSortBy < TDCC_CUSTOMCOLUMN_LAST)
return (ID_SORT_BYCUSTOMCOLUMN_FIRST + (nSortBy - TDCC_CUSTOMCOLUMN_FIRST));
// all else
ASSERT (0);
return 0;
}
void CToDoListWnd::OnNewtaskAttopSelected()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATTOPOFSELTASKPARENT));
}
void CToDoListWnd::OnNewtaskAtbottomSelected()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATBOTTOMOFSELTASKPARENT));
}
void CToDoListWnd::OnNewtaskAfterselectedtask()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTAFTERSELTASK));
}
void CToDoListWnd::OnNewtaskBeforeselectedtask()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTBEFORESELTASK));
}
void CToDoListWnd::OnNewsubtaskAtbottom()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATBOTTOMOFSELTASK));
}
void CToDoListWnd::OnNewsubtaskAttop()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATTOPOFSELTASK));
}
void CToDoListWnd::OnNewtaskAttop()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATTOP));
}
void CToDoListWnd::OnNewtaskAtbottom()
{
VERIFY (CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATBOTTOM));
}
BOOL CToDoListWnd::CreateNewTask(LPCTSTR szTitle, TDC_INSERTWHERE nInsertWhere, BOOL bEdit)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
return (tdc.CreateNewTask(szTitle, nInsertWhere, bEdit) != NULL);
}
void CToDoListWnd::OnUpdateSort(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.IsSortable() && tdc.GetTaskCount());
}
void CToDoListWnd::OnEditTaskcolor()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly() && tdc.HasSelection())
{
CEnColorDialog dialog(tdc.GetSelectedTaskColor(), CC_FULLOPEN | CC_RGBINIT);
if (dialog.DoModal() == IDOK)
tdc.SetSelectedTaskColor(dialog.GetColor());
}
}
void CToDoListWnd::OnEditCleartaskcolor()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly() && tdc.HasSelection())
tdc.ClearSelectedTaskColor();
}
void CToDoListWnd::OnUpdateEditCleartaskcolor(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.HasSelection() &&
Prefs().GetTextColorOption() == COLOROPT_DEFAULT &&
tdc.GetSelectedTaskColor() != 0);
}
void CToDoListWnd::OnEditTaskdone()
{
GetToDoCtrl().SetSelectedTaskDone(!GetToDoCtrl().IsSelectedTaskDone());
}
void CToDoListWnd::OnEditTasktext()
{
GetToDoCtrl().EditSelectedTask();
}
void CToDoListWnd::OnTrayIconClick(NMHDR* /*pNMHDR*/, LRESULT* pResult)
{
SetFocus();
Show(Prefs().GetToggleTrayVisibility());
*pResult = 0;
}
LRESULT CToDoListWnd::OnToDoListShowWindow(WPARAM /*wp*/, LPARAM /*lp*/)
{
Show(FALSE);
return 0;
}
LRESULT CToDoListWnd::OnToDoListGetVersion(WPARAM /*wp*/, LPARAM /*lp*/)
{
return GetVersion();
}
LRESULT CToDoListWnd::OnToDoListRefreshPrefs(WPARAM /*wp*/, LPARAM /*lp*/)
{
// sent by the app object if registry settings have changed
ResetPrefs();
// mark all tasklists as needing update
m_mgrToDoCtrls.SetAllNeedPreferenceUpdate(TRUE);
// then update active tasklist
UpdateToDoCtrlPreferences();
return 0;
}
void CToDoListWnd::OnTrayIconDblClk(NMHDR* /*pNMHDR*/, LRESULT* pResult)
{
Show(FALSE);
*pResult = 0;
}
void CToDoListWnd::OnTrayiconCreatetask()
{
Show(FALSE);
// create a task at the top of the tree
GetToDoCtrl().CreateNewTask(CEnString(IDS_TASK), TDC_INSERTATTOP);
}
void CToDoListWnd::OnTrayIconRClick(NMHDR* pNMHDR, LRESULT* pResult)
{
SetForegroundWindow();
// show context menu
CEnMenu menu;
if (menu.LoadMenu(IDR_MISC, m_trayIcon.GetSafeHwnd(), TRUE))
{
CMenu* pSubMenu = menu.GetSubMenu(TRAYICON);
pSubMenu->SetDefaultItem(ID_TRAYICON_SHOW);
if (pSubMenu)
{
m_mgrToDoCtrls.PreparePopupMenu(*pSubMenu, ID_TRAYICON_SHOWDUETASKS1);
NM_TRAYICON* pNMTI = (NM_TRAYICON*)pNMHDR;
// in order to ensure that multiple password dialogs cannot
// appear we must make sure that all the command handling is
// done before we return from here
UINT nCmdID = ::TrackPopupMenu(*pSubMenu, TPM_RETURNCMD | TPM_LEFTALIGN | TPM_LEFTBUTTON,
pNMTI->ptAction.x, pNMTI->ptAction.y, 0, *this, NULL);
PostMessage(WM_NULL);
if (nCmdID != (UINT)-1)
SendMessage(WM_COMMAND, nCmdID);
}
}
*pResult = 0;
}
void CToDoListWnd::OnClose()
{
if (!m_bEndingSession)
{
int nSysTrayOption = Prefs().GetSysTrayOption();
if (nSysTrayOption == STO_ONCLOSE || nSysTrayOption == STO_ONMINCLOSE)
MinimizeToTray();
else // shutdown but user can cancel
DoExit();
}
// else we've already shutdown
}
void CToDoListWnd::MinimizeToTray()
{
// end whatever the user is doing
GetToDoCtrl().Flush();
// save prev state so we can restore properly
CPreferences().WriteProfileInt(_T("Pos"), _T("Maximized"), IsZoomed());
if (Prefs().GetAutoSaveOnSwitchApp())
{
// save all
SaveAll(TDLS_FLUSH | TDLS_AUTOSAVE);
}
// hide main window
Gui::MinToTray(*this); // courtesy of floyd
m_bVisible = FALSE;
// hide find dialog
ShowFindDialog(FALSE);
}
void CToDoListWnd::ShowFindDialog(BOOL bShow)
{
if (bShow)
{
if (m_bVisible && m_findDlg.GetSafeHwnd() && IsWindowVisible())
m_findDlg.Show(TRUE);
}
else // hide
{
if (m_findDlg.GetSafeHwnd())
{
m_bFindShowing = m_findDlg.IsWindowVisible();
m_findDlg.Show(FALSE);
}
else
m_bFindShowing = FALSE;
}
}
void CToDoListWnd::OnTrayiconClose()
{
DoExit();
}
LRESULT CToDoListWnd::OnToDoCtrlNotifyListChange(WPARAM /*wp*/, LPARAM lp)
{
// decide whether the filter controls need updating
switch (lp)
{
case TDCA_ALLOCTO:
case TDCA_ALLOCBY:
case TDCA_STATUS:
case TDCA_CATEGORY:
case TDCA_VERSION:
case TDCA_TAGS:
RefreshFilterControls();
break;
}
return 0L;
}
LRESULT CToDoListWnd::OnToDoCtrlNotifyViewChange(WPARAM wp, LPARAM lp)
{
if (GetTDCCount())
{
if (lp != (LPARAM)wp)
{
CFocusWatcher::UpdateFocus();
m_filterBar.RefreshFilterControls(GetToDoCtrl());
}
else
{
int breakpoint = 0;
}
}
return 0L;
}
LRESULT CToDoListWnd::OnToDoCtrlNotifyTimeTrack(WPARAM /*wp*/, LPARAM lp)
{
BOOL bTrack = lp;
if (bTrack && Prefs().GetExclusiveTimeTracking())
{
// end time tracking on every other tasklist
int nSel = GetSelToDoCtrl();
ASSERT (nSel != -1);
for (int nCtrl = 0; nCtrl < GetTDCCount(); nCtrl++)
{
if (nCtrl != nSel)
GetToDoCtrl(nCtrl).EndTimeTracking();
}
}
return 0L;
}
LRESULT CToDoListWnd::OnToDoCtrlNotifyRecreateRecurringTask(WPARAM wp, LPARAM lp)
{
DWORD dwTaskID = wp, dwNewTaskID = lp;
// is there a reminder that we need to copy for the new task
CFilteredToDoCtrl& tdc = GetToDoCtrl();
TDCREMINDER rem;
int nRem = m_reminders.FindReminder(dwTaskID, &tdc);
if (nRem != -1)
{
// get the existing reminder
m_reminders.GetReminder(nRem, rem);
// init for new task
rem.bEnabled = TRUE;
rem.dDaysSnooze = 0.0;
rem.dwTaskID = dwNewTaskID;
// add for the new task ID
m_reminders.SetReminder(rem);
// delete the original only if the task id has changed
if (dwNewTaskID != dwTaskID)
m_reminders.RemoveReminder(dwTaskID, rem.pTDC);
}
return 0L;
}
LRESULT CToDoListWnd::OnToDoCtrlNotifyMod(WPARAM wp, LPARAM lp)
{
int nTDC = m_mgrToDoCtrls.FindToDoCtrl((HWND)wp);
if (nTDC == -1)
{
// could be a notification from a TDC not yet added
return 0L;
}
BOOL bWasMod = m_mgrToDoCtrls.GetModifiedStatus(nTDC);
m_mgrToDoCtrls.SetModifiedStatus(nTDC, TRUE);
// update the caption only if the control was not previously modified
// or the project name changed
if (!bWasMod)
UpdateCaption();
TDC_ATTRIBUTE nAttrib = (TDC_ATTRIBUTE)lp;
switch (nAttrib)
{
case TDCA_PROJNAME:
{
// update caption if not already done
if (bWasMod)
UpdateCaption();
// update tab order
if (Prefs().GetKeepTabsOrdered())
RefreshTabOrder();
}
break;
// update due items
case TDCA_DONEDATE:
case TDCA_DUEDATE:
OnTimerDueItems(nTDC);
break;
// reminders
case TDCA_DELETE:
m_reminders.RemoveDeletedTaskReminders(&GetToDoCtrl(nTDC));
break;
}
// status bar
UpdateStatusbar();
// refresh toolbar states
PostMessage(WM_IDLEUPDATECMDUI, TRUE);
// do we need to update the current todoctrl's
// custom attributes on the find dialog?
if (m_findDlg.GetSafeHwnd() && nAttrib == TDCA_CUSTOMATTRIBDEFS)
{
UpdateFindDialogCustomAttributes(&GetToDoCtrl());
}
return 0L;
}
void CToDoListWnd::UpdateCaption()
{
int nSel = GetSelToDoCtrl();
CString sProjectName = m_mgrToDoCtrls.UpdateTabItemText(nSel);
CFilteredToDoCtrl& tdc = GetToDoCtrl();
UINT nIDStatus = 0;
if (m_mgrToDoCtrls.GetReadOnlyStatus(nSel) > 0)
nIDStatus = IDS_STATUSREADONLY;
else if (tdc.CompareFileFormat() == TDCFF_NEWER)
nIDStatus = IDS_STATUSNEWERFORMAT;
else if (m_mgrToDoCtrls.GetReadOnlyStatus(nSel) == 0 &&
m_mgrToDoCtrls.PathSupportsSourceControl(nSel))
{
if (tdc.IsCheckedOut())
nIDStatus = IDS_STATUSCHECKEDOUT;
else
nIDStatus = IDS_STATUSCHECKEDIN;
}
else if (!Prefs().GetEnableSourceControl() && m_mgrToDoCtrls.IsSourceControlled(nSel))
{
nIDStatus = IDS_STATUSSOURCECONTROLLED;
}
CString sCaption, sCopyright(MAKEINTRESOURCE(IDS_COPYRIGHT));
CEnString sStatus(nIDStatus);
// add current focus text as required
if (nIDStatus)
{
if (m_bShowStatusBar || m_sCurrentFocus.IsEmpty())
sCaption.Format(_T("%s [%s] - %s"), sProjectName, sStatus, sCopyright);
else
sCaption.Format(_T("%s [%s][%s] - %s"), sProjectName, m_sCurrentFocus, sStatus, sCopyright);
}
else
{
if (m_bShowStatusBar || m_sCurrentFocus.IsEmpty())
sCaption.Format(_T("%s - %s"), sProjectName, sCopyright);
else
sCaption.Format(_T("%s [%s] - %s"), sProjectName, m_sCurrentFocus, sCopyright);
}
// prepend task pathname if tasklist not visible
if (m_nMaxState == TDCMS_MAXCOMMENTS)
{
// quote the path to help it stand-out
CString sTaskPath;
sTaskPath.Format(_T("\"%s\""), tdc.GetSelectedTaskPath(TRUE, 100));
if (!sTaskPath.IsEmpty())
sCaption = sTaskPath + " - " + sCaption;
}
SetWindowText(sCaption);
// set tray tip too
UpdateTooltip();
}
void CToDoListWnd::UpdateTooltip()
{
// base the tooltip on our current caption
CString sTooltip;
GetWindowText(sTooltip);
// move copyright to next line and remove '-'
sTooltip.Replace(_T(" - "), _T("\n"));
// prefix with selected task as first line
CFilteredToDoCtrl& tdc = GetToDoCtrl();
DWORD dwSelID = tdc.GetSelectedTaskID();
if (dwSelID)
{
CString sSelItem = tdc.GetTaskTitle(dwSelID);
// maximum length of tooltip is 128 including null
if (sSelItem.GetLength() > (128 - sTooltip.GetLength() - 6))
{
sSelItem = sSelItem.Left(128 - sTooltip.GetLength() - 6);
sSelItem += _T("...");
}
else if (tdc.GetSelectedCount() > 1)
sSelItem += _T(", ...");
sTooltip = sSelItem + _T("\n") + sTooltip;
}
m_trayIcon.SetTip(sTooltip);
}
BOOL CToDoListWnd::Export2Html(const CTaskFile& tasks, LPCTSTR szFilePath, LPCTSTR szStylesheet) const
{
CWaitCursor cursor;
if (FileMisc::FileExists(szStylesheet))
{
return tasks.TransformToFile(szStylesheet, szFilePath, Prefs().GetHtmlCharSet());
}
// else default export
return m_mgrImportExport.ExportTaskListToHtml(&tasks, szFilePath);
}
void CToDoListWnd::OnSaveas()
{
int nSel = GetSelToDoCtrl();
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSel);
// use tab text as hint to user
CString sFilePath = m_mgrToDoCtrls.GetFilePath(nSel, FALSE);
CPreferences prefs;
// display the dialog
CFileSaveDialog dialog(IDS_SAVETASKLISTAS_TITLE,
GetDefaultFileExt(),
sFilePath,
EOFN_DEFAULTSAVE,
GetFileFilter(FALSE),
this);
// always use the tasklist's own format for initializing the file dialog
dialog.m_ofn.nFilterIndex = tdc.IsUnicode() ? 2 : 1;
int nRes = dialog.DoModal(&prefs);
if (nRes == IDOK)
{
BOOL bWasUnicode = tdc.IsUnicode();
BOOL bUnicode = (dialog.m_ofn.nFilterIndex == 2);
tdc.SetUnicode(bUnicode);
// save this choice as the best hint for the next new tasklist
CPreferences().WriteProfileInt(PREF_KEY, _T("UnicodeNewTasklists"), bUnicode);
if (SaveTaskList(nSel, dialog.GetPathName()) == TDCO_SUCCESS)
{
//m_mgrToDoCtrls.ClearWebDetails(nSel); // because it's no longer a remote file
}
else // restore previous file format
tdc.SetUnicode(bWasUnicode);
UpdateStatusbar();
}
}
void CToDoListWnd::OnUpdateSaveas(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.GetTaskCount() || tdc.IsModified());
}
void CToDoListWnd::OnContextMenu(CWnd* pWnd, CPoint point)
{
static UINT nActiveMenuID = 0; // prevent reentrancy
UINT nMenuID = 0;
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (pWnd == &m_tabCtrl)
{
// if point.x,y are both -1 then we just use the cursor pos
// which is what windows appears to do mostly/sometimes
if (point.x == -1 && point.y == -1)
{
CRect rTab;
if (m_tabCtrl.GetItemRect(m_tabCtrl.GetCurSel(), rTab))
{
point = rTab.CenterPoint();
m_tabCtrl.ClientToScreen(&point);
// load popup menu
nMenuID = TABCTRLCONTEXT;
}
}
else
{
// activate clicked tab
TCHITTESTINFO tcht = { { point.x, point.y }, TCHT_NOWHERE };
m_tabCtrl.ScreenToClient(&tcht.pt);
int nTab = m_tabCtrl.HitTest(&tcht);
if (nTab != -1 && !(tcht.flags & TCHT_NOWHERE))
{
if (nTab != m_tabCtrl.GetCurSel())
{
if (!SelectToDoCtrl(nTab, TRUE))
return; // user cancelled
}
m_tabCtrl.SetFocus(); // give user feedback
// load popup menu
nMenuID = TABCTRLCONTEXT;
}
}
}
else if (pWnd == (CWnd*)&tdc) // try active todoctrl
{
TDC_HITTEST nHit = tdc.HitTest(point);
switch (nHit)
{
case TDCHT_NOWHERE:
break;
case TDCHT_TASKLIST:
case TDCHT_TASK:
if (tdc.WantTaskContextMenu())
{
nMenuID = TASKCONTEXT;
// if point.x,y are both -1 then we request the current
// selection position
if (point.x == -1 && point.y == -1)
{
CRect rSelection;
if (tdc.GetSelectionBoundingRect(rSelection))
{
tdc.ClientToScreen(rSelection);
point.x = min(rSelection.left + 50, rSelection.CenterPoint().x);
point.y = rSelection.top + 8;
}
else
{
nMenuID = 0; // no context menu
}
}
}
break;
case TDCHT_COLUMNHEADER:
nMenuID = HEADERCONTEXT;
break;
}
}
// show the menu
if (nMenuID && nMenuID != nActiveMenuID)
{
CEnMenu menu;
if (menu.LoadMenu(IDR_MISC, NULL, TRUE))
{
CMenu* pPopup = menu.GetSubMenu(nMenuID);
if (pPopup)
{
// some special handling
switch (nMenuID)
{
case TASKCONTEXT:
m_nContextColumnID = tdc.ColumnHitTest(point);
PrepareEditMenu(pPopup);
break;
case TABCTRLCONTEXT:
PrepareSourceControlMenu(pPopup);
break;
}
CToolbarHelper::PrepareMenuItems(pPopup, this);
nActiveMenuID = nMenuID;
pPopup->TrackPopupMenu(TPM_LEFTALIGN | TPM_LEFTBUTTON, point.x, point.y, this);
nActiveMenuID = 0;
}
}
}
else
CFrameWnd::OnContextMenu(pWnd, point);
}
void CToDoListWnd::OnTrayiconShow()
{
Show(FALSE);
}
void CToDoListWnd::OnTrayiconShowDueTasks(UINT nCmdID)
{
int nTDC = nCmdID - ID_TRAYICON_SHOWDUETASKS1;
int nSelTDC = GetSelToDoCtrl();
// verify password if encrypted tasklist is active
// unless app is already visible
if (!m_bVisible || IsIconic() || (nTDC != nSelTDC))
{
if (!VerifyToDoCtrlPassword(nTDC))
return;
}
CFilteredToDoCtrl& tdc = GetToDoCtrl(nTDC);
if (!DoDueTaskNotification(nTDC, PFP_DUETODAY))
{
CEnString sMessage(IDS_NODUETODAY, m_mgrToDoCtrls.GetFriendlyProjectName(nTDC));
MessageBox(sMessage);//, IDS_DUETASKS_TITLE);
}
}
LRESULT CToDoListWnd::OnHotkey(WPARAM /*wp*/, LPARAM /*lp*/)
{
Show(TRUE);
return 0L;
}
BOOL CToDoListWnd::VerifyToDoCtrlPassword() const
{
return VerifyToDoCtrlPassword(GetSelToDoCtrl());
}
BOOL CToDoListWnd::VerifyToDoCtrlPassword(int nIndex) const
{
if (m_bPasswordPrompting)
{
CEnString sExplanation(IDS_SELECTENCRYPTED,
m_mgrToDoCtrls.GetFriendlyProjectName(nIndex));
return GetToDoCtrl(nIndex).VerifyPassword(sExplanation);
}
// else
return TRUE;
}
void CToDoListWnd::Show(BOOL bAllowToggle)
{
if (GetSelToDoCtrl() == -1)
return;
if (!m_bVisible || !IsWindowVisible()) // restore from the tray
{
SetForegroundWindow();
if (!VerifyToDoCtrlPassword())
return;
m_bVisible = TRUE;
Gui::RestoreFromTray(*this, CPreferences().GetProfileInt(_T("Pos"), _T("Maximized"), FALSE));
// restore find dialog
if (m_bFindShowing)
m_findDlg.Show();
}
else if (IsIconic())
{
SetForegroundWindow();
ShowWindow(SW_RESTORE); // this will force a password check
}
// if we're already visible then either bring to the foreground
// or hide if we're right at the top of the z-order
else if (!bAllowToggle || Gui::IsObscured(*this) || !Gui::HasFocus(*this, TRUE))
SetForegroundWindow();
else if (Prefs().GetSysTrayOption() == STO_NONE)
ShowWindow(SW_MINIMIZE);
else // hide to system tray
MinimizeToTray();
// refresh all tasklists if we are visible
if (m_bVisible && !IsIconic())
{
const CPreferencesDlg& userPrefs = Prefs();
if (userPrefs.GetReadonlyReloadOption() != RO_NO)
OnTimerReadOnlyStatus();
if (userPrefs.GetTimestampReloadOption() != RO_NO)
OnTimerTimestampChange();
if (userPrefs.GetEnableSourceControl())
OnTimerCheckoutStatus();
}
GetToDoCtrl().SetFocusToTasks();
}
#ifdef _DEBUG
void CToDoListWnd::OnDebugEndSession()
{
SendMessage(WM_QUERYENDSESSION);
SendMessage(WM_ENDSESSION, 1, 0);
}
void CToDoListWnd::OnDebugShowSetupDlg()
{
CTDLWelcomeWizard dialog;
dialog.DoModal();
}
#endif
void CToDoListWnd::TranslateUIElements()
{
// show progress bar
DOPROGRESS(IDS_UPDATINGDICTIONARY)
// disable translation of top-level menu names in
// IDR_MISC, IDR_PLACEHOLDERS, IDR_TREEDRAGDROP
CLocalizer::IgnoreString(_T("TrayIcon"));
CLocalizer::IgnoreString(_T("TaskContext"));
CLocalizer::IgnoreString(_T("TabCtrl"));
CLocalizer::IgnoreString(_T("TasklistHeader"));
CLocalizer::IgnoreString(_T("CommentsPopup"));
CLocalizer::IgnoreString(_T("ToolsDialog"));
CLocalizer::IgnoreString(_T("TreeDragDrop"));
// disable light box manager temporarily
// to avoid the disabling effect whenever
// a dialog is created
if (Prefs().GetEnableLightboxMgr())
CLightBoxMgr::Release();
CLocalizer::ForceTranslateAllUIElements(IDS_FIRSTSTRING, IDS_LASTSTRING);
// force a redraw of whole UI
if (IsWindowVisible())
{
ShowWindow(SW_HIDE);
ShowWindow(SW_SHOW);
}
SetForegroundWindow();
// restore light box manager
if (Prefs().GetEnableLightboxMgr())
CLightBoxMgr::Initialize(this, m_theme.crAppBackDark);
}
void CToDoListWnd::OnUpdateRecentFileMenu(CCmdUI* pCmdUI)
{
// check that this is not occurring because our CFrameWnd
// base class is routing this to the first item in a submenu
if (pCmdUI->m_pMenu &&
pCmdUI->m_pMenu->GetMenuItemID(pCmdUI->m_nIndex) == (UINT)-1)
return;
m_mruList.UpdateMenu(pCmdUI);
}
BOOL CToDoListWnd::OnOpenRecentFile(UINT nID)
{
ASSERT(nID >= ID_FILE_MRU_FILE1);
ASSERT(nID < ID_FILE_MRU_FILE1 + (UINT)m_mruList.GetSize());
int nIndex = nID - ID_FILE_MRU_FILE1;
CString sTaskList = m_mruList[nIndex];
TDC_FILE nOpen = OpenTaskList(sTaskList);
if (nOpen == TDCO_SUCCESS)
{
Resize();
UpdateWindow();
}
else
{
HandleLoadTasklistError(nOpen, sTaskList);
if (nOpen != TDCO_CANCELLED)
m_mruList.Remove(nIndex);
}
RefreshTabOrder();
// always return TRUE to say we handled it
return TRUE;
}
void CToDoListWnd::RefreshTabOrder()
{
if (Prefs().GetKeepTabsOrdered())
{
int nSelOrg = GetSelToDoCtrl();
int nSel = m_mgrToDoCtrls.SortToDoCtrlsByName();
if (nSel != nSelOrg)
SelectToDoCtrl(nSel, FALSE);
}
}
TDC_FILE CToDoListWnd::DelayOpenTaskList(LPCTSTR szFilePath)
{
ASSERT (Prefs().GetEnableDelayedLoading()); // sanity check
// decode/prepare filepath
CString sFilePath(szFilePath);
TSM_TASKLISTINFO storageInfo;
TDC_PREPAREPATH nPathType = PrepareFilePath(sFilePath, &storageInfo);
if (nPathType == TDCPP_NONE)
return TDCO_NOTEXIST;
// see if the tasklist is already open
if (SelectToDoCtrl(sFilePath, TRUE))
return TDCO_SUCCESS;
// delay load the file, visible but disabled
CFilteredToDoCtrl* pTDC = NewToDoCtrl(TRUE, FALSE);
// if this is a 'special' temp file then assume TDL automatically
// named it when handling WM_ENDSESSION.
BOOL bDelayLoad = !IsEndSessionFilePath(sFilePath);
COleDateTime dtEarliest;
if (bDelayLoad)
bDelayLoad = pTDC->DelayLoad(sFilePath, dtEarliest);
if (bDelayLoad)
{
// now we have to check for whether the tasklist has due tasks
// and the user wants notification
int nNotifyDueBy = Prefs().GetNotifyDueByOnLoad();
if (nNotifyDueBy != PFP_DONTNOTIFY && CDateHelper::IsDateSet(dtEarliest.m_dt))
{
// check the date against when the user wants notifying
DH_DATE nDate = DHD_TODAY;
switch (nNotifyDueBy)
{
case PFP_DUETODAY: /*nDate = DHD_TODAY;*/ break;
case PFP_DUETOMORROW: nDate = DHD_TOMORROW; break;
case PFP_DUETHISWEEK: nDate = DHD_ENDTHISWEEK; break;
case PFP_DUENEXTWEEK: nDate = DHD_ENDNEXTWEEK; break;
case PFP_DUETHISMONTH: nDate = DHD_ENDTHISMONTH; break;
case PFP_DUENEXTMONTH: nDate = DHD_ENDNEXTMONTH; break;
default: ASSERT (0);
}
COleDateTime dtDueWhen = CDateHelper::GetDate(nDate);
bDelayLoad = (dtDueWhen < dtEarliest);
}
}
// if the delay load failed for any reason we need to delete the tasklist
// and fallback on the default load mechanism
if (!bDelayLoad)
{
pTDC->DestroyWindow();
delete pTDC;
// note: we use the original filepath in case it is
// actually storage info
return OpenTaskList(szFilePath, FALSE);
}
int nCtrl = m_mgrToDoCtrls.AddToDoCtrl(pTDC, &storageInfo, FALSE); // FALSE == not yet loaded
// update due item status
if (CDateHelper::IsDateSet(dtEarliest))
{
TDCM_DUESTATUS nStatus = TDCM_FUTURE;
COleDateTime dtToday = COleDateTime::GetCurrentTime();
if (floor(dtEarliest) < floor(dtToday))
nStatus = TDCM_PAST;
else if (floor(dtEarliest) == floor(dtToday))
nStatus = TDCM_TODAY;
m_mgrToDoCtrls.SetDueItemStatus(nCtrl, nStatus);
}
return TDCO_SUCCESS;
}
CString CToDoListWnd::GetEndSessionFilePath()
{
return FileMisc::GetTempFileName(_T("tde"));
}
BOOL CToDoListWnd::IsEndSessionFilePath(const CString& sFilePath)
{
if (!FileMisc::FileExists(sFilePath))
return FALSE;
if (!FileMisc::HasExtension(sFilePath, _T("tmp")))
return FALSE;
if (!FileMisc::IsTempFile(sFilePath))
return FALSE;
if (FileMisc::GetFileNameFromPath(sFilePath).Find(_T("tde")) != 0)
return FALSE;
// passed all the tests
return TRUE;
}
TDC_ARCHIVE CToDoListWnd::GetAutoArchiveOptions(LPCTSTR szFilePath, CString& sArchivePath, BOOL& bRemoveFlagged) const
{
TDC_ARCHIVE nRemove = TDC_REMOVENONE;
bRemoveFlagged = FALSE;
const CPreferencesDlg& userPrefs = Prefs();
if (userPrefs.GetAutoArchive())
{
if (userPrefs.GetRemoveArchivedTasks())
{
if (userPrefs.GetRemoveOnlyOnAbsoluteCompletion())
nRemove = TDC_REMOVEIFSIBLINGSANDSUBTASKSCOMPLETE;
else
nRemove = TDC_REMOVEALL;
bRemoveFlagged = !userPrefs.GetDontRemoveFlagged();
}
sArchivePath = m_mgrToDoCtrls.GetArchivePath(szFilePath);
}
else
sArchivePath.Empty();
return nRemove;
}
TDC_FILE CToDoListWnd::OpenTaskList(LPCTSTR szFilePath, BOOL bNotifyDueTasks)
{
CString sFilePath(szFilePath);
TDC_PREPAREPATH nType = PrepareFilePath(sFilePath);
if (nType == TDCPP_NONE)
return TDCO_NOTEXIST;
// see if the tasklist is already open
int nExist = m_mgrToDoCtrls.FindToDoCtrl(sFilePath);
if (nExist != -1)
{
// reload provided there are no existing changes
// and the timestamp has changed
if (!m_mgrToDoCtrls.GetModifiedStatus(nExist) &&
m_mgrToDoCtrls.RefreshLastModified(nExist))
{
ReloadTaskList(nExist, bNotifyDueTasks);
}
// then select
if (SelectToDoCtrl(nExist, TRUE))
return TDCO_SUCCESS;
}
// create a new todoltrl for this tasklist
const CPreferencesDlg& userPrefs = Prefs();
CFilteredToDoCtrl* pTDC = NewToDoCtrl();
CHoldRedraw hr(pTDC->GetSafeHwnd());
// handles simple and storage tasklists
// we use szFilePath because it may be storage Info not a true path
TSM_TASKLISTINFO storageInfo;
TDC_FILE nOpen = OpenTaskList(pTDC, sFilePath, &storageInfo);
if (nOpen == TDCO_SUCCESS)
{
int nTDC = AddToDoCtrl(pTDC, &storageInfo);
// notify readonly
CheckNotifyReadonly(nTDC);
// reload any reminders
m_reminders.AddToDoCtrl(*pTDC);
// notify user of due tasks if req
if (bNotifyDueTasks)
DoDueTaskNotification(nTDC, userPrefs.GetNotifyDueByOnLoad());
// save checkout status
if (userPrefs.GetAutoCheckOut())
m_mgrToDoCtrls.SetLastCheckoutStatus(nTDC, pTDC->IsCheckedOut());
// check for automatic naming when handling WM_ENDSESSION.
// so we clear the filename and mark it as modified
if (IsEndSessionFilePath(sFilePath))
{
pTDC->ClearFilePath();
pTDC->SetModified();
}
UpdateCaption();
UpdateStatusbar();
OnTimerDueItems(nTDC);
// update search
if (userPrefs.GetRefreshFindOnLoad() && m_findDlg.GetSafeHwnd())
m_findDlg.RefreshSearch();
}
else if (GetTDCCount() >= 1) // only delete if there's another ctrl existing
{
pTDC->DestroyWindow();
delete pTDC;
}
else // re-add
{
AddToDoCtrl(pTDC);
}
return nOpen;
}
TDC_FILE CToDoListWnd::OpenTaskList(CFilteredToDoCtrl* pTDC, LPCTSTR szFilePath, TSM_TASKLISTINFO* pInfo)
{
CString sFilePath(szFilePath);
CTaskFile tasks;
TDC_FILE nOpen = TDCO_UNSET;
TSM_TASKLISTINFO storageInfo;
TDC_PREPAREPATH nType = PrepareFilePath(sFilePath, &storageInfo);
// handle bad path
if ((szFilePath && *szFilePath) && sFilePath.IsEmpty())
return TDCO_NOTEXIST;
DOPROGRESS(IDS_LOADINGPROGRESS)
if (sFilePath.IsEmpty())
{
sFilePath = pTDC->GetFilePath(); // ie. reload
}
else
{
switch (nType)
{
case TDCPP_STORAGE:
{
CPreferences prefs;
// retrieve file from storage
if (m_mgrStorage.RetrieveTasklist(&storageInfo, &tasks, -1, &prefs))
{
// handle returned tasks
if (tasks.GetTaskCount())
{
// merge returned tasks with this tasklist
// TODO
nOpen = TDCO_SUCCESS;
}
// return update storage info
if (pInfo)
*pInfo = storageInfo;
}
else
nOpen = TDCO_CANCELLED;
}
break;
case TDCPP_FILE:
{
BOOL bSrcControl = m_mgrToDoCtrls.PathSupportsSourceControl(szFilePath);
pTDC->SetStyle(TDCS_ENABLESOURCECONTROL, bSrcControl);
pTDC->SetStyle(TDCS_CHECKOUTONLOAD, bSrcControl ? Prefs().GetAutoCheckOut() : FALSE);
}
break;
case TDCPP_NONE:
default:
ASSERT(0);
break;
}
}
// has the load already been handled?
if (nOpen == TDCO_UNSET)
nOpen = pTDC->Load(sFilePath, tasks);
if (nOpen == TDCO_SUCCESS)
{
// update readonly status
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(*pTDC);
const CPreferencesDlg& userPrefs = Prefs();
// certain operations cannot be performed on 'storage' tasklists
if (nType == TDCPP_FILE)
{
// archive completed tasks?
if (!pTDC->IsReadOnly())
{
CString sArchivePath;
BOOL bRemoveFlagged;
TDC_ARCHIVE nRemove = GetAutoArchiveOptions(szFilePath, sArchivePath, bRemoveFlagged);
if (!sArchivePath.IsEmpty())
pTDC->ArchiveDoneTasks(sArchivePath, nRemove, bRemoveFlagged);
}
if (userPrefs.GetAddFilesToMRU())
m_mruList.Add(sFilePath);
}
if (userPrefs.GetExpandTasksOnLoad())
pTDC->ExpandTasks(TDCEC_ALL);
// update find dialog with this ToDoCtrl's custom attributes
UpdateFindDialogCustomAttributes(pTDC);
}
else
pTDC->SetModified(FALSE);
return nOpen;
}
void CToDoListWnd::CheckNotifyReadonly(int nIndex) const
{
ASSERT(nIndex != -1);
const CPreferencesDlg& userPrefs = Prefs();
if (nIndex >= 0 && userPrefs.GetNotifyReadOnly())
{
CEnString sMessage;
CString sDisplayPath = m_mgrToDoCtrls.GetDisplayPath(nIndex);
CString sFilePath = m_mgrToDoCtrls.GetFilePath(nIndex);
const CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
if (CDriveInfo::IsReadonlyPath(sFilePath) > 0)
sMessage.Format(IDS_OPENREADONLY, sDisplayPath);
else if (!userPrefs.GetEnableSourceControl() && m_mgrToDoCtrls.IsSourceControlled(nIndex))
sMessage.Format(IDS_OPENSOURCECONTROLLED, sDisplayPath);
else if (tdc.CompareFileFormat() == TDCFF_NEWER)
sMessage.Format(IDS_OPENNEWER, sDisplayPath);
if (!sMessage.IsEmpty())
MessageBox(sMessage, IDS_OPENTASKLIST_TITLE);
}
}
void CToDoListWnd::UpdateFindDialogCustomAttributes(const CFilteredToDoCtrl* pTDC)
{
if (pTDC == NULL && GetTDCCount() == 0)
return; // nothing to do
CTDCCustomAttribDefinitionArray aTDCAttribDefs, aAllAttribDefs;
// all tasklists
int nTDC = GetTDCCount();
while (nTDC--)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl(nTDC);
tdc.GetCustomAttributeDefs(aTDCAttribDefs);
CTDCCustomAttributeHelper::AppendUniqueAttributes(aTDCAttribDefs, aAllAttribDefs);
}
// active tasklist
if (pTDC == NULL)
{
ASSERT(GetTDCCount() > 0);
pTDC = &GetToDoCtrl();
}
ASSERT (pTDC);
pTDC->GetCustomAttributeDefs(aTDCAttribDefs);
// do the update
m_findDlg.SetCustomAttributes(aTDCAttribDefs, aAllAttribDefs);
}
BOOL CToDoListWnd::DoDueTaskNotification(int nDueBy)
{
return DoDueTaskNotification(GetSelToDoCtrl(), nDueBy);
}
BOOL CToDoListWnd::DoDueTaskNotification(int nTDC, int nDueBy)
{
// check userPrefs
if (nDueBy == -1)
return TRUE; // nothing to do
if (nTDC != -1 && !VerifyTaskListOpen(nTDC, FALSE))
return TRUE; // no error. user cancelled
CFilteredToDoCtrl& tdc = m_mgrToDoCtrls.GetToDoCtrl(nTDC);
const CPreferencesDlg& userPrefs = Prefs();
// preferences
BOOL bParentTitleCommentsOnly = userPrefs.GetExportParentTitleCommentsOnly();
BOOL bDueTaskTitlesOnly = userPrefs.GetDueTaskTitlesOnly();
CString sStylesheet = userPrefs.GetDueTaskStylesheet();
BOOL bTransform = FileMisc::FileExists(sStylesheet);
BOOL bHtmlNotify = userPrefs.GetDisplayDueTasksInHtml();
DWORD dwFlags = TDCGTF_FILENAME;
if (bHtmlNotify)
dwFlags |= TDCGTF_HTMLCOMMENTS;
if (bTransform)
dwFlags |= TDCGTF_TRANSFORM;
// due task notification preference overrides Export preference
if (bDueTaskTitlesOnly)
dwFlags |= TDCGTF_TITLESONLY;
else if (bParentTitleCommentsOnly)
dwFlags |= TDCGTF_PARENTTITLECOMMENTSONLY;
TDC_GETTASKS nFilter = TDCGT_DUE;
UINT nIDDueBy = IDS_DUETODAY;
switch (nDueBy)
{
case PFP_DUETODAY:
break; // done
case PFP_DUETOMORROW:
nIDDueBy = IDS_DUETOMORROW;
nFilter = TDCGT_DUETOMORROW;
break;
case PFP_DUETHISWEEK:
nIDDueBy = IDS_DUETHISWEEK;
nFilter = TDCGT_DUETHISWEEK;
break;
case PFP_DUENEXTWEEK:
nIDDueBy = IDS_DUENEXTWEEK;
nFilter = TDCGT_DUENEXTWEEK;
break;
case PFP_DUETHISMONTH:
nIDDueBy = IDS_DUETHISMONTH;
nFilter = TDCGT_DUETHISMONTH;
break;
case PFP_DUENEXTMONTH:
nIDDueBy = IDS_DUENEXTMONTH;
nFilter = TDCGT_DUENEXTMONTH;
break;
default:
ASSERT (0);
return FALSE;
}
TDCGETTASKS filter(nFilter, dwFlags);
filter.sAllocTo = userPrefs.GetDueTaskPerson();
CTaskFile tasks;
if (!tdc.GetTasks(tasks, filter))
return FALSE;
// set an appropriate title
tasks.SetReportAttributes(CEnString(nIDDueBy));
tasks.SetCharSet(userPrefs.GetHtmlCharSet());
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
// nasty hack to prevent exporters adding space for notes
BOOL bSpaceForNotes = userPrefs.GetExportSpaceForNotes();
if (bSpaceForNotes)
CPreferences().WriteProfileInt(PREF_KEY, _T("ExportSpaceForNotes"), FALSE);
// different file for each
CString sTempFile;
sTempFile.Format(_T("ToDoList.due.%d"), nTDC);
sTempFile = FileMisc::GetTempFileName(sTempFile, bHtmlNotify ? _T("html") : _T("txt"));
BOOL bRes = FALSE;
if (bHtmlNotify) // display in browser?
{
if (bTransform)
bRes = tasks.TransformToFile(sStylesheet, sTempFile, userPrefs.GetHtmlCharSet());
else
bRes = m_mgrImportExport.ExportTaskListToHtml(&tasks, sTempFile);
}
else
bRes = m_mgrImportExport.ExportTaskListToText(&tasks, sTempFile);
if (bRes)
{
Show(FALSE);
m_mgrToDoCtrls.ShowDueTaskNotification(nTDC, sTempFile, bHtmlNotify);
}
// undo hack
if (bSpaceForNotes)
CPreferences().WriteProfileInt(PREF_KEY, _T("ExportSpaceForNotes"), TRUE);
return TRUE;
}
CString CToDoListWnd::GetTitle()
{
static CString sTitle(_T("ToDoList 6.8.10 Feature Release"));
CLocalizer::IgnoreString(sTitle);
return sTitle;
}
void CToDoListWnd::OnAbout()
{
CString sVersion;
sVersion.Format(_T("%s (c) AbstractSpoon 2003-14"), GetTitle());
CLocalizer::IgnoreString(sVersion);
CAboutDlg dialog(IDR_MAINFRAME,
ABS_LISTCOPYRIGHT,
sVersion,
CEnString(IDS_ABOUTHEADING),
CEnString(IDS_ABOUTCONTRIBUTION),
// format the link into the license so that it is not translated
CEnString(IDS_LICENSE, _T("\"http://www.opensource.org/licenses/eclipse-1.0.php\"")),
1, 1, 12, 1, 250);
dialog.DoModal();
}
void CToDoListWnd::OnPreferences()
{
DoPreferences();
}
void CToDoListWnd::DoPreferences(int nInitPage)
{
// take a copy of current userPrefs to check changes against
const CPreferencesDlg curPrefs;
// kill timers
SetTimer(TIMER_READONLYSTATUS, FALSE);
SetTimer(TIMER_TIMESTAMPCHANGE, FALSE);
SetTimer(TIMER_CHECKOUTSTATUS, FALSE);
SetTimer(TIMER_AUTOSAVE, FALSE);
SetTimer(TIMER_TIMETRACKING, FALSE);
SetTimer(TIMER_AUTOMINIMIZE, FALSE);
// restore translation of dynamic menu items shortcut prefs
EnableDynamicMenuTranslation(TRUE);
ASSERT(m_pPrefs);
UINT nRet = m_pPrefs->DoModal(nInitPage);
// updates userPrefs
RedrawWindow();
ResetPrefs();
const CPreferencesDlg& userPrefs = Prefs();
if (nRet == IDOK)
{
SetUITheme(userPrefs.GetUITheme());
// lightbox mgr
if (curPrefs.GetEnableLightboxMgr() != userPrefs.GetEnableLightboxMgr())
{
if (userPrefs.GetEnableLightboxMgr())
CLightBoxMgr::Initialize(this, m_theme.crAppBackDark);
else
CLightBoxMgr::Release();
}
// mark all todoctrls as needing refreshing
m_mgrToDoCtrls.SetAllNeedPreferenceUpdate(TRUE);
// delete fonts if they appear to have changed
// and recreate in UpdateToDoCtrlPrefs
CString sFaceName;
int nFontSize;
if (!userPrefs.GetTreeFont(sFaceName, nFontSize) || !GraphicsMisc::SameFont(m_fontTree, sFaceName, nFontSize))
GraphicsMisc::VerifyDeleteObject(m_fontTree);
if (!userPrefs.GetCommentsFont(sFaceName, nFontSize) || !GraphicsMisc::SameFont(m_fontComments, sFaceName, nFontSize))
GraphicsMisc::VerifyDeleteObject(m_fontComments);
BOOL bResizeDlg = FALSE;
// topmost
BOOL bTopMost = (userPrefs.GetAlwaysOnTop() && !IsZoomed());
SetWindowPos(bTopMost ? &wndTopMost : &wndNoTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
// tray icon
m_trayIcon.ShowTrayIcon(userPrefs.GetUseSysTray());
// support for .tdl
if (curPrefs.GetEnableTDLExtension() != userPrefs.GetEnableTDLExtension())
{
CFileRegister filereg(_T("tdl"), _T("tdl_Tasklist"));
if (userPrefs.GetEnableTDLExtension())
filereg.RegisterFileType(_T("Tasklist"), 0);
else
filereg.UnRegisterFileType();
}
// support for tdl web protocol
if (curPrefs.GetEnableTDLProtocol() != userPrefs.GetEnableTDLProtocol())
EnableTDLProtocol(userPrefs.GetEnableTDLProtocol());
// language
CString sLangFile = userPrefs.GetLanguageFile();
BOOL bAdd2Dict = userPrefs.GetEnableAdd2Dictionary();
if (UpdateLanguageTranslationAndRestart(sLangFile, bAdd2Dict, curPrefs))
{
DoExit(TRUE);
return;
}
// default task attributes
UpdateDefaultTaskAttributes(userPrefs);
// source control
BOOL bSourceControl = userPrefs.GetEnableSourceControl();
if (curPrefs.GetEnableSourceControl() != bSourceControl ||
curPrefs.GetSourceControlLanOnly() != userPrefs.GetSourceControlLanOnly())
{
// update all open files to ensure they're in the right state
int nCtrl = GetTDCCount();
while (nCtrl--)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
// check files in if we're disabling sc and this file is
// checked out. however although we
// are checking in, the file cannot be edited by the user
// until they remove the file from under source control
if (!bSourceControl && tdc.IsCheckedOut())
{
if (tdc.IsModified())
tdc.Save();
tdc.CheckIn();
}
// else checkout if we're enabling and auto-checkout is also enabled
else if (bSourceControl)
{
// there can be two reasons for wanting to check out a file
// either the autocheckout preference is set or its a local
// file which is not checked out but has been modified and source
// control now covers all files in which case we save it first
BOOL bPathSupports = m_mgrToDoCtrls.PathSupportsSourceControl(nCtrl);
BOOL bNeedsSave = bPathSupports && !tdc.IsCheckedOut() && tdc.IsModified();
BOOL bWantCheckOut = bNeedsSave || (bPathSupports && userPrefs.GetAutoCheckOut());
if (bNeedsSave)
tdc.Save(); // save silently
tdc.SetStyle(TDCS_ENABLESOURCECONTROL, bPathSupports);
tdc.SetStyle(TDCS_CHECKOUTONLOAD, bPathSupports && userPrefs.GetAutoCheckOut());
if (bWantCheckOut && !tdc.IsCheckedOut())
tdc.CheckOut();
}
// re-sync
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.RefreshModifiedStatus(nCtrl);
m_mgrToDoCtrls.RefreshLastModified(nCtrl);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
}
}
m_toolbar.GetToolBarCtrl().HideButton(ID_TOOLS_TOGGLECHECKIN, !bSourceControl);
// check box in front of task title.
// this is tricky because the checkbox won't display if the completion
// column is visible. ie. the completion column take precedence.
// so if the user has just turned on the checkbox in front of
// the task title then we need to turn off the completion column
// on all those tasklists currently showing it. But we only need to do
// this on those tasklists which are managing their own columns else
// it will be handled when we update the preferences in due course.
if (userPrefs.GetTreeCheckboxes() && !curPrefs.GetTreeCheckboxes())
{
int nCtrl = GetTDCCount();
while (nCtrl--)
{
if (m_mgrToDoCtrls.HasOwnColumns(nCtrl))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
if (tdc.IsColumnShowing(TDCC_DONE))
{
CTDCColumnIDArray aColumns;
int nCol = tdc.GetVisibleColumns(aColumns);
while (nCol--)
{
if (aColumns[nCol] == TDCC_DONE)
{
aColumns.RemoveAt(nCol);
break;
}
}
tdc.SetVisibleColumns(aColumns);
}
}
}
}
// same again for task icons
if (userPrefs.GetTreeTaskIcons() && !curPrefs.GetTreeTaskIcons())
{
int nCtrl = GetTDCCount();
while (nCtrl--)
{
if (m_mgrToDoCtrls.HasOwnColumns(nCtrl))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
if (tdc.IsColumnShowing(TDCC_ICON))
{
CTDCColumnIDArray aColumns;
int nCol = tdc.GetVisibleColumns(aColumns);
while (nCol--)
{
if (aColumns[nCol] == TDCC_ICON)
{
aColumns.RemoveAt(nCol);
break;
}
}
tdc.SetVisibleColumns(aColumns);
}
}
}
}
// menu icons
UINT nPrevID = MapNewTaskPos(curPrefs.GetNewTaskPos(), FALSE);
m_mgrMenuIcons.ChangeImageID(nPrevID, GetNewTaskCmdID());
nPrevID = MapNewTaskPos(curPrefs.GetNewSubtaskPos(), TRUE);
m_mgrMenuIcons.ChangeImageID(nPrevID, GetNewSubtaskCmdID());
// reload menu
LoadMenubar();
// tab bar
bResizeDlg |= (curPrefs.GetAutoHideTabbar() != userPrefs.GetAutoHideTabbar());
if (curPrefs.GetStackTabbarItems() != userPrefs.GetStackTabbarItems())
{
BOOL bStackTabbar = userPrefs.GetStackTabbarItems();
bResizeDlg = TRUE;
m_tabCtrl.ModifyStyle(bStackTabbar ? 0 : TCS_MULTILINE, bStackTabbar ? TCS_MULTILINE : 0);
}
else
m_tabCtrl.Invalidate(); // handle priority colour changes
// visible filter controls
if (m_bShowFilterBar)
bResizeDlg = TRUE;
BOOL bEnableMultiSel = userPrefs.GetMultiSelFilters();
BOOL bPrevMultiSel = curPrefs.GetMultiSelFilters();
if (bPrevMultiSel != bEnableMultiSel)
{
m_filterBar.EnableMultiSelection(bEnableMultiSel);
// if it was was previously multisel (but not now) then
// refresh the filter because we may have gone from
// multiple selection down to only one
OnViewRefreshfilter();
}
m_filterBar.ShowDefaultFilters(userPrefs.GetShowDefaultFilters());
// title filter option
PUIP_MATCHTITLE nTitleOption = userPrefs.GetTitleFilterOption();
PUIP_MATCHTITLE nPrevTitleOption = curPrefs.GetTitleFilterOption();
if (nPrevTitleOption != nTitleOption)
OnViewRefreshfilter();
// inherited parent task attributes for new tasks
CTDCAttributeArray aParentAttrib;
BOOL bUpdateAttrib;
userPrefs.GetParentAttribsUsed(aParentAttrib, bUpdateAttrib);
CFilteredToDoCtrl::SetInheritedParentAttributes(aParentAttrib, bUpdateAttrib);
// hotkey
UpdateGlobalHotkey();
// time periods
CTimeHelper::SetHoursInOneDay(userPrefs.GetHoursInOneDay());
CTimeHelper::SetDaysInOneWeek(userPrefs.GetDaysInOneWeek());
CDateHelper::SetWeekendDays(userPrefs.GetWeekendDays());
RefreshTabOrder();
// time tracking
if (curPrefs.GetTrackNonActiveTasklists() != userPrefs.GetTrackNonActiveTasklists())
RefreshPauseTimeTracking();
UpdateCaption();
UpdateTabSwitchTooltip();
// colours
if (m_findDlg.GetSafeHwnd())
m_findDlg.RefreshUserPreferences();
// active tasklist userPrefs
UpdateToDoCtrlPreferences();
// then refresh filter bar for any new default cats, statuses, etc
m_filterBar.RefreshFilterControls(GetToDoCtrl());
if (bResizeDlg)
Resize();
// Stickies Support
CString sStickiesPath;
if (userPrefs.GetUseStickies(sStickiesPath))
VERIFY(m_reminders.UseStickies(TRUE, sStickiesPath));
else
m_reminders.UseStickies(FALSE);
// Recently modified period
CFilteredToDoCtrl::SetRecentlyModifiedPeriod(userPrefs.GetRecentlyModifiedPeriod());
// don't ask me for the full details on this but it seems as
// though the CSysImageList class is waiting to process a
// message before we can switch image sizes so we put it
// right at the end after everything is done.
Misc::ProcessMsgLoop();
AppendTools2Toolbar(TRUE);
}
// finally set or terminate the various status check timers
SetTimer(TIMER_READONLYSTATUS, (userPrefs.GetReadonlyReloadOption() != RO_NO));
SetTimer(TIMER_TIMESTAMPCHANGE, (userPrefs.GetTimestampReloadOption() != RO_NO));
SetTimer(TIMER_AUTOSAVE, userPrefs.GetAutoSaveFrequency());
SetTimer(TIMER_CHECKOUTSTATUS, (userPrefs.GetCheckoutOnCheckin() || userPrefs.GetAutoCheckinFrequency()));
SetTimer(TIMER_TIMETRACKING, TRUE);
SetTimer(TIMER_AUTOMINIMIZE, userPrefs.GetAutoMinimizeFrequency());
// re-disable dynamic menu translation
EnableDynamicMenuTranslation(FALSE);
}
BOOL CToDoListWnd::UpdateLanguageTranslationAndRestart(const CString& sLangFile, BOOL bAdd2Dict,
const CPreferencesDlg& curPrefs)
{
BOOL bDefLang = (CTDLLanguageComboBox::GetDefaultLanguage() == sLangFile);
if (curPrefs.GetLanguageFile() != sLangFile)
{
if (bDefLang || FileMisc::FileExists(sLangFile))
{
// if the language file exists and has changed then inform the user that they to restart
// Note: restarting will also handle bAdd2Dict
if (MessageBox(IDS_RESTARTTOCHANGELANGUAGE, 0, MB_YESNO) == IDYES)
{
return TRUE;
}
}
}
// else if the the user wants to enable/disable 'Add2Dictionary'
else if (bAdd2Dict != curPrefs.GetEnableAdd2Dictionary())
{
if (bAdd2Dict && !bDefLang)
{
CLocalizer::SetTranslationOption(ITTTO_ADD2DICTIONARY);
TranslateUIElements();
}
else // disable 'Add2Dictionary'
CLocalizer::SetTranslationOption(ITTTO_TRANSLATEONLY);
}
// no need to restart
return FALSE;
}
void CToDoListWnd::UpdateDefaultTaskAttributes(const CPreferencesDlg& prefs)
{
m_tdiDefault.sTitle = CEnString(IDS_TASK);
m_tdiDefault.color = prefs.GetDefaultColor();
m_tdiDefault.dateStart.m_dt = prefs.GetAutoDefaultStartDate() ? -1 : 0;
m_tdiDefault.sAllocBy = prefs.GetDefaultAllocBy();
m_tdiDefault.sStatus = prefs.GetDefaultStatus();
m_tdiDefault.dTimeEstimate = prefs.GetDefaultTimeEst(m_tdiDefault.nTimeEstUnits);
m_tdiDefault.dTimeSpent = prefs.GetDefaultTimeSpent(m_tdiDefault.nTimeSpentUnits);
m_tdiDefault.nTimeSpentUnits = m_tdiDefault.nTimeEstUnits; // to match
m_tdiDefault.sCreatedBy = prefs.GetDefaultCreatedBy();
m_tdiDefault.dCost = prefs.GetDefaultCost();
m_tdiDefault.sCommentsTypeID = prefs.GetDefaultCommentsFormat();
m_tdiDefault.nPriority = prefs.GetDefaultPriority();
m_tdiDefault.nRisk = prefs.GetDefaultRisk();
prefs.GetDefaultCategories(m_tdiDefault.aCategories);
prefs.GetDefaultAllocTo(m_tdiDefault.aAllocTo);
prefs.GetDefaultTags(m_tdiDefault.aTags);
m_mgrImportExport.SetDefaultTaskAttributes(m_tdiDefault);
}
BOOL CToDoListWnd::LoadMenubar()
{
m_menubar.DestroyMenu();
if (!m_menubar.LoadMenu(IDR_MAINFRAME))
return FALSE;
SetMenu(&m_menubar);
m_hMenuDefault = m_menubar;
#ifdef _DEBUG
// add menu option to simulate WM_QUERYENDSESSION
m_menubar.InsertMenu((UINT)-1, MFT_STRING, ID_DEBUGENDSESSION, _T("EndSession"));
CLocalizer::EnableTranslation(ID_DEBUGENDSESSION, FALSE);
// and option to display setup dialog on demand
m_menubar.InsertMenu((UINT)-1, MFT_STRING, ID_DEBUGSHOWSETUPDLG, _T("Show Setup"));
CLocalizer::EnableTranslation(ID_DEBUGSHOWSETUPDLG, FALSE);
#endif
if (Prefs().GetShowTasklistCloseButton())
m_menubar.AddMDIButton(MEB_CLOSE, ID_CLOSE);
if (CThemed::IsThemeActive())
m_menubar.SetBackgroundColor(m_theme.crMenuBack);
DrawMenuBar();
// disable translation of dynamic menus
EnableDynamicMenuTranslation(FALSE);
return TRUE;
}
void CToDoListWnd::EnableDynamicMenuTranslation(BOOL bEnable)
{
CLocalizer::EnableTranslation(ID_FILE_MRU_FIRST, ID_FILE_MRU_LAST, bEnable);
CLocalizer::EnableTranslation(ID_WINDOW1, ID_WINDOW16, bEnable);
CLocalizer::EnableTranslation(ID_TOOLS_USERTOOL1, ID_TOOLS_USERTOOL16, bEnable);
CLocalizer::EnableTranslation(ID_FILE_OPEN_USERSTORAGE1, ID_FILE_OPEN_USERSTORAGE16, bEnable);
CLocalizer::EnableTranslation(ID_FILE_SAVE_USERSTORAGE1, ID_FILE_SAVE_USERSTORAGE16, bEnable);
CLocalizer::EnableTranslation(ID_TRAYICON_SHOWDUETASKS1, ID_TRAYICON_SHOWDUETASKS20, bEnable);
}
void CToDoListWnd::UpdateGlobalHotkey()
{
static DWORD dwPrevHotkey = 0;
DWORD dwHotkey = Prefs().GetGlobalHotkey();
if (dwPrevHotkey == dwHotkey)
return;
if (dwHotkey == 0) // disabled
::UnregisterHotKey(*this, 1);
else
{
// map modifiers to those wanted by RegisterHotKey
DWORD dwPrefMods = HIWORD(dwHotkey);
DWORD dwVKey = LOWORD(dwHotkey);
DWORD dwMods = (dwPrefMods & HOTKEYF_ALT) ? MOD_ALT : 0;
dwMods |= (dwPrefMods & HOTKEYF_CONTROL) ? MOD_CONTROL : 0;
dwMods |= (dwPrefMods & HOTKEYF_SHIFT) ? MOD_SHIFT : 0;
RegisterHotKey(*this, 1, dwMods, dwVKey);
}
dwPrevHotkey = dwHotkey;
}
void CToDoListWnd::RefreshPauseTimeTracking()
{
// time tracking
int nCtrl = GetTDCCount();
int nSel = GetSelToDoCtrl();
BOOL bTrackActiveOnly = !Prefs().GetTrackNonActiveTasklists();
while (nCtrl--)
{
BOOL bSel = (nCtrl == nSel);
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
tdc.PauseTimeTracking(bSel ? FALSE : bTrackActiveOnly);
}
}
BOOL CToDoListWnd::ProcessStartupOptions(const TDCSTARTUP& startup)
{
// 1. check if we can handle a task link
if (startup.HasFlag(TLD_TASKLINK))
{
CStringArray aFiles;
if (startup.GetFilePaths(aFiles))
{
CString sFile;
DWORD dwTaskID = 0;
CFilteredToDoCtrl::ParseTaskLink(aFiles[0], dwTaskID, sFile);
return DoTaskLink(sFile, dwTaskID);
}
// else
return FALSE;
}
// 2. execute a command
if (startup.HasCommandID())
{
SendMessage(WM_COMMAND, MAKEWPARAM(startup.GetCommandID(), 0), 0);
return TRUE;
}
// 3. try open/import file
if (startup.HasFilePath())
{
int nFirstSel = -1;
BOOL bSuccess = FALSE;
CStringArray aFilePaths;
int nNumFiles = startup.GetFilePaths(aFilePaths);
for (int nFile = 0; nFile < nNumFiles; nFile++)
{
const CString& sFilePath = aFilePaths[nFile];
if (startup.HasFlag(TLD_IMPORTFILE))
{
if (ImportFile(sFilePath, TRUE))
{
bSuccess = TRUE;
}
}
else
{
BOOL bCanDelayLoad = Prefs().GetEnableDelayedLoading();
// open the first tasklist fully and the rest delayed
if (!bSuccess || !Prefs().GetEnableDelayedLoading())
{
if (OpenTaskList(sFilePath, FALSE) == TDCO_SUCCESS)
{
bSuccess = TRUE;
}
}
else if (DelayOpenTaskList(sFilePath) == TDCO_SUCCESS)
{
bSuccess = TRUE;
}
}
// snapshot the first success for subsequent selection
if (bSuccess && (nFirstSel == -1))
nFirstSel = GetSelToDoCtrl();
}
// exit on failure
if (!bSuccess)
return FALSE;
// set selection to first tasklist loaded
ASSERT((nFirstSel != -1) && (nFirstSel < GetTDCCount()));
SelectToDoCtrl(nFirstSel, FALSE);
}
CFilteredToDoCtrl& tdc = GetToDoCtrl();
BOOL bRes = FALSE;
if (startup.HasFlag(TLD_NEWTASK))
{
CEnString sNewTask;
BOOL bEditTask = FALSE;
// we edit the task name if no name was supplied
if (!startup.GetNewTask(sNewTask))
{
sNewTask.LoadString(IDS_TASK);
bEditTask = TRUE;
}
// do we have a parent task ?
if (tdc.SelectTask(startup.GetParentTaskID()))
{
bRes = CreateNewTask(sNewTask, TDC_INSERTATTOPOFSELTASK, FALSE);
}
// or a sibling task ?
else if (tdc.SelectTask(startup.GetSiblingTaskID()))
{
bRes = CreateNewTask(sNewTask, TDC_INSERTAFTERSELTASK, FALSE);
}
else
{
bRes = CreateNewTask(sNewTask, TDC_INSERTATTOP, FALSE);
}
// creation date
double dDate;
if (startup.GetCreationDate(dDate))
tdc.SetSelectedTaskDate(TDCD_CREATE, dDate);
// edit task title?
if (bRes && bEditTask)
PostMessage(WM_COMMAND, ID_EDIT_TASKTEXT);
}
else if (startup.GetTaskID())
{
bRes = tdc.SelectTask(startup.GetTaskID());
}
else // works on the currently selected item(s)
{
bRes = (tdc.GetSelectedCount() > 0);
}
// rest of task attributes
if (bRes)
{
CStringArray aItems;
CString sItem;
int nItem;
double dItem;
if (startup.GetComments(sItem))
tdc.SetSelectedTaskComments(sItem, sItem);
if (startup.GetExternalID(sItem))
tdc.SetSelectedTaskExtID(sItem);
if (startup.GetVersion(sItem))
tdc.SetSelectedTaskVersion(sItem);
if (startup.GetAllocTo(aItems))
tdc.SetSelectedTaskAllocTo(aItems);
if (startup.GetAllocBy(sItem))
tdc.SetSelectedTaskAllocBy(sItem);
if (startup.GetCategories(aItems))
tdc.SetSelectedTaskCategories(aItems);
if (startup.GetTags(aItems))
tdc.SetSelectedTaskTags(aItems);
if (startup.GetStatus(sItem))
tdc.SetSelectedTaskStatus(sItem);
if (startup.GetFileRef(sItem))
tdc.SetSelectedTaskFileRef(sItem);
if (startup.GetPriority(nItem))
tdc.SetSelectedTaskPriority(nItem);
if (startup.GetRisk(nItem))
tdc.SetSelectedTaskRisk(nItem);
if (startup.GetPercentDone(nItem))
tdc.SetSelectedTaskPercentDone(nItem);
if (startup.GetCost(dItem))
tdc.SetSelectedTaskCost(dItem);
if (startup.GetTimeEst(dItem))
tdc.SetSelectedTaskTimeEstimate(dItem); // in hours
if (startup.GetTimeSpent(dItem))
tdc.SetSelectedTaskTimeSpent(dItem); // in hours
if (startup.GetStartDate(dItem))
tdc.SetSelectedTaskDate(TDCD_START, dItem);
if (startup.GetDueDate(dItem))
tdc.SetSelectedTaskDate(TDCD_DUE, dItem);
if (startup.GetDoneDate(dItem))
tdc.SetSelectedTaskDate(TDCD_DONE, dItem);
}
return bRes;
}
BOOL CToDoListWnd::OnCopyData(CWnd* /*pWnd*/, COPYDATASTRUCT* pCopyDataStruct)
{
BOOL bRes = FALSE;
switch (pCopyDataStruct->dwData)
{
case TDL_STARTUP:
{
ASSERT(pCopyDataStruct->cbData == sizeof(TDCSTARTUP));
const TDCSTARTUP* pStartup = (TDCSTARTUP*)(pCopyDataStruct->lpData);
if (pStartup)
bRes = ProcessStartupOptions(*pStartup);
}
break;
}
return bRes;
}
BOOL CToDoListWnd::ImportFile(LPCTSTR szFilePath, BOOL bSilent)
{
int nImporter = m_mgrImportExport.FindImporter(szFilePath);
if (nImporter == -1)
return FALSE;
CTaskFile tasks;
m_mgrImportExport.ImportTaskList(szFilePath, &tasks, nImporter, bSilent);
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (tdc.InsertTasks(tasks, TDC_INSERTATTOP))
UpdateCaption();
return TRUE;
}
void CToDoListWnd::OnEditCopy()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.Flush();
tdc.CopySelectedTask();
}
void CToDoListWnd::OnEditCut()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.Flush();
tdc.CutSelectedTask();
}
void CToDoListWnd::OnUpdateEditCut(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(!tdc.IsReadOnly() && nSelCount);
}
void CToDoListWnd::OnEditPasteSub()
{
CWaitCursor wait;
GetToDoCtrl().PasteTasks(TDCP_ONSELTASK);
}
void CToDoListWnd::OnUpdateEditPasteSub(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.CanPaste() && nSelCount == 1);
}
void CToDoListWnd::OnEditPasteAfter()
{
CWaitCursor wait;
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
if (nSelCount == 0)
tdc.PasteTasks(TDCP_ATBOTTOM);
else
tdc.PasteTasks(TDCP_BELOWSELTASK);
}
void CToDoListWnd::OnUpdateEditPasteAfter(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
// modify the text appropriately if the tasklist is empty
if (nSelCount == 0)
pCmdUI->SetText(CEnString(IDS_PASTETOPLEVELTASK));
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.CanPaste());
}
void CToDoListWnd::OnEditPasteAsRef()
{
CWaitCursor wait;
GetToDoCtrl().PasteTasks(TDCP_ONSELTASK, TRUE);
}
void CToDoListWnd::OnUpdateEditPasteAsRef(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
//int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.CanPaste()/* && nSelCount == 1*/);
}
void CToDoListWnd::OnEditCopyastext()
{
CopySelectedTasksToClipboard(TDCTC_ASTEXT);
}
void CToDoListWnd::OnEditCopyashtml()
{
CopySelectedTasksToClipboard(TDCTC_ASHTML);
}
void CToDoListWnd::CopySelectedTasksToClipboard(TDC_TASKS2CLIPBOARD nAsFormat)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.Flush();
BOOL bParentTitleCommentsOnly = Prefs().GetExportParentTitleCommentsOnly();
DWORD dwFlags = (bParentTitleCommentsOnly ? TDCGTF_PARENTTITLECOMMENTSONLY : 0);
CTaskFile tasks;
CString sTasks;
tdc.GetSelectedTasks(tasks, TDCGETTASKS(TDCGT_ALL, dwFlags));
switch (nAsFormat)
{
case TDCTC_ASHTML:
sTasks = m_mgrImportExport.ExportTaskListToHtml(&tasks);
break;
case TDCTC_ASTEXT:
sTasks = m_mgrImportExport.ExportTaskListToText(&tasks);
break;
case TDCTC_ASLINK:
sTasks.Format(_T("tdl://%ld"), tdc.GetSelectedTaskID());
break;
case TDCTC_ASDEPENDS:
sTasks.Format(_T("%ld"), tdc.GetSelectedTaskID());
break;
case TDCTC_ASLINKFULL:
sTasks.Format(_T("tdl://%s?%ld"),
tdc.GetFilePath(),
tdc.GetSelectedTaskID());
sTasks.Replace(_T(" "), _T("%20"));
break;
case TDCTC_ASDEPENDSFULL:
sTasks.Format(_T("%s?%ld"),
tdc.GetFilePath(),
tdc.GetSelectedTaskID());
break;
case TDCTC_ASPATH:
sTasks = tdc.GetSelectedTaskPath(TRUE);
break;
}
Misc::CopyTexttoClipboard(sTasks, *this);
}
void CToDoListWnd::OnUpdateEditCopy(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() > 0);
}
BOOL CToDoListWnd::CanCreateNewTask(TDC_INSERTWHERE nInsertWhere) const
{
return GetToDoCtrl().CanCreateNewTask(nInsertWhere);
}
void CToDoListWnd::OnUpdateNewtaskAttopSelected(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATTOPOFSELTASKPARENT));
}
void CToDoListWnd::OnUpdateNewtaskAtbottomSelected(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATBOTTOMOFSELTASKPARENT));
}
void CToDoListWnd::OnUpdateNewtaskAfterselectedtask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTAFTERSELTASK));
}
void CToDoListWnd::OnUpdateNewtaskBeforeselectedtask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTBEFORESELTASK));
}
void CToDoListWnd::OnUpdateNewsubtaskAttop(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATTOPOFSELTASK));
}
void CToDoListWnd::OnUpdateNewsubtaskAtBottom(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATBOTTOMOFSELTASK));
}
void CToDoListWnd::OnUpdateNewtaskAttop(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATTOP));
}
void CToDoListWnd::OnUpdateNewtaskAtbottom(CCmdUI* pCmdUI)
{
pCmdUI->Enable(CanCreateNewTask(TDC_INSERTATBOTTOM));
}
void CToDoListWnd::OnMaximizeTasklist()
{
// toggle max state on or off
switch (m_nMaxState)
{
case TDCMS_MAXTASKLIST:
// turn off maximize tasklist by restoring previous max state
m_nMaxState = m_nPrevMaxState;
m_nPrevMaxState = TDCMS_NORMAL; // reset
break;
case TDCMS_MAXCOMMENTS:
// turn on maximize tasklist and save previous max state
m_nMaxState = TDCMS_MAXTASKLIST;
m_nPrevMaxState = TDCMS_MAXCOMMENTS;
break;
case TDCMS_NORMAL:
// turn on maximize tasklist
m_nMaxState = TDCMS_MAXTASKLIST;
m_nPrevMaxState = TDCMS_NORMAL; // reset
break;
}
// update active tasklist
GetToDoCtrl().SetMaximizeState(m_nMaxState);
Invalidate();
// and caption
UpdateCaption();
}
void CToDoListWnd::OnUpdateMaximizeTasklist(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_nMaxState == TDCMS_MAXTASKLIST ? 1 : 0);
}
void CToDoListWnd::OnMaximizeComments()
{
// toggle max state on or off
switch (m_nMaxState)
{
case TDCMS_MAXCOMMENTS:
// toggle off maximize comments by restoring previous max state
m_nMaxState = m_nPrevMaxState;
m_nPrevMaxState = TDCMS_NORMAL; // reset
break;
case TDCMS_MAXTASKLIST:
// turn on maximize comments and save previous max state
m_nMaxState = TDCMS_MAXCOMMENTS;
m_nPrevMaxState = TDCMS_MAXTASKLIST;
break;
case TDCMS_NORMAL:
// turn on maximize comments
m_nMaxState = TDCMS_MAXCOMMENTS;
m_nPrevMaxState = TDCMS_NORMAL; // reset
break;
}
// update active tasklist
GetToDoCtrl().SetMaximizeState(m_nMaxState);
Invalidate();
// and caption
UpdateCaption();
}
void CToDoListWnd::OnUpdateMaximizeComments(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_nMaxState == TDCMS_MAXCOMMENTS ? 1 : 0);
}
void CToDoListWnd::OnReload()
{
int nSel = GetSelToDoCtrl();
if (m_mgrToDoCtrls.GetModifiedStatus(nSel))
{
if (IDYES != MessageBox(IDS_CONFIRMRELOAD, IDS_CONFIRMRELOAD_TITLE, MB_YESNOCANCEL | MB_DEFBUTTON2))
{
return;
}
}
// else reload
ReloadTaskList(nSel);
RefreshTabOrder();
}
BOOL CToDoListWnd::ReloadTaskList(int nIndex, BOOL bNotifyDueTasks, BOOL bNotifyError)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
TDC_FILE nRes = OpenTaskList(&tdc);
if (nRes == TDCO_SUCCESS)
{
const CPreferencesDlg& userPrefs = Prefs();
// update file status
if (userPrefs.GetAutoCheckOut())
m_mgrToDoCtrls.SetLastCheckoutStatus(nIndex, tdc.IsCheckedOut());
m_mgrToDoCtrls.RefreshLastModified(nIndex);
m_mgrToDoCtrls.SetModifiedStatus(nIndex, FALSE);
m_mgrToDoCtrls.UpdateTabItemText(nIndex);
// notify user of due tasks if req
if (bNotifyDueTasks)
DoDueTaskNotification(nIndex, userPrefs.GetNotifyDueByOnLoad());
UpdateCaption();
UpdateStatusbar();
}
else if (bNotifyError)
{
HandleLoadTasklistError(nRes, tdc.GetFilePath());
}
return (nRes == TDCO_SUCCESS);
}
void CToDoListWnd::OnUpdateReload(CCmdUI* pCmdUI)
{
int nSel = GetSelToDoCtrl();
pCmdUI->Enable(!m_mgrToDoCtrls.GetFilePath(nSel).IsEmpty());
}
void CToDoListWnd::OnSize(UINT nType, int cx, int cy)
{
CFrameWnd::OnSize(nType, cx, cy);
// ensure m_cbQuickFind is positioned correctly
int nPos = m_toolbar.CommandToIndex(ID_EDIT_FINDTASKS) + 2;
CRect rNewPos, rOrgPos;
m_toolbar.GetItemRect(nPos, rNewPos);
m_toolbar.ClientToScreen(rNewPos);
m_cbQuickFind.CWnd::GetWindowRect(rOrgPos);
// check if it needs to be moved
if (rNewPos.TopLeft() != rOrgPos.TopLeft())
{
m_toolbar.ScreenToClient(rNewPos);
rNewPos.bottom = rNewPos.top + 200;
m_cbQuickFind.MoveWindow(rNewPos);
}
// topmost?
BOOL bMaximized = (nType == SIZE_MAXIMIZED);
if (nType != SIZE_MINIMIZED)
Resize(cx, cy, bMaximized);
// if not maximized then set topmost if that's the preference
BOOL bTopMost = (Prefs().GetAlwaysOnTop() && !bMaximized) ? 1 : 0;
// do nothing if no change
BOOL bIsTopMost = (GetExStyle() & WS_EX_TOPMOST) ? 1 : 0;
if (bTopMost != bIsTopMost)
SetWindowPos(bTopMost ? &wndTopMost : &wndNoTopMost, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
}
BOOL CToDoListWnd::CalcToDoCtrlRect(CRect& rect, int cx, int cy, BOOL bMaximized)
{
if (!cx && !cy)
{
CRect rClient;
GetClientRect(rClient);
cx = rClient.right;
cy = rClient.bottom;
bMaximized = IsZoomed();
// check again
if (!cx && !cy)
return FALSE;
}
CRect rTaskList(0, BEVEL, cx - BEVEL, cy);
// toolbar
if (m_bShowToolbar)
rTaskList.top += m_toolbar.GetHeight() + TB_VOFFSET;
// resize tabctrl
CDeferWndMove dwm(0); // dummy
CPoint ptOrg(0, rTaskList.top);
int nTabHeight = ReposTabBar(dwm, ptOrg, cx, TRUE);
if (nTabHeight)
rTaskList.top += nTabHeight + 1; // hide the bottom of the tab ctrl
// filter controls
int nInset = (CThemed().IsNonClientThemed() ? BORDER : BEVEL);
int nFilterWidth = cx - 2 * nInset;
int nFilterHeight = m_bShowFilterBar ? m_filterBar.CalcHeight(nFilterWidth) : 0;
if (nFilterHeight)
rTaskList.top += nFilterHeight;// + 4;
// statusbar
if (m_bShowStatusBar)
{
CRect rStatus;
m_statusBar.GetWindowRect(rStatus);
ScreenToClient(rStatus);
rTaskList.bottom = rStatus.top - BORDER;
}
else
rTaskList.bottom = cy - BORDER;
// shrink slightly so that edit controls do not merge with window border
rTaskList.DeflateRect(nInset, nInset, nInset, nInset);
rect = rTaskList;
return TRUE;
}
void CToDoListWnd::Resize(int cx, int cy, BOOL bMaximized)
{
static int nLastCx = 0, nLastCy = 0;
if (!cx && !cy)
{
CRect rClient;
GetClientRect(rClient);
cx = rClient.right;
cy = rClient.bottom;
bMaximized = IsZoomed();
// check again
if (!cx && !cy)
return;
}
if (cx == nLastCx && cy == nLastCy && !GetTDCCount())
return;
nLastCx = cx;
nLastCy = cy;
// resize in one go
CDlgUnits dlu(*this);
CDeferWndMove dwm(6);
CRect rTaskList(0, BEVEL, cx - BEVEL, cy);
// toolbar
if (m_bShowToolbar) // showing toolbar
rTaskList.top += m_toolbar.Resize(cx, CPoint(0, TB_VOFFSET)) + TB_VOFFSET;
// resize tabctrl
CPoint ptOrg(0, rTaskList.top);
int nTabHeight = ReposTabBar(dwm, ptOrg, cx);
if (nTabHeight)
rTaskList.top += nTabHeight + 1; // hide the bottom of the tab ctrl
// filter controls
int nInset = (CThemed().IsNonClientThemed() ? BORDER : BEVEL);
int nFilterWidth = cx - 2 * nInset;
int nFilterHeight = m_bShowFilterBar ? m_filterBar.CalcHeight(nFilterWidth) : 0;
dwm.MoveWindow(&m_filterBar, nInset, rTaskList.top, nFilterWidth, nFilterHeight);
if (nFilterHeight)
rTaskList.top += nFilterHeight;// + 4;
// statusbar has already been automatically resized unless it's invisible
CRect rStatus(0, cy, cx, cy);
if (m_bShowStatusBar)
{
m_statusBar.GetWindowRect(rStatus);
ScreenToClient(rStatus);
}
else
dwm.MoveWindow(&m_statusBar, rStatus, FALSE);
// finally the active todoctrl
if (GetTDCCount())
{
if (m_bShowStatusBar)
rTaskList.bottom = rStatus.top - BORDER;
else
rTaskList.bottom = rStatus.bottom - BORDER;
// shrink slightly so that edit controls do not merge with window border
rTaskList.DeflateRect(nInset, nInset, nInset, nInset);
dwm.MoveWindow(&GetToDoCtrl(), rTaskList);
#ifdef _DEBUG
CRect rect;
CalcToDoCtrlRect(rect, cx, cy, IsZoomed());
ASSERT(rect == rTaskList);
#endif
}
}
BOOL CToDoListWnd::WantTasklistTabbarVisible() const
{
BOOL bWantTabbar = (GetTDCCount() > 1 || !Prefs().GetAutoHideTabbar());
bWantTabbar &= m_bShowTasklistBar;
return bWantTabbar;
}
int CToDoListWnd::ReposTabBar(CDeferWndMove& dwm, const CPoint& ptOrg, int nWidth, BOOL bCalcOnly)
{
CRect rTabs(0, 0, nWidth, 0);
m_tabCtrl.AdjustRect(TRUE, rTabs);
int nTabHeight = rTabs.Height() - 4;
rTabs = dwm.OffsetCtrl(this, IDC_TABCONTROL); // not actually a move
rTabs.right = nWidth + 1;
rTabs.bottom = rTabs.top + nTabHeight;
rTabs.OffsetRect(0, ptOrg.y - rTabs.top + 1); // add a pixel between tabbar and toolbar
BOOL bNeedTabCtrl = WantTasklistTabbarVisible();
if (!bCalcOnly)
{
dwm.MoveWindow(&m_tabCtrl, rTabs);
// hide and disable tabctrl if not needed
m_tabCtrl.ShowWindow(bNeedTabCtrl ? SW_SHOW : SW_HIDE);
m_tabCtrl.EnableWindow(bNeedTabCtrl);
if (bNeedTabCtrl)
UpdateTabSwitchTooltip();
}
return bNeedTabCtrl ? rTabs.Height() : 0;
}
void CToDoListWnd::OnPrint()
{
DoPrint();
}
void CToDoListWnd::DoPrint(BOOL bPreview)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelTDC = GetSelToDoCtrl();
// pass the project name as the title field
CString sTitle = m_mgrToDoCtrls.GetFriendlyProjectName(nSelTDC);
// export to html and then print in IE
CTDLPrintDialog dialog(sTitle, bPreview, tdc.GetView());
if (dialog.DoModal() != IDOK)
return;
RedrawWindow();
// always use the same file
CString sTempFile = FileMisc::GetTempFileName(_T("ToDoList.print"), _T("html"));
// stylesheets don't seem to like the way we do html comments
CString sStylesheet = dialog.GetStylesheet();
BOOL bTransform = FileMisc::FileExists(sStylesheet);
sTitle = dialog.GetTitle();
// export
DOPROGRESS(bPreview ? IDS_PPREVIEWPROGRESS : IDS_PRINTPROGRESS)
CTaskFile tasks;
GetTasks(tdc, TRUE, bTransform, dialog.GetTaskSelection(), tasks, NULL);
// add title and date, and style
COleDateTime date;
if (dialog.GetWantDate())
date = COleDateTime::GetCurrentTime();
tasks.SetReportAttributes(sTitle, date);
// add export style
if (!bTransform)
{
TDLPD_STYLE nStyle = dialog.GetExportStyle();
tasks.SetMetaData(TDL_EXPORTSTYLE, Misc::Format(nStyle));
}
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
if (!Export2Html(tasks, sTempFile, sStylesheet))
return;
// print from browser
CRect rHidden(-20, -20, -10, -10); // create IE off screen
if (m_IE.GetSafeHwnd() || m_IE.Create(NULL, WS_CHILD | WS_VISIBLE, rHidden, this, (UINT)IDC_STATIC))
{
double dFileSize = FileMisc::GetFileSize(sTempFile);
BOOL bPrintBkgnd = Prefs().GetColorTaskBackground();
if (bPreview)
m_IE.PrintPreview(sTempFile, bPrintBkgnd);
else
m_IE.Print(sTempFile, bPrintBkgnd);
}
else // try sending to browser
{
int nRes = (int)::ShellExecute(*this, bPreview ? _T("print") : NULL, sTempFile, NULL, NULL, SW_HIDE);
if (nRes < 32)
MessageBox(IDS_PRINTFAILED, IDS_PRINTFAILED_TITLE, MB_OK);
}
}
void CToDoListWnd::OnUpdatePrint(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetTaskCount());
}
int CToDoListWnd::AddToDoCtrl(CFilteredToDoCtrl* pTDC, TSM_TASKLISTINFO* pInfo, BOOL bResizeDlg)
{
// add tdc first to ensure tab controls has some
// items before we query it for its size
int nSel = m_mgrToDoCtrls.AddToDoCtrl(pTDC, pInfo);
// make sure size is right
CRect rTDC;
if (CalcToDoCtrlRect(rTDC))
pTDC->MoveWindow(rTDC);
SelectToDoCtrl(nSel, FALSE);
pTDC->SetFocusToTasks();
// make sure the tab control is correctly sized
if (bResizeDlg)
Resize();
// if this is the only control then set or terminate the various status
// check timers
if (GetTDCCount() == 1)
{
const CPreferencesDlg& userPrefs = Prefs();
SetTimer(TIMER_READONLYSTATUS, userPrefs.GetReadonlyReloadOption() != RO_NO);
SetTimer(TIMER_TIMESTAMPCHANGE, userPrefs.GetTimestampReloadOption() != RO_NO);
SetTimer(TIMER_AUTOSAVE, userPrefs.GetAutoSaveFrequency());
SetTimer(TIMER_CHECKOUTSTATUS,
userPrefs.GetCheckoutOnCheckin() || userPrefs.GetAutoCheckinFrequency());
}
// make sure everything looks okay
Invalidate();
UpdateWindow();
return nSel;
}
void CToDoListWnd::SetTimer(UINT nTimerID, BOOL bOn)
{
if (bOn)
{
UINT nPeriod = 0;
switch (nTimerID)
{
case TIMER_READONLYSTATUS:
nPeriod = INTERVAL_READONLYSTATUS;
break;
case TIMER_TIMESTAMPCHANGE:
nPeriod = INTERVAL_TIMESTAMPCHANGE;
break;
case TIMER_AUTOSAVE:
nPeriod = (Prefs().GetAutoSaveFrequency() * ONE_MINUTE);
break;
case TIMER_CHECKOUTSTATUS:
nPeriod = INTERVAL_CHECKOUTSTATUS;
break;
case TIMER_DUEITEMS:
nPeriod = INTERVAL_DUEITEMS;
break;
case TIMER_TIMETRACKING:
nPeriod = INTERVAL_TIMETRACKING;
break;
case TIMER_AUTOMINIMIZE:
nPeriod = (Prefs().GetAutoMinimizeFrequency() * ONE_MINUTE);
break;
}
if (nPeriod)
{
UINT nID = CFrameWnd::SetTimer(nTimerID, nPeriod, NULL);
ASSERT (nID);
}
}
else
KillTimer(nTimerID);
}
void CToDoListWnd::OnTimer(UINT nIDEvent)
{
CFrameWnd::OnTimer(nIDEvent);
// if we are disabled (== modal dialog visible) then do not respond
if (!IsWindowEnabled())
return;
// don't check whilst in the middle of saving or closing
if (m_bSaving || m_bClosing)
return;
// if no controls are active kill the timers
if (!GetTDCCount())
{
SetTimer(TIMER_READONLYSTATUS, FALSE);
SetTimer(TIMER_TIMESTAMPCHANGE, FALSE);
SetTimer(TIMER_AUTOSAVE, FALSE);
SetTimer(TIMER_CHECKOUTSTATUS, FALSE);
SetTimer(TIMER_DUEITEMS, FALSE);
SetTimer(TIMER_TIMETRACKING, FALSE);
SetTimer(TIMER_AUTOMINIMIZE, FALSE);
return;
}
switch (nIDEvent)
{
case TIMER_READONLYSTATUS:
OnTimerReadOnlyStatus();
break;
case TIMER_TIMESTAMPCHANGE:
OnTimerTimestampChange();
break;
case TIMER_AUTOSAVE:
OnTimerAutoSave();
break;
case TIMER_CHECKOUTSTATUS:
OnTimerCheckoutStatus();
break;
case TIMER_DUEITEMS:
OnTimerDueItems();
break;
case TIMER_TIMETRACKING:
OnTimerTimeTracking();
break;
case TIMER_AUTOMINIMIZE:
OnTimerAutoMinimize();
break;
}
}
BOOL CToDoListWnd::IsActivelyTimeTracking() const
{
// cycle thru tasklists until we find a time tracker
int nCtrl = GetTDCCount();
while (nCtrl--)
{
if (GetToDoCtrl(nCtrl).IsActivelyTimeTracking())
return TRUE;
}
// else
return FALSE;
}
void CToDoListWnd::OnTimerTimeTracking()
{
AF_NOREENTRANT // macro helper
static BOOL bWasTimeTracking = FALSE;
BOOL bNowTimeTracking = IsActivelyTimeTracking();
if (bWasTimeTracking != bNowTimeTracking)
{
UINT nIDTrayIcon = (bNowTimeTracking ? IDI_TRAYTRACK_STD : IDI_TRAY_STD);
m_trayIcon.SetIcon(nIDTrayIcon);
// set the main window icon also as this helps the user know what's going on
HICON hIcon = GraphicsMisc::LoadIcon(nIDTrayIcon);
SetIcon(hIcon, FALSE);
}
bWasTimeTracking = bNowTimeTracking;
}
void CToDoListWnd::OnTimerDueItems(int nCtrl)
{
AF_NOREENTRANT // macro helper
int nFrom = (nCtrl == -1) ? 0 : nCtrl;
int nTo = (nCtrl == -1) ? GetTDCCount() - 1 : nCtrl;
BOOL bRepaint = FALSE;
for (nCtrl = nFrom; nCtrl <= nTo; nCtrl++)
{
// first we search for overdue items on each tasklist and if that
// fails to find anything we then search for items due today
// but only if the tasklist is fully loaded
if (m_mgrToDoCtrls.IsLoaded(nCtrl))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
TDCM_DUESTATUS nStatus = TDCM_NONE;
if (tdc.HasOverdueTasks()) // takes priority
nStatus = TDCM_PAST;
else if (tdc.HasDueTodayTasks())
nStatus = TDCM_TODAY;
if (nStatus != m_mgrToDoCtrls.GetDueItemStatus(nCtrl))
{
m_mgrToDoCtrls.SetDueItemStatus(nCtrl, nStatus);
bRepaint = TRUE;
}
}
}
if (bRepaint)
m_tabCtrl.Invalidate(FALSE);
}
void CToDoListWnd::OnTimerReadOnlyStatus(int nCtrl)
{
AF_NOREENTRANT // macro helper
const CPreferencesDlg& userPrefs = Prefs();
// work out whether we should check remote files or not
BOOL bCheckRemoteFiles = (nCtrl != -1);
if (!bCheckRemoteFiles)
{
static int nElapsed = 0;
UINT nRemoteFileCheckInterval = userPrefs.GetRemoteFileCheckFrequency() * 1000; // in ms
nElapsed %= nRemoteFileCheckInterval;
bCheckRemoteFiles = !nElapsed;
nElapsed += INTERVAL_READONLYSTATUS;
}
int nReloadOption = userPrefs.GetReadonlyReloadOption();
ASSERT (nReloadOption != RO_NO);
// process files
CString sFileList;
int nFrom = (nCtrl == -1) ? 0 : nCtrl;
int nTo = (nCtrl == -1) ? GetTDCCount() - 1 : nCtrl;
for (nCtrl = nFrom; nCtrl <= nTo; nCtrl++)
{
// don't check delay-loaded tasklists
if (!m_mgrToDoCtrls.IsLoaded(nCtrl))
continue;
// don't check removeable drives
int nType = m_mgrToDoCtrls.GetFilePathType(nCtrl);
if (nType == TDCM_UNDEF || nType == TDCM_REMOVABLE)
continue;
// check remote files?
if (!bCheckRemoteFiles && nType == TDCM_REMOTE)
continue;
if (m_mgrToDoCtrls.RefreshReadOnlyStatus(nCtrl))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
BOOL bReadOnly = m_mgrToDoCtrls.GetReadOnlyStatus(nCtrl);
BOOL bReload = FALSE;
if (nReloadOption == RO_ASK)
{
CString sFilePath = tdc.GetFilePath();
CEnString sMessage(bReadOnly ? IDS_WRITABLETOREADONLY : IDS_READONLYTOWRITABLE, sFilePath);
if (!bReadOnly) // might been modified
sMessage += CEnString(IDS_WANTRELOAD);
UINT nRet = MessageBox(sMessage, IDS_STATUSCHANGE_TITLE, !bReadOnly ? MB_YESNOCANCEL : MB_OK);
bReload = (nRet == IDYES || nRet == IDOK);
}
else
bReload = !bReadOnly; // now writable
if (bReload && ReloadTaskList(nCtrl, FALSE, (nReloadOption == RO_ASK)))
{
// notify the user if we haven't already
if (nReloadOption == RO_NOTIFY)
{
sFileList += tdc.GetFriendlyProjectName();
sFileList += "\n";
}
}
else // update the UI
{
if (nCtrl == m_tabCtrl.GetCurSel())
UpdateCaption();
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
}
}
}
// do we need to notify the user?
if (!sFileList.IsEmpty())
{
CEnString sMessage(IDS_TASKLISTSRELOADED, sFileList);
m_trayIcon.ShowBalloon(sMessage, CEnString(IDS_TIMESTAMPCHANGE_BALLOONTITLE), NIIF_INFO);
}
}
void CToDoListWnd::OnTimerTimestampChange(int nCtrl)
{
AF_NOREENTRANT // macro helper
const CPreferencesDlg& userPrefs = Prefs();
int nReloadOption = userPrefs.GetTimestampReloadOption();
ASSERT (nReloadOption != RO_NO);
// work out whether we should check remote files or not
BOOL bCheckRemoteFiles = (nCtrl != -1);
if (!bCheckRemoteFiles)
{
static int nElapsed = 0;
UINT nRemoteFileCheckInterval = userPrefs.GetRemoteFileCheckFrequency() * 1000; // in ms
nElapsed %= nRemoteFileCheckInterval;
bCheckRemoteFiles = !nElapsed;
nElapsed += INTERVAL_TIMESTAMPCHANGE;
}
// process files
CString sFileList;
int nFrom = (nCtrl == -1) ? 0 : nCtrl;
int nTo = (nCtrl == -1) ? GetTDCCount() - 1 : nCtrl;
for (nCtrl = nFrom; nCtrl <= nTo; nCtrl++)
{
// don't check delay-loaded tasklists
if (!m_mgrToDoCtrls.IsLoaded(nCtrl))
continue;
// don't check removeable drives
int nType = m_mgrToDoCtrls.GetFilePathType(nCtrl);
if (nType == TDCM_UNDEF || nType == TDCM_REMOVABLE)
continue;
// check remote files?
if (!bCheckRemoteFiles && nType == TDCM_REMOTE)
continue;
if (m_mgrToDoCtrls.RefreshLastModified(nCtrl))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
BOOL bReload = TRUE; // default
if (nReloadOption == RO_ASK)
{
CString sFilePath = tdc.GetFilePath();
CEnString sMessage(IDS_MODIFIEDELSEWHERE, sFilePath);
sMessage += CEnString(IDS_WANTRELOAD);
bReload = (MessageBox(sMessage, IDS_TIMESTAMPCHANGE_TITLE, MB_YESNOCANCEL) == IDYES);
}
if (bReload && ReloadTaskList(nCtrl, FALSE, (nReloadOption == RO_ASK)))
{
// notify the user if we haven't already
if (nReloadOption == RO_NOTIFY)
{
sFileList += tdc.GetFriendlyProjectName();
sFileList += "\n";
}
}
else
{
// update UI
if (nCtrl == m_tabCtrl.GetCurSel())
UpdateCaption();
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
}
}
}
// do we need to notify the user?
if (!sFileList.IsEmpty())
{
CEnString sMessage(IDS_TASKLISTSRELOADED, sFileList);
m_trayIcon.ShowBalloon(sMessage, CEnString(IDS_TIMESTAMPCHANGE_BALLOONTITLE), NIIF_INFO);
}
}
void CToDoListWnd::OnTimerAutoSave()
{
AF_NOREENTRANT // macro helper
// don't save if the user is editing a task label
if (!GetToDoCtrl().IsTaskLabelEditing())
SaveAll(TDLS_AUTOSAVE);
}
void CToDoListWnd::OnTimerAutoMinimize()
{
AF_NOREENTRANT // macro helper
if (!IsWindowVisible() || IsIconic())
return;
LASTINPUTINFO lii = { sizeof(LASTINPUTINFO), 0 };
if (GetLastInputInfo(&lii))
{
double dElapsed = (GetTickCount() - lii.dwTime);
dElapsed /= ONE_MINUTE; // convert to minutes
// check
if (dElapsed > (double)Prefs().GetAutoMinimizeFrequency())
ShowWindow(SW_MINIMIZE);
}
}
void CToDoListWnd::OnTimerCheckoutStatus(int nCtrl)
{
AF_NOREENTRANT // macro helper
const CPreferencesDlg& userPrefs = Prefs();
// work out whether we should check remote files or not
BOOL bCheckRemoteFiles = (nCtrl != -1);
if (!bCheckRemoteFiles)
{
static int nElapsed = 0;
UINT nRemoteFileCheckInterval = userPrefs.GetRemoteFileCheckFrequency() * 1000; // in ms
nElapsed %= nRemoteFileCheckInterval;
bCheckRemoteFiles = !nElapsed;
nElapsed += INTERVAL_CHECKOUTSTATUS;
}
// process files
CString sFileList;
int nFrom = (nCtrl == -1) ? 0 : nCtrl;
int nTo = (nCtrl == -1) ? GetTDCCount() - 1 : nCtrl;
for (nCtrl = nFrom; nCtrl <= nTo; nCtrl++)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
if (!m_mgrToDoCtrls.PathSupportsSourceControl(nCtrl))
continue;
// try to check out only if the previous attempt failed
if (!tdc.IsCheckedOut() && userPrefs.GetCheckoutOnCheckin())
{
// we only try to check if the previous checkout failed
if (!m_mgrToDoCtrls.GetLastCheckoutStatus(nCtrl))
{
if (tdc.CheckOut())
{
// update timestamp
m_mgrToDoCtrls.RefreshLastModified(nCtrl);
m_mgrToDoCtrls.SetLastCheckoutStatus(nCtrl, TRUE);
if (nCtrl == m_tabCtrl.GetCurSel())
UpdateCaption();
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
// notify the user if the trayicon is visible
sFileList += tdc.GetFriendlyProjectName();
sFileList += "\n";
}
else // make sure we try again later
m_mgrToDoCtrls.SetLastCheckoutStatus(nCtrl, FALSE);
}
}
// only checkin if sufficient time has elapsed since last mod
// and there are no mods outstanding
else if (tdc.IsCheckedOut() && userPrefs.GetAutoCheckinFrequency())
{
if (!tdc.IsModified())
{
double dElapsed = COleDateTime::GetCurrentTime() - tdc.GetLastTaskModified();
dElapsed *= 24 * 60; // convert to minutes
if (dElapsed > (double)userPrefs.GetAutoCheckinFrequency())
{
if (tdc.CheckIn())
{
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.RefreshLastModified(nCtrl);
m_mgrToDoCtrls.SetLastCheckoutStatus(nCtrl, TRUE);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
UpdateCaption();
}
}
}
}
}
// do we need to notify the user?
if (!sFileList.IsEmpty())
{
CEnString sMessage(IDS_TASKLISTSCHECKEDOUT, sFileList);
m_trayIcon.ShowBalloon(sMessage, _T("Source Control Change(s)"), NIIF_INFO);
}
}
void CToDoListWnd::OnNeedTooltipText(NMHDR* pNMHDR, LRESULT* pResult)
{
static CString sTipText;
sTipText.Empty();
switch (pNMHDR->idFrom)
{
case ID_TOOLS_USERTOOL1:
case ID_TOOLS_USERTOOL2:
case ID_TOOLS_USERTOOL3:
case ID_TOOLS_USERTOOL4:
case ID_TOOLS_USERTOOL5:
case ID_TOOLS_USERTOOL6:
case ID_TOOLS_USERTOOL7:
case ID_TOOLS_USERTOOL8:
case ID_TOOLS_USERTOOL9:
case ID_TOOLS_USERTOOL10:
case ID_TOOLS_USERTOOL11:
case ID_TOOLS_USERTOOL12:
case ID_TOOLS_USERTOOL13:
case ID_TOOLS_USERTOOL14:
case ID_TOOLS_USERTOOL15:
case ID_TOOLS_USERTOOL16:
{
USERTOOL ut;
if (Prefs().GetUserTool(pNMHDR->idFrom - ID_TOOLS_USERTOOL1, ut))
sTipText = ut.sToolName;
}
break;
default:
// tab control popups
if (pNMHDR->idFrom >= 0 && pNMHDR->idFrom < (UINT)m_mgrToDoCtrls.GetCount())
{
sTipText = m_mgrToDoCtrls.GetTabItemTooltip(pNMHDR->idFrom);
}
break;
}
if (!sTipText.IsEmpty())
{
TOOLTIPTEXT* pTTT = (TOOLTIPTEXT*)pNMHDR;
pTTT->lpszText = (LPTSTR)(LPCTSTR)sTipText;
}
*pResult = 0;
}
void CToDoListWnd::OnUpdateUserTool(CCmdUI* pCmdUI)
{
if (pCmdUI->m_pMenu && pCmdUI->m_nID == ID_TOOLS_USERTOOL1) // only handle first item
{
CUserToolArray aTools;
Prefs().GetUserTools(aTools);
CToolsHelper th(Prefs().GetEnableTDLExtension(), ID_TOOLS_USERTOOL1);
th.UpdateMenu(pCmdUI, aTools, m_mgrMenuIcons);
}
else if (m_bShowToolbar)
{
int nTool = pCmdUI->m_nID - ID_TOOLS_USERTOOL1;
ASSERT (nTool >= 0 && nTool < 16);
USERTOOL ut;
if (Prefs().GetUserTool(nTool, ut))
pCmdUI->Enable(TRUE);
}
}
void CToDoListWnd::OnUserTool(UINT nCmdID)
{
int nTool = nCmdID - ID_TOOLS_USERTOOL1;
USERTOOL ut;
ASSERT (nTool >= 0 && nTool < 16);
const CPreferencesDlg& prefs = Prefs();
if (prefs.GetUserTool(nTool, ut))
{
// Save all tasklists before executing the user tool
if (prefs.GetAutoSaveOnRunTools())
{
if (SaveAll(TDLS_FLUSH) == TDCO_CANCELLED)
return;
}
USERTOOLARGS args;
PopulateToolArgs(args);
CToolsHelper th(prefs.GetEnableTDLExtension(), ID_TOOLS_USERTOOL1);
th.RunTool(ut, args, this);
}
}
void CToDoListWnd::AddUserStorageToMenu(CMenu* pMenu)
{
if (pMenu)
{
const UINT MENUSTARTID = pMenu->GetMenuItemID(0);
// delete existing entries
int nStore = 16;
while (nStore--)
{
pMenu->DeleteMenu(nStore, MF_BYPOSITION);
}
// if we have any tools to add we do it here
int nNumStorage = min(m_mgrStorage.GetNumStorage(), 16);
if (nNumStorage)
{
UINT nFlags = (MF_BYPOSITION | MF_STRING);
for (int nStore = 0; nStore < nNumStorage; nStore++)
{
CString sMenuItem, sText = m_mgrStorage.GetStorageMenuText(nStore);
if (nStore < 9)
sMenuItem.Format(_T("&%d %s"), nStore + 1, sText);
else
sMenuItem = sText;
// special case: if this is a 'Save' menu item
// then the we disable it if it's the same as
// the current tasklist's storage
UINT nExtraFlags = 0;
/*
if (MENUSTARTID == ID_FILE_SAVE_USERSTORAGE1)
{
int nTDC = GetSelToDoCtrl();
TSM_TASKLISTINFO storageInfo;
if (m_mgrToDoCtrls.GetStorageDetails(nTDC, storageInfo))
{
if (m_mgrStorage.FindStorage(storageInfo.sStorageID) == nStore)
nExtraFlags = (MF_GRAYED | MF_CHECKED);
}
}
*/
pMenu->InsertMenu(nStore, nFlags | nExtraFlags, MENUSTARTID + nStore, sMenuItem);
// add icon if available
HICON hIcon = m_mgrStorage.GetStorageIcon(nStore);
if (hIcon)
m_mgrMenuIcons.AddImage(MENUSTARTID + nStore, hIcon);
}
}
else // if nothing to add just re-add placeholder
{
pMenu->InsertMenu(0, MF_BYPOSITION | MF_STRING | MF_GRAYED, MENUSTARTID, _T("3rd Party Storage"));
}
}
}
void CToDoListWnd::OnFileOpenFromUserStorage(UINT nCmdID)
{
int nStorage = nCmdID - ID_FILE_OPEN_USERSTORAGE1;
ASSERT (nStorage >= 0 && nStorage < 16);
TSM_TASKLISTINFO storageInfo;
CPreferences prefs;
CTaskFile tasks;
if (m_mgrStorage.RetrieveTasklist(&storageInfo, &tasks, nStorage, &prefs))
{
if (tasks.GetTaskCount())
{
VERIFY(CreateNewTaskList(FALSE));
CFilteredToDoCtrl& tdc = GetToDoCtrl();
VERIFY(tdc.InsertTasks(tasks, TDC_INSERTATTOP));
// attach the returned storage info to this tasklist
m_mgrToDoCtrls.SetStorageDetails(GetSelToDoCtrl(), storageInfo);
}
else if (storageInfo.szLocalFileName[0])
{
TDC_FILE nOpen = OpenTaskList(storageInfo.szLocalFileName, TRUE);
if (nOpen == TDCO_SUCCESS)
{
// attach the returned storage info to this tasklist
int nTDC = m_mgrToDoCtrls.FindToDoCtrl(storageInfo.szLocalFileName);
ASSERT(nTDC == GetSelToDoCtrl());
m_mgrToDoCtrls.SetStorageDetails(nTDC, storageInfo);
}
else
HandleLoadTasklistError(nOpen, storageInfo.szDisplayName);
}
else
{
// TODO
}
UpdateCaption();
UpdateStatusbar();
Resize();
UpdateWindow();
}
}
void CToDoListWnd::OnFileSaveToUserStorage(UINT nCmdID)
{
int nStorage = (nCmdID - ID_FILE_SAVE_USERSTORAGE1);
ASSERT (nStorage >= 0 && nStorage < 16);
CFilteredToDoCtrl& tdc = GetToDoCtrl();
CString sLocalPath, sTdcPath = tdc.GetFilePath();
// retrieve any existing storage info for this tasklist
int nTDC = GetSelToDoCtrl();
TSM_TASKLISTINFO storageInfo;
if (m_mgrToDoCtrls.GetStorageDetails(nTDC, storageInfo))
{
sLocalPath = storageInfo.szLocalFileName;
// clear the storage info ID if we are changing
// the destination
if (nStorage != m_mgrStorage.FindStorage(storageInfo.sStorageID))
storageInfo.Reset();
}
if (sLocalPath.IsEmpty())
{
sLocalPath = sTdcPath = tdc.GetFilePath();
if (sLocalPath.IsEmpty())
sLocalPath = FileMisc::GetTempFileName(m_mgrToDoCtrls.GetFriendlyProjectName(nTDC), _T("tdl"));
}
CTaskFile tasks;
VERIFY (tdc.Save(tasks, sLocalPath) == TDCO_SUCCESS);
// restore previous task path
if (sTdcPath != sLocalPath)
{
tdc.SetFilePath(sTdcPath);
}
else // prevent this save triggering a reload
{
m_mgrToDoCtrls.RefreshLastModified(nTDC);
}
_tcsncpy(storageInfo.szLocalFileName, sLocalPath, _MAX_PATH);
CPreferences prefs;
if (m_mgrStorage.StoreTasklist(&storageInfo, &tasks, nStorage, &prefs))
{
m_mgrToDoCtrls.SetStorageDetails(nTDC, storageInfo);
UpdateCaption();
UpdateStatusbar();
Resize();
UpdateWindow();
}
// else assume that the plugin handled any problems
}
void CToDoListWnd::OnInitMenuPopup(CMenu* pPopupMenu, UINT nIndex, BOOL bSysMenu)
{
// test for top-level menus
if (bSysMenu)
return;
if (m_menubar.GetSubMenu(FILEALL) == pPopupMenu)
{
// insert Min to sys tray if appropriate
BOOL bHasMinToTray = (::GetMenuString(*pPopupMenu, ID_MINIMIZETOTRAY, NULL, 0, MF_BYCOMMAND) != 0);
if (Prefs().GetSysTrayOption() == STO_ONCLOSE || Prefs().GetSysTrayOption() == STO_ONMINCLOSE)
{
if (!bHasMinToTray)
pPopupMenu->InsertMenu(ID_EXIT, MF_BYCOMMAND, ID_MINIMIZETOTRAY, CEnString(ID_MINIMIZETOTRAY));
}
else if (bHasMinToTray) // then remove
{
pPopupMenu->DeleteMenu(ID_MINIMIZETOTRAY, MF_BYCOMMAND);
}
}
else if (m_menubar.GetSubMenu(EDITTASK) == pPopupMenu)
{
// remove relevant commands from the edit menu
PrepareEditMenu(pPopupMenu);
}
else if (m_menubar.GetSubMenu(SORTTASK) == pPopupMenu)
{
// remove relevant commands from the sort menu
PrepareSortMenu(pPopupMenu);
}
else if (m_menubar.GetSubMenu(TOOLS) == pPopupMenu)
{
// remove relevant commands from the sort menu
PrepareSourceControlMenu(pPopupMenu);
}
else // all other sub-menus
{
// test for 'Open From...'
if (pPopupMenu->GetMenuItemID(0) == ID_FILE_OPEN_USERSTORAGE1)
{
AddUserStorageToMenu(pPopupMenu);
}
// test for 'save To...'
else if (pPopupMenu->GetMenuItemID(0) == ID_FILE_SAVE_USERSTORAGE1)
{
AddUserStorageToMenu(pPopupMenu);
}
}
CFrameWnd::OnInitMenuPopup(pPopupMenu, nIndex, bSysMenu);
}
void CToDoListWnd::OnViewToolbar()
{
m_bShowToolbar = !m_bShowToolbar;
m_toolbar.ShowWindow(m_bShowToolbar ? SW_SHOW : SW_HIDE);
m_toolbar.EnableWindow(m_bShowToolbar);
Resize();
Invalidate(TRUE);
}
void CToDoListWnd::OnUpdateViewToolbar(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowToolbar);
}
void CToDoListWnd::AppendTools2Toolbar(BOOL bAppend)
{
CToolsHelper th(Prefs().GetEnableTDLExtension(), ID_TOOLS_USERTOOL1);
if (bAppend)
{
// then re-add
CUserToolArray aTools;
Prefs().GetUserTools(aTools);
th.AppendToolsToToolbar(aTools, m_toolbar, ID_PREFERENCES);
// refresh tooltips
m_tbHelper.Release();
m_tbHelper.Initialize(&m_toolbar, this);
}
else // remove
{
th.RemoveToolsFromToolbar(m_toolbar, ID_PREFERENCES);
}
// resize toolbar to accept the additional buttons
Resize();
}
void CToDoListWnd::OnSort()
{
GetToDoCtrl().Resort(TRUE);
}
void CToDoListWnd::OnWindowPosChanged(WINDOWPOS FAR* lpwndpos)
{
CFrameWnd::OnWindowPosChanged(lpwndpos);
}
void CToDoListWnd::OnArchiveCompletedtasks()
{
CWaitCursor cursor;
int nSelTDC = GetSelToDoCtrl();
if (m_mgrToDoCtrls.ArchiveDoneTasks(nSelTDC))
{
// auto-reload archive if it's loaded
CString sArchivePath = m_mgrToDoCtrls.GetArchivePath(nSelTDC);
int nArchiveTDC = m_mgrToDoCtrls.FindToDoCtrl(sArchivePath);
if (nArchiveTDC != -1 && m_mgrToDoCtrls.IsLoaded(nArchiveTDC))
ReloadTaskList(nArchiveTDC, FALSE, FALSE);
UpdateCaption();
}
}
void CToDoListWnd::OnUpdateArchiveCompletedtasks(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && !m_mgrToDoCtrls.GetArchivePath(tdc.GetFilePath()).IsEmpty());
}
void CToDoListWnd::OnArchiveSelectedTasks()
{
CWaitCursor cursor;
int nSelTDC = GetSelToDoCtrl();
if (m_mgrToDoCtrls.ArchiveSelectedTasks(nSelTDC))
{
// auto-reload archive if it's loaded
CString sArchivePath = m_mgrToDoCtrls.GetArchivePath(nSelTDC);
int nArchiveTDC = m_mgrToDoCtrls.FindToDoCtrl(sArchivePath);
if (nArchiveTDC != -1 && m_mgrToDoCtrls.IsLoaded(nArchiveTDC))
ReloadTaskList(nArchiveTDC, FALSE, FALSE);
UpdateCaption();
}
}
void CToDoListWnd::OnUpdateArchiveSelectedCompletedTasks(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && !m_mgrToDoCtrls.GetArchivePath(tdc.GetFilePath()).IsEmpty());
}
void CToDoListWnd::OnMovetaskdown()
{
GetToDoCtrl().MoveSelectedTask(TDCM_DOWN);
}
void CToDoListWnd::OnUpdateMovetaskdown(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanMoveSelectedTask(TDCM_DOWN));
}
void CToDoListWnd::OnMovetaskup()
{
GetToDoCtrl().MoveSelectedTask(TDCM_UP);
}
void CToDoListWnd::OnUpdateMovetaskup(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanMoveSelectedTask(TDCM_UP));
}
void CToDoListWnd::OnMovetaskright()
{
GetToDoCtrl().MoveSelectedTask(TDCM_RIGHT);
}
void CToDoListWnd::OnUpdateMovetaskright(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanMoveSelectedTask(TDCM_RIGHT));
}
void CToDoListWnd::OnMovetaskleft()
{
GetToDoCtrl().MoveSelectedTask(TDCM_LEFT);
}
void CToDoListWnd::OnUpdateMovetaskleft(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanMoveSelectedTask(TDCM_LEFT));
}
CFilteredToDoCtrl& CToDoListWnd::GetToDoCtrl()
{
return GetToDoCtrl(GetSelToDoCtrl());
}
CFilteredToDoCtrl& CToDoListWnd::GetToDoCtrl(int nIndex)
{
return m_mgrToDoCtrls.GetToDoCtrl(nIndex);
}
const CFilteredToDoCtrl& CToDoListWnd::GetToDoCtrl() const
{
return GetToDoCtrl(GetSelToDoCtrl());
}
const CFilteredToDoCtrl& CToDoListWnd::GetToDoCtrl(int nIndex) const
{
return m_mgrToDoCtrls.GetToDoCtrl(nIndex);
}
CFilteredToDoCtrl* CToDoListWnd::NewToDoCtrl(BOOL bVisible, BOOL bEnabled)
{
CFilteredToDoCtrl* pTDC = NULL;
// if the active tasklist is untitled and unmodified then reuse it
if (GetTDCCount())
{
int nSel = GetSelToDoCtrl();
CFilteredToDoCtrl& tdc = GetToDoCtrl();
// make sure that we don't accidently reuse a just edited tasklist
tdc.Flush();
if (m_mgrToDoCtrls.IsPristine(nSel))
{
pTDC = &tdc;
m_mgrToDoCtrls.RemoveToDoCtrl(nSel, FALSE);
// make sure we reset the selection to a valid index
if (GetTDCCount())
{
// try leaving the selection as-is
if (nSel >= GetTDCCount())
nSel--; // try the preceding item
SelectToDoCtrl(nSel, FALSE);
}
return pTDC;
}
}
// else
pTDC = new CFilteredToDoCtrl(m_mgrContent, Prefs().GetDefaultCommentsFormat());
if (pTDC && CreateToDoCtrl(pTDC, bVisible, bEnabled))
return pTDC;
// else
delete pTDC;
return NULL;
}
BOOL CToDoListWnd::CreateToDoCtrl(CFilteredToDoCtrl* pTDC, BOOL bVisible, BOOL bEnabled)
{
// create somewhere out in space
CRect rCtrl(-32010, -32010, -32000, -32000);
if (pTDC->Create(rCtrl, this, IDC_TODOLIST, bVisible, bEnabled))
{
// set font to our font
CDialogHelper::SetFont(pTDC, m_fontMain, FALSE);
if (!m_ilCheckboxes.GetSafeHandle())
InitCheckboxImageList();
UpdateToDoCtrlPreferences(pTDC);
if (CThemed::IsThemeActive())
pTDC->SetUITheme(m_theme);
for (int nExt = 0; nExt < m_mgrUIExtensions.GetNumUIExtensions(); nExt++)
pTDC->AddView(m_mgrUIExtensions.GetUIExtension(nExt));
return TRUE;
}
return FALSE;
}
BOOL CToDoListWnd::InitCheckboxImageList()
{
if (m_ilCheckboxes.GetSafeHandle())
return TRUE;
const int nStates[] = { -1, CBS_UNCHECKEDNORMAL, CBS_CHECKEDNORMAL };//, CBS_MIXEDNORMAL };
const int nNumStates = sizeof(nStates) / sizeof(int);
CThemed th;
if (th.Open(this, _T("BUTTON")) && th.AreControlsThemed())
{
th.BuildImageList(m_ilCheckboxes, BP_CHECKBOX, nStates, nNumStates);
}
// unthemed + fallback
if (!m_ilCheckboxes.GetSafeHandle())
{
CBitmap bitmap;
bitmap.LoadBitmap(IDB_CHECKBOXES);
BITMAP BM;
bitmap.GetBitmap(&BM);
if (m_ilCheckboxes.Create(BM.bmWidth / nNumStates, BM.bmHeight, ILC_COLOR32 | ILC_MASK, 0, 1))
m_ilCheckboxes.Add(&bitmap, 255);
}
return (NULL != m_ilCheckboxes.GetSafeHandle());
}
void CToDoListWnd::OnMBtnClickTabcontrol(NMHDR* pNMHDR, LRESULT* pResult)
{
NMTCMBTNCLK* pTCHDR = (NMTCMBTNCLK*)pNMHDR;
// check valid tab
if (pTCHDR->iTab >= 0)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(pTCHDR->iTab);
tdc.Flush();
CloseToDoCtrl(pTCHDR->iTab);
if (!GetTDCCount())
CreateNewTaskList(FALSE);
}
*pResult = 0;
}
void CToDoListWnd::OnSelchangeTabcontrol(NMHDR* /*pNMHDR*/, LRESULT* pResult)
{
// show the incoming selection
int nCurSel = GetSelToDoCtrl();
// check password if necessary
if (m_nLastSelItem != -1 && !VerifyToDoCtrlPassword())
{
m_tabCtrl.SetCurSel(m_nLastSelItem);
return;
}
int nDueBy = Prefs().GetNotifyDueByOnSwitch();
if (nCurSel != -1)
{
// make sure it's loaded
if (!VerifyTaskListOpen(nCurSel, (nDueBy == -1)))
{
// restore the previous tab
m_tabCtrl.SetCurSel(m_nLastSelItem);
return;
}
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCurSel);
UpdateToDoCtrlPreferences(&tdc);
// update the filter selection
RefreshFilterControls();
// update status bar
UpdateStatusbar();
UpdateCaption();
// make sure size is right
CRect rTDC;
if (CalcToDoCtrlRect(rTDC))
tdc.MoveWindow(rTDC, FALSE);
// refresh view state
tdc.SetMaximizeState(m_nMaxState);
tdc.EnableWindow(TRUE);
tdc.ShowWindow(SW_SHOW);
tdc.PauseTimeTracking(FALSE); // always
}
// hide the outgoing selection
if (m_nLastSelItem != -1)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(m_nLastSelItem);
tdc.ShowWindow(SW_HIDE);
tdc.EnableWindow(FALSE);
tdc.PauseTimeTracking(!Prefs().GetTrackNonActiveTasklists());
m_nLastSelItem = -1; // reset
}
if (nCurSel != -1)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCurSel);
// update find dialog with this ToDoCtrl's custom attributes
UpdateFindDialogCustomAttributes(&tdc);
// leave focus setting till last else the 'old' tasklist flashes
tdc.SetFocusToTasks();
// notify user of due tasks if req
DoDueTaskNotification(nCurSel, nDueBy);
UpdateAeroFeatures();
}
InitMenuIconManager();
*pResult = 0;
}
void CToDoListWnd::RefreshFilterControls()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
// if this tasklist's filter is an unnamed custom one
RefreshFilterBarCustomFilters();
// get existing filter bar size so we can determine if a
// resize if required
CRect rFilter;
m_filterBar.GetClientRect(rFilter);
m_filterBar.RefreshFilterControls(tdc);
CRect rClient;
GetClientRect(rClient);
if (m_filterBar.CalcHeight(rClient.Width()) != rFilter.Height())
Resize();
}
void CToDoListWnd::OnSelchangingTabcontrol(NMHDR* /*pNMHDR*/, LRESULT* pResult)
{
// cache the outgoing selection
m_nLastSelItem = GetSelToDoCtrl();
// and flush
if (m_nLastSelItem != -1)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(m_nLastSelItem);
tdc.Flush();
// and save
if (Prefs().GetAutoSaveOnSwitchTasklist() && !tdc.GetFilePath().IsEmpty() && tdc.IsModified())
{
tdc.Save();
m_mgrToDoCtrls.SetModifiedStatus(m_nLastSelItem, FALSE);
m_mgrToDoCtrls.RefreshLastModified(m_nLastSelItem);
m_mgrToDoCtrls.RefreshReadOnlyStatus(m_nLastSelItem);
m_mgrToDoCtrls.RefreshPathType(m_nLastSelItem);
}
}
*pResult = 0;
}
TDC_FILE CToDoListWnd::ConfirmSaveTaskList(int nIndex, DWORD dwFlags)
{
BOOL bClosingWindows = Misc::HasFlag(dwFlags, TDLS_CLOSINGWINDOWS);
BOOL bClosingTaskList = Misc::HasFlag(dwFlags, TDLS_CLOSINGTASKLISTS) || bClosingWindows; // sanity check
TDC_FILE nSave = TDCO_SUCCESS;
// save changes
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
if (tdc.IsModified())
{
BOOL bFirstTimeSave = tdc.GetFilePath().IsEmpty();
// if we are closing Windows, we don't bother asking
// we just save and get out as fast as poss
if (bClosingWindows)
{
// if it's a first time save we just save to a temp file
if (bFirstTimeSave)
tdc.Save(GetEndSessionFilePath());
else
tdc.Save();
return TDCO_SUCCESS;
}
// else we obey the user's preferences
else if (bClosingTaskList && (bFirstTimeSave || Prefs().GetConfirmSaveOnExit()))
{
// make sure app is visible
Show(FALSE);
// save tasklist
CString sName = m_mgrToDoCtrls.GetFriendlyProjectName(nIndex);
CEnString sMessage(IDS_SAVEBEFORECLOSE, sName);
// don't allow user to cancel if closing down
int nRet = MessageBox(sMessage, IDS_CONFIRMSAVE_TITLE, bClosingWindows ? MB_YESNO : MB_YESNOCANCEL);
if (nRet == IDYES)
{
// note: we omit the auto save parameter here because we want the user to
// be notified of any problems
nSave = SaveTaskList(nIndex);
// if the save failed (as opposed to cancelled) then we must
// propagate this upwards
if (nSave != TDCO_SUCCESS && nSave != TDCO_CANCELLED)
return nSave;
// user can still cancel save dialog even if closing down
// but not if closing windows
else if (nSave == TDCO_CANCELLED)
nRet = bClosingWindows ? IDNO : IDCANCEL;
}
ASSERT (!(bClosingWindows && nRet == IDCANCEL)); // sanity check
if (nRet == IDCANCEL)
return TDCO_CANCELLED;
else
{
tdc.SetModified(FALSE); // so we don't get prompted again
m_mgrToDoCtrls.SetModifiedStatus(nIndex, FALSE);
}
}
else
nSave = SaveTaskList(nIndex, NULL, Misc::HasFlag(dwFlags, TDLS_AUTOSAVE));
}
return nSave; // user did not cancel
}
BOOL CToDoListWnd::CloseToDoCtrl(int nIndex)
{
ASSERT (nIndex >= 0);
ASSERT (nIndex < GetTDCCount());
CFilteredToDoCtrl& tdcSel = GetToDoCtrl();
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
tdc.Flush(TRUE);
if (ConfirmSaveTaskList(nIndex, TDLS_CLOSINGTASKLISTS) != TDCO_SUCCESS)
return FALSE;
// remove any find results associated with this tasklist
if (m_findDlg.GetSafeHwnd())
m_findDlg.DeleteResults(&tdc);
CWaitCursor cursor;
// save off current reminders
m_reminders.CloseToDoCtrl(tdc);
int nNewSel = m_mgrToDoCtrls.RemoveToDoCtrl(nIndex, TRUE);
if (nNewSel != -1)
{
// if we're closing TDL then the main window will not
// be visible at this point so we don't have to worry about
// encrypted tasklists becoming visible. however if as a result
// of this closure an encrypted tasklist would become visible
// then we need to prompt for a password and if this fails
// we must create another tasklist to hide the encrypted one.
// unless the tasklist being closed was not active and the
// new selection hasn't actually changed
BOOL bCheckPassword = !m_bClosing && (&GetToDoCtrl(nNewSel) != &tdcSel);
if (!SelectToDoCtrl(nNewSel, bCheckPassword))
{
CreateNewTaskList(FALSE);
RefreshTabOrder();
}
if (!m_bClosing)
Resize();
}
return TRUE;
}
void CToDoListWnd::OnCloseTasklist()
{
int nSel = GetSelToDoCtrl();
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSel);
// make sure there are no edits pending
tdc.Flush(TRUE);
// check if its a pristine tasklist and the last tasklist and
// if so only close it if the default comments type has changed
if (m_mgrToDoCtrls.IsPristine(nSel) && GetTDCCount() == 1)
return;
CloseToDoCtrl(nSel);
// if empty then create a new dummy item
if (!GetTDCCount())
CreateNewTaskList(FALSE);
else
Resize();
}
BOOL CToDoListWnd::SelectToDoCtrl(LPCTSTR szFilePath, BOOL bCheckPassword, int nNotifyDueTasksBy)
{
int nCtrl = m_mgrToDoCtrls.FindToDoCtrl(szFilePath);
if (nCtrl != -1)
{
SelectToDoCtrl(nCtrl, bCheckPassword, nNotifyDueTasksBy);
return TRUE;
}
return FALSE;
}
int CToDoListWnd::GetSelToDoCtrl() const
{
if (m_tabCtrl.GetSafeHwnd())
return m_tabCtrl.GetCurSel();
else
return -1;
}
BOOL CToDoListWnd::VerifyTaskListOpen(int nIndex, BOOL bWantNotifyDueTasks)
{
if (!m_mgrToDoCtrls.IsLoaded(nIndex))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
TSM_TASKLISTINFO storageInfo;
CString sFilePath = tdc.GetFilePath();
if (m_mgrToDoCtrls.GetStorageDetails(nIndex, storageInfo))
sFilePath = storageInfo.EncodeInfo();
if (OpenTaskList(&tdc, sFilePath, &storageInfo) == TDCO_SUCCESS)
{
// make sure hidden windows stay hidden
if (nIndex != GetSelToDoCtrl())
tdc.ShowWindow(SW_HIDE);
// notify readonly
Resize();
CheckNotifyReadonly(nIndex);
m_mgrToDoCtrls.SetLoaded(nIndex);
m_mgrToDoCtrls.UpdateTabItemText(nIndex);
// update storage info
m_mgrToDoCtrls.SetStorageDetails(nIndex, storageInfo);
if (bWantNotifyDueTasks)
DoDueTaskNotification(nIndex, Prefs().GetNotifyDueByOnLoad());
return TRUE;
}
return FALSE;
}
return TRUE;
}
BOOL CToDoListWnd::SelectToDoCtrl(int nIndex, BOOL bCheckPassword, int nNotifyDueTasksBy)
{
ASSERT (nIndex >= 0);
ASSERT (nIndex < GetTDCCount());
// load and show the chosen item
// we don't need to do a 'open' due task notification if the caller
// has asked for a notification anyway
if (!m_bClosing)
{
// if the tasklist is not loaded and we verify its loading
// then we know that the password (if there is one) has been
// verified and doesn't need checking again
if (!m_mgrToDoCtrls.IsLoaded(nIndex) )
{
if (!VerifyTaskListOpen(nIndex, nNotifyDueTasksBy == -1))
{
// TODO
return FALSE;
}
else
bCheckPassword = FALSE;
}
}
// validate password first if necessary
if (bCheckPassword && !VerifyToDoCtrlPassword(nIndex))
return FALSE;
int nCurSel = GetSelToDoCtrl(); // cache this
// resize tdc first
CFilteredToDoCtrl& tdc = GetToDoCtrl(nIndex);
CRect rTDC;
if (CalcToDoCtrlRect(rTDC))
tdc.MoveWindow(rTDC);
m_tabCtrl.SetCurSel(nIndex); // this changes the selected CToDoCtrl
m_tabCtrl.UpdateWindow();
if (!m_bClosing)
UpdateToDoCtrlPreferences();
const CPreferencesDlg& userPrefs = Prefs();
tdc.EnableWindow(TRUE);
tdc.SetFocusToTasks();
tdc.ShowWindow(SW_SHOW);
tdc.PauseTimeTracking(FALSE); // always
tdc.SetMaximizeState(m_nMaxState);
// if the tasklist is encrypted and todolist always prompts for password
// then disable Flip3D and Aero Peek
UpdateAeroFeatures();
// before hiding the previous selection
if (nCurSel != -1 && nCurSel != nIndex)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCurSel);
tdc.ShowWindow(SW_HIDE);
tdc.EnableWindow(FALSE);
tdc.PauseTimeTracking(!userPrefs.GetTrackNonActiveTasklists());
}
if (!m_bClosing)
{
if (userPrefs.GetReadonlyReloadOption() != RO_NO)
OnTimerReadOnlyStatus(nIndex);
if (userPrefs.GetTimestampReloadOption() != RO_NO)
OnTimerTimestampChange(nIndex);
if (userPrefs.GetEnableSourceControl())
OnTimerCheckoutStatus(nIndex);
UpdateCaption();
UpdateStatusbar();
// update the filter selection
RefreshFilterControls();
// and the menu icon manager
InitMenuIconManager();
// and current directory
UpdateCwd();
DoDueTaskNotification(nNotifyDueTasksBy);
}
return TRUE;
}
void CToDoListWnd::UpdateAeroFeatures()
{
#ifdef _DEBUG
BOOL bEnable = !GetToDoCtrl().IsEncrypted();
#else
BOOL bEnable = (!m_bPasswordPrompting || !GetToDoCtrl().IsEncrypted());
#endif
// Disable peek and other dynamic views if the active tasklist is encrypted
GraphicsMisc::EnableFlip3D(*this, bEnable);
if (!GraphicsMisc::EnableAeroPeek(*this, bEnable))
GraphicsMisc::ForceIconicRepresentation(*this, !bEnable);
}
void CToDoListWnd::UpdateToDoCtrlPreferences()
{
// check if this has already been done since the last userPrefs change
int nSel = GetSelToDoCtrl();
if (m_mgrToDoCtrls.GetNeedsPreferenceUpdate(nSel))
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSel);
UpdateToDoCtrlPreferences(&tdc);
// we do column visibility a bit different because
// the manager knows whether the columns have been fiddled
// with or not
CTDCColumnIDArray aColumns;
m_mgrToDoCtrls.RefreshColumns(nSel, aColumns);
// and filter bar relies on this tdc's visible columns
m_filterBar.SetVisibleFilters(aColumns);
// reset flag
m_mgrToDoCtrls.SetNeedsPreferenceUpdate(nSel, FALSE);
}
}
void CToDoListWnd::UpdateToDoCtrlPreferences(CFilteredToDoCtrl* pTDC)
{
const CPreferencesDlg& userPrefs = Prefs();
CFilteredToDoCtrl& tdc = *pTDC;
tdc.NotifyBeginPreferencesUpdate();
CTDCStylesMap styles;
styles.InitHashTable(TDCS_LAST);
styles[TDCS_ALLOWPARENTTIMETRACKING] = userPrefs.GetAllowParentTimeTracking();
styles[TDCS_ALLOWREFERENCEEDITING] = userPrefs.GetAllowReferenceEditing();
styles[TDCS_ALWAYSHIDELISTPARENTS] = userPrefs.GetAlwaysHideListParents();
styles[TDCS_AUTOADJUSTDEPENDENCYDATES] = userPrefs.GetAutoAdjustDependentsDates();
styles[TDCS_AUTOCALCPERCENTDONE] = userPrefs.GetAutoCalcPercentDone();
styles[TDCS_AUTOCALCTIMEESTIMATES] = userPrefs.GetAutoCalcTimeEstimates();
styles[TDCS_AUTOREPOSCTRLS] = userPrefs.GetAutoReposCtrls();
styles[TDCS_AVERAGEPERCENTSUBCOMPLETION] = userPrefs.GetAveragePercentSubCompletion();
styles[TDCS_CALCREMAININGTIMEBYDUEDATE] = (userPrefs.GetTimeRemainingCalculation() == PTCP_REMAININGTTIMEISDUEDATE);
styles[TDCS_CALCREMAININGTIMEBYPERCENT] = (userPrefs.GetTimeRemainingCalculation() == PTCP_REMAININGTTIMEISPERCENTAGE);
styles[TDCS_CALCREMAININGTIMEBYSPENT] = (userPrefs.GetTimeRemainingCalculation() == PTCP_REMAININGTTIMEISSPENT);
styles[TDCS_COLORTEXTBYATTRIBUTE] = (userPrefs.GetTextColorOption() == COLOROPT_ATTRIB);
styles[TDCS_COLORTEXTBYNONE] = (userPrefs.GetTextColorOption() == COLOROPT_NONE);
styles[TDCS_COLORTEXTBYPRIORITY] = (userPrefs.GetTextColorOption() == COLOROPT_PRIORITY);
styles[TDCS_COLUMNHEADERSORTING] = userPrefs.GetEnableColumnHeaderSorting();
styles[TDCS_COMMENTSUSETREEFONT] = userPrefs.GetCommentsUseTreeFont();
styles[TDCS_CONFIRMDELETE] = userPrefs.GetConfirmDelete();
styles[TDCS_DISPLAYHMSTIMEFORMAT] = userPrefs.GetUseHMSTimeFormat();
styles[TDCS_DONEHAVELOWESTPRIORITY] = userPrefs.GetDoneTasksHaveLowestPriority();
styles[TDCS_DONEHAVELOWESTRISK] = userPrefs.GetDoneTasksHaveLowestRisk();
styles[TDCS_DUEHAVEHIGHESTPRIORITY] = userPrefs.GetDueTasksHaveHighestPriority();
styles[TDCS_FOCUSTREEONENTER] = userPrefs.GetFocusTreeOnEnter();
styles[TDCS_FULLROWSELECTION] = userPrefs.GetFullRowSelection();
styles[TDCS_HIDEDONETIMEFIELD] = userPrefs.GetHideDoneTimeField();
styles[TDCS_HIDEDUETIMEFIELD] = userPrefs.GetHideDueTimeField();
styles[TDCS_HIDEPERCENTFORDONETASKS] = userPrefs.GetHidePercentForDoneTasks();
styles[TDCS_HIDEPRIORITYNUMBER] = userPrefs.GetHidePriorityNumber();
styles[TDCS_HIDESTARTDUEFORDONETASKS] = userPrefs.GetHideStartDueForDoneTasks();
styles[TDCS_HIDESTARTTIMEFIELD] = userPrefs.GetHideStartTimeField();
styles[TDCS_HIDEZEROPERCENTDONE] = userPrefs.GetHideZeroPercentDone();
styles[TDCS_HIDEZEROTIMECOST] = userPrefs.GetHideZeroTimeCost();
styles[TDCS_INCLUDEDONEINAVERAGECALC] = userPrefs.GetIncludeDoneInAverageCalc();
styles[TDCS_INCLUDEDONEINPRIORITYCALC] = userPrefs.GetIncludeDoneInPriorityRiskCalc();
styles[TDCS_INCLUDEDONEINRISKCALC] = userPrefs.GetIncludeDoneInPriorityRiskCalc();
styles[TDCS_INCLUDEUSERINCHECKOUT] = userPrefs.GetIncludeUserNameInCheckout();
styles[TDCS_LOGTASKTIMESEPARATELY] = userPrefs.GetLogTaskTimeSeparately();
styles[TDCS_LOGTIMETRACKING] = userPrefs.GetLogTimeTracking();
styles[TDCS_NODUEDATEISDUETODAY] = userPrefs.GetNoDueDateIsDueToday();
styles[TDCS_PAUSETIMETRACKINGONSCRNSAVER] = !userPrefs.GetTrackOnScreenSaver();
styles[TDCS_REFILTERONMODIFY] = userPrefs.GetReFilterOnModify();
styles[TDCS_RESORTONMODIFY] = userPrefs.GetReSortOnModify();
styles[TDCS_RESTOREFILTERS] = userPrefs.GetRestoreTasklistFilters();
styles[TDCS_RIGHTSIDECOLUMNS] = userPrefs.GetShowColumnsOnRight();
styles[TDCS_ROUNDTIMEFRACTIONS] = userPrefs.GetRoundTimeFractions();
styles[TDCS_SHAREDCOMMENTSHEIGHT] = userPrefs.GetSharedCommentsHeight();
styles[TDCS_SHOWCOMMENTSALWAYS] = userPrefs.GetShowCommentsAlways();
styles[TDCS_SHOWCOMMENTSINLIST] = userPrefs.GetShowComments();
styles[TDCS_SHOWCTRLSASCOLUMNS] = userPrefs.GetShowCtrlsAsColumns();
styles[TDCS_SHOWDATESINISO] = userPrefs.GetDisplayDatesInISO();
styles[TDCS_SHOWDEFAULTTASKICONS] = userPrefs.GetShowDefaultTaskIcons();
styles[TDCS_SHOWFIRSTCOMMENTLINEINLIST] = userPrefs.GetDisplayFirstCommentLine();
styles[TDCS_SHOWINFOTIPS] = userPrefs.GetShowInfoTips();
styles[TDCS_SHOWNONFILEREFSASTEXT] = userPrefs.GetShowNonFilesAsText();
styles[TDCS_SHOWPARENTSASFOLDERS] = userPrefs.GetShowParentsAsFolders();
styles[TDCS_SHOWPATHINHEADER] = userPrefs.GetShowPathInHeader();
styles[TDCS_SHOWPERCENTASPROGRESSBAR] = userPrefs.GetShowPercentAsProgressbar();
styles[TDCS_SHOWPROJECTNAME] = m_bShowProjectName;
styles[TDCS_SHOWSUBTASKCOMPLETION] = userPrefs.GetShowSubtaskCompletion();
styles[TDCS_SHOWTREELISTBAR] = m_bShowTreeListBar;
styles[TDCS_SHOWWEEKDAYINDATES] = userPrefs.GetShowWeekdayInDates();
styles[TDCS_SORTDONETASKSATBOTTOM] = userPrefs.GetSortDoneTasksAtBottom();
styles[TDCS_SORTVISIBLETASKSONLY] = FALSE;//userPrefs.GetSortVisibleOnly();
styles[TDCS_STRIKETHOUGHDONETASKS] = userPrefs.GetStrikethroughDone();
styles[TDCS_TASKCOLORISBACKGROUND] = userPrefs.GetColorTaskBackground();
styles[TDCS_TRACKSELECTEDTASKONLY] = !userPrefs.GetTrackNonSelectedTasks();
styles[TDCS_TREATSUBCOMPLETEDASDONE] = userPrefs.GetTreatSubCompletedAsDone();
styles[TDCS_TREECHECKBOXES] = userPrefs.GetTreeCheckboxes();
styles[TDCS_TREETASKICONS] = userPrefs.GetTreeTaskIcons();
styles[TDCS_USEEARLIESTDUEDATE] = (userPrefs.GetDueDateCalculation() == PTCP_EARLIESTDUEDATE);
styles[TDCS_USEEARLIESTSTARTDATE] = (userPrefs.GetStartDateCalculation() == PTCP_EARLIESTSTARTDATE);
styles[TDCS_USEHIGHESTPRIORITY] = userPrefs.GetUseHighestPriority();
styles[TDCS_USEHIGHESTRISK] = userPrefs.GetUseHighestRisk();
styles[TDCS_USELATESTDUEDATE] = (userPrefs.GetDueDateCalculation() == PTCP_LATESTDUEDATE);
styles[TDCS_USELATESTSTARTDATE] = (userPrefs.GetStartDateCalculation() == PTCP_LATESTSTARTDATE);
styles[TDCS_USEPERCENTDONEINTIMEEST] = userPrefs.GetUsePercentDoneInTimeEst();
styles[TDCS_USES3RDPARTYSOURCECONTROL] = userPrefs.GetUsing3rdPartySourceControl();
styles[TDCS_WARNADDDELETEARCHIVE] = userPrefs.GetWarnAddDeleteArchive();
styles[TDCS_WEIGHTPERCENTCALCBYNUMSUB] = userPrefs.GetWeightPercentCompletionByNumSubtasks();
// source control
BOOL bSrcControl = m_mgrToDoCtrls.PathSupportsSourceControl(tdc.GetFilePath());
styles[TDCS_ENABLESOURCECONTROL] = bSrcControl;
styles[TDCS_CHECKOUTONLOAD] = bSrcControl && userPrefs.GetAutoCheckOut();
// set the styles in one hit
tdc.SetStyles(styles);
// layout
tdc.SetLayoutPositions((TDC_UILOCATION)userPrefs.GetControlsPos(),
(TDC_UILOCATION)userPrefs.GetCommentsPos(),
TRUE);
// info tips
tdc.SetMaxInfotipCommentsLength(userPrefs.GetMaxInfoTipCommentsLength());
// update default task preferences
tdc.SetDefaultTaskAttributes(m_tdiDefault);
// default string lists
CStringArray aItems;
if (userPrefs.GetDefaultListItems(PTDP_CATEGORY, aItems))
aItems.Append(m_tdiDefault.aCategories);
tdc.SetDefaultCategoryNames(aItems, userPrefs.GetCategoryListIsReadonly());
if (userPrefs.GetDefaultListItems(PTDP_ALLOCTO, aItems))
aItems.Append(m_tdiDefault.aAllocTo);
tdc.SetDefaultAllocToNames(aItems, userPrefs.GetAllocToListIsReadonly());
userPrefs.GetDefaultListItems(PTDP_STATUS, aItems);
tdc.SetDefaultStatusNames(aItems, userPrefs.GetStatusListIsReadonly());
userPrefs.GetDefaultListItems(PTDP_ALLOCBY, aItems);
tdc.SetDefaultAllocByNames(aItems, userPrefs.GetAllocByListIsReadonly());
// fonts
if (!m_fontTree.GetSafeHandle() || !m_fontComments.GetSafeHandle())
{
CString sFaceName;
int nFontSize;
if (!m_fontTree.GetSafeHandle() && userPrefs.GetTreeFont(sFaceName, nFontSize))
m_fontTree.Attach(GraphicsMisc::CreateFont(sFaceName, nFontSize));
if (!m_fontComments.GetSafeHandle() && userPrefs.GetCommentsFont(sFaceName, nFontSize))
m_fontComments.Attach(GraphicsMisc::CreateFont(sFaceName, nFontSize));
}
tdc.SetTreeFont(m_fontTree);
tdc.SetCommentsFont(m_fontComments);
// colours
tdc.SetGridlineColor(userPrefs.GetGridlineColor());
tdc.SetCompletedTaskColor(userPrefs.GetDoneTaskColor());
tdc.SetAlternateLineColor(userPrefs.GetAlternateLineColor());
tdc.SetFlaggedTaskColor(userPrefs.GetFlaggedTaskColor());
tdc.SetReferenceTaskColor(userPrefs.GetReferenceTaskColor());
tdc.SetPriorityColors(m_aPriorityColors);
COLORREF color, crToday;
userPrefs.GetStartedTaskColors(color, crToday);
tdc.SetStartedTaskColors(color, crToday);
userPrefs.GetDueTaskColors(color, crToday);
tdc.SetDueTaskColors(color, crToday);
CTDCColorMap mapColors;
CAttribColorArray aColors;
TDC_ATTRIBUTE nAttrib = TDCA_NONE;
int nColor = userPrefs.GetAttributeColors(nAttrib, aColors);
while (nColor--)
{
ATTRIBCOLOR& ac = aColors[nColor];
mapColors[ac.sAttrib] = ac.color;
}
tdc.SetAttributeColors(nAttrib, mapColors);
// drag drop
tdc.SetSubtaskDragDropPos(userPrefs.GetNewSubtaskPos() == PUIP_TOP);
// misc
tdc.SetMaxColumnWidth(userPrefs.GetMaxColumnWidth());
tdc.SetCheckImageList(m_ilCheckboxes);
tdc.SetPercentDoneIncrement(userPrefs.GetPercentDoneIncrement());
CString sStatus;
userPrefs.GetCompletionStatus(sStatus);
tdc.SetCompletionStatus(sStatus);
tdc.Flush(); // clear any outstanding issues
// we're done
tdc.NotifyEndPreferencesUpdate();
}
void CToDoListWnd::UpdateTabSwitchTooltip()
{
CToolTipCtrl* pTT = m_tabCtrl.GetToolTips();
if (pTT)
{
// get the string for tab switching
CString sShortcut = m_mgrShortcuts.GetShortcutTextByCmd(ID_VIEW_NEXT);
if (sShortcut.IsEmpty())
sShortcut = m_mgrShortcuts.GetShortcutTextByCmd(ID_VIEW_PREV);
pTT->DelTool(&m_tabCtrl); // always
if (!sShortcut.IsEmpty())
{
CEnString sTip(IDS_TABSWITCHTOOLTIP, sShortcut);
pTT->AddTool(&m_tabCtrl, sTip);
}
}
}
void CToDoListWnd::OnSaveall()
{
SaveAll(TDLS_INCLUDEUNSAVED | TDLS_FLUSH);
}
void CToDoListWnd::OnUpdateSaveall(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_mgrToDoCtrls.AnyIsModified());
}
void CToDoListWnd::OnCloseall()
{
// save first
TDC_FILE nSaveAll = SaveAll(TDLS_INCLUDEUNSAVED | TDLS_CLOSINGTASKLISTS | TDLS_FLUSH);
if (nSaveAll != TDCO_SUCCESS)
return;
// remove tasklists
int nCtrl = GetTDCCount();
while (nCtrl--)
m_mgrToDoCtrls.RemoveToDoCtrl(nCtrl, TRUE);
// if empty then create a new dummy item
if (!GetTDCCount())
CreateNewTaskList(FALSE);
else
Resize();
}
void CToDoListWnd::OnUpdateCloseall(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetTDCCount());
}
BOOL CToDoListWnd::DoQueryEndSession(BOOL bQuery, BOOL bEnding)
{
HWND hWnd = GetSafeHwnd();
// what we do here depends on whether we're on Vista or not
// we test for this by trying to load the new API functions
if (bQuery)
{
CEnString sReason(IDS_SHUTDOWNBLOCKREASON);
// if Vista and handling WM_QUERYENDSESSION
// we register our reason and return TRUE to
// get more time to clean up in WM_ENDSESSION
if (Misc::ShutdownBlockReasonCreate(hWnd, sReason))
return TRUE;
// else we're XP so we return TRUE to let shutdown continue
return TRUE;
}
// else do a proper shutdown
m_bEndingSession = TRUE;
DoExit(FALSE, bEnding);
// cleanup our shutdown reason if not handled in DoExit
Misc::ShutdownBlockReasonDestroy(hWnd);
// and return anything because it's ignored
return TRUE;
}
BOOL CToDoListWnd::OnQueryEndSession()
{
if (!CFrameWnd::OnQueryEndSession())
return FALSE;
return DoQueryEndSession(TRUE, FALSE);
}
void CToDoListWnd::OnEndSession(BOOL bEnding)
{
CFrameWnd::OnEndSession(bEnding);
DoQueryEndSession(FALSE, bEnding);
}
void CToDoListWnd::OnExit()
{
DoExit();
}
void CToDoListWnd::OnMinimizeToTray()
{
MinimizeToTray();
}
BOOL CToDoListWnd::DoExit(BOOL bRestart, BOOL bClosingWindows)
{
ASSERT (!(bClosingWindows && bRestart));
// save all first to ensure new tasklists get reloaded on startup
DWORD dwSaveFlags = TDLS_INCLUDEUNSAVED | TDLS_CLOSINGTASKLISTS | TDLS_FLUSH;
if (bClosingWindows)
dwSaveFlags |= TDLS_CLOSINGWINDOWS;
TDC_FILE nSaveAll = SaveAll(dwSaveFlags);
if (nSaveAll != TDCO_SUCCESS)
return FALSE; // user cancelled
// save settings before we close the tasklists
// to snapshot the currently open tasklists
SaveSettings();
m_bClosing = TRUE;
// hide the window as soon as possible so users do not
// see the machinations of closing down
if (m_bFindShowing)
m_findDlg.ShowWindow(SW_HIDE);
ShowWindow(SW_HIDE);
// remove tasklists
int nCtrl = GetTDCCount();
while (nCtrl--)
VERIFY(CloseToDoCtrl(nCtrl)); // shouldn't fail now
// if there are any still open then the user must have cancelled else destroy the window
ASSERT (GetTDCCount() == 0);
if (GetTDCCount() == 0)
{
// this will save any left over settings
// when it goes out of scope
{
CPreferences prefs;
m_mgrImportExport.Release();
m_tbHelper.Release();
m_mgrShortcuts.Release(&prefs);
m_mgrImportExport.Release();
m_mgrUIExtensions.Release();
m_mgrStorage.Release();
CFocusWatcher::Release();
CMouseWheelMgr::Release();
CEditShortcutMgr::Release();
}
// cleanup our shutdown reason
Misc::ShutdownBlockReasonDestroy(*this);
DestroyWindow();
if (bRestart)
{
CString sParams = AfxGetApp()->m_lpCmdLine;
::ShellExecute(NULL, NULL, FileMisc::GetModuleFileName(), sParams, NULL, SW_SHOW);
}
return TRUE;
}
// cancel
m_bClosing = FALSE;
return FALSE;
}
void CToDoListWnd::OnImportTasklist()
{
CString sImportPath;
TDLID_IMPORTTO nImportTo = TDIT_NEWTASKLIST;
int nImporter = -1;
do
{
CTDLImportDialog dialog(m_mgrImportExport);
if (dialog.DoModal() == IDOK)
{
// check file can be opened
nImportTo = dialog.GetImportTo();
nImporter = dialog.GetImporterIndex();
if (dialog.GetImportFromClipboard())
{
sImportPath = FileMisc::GetTempFileName(_T("ToDoList.import"), _T("txt"));
FileMisc::SaveFile(sImportPath, dialog.GetImportClipboardText());
}
else
sImportPath = dialog.GetImportFilePath();
// check file accessibility
if (sImportPath.IsEmpty() || FileMisc::CanOpenFile(sImportPath, TRUE))
break;
// else
MessageBox(CEnString(IDS_IMPORTFILE_CANTOPEN, sImportPath), IDS_IMPORTTASKLIST_TITLE);
sImportPath.Empty();
}
else // cancel
return;
}
while (sImportPath.IsEmpty());
// load/import tasks
DOPROGRESS(IDS_IMPORTPROGRESS)
// do the import
CTaskFile tasks;
BOOL bCancel = !m_mgrImportExport.ImportTaskList(sImportPath, &tasks, nImporter);
if (bCancel)
return;
if (!tasks.GetTaskCount())
{
// notify user
MessageBox(IDS_NOTASKSIMPORTED);
}
else
{
if (nImportTo == TDIT_NEWTASKLIST)
VERIFY(CreateNewTaskList(FALSE));
CFilteredToDoCtrl& tdc = GetToDoCtrl();
TDC_INSERTWHERE nWhere = (nImportTo == TDIT_SELECTEDTASK) ? TDC_INSERTATTOPOFSELTASK : TDC_INSERTATTOP;
VERIFY(tdc.InsertTasks(tasks, nWhere));
// if a new tasklist then set the project name
if (nImportTo == TDIT_NEWTASKLIST)
tdc.SetProjectName(tasks.GetProjectName());
m_mgrToDoCtrls.RefreshModifiedStatus(GetSelToDoCtrl());
UpdateCaption();
}
}
void CToDoListWnd::OnSetPriority(UINT nCmdID)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly() && tdc.HasSelection())
{
if (nCmdID == ID_EDIT_SETPRIORITYNONE)
tdc.SetSelectedTaskPriority(-2);
else
tdc.SetSelectedTaskPriority(nCmdID - ID_EDIT_SETPRIORITY0);
}
}
void CToDoListWnd::OnUpdateSetPriority(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(!tdc.IsReadOnly() && nSelCount);
int nPriority = pCmdUI->m_nID - ID_EDIT_SETPRIORITY0;
if (pCmdUI->m_nID == ID_EDIT_SETPRIORITYNONE)
nPriority = -2;
pCmdUI->SetCheck(nPriority == tdc.GetSelectedTaskPriority());
}
void CToDoListWnd::OnEditSetfileref()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly() && tdc.HasSelection())
{
CPreferences prefs;
CFileOpenDialog dialog(IDS_SETFILEREF_TITLE,
NULL,
tdc.GetSelectedTaskFileRef(),
EOFN_DEFAULTOPEN,
CEnString(IDS_ALLFILEFILTER));
if (dialog.DoModal(&prefs) == IDOK)
tdc.SetSelectedTaskFileRef(dialog.GetPathName());
}
}
void CToDoListWnd::OnUpdateEditSetfileref(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.HasSelection());
}
void CToDoListWnd::OnEditOpenfileref()
{
GetToDoCtrl().GotoSelectedTaskFileRef();
}
void CToDoListWnd::OnUpdateEditOpenfileref(CCmdUI* pCmdUI)
{
CEnString sFileRef = GetToDoCtrl().GetSelectedTaskFileRef();
if (sFileRef.IsEmpty())
pCmdUI->Enable(FALSE);
else
{
pCmdUI->Enable(TRUE);
// restrict file length to 250 pixels
CClientDC dc(this);
sFileRef.FormatDC(&dc, 250, ES_PATH);
pCmdUI->SetText(sFileRef);
}
}
LRESULT CToDoListWnd::OnPreferencesDefaultListChange(WPARAM wp, LPARAM lp)
{
// decode params
int nList = LOWORD(wp);
BOOL bAdd = HIWORD(wp);
LPCTSTR szItem = (LPCTSTR)lp;
switch (nList)
{
case PTDP_ALLOCBY:
break;
case PTDP_ALLOCTO:
break;
case PTDP_STATUS:
break;
case PTDP_CATEGORY:
break;
}
return 0L;
}
void CToDoListWnd::PopulateToolArgs(USERTOOLARGS& args) const
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
args.sTasklist = tdc.GetFilePath();
args.sTaskTitle = tdc.GetSelectedTaskTitle();
args.sTaskExtID = tdc.GetSelectedTaskExtID();
args.sTaskComments = tdc.GetSelectedTaskComments();
args.sTaskFileLink = tdc.GetSelectedTaskFileRef();
args.sTaskAllocBy = tdc.GetSelectedTaskAllocBy();
CDWordArray aIDs;
DWORD dwTemp;
if (tdc.GetSelectedTaskIDs(aIDs, dwTemp, FALSE))
args.sTaskIDs = Misc::FormatArray(aIDs, _T("|"));
CStringArray aAllocTo;
if (tdc.GetSelectedTaskAllocTo(aAllocTo))
args.sTaskAllocTo = Misc::FormatArray(aAllocTo, _T("|"));
}
LRESULT CToDoListWnd::OnPreferencesTestTool(WPARAM /*wp*/, LPARAM lp)
{
USERTOOL* pTool = (USERTOOL*)lp;
if (pTool)
{
USERTOOLARGS args;
PopulateToolArgs(args);
CToolsHelper th(Prefs().GetEnableTDLExtension(), ID_TOOLS_USERTOOL1);
th.TestTool(*pTool, args, m_pPrefs);
}
return 0;
}
LRESULT CToDoListWnd::OnPreferencesClearMRU(WPARAM /*wp*/, LPARAM /*lp*/)
{
m_mruList.RemoveAll();
m_mruList.WriteList(CPreferences());
return 0;
}
LRESULT CToDoListWnd::OnPreferencesCleanupDictionary(WPARAM /*wp*/, LPARAM lp)
{
LPCTSTR szLangFile = (LPCTSTR)lp;
ASSERT(szLangFile && *szLangFile);
if (szLangFile && *szLangFile)
{
DOPROGRESS(IDS_CLEANINGDICTPROGRESS)
CString sMasterDict = FileMisc::GetFolderFromFilePath(szLangFile);
sMasterDict += _T("\\YourLanguage.csv");
return CLocalizer::CleanupDictionary(sMasterDict);
}
return 0;
}
void CToDoListWnd::PrepareSortMenu(CMenu* pMenu)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (Prefs().GetShowEditMenuAsColumns())
{
int nCountLastSep = 0;
for (int nItem = 0; nItem < (int)pMenu->GetMenuItemCount(); nItem++)
{
BOOL bDelete = FALSE;
BOOL bIsSeparator = FALSE;
UINT nMenuID = pMenu->GetMenuItemID(nItem);
switch (nMenuID)
{
case ID_SORT_BYCOST: bDelete = !tdc.IsColumnShowing(TDCC_COST); break;
case ID_SORT_BYFLAG: bDelete = !tdc.IsColumnShowing(TDCC_FLAG); break;
case ID_SORT_BYDONEDATE: bDelete = !tdc.IsColumnShowing(TDCC_DONEDATE); break;
case ID_SORT_BYPRIORITY: bDelete = !tdc.IsColumnShowing(TDCC_PRIORITY); break;
case ID_SORT_BYPERCENT: bDelete = !tdc.IsColumnShowing(TDCC_PERCENT); break;
case ID_SORT_BYSTARTDATE: bDelete = !tdc.IsColumnShowing(TDCC_STARTDATE); break;
case ID_SORT_BYDUEDATE: bDelete = !tdc.IsColumnShowing(TDCC_DUEDATE); break;
case ID_SORT_BYTIMEEST: bDelete = !tdc.IsColumnShowing(TDCC_TIMEEST); break;
case ID_SORT_BYID: bDelete = !tdc.IsColumnShowing(TDCC_ID); break;
case ID_SORT_BYDONE: bDelete = !tdc.IsColumnShowing(TDCC_DONE); break;
case ID_SORT_BYALLOCBY: bDelete = !tdc.IsColumnShowing(TDCC_ALLOCBY); break;
case ID_SORT_BYSTATUS: bDelete = !tdc.IsColumnShowing(TDCC_STATUS); break;
case ID_SORT_BYCATEGORY: bDelete = !tdc.IsColumnShowing(TDCC_CATEGORY); break;
case ID_SORT_BYTIMESPENT: bDelete = !tdc.IsColumnShowing(TDCC_TIMESPENT); break;
case ID_SORT_BYALLOCTO: bDelete = !tdc.IsColumnShowing(TDCC_ALLOCTO); break;
case ID_SORT_BYCREATIONDATE:bDelete = !tdc.IsColumnShowing(TDCC_CREATIONDATE); break;
case ID_SORT_BYCREATEDBY: bDelete = !tdc.IsColumnShowing(TDCC_CREATEDBY); break;
case ID_SORT_BYMODIFYDATE: bDelete = !tdc.IsColumnShowing(TDCC_LASTMOD); break;
case ID_SORT_BYRISK: bDelete = !tdc.IsColumnShowing(TDCC_RISK); break;
case ID_SORT_BYEXTERNALID: bDelete = !tdc.IsColumnShowing(TDCC_EXTERNALID); break;
case ID_SORT_BYVERSION: bDelete = !tdc.IsColumnShowing(TDCC_VERSION); break;
case ID_SORT_BYRECURRENCE: bDelete = !tdc.IsColumnShowing(TDCC_RECURRENCE); break;
case ID_SORT_BYREMAINING: bDelete = !tdc.IsColumnShowing(TDCC_REMAINING); break;
case ID_SORT_BYRECENTEDIT: bDelete = !tdc.IsColumnShowing(TDCC_RECENTEDIT); break;
case ID_SORT_BYICON: bDelete = !tdc.IsColumnShowing(TDCC_ICON); break;
case ID_SORT_BYFILEREF: bDelete = !tdc.IsColumnShowing(TDCC_FILEREF); break;
case ID_SORT_BYTIMETRACKING:bDelete = !tdc.IsColumnShowing(TDCC_TRACKTIME); break;
case ID_SORT_BYPATH: bDelete = !tdc.IsColumnShowing(TDCC_PATH); break;
case ID_SORT_BYTAG: bDelete = !tdc.IsColumnShowing(TDCC_TAGS); break;
case ID_SORT_BYDEPENDENCY: bDelete = !tdc.IsColumnShowing(TDCC_DEPENDENCY); break;
// case ID_SORT_BYCOLOR: bDelete = (Prefs().GetTextColorOption() != COLOROPT_DEFAULT); break;
case ID_SEPARATOR:
bIsSeparator = TRUE;
bDelete = (nCountLastSep == 0);
nCountLastSep = 0;
break;
default: bDelete = FALSE; break;
}
// delete the item else increment the count since the last separator
if (bDelete)
{
pMenu->DeleteMenu(nItem, MF_BYPOSITION);
nItem--;
}
else if (!bIsSeparator)
nCountLastSep++;
}
}
// custom sort columns
// first delete all custom columns and the related separator
int nPosUnsorted = -1;
for (int nItem = 0; nItem < (int)pMenu->GetMenuItemCount(); nItem++)
{
UINT nMenuID = pMenu->GetMenuItemID(nItem);
if (nMenuID >= ID_SORT_BYCUSTOMCOLUMN_FIRST && nMenuID <= ID_SORT_BYCUSTOMCOLUMN_LAST)
{
pMenu->DeleteMenu(nItem, MF_BYPOSITION);
nItem--;
}
}
// separator is just before the separator before 'unsorted entry'
int nInsert = CEnMenu::GetMenuItemPos(pMenu->GetSafeHmenu(), ID_SORT_NONE) - 1;
ASSERT(nInsert >= 0);
// delete separator if exist
if (nInsert > 0 && pMenu->GetMenuItemID(nInsert - 1) == 0)
{
nInsert--;
pMenu->DeleteMenu(nInsert, MF_BYPOSITION);
}
// then re-add
CTDCCustomAttribDefinitionArray aAttribDefs;
if (tdc.GetCustomAttributeDefs(aAttribDefs))
{
// re-add separator on demand
BOOL bWantSep = TRUE;
for (int nCol = 0; nCol < aAttribDefs.GetSize(); nCol++)
{
const TDCCUSTOMATTRIBUTEDEFINITION& attribDef = aAttribDefs[nCol];
if (attribDef.bEnabled && attribDef.SupportsFeature(TDCCAF_SORT))
{
if (bWantSep)
{
bWantSep = FALSE;
pMenu->InsertMenu(nInsert, MF_BYPOSITION);
nInsert++;
}
UINT nMenuID = (attribDef.GetColumnID() - TDCC_CUSTOMCOLUMN_FIRST) + ID_SORT_BYCUSTOMCOLUMN_FIRST;
CEnString sColumn(IDS_CUSTOMCOLUMN, attribDef.sLabel);
pMenu->InsertMenu(nInsert, MF_BYPOSITION, nMenuID, sColumn);
nInsert++;
}
}
}
}
void CToDoListWnd::PrepareSourceControlMenu(CMenu* pMenu)
{
if (Prefs().GetEnableSourceControl())
return;
int nCountLastSep = 0;
for (int nItem = 0; nItem < (int)pMenu->GetMenuItemCount(); nItem++)
{
BOOL bDelete = FALSE;
BOOL bIsSeparator = FALSE;
UINT nMenuID = pMenu->GetMenuItemID(nItem);
switch (nMenuID)
{
case -1: // its a popup so recursively handle it
{
CMenu* pPopup = pMenu->GetSubMenu(nItem);
PrepareEditMenu(pPopup);
// if the popup is now empty remove it too
bDelete = !pPopup->GetMenuItemCount();
}
break;
case ID_TOOLS_CHECKIN:
case ID_TOOLS_CHECKOUT:
bDelete = TRUE;
break;
case ID_SEPARATOR:
bIsSeparator = TRUE;
bDelete = (nCountLastSep == 0);
nCountLastSep = 0;
break;
default: bDelete = FALSE; break;
}
// delete the item else increment the count since the last separator
if (bDelete)
{
pMenu->DeleteMenu(nItem, MF_BYPOSITION);
nItem--;
}
else if (!bIsSeparator)
nCountLastSep++;
}
}
void CToDoListWnd::PrepareEditMenu(CMenu* pMenu)
{
if (!Prefs().GetShowEditMenuAsColumns())
return;
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nCountLastSep = 0;
for (int nItem = 0; nItem < (int)pMenu->GetMenuItemCount(); nItem++)
{
BOOL bDelete = FALSE;
BOOL bIsSeparator = FALSE;
UINT nMenuID = pMenu->GetMenuItemID(nItem);
switch (nMenuID)
{
case -1: // its a popup so recursively handle it
{
CMenu* pPopup = pMenu->GetSubMenu(nItem);
if (pPopup)
{
PrepareEditMenu(pPopup);
// if the popup is now empty remove it too
bDelete = !pPopup->GetMenuItemCount();
}
}
break;
case ID_EDIT_TASKCOLOR:
case ID_EDIT_CLEARTASKCOLOR:
bDelete = (Prefs().GetTextColorOption() != COLOROPT_DEFAULT);
break;
case ID_EDIT_DECTASKPRIORITY:
case ID_EDIT_INCTASKPRIORITY:
case ID_EDIT_SETPRIORITYNONE:
case ID_EDIT_SETPRIORITY0:
case ID_EDIT_SETPRIORITY1:
case ID_EDIT_SETPRIORITY2:
case ID_EDIT_SETPRIORITY3:
case ID_EDIT_SETPRIORITY4:
case ID_EDIT_SETPRIORITY5:
case ID_EDIT_SETPRIORITY6:
case ID_EDIT_SETPRIORITY7:
case ID_EDIT_SETPRIORITY8:
case ID_EDIT_SETPRIORITY9:
case ID_EDIT_SETPRIORITY10:
bDelete = !tdc.IsColumnShowing(TDCC_PRIORITY);
break;
case ID_EDIT_OFFSETDATES:
bDelete = !(tdc.IsColumnShowing(TDCC_STARTDATE) ||
tdc.IsColumnShowing(TDCC_DUEDATE) ||
tdc.IsColumnShowing(TDCC_DONEDATE));
break;
case ID_EDIT_CLOCK_TASK:
bDelete = !(tdc.IsColumnShowing(TDCC_TRACKTIME) ||
tdc.IsColumnShowing(TDCC_TIMESPENT));
break;
case ID_SHOWTIMELOGFILE:
bDelete = !((tdc.IsColumnShowing(TDCC_TRACKTIME) ||
tdc.IsColumnShowing(TDCC_TIMESPENT)) &&
Prefs().GetLogTimeTracking());
break;
case ID_EDIT_DECTASKPERCENTDONE: bDelete = !tdc.IsColumnShowing(TDCC_PERCENT); break;
case ID_EDIT_INCTASKPERCENTDONE: bDelete = !tdc.IsColumnShowing(TDCC_PERCENT); break;
case ID_EDIT_OPENFILEREF: bDelete = !tdc.IsColumnShowing(TDCC_FILEREF); break;
case ID_EDIT_SETFILEREF: bDelete = !tdc.IsColumnShowing(TDCC_FILEREF); break;
case ID_EDIT_FLAGTASK: bDelete = !tdc.IsColumnShowing(TDCC_FLAG); break;
case ID_EDIT_RECURRENCE: bDelete = !tdc.IsColumnShowing(TDCC_RECURRENCE); break;
case ID_EDIT_GOTODEPENDENCY: bDelete = !tdc.IsColumnShowing(TDCC_DEPENDENCY); break;
case ID_EDIT_SETTASKICON:
case ID_EDIT_CLEARTASKICON:
bDelete = !(tdc.IsColumnShowing(TDCC_ICON) || Prefs().GetTreeTaskIcons());
break;
case ID_SEPARATOR:
bIsSeparator = TRUE;
bDelete = (nCountLastSep == 0);
nCountLastSep = 0;
break;
default: bDelete = FALSE; break;
}
// delete the item else increment the count since the last separator
if (bDelete)
{
pMenu->DeleteMenu(nItem, MF_BYPOSITION);
nItem--;
}
else if (!bIsSeparator)
nCountLastSep++;
}
// make sure last item is not a separator
int nLastItem = (int)pMenu->GetMenuItemCount() - 1;
if (pMenu->GetMenuItemID(nLastItem) == 0)
pMenu->DeleteMenu(nLastItem, MF_BYPOSITION);
}
void CToDoListWnd::OnViewNext()
{
if (GetTDCCount() < 2)
return;
int nNext = GetSelToDoCtrl() + 1;
if (nNext >= GetTDCCount())
nNext = 0;
SelectToDoCtrl(nNext, TRUE, Prefs().GetNotifyDueByOnSwitch());
}
void CToDoListWnd::OnUpdateViewNext(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetTDCCount() > 1);
}
void CToDoListWnd::OnViewPrev()
{
if (GetTDCCount() < 2)
return;
int nPrev = GetSelToDoCtrl() - 1;
if (nPrev < 0)
nPrev = GetTDCCount() - 1;
SelectToDoCtrl(nPrev, TRUE, Prefs().GetNotifyDueByOnSwitch());
}
void CToDoListWnd::OnUpdateViewPrev(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetTDCCount() > 1);
}
void CToDoListWnd::OnSysCommand(UINT nID, LPARAM lParam)
{
switch (nID)
{
case SC_MINIMIZE:
// we don't minimize if we're going to be hiding to then system tray
{
m_hwndLastFocus = ::GetFocus();
int nSysTrayOption = Prefs().GetSysTrayOption();
if (nSysTrayOption == STO_ONMINIMIZE || nSysTrayOption == STO_ONMINCLOSE)
MinimizeToTray();
else
{
// SPECIAL FIX: Apparently when Ultramon hooks the minimize
// button it ends up sending us a close message!
ShowWindow(SW_MINIMIZE);
}
return; // eat it
}
case SC_HOTKEY:
Show(FALSE);
return;
case SC_CLOSE:
// don't allow closing whilst reloading tasklists
if (m_bReloading)
return;
break;
case SC_RESTORE:
case SC_MAXIMIZE:
PostMessage(WM_APPRESTOREFOCUS, 0L, (LPARAM)m_hwndLastFocus);
break;
}
CFrameWnd::OnSysCommand(nID, lParam);
}
void CToDoListWnd::OnUpdateImport(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!GetToDoCtrl().IsReadOnly());
}
UINT CToDoListWnd::MapNewTaskPos(int nPos, BOOL bSubtask)
{
if (!bSubtask) // task
{
switch (nPos)
{
case PUIP_TOP: return ID_NEWTASK_ATTOP;
case PUIP_BOTTOM: return ID_NEWTASK_ATBOTTOM;
case PUIP_BELOW: return ID_NEWTASK_AFTERSELECTEDTASK;
case PUIP_ABOVE:
default: return ID_NEWTASK_BEFORESELECTEDTASK;
}
}
else // subtask
{
if (nPos == PUIP_BOTTOM)
return ID_NEWSUBTASK_ATBOTTOM;
else
return ID_NEWSUBTASK_ATTOP;
}
}
UINT CToDoListWnd::GetNewTaskCmdID()
{
return MapNewTaskPos(Prefs().GetNewTaskPos(), FALSE);
}
UINT CToDoListWnd::GetNewSubtaskCmdID()
{
return MapNewTaskPos(Prefs().GetNewSubtaskPos(), TRUE);
}
void CToDoListWnd::OnNewtask()
{
// convert to users choice
SendMessage(WM_COMMAND, GetNewTaskCmdID());
}
void CToDoListWnd::OnUpdateNewtask(CCmdUI* pCmdUI)
{
switch (GetNewTaskCmdID())
{
case ID_NEWTASK_ATTOPSELECTED:
OnUpdateNewtaskAttopSelected(pCmdUI);
break;
case ID_NEWTASK_ATBOTTOMSELECTED:
OnUpdateNewtaskAtbottomSelected(pCmdUI);
break;
case ID_NEWTASK_AFTERSELECTEDTASK:
OnUpdateNewtaskAfterselectedtask(pCmdUI);
break;
case ID_NEWTASK_BEFORESELECTEDTASK:
OnUpdateNewtaskBeforeselectedtask(pCmdUI);
break;
case ID_NEWTASK_ATTOP:
OnUpdateNewtaskAttop(pCmdUI);
break;
case ID_NEWTASK_ATBOTTOM:
OnUpdateNewtaskAtbottom(pCmdUI);
break;
}
}
void CToDoListWnd::OnNewsubtask()
{
// convert to users choice
SendMessage(WM_COMMAND, GetNewSubtaskCmdID());
}
void CToDoListWnd::OnUpdateNewsubtask(CCmdUI* pCmdUI)
{
switch (GetNewSubtaskCmdID())
{
case ID_NEWSUBTASK_ATTOP:
OnUpdateNewsubtaskAttop(pCmdUI);
break;
case ID_NEWSUBTASK_ATBOTTOM:
OnUpdateNewsubtaskAtBottom(pCmdUI);
break;
}
}
void CToDoListWnd::OnToolsCheckout()
{
int nSel = GetSelToDoCtrl();
// sanity check
if (!m_mgrToDoCtrls.CanCheckOut(nSel))
return;
CAutoFlag af(m_bSaving, TRUE);
CString sCheckedOutTo;
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSel);
if (tdc.CheckOut(sCheckedOutTo))
{
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nSel);
m_mgrToDoCtrls.SetLastCheckoutStatus(nSel, TRUE);
m_mgrToDoCtrls.RefreshLastModified(nSel);
UpdateCaption();
// update menu icon mgr
m_mgrMenuIcons.ChangeImageID(ID_TOOLS_CHECKOUT, ID_TOOLS_CHECKIN);
}
else // failed
{
m_mgrToDoCtrls.SetLastCheckoutStatus(nSel, FALSE);
// notify user
CEnString sMessage;
if (!sCheckedOutTo.IsEmpty())
{
sMessage.Format(IDS_CHECKEDOUTBYOTHER, tdc.GetFilePath(), sCheckedOutTo);
}
else
{
// if sCheckedOutTo is empty then the error is most likely because
// the file has been deleted or cannot be opened for editing
sMessage.Format(IDS_CHECKOUTFAILED, tdc.GetFilePath());
}
MessageBox(sMessage, IDS_CHECKOUT_TITLE, MB_OK | MB_ICONEXCLAMATION);
}
}
void CToDoListWnd::OnUpdateToolsCheckout(CCmdUI* pCmdUI)
{
int nSel = GetSelToDoCtrl();
pCmdUI->Enable(m_mgrToDoCtrls.CanCheckOut(nSel));
}
void CToDoListWnd::OnToolsToggleCheckin()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (tdc.IsCheckedOut())
OnToolsCheckin();
else
OnToolsCheckout();
}
void CToDoListWnd::OnUpdateToolsToggleCheckin(CCmdUI* pCmdUI)
{
int nSel = GetSelToDoCtrl();
BOOL bEnable = m_mgrToDoCtrls.CanCheckInOut(nSel);
pCmdUI->Enable(bEnable);
pCmdUI->SetCheck(bEnable && GetToDoCtrl().IsCheckedOut() ? 1 : 0);
}
void CToDoListWnd::OnToolsCheckin()
{
int nSel = GetSelToDoCtrl();
// sanity check
if (!m_mgrToDoCtrls.CanCheckIn(nSel))
return;
CAutoFlag af(m_bSaving, TRUE);
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSel);
ASSERT (!tdc.GetFilePath().IsEmpty());
tdc.Flush();
// save modifications
TDC_FILE nSave = TDCO_SUCCESS;
if (tdc.IsModified())
{
if (Prefs().GetConfirmSaveOnExit())
{
CString sName = m_mgrToDoCtrls.GetFriendlyProjectName(nSel);
CEnString sMessage(IDS_SAVEBEFORECHECKIN, sName);
int nRet = MessageBox(sMessage, IDS_CHECKIN_TITLE, MB_YESNOCANCEL);
switch (nRet)
{
case IDYES:
nSave = SaveTaskList(nSel);
break;
case IDNO:
ReloadTaskList(nSel, FALSE, FALSE);
break;
case IDCANCEL:
return;
}
}
else
SaveTaskList(nSel);
}
if (nSave == TDCO_SUCCESS && tdc.CheckIn())
{
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nSel);
m_mgrToDoCtrls.RefreshLastModified(nSel);
UpdateCaption();
// update menu icon mgr
m_mgrMenuIcons.ChangeImageID(ID_TOOLS_CHECKIN, ID_TOOLS_CHECKOUT);
}
m_mgrToDoCtrls.SetLastCheckoutStatus(nSel, TRUE); // so we won't try to immediately check it out again
}
void CToDoListWnd::OnUpdateToolsCheckin(CCmdUI* pCmdUI)
{
int nSel = GetSelToDoCtrl();
pCmdUI->Enable(m_mgrToDoCtrls.CanCheckIn(nSel));
}
void CToDoListWnd::OnExport()
{
const CPreferencesDlg& userPrefs = Prefs();
int nTDCCount = GetTDCCount(), nSelTDC = GetSelToDoCtrl();
ASSERT (nTDCCount >= 1);
CTDLExportDlg dialog(m_mgrImportExport,
nTDCCount == 1,
GetToDoCtrl().GetView(),
userPrefs.GetExportVisibleColsOnly(),
m_mgrToDoCtrls.GetFilePath(nSelTDC, FALSE),
userPrefs.GetAutoExportFolderPath());
// keep showing the dialog until either the user
// selects a filename which does not match a tasklist
// or they confirm that they want to overwrite the tasklist
CString sExportPath;
BOOL bOverWrite = FALSE;
int nFormat = -1;
while (!bOverWrite)
{
if (dialog.DoModal() != IDOK)
return;
sExportPath = dialog.GetExportPath();
nFormat = dialog.GetExportFormat();
// interested in overwriting single files
if (nTDCCount == 1 || !dialog.GetExportAllTasklists() || dialog.GetExportOneFile())
{
// check with user if they are about to override a tasklist
if (m_mgrToDoCtrls.FindToDoCtrl(sExportPath) != -1)
{
UINT nRet = MessageBox(IDS_CONFIRM_EXPORT_OVERWRITE, 0, MB_YESNOCANCEL, sExportPath);
if (nRet == IDCANCEL)
return;
// else
bOverWrite = (nRet == IDYES);
}
else
bOverWrite = TRUE; // nothing to fear
}
else // check all open tasklists
{
CString sFilePath, sExt = m_mgrImportExport.GetExporterFileExtension(nFormat);
for (int nCtrl = 0; nCtrl < nTDCCount; nCtrl++)
{
CString sPath = m_mgrToDoCtrls.GetFilePath(nCtrl);
CString sFName;
FileMisc::SplitPath(sPath, NULL, NULL, &sFName);
FileMisc::MakePath(sFilePath, NULL, sExportPath, sFName, sExt);
if (m_mgrToDoCtrls.FindToDoCtrl(sFilePath) != -1)
{
UINT nRet = MessageBox(IDS_CONFIRM_EXPORT_OVERWRITE, 0, MB_YESNOCANCEL, sFilePath);
if (nRet == IDCANCEL)
return;
// else
bOverWrite = (nRet == IDYES);
break;
}
}
// no matches?
bOverWrite = TRUE; // nothing to fear
}
}
UpdateWindow();
// export
DOPROGRESS(IDS_EXPORTPROGRESS)
BOOL bHtmlComments = (nFormat == EXPTOHTML);
if (nTDCCount == 1 || !dialog.GetExportAllTasklists())
{
// set the html image folder to be the output path with
// an different extension
CString sImgFolder;
if (bHtmlComments)
{
sImgFolder = sExportPath;
FileMisc::ReplaceExtension(sImgFolder, _T("html_images"));
FileMisc::DeleteFolderContents(sImgFolder, TRUE);
}
CFilteredToDoCtrl& tdc = GetToDoCtrl(nSelTDC);
CTaskFile tasks;
// Note: don't need to verify password if encrypted tasklist is active
GetTasks(tdc, bHtmlComments, FALSE, dialog.GetTaskSelection(), tasks, sImgFolder);
// add project name as report title
CString sTitle = m_mgrToDoCtrls.GetFriendlyProjectName(nSelTDC);
tasks.SetReportAttributes(sTitle);
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
if (m_mgrImportExport.ExportTaskList(&tasks, sExportPath, nFormat, FALSE))
{
// and preview
if (userPrefs.GetPreviewExport())
FileMisc::Run(*this, sExportPath, NULL, SW_SHOWNORMAL);
}
}
// multiple tasklists
else if (dialog.GetExportOneFile())
{
CMultiTaskFile taskFiles;
for (int nCtrl = 0; nCtrl < nTDCCount; nCtrl++)
{
// verify password
if (nCtrl != nSelTDC && !VerifyToDoCtrlPassword(nCtrl))
continue;
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
tdc.LockWindowUpdate();
// make sure it's loaded
if (VerifyTaskListOpen(nCtrl, FALSE))
{
CTaskFile& tasks = taskFiles.GetTaskFile(nCtrl);
tasks.Reset();
// set the html image folder to be the output path with
// an different extension
CString sImgFolder;
if (bHtmlComments)
{
sImgFolder = sExportPath;
FileMisc::ReplaceExtension(sImgFolder, _T("html_images"));
}
GetTasks(tdc, bHtmlComments, FALSE, dialog.GetTaskSelection(), tasks, sImgFolder);
// add project name as report title
CString sTitle = m_mgrToDoCtrls.GetFriendlyProjectName(nCtrl);
tasks.SetReportAttributes(sTitle);
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
}
tdc.UnlockWindowUpdate();
}
Resize();
if (m_mgrImportExport.ExportTaskLists(&taskFiles, sExportPath, nFormat, FALSE))
{
// and preview
if (userPrefs.GetPreviewExport())
FileMisc::Run(*this, sExportPath, NULL, SW_SHOWNORMAL);
}
}
else // separate files
{
CString sExt = m_mgrImportExport.GetExporterFileExtension(nFormat);
for (int nCtrl = 0; nCtrl < nTDCCount; nCtrl++)
{
// verify password
if (nCtrl != nSelTDC && !VerifyToDoCtrlPassword(nCtrl))
continue;
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
tdc.LockWindowUpdate();
// make sure it's loaded
if (VerifyTaskListOpen(nCtrl, FALSE))
{
// build filepath if required (if exporter has an extension)
CString sFilePath;
BOOL bOverWrite = -1;
if (!sExt.IsEmpty())
{
CString sPath = m_mgrToDoCtrls.GetFilePath(nCtrl);
// if the file has not been saved before we use the tab text
// and prompt the user to confirm
if (sPath.IsEmpty())
{
sPath = m_mgrToDoCtrls.GetFilePath(nCtrl, FALSE);
FileMisc::MakePath(sFilePath, NULL, sExportPath, sPath, sExt);
CFileSaveDialog dlgFile(IDS_EXPORTFILEAS_TITLE,
sExt,
sFilePath,
EOFN_DEFAULTSAVE,
m_mgrImportExport.GetExporterFileFilter(nFormat));
CPreferences prefs;
if (dlgFile.DoModal(&prefs) == IDOK)
sFilePath = dlgFile.GetPathName();
else
continue; // next tasklist
}
else
{
CString sFName;
FileMisc::SplitPath(sPath, NULL, NULL, &sFName);
FileMisc::MakePath(sFilePath, NULL, sExportPath, sFName, sExt);
}
}
// set the html image folder to be the output path with
// an different extension
CString sImgFolder;
if (bHtmlComments)
{
sImgFolder = sFilePath;
FileMisc::ReplaceExtension(sImgFolder, _T("html_images"));
}
CTaskFile tasks;
GetTasks(tdc, bHtmlComments, FALSE, dialog.GetTaskSelection(), tasks, sImgFolder);
// add project name as report title
CString sTitle = m_mgrToDoCtrls.GetFriendlyProjectName(nCtrl);
tasks.SetReportAttributes(sTitle);
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
m_mgrImportExport.ExportTaskList(&tasks, sFilePath, nFormat, FALSE);
}
tdc.UnlockWindowUpdate();
}
}
}
int CToDoListWnd::GetTasks(CFilteredToDoCtrl& tdc, BOOL bHtmlComments, BOOL bTransform,
TSD_TASKS nWhatTasks, TDCGETTASKS& filter, DWORD dwSelFlags,
CTaskFile& tasks, LPCTSTR szHtmlImageDir) const
{
// preferences
const CPreferencesDlg& userPrefs = Prefs();
// project name
tasks.SetProjectName(tdc.GetFriendlyProjectName());
tasks.SetCharSet(userPrefs.GetHtmlCharSet());
// export flags
filter.dwFlags |= TDCGTF_FILENAME;
if (userPrefs.GetExportParentTitleCommentsOnly())
filter.dwFlags |= TDCGTF_PARENTTITLECOMMENTSONLY;
if (bHtmlComments)
{
filter.dwFlags |= TDCGTF_HTMLCOMMENTS;
tasks.SetHtmlImageFolder(szHtmlImageDir);
// And delete all existing images in that folder
if (szHtmlImageDir && *szHtmlImageDir)
FileMisc::DeleteFolderContents(szHtmlImageDir, TRUE);
if (bTransform)
{
ASSERT(bHtmlComments);
filter.dwFlags |= TDCGTF_TRANSFORM;
}
}
// get the tasks
tdc.Flush();
switch (nWhatTasks)
{
case TSDT_ALL:
{
// if there's a filter present then we toggle it off
// grab the tasks and then toggle it back on
BOOL bNeedToggle = (tdc.HasFilter() || tdc.HasCustomFilter());
if (bNeedToggle)
{
::LockWindowUpdate(tdc.GetSafeHwnd());
tdc.ToggleFilter();
}
tdc.GetTasks(tasks, filter);
if (bNeedToggle)
{
tdc.ToggleFilter();
::LockWindowUpdate(NULL);
}
}
break;
case TSDT_FILTERED:
// if no filter is present then this just gets the whole lot
tdc.GetFilteredTasks(tasks, filter);
break;
case TSDT_SELECTED:
tdc.GetSelectedTasks(tasks, filter, dwSelFlags);
break;
}
// delete the HTML image folder if it is empty
// this will fail if it is not empty.
if (bHtmlComments && szHtmlImageDir && *szHtmlImageDir)
RemoveDirectory(szHtmlImageDir);
return tasks.GetTaskCount();
}
int CToDoListWnd::GetTasks(CFilteredToDoCtrl& tdc, BOOL bHtmlComments, BOOL bTransform,
const CTaskSelectionDlg& taskSel, CTaskFile& tasks, LPCTSTR szHtmlImageDir) const
{
DWORD dwSelFlags = 0;
TSD_TASKS nWhatTasks = taskSel.GetWantWhatTasks();
if (taskSel.GetWantSelectedTasks())
{
if (!taskSel.GetWantSelectedSubtasks())
dwSelFlags |= TDCGSTF_NOTSUBTASKS;
if (taskSel.GetWantSelectedParentTask())
dwSelFlags |= TDCGSTF_IMMEDIATEPARENT;
}
TDC_GETTASKS nFilter = TDCGT_ALL;
// build filter
if (taskSel.GetWantCompletedTasks() && !taskSel.GetWantInCompleteTasks())
nFilter = TDCGT_DONE;
else if (!taskSel.GetWantCompletedTasks() && taskSel.GetWantInCompleteTasks())
nFilter = TDCGT_NOTDONE;
TDCGETTASKS filter(nFilter);
// attributes to export
switch (taskSel.GetAttributeOption())
{
case TSDA_ALL:
break;
case TSDA_VISIBLE:
{
CTDCColumnIDArray aCols;
tdc.GetVisibleColumns(aCols);
MapColumnsToAttributes(aCols, filter.aAttribs);
if (taskSel.GetWantCommentsWithVisible())
filter.aAttribs.Add(TDCA_COMMENTS);
filter.aAttribs.Add(TDCA_CUSTOMATTRIB); // always
}
break;
case TSDA_USER:
taskSel.GetUserAttributes(filter.aAttribs);
filter.dwFlags |= TDCGTF_USERCOLUMNS;
break;
}
// get the tasks
return GetTasks(tdc, bHtmlComments, bTransform, nWhatTasks, filter, dwSelFlags, tasks, szHtmlImageDir);
}
void CToDoListWnd::OnUpdateExport(CCmdUI* pCmdUI)
{
// make sure at least one control has items
int nCtrl = GetTDCCount();
while (nCtrl--)
{
if (GetToDoCtrl().GetTaskCount())
{
pCmdUI->Enable(TRUE);
return;
}
}
// else
pCmdUI->Enable(FALSE);
}
void CToDoListWnd::OnToolsTransformactivetasklist()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
// pass the project name as the title field
CString sTitle = tdc.GetProjectName();
CTDLTransformDialog dialog(sTitle, tdc.GetView());
if (dialog.DoModal() != IDOK)
return;
CString sStylesheet = dialog.GetStylesheet();
sTitle = dialog.GetTitle();
// output path
CString sOutputPath(tdc.GetFilePath());
{
if (!sOutputPath.IsEmpty())
FileMisc::ReplaceExtension(sOutputPath, _T("html"));
CPreferences prefs;
CFileSaveDialog dialog(IDS_SAVETRANSFORM_TITLE,
_T("html"),
sOutputPath,
OFN_OVERWRITEPROMPT,
CEnString(IDS_TRANSFORMFILEFILTER),
this);
if (dialog.DoModal(&prefs) != IDOK)
return; // user elected not to proceed
sOutputPath = dialog.GetPathName();
}
// export
DOPROGRESS(IDS_TRANSFORMPROGRESS)
// set the html image folder to be the same as the
// output path without the extension
CString sHtmlImgFolder(sOutputPath);
FileMisc::ReplaceExtension(sHtmlImgFolder, _T("html_images"));
CTaskFile tasks;
GetTasks(tdc, TRUE, TRUE, dialog.GetTaskSelection(), tasks, sHtmlImgFolder);
// add title and date
COleDateTime date;
if (dialog.GetWantDate())
date = COleDateTime::GetCurrentTime();
tasks.SetReportAttributes(sTitle, date);
// save intermediate tasklist to file as required
LogIntermediateTaskList(tasks, tdc.GetFilePath());
if (tasks.TransformToFile(sStylesheet, sOutputPath, Prefs().GetHtmlCharSet()))
{
// preview
if (Prefs().GetPreviewExport())
FileMisc::Run(*this, sOutputPath, NULL, SW_SHOWNORMAL);
}
}
BOOL CToDoListWnd::LogIntermediateTaskList(CTaskFile& tasks, LPCTSTR szRefPath)
{
if (FileMisc::IsLoggingEnabled())
{
CString sRefName = FileMisc::RemoveExtension(FileMisc::GetFileNameFromPath(szRefPath));
CString sTempTaskPath = FileMisc::GetTempFileName(sRefName, _T("intermediate.txt"));
return tasks.Save(sTempTaskPath);
}
// else
return TRUE;
}
void CToDoListWnd::OnNexttopleveltask()
{
GetToDoCtrl().GotoNextTopLevelTask(TDCG_NEXT);
}
void CToDoListWnd::OnPrevtopleveltask()
{
GetToDoCtrl().GotoNextTopLevelTask(TDCG_PREV);
}
void CToDoListWnd::OnUpdateNexttopleveltask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanGotoNextTopLevelTask(TDCG_NEXT));
}
void CToDoListWnd::OnUpdatePrevtopleveltask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanGotoNextTopLevelTask(TDCG_PREV));
}
void CToDoListWnd::OnGotoNexttask()
{
GetToDoCtrl().GotoNextTask(TDCG_NEXT);
}
void CToDoListWnd::OnGotoPrevtask()
{
GetToDoCtrl().GotoNextTask(TDCG_PREV);
}
void CToDoListWnd::OnUpdateGotoPrevtask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanGotoNextTask(TDCG_PREV));
}
void CToDoListWnd::OnUpdateGotoNexttask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanGotoNextTask(TDCG_NEXT));
}
//------------------------------------------------------------------------
BOOL CToDoListWnd::InitFindDialog(BOOL bShow)
{
if (!m_findDlg.GetSafeHwnd())
{
UpdateFindDialogCustomAttributes();
VERIFY(m_findDlg.Initialize(this));
if (CThemed::IsThemeActive())
m_findDlg.SetUITheme(m_theme);
if (bShow)
m_findDlg.Show(bShow);
}
return (m_findDlg.GetSafeHwnd() != NULL);
}
void CToDoListWnd::OnFindTasks()
{
InitFindDialog();
if (IsWindowVisible())
{
// remove results from encrypted tasklists but not from the
// active tasklist
if (!m_findDlg.IsWindowVisible())
{
int nSelTDC = GetSelToDoCtrl();
int nTDC = GetTDCCount();
while (nTDC--)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl(nTDC);
if (nTDC != nSelTDC && tdc.IsEncrypted())
m_findDlg.DeleteResults(&tdc);
}
}
m_findDlg.Show();
}
m_bFindShowing = TRUE;
}
LRESULT CToDoListWnd::OnFindDlgClose(WPARAM /*wp*/, LPARAM /*lp*/)
{
m_bFindShowing = FALSE;
GetToDoCtrl().SetFocusToTasks();
return 0L;
}
LRESULT CToDoListWnd::OnFindDlgFind(WPARAM /*wp*/, LPARAM /*lp*/)
{
// set up the search
BOOL bSearchAll = m_findDlg.GetSearchAllTasklists();
int nSelTaskList = GetSelToDoCtrl();
int nFrom = bSearchAll ? 0 : nSelTaskList;
int nTo = bSearchAll ? GetTDCCount() - 1 : nSelTaskList;
// search params
SEARCHPARAMS params;
if (m_findDlg.GetSearchParams(params))
{
int nSel = GetSelToDoCtrl();
int bFirst = TRUE;
for (int nCtrl = nFrom; nCtrl <= nTo; nCtrl++)
{
// load or verify password unless tasklist is already active
if (nCtrl != nSel)
{
// load if necessary (in which case the password will have been checked)
if (!m_mgrToDoCtrls.IsLoaded(nCtrl))
{
if (!VerifyTaskListOpen(nCtrl, FALSE))
continue;
}
else if (!VerifyToDoCtrlPassword(nCtrl))
continue;
}
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
CResultArray aResults;
CHoldRedraw hr(m_bFindShowing ? m_findDlg : NULL);
if (tdc.FindTasks(params, aResults))
{
// use tasklist title from tabctrl
CString sTitle = m_mgrToDoCtrls.GetTabItemText(nCtrl);
m_findDlg.AddHeaderRow(sTitle, !bFirst);
for (int nResult = 0; nResult < aResults.GetSize(); nResult++)
AddFindResult(aResults[nResult], &tdc);
bFirst = FALSE;
}
}
}
// auto-select single results
/* if (m_findDlg.GetResultCount() == 1)
{
CFTDResultsArray results;
m_findDlg.GetAllResults(results);
ASSERT (results.GetSize() == 1);
if (OnFindSelectResult(0, (LPARAM)&results[0]))
m_findDlg.Show(FALSE);
}
else*/
m_findDlg.SetActiveWindow();
return 0;
}
void CToDoListWnd::AddFindResult(const SEARCHRESULT& result, const CFilteredToDoCtrl* pTDC)
{
CString sTitle = pTDC->GetTaskTitle(result.dwTaskID);
//CString sPath = pTDC->GetTaskPath(result.dwID);
m_findDlg.AddResult(result, sTitle, /*sPath,*/ pTDC);
}
LRESULT CToDoListWnd::OnFindSelectResult(WPARAM /*wp*/, LPARAM lp)
{
// extract Task ID
FTDRESULT* pResult = (FTDRESULT*)lp;
ASSERT (pResult->dwTaskID);
int nCtrl = m_mgrToDoCtrls.FindToDoCtrl(pResult->pTDC);
ASSERT(nCtrl != -1);
if (m_tabCtrl.GetCurSel() != nCtrl)
{
if (!SelectToDoCtrl(nCtrl, TRUE))
return 0L;
}
// we can't use pResult->pTDC because it's const
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
tdc.SetFocusToTasks();
if (tdc.GetSelectedTaskID() != pResult->dwTaskID)
{
if (!tdc.SelectTask(pResult->dwTaskID))
{
// perhaps the task is filtered out so we toggle the filter
// and try again
if (tdc.HasFilter())
{
tdc.ToggleFilter();
// if that also fails, we restore the filter
if (!tdc.SelectTask(pResult->dwTaskID))
tdc.ToggleFilter();
}
}
Invalidate();
UpdateWindow();
}
return 1L;
}
LRESULT CToDoListWnd::OnFindSelectAll(WPARAM /*wp*/, LPARAM /*lp*/)
{
if (!m_findDlg.GetResultCount())
return 0;
CWaitCursor cursor;
for (int nTDC = 0; nTDC < GetTDCCount(); nTDC++)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nTDC);
tdc.DeselectAll();
// collate the taskIDs
CDWordArray aTaskIDs;
m_findDlg.GetResultIDs(&tdc, aTaskIDs);
// select them in one hit
if (aTaskIDs.GetSize())
tdc.MultiSelectItems(aTaskIDs, TSHS_SELECT, (nTDC == GetSelToDoCtrl()));
}
// if find dialog is floating then hide it
if (!m_findDlg.IsDocked())
m_findDlg.Show(FALSE);
return 0;
}
LRESULT CToDoListWnd::OnFindApplyAsFilter(WPARAM /*wp*/, LPARAM lp)
{
CString sCustom((LPCTSTR)lp);
SEARCHPARAMS filter;
m_findDlg.GetSearchParams(filter);
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.SetCustomFilter(filter, sCustom);
tdc.SetFocusToTasks();
RefreshFilterBarCustomFilters();
m_filterBar.RefreshFilterControls(tdc);
// if find dialog is floating then hide it
if (!m_findDlg.IsDocked())
m_findDlg.Show(FALSE);
return 0;
}
LRESULT CToDoListWnd::OnFindAddSearch(WPARAM /*wp*/, LPARAM /*lp*/)
{
RefreshFilterBarCustomFilters();
return 0;
}
LRESULT CToDoListWnd::OnFindDeleteSearch(WPARAM /*wp*/, LPARAM /*lp*/)
{
RefreshFilterBarCustomFilters();
return 0;
}
void CToDoListWnd::RefreshFilterBarCustomFilters()
{
CStringArray aFilters;
m_findDlg.GetSavedSearches(aFilters);
// check for unnamed filter
if (m_findDlg.GetSafeHwnd())
{
CEnString sUnNamed(IDS_UNNAMEDFILTER);
if (m_findDlg.GetActiveSearch().IsEmpty() && Misc::Find(aFilters, sUnNamed, FALSE, FALSE) == -1)
aFilters.Add(sUnNamed);
}
m_filterBar.AddCustomFilters(aFilters);
}
//------------------------------------------------------------------------
LRESULT CToDoListWnd::OnDropFile(WPARAM wParam, LPARAM lParam)
{
TLDT_DATA* pData = (TLDT_DATA*)wParam;
CWnd* pTarget = (CWnd*)lParam;
if (pTarget == this) // we're the target
{
CString sFile = pData->pFilePaths ? pData->pFilePaths->GetAt(0) : _T("");
if (FileMisc::HasExtension(sFile, _T("tdl")) || FileMisc::HasExtension(sFile, _T("xml"))) // tasklist
{
TDC_FILE nRes = OpenTaskList(sFile);
HandleLoadTasklistError(nRes, sFile);
}
}
return 0L;
}
void CToDoListWnd::OnViewMovetasklistright()
{
m_mgrToDoCtrls.MoveToDoCtrl(GetSelToDoCtrl(), 1);
}
void CToDoListWnd::OnUpdateViewMovetasklistright(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!Prefs().GetKeepTabsOrdered() &&
m_mgrToDoCtrls.CanMoveToDoCtrl(GetSelToDoCtrl(), 1));
}
void CToDoListWnd::OnViewMovetasklistleft()
{
m_mgrToDoCtrls.MoveToDoCtrl(GetSelToDoCtrl(), -1);
}
void CToDoListWnd::OnUpdateViewMovetasklistleft(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!Prefs().GetKeepTabsOrdered() &&
m_mgrToDoCtrls.CanMoveToDoCtrl(GetSelToDoCtrl(), -1));
}
void CToDoListWnd::OnToolsShowtasksDue(UINT nCmdID)
{
int nDueBy = PFP_DUETODAY;
UINT nIDDueBy = IDS_NODUETODAY;
switch (nCmdID)
{
case ID_TOOLS_SHOWTASKS_DUETODAY:
break; // done
case ID_TOOLS_SHOWTASKS_DUETOMORROW:
nIDDueBy = IDS_NODUETOMORROW;
nDueBy = PFP_DUETOMORROW;
break;
case ID_TOOLS_SHOWTASKS_DUEENDTHISWEEK:
nIDDueBy = IDS_NODUETHISWEEK;
nDueBy = PFP_DUETHISWEEK;
break;
case ID_TOOLS_SHOWTASKS_DUEENDNEXTWEEK:
nIDDueBy = IDS_NODUENEXTWEEK;
nDueBy = PFP_DUENEXTWEEK;
break;
case ID_TOOLS_SHOWTASKS_DUEENDTHISMONTH:
nIDDueBy = IDS_NODUETHISMONTH;
nDueBy = PFP_DUETHISMONTH;
break;
case ID_TOOLS_SHOWTASKS_DUEENDNEXTMONTH:
nIDDueBy = IDS_NODUENEXTMONTH;
nDueBy = PFP_DUENEXTMONTH;
break;
default:
ASSERT(0);
return;
}
if (!DoDueTaskNotification(nDueBy))
{
MessageBox(nIDDueBy, 0, MB_OK, m_mgrToDoCtrls.GetFriendlyProjectName(GetSelToDoCtrl()));
}
}
void CToDoListWnd::ResetPrefs()
{
delete m_pPrefs;
m_pPrefs = new CPreferencesDlg(&m_mgrShortcuts, IDR_MAINFRAME, &m_mgrContent, &m_mgrImportExport);
// update
m_mgrToDoCtrls.SetPrefs(m_pPrefs);
// grab current colors
Prefs().GetPriorityColors(m_aPriorityColors);
m_filterBar.SetPriorityColors(m_aPriorityColors);
}
const CPreferencesDlg& CToDoListWnd::Prefs() const
{
ASSERT (m_pPrefs);
return *m_pPrefs;
}
void CToDoListWnd::OnSpellcheckcomments()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.SpellcheckSelectedTask(FALSE);
}
void CToDoListWnd::OnUpdateSpellcheckcomments(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanSpellcheckSelectedTaskComments());
}
void CToDoListWnd::OnSpellchecktitle()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.SpellcheckSelectedTask(TRUE);
}
void CToDoListWnd::OnUpdateSpellchecktitle(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.GetSelectedTaskTitle().IsEmpty());
}
void CToDoListWnd::OnFileEncrypt()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly())
{
BOOL bWasEncrypted = tdc.IsEncrypted();
CString sPassword = tdc.GetPassword();
// if the tasklist is already encrypted then verify password
// before allowing change
if (!bWasEncrypted || VerifyToDoCtrlPassword())
tdc.EnableEncryption(!tdc.IsEncrypted());
// make sure we disable encryption on the archive too
if (bWasEncrypted)
{
CString sArchive = m_mgrToDoCtrls.GetArchivePath(GetSelToDoCtrl());
if (FileMisc::FileExists(sArchive))
{
CTaskFile archive(sPassword);
if (archive.Load(sArchive))
{
archive.SetPassword(NULL); // remove password
archive.Save(sArchive);
}
}
}
}
UpdateAeroFeatures();
}
void CToDoListWnd::OnUpdateFileEncrypt(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanEncrypt() && !tdc.IsReadOnly());
pCmdUI->SetCheck(tdc.IsEncrypted() ? 1 : 0);
}
void CToDoListWnd::OnFileResetversion()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly())
{
tdc.ResetFileVersion();
tdc.SetModified();
UpdateStatusbar();
UpdateCaption();
}
}
void CToDoListWnd::OnUpdateFileResetversion(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly());
}
void CToDoListWnd::OnSpellcheckTasklist()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.Spellcheck();
}
void CToDoListWnd::OnUpdateSpellcheckTasklist(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.GetTaskCount());
}
TDC_FILE CToDoListWnd::SaveAll(DWORD dwFlags)
{
TDC_FILE nSaveAll = TDCO_SUCCESS;
int nCtrl = GetTDCCount();
BOOL bIncUnsaved = Misc::HasFlag(dwFlags, TDLS_INCLUDEUNSAVED);
BOOL bClosingWindows = Misc::HasFlag(dwFlags, TDLS_CLOSINGWINDOWS);
BOOL bClosingAll = Misc::HasFlag(dwFlags, TDLS_CLOSINGTASKLISTS);
// scoped to end status bar progress
// before calling UpdateStatusbar
{
DOPROGRESS(IDS_SAVINGPROGRESS)
while (nCtrl--)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nCtrl);
// bypass unsaved tasklists unless closing Windows
if (!bClosingWindows && !bIncUnsaved && tdc.GetFilePath().IsEmpty())
continue;
if (Misc::HasFlag(dwFlags, TDLS_FLUSH))
tdc.Flush(bClosingAll);
TDC_FILE nSave = ConfirmSaveTaskList(nCtrl, dwFlags);
if (nSave == TDCO_CANCELLED) // user cancelled
return TDCO_CANCELLED;
// else cache any failure w/o overwriting previous
if (nSaveAll == TDCO_SUCCESS)
nSaveAll = nSave;
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
}
}
if (!bClosingWindows)
{
UpdateCaption();
UpdateStatusbar();
}
return nSaveAll;
}
void CToDoListWnd::OnEditTimeTrackTask()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.TimeTrackSelectedTask();
}
void CToDoListWnd::OnUpdateEditTimeTrackTask(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanTimeTrackSelectedTask());
pCmdUI->SetCheck(tdc.IsSelectedTaskBeingTimeTracked() ? 1 : 0);
}
void CToDoListWnd::OnDrawItem(int nIDCtl, LPDRAWITEMSTRUCT lpDrawItemStruct)
{
if (nIDCtl == IDC_TABCONTROL)
{
TDCM_DUESTATUS nStatus = m_mgrToDoCtrls.GetDueItemStatus(lpDrawItemStruct->itemID);
if (nStatus == TDCM_PAST || nStatus == TDCM_TODAY)
{
// determine appropriate due colour
COLORREF crDue, crDueToday;
GetToDoCtrl(lpDrawItemStruct->itemID).GetDueTaskColors(crDue, crDueToday);
COLORREF crTag = (nStatus == TDCM_PAST) ? crDue : crDueToday;
if (crTag != CLR_NONE)
{
CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
const CRect& rect = lpDrawItemStruct->rcItem;
// draw a little tag in the top left corner
for (int nHPos = 0; nHPos < 6; nHPos++)
{
for (int nVPos = 0; nVPos < 6 - nHPos; nVPos++)
{
pDC->SetPixelV(rect.left + nHPos, rect.top + nVPos, crTag);
}
}
// draw a black line between the two
pDC->SelectStockObject(BLACK_PEN);
pDC->MoveTo(rect.left, rect.top + 6);
pDC->LineTo(rect.left + 7, rect.top - 1);
}
}
return;
}
else if (nIDCtl == 0 && lpDrawItemStruct->itemID == ID_CLOSE)
{
if (m_menubar.DrawMDIButton(lpDrawItemStruct))
return;
}
CFrameWnd::OnDrawItem(nIDCtl, lpDrawItemStruct);
}
void CToDoListWnd::OnMeasureItem(int nIDCtl, LPMEASUREITEMSTRUCT lpMeasureItemStruct)
{
if (nIDCtl == 0 && lpMeasureItemStruct->itemID == ID_CLOSE)
{
if (m_menubar.MeasureMDIButton(lpMeasureItemStruct))
return;
}
CFrameWnd::OnMeasureItem(nIDCtl, lpMeasureItemStruct);
}
void CToDoListWnd::OnViewNextSel()
{
GetToDoCtrl().SelectNextTasksInHistory();
}
void CToDoListWnd::OnUpdateViewNextSel(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanSelectNextTasksInHistory());
}
void CToDoListWnd::OnViewPrevSel()
{
GetToDoCtrl().SelectPrevTasksInHistory();
}
void CToDoListWnd::OnUpdateViewPrevSel(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanSelectPrevTasksInHistory());
}
void CToDoListWnd::OnSplitTaskIntoPieces(UINT nCmdID)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nNumPieces = 2 + (nCmdID - ID_NEWTASK_SPLITTASKINTO_TWO);
tdc.SplitSelectedTask(nNumPieces);
}
void CToDoListWnd::OnUpdateSplitTaskIntoPieces(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanSplitSelectedTask());
}
void CToDoListWnd::OnViewExpandtask()
{
GetToDoCtrl().ExpandTasks(TDCEC_SELECTED, TRUE);
}
void CToDoListWnd::OnUpdateViewExpandtask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_SELECTED, TRUE));
}
void CToDoListWnd::OnViewCollapsetask()
{
GetToDoCtrl().ExpandTasks(TDCEC_SELECTED, FALSE);
}
void CToDoListWnd::OnUpdateViewCollapsetask(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_SELECTED, FALSE));
}
void CToDoListWnd::OnViewExpandall()
{
GetToDoCtrl().ExpandTasks(TDCEC_ALL, TRUE);
}
void CToDoListWnd::OnUpdateViewExpandall(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_ALL, TRUE));
}
void CToDoListWnd::OnViewCollapseall()
{
GetToDoCtrl().ExpandTasks(TDCEC_ALL, FALSE);
}
void CToDoListWnd::OnUpdateViewCollapseall(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_ALL, FALSE));
}
void CToDoListWnd::OnViewExpandDuetasks()
{
GetToDoCtrl().ExpandTasks(TDCEC_DUE, TRUE);
}
void CToDoListWnd::OnUpdateViewExpandDuetasks(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_DUE, TRUE));
}
void CToDoListWnd::OnViewCollapseDuetasks()
{
GetToDoCtrl().ExpandTasks(TDCEC_DUE, FALSE);
}
void CToDoListWnd::OnUpdateViewCollapseDuetasks(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_DUE, FALSE));
}
void CToDoListWnd::OnViewExpandStartedtasks()
{
GetToDoCtrl().ExpandTasks(TDCEC_STARTED, TRUE);
}
void CToDoListWnd::OnUpdateViewExpandStartedtasks(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_STARTED, TRUE));
}
void CToDoListWnd::OnViewCollapseStartedtasks()
{
GetToDoCtrl().ExpandTasks(TDCEC_STARTED, FALSE);
}
void CToDoListWnd::OnUpdateViewCollapseStartedtasks(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanExpandTasks(TDCEC_STARTED, FALSE));
}
void CToDoListWnd::OnViewToggletaskexpanded()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.ExpandTasks(TDCEC_SELECTED, tdc.CanExpandTasks(TDCEC_SELECTED, TRUE));
}
void CToDoListWnd::OnUpdateViewToggletaskexpanded(CCmdUI* pCmdUI)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanExpandTasks(TDCEC_SELECTED, TRUE) || tdc.CanExpandTasks(TDCEC_SELECTED, FALSE));
}
void CToDoListWnd::OnWindow(UINT nCmdID)
{
int nTDC = (int)(nCmdID - ID_WINDOW1);
if (nTDC < GetTDCCount())
SelectToDoCtrl(nTDC, (nTDC != GetSelToDoCtrl()), Prefs().GetNotifyDueByOnSwitch());
}
void CToDoListWnd::OnUpdateWindow(CCmdUI* pCmdUI)
{
if (pCmdUI->m_pMenu)
{
ASSERT (ID_WINDOW1 == pCmdUI->m_nID);
const UINT MAXWINDOWS = 16;
int nWnd;
// delete existing tool entries first
for (nWnd = 0; nWnd < MAXWINDOWS; nWnd++)
pCmdUI->m_pMenu->DeleteMenu(ID_WINDOW1 + nWnd, MF_BYCOMMAND);
int nSel = GetSelToDoCtrl();
int nPos = 0, nTDCCount = GetTDCCount();
ASSERT (nTDCCount);
nTDCCount = min(nTDCCount, MAXWINDOWS);
for (nWnd = 0; nWnd < nTDCCount; nWnd++)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl(nWnd);
CString sMenuItem;
sMenuItem.Format(_T("&%d (%s)"), (nPos + 1) % 10, tdc.GetFriendlyProjectName());
UINT nFlags = MF_BYPOSITION | MF_STRING | (nSel == nWnd ? MF_CHECKED : MF_UNCHECKED);
pCmdUI->m_pMenu->InsertMenu(pCmdUI->m_nIndex++, nFlags, ID_WINDOW1 + nWnd, sMenuItem);
nPos++;
}
// update end menu count
pCmdUI->m_nIndex--; // point to last menu added
pCmdUI->m_nIndexMax = pCmdUI->m_pMenu->GetMenuItemCount();
pCmdUI->m_bEnableChanged = TRUE; // all the added items are enabled
}
}
#if _MSC_VER >= 1400
void CToDoListWnd::OnActivateApp(BOOL bActive, DWORD dwThreadID)
#else
void CToDoListWnd::OnActivateApp(BOOL bActive, HTASK hTask)
#endif
{
// don't activate when in the middle of loading
if (m_bReloading && !bActive)
return;
#if _MSC_VER >= 1400
CFrameWnd::OnActivateApp(bActive, dwThreadID);
#else
CFrameWnd::OnActivateApp(bActive, hTask);
#endif
// don't do any further processing if closing
if (m_bClosing)
return;
if (!bActive)
{
// save focus to restore when we next get activated
HWND hFocus = ::GetFocus();
if (hFocus)
m_hwndLastFocus = hFocus;
// save tasklists if required
if (Prefs().GetAutoSaveOnSwitchApp())
SaveAll(TDLS_FLUSH | TDLS_AUTOSAVE);
}
else
{
if (GetTDCCount() && (!m_hwndLastFocus || Prefs().GetAutoFocusTasklist()))
PostMessage(WM_APPRESTOREFOCUS, 0L, (LPARAM)GetToDoCtrl().GetSafeHwnd());
else if (m_hwndLastFocus)
{
// delay the restoration of focus else it gets lost
PostMessage(WM_APPRESTOREFOCUS, 0L, (LPARAM)m_hwndLastFocus);
}
UpdateCwd();
}
}
LRESULT CToDoListWnd::OnAppRestoreFocus(WPARAM /*wp*/, LPARAM lp)
{
HWND hWnd = (HWND)lp;
if (GetTDCCount() && hWnd == GetToDoCtrl().GetSafeHwnd())
GetToDoCtrl().SetFocusToTasks();
else
return (LRESULT)::SetFocus(hWnd);
return 0L;
}
void CToDoListWnd::UpdateCwd()
{
// set cwd to active tasklist
if (GetTDCCount())
{
CString sFolder = FileMisc::GetFolderFromFilePath(m_mgrToDoCtrls.GetFilePath(GetSelToDoCtrl()));
if (FileMisc::FolderExists(sFolder))
SetCurrentDirectory(sFolder);
}
}
BOOL CToDoListWnd::OnCommand(WPARAM wParam, LPARAM lParam)
{
UpdateWindow();
return CFrameWnd::OnCommand(wParam, lParam);
}
void CToDoListWnd::OnEnable(BOOL bEnable)
{
CFrameWnd::OnEnable(bEnable);
// save current focus because modal window is being shown
if (!bEnable)
{
HWND hFocus = ::GetFocus();
if (hFocus)
m_hwndLastFocus = hFocus;
}
// then restore it when we are enabled
else if (m_hwndLastFocus)
{
UpdateWindow();
PostMessage(WM_APPRESTOREFOCUS, 0L, (LPARAM)m_hwndLastFocus);
}
}
void CToDoListWnd::OnViewSorttasklisttabs()
{
int nSel = m_mgrToDoCtrls.SortToDoCtrlsByName();
SelectToDoCtrl(nSel, FALSE);
}
void CToDoListWnd::OnUpdateViewSorttasklisttabs(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetTDCCount() > 1 && !Prefs().GetKeepTabsOrdered());
}
void CToDoListWnd::OnEditInctaskpercentdone()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.IncrementSelectedTaskPercentDone(TRUE);
}
void CToDoListWnd::OnUpdateEditInctaskpercentdone(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnEditDectaskpercentdone()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.IncrementSelectedTaskPercentDone(FALSE);
}
void CToDoListWnd::OnUpdateEditDectaskpercentdone(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnEditDectaskpriority()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.IncrementSelectedTaskPriority(FALSE);
}
void CToDoListWnd::OnUpdateEditDectaskpriority(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnEditInctaskpriority()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.IncrementSelectedTaskPriority(TRUE);
}
void CToDoListWnd::OnUpdateEditInctaskpriority(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnEditFlagtask()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.SetSelectedTaskFlag(!tdc.IsSelectedTaskFlagged());
}
void CToDoListWnd::OnUpdateEditFlagtask(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
pCmdUI->SetCheck(tdc.IsSelectedTaskFlagged() ? 1 : 0);
}
void CToDoListWnd::OnEditGotoDependency()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.GotoSelectedTaskDependency();
}
void CToDoListWnd::OnUpdateEditGotoDependency(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
CStringArray aDepends;
pCmdUI->Enable(tdc.GetSelectedTaskDependencies(aDepends) > 0);
}
void CToDoListWnd::OnEditRecurrence()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.EditSelectedTaskRecurrence();
}
void CToDoListWnd::OnUpdateEditRecurrence(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnFileOpenarchive()
{
CString sArchivePath = m_mgrToDoCtrls.GetArchivePath(GetSelToDoCtrl());
BOOL bArchiveExists = FileMisc::FileExists(sArchivePath);
if (bArchiveExists)
OpenTaskList(sArchivePath, FALSE);
}
void CToDoListWnd::OnUpdateFileOpenarchive(CCmdUI* pCmdUI)
{
CString sArchivePath = m_mgrToDoCtrls.GetArchivePath(GetSelToDoCtrl());
BOOL bArchiveExists = FileMisc::FileExists(sArchivePath);
pCmdUI->Enable(bArchiveExists);
}
void CToDoListWnd::PrepareFilter(FTDCFILTER& filter) const
{
if (filter.nShow != FS_CUSTOM)
{
// handle title filter option
switch (Prefs().GetTitleFilterOption())
{
case PUIP_MATCHONTITLECOMMENTS:
filter.nTitleOption = FT_FILTERONTITLECOMMENTS;
break;
case PUIP_MATCHONANYTEXT:
filter.nTitleOption = FT_FILTERONANYTEXT;
break;
case PUIP_MATCHONTITLE:
default:
filter.nTitleOption = FT_FILTERONTITLEONLY;
break;
}
}
}
void CToDoListWnd::OnViewShowfilterbar()
{
m_bShowFilterBar = !m_bShowFilterBar;
m_filterBar.ShowWindow(m_bShowFilterBar ? SW_SHOW : SW_HIDE);
Resize();
Invalidate(TRUE);
}
void CToDoListWnd::OnUpdateViewShowfilterbar(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowFilterBar ? 1 : 0);
}
void CToDoListWnd::OnViewClearfilter()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (tdc.HasFilter() || tdc.HasCustomFilter())
{
tdc.ClearFilter();
// reenable the filter controls
//m_filterBar.RemoveCustomFilters();
m_filterBar.RefreshFilterControls(tdc);
RefreshFilterControls();
UpdateStatusbar();
}
}
void CToDoListWnd::OnUpdateViewClearfilter(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.HasFilter() || tdc.HasCustomFilter());
}
void CToDoListWnd::OnViewTogglefilter()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.ToggleFilter();
RefreshFilterControls();
UpdateStatusbar();
// reenable the filter controls
m_filterBar.SetCustomFilter(tdc.HasCustomFilter(), tdc.GetCustomFilterName());
}
void CToDoListWnd::OnUpdateViewTogglefilter(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.HasFilter() || tdc.HasLastFilter() || tdc.HasCustomFilter());
}
LRESULT CToDoListWnd::OnSelchangeFilter(WPARAM wp, LPARAM lp)
{
FTDCFILTER* pFilter = (FTDCFILTER*)wp;
CString sCustom((LPCTSTR)lp);
ASSERT(pFilter);
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (pFilter->nShow == FS_CUSTOM)
{
SEARCHPARAMS params;
if (sCustom.IsEmpty())
m_findDlg.GetSearchParams(params);
else
m_findDlg.GetSearchParams(sCustom, params);
tdc.SetCustomFilter(params, sCustom);
}
else
{
PrepareFilter(*pFilter);
tdc.SetFilter(*pFilter);
}
m_filterBar.RefreshFilterControls(tdc);
UpdateStatusbar();
return 0L;
}
void CToDoListWnd::OnViewFilter()
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
CStringArray aCustom;
m_filterBar.GetCustomFilters(aCustom);
CTDLFilterDlg dialog(Prefs().GetMultiSelFilters());
if (dialog.DoModal(aCustom, tdc, m_aPriorityColors) == IDOK)
{
FTDCFILTER filter;
CString sCustom;
dialog.GetFilter(filter, sCustom);
OnSelchangeFilter((WPARAM)&filter, (LPARAM)(LPCTSTR)sCustom);
}
}
void CToDoListWnd::OnUpdateViewFilter(CCmdUI* pCmdUI)
{
UNREFERENCED_PARAMETER(pCmdUI);
// pCmdUI->Enable(!m_bShowFilterBar);
}
void CToDoListWnd::OnViewRefreshfilter()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
FTDCFILTER filterTDC, filter;
tdc.GetFilter(filterTDC);
m_filterBar.GetFilter(filter);
PrepareFilter(filter);
// if the filter has changed then set the new one else
// refresh the current one
if (filterTDC == filter)
tdc.RefreshFilter();
else
{
tdc.SetFilter(filter);
if (Prefs().GetExpandTasksOnLoad())
tdc.ExpandTasks(TDCEC_ALL);
}
UpdateStatusbar();
}
void CToDoListWnd::OnUpdateViewRefreshfilter(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
FTDCFILTER filterTDC, filter;
tdc.GetFilter(filterTDC);
m_filterBar.GetFilter(filter);
pCmdUI->Enable(tdc.HasFilter() || (filter != filterTDC) || tdc.HasCustomFilter());
}
void CToDoListWnd::OnTabctrlPreferences()
{
DoPreferences(PREFPAGE_UI);
}
void CToDoListWnd::OnTasklistSelectColumns()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSel = GetSelToDoCtrl();
CTDCColumnIDArray aColumns, aDefault;
tdc.GetVisibleColumns(aColumns);
Prefs().GetDefaultColumns(aDefault);
CTDLColumnSelectionDlg dialog(aColumns, aDefault, TRUE);
if (dialog.DoModal() == IDOK)
{
dialog.GetVisibleColumns(aColumns);
tdc.SetVisibleColumns(aColumns);
m_filterBar.SetVisibleFilters(aColumns);
if (dialog.GetApplyActiveTasklist())
m_mgrToDoCtrls.SetHasOwnColumns(nSel, TRUE);
else
{
// update preferences
ASSERT(m_pPrefs);
m_pPrefs->SetDefaultColumns(aColumns);
// and flag other tasklists as requiring updates
m_mgrToDoCtrls.SetAllNeedPreferenceUpdate(TRUE, nSel);
m_mgrToDoCtrls.SetHasOwnColumns(nSel, FALSE);
}
// reload the menu if we dynamically alter it
if (Prefs().GetShowEditMenuAsColumns())
LoadMenubar();
Resize();
}
}
void CToDoListWnd::OnViewProjectname()
{
m_bShowProjectName = !m_bShowProjectName;
// mark all tasklists as needing update
m_mgrToDoCtrls.SetAllNeedPreferenceUpdate(TRUE);
// then update active tasklist
GetToDoCtrl().SetStyle(TDCS_SHOWPROJECTNAME, m_bShowProjectName);
m_mgrToDoCtrls.SetNeedsPreferenceUpdate(GetSelToDoCtrl(), FALSE);
}
void CToDoListWnd::OnUpdateViewProjectname(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowProjectName ? 1 : 0);
}
void CToDoListWnd::OnEditOffsetdates()
{
COffsetDatesDlg dialog;
if (dialog.DoModal() == IDOK)
{
ODD_UNITS nUnits;
int nAmount = dialog.GetOffsetAmount(nUnits);
if (!nAmount)
return;
DWORD dwWhat = dialog.GetOffsetWhat();
BOOL bSubtasks = dialog.GetOffsetSubtasks();
// translate units
int nTDCUnits = (nUnits == ODDU_DAYS) ? TDITU_DAYS :
((nUnits == ODDU_WEEKS) ? TDITU_WEEKS : TDITU_MONTHS);
// do the offsets
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (dwWhat & ODD_STARTDATE)
tdc.OffsetSelectedTaskDate(TDCD_START, nAmount, nTDCUnits, bSubtasks);
if (dwWhat & ODD_DUEDATE)
tdc.OffsetSelectedTaskDate(TDCD_DUE, nAmount, nTDCUnits, bSubtasks);
if (dwWhat & ODD_DONEDATE)
tdc.OffsetSelectedTaskDate(TDCD_DONE, nAmount, nTDCUnits, bSubtasks);
}
}
void CToDoListWnd::OnUpdateEditOffsetdates(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
void CToDoListWnd::OnPrintpreview()
{
DoPrint(TRUE);
}
void CToDoListWnd::OnShowTimelogfile()
{
CString sLogPath = GetToDoCtrl().GetSelectedTaskTimeLogPath();
if (!sLogPath.IsEmpty())
FileMisc::Run(*this, sLogPath, NULL, SW_HIDE);
}
void CToDoListWnd::OnUpdateShowTimelogfile(CCmdUI* pCmdUI)
{
const CPreferencesDlg& userPrefs = Prefs();
int nTasks = GetToDoCtrl().GetSelectedCount();
BOOL bEnable = FALSE;
if (userPrefs.GetLogTimeTracking() &&
(nTasks == 1 || !userPrefs.GetLogTaskTimeSeparately()))
{
CString sLogPath = GetToDoCtrl().GetSelectedTaskTimeLogPath();
bEnable = FileMisc::FileExists(sLogPath);
}
pCmdUI->Enable(bEnable);
}
void CToDoListWnd::OnAddtimetologfile()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
DWORD dwTaskID = tdc.GetSelectedTaskID();
CString sTitle = tdc.GetSelectedTaskTitle();
CTDLAddLoggedTimeDlg dialog(dwTaskID, sTitle);
if (dialog.DoModal() == IDOK)
tdc.AddTimeToTaskLogFile(dwTaskID, dialog.GetLoggedTime(), dialog.GetWhen(), dialog.GetAddToTimeSpent());
}
void CToDoListWnd::OnUpdateAddtimetologfile(CCmdUI* pCmdUI)
{
BOOL bEnable = (Prefs().GetLogTimeTracking() && GetToDoCtrl().GetSelectedCount() == 1);
pCmdUI->Enable(bEnable);
}
LRESULT CToDoListWnd::OnToDoCtrlDoLengthyOperation(WPARAM wParam, LPARAM lParam)
{
if (wParam) // start op
{
m_sbProgress.BeginProgress(m_statusBar, (LPCTSTR)lParam);
}
else // end op
{
m_sbProgress.EndProgress();
}
return 0L;
}
BOOL CToDoListWnd::DoTaskLink(const CString& sPath, DWORD dwTaskID)
{
// handle no file path => active tasklist
if (sPath.IsEmpty())
{
ASSERT(dwTaskID);
GetToDoCtrl().SelectTask(dwTaskID);
return TRUE; // handled regardless of result
}
else
{
// build the full path to the file
// from the folder of the active tasklist
int nSelTDC = GetSelToDoCtrl();
CString sActivePath = m_mgrToDoCtrls.GetFilePath(nSelTDC);
CString sFolder = FileMisc::GetFolderFromFilePath(sActivePath);
CString sFile(sPath);
FileMisc::MakeFullPath(sFile, sFolder);
// do we have this tasklist ?
int nTDC = m_mgrToDoCtrls.FindToDoCtrl(sFile);
if (nTDC != -1)
{
if (SelectToDoCtrl(nTDC, (nTDC != nSelTDC)) && dwTaskID)
GetToDoCtrl().SelectTask(dwTaskID);
return TRUE; // handled regardless of result
}
else if (!Prefs().GetMultiInstance())
{
TDC_FILE nRet = OpenTaskList(sFile);
if (nRet == TDCO_SUCCESS)
{
if (dwTaskID)
GetToDoCtrl().SelectTask(dwTaskID);
}
else
HandleLoadTasklistError(nRet, sFile);
return TRUE; // handled regardless of result
}
}
// not handled
return FALSE;
}
LRESULT CToDoListWnd::OnToDoCtrlDoTaskLink(WPARAM wParam, LPARAM lParam)
{
DWORD dwTaskID = wParam;
CString sFile((LPCTSTR)lParam);
// can we handle it ?
if (DoTaskLink(sFile, dwTaskID))
return TRUE;
// Pass to our app startup code to look
// for another instance who can handle it
CString sCommandline;
sCommandline.Format(_T("%s -l \"%s?%ld\""),
FileMisc::GetAppFileName(),
sFile,
dwTaskID);
return FileMisc::Run(*this, sCommandline);
}
LRESULT CToDoListWnd::OnTodoCtrlFailedLink(WPARAM /*wParam*/, LPARAM lParam)
{
LPCTSTR szLink = (LPCTSTR)lParam;
// if it's an Outlook link then prompt to install
// the Outlook URL handler
if (COutlookHelper::IsOutlookUrl(szLink))
{
// if the handler installs properly give the url another go
if (COutlookHelper::QueryInstallUrlHandler(IDS_QUERYINSTALLOUTLOOKHANDLER))
FileMisc::Run(*this, szLink);
return TRUE; // we handled it regardless
}
else // see if it's a task link
{
CString sFile;
DWORD dwTaskID = 0;
CFilteredToDoCtrl::ParseTaskLink(szLink, dwTaskID, sFile);
if (DoTaskLink(sFile, dwTaskID))
return TRUE; // we handled it
}
// all else
AfxMessageBox(IDS_COMMENTSGOTOERRMSG);
return 0L;
}
LRESULT CToDoListWnd::OnToDoCtrlTaskIsDone(WPARAM wParam, LPARAM lParam)
{
ASSERT (lParam);
CString sFile((LPCTSTR)lParam);
if (!sFile.IsEmpty())
{
// build the full path to the file
if (::PathIsRelative(sFile))
{
// append it to the folder containing the active tasklist
CString sPathName = m_mgrToDoCtrls.GetFilePath(GetSelToDoCtrl());
CString sDrive, sFolder;
FileMisc::SplitPath(sPathName, &sDrive, &sFolder);
FileMisc::MakePath(sFile, sDrive, sFolder, sFile);
}
// else its a full path already
int nTDC = m_mgrToDoCtrls.FindToDoCtrl(sFile);
if (nTDC != -1) // already loaded
return GetToDoCtrl(nTDC).IsTaskDone(wParam);
else
{
// we must load the tasklist ourselves
CTaskFile tasks;
if (tasks.Load(sFile))
{
HTASKITEM ht = tasks.FindTask(wParam);
return ht ? tasks.IsTaskDone(ht) : FALSE;
}
}
}
return 0L;
}
LRESULT CToDoListWnd::OnPowerBroadcast(WPARAM wp, LPARAM /*lp*/)
{
const CPreferencesDlg& userPrefs = Prefs();
switch (wp)
{
case PBT_APMSUSPEND:
case PBT_APMSTANDBY:
case PBT_APMQUERYSUSPEND:
case PBT_APMQUERYSTANDBY:
// Terminate all timers
SetTimer(TIMER_DUEITEMS, FALSE);
SetTimer(TIMER_READONLYSTATUS, FALSE);
SetTimer(TIMER_TIMESTAMPCHANGE, FALSE);
SetTimer(TIMER_CHECKOUTSTATUS, FALSE);
SetTimer(TIMER_AUTOSAVE, FALSE);
SetTimer(TIMER_TIMETRACKING, FALSE);
break;
case PBT_APMQUERYSUSPENDFAILED:
case PBT_APMQUERYSTANDBYFAILED:
case PBT_APMRESUMECRITICAL:
case PBT_APMRESUMESUSPEND:
case PBT_APMRESUMESTANDBY:
// reset time tracking as required
if (!userPrefs.GetTrackHibernated())
{
int nCtrl = GetTDCCount();
while (nCtrl--)
GetToDoCtrl(nCtrl).ResetTimeTracking();
}
// restart timers
SetTimer(TIMER_DUEITEMS, TRUE);
SetTimer(TIMER_READONLYSTATUS, userPrefs.GetReadonlyReloadOption() != RO_NO);
SetTimer(TIMER_TIMESTAMPCHANGE, userPrefs.GetTimestampReloadOption() != RO_NO);
SetTimer(TIMER_AUTOSAVE, userPrefs.GetAutoSaveFrequency());
SetTimer(TIMER_CHECKOUTSTATUS, userPrefs.GetCheckoutOnCheckin() || userPrefs.GetAutoCheckinFrequency());
SetTimer(TIMER_TIMETRACKING, TRUE);
// check for updates
if (Prefs().GetAutoCheckForUpdates())
CheckForUpdates(FALSE);
break;
}
return TRUE; // allow
}
LRESULT CToDoListWnd::OnGetFont(WPARAM /*wp*/, LPARAM /*lp*/)
{
return (LRESULT)m_fontMain.GetSafeHandle();
}
void CToDoListWnd::OnViewStatusBar()
{
m_bShowStatusBar = !m_bShowStatusBar;
m_statusBar.ShowWindow(m_bShowStatusBar ? SW_SHOW : SW_HIDE);
SendMessage(WM_SIZE, SIZE_RESTORED, 0L);
//Resize();
if (m_bShowStatusBar)
UpdateStatusbar();
else
UpdateCaption();
}
void CToDoListWnd::OnUpdateViewStatusBar(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowStatusBar ? 1 : 0) ;
}
BOOL CToDoListWnd::OnQueryOpen()
{
if (CFrameWnd::OnQueryOpen())
{
// fail if the active tasklist is encrypted because we have to verify the password
// and we're not allowed to display a dialog in this message handler
if (!m_bQueryOpenAllow && GetToDoCtrl().IsEncrypted())
{
PostMessage(WM_TDL_RESTORE);
return FALSE;
}
// all others
return TRUE;
}
return FALSE;
}
LRESULT CToDoListWnd::OnToDoListRestore(WPARAM /*wp*/, LPARAM /*lp*/)
{
ASSERT (IsIconic() && GetToDoCtrl().IsEncrypted()); // sanity check
if (IsIconic())
{
if (VerifyToDoCtrlPassword())
{
CAutoFlag af(m_bQueryOpenAllow, TRUE);
ShowWindow(SW_RESTORE);
}
}
return 0L;
}
void CToDoListWnd::OnCopyTaskasLink()
{
CopySelectedTasksToClipboard(TDCTC_ASLINK);
}
void CToDoListWnd::OnUpdateCopyTaskasLink(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() == 1);
}
void CToDoListWnd::OnCopyTaskasDependency()
{
CopySelectedTasksToClipboard(TDCTC_ASDEPENDS);
}
void CToDoListWnd::OnUpdateCopyTaskasDependency(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() == 1);
}
void CToDoListWnd::OnCopyTaskasLinkFull()
{
CopySelectedTasksToClipboard(TDCTC_ASLINKFULL);
}
void CToDoListWnd::OnUpdateCopyTaskasLinkFull(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() == 1);
}
void CToDoListWnd::OnCopyTaskasDependencyFull()
{
CopySelectedTasksToClipboard(TDCTC_ASDEPENDSFULL);
}
void CToDoListWnd::OnUpdateCopyTaskasDependencyFull(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() == 1);
}
void CToDoListWnd::OnCopyTaskasPath()
{
CopySelectedTasksToClipboard(TDCTC_ASPATH);
}
void CToDoListWnd::OnUpdateCopyTaskasPath(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetSelectedCount() == 1);
}
BOOL CToDoListWnd::PreCreateWindow(CREATESTRUCT& cs)
{
if (CFrameWnd::PreCreateWindow(cs))
{
cs.dwExStyle &= ~WS_EX_CLIENTEDGE;
// Check if our class is already defined
LPCTSTR pszClassName = _T("ToDoListFrame");
WNDCLASS wndcls;
if (!::GetClassInfo(AfxGetInstanceHandle(), pszClassName, &wndcls))
{
// Get the current requested window class
VERIFY(GetClassInfo(AfxGetInstanceHandle(), cs.lpszClass, &wndcls));
// We want to register this info with our name
wndcls.lpszClassName = pszClassName;
// Need to preset the icon otherwise the function GetIconWndClass
// calling us will overwrite our class.
wndcls.hIcon = GraphicsMisc::LoadIcon(IDR_MAINFRAME);
// Register our class now and check the outcome
if (!::RegisterClass(&wndcls))
{
ASSERT(0);
return FALSE;
}
}
// Now use our class
cs.lpszClass = pszClassName;
return TRUE;
}
// else
return FALSE;
}
void CToDoListWnd::OnWindowPosChanging(WINDOWPOS FAR* lpwndpos)
{
CFrameWnd::OnWindowPosChanging(lpwndpos);
}
void CToDoListWnd::OnToolsCheckforupdates()
{
CheckForUpdates(TRUE);
}
void CToDoListWnd::OnEditInsertdatetime()
{
DoInsertDateAndTime(TRUE, TRUE);
}
void CToDoListWnd::OnEditInsertdate()
{
DoInsertDateAndTime(TRUE, FALSE);
}
void CToDoListWnd::OnEditInserttime()
{
DoInsertDateAndTime(FALSE, TRUE);
}
void CToDoListWnd::DoInsertDateAndTime(BOOL bDate, BOOL bTime)
{
COleDateTime date = COleDateTime::GetCurrentTime();
const CPreferencesDlg& userPrefs = Prefs();
CString sInsert;
if (bDate) // date only or date and time
{
DWORD dwFmt = (bTime ? DHFD_TIME : 0);
if (userPrefs.GetShowWeekdayInDates())
dwFmt |= DHFD_DOW;
if (userPrefs.GetDisplayDatesInISO())
dwFmt |= DHFD_ISO;
sInsert = CDateHelper::FormatDate(date, dwFmt);
}
else // time only
{
if (userPrefs.GetDisplayDatesInISO())
sInsert = CTimeHelper::FormatISOTime(date.GetHour(), date.GetMinute(), date.GetSecond(), TRUE);
else
sInsert = CTimeHelper::Format24HourTime(date.GetHour(), date.GetMinute(), date.GetSecond(), TRUE);
}
// add trailing space
sInsert += ' ';
GetToDoCtrl().PasteText(sInsert);
}
void CToDoListWnd::OnUpdateEditInsertdatetime(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanPasteText());
}
void CToDoListWnd::OnUpdateEditInserttime(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanPasteText());
}
void CToDoListWnd::OnUpdateEditInsertdate(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().CanPasteText());
}
void CToDoListWnd::OnSysColorChange()
{
CFrameWnd::OnSysColorChange();
InitMenuIconManager();
SetUITheme(m_sThemeFile);
}
void CToDoListWnd::UpdateSBPaneAndTooltip(UINT nIDPane, UINT nIDTextFormat, const CString& sValue, UINT nIDTooltip, TDC_COLUMN nTDCC)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
const CPreferencesDlg& userPrefs = Prefs();
CEnString sText, sTooltip;
if (!userPrefs.GetShowCtrlsAsColumns() || tdc.IsColumnShowing(nTDCC))
{
sText.Format(nIDTextFormat, sValue);
sTooltip.LoadString(nIDTooltip);
}
else
{
sText.Empty();
sTooltip.Empty();
}
int nPane = m_statusBar.CommandToIndex(nIDPane);
m_statusBar.SetPaneText(nPane, sText);
m_statusBar.SetPaneTooltipIndex(nPane, sTooltip);
}
void CToDoListWnd::UpdateStatusBarInfo(const CFilteredToDoCtrl& tdc, TDCSTATUSBARINFO& sbi) const
{
sbi.nSelCount = tdc.GetSelectedCount();
sbi.dwSelTaskID = tdc.GetSelectedTaskID();
sbi.dCost = tdc.CalcSelectedTaskCost();
const CPreferencesDlg& userPrefs = Prefs();
userPrefs.GetDefaultTimeEst(sbi.nTimeEstUnits);
sbi.dTimeEst = tdc.CalcSelectedTaskTimeEstimate(sbi.nTimeEstUnits);
userPrefs.GetDefaultTimeSpent(sbi.nTimeSpentUnits);
sbi.dTimeSpent = tdc.CalcSelectedTaskTimeSpent(sbi.nTimeSpentUnits);
}
void CToDoListWnd::OnUpdateSBSelectionCount(CCmdUI* /*pCmdUI*/)
{
if (GetTDCCount())
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
// keep track of previous information to avoid unnecessary processing
static TDCSTATUSBARINFO sbiPrev;
TDCSTATUSBARINFO sbi;
UpdateStatusBarInfo(tdc, sbi);
if (sbi == sbiPrev)
return;
sbiPrev = sbi;
// number of selected tasks
CEnString sText;
if (sbi.nSelCount == 1)
{
ASSERT(sbi.dwSelTaskID);
sText.Format(ID_SB_SELCOUNTONE, sbi.dwSelTaskID);
}
else
sText.Format(ID_SB_SELCOUNT, sbi.nSelCount);
m_statusBar.SetPaneText(m_statusBar.CommandToIndex(ID_SB_SELCOUNT), sText);
// times
const CPreferencesDlg& userPrefs = Prefs();
// estimate
if (userPrefs.GetUseHMSTimeFormat())
sText = CTimeHelper().FormatTimeHMS(sbi.dTimeEst, sbi.nTimeEstUnits);
else
sText = CTimeHelper().FormatTime(sbi.dTimeEst, sbi.nTimeEstUnits, 2);
UpdateSBPaneAndTooltip(ID_SB_SELTIMEEST, ID_SB_SELTIMEEST, sText, IDS_SB_SELTIMEEST_TIP, TDCC_TIMEEST);
// spent
if (userPrefs.GetUseHMSTimeFormat())
sText = CTimeHelper().FormatTimeHMS(sbi.dTimeSpent, sbi.nTimeSpentUnits);
else
sText = CTimeHelper().FormatTime(sbi.dTimeSpent, sbi.nTimeSpentUnits, 2);
UpdateSBPaneAndTooltip(ID_SB_SELTIMESPENT, ID_SB_SELTIMESPENT, sText, IDS_SB_SELTIMESPENT_TIP, TDCC_TIMESPENT);
// cost
sText = Misc::Format(sbi.dCost, 2);
UpdateSBPaneAndTooltip(ID_SB_SELCOST, ID_SB_SELCOST, sText, IDS_SB_SELCOST_TIP, TDCC_COST);
// set tray tip too
UpdateTooltip();
}
}
void CToDoListWnd::OnUpdateSBTaskCount(CCmdUI* /*pCmdUI*/)
{
if (GetTDCCount())
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
UINT nVisibleTasks;
UINT nTotalTasks = tdc.GetTaskCount(&nVisibleTasks);
CEnString sText;
sText.Format(IDS_SB_TASKCOUNT, nVisibleTasks, nTotalTasks);
int nIndex = m_statusBar.CommandToIndex(ID_SB_TASKCOUNT);
m_statusBar.SetPaneText(nIndex, sText);
m_statusBar.SetPaneTooltipIndex(nIndex, CEnString(IDS_SB_TASKCOUNT_TIP));
}
}
void CToDoListWnd::OnEditSelectall()
{
GetToDoCtrl().SelectAll();
}
void CToDoListWnd::OnUpdateEditSelectall(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetToDoCtrl().GetTaskCount());
}
void CToDoListWnd::OnCloseallbutthis()
{
int nThis = GetSelToDoCtrl();
int nCtrl = GetTDCCount();
// remove tasklists
while (nCtrl--)
{
if (nCtrl != nThis)
{
if (ConfirmSaveTaskList(nCtrl, TDLS_CLOSINGTASKLISTS) != TDCO_SUCCESS)
continue; // user cancelled
m_mgrToDoCtrls.RemoveToDoCtrl(nCtrl, TRUE);
}
}
}
void CToDoListWnd::OnUpdateCloseallbutthis(CCmdUI* pCmdUI)
{
pCmdUI->Enable(GetTDCCount() > 1);
}
void CToDoListWnd::DoSendTasks(BOOL bSelected)
{
CTDLSendTasksDlg dialog(bSelected, GetToDoCtrl().GetView());
if (dialog.DoModal() == IDOK)
{
// get tasks
CFilteredToDoCtrl& tdc = GetToDoCtrl();
CTaskFile tasks;
GetTasks(tdc, FALSE, FALSE, dialog.GetTaskSelection(), tasks, NULL);
// package them up
CString sAttachment, sBody;
TD_SENDAS nSendAs = dialog.GetSendAs();
switch (nSendAs)
{
case TDSA_TASKLIST:
{
CString sFilename, sExt;
FileMisc::SplitPath(tdc.GetFilePath(), NULL, NULL, &sFilename, &sExt);
sAttachment = FileMisc::GetTempFileName(sFilename, sExt);
if (!tasks.Save(sAttachment))
{
// TODO
return;
}
sBody.LoadString(IDS_TASKLISTATTACHED);
}
break;
case TDSA_BODYTEXT:
sBody = m_mgrImportExport.ExportTaskListToText(&tasks);
break;
}
// form subject
CString sSubject = tdc.GetFriendlyProjectName();
// recipients
CStringArray aTo;
tdc.GetSelectedTaskAllocTo(aTo);
CString sTo = Misc::FormatArray(aTo, _T(";"));
// prefix with task name if necessary
if (dialog.GetTaskSelection().GetWantSelectedTasks() && tdc.GetSelectedCount() == 1)
{
sSubject = tdc.GetSelectedTaskTitle() + _T(" - ") + sSubject;
}
CSendFileTo().SendMail(*this, sTo, sSubject, sBody, sAttachment);
}
}
void CToDoListWnd::OnSendTasks()
{
DoSendTasks(FALSE);
}
void CToDoListWnd::OnUpdateSendTasks(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.GetTaskCount());
}
void CToDoListWnd::OnSendSelectedTasks()
{
DoSendTasks(TRUE);
}
void CToDoListWnd::OnUpdateSendSelectedTasks(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.GetTaskCount());
}
void CToDoListWnd::OnEditUndo()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.UndoLastAction(TRUE);
UpdateStatusbar();
}
void CToDoListWnd::OnUpdateEditUndo(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanUndoLastAction(TRUE));
}
void CToDoListWnd::OnEditRedo()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.UndoLastAction(FALSE);
UpdateStatusbar();
}
void CToDoListWnd::OnUpdateEditRedo(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanUndoLastAction(FALSE));
}
void CToDoListWnd::OnViewCycleTaskViews()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.SetNextView();
tdc.SetFocusToTasks();
}
void CToDoListWnd::OnUpdateViewCycleTaskViews(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_nMaxState != TDCMS_MAXCOMMENTS);
}
void CToDoListWnd::OnViewToggleTreeandList()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
FTC_VIEW nView = tdc.GetView();
switch (nView)
{
case FTCV_TASKTREE:
nView = FTCV_TASKLIST;
break;
case FTCV_TASKLIST:
default:
nView = FTCV_TASKTREE;
break;
}
tdc.SetView(nView);
tdc.SetFocusToTasks();
}
void CToDoListWnd::OnUpdateViewToggleTreeandList(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_nMaxState != TDCMS_MAXCOMMENTS);
}
void CToDoListWnd::OnViewToggletasksandcomments()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.TasksHaveFocus())
tdc.SetFocusToTasks();
else
tdc.SetFocusToComments();
}
void CToDoListWnd::OnUpdateViewToggletasksandcomments(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_nMaxState == TDCMS_NORMAL ||
(m_nMaxState == TDCMS_MAXTASKLIST && Prefs().GetShowCommentsAlways()));
}
void CToDoListWnd::OnUpdateQuickFind(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_bShowToolbar);
}
void CToDoListWnd::OnQuickFind()
{
if (m_bShowToolbar)
m_cbQuickFind.SetFocus();
}
void CToDoListWnd::OnQuickFindNext()
{
if (!m_sQuickFind.IsEmpty())
{
if (!GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTNEXT))
GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTFIRST); // return to start
}
}
void CToDoListWnd::OnUpdateQuickFindNext(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!m_sQuickFind.IsEmpty());
}
LRESULT CToDoListWnd::OnQuickFindItemAdded(WPARAM /*wp*/, LPARAM /*lp*/)
{
// keep only the last 20 items
int nItem = m_cbQuickFind.GetCount();
while (nItem > 20)
{
nItem--;
m_cbQuickFind.DeleteString(nItem);
}
return 0L;
}
void CToDoListWnd::OnQuickFindPrev()
{
if (!m_sQuickFind.IsEmpty())
{
if (!GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTPREV))
GetToDoCtrl().SelectTask(m_sQuickFind, TDC_SELECTLAST); // return to end
}
}
void CToDoListWnd::OnUpdateQuickFindPrev(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!m_sQuickFind.IsEmpty());
}
void CToDoListWnd::OnMove(int x, int y)
{
CFrameWnd::OnMove(x, y);
}
void CToDoListWnd::OnEditSettaskicon()
{
GetToDoCtrl().EditSelectedTaskIcon();
}
void CToDoListWnd::OnUpdateEditSettaskicon(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly());
}
LRESULT CToDoListWnd::OnToDoCtrlReminder(WPARAM /*wp*/, LPARAM lp)
{
AF_NOREENTRANT_RET(0L) // macro helper
// ignore if we are showing a dialog
if (!IsWindowEnabled())
return 0L;
Show(FALSE);
CTDLShowReminderDlg dialog;
TDCREMINDER* pReminder = (TDCREMINDER*)lp;
int nRet = dialog.DoModal(*pReminder);
switch (nRet)
{
case IDSNOOZE:
{
double dNow = COleDateTime::GetCurrentTime();
if (pReminder->bRelative)
{
if (pReminder->nRelativeFromWhen == TDCR_DUEDATE)
{
// in case the user didn't handle the notification immediately we need
// to soak up any additional elapsed time in the snooze
COleDateTime dDue = pReminder->pTDC->GetTaskDate(pReminder->dwTaskID, TDCD_DUE);
pReminder->dDaysSnooze = (dNow - dDue + pReminder->dRelativeDaysLeadIn);
}
else // from start
{
// in case the user didn't handle the notification immediately we need
// to soak up any additional elapsed time in the snooze
COleDateTime dStart = pReminder->pTDC->GetTaskDate(pReminder->dwTaskID, TDCD_START);
pReminder->dDaysSnooze = (dNow - dStart + pReminder->dRelativeDaysLeadIn);
}
}
else // absolute
{
// in case the user didn't handle the notification immediately we need
// to soak up any additional elapsed time in the snooze
pReminder->dDaysSnooze = dNow - pReminder->dtAbsolute;
}
// then we add the user's snooze
pReminder->dDaysSnooze += dialog.GetSnoozeDays();
}
return 0L; // don't delete (default)
case IDGOTOTASK:
{
int nTDC = m_mgrToDoCtrls.FindToDoCtrl(pReminder->pTDC);
ASSERT(nTDC != -1);
SelectToDoCtrl(nTDC, TRUE);
GetToDoCtrl().SelectTask(pReminder->dwTaskID);
}
// fall thru
case IDCANCEL:
default:
// delete unless it's a recurring task in which case we
// disable it so that it can later be copied when the
// recurring task is completed
if (GetToDoCtrl().IsTaskRecurring(pReminder->dwTaskID))
{
pReminder->bEnabled = FALSE;
return 0L; // don't delete
}
// else
return 1L; // delete
}
}
LRESULT CToDoListWnd::OnToDoCtrlTaskHasReminder(WPARAM wParam, LPARAM lParam)
{
int nRem = m_reminders.FindReminder(wParam, (CFilteredToDoCtrl*)lParam, FALSE);
return (nRem != -1);
}
LRESULT CToDoListWnd::OnDoubleClkReminderCol(WPARAM /*wp*/, LPARAM /*lp*/)
{
OnEditSetReminder();
return 0L;
}
void CToDoListWnd::OnEditSetReminder()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
CTDLSetReminderDlg dialog;
CString sTitle = tdc.GetSelectedTaskTitle();
CDWordArray aTaskIDs;
int nNumSel = tdc.GetSelectedTaskIDs(aTaskIDs, TRUE);
if (nNumSel == 0)
return;
// get the first reminder as a reference
BOOL bNewReminder = TRUE;
TDCREMINDER rem;
for (int nTask = 0; nTask < nNumSel; nTask++)
{
DWORD dwTaskID = aTaskIDs[nTask];
int nRem = m_reminders.FindReminder(dwTaskID, &tdc);
if (nRem != -1)
{
m_reminders.GetReminder(nRem, rem);
bNewReminder = FALSE;
break;
}
}
// handle new task
if (bNewReminder)
{
rem.dwTaskID = aTaskIDs[0];
rem.pTDC = &tdc;
}
if (dialog.DoModal(rem, bNewReminder) == IDOK)
{
// apply reminder to selected tasks
for (int nTask = 0; nTask < nNumSel; nTask++)
{
rem.dwTaskID = aTaskIDs[nTask];
m_reminders.SetReminder(rem);
}
tdc.RedrawReminders();
}
}
void CToDoListWnd::OnUpdateEditSetReminder(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
BOOL bEnable = (tdc.GetSelectedCount() > 0) && !tdc.SelectedTasksAreAllDone();
pCmdUI->Enable(bEnable);
}
void CToDoListWnd::OnEditClearReminder()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
CDWordArray aTaskIDs;
int nTask = tdc.GetSelectedTaskIDs(aTaskIDs, TRUE);
while (nTask--)
{
DWORD dwTaskID = aTaskIDs[nTask];
m_reminders.RemoveReminder(dwTaskID, &tdc);
}
tdc.RedrawReminders();
}
void CToDoListWnd::OnUpdateEditClearReminder(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
BOOL bEnable = FALSE;
// check at least one selected item has a reminder
CDWordArray aTaskIDs;
int nTask = tdc.GetSelectedTaskIDs(aTaskIDs, TRUE);
while (nTask--)
{
DWORD dwTaskID = aTaskIDs[nTask];
if (m_reminders.FindReminder(dwTaskID, &tdc) != -1)
{
bEnable = TRUE;
break;
}
}
pCmdUI->Enable(bEnable);
}
void CToDoListWnd::OnEditCleartaskicon()
{
GetToDoCtrl().ClearSelectedTaskIcon();
}
void CToDoListWnd::OnUpdateEditCleartaskicon(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
int nSelCount = tdc.GetSelectedCount();
pCmdUI->Enable(nSelCount && !tdc.IsReadOnly() && tdc.SelectedTasksHaveIcons());
}
void CToDoListWnd::OnSortMulti()
{
TDSORTCOLUMNS sort;
CTDCColumnIDArray aColumns;
CTDCCustomAttribDefinitionArray aAttribDefs;
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.GetSortBy(sort);
tdc.GetVisibleColumns(aColumns);
tdc.GetCustomAttributeDefs(aAttribDefs);
CTDLMultiSortDlg dialog(sort, aColumns, aAttribDefs);
if (dialog.DoModal() == IDOK)
{
dialog.GetSortBy(sort);
tdc.MultiSort(sort);
}
}
void CToDoListWnd::OnUpdateSortMulti(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(GetToDoCtrl().IsMultiSorting() ? 1 : 0);
}
void CToDoListWnd::OnToolsRemovefromsourcecontrol()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (tdc.RemoveFromSourceControl())
{
int nCtrl = GetSelToDoCtrl();
m_mgrToDoCtrls.UpdateToDoCtrlReadOnlyUIState(nCtrl);
m_mgrToDoCtrls.UpdateTabItemText(nCtrl);
m_mgrToDoCtrls.SetModifiedStatus(nCtrl, FALSE);
m_mgrToDoCtrls.RefreshLastModified(nCtrl);
m_mgrToDoCtrls.RefreshReadOnlyStatus(nCtrl);
m_mgrToDoCtrls.RefreshPathType(nCtrl);
}
}
void CToDoListWnd::OnUpdateToolsRemovefromsourcecontrol(CCmdUI* pCmdUI)
{
int nCtrl = GetSelToDoCtrl();
BOOL bEnable = m_mgrToDoCtrls.IsSourceControlled(nCtrl);
// bEnable &= !Prefs().GetEnableSourceControl();
// bEnable &= m_mgrToDoCtrls.PathSupportsSourceControl(nCtrl);
if (bEnable)
{
// make sure no-one has the file checked out
CFilteredToDoCtrl& tdc = GetToDoCtrl();
bEnable &= !tdc.IsCheckedOut();
}
pCmdUI->Enable(bEnable);
}
void CToDoListWnd::OnViewShowTasklistTabbar()
{
m_bShowTasklistBar = !m_bShowTasklistBar;
Resize();
Invalidate(TRUE);
}
void CToDoListWnd::OnUpdateViewShowTasklistTabbar(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowTasklistBar ? 1 : 0);
}
void CToDoListWnd::OnViewShowTreeListTabbar()
{
m_bShowTreeListBar = !m_bShowTreeListBar;
GetToDoCtrl().SetStyle(TDCS_SHOWTREELISTBAR, m_bShowTreeListBar);
// refresh all the other tasklists
m_mgrToDoCtrls.SetAllNeedPreferenceUpdate(TRUE, GetSelToDoCtrl());
}
void CToDoListWnd::OnUpdateViewShowTreeListTabbar(CCmdUI* pCmdUI)
{
pCmdUI->SetCheck(m_bShowTreeListBar ? 1 : 0);
}
void CToDoListWnd::OnFileChangePassword()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly() && tdc.IsEncrypted() && VerifyToDoCtrlPassword())
{
tdc.EnableEncryption(FALSE); // clears the password
tdc.EnableEncryption(TRUE); // forces it to be re-got
}
}
void CToDoListWnd::OnUpdateFileChangePassword(CCmdUI* pCmdUI)
{
const CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(!tdc.IsReadOnly() && tdc.IsEncrypted());
}
void CToDoListWnd::OnTasklistCustomColumns()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
if (!tdc.IsReadOnly())
{
CTDLCustomAttributeDlg dialog(tdc, m_theme);
if (dialog.DoModal() == IDOK)
{
CTDCCustomAttribDefinitionArray aAttrib;
dialog.GetAttributes(aAttrib);
tdc.SetCustomAttributeDefs(aAttrib);
}
}
}
void CToDoListWnd::OnUpdateTasklistCustomcolumns(CCmdUI* pCmdUI)
{
pCmdUI->Enable(!GetToDoCtrl().IsReadOnly());
}
void CToDoListWnd::OnEditClearAttribute()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
TDC_ATTRIBUTE nAttrib = MapColumnToAttribute(m_nContextColumnID);
tdc.ClearSelectedTaskAttribute(nAttrib);
}
void CToDoListWnd::OnUpdateEditClearAttribute(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
TDC_ATTRIBUTE nAttrib = MapColumnToAttribute(m_nContextColumnID);
pCmdUI->Enable(tdc.CanClearSelectedTaskAttribute(nAttrib));
}
void CToDoListWnd::OnEditClearFocusedAttribute()
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
tdc.ClearSelectedTaskFocusedAttribute();
}
void CToDoListWnd::OnUpdateEditClearFocusedAttribute(CCmdUI* pCmdUI)
{
CFilteredToDoCtrl& tdc = GetToDoCtrl();
pCmdUI->Enable(tdc.CanClearSelectedTaskFocusedAttribute());
}