// MainFrm.cpp : implementation of the CMainFrame class // #include "stdafx.h" #include "CP_Main.h" #include "MainFrm.h" #include "afxole.h" #include "Misc.h" #include "CopyProperties.h" #include "InternetUpdate.h" #include ".\mainfrm.h" #include "HyperLink.h" #include "tinyxml\tinyxml.h" #include "Path.h" #include "DittoCopyBuffer.h" #include "HotKeys.h" #include "GlobalClips.h" #include "OptionsSheet.h" #include "DeleteClipData.h" #include "DatabaseUtilities.h" #ifdef _DEBUG #define new DEBUG_NEW #undef THIS_FILE static char THIS_FILE[] = __FILE__; #endif #define WM_ICON_NOTIFY WM_APP+10 #define MYWM_NOTIFYICON (WM_USER+1) #define WM_TRAYNOTIFY WM_USER + 100 IMPLEMENT_DYNAMIC(CMainFrame, CFrameWnd) BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd) //{{AFX_MSG_MAP(CMainFrame) ON_WM_CREATE() ON_COMMAND(ID_FIRST_OPTION, OnFirstOption) ON_COMMAND(ID_FIRST_EXIT, OnFirstExit) ON_WM_TIMER() ON_COMMAND(ID_FIRST_SHOWQUICKPASTE, OnFirstShowquickpaste) ON_COMMAND(ID_FIRST_TOGGLECONNECTCV, OnFirstToggleConnectCV) ON_UPDATE_COMMAND_UI(ID_FIRST_TOGGLECONNECTCV, OnUpdateFirstToggleConnectCV) ON_COMMAND(ID_FIRST_HELP, OnFirstHelp) //}}AFX_MSG_MAP ON_MESSAGE(WM_HOTKEY, OnHotKey) ON_MESSAGE(WM_SHOW_TRAY_ICON, OnShowTrayIcon) ON_MESSAGE(WM_CLIPBOARD_COPIED, OnClipboardCopied) ON_WM_CLOSE() ON_MESSAGE(WM_ADD_TO_DATABASE_FROM_SOCKET, OnAddToDatabaseFromSocket) ON_MESSAGE(WM_SEND_RECIEVE_ERROR, OnErrorOnSendRecieve) ON_COMMAND(ID_FIRST_IMPORT, OnFirstImport) ON_MESSAGE(WM_EDIT_WND_CLOSING, OnEditWndClose) ON_WM_DESTROY() ON_COMMAND(ID_FIRST_NEWCLIP, OnFirstNewclip) ON_MESSAGE(WM_SET_CONNECTED, OnSetConnected) ON_MESSAGE(WM_OPEN_CLOSE_WINDWOW, OnOpenCloseWindow) ON_MESSAGE(WM_LOAD_ClIP_ON_CLIPBOARD, OnLoadClipOnClipboard) ON_COMMAND(ID_FIRST_GLOBALHOTKEYS, &CMainFrame::OnFirstGlobalhotkeys) ON_MESSAGE(WM_GLOBAL_CLIPS_CLOSED, OnGlobalClipsClosed) ON_MESSAGE(WM_OPTIONS_CLOSED, OnOptionsClosed) ON_MESSAGE(WM_SHOW_OPTIONS, OnShowOptions) ON_COMMAND(ID_FIRST_DELETECLIPDATA, &CMainFrame::OnFirstDeleteclipdata) ON_MESSAGE(WM_DELETE_CLIPS_CLOSED, OnDeleteClipDataClosed) ON_COMMAND(ID_FIRST_SAVECURRENTCLIPBOARD, &CMainFrame::OnFirstSavecurrentclipboard) ON_MESSAGE(WM_SAVE_CLIPBOARD, &CMainFrame::OnSaveClipboardMessage) ON_MESSAGE(WM_READD_TASKBAR_ICON, OnReAddTaskBarIcon) ON_MESSAGE(WM_REOPEN_DATABASE, &CMainFrame::OnReOpenDatabase) ON_MESSAGE(WM_SHOW_MSG_WINDOW, &CMainFrame::OnShowMsgWindow) ON_MESSAGE(WM_SHOW_DITTO_GROUP, &CMainFrame::OnShowDittoGroup) ON_COMMAND(ID_FIRST_FIXUPSTICKYCLIPORDER, &CMainFrame::OnFirstFixupstickycliporder) ON_MESSAGE(WM_DISPLAYCHANGE, &CMainFrame::OnResolutionChange) ON_MESSAGE(WM_TRAYNOTIFY, &CMainFrame::OnTrayNotification) ON_MESSAGE(WM_PLAIN_TEXT_PASTE, &CMainFrame::OnPlainTextPaste) END_MESSAGE_MAP() static UINT indicators[] = { ID_SEPARATOR, // status line indicator ID_INDICATOR_CAPS, ID_INDICATOR_NUM, ID_INDICATOR_SCRL, }; ///////////////////////////////////////////////////////////////////////////// // CMainFrame construction/destruction CMainFrame::CMainFrame() { m_pEditFrameWnd = NULL; m_keyStateModifiers = 0; m_startKeyStateTime = 0; m_bMovedSelectionMoveKeyState = false; m_keyModifiersTimerCount = 0; m_pGlobalClips = NULL; m_pOptions = NULL; m_pDeleteClips = NULL; m_doubleClickGroupId = -1; m_doubleClickGroupStartTime = 0; } CMainFrame::~CMainFrame() { CGetSetOptions::SetMainHWND(0); } int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct) { if(CFrameWnd::OnCreate(lpCreateStruct) == - 1) { return - 1; } m_PowerManager.Start(m_hWnd); ////Center the main window so message boxes are in the center CRect rcScreen = DefaultMonitorRect(); CPoint cpCenter = rcScreen.CenterPoint(); MoveWindow(cpCenter.x, cpCenter.x, 1, 1); m_startupScreenWidth = GetScreenWidth(); m_startupScreenHeight = GetScreenHeight(); //Then set the main window to transparent so it's never shown //if it is shown then only the task tray icon //m_Transparency.SetTransparent(m_hWnd, 0, true); SetWindowText(_T("")); Log(_T("Setting polling timer to track focus")); SetTimer(ACTIVE_WINDOW_TIMER, g_Opt.FocusWndTimerTimeout(), 0); SetTimer(READ_RANDOM_DB_FILE, g_Opt.ReadRandomFileInterval() * 1000, 0); SetWindowText(_T("Ditto")); m_trayIcon.Create(this, IDR_MENU, _T("Ditto"), CTrayNotifyIcon::LoadIcon(IDR_MAINFRAME), WM_TRAYNOTIFY, 0, 1); m_trayIcon.SetDefaultMenuItem(ID_FIRST_SHOWQUICKPASTE, FALSE); m_trayIcon.MinimiseToTray(this); if (CGetSetOptions::GetShowStartupMessage()) { CString msg = theApp.m_Language.GetString(_T("StartupMsg"), _T("Ditto is running minimized, Ditto can be opened by hot keys or by clicking the task tray icon")); m_trayIcon.SetBalloonDetails(msg, _T("Ditto"), CTrayNotifyIcon::BalloonStyle::Info, CGetSetOptions::GetBalloonTimeout()); } theApp.m_Language.UpdateTrayIconRightClickMenu(&m_trayIcon.GetMenu()); //Only if in release #ifndef _DEBUG { //If not showing the icon show it for 40 seconds so they can get to the option //in case they can't remember the hot keys or something like that if(!(CGetSetOptions::GetShowIconInSysTray())) { SetTimer(HIDE_ICON_TIMER, 40000, 0); } } #endif //SetTimer(CLOSE_WINDOW_TIMER, ONE_HOUR*24, 0); SetTimer(REMOVE_OLD_TEMP_FILES, ONE_HOUR * 6, 0); SetTimer(REMOVE_OLD_ENTRIES_TIMER, ONE_MINUTE*15, 0); //found on some computers GetTickCount gettickcount returns a smaller value than other, can't explain //check here to see if we need to make an adjustment IdleSeconds(); m_ulCopyGap = CGetSetOptions::GetCopyGap(); theApp.AfterMainCreate(); m_thread.Start(this); return 0; } LRESULT CMainFrame::OnPlainTextPaste(WPARAM wParam, LPARAM lParam) { DoTextOnlyPaste(); return 1; } LRESULT CMainFrame::OnTrayNotification(WPARAM wParam, LPARAM lParam) { if (WM_MOUSEFIRST <= LOWORD(lParam) && LOWORD(lParam) <= WM_MOUSELAST) { theApp.m_activeWnd.TrackActiveWnd(true); } //click on balloon if (lParam == 0x405) { SetTimer(DELAYED_SHOW_DITTO_TIMER, 100, NULL); } m_trayIcon.OnTrayNotification(wParam, lParam); return 0L; } BOOL CMainFrame::PreCreateWindow(CREATESTRUCT &cs) { if(cs.hMenu != NULL) { ::DestroyMenu(cs.hMenu); // delete menu if loaded cs.hMenu = NULL; // no menu for this window } if(!CFrameWnd::PreCreateWindow(cs)) { return FALSE; } WNDCLASS wc; wc.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW; wc.lpfnWndProc = AfxWndProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = AfxGetInstanceHandle(); wc.hIcon = NULL; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH); wc.lpszMenuName = NULL; wc.lpszClassName = _T("Ditto"); // Create the QPaste window class if(!AfxRegisterClass(&wc)) { return FALSE; } cs.lpszClass = wc.lpszClassName; return TRUE; } ///////////////////////////////////////////////////////////////////////////// // CMainFrame diagnostics #ifdef _DEBUG void CMainFrame::AssertValid()const { CFrameWnd::AssertValid(); } void CMainFrame::Dump(CDumpContext &dc)const { CFrameWnd::Dump(dc); } #endif //_DEBUG ///////////////////////////////////////////////////////////////////////////// // CMainFrame message handlers void CMainFrame::OnFirstExit() { this->SendMessage(WM_CLOSE, 0, 0); } LRESULT CMainFrame::OnHotKey(WPARAM wParam, LPARAM lParam) { if(theApp.m_pDittoHotKey && wParam == theApp.m_pDittoHotKey->m_Atom || theApp.m_pDittoHotKey2 && wParam == theApp.m_pDittoHotKey2->m_Atom || theApp.m_pDittoHotKey3 && wParam == theApp.m_pDittoHotKey3->m_Atom) { //If they still have the shift/ctrl keys down if(m_keyStateModifiers != 0 && m_quickPaste.IsWindowVisibleEx()) { Log(_T("On Show Ditto HotKey, key state modifiers are still down, moving selection")); if(m_bMovedSelectionMoveKeyState == false) { Log(_T("Setting flag m_bMovedSelectionMoveKeyState to true, will paste when modifer keys are up")); } m_quickPaste.MoveSelection(true); m_bMovedSelectionMoveKeyState = true; } else if(g_Opt.m_HideDittoOnHotKeyIfAlreadyShown && m_quickPaste.IsWindowTopLevel() && g_Opt.GetShowPersistent() == FALSE) { Log(_T("On Show Ditto HotKey, window is alread visible, hiding window")); m_quickPaste.HideQPasteWnd(); } else { Log(_T("On Show Ditto HotKey, showing window")); StartKeyModifyerTimer(); ShowQPasteWithActiveWindowCheck(); } //KillTimer(CLOSE_WINDOW_TIMER); //SetTimer(CLOSE_WINDOW_TIMER, ONE_HOUR *24, 0); } else if(theApp.m_pPosOne && wParam == theApp.m_pPosOne->m_Atom) { Log(_T("Pos 1 hot key")); DoFirstTenPositionsPaste(0); } else if(theApp.m_pPosTwo && wParam == theApp.m_pPosTwo->m_Atom) { Log(_T("Pos 2 hot key")); DoFirstTenPositionsPaste(1); } else if(theApp.m_pPosThree && wParam == theApp.m_pPosThree->m_Atom) { Log(_T("Pos 3 hot key")); DoFirstTenPositionsPaste(2); } else if(theApp.m_pPosFour && wParam == theApp.m_pPosFour->m_Atom) { Log(_T("Pos 4 hot key")); DoFirstTenPositionsPaste(3); } else if(theApp.m_pPosFive && wParam == theApp.m_pPosFive->m_Atom) { Log(_T("Pos 5 hot key")); DoFirstTenPositionsPaste(4); } else if(theApp.m_pPosSix && wParam == theApp.m_pPosSix->m_Atom) { Log(_T("Pos 6 hot key")); DoFirstTenPositionsPaste(5); } else if(theApp.m_pPosSeven && wParam == theApp.m_pPosSeven->m_Atom) { Log(_T("Pos 7 hot key")); DoFirstTenPositionsPaste(6); } else if(theApp.m_pPosEight && wParam == theApp.m_pPosEight->m_Atom) { Log(_T("Pos 8 hot key")); DoFirstTenPositionsPaste(7); } else if(theApp.m_pPosNine && wParam == theApp.m_pPosNine->m_Atom) { Log(_T("Pos 9 hot key")); DoFirstTenPositionsPaste(8); } else if(theApp.m_pPosTen && wParam == theApp.m_pPosTen->m_Atom) { Log(_T("Pos 10 hot key")); DoFirstTenPositionsPaste(9); } else if(theApp.m_pCopyBuffer1 && wParam == theApp.m_pCopyBuffer1->m_Atom) { Log(_T("Copy buffer 1 hot key")); theApp.m_CopyBuffer.StartCopy(0); } else if(theApp.m_pPasteBuffer1 && wParam == theApp.m_pPasteBuffer1->m_Atom) { Log(_T("Paste buffer 1 hot key")); theApp.m_CopyBuffer.PastCopyBuffer(0); } else if(theApp.m_pCutBuffer1 && wParam == theApp.m_pCutBuffer1->m_Atom) { Log(_T("Cut buffer 1 hot key")); theApp.m_CopyBuffer.StartCopy(0, true); } else if(theApp.m_pCopyBuffer2 && wParam == theApp.m_pCopyBuffer2->m_Atom) { Log(_T("Copy buffer 2 hot key")); theApp.m_CopyBuffer.StartCopy(1); } else if(theApp.m_pPasteBuffer2 && wParam == theApp.m_pPasteBuffer2->m_Atom) { Log(_T("Paste buffer 2 hot key")); theApp.m_CopyBuffer.PastCopyBuffer(1); } else if(theApp.m_pCutBuffer2 && wParam == theApp.m_pCutBuffer2->m_Atom) { Log(_T("Cut buffer 2 hot key")); theApp.m_CopyBuffer.StartCopy(1, true); } else if(theApp.m_pCopyBuffer3 && wParam == theApp.m_pCopyBuffer3->m_Atom) { Log(_T("Copy buffer 3 hot key")); theApp.m_CopyBuffer.StartCopy(2); } else if(theApp.m_pPasteBuffer3 && wParam == theApp.m_pPasteBuffer3->m_Atom) { Log(_T("Paste buffer 3 hot key")); theApp.m_CopyBuffer.PastCopyBuffer(2); } else if(theApp.m_pCutBuffer3 && wParam == theApp.m_pCutBuffer3->m_Atom) { Log(_T("Cut buffer 3 hot key")); theApp.m_CopyBuffer.StartCopy(2, true); } else if(theApp.m_pTextOnlyPaste && wParam == theApp.m_pTextOnlyPaste->m_Atom) { DoTextOnlyPaste(); } else if(theApp.m_pSaveClipboard && wParam == theApp.m_pSaveClipboard->m_Atom) { OnFirstSavecurrentclipboard(); } else { for(int i = 0; i < g_HotKeys.GetCount(); i++) { if(g_HotKeys[i] != NULL && g_HotKeys[i]->m_Atom == wParam && g_HotKeys[i]->m_clipId > 0) { if(g_HotKeys[i]->m_hkType == CHotKey::PASTE_OPEN_CLIP) { Log(StrF(_T("Pasting clip from global shortcut, clipId: %d"), g_HotKeys[i]->m_clipId)); PasteOrShowGroup(g_HotKeys[i]->m_clipId, -1, FALSE, TRUE, false); } else if(g_HotKeys[i]->m_hkType == CHotKey::MOVE_TO_GROUP) { Log(StrF(_T("Global hot key to save clip to group Id: %d, Sending copy to save selection to this group"), g_HotKeys[i]->m_clipId)); KillTimer(GROUP_DOUBLE_CLICK); m_doubleClickGroupId = -1; m_doubleClickGroupStartTime = 0; theApp.SetActiveGroupId(g_HotKeys[i]->m_clipId); theApp.m_activeWnd.SendCopy(CopyReasonEnum::COPY_TO_GROUP); } break; } } } return TRUE; } void CMainFrame::ShowQPasteWithActiveWindowCheck() { //Before we show our window find the current focused window for paste into theApp.m_activeWnd.TrackActiveWnd(true); if (CGetSetOptions::GetOpenToGroupByActiveExe() && theApp.m_activeWnd.ActiveWnd() != NULL) { CString exeName = GetProcessName(theApp.m_activeWnd.ActiveWnd()); if (exeName != _T("")) { theApp.TryEnterOldGroupState(); CString query = StrF(_T("SELECT lID FROM Main WHERE bIsGroup = 1 AND mText = '%s' COLLATE NOCASE"), exeName); CppSQLite3Query q = theApp.m_db.execQueryEx(query); if (q.eof() == false) { int groupId = q.getIntField(_T("lID")); //this will revert back to the old group on hide of ditto theApp.EnterGroupID(groupId, TRUE, TRUE); Log(StrF(_T("Opening Ditto to Group based on found group name, name: %s, GroupId: %d"), exeName, groupId)); } else { theApp.TryEnterOldGroupState(); } } } m_quickPaste.ShowQPasteWnd(this, false, true, FALSE); } void CMainFrame::DoTextOnlyPaste() { CClipboardSaveRestore textOnlyPaste; Log(_T("Text Only paste, saving clipboard to be restored later")); textOnlyPaste.Save(TRUE); Log(_T("Text Only paste, Add cf_text or cf_unicodetext to clipboard")); textOnlyPaste.RestoreTextOnly(); DWORD pasteDelay = g_Opt.GetTextOnlyPasteDelay(); Log(StrF(_T("Text Only paste, delaying %d ms before sending paste"), pasteDelay)); Sleep(pasteDelay); Log(_T("Text Only paste, Sending paste")); theApp.m_activeWnd.SendPaste(false); Log(_T("Text Only paste, Post sending paste")); } void CMainFrame::DoFirstTenPositionsPaste(int nPos) { try { CString csSort = _T(""); CString strFilter = _T(""); bool pastedFromGroup = false; if (theApp.m_GroupID < 0 || CGetSetOptions::GetUseUISelectedGroupForLastTenCopies() == FALSE) { //do not change this this directly relates to the views in the Main table csSort = "Main.bIsGroup ASC, " "Main.stickyClipOrder DESC, " "Main.clipOrder DESC"; if (g_Opt.m_bShowAllClipsInMainList) { if (CGetSetOptions::GetShowGroupsInMainList()) { //found to be slower on large databases strFilter = "((Main.bIsGroup = 1 AND Main.lParentID = -1) OR Main.bIsGroup = 0)"; } else { strFilter = "(Main.bIsGroup = 0)"; } } else { strFilter = "((Main.bIsGroup = 1 AND Main.lParentID = -1) OR (Main.bIsGroup = 0 AND Main.lParentID = -1))"; } } else { pastedFromGroup = true; //do not change this this directly relates to the views in the Main table csSort = "Main.bIsGroup ASC, " "Main.stickyClipGroupOrder DESC, " "Main.clipGroupOrder DESC"; if (theApp.m_GroupID >= 0) { strFilter.Format(_T("Main.lParentID = %d"), theApp.m_GroupID); } } CString query = StrF(_T("SELECT lID, bIsGroup FROM Main WHERE %s ORDER BY %s LIMIT 1 OFFSET %d"), strFilter, csSort, nPos); Log(StrF(_T("Doing Last Ten Paste, Index: %d Query: %s"), nPos, query)); CppSQLite3Query q = theApp.m_db.execQueryEx(query); if(q.eof() == false) { PasteOrShowGroup(q.getIntField(_T("lID")), CGetSetOptions::GetMoveClipsOnGlobal10(), false, g_Opt.m_bSendPasteOnFirstTenHotKeys, pastedFromGroup); } } CATCH_SQLITE_EXCEPTION } void CMainFrame::StartKeyModifyerTimer() { m_keyModifiersTimerCount = 0; m_bMovedSelectionMoveKeyState = false; m_startKeyStateTime = GetTickCount(); m_keyStateModifiers = CAccels::GetKeyStateModifiers(); SetTimer(KEY_STATE_MODIFIERS, 50, NULL); } void CMainFrame::PasteOrShowGroup(int dbId, BOOL updateClipTime, BOOL activeTarget, BOOL sendPaste, bool pastedFromGroup) { try { bool isGroup = false; CppSQLite3Query q = theApp.m_db.execQueryEx(_T("SELECT bIsGroup FROM Main WHERE lID = %d"), dbId); if(q.eof() == false) { if(q.getIntField(_T("bIsGroup")) > 0) { isGroup = true; } } if(isGroup) { DWORD maxDiff = (DWORD)CGetSetOptions::GetGroupDoubleClickTimeMS(); DWORD diff = GetTickCount() - m_doubleClickGroupStartTime; if(m_doubleClickGroupId == dbId && diff < maxDiff) { Log(StrF(_T("Second Press of group hot key, group Id: %d, Sending copy to save selection to this group"), dbId)); KillTimer(GROUP_DOUBLE_CLICK); m_doubleClickGroupId = -1; m_doubleClickGroupStartTime = 0; theApp.SetActiveGroupId(dbId); theApp.m_activeWnd.SendCopy(CopyReasonEnum::COPY_TO_GROUP); } else { m_doubleClickGroupId = dbId; m_doubleClickGroupStartTime = GetTickCount(); int doubleClickTime = CGetSetOptions::GetGroupDoubleClickTimeMS(); SetTimer(GROUP_DOUBLE_CLICK, doubleClickTime, 0); Log(StrF(_T("First Press of group hot key, group Id: %d, timout: %d"), dbId, doubleClickTime)); } } else { KillTimer(GROUP_DOUBLE_CLICK); m_doubleClickGroupId = -1; m_doubleClickGroupStartTime = 0; BOOL bItWas = g_Opt.m_bUpdateTimeOnPaste; if (updateClipTime != -1) { g_Opt.m_bUpdateTimeOnPaste = updateClipTime; } CProcessPaste paste; paste.m_pastedFromGroup = pastedFromGroup; paste.GetClipIDs().Add(dbId); if (activeTarget != -1) { paste.m_bActivateTarget = activeTarget ? true : false;; } if (sendPaste != -1) { paste.m_bSendPaste = sendPaste ? true : false; } paste.DoPaste(); theApp.OnPasteCompleted(); if (updateClipTime != -1) { g_Opt.m_bUpdateTimeOnPaste = bItWas; } } } CATCH_SQLITE_EXCEPTION } void CMainFrame::DoDittoCopyBufferPaste(int nCopyBuffer) { try { CppSQLite3Query q = theApp.m_db.execQueryEx(_T("SELECT lID FROM Main WHERE CopyBuffer = %d"), nCopyBuffer); if(q.eof() == false) { //Don't move these to the top BOOL bItWas = g_Opt.m_bUpdateTimeOnPaste; g_Opt.m_bUpdateTimeOnPaste = FALSE; CProcessPaste paste; paste.GetClipIDs().Add(q.getIntField(_T("lID"))); paste.m_bActivateTarget = false; paste.DoPaste(); theApp.OnPasteCompleted(); g_Opt.m_bUpdateTimeOnPaste = bItWas; } } CATCH_SQLITE_EXCEPTION } void CMainFrame::OnTimer(UINT_PTR nIDEvent) { switch(nIDEvent) { case HIDE_ICON_TIMER: { m_trayIcon.Hide(); KillTimer(nIDEvent); } break; case CLOSE_WINDOW_TIMER: { //m_quickPaste.CloseQPasteWnd(); } break; case REMOVE_OLD_ENTRIES_TIMER: { m_thread.FireDeleteEntries(); } break; case REMOVE_OLD_TEMP_FILES: { m_thread.FireRemoveTempFiles(); } break; case KEY_STATE_MODIFIERS: m_keyModifiersTimerCount++; if(m_keyStateModifiers != 0) { BYTE keyState = CAccels::GetKeyStateModifiers(); //Have they release the key state modifiers yet(ctrl, shift, alt) if((m_keyStateModifiers &keyState) == 0) { KillTimer(KEY_STATE_MODIFIERS); long waitTime = (long)(GetTickCount() - m_startKeyStateTime); if(m_bMovedSelectionMoveKeyState || m_keyModifiersTimerCount > g_Opt.GetKeyStateWaitTimerCount()) { Log(StrF(_T("Timer KEY_STATE_MODIFIERS timeout count hit(%d), count (%d), time (%d), Move Selection from Modifer (%d) sending paste"), g_Opt.GetKeyStateWaitTimerCount(), m_keyModifiersTimerCount, waitTime, m_bMovedSelectionMoveKeyState)); m_quickPaste.OnKeyStateUp(); } else { Log(StrF(_T("Timer KEY_STATE_MODIFIERS count NOT hit(%d), count (%d) time (%d)"), g_Opt.GetKeyStateWaitTimerCount(), m_keyModifiersTimerCount, waitTime)); m_quickPaste.SetKeyModiferState(false); } m_keyStateModifiers = 0; m_keyModifiersTimerCount = 0; m_bMovedSelectionMoveKeyState = 0; } } else { KillTimer(KEY_STATE_MODIFIERS); } break; case ACTIVE_WINDOW_TIMER: { if(theApp.m_bShowingQuickPaste) { theApp.m_activeWnd.TrackActiveWnd(false); } } break; case READ_RANDOM_DB_FILE: { m_thread.FireReadDbFile(); } break; case GROUP_DOUBLE_CLICK: { KillTimer(GROUP_DOUBLE_CLICK); Log(StrF(_T("Processing single click of groupId %d in timer, opening ditto to this group"), m_doubleClickGroupId)); DWORD maxDiff = (DWORD)(CGetSetOptions::GetGroupDoubleClickTimeMS() * 1.5); DWORD diff = GetTickCount() - m_doubleClickGroupStartTime; if(diff < maxDiff) { if(m_doubleClickGroupId > -1) { if (theApp.EnterGroupID(m_doubleClickGroupId, FALSE, TRUE)) { theApp.m_activeWnd.TrackActiveWnd(true); StartKeyModifyerTimer(); m_quickPaste.ShowQPasteWnd(this, false, true, FALSE); } } } else { Log(StrF(_T("Something happened and we didn't process the group timer in time, Id: %d, Diff ms: %d, maxDiff: %d"), m_doubleClickGroupId, diff, maxDiff)); } m_doubleClickGroupId = -1; m_doubleClickGroupStartTime = 0; } break; case SCREEN_RESOLUTION_CHANGED: { KillTimer(SCREEN_RESOLUTION_CHANGED); m_quickPaste.OnScreenResolutionChange(); } break; case DELAYED_SHOW_DITTO_TIMER: { KillTimer(DELAYED_SHOW_DITTO_TIMER); m_quickPaste.ShowQPasteWnd(this, false, false, FALSE); } break; } CFrameWnd::OnTimer(nIDEvent); } LRESULT CMainFrame::OnShowTrayIcon(WPARAM wParam, LPARAM lParam) { if(lParam) { if(!m_trayIcon.IsHidden()) { KillTimer(HIDE_ICON_TIMER); SetTimer(HIDE_ICON_TIMER, 40000, 0); } } if(wParam) { m_trayIcon.Show(); } else { m_trayIcon.Hide(); } return TRUE; } void CMainFrame::OnFirstShowquickpaste() { m_quickPaste.ShowQPasteWnd(this, false, false, FALSE); } void CMainFrame::OnFirstToggleConnectCV() { theApp.ToggleConnectCV(); } void CMainFrame::OnUpdateFirstToggleConnectCV(CCmdUI *pCmdUI) { theApp.UpdateMenuConnectCV(pCmdUI->m_pMenu, ID_FIRST_TOGGLECONNECTCV); } LRESULT CMainFrame::OnClipboardCopied(WPARAM wParam, LPARAM lParam) { Log(_T("Start of function OnClipboardCopied, adding clip to thread for processing")); CClip *pClip = (CClip*)wParam; if(pClip != NULL) { m_thread.AddClipToSave(pClip); } Log(_T("End of function OnClipboardCopied")); return TRUE; } BOOL CMainFrame::PreTranslateMessage(MSG *pMsg) { //forward the mouse wheel onto the window under the cursor //normally windows only sends it to the window with focus, bypass this if ((pMsg->message == WM_MOUSEWHEEL || pMsg->message == WM_MOUSEHWHEEL) && ::GetCapture() == nullptr) { POINT mouse; GetCursorPos(&mouse); HWND hwndFromPoint = ::WindowFromPoint(mouse); if (pMsg->hwnd != hwndFromPoint) { DWORD winProcessId = 0; ::GetWindowThreadProcessId(hwndFromPoint, &winProcessId); if (winProcessId == ::GetCurrentProcessId()) //no-fail! { pMsg->hwnd = hwndFromPoint; } } if (GetKeyState(VK_SHIFT) & 0x8000) pMsg->message = WM_MOUSEHWHEEL; } return CFrameWnd::PreTranslateMessage(pMsg); } void CMainFrame::OnClose() { CloseAllOpenDialogs(); if(m_pEditFrameWnd) { if(m_pEditFrameWnd->CloseAll() == false) { return ; } } Log(_T("OnClose - before stop MainFrm thread")); m_thread.Stop(); Log(_T("OnClose - after stop MainFrm thread")); theApp.BeforeMainClose(); m_PowerManager.Close(); CFrameWnd::OnClose(); } bool CMainFrame::CloseAllOpenDialogs() { bool bRet = false; DWORD dwordProcessId; DWORD dwordChildWindowProcessId; GetWindowThreadProcessId(this->m_hWnd, &dwordProcessId); ASSERT(dwordProcessId); CWnd *pTempWnd = GetDesktopWindow()->GetWindow(GW_CHILD); while((pTempWnd = pTempWnd->GetWindow(GW_HWNDNEXT)) != NULL) { if(pTempWnd->GetSafeHwnd() == NULL) { break; } GetWindowThreadProcessId(pTempWnd->GetSafeHwnd(), &dwordChildWindowProcessId); if(dwordChildWindowProcessId == dwordProcessId) { TCHAR szTemp[100]; GetClassName(pTempWnd->GetSafeHwnd(), szTemp, 100); // #32770 is class name for dialogs so don't process the message if it is a dialog if(STRCMP(szTemp, _T("#32770")) == 0) { pTempWnd->SendMessage(WM_CLOSE, 0, 0); bRet = true; } } } MSG msg; while(PeekMessage(&msg, NULL, NULL, NULL, PM_REMOVE)) { TranslateMessage(&msg); DispatchMessage(&msg); } return bRet; } LRESULT CMainFrame::OnLoadClipOnClipboard(WPARAM wParam, LPARAM lParam) { CClip *pClip = (CClip*)wParam; if(pClip == NULL) { LogSendRecieveInfo("---------ERROR OnLoadClipOnClipboard pClip == NULL"); return FALSE; } if(pClip) { CProcessPaste paste; paste.m_bSendPaste = false; paste.m_bActivateTarget = false; paste.m_pasteOptions.m_delayRenderLockout = GetTickCount(); LogSendRecieveInfo("---------OnLoadClipOnClipboard - Before PutFormats on clipboard"); paste.m_pOle->PutFormatOnClipboard(&pClip->m_Formats); paste.m_pOle->CacheGlobalData(theApp.m_cfIgnoreClipboard, NewGlobalP("Ignore", sizeof("Ignore"))); LogSendRecieveInfo("---------OnLoadClipOnClipboard - After PutFormats on clipboard"); LogSendRecieveInfo(StrF(_T("---------OnLoadClipOnClipboard - Setting clip id: %d on ole clipboard"), pClip->m_id)); paste.GetClipIDs().Add(pClip->m_id); paste.DoPaste(); LogSendRecieveInfo(StrF(_T("---------OnLoadClipOnClipboard - After paste clip id: %d on ole clipboard"), pClip->m_id)); } delete pClip; return TRUE; } LRESULT CMainFrame::OnAddToDatabaseFromSocket(WPARAM wParam, LPARAM lParam) { CClipList *pClipList = (CClipList*)wParam; if(pClipList == NULL) { LogSendRecieveInfo("---------OnAddToDatabaseFromSocket - ERROR pClipList == NULL"); return FALSE; } DWORD flags = (DWORD)lParam; if(flags & REMOTE_CLIP_ADD_TO_CLIPBOARD) { CClip *pClip = pClipList->GetTail(); if(pClip) { LogSendRecieveInfo("OnAddToDatabaseFromSocket - Adding clip from socket setting clip to be put on clipboard"); pClip->m_param1 |= REMOTE_CLIP_ADD_TO_CLIPBOARD; } } if (flags & REMOTE_CLIP_MANUAL_SEND) { CClip *pClip = pClipList->GetTail(); if (pClip) { LogSendRecieveInfo("OnAddToDatabaseFromSocket - Adding clip from socket setting clip was a manual send from other side"); pClip->m_param1 |= REMOTE_CLIP_MANUAL_SEND; } } m_thread.AddRemoteClipToSave(pClipList); delete pClipList; return TRUE; } LRESULT CMainFrame::OnErrorOnSendRecieve(WPARAM wParam, LPARAM lParam) { CString csNewText = (TCHAR*)wParam; ShowErrorMessage(_T("Ditto - Send/Receive Error"), csNewText); return TRUE; } CString WndName(HWND hParent) { TCHAR cWindowText[200]; ::GetWindowText(hParent, cWindowText, 100); int nCount = 0; while(STRLEN(cWindowText) <= 0) { hParent = ::GetParent(hParent); if(hParent == NULL) { break; } ::GetWindowText(hParent, cWindowText, 100); nCount++; if(nCount > 100) { Log(_T("GetTargetName reached maximum search depth of 100")); break; } } return cWindowText; } void CMainFrame::OnFirstHelp() { CHyperLink::GotoURL(_T("https://sourceforge.net/p/ditto-cp/wiki/"), SW_SHOW); } void CMainFrame::ShowErrorMessage(CString csTitle, CString csMessage) { Log(StrF(_T("ShowErrorMessage %s - %s"), csTitle, csMessage)); m_trayIcon.SetBalloonDetails(csMessage, csTitle, CTrayNotifyIcon::BalloonStyle::Error, CGetSetOptions::GetBalloonTimeout()); } void CMainFrame::OnFirstImport() { theApp.ImportClips(theApp.m_MainhWnd); } void CMainFrame::ShowEditWnd(CClipIDs &Ids) { CWaitCursor wait; bool bCreatedWindow = false; if(m_pEditFrameWnd == NULL) { m_pEditFrameWnd = new CEditFrameWnd; m_pEditFrameWnd->LoadFrame(IDR_MAINFRAME); bCreatedWindow = true; } if(m_pEditFrameWnd) { m_pEditFrameWnd->EditIds(Ids); m_pEditFrameWnd->SetNotifyWnd(m_hWnd); if(bCreatedWindow) { CSize sz; CPoint pt; CGetSetOptions::GetEditWndSize(sz); CGetSetOptions::GetEditWndPoint(pt); CRect cr(pt, sz); EnsureWindowVisible(&cr); m_pEditFrameWnd->MoveWindow(cr); } m_pEditFrameWnd->ShowWindow(SW_SHOW); m_pEditFrameWnd->SetForegroundWindow(); m_pEditFrameWnd->SetFocus(); } } LRESULT CMainFrame::OnEditWndClose(WPARAM wParam, LPARAM lParam) { m_pEditFrameWnd = NULL; return TRUE; } LRESULT CMainFrame::OnSetConnected(WPARAM wParam, LPARAM lParam) { if(wParam) { theApp.SetConnectCV(true); } else if(lParam) { theApp.SetConnectCV(false); } return TRUE; } LRESULT CMainFrame::OnOpenCloseWindow(WPARAM wParam, LPARAM lParam) { if(wParam) { StartKeyModifyerTimer(); ShowQPasteWithActiveWindowCheck(); } else if(lParam) { m_quickPaste.HideQPasteWnd(); } return TRUE; } void CMainFrame::OnDestroy() { CFrameWnd::OnDestroy(); if(m_pEditFrameWnd) { m_pEditFrameWnd->DestroyWindow(); } } void CMainFrame::OnFirstNewclip() { CClipIDs IDs; IDs.Add( - 1); theApp.EditItems(IDs, true); } void CMainFrame::OnFirstOption() { if(m_pOptions != NULL) { ::SetForegroundWindow(m_pOptions->m_hWnd); } else { m_pOptions = new COptionsSheet(_T("")); if(m_pOptions != NULL) { ((COptionsSheet*)m_pOptions)->SetNotifyWnd(m_hWnd); m_pOptions->Create(); m_pOptions->ShowWindow(SW_SHOW); } } } void CMainFrame::OnFirstGlobalhotkeys() { if(m_pGlobalClips != NULL) { ::SetForegroundWindow(m_pGlobalClips->m_hWnd); } else { m_pGlobalClips = new GlobalClips(); CAlphaBlend tran; tran.SetTransparent(m_hWnd, 0, 1); if(m_pGlobalClips != NULL) { ((GlobalClips*)m_pGlobalClips)->SetNotifyWnd(m_hWnd); m_pGlobalClips->Create(IDD_GLOBAL_CLIPS, NULL); m_pGlobalClips->ShowWindow(SW_SHOW); } } } LRESULT CMainFrame::OnShowOptions(WPARAM wParam, LPARAM lParam) { OnFirstOption(); return 0; } LRESULT CMainFrame::OnOptionsClosed(WPARAM wParam, LPARAM lParam) { BOOL themeChanged = (BOOL)wParam; m_trayIcon.MinimiseToTray(this); CAlphaBlend tran; tran.SetTransparent(m_hWnd, 255, 0); delete m_pOptions; m_pOptions = NULL; if (themeChanged) { g_Opt.m_Theme.Load(g_Opt.GetTheme()); m_quickPaste.CloseQPasteWnd(); } else { if (m_quickPaste.m_pwndPaste != NULL) { m_quickPaste.m_pwndPaste->PostMessage(NM_POST_OPTIONS_WINDOW); } } m_trayIcon.SetMenu(NULL, IDR_MENU); theApp.m_Language.UpdateTrayIconRightClickMenu(&m_trayIcon.GetMenu()); return 0; } LRESULT CMainFrame::OnGlobalClipsClosed(WPARAM wParam, LPARAM lParam) { m_trayIcon.MinimiseToTray(this); CAlphaBlend tran; tran.SetTransparent(m_hWnd, 255, 0); delete m_pGlobalClips; m_pGlobalClips = NULL; return 0; } void CMainFrame::RefreshShowInTaskBar() { BOOL windowVisible = m_quickPaste.IsWindowVisibleEx(); m_quickPaste.CloseQPasteWnd(); if (windowVisible) { m_quickPaste.ShowQPasteWnd(this, true, false, true); } } LRESULT CMainFrame::OnDeleteClipDataClosed(WPARAM wParam, LPARAM lParam) { m_trayIcon.MinimiseToTray(this); CAlphaBlend tran; tran.SetTransparent(m_hWnd, 255, 0); delete m_pDeleteClips; m_pDeleteClips = NULL; return 0; } void CMainFrame::OnFirstDeleteclipdata() { //this->ShowWindow(SW_HIDE); if (m_pDeleteClips != NULL) { ::SetForegroundWindow(m_pDeleteClips->m_hWnd); } else { m_pDeleteClips = new CDeleteClipData(); CAlphaBlend tran; tran.SetTransparent(m_hWnd, 0, 1); if (m_pDeleteClips != NULL) { ((CDeleteClipData*) m_pDeleteClips)->SetNotifyWnd(m_hWnd); m_pDeleteClips->Create(IDD_DELETE_CLIP_DATA, NULL); m_pDeleteClips->ShowWindow(SW_SHOW); } } } LRESULT CMainFrame::OnSaveClipboardMessage(WPARAM wParam, LPARAM lParam) { OnFirstSavecurrentclipboard(); return TRUE; } void CMainFrame::OnFirstSavecurrentclipboard() { Log(_T("Start Saving the current clipboard to the database")); CClip* pClip = new CClip; if(pClip) { CClipTypes* pTypes = theApp.LoadTypesFromDB(); if(pTypes) { if(pClip->LoadFromClipboard(pTypes, false, _T(""))) { Log(_T("Loaded clips from the clipboard, sending message to save to the db")); ::PostMessage(m_hWnd, WM_CLIPBOARD_COPIED, (WPARAM)pClip, 0); } else { Log(_T("Failed to load clips from the clipboard, not saving to db")); delete pClip; pClip = NULL; } } else { Log(_T("Failed to load supported types from the db, not saving to the db")); } } Log(_T("Start Saving the current clipboard to the database")); } LRESULT CMainFrame::OnReAddTaskBarIcon(WPARAM wParam, LPARAM lParam) { if(CGetSetOptions::GetShowIconInSysTray()) { m_trayIcon.SetIcon(CTrayNotifyIcon::LoadIcon(IDR_MAINFRAME)); } return TRUE; } LRESULT CMainFrame::OnReOpenDatabase(WPARAM wParam, LPARAM lParam) { Log(StrF(_T("OnReOpenDatabase, Start closing and reopening database Delay: %d"), CGetSetOptions::GetWindowsResumeDelayReOpenDbMS())); try { Sleep(CGetSetOptions::GetWindowsResumeDelayReOpenDbMS()); m_quickPaste.CloseQPasteWnd(); theApp.m_db.close(); OpenDatabase(CGetSetOptions::GetDBPath()); } CATCH_SQLITE_EXCEPTION Log(StrF(_T("OnReOpenDatabase, End closing and reopening database Delay: %d"), CGetSetOptions::GetWindowsResumeDelayReOpenDbMS())); return TRUE; } LRESULT CMainFrame::OnShowMsgWindow(WPARAM wParam, LPARAM lParam) { CString *pMsg = (CString*)wParam; int clipId = (int)lParam; m_trayIcon.SetBalloonDetails(pMsg->GetBuffer(), _T("Ditto"), CTrayNotifyIcon::BalloonStyle::Info, CGetSetOptions::GetBalloonTimeout()); delete pMsg; return TRUE; } LRESULT CMainFrame::OnShowDittoGroup(WPARAM wParam, LPARAM lParam) { int groupId = (int)wParam; CppSQLite3Query q = theApp.m_db.execQueryEx(_T("SELECT bIsGroup FROM Main WHERE lID = %d"), groupId); if(q.eof() == false) { if(q.getIntField(_T("bIsGroup")) > 0) { PasteOrShowGroup(groupId, FALSE, FALSE, FALSE, false); } } return TRUE; } void CMainFrame::OnFirstFixupstickycliporder() { ReOrderStickyClips(-1, theApp.m_db); } LRESULT CMainFrame::OnResolutionChange(WPARAM wParam, LPARAM lParam) { if (m_startupScreenWidth != GetScreenWidth() || m_startupScreenHeight != GetScreenHeight()) { m_startupScreenWidth = GetScreenWidth(); m_startupScreenHeight = GetScreenHeight(); SetTimer(SCREEN_RESOLUTION_CHANGED, 1000, NULL); } return TRUE; }