CopyThread.cpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. // CopyThread.cpp : implementation file
  2. //
  3. #include "stdafx.h"
  4. #include "cp_main.h"
  5. #include "CopyThread.h"
  6. #ifdef _DEBUG
  7. #define new DEBUG_NEW
  8. #undef THIS_FILE
  9. static char THIS_FILE[] = __FILE__;
  10. #endif
  11. /////////////////////////////////////////////////////////////////////////////
  12. // CCopyThread
  13. IMPLEMENT_DYNCREATE(CCopyThread, CWinThread)
  14. CCopyThread::CCopyThread():
  15. m_bQuit(false),
  16. m_bConfigChanged(false),
  17. m_pClipboardViewer(NULL),
  18. m_connectOnStartup(true)
  19. {
  20. m_bAutoDelete = false;
  21. }
  22. CCopyThread::~CCopyThread()
  23. {
  24. m_LocalConfig.DeleteTypes();
  25. m_SharedConfig.DeleteTypes();
  26. DELETE_PTR(m_pClipboardViewer);
  27. }
  28. BOOL CCopyThread::InitInstance()
  29. {
  30. m_pClipboardViewer = new CClipboardViewer(this);
  31. m_pClipboardViewer->m_connectOnStartup = m_connectOnStartup;
  32. // the window is created within this thread and therefore uses its message queue
  33. m_pClipboardViewer->Create();
  34. return TRUE;
  35. }
  36. int CCopyThread::ExitInstance()
  37. {
  38. m_pClipboardViewer->Disconnect(false);
  39. return CWinThread::ExitInstance();
  40. }
  41. // Called within Copy Thread:
  42. void CCopyThread::OnClipboardChange()
  43. {
  44. Log(_T("OnClipboardChange - Start"));
  45. SyncConfig(); // synchronize with the main thread's copy configuration
  46. // if we are told not to copy on change, then we have nothing to do.
  47. if(!m_LocalConfig.m_bCopyOnChange)
  48. return;
  49. CClip* pClip = new CClip;
  50. CClipTypes* pSupportedTypes = m_LocalConfig.m_pSupportedTypes;
  51. bool bDeleteMemory = false;
  52. //If we are copying from a Ditto Buffer then save all to the database, so when we paste this it will paste
  53. //just like you were using Ctrl-V
  54. if(theApp.m_CopyBuffer.Active())
  55. {
  56. Log(_T("LoadFromClipboard - Copy buffer Active Start"));
  57. pSupportedTypes = new CClipTypes;
  58. if(pSupportedTypes)
  59. {
  60. bDeleteMemory = true;
  61. COleDataObject oleData;
  62. if(oleData.AttachClipboard())
  63. {
  64. oleData.BeginEnumFormats();
  65. FORMATETC format;
  66. while(oleData.GetNextFormat(&format))
  67. {
  68. pSupportedTypes->Add(format.cfFormat);
  69. }
  70. oleData.Release();
  71. }
  72. }
  73. else
  74. {
  75. pSupportedTypes = m_LocalConfig.m_pSupportedTypes;
  76. }
  77. Log(_T("LoadFromClipboard - Copy buffer Active End"));
  78. }
  79. Log(_T("LoadFromClipboard - Before"));
  80. bool bResult = pClip->LoadFromClipboard(pSupportedTypes);
  81. Log(_T("LoadFromClipboard - After"));
  82. if(!bResult)
  83. {
  84. DWORD delay = CGetSetOptions::GetNoFormatsRetryDelay();
  85. if(delay > 0)
  86. {
  87. Log(StrF(_T("LoadFromClipboard didn't find any clips to save, sleeping %dms, then trying again"), delay));
  88. Sleep(delay);
  89. Log(_T("LoadFromClipboard #2 - Before"));
  90. bResult = pClip->LoadFromClipboard(pSupportedTypes);
  91. Log(_T("LoadFromClipboard #2 - After"));
  92. }
  93. else
  94. {
  95. Log(_T("LoadFromClipboard didn't find any clips to save, retry setting is not set, not retrying"));
  96. }
  97. }
  98. if(bDeleteMemory)
  99. {
  100. delete pSupportedTypes;
  101. pSupportedTypes = NULL;
  102. }
  103. if(!bResult)
  104. {
  105. delete pClip;
  106. return; // error
  107. }
  108. if(m_LocalConfig.m_bAsyncCopy)
  109. ::PostMessage(m_LocalConfig.m_hClipHandler, WM_CLIPBOARD_COPIED, (WPARAM)pClip, 0);
  110. else
  111. ::SendMessage(m_LocalConfig.m_hClipHandler, WM_CLIPBOARD_COPIED, (WPARAM)pClip, 0);
  112. Log(_T("OnClipboardChange - End"));
  113. }
  114. void CCopyThread::SyncConfig()
  115. {
  116. // atomic read
  117. if(m_bConfigChanged)
  118. {
  119. CClipTypes* pTypes = NULL;
  120. ATL::CCritSecLock csLock(m_cs.m_sect);
  121. pTypes = m_LocalConfig.m_pSupportedTypes;
  122. m_LocalConfig = m_SharedConfig;
  123. // NULL means that it shouldn't have been sync'ed
  124. if( m_SharedConfig.m_pSupportedTypes == NULL )
  125. { // let m_LocalConfig keep its types
  126. m_LocalConfig.m_pSupportedTypes = pTypes; // undo sync
  127. pTypes = NULL; // nothing to delete
  128. }
  129. else
  130. m_SharedConfig.m_pSupportedTypes = NULL; // now owned by LocalConfig
  131. // delete old types
  132. if( pTypes )
  133. {
  134. delete pTypes;
  135. }
  136. }
  137. }
  138. bool CCopyThread::IsClipboardViewerConnected()
  139. {
  140. return m_pClipboardViewer->m_bIsConnected;
  141. }
  142. bool CCopyThread::GetConnectCV()
  143. {
  144. return m_pClipboardViewer->GetConnect();
  145. }
  146. void CCopyThread::SetConnectCV(bool bConnect)
  147. {
  148. if(m_pClipboardViewer != NULL && m_pClipboardViewer->m_hWnd != NULL)
  149. {
  150. ::SendMessage( m_pClipboardViewer->m_hWnd, WM_SETCONNECT, bConnect, 0 );
  151. }
  152. }
  153. void CCopyThread::SetSupportedTypes( CClipTypes* pTypes )
  154. {
  155. ATL::CCritSecLock csLock(m_cs.m_sect);
  156. if(m_SharedConfig.m_pSupportedTypes)
  157. {
  158. DELETE_PTR(m_SharedConfig.m_pSupportedTypes);
  159. }
  160. m_SharedConfig.m_pSupportedTypes = pTypes;
  161. m_bConfigChanged = true;
  162. }
  163. HWND CCopyThread::SetClipHandler(HWND hWnd)
  164. {
  165. ATL::CCritSecLock csLock(m_cs.m_sect);
  166. HWND hRet = m_SharedConfig.m_hClipHandler;
  167. m_SharedConfig.m_hClipHandler = hWnd;
  168. m_bConfigChanged = (hRet != hWnd);
  169. return hRet;
  170. }
  171. HWND CCopyThread::GetClipHandler()
  172. {
  173. ATL::CCritSecLock csLock(m_cs.m_sect);
  174. HWND hRet = m_SharedConfig.m_hClipHandler;
  175. return hRet;
  176. }
  177. bool CCopyThread::SetCopyOnChange(bool bVal)
  178. {
  179. ATL::CCritSecLock csLock(m_cs.m_sect);
  180. bool bRet = m_SharedConfig.m_bCopyOnChange;
  181. m_SharedConfig.m_bCopyOnChange = bVal;
  182. m_bConfigChanged = (bRet != bVal);
  183. return bRet;
  184. }
  185. bool CCopyThread::GetCopyOnChange()
  186. {
  187. ATL::CCritSecLock csLock(m_cs.m_sect);
  188. bool bRet = m_SharedConfig.m_bCopyOnChange;
  189. return bRet;
  190. }
  191. bool CCopyThread::SetAsyncCopy(bool bVal)
  192. {
  193. ATL::CCritSecLock csLock(m_cs.m_sect);
  194. bool bRet = m_SharedConfig.m_bAsyncCopy;
  195. m_SharedConfig.m_bAsyncCopy = bVal;
  196. m_bConfigChanged = (bRet != bVal);
  197. return bRet;
  198. }
  199. bool CCopyThread::GetAsyncCopy()
  200. {
  201. ATL::CCritSecLock csLock(m_cs.m_sect);
  202. bool bRet = m_SharedConfig.m_bAsyncCopy;
  203. return bRet;
  204. }
  205. void CCopyThread::Init(CCopyConfig cfg)
  206. {
  207. ASSERT(m_LocalConfig.m_pSupportedTypes == NULL);
  208. m_LocalConfig = m_SharedConfig = cfg;
  209. // let m_LocalConfig own the m_pSupportedTypes
  210. m_SharedConfig.m_pSupportedTypes = NULL;
  211. }
  212. bool CCopyThread::Quit()
  213. {
  214. m_bQuit = true;
  215. m_pClipboardViewer->PostMessage( WM_QUIT );
  216. return CWinThread::PostThreadMessage( WM_QUIT, NULL, NULL ) != FALSE;
  217. }