XTOutBarCtrl.cpp
上传用户:szled88
上传日期:2015-04-09
资源大小:43957k
文件大小:85k
源码类别:

对话框与窗口

开发平台:

Visual C++

  1. // XTOutBarCtrl.cpp : implementation of the CXTOutBarCtrl class.
  2. //
  3. // This file is a part of the XTREME CONTROLS MFC class library.
  4. // (c)1998-2008 Codejock Software, All Rights Reserved.
  5. //
  6. // THIS SOURCE FILE IS THE PROPERTY OF CODEJOCK SOFTWARE AND IS NOT TO BE
  7. // RE-DISTRIBUTED BY ANY MEANS WHATSOEVER WITHOUT THE EXPRESSED WRITTEN
  8. // CONSENT OF CODEJOCK SOFTWARE.
  9. //
  10. // THIS SOURCE CODE CAN ONLY BE USED UNDER THE TERMS AND CONDITIONS OUTLINED
  11. // IN THE XTREME TOOLKIT PRO LICENSE AGREEMENT. CODEJOCK SOFTWARE GRANTS TO
  12. // YOU (ONE SOFTWARE DEVELOPER) THE LIMITED RIGHT TO USE THIS SOFTWARE ON A
  13. // SINGLE COMPUTER.
  14. //
  15. // CONTACT INFORMATION:
  16. // support@codejock.com
  17. // http://www.codejock.com
  18. //
  19. /////////////////////////////////////////////////////////////////////////////
  20. #include "stdafx.h"
  21. #include "Common/XTPResourceManager.h"
  22. #include "Common/XTPColorManager.h"
  23. #include "Common/XTPDrawHelpers.h"
  24. #include "Resource.h"
  25. #include "XTDefines.h"
  26. #include "XTGlobal.h"
  27. #include "XTVC50Helpers.h"
  28. #include "XTFlatComboBox.h"
  29. #include "XTOutBarCtrl.h"
  30. #include "XTOutBarCtrlTheme.h"
  31. #include "XTFunctions.h"
  32. #ifdef _DEBUG
  33. #define new DEBUG_NEW
  34. #undef THIS_FILE
  35. static char THIS_FILE[] = __FILE__;
  36. #endif
  37. /////////////////////////////////////////////////////////////////////////////
  38. // CXTOutBarEditItem
  39. CXTOutBarEditItem::CXTOutBarEditItem()
  40. {
  41. m_iIndex = -1;
  42. m_pParentWnd = NULL;
  43. m_bSmallIcons = false;
  44. m_bEscapeKey = false;
  45. m_bIsFolder = false;
  46. }
  47. CXTOutBarEditItem::~CXTOutBarEditItem()
  48. {
  49. }
  50. IMPLEMENT_DYNAMIC(CXTOutBarEditItem, CXTEdit)
  51. BEGIN_MESSAGE_MAP(CXTOutBarEditItem, CXTEdit)
  52. //{{AFX_MSG_MAP(CXTOutBarEditItem)
  53. ON_WM_KILLFOCUS()
  54. ON_WM_CHAR()
  55. ON_WM_SETCURSOR()
  56. //}}AFX_MSG_MAP
  57. END_MESSAGE_MAP()
  58. void CXTOutBarEditItem::OnKillFocus(CWnd* /*pNewWnd*/)
  59. {
  60. PostMessage(WM_CLOSE);
  61. if (!m_bEscapeKey)
  62. {
  63. GetWindowText(m_strText);
  64. if (!m_strText.IsEmpty())
  65. {
  66. m_pParentWnd->EndLabelEdit(this, m_bIsFolder);
  67. }
  68. }
  69. }
  70. BOOL CXTOutBarEditItem::PreTranslateMessage(MSG* pMsg)
  71. {
  72. switch (pMsg->wParam)
  73. {
  74. case VK_ESCAPE:
  75. {
  76. m_bEscapeKey = TRUE; // fall thru
  77. }
  78. case VK_RETURN:
  79. {
  80. PostMessage(WM_CLOSE);
  81. return TRUE;
  82. }
  83. }
  84. return CXTEdit::PreTranslateMessage(pMsg);
  85. }
  86. void CXTOutBarEditItem::PostNcDestroy()
  87. {
  88. delete this;
  89. }
  90. BOOL CXTOutBarEditItem::Create(LPCTSTR lpszText, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, UINT nIndex, bool bIsFolder, bool bSmallIcons)
  91. {
  92. ASSERT_VALID(pParentWnd); // must be valid.
  93. m_pParentWnd = (CXTOutBarCtrl*)pParentWnd;
  94. ASSERT_KINDOF(CXTOutBarCtrl, m_pParentWnd); // must be an outlook bar control.
  95. // get the inside size of the outlook bar.
  96. CRect rcInside;
  97. m_pParentWnd->GetInsideRect(rcInside);
  98. rcInside.DeflateRect(5, 5);
  99. // adjust the size if necessary.
  100. m_rcOriginal = rect;
  101. m_bSmallIcons = bSmallIcons;
  102. m_bIsFolder = bIsFolder;
  103. m_iIndex = nIndex;
  104. m_strText = lpszText;
  105. if (m_rcOriginal.Width() > rcInside.Width())
  106. {
  107. if (m_bSmallIcons != TRUE)
  108. {
  109. m_rcOriginal.left = rcInside.left;
  110. }
  111. m_rcOriginal.right = rcInside.right;
  112. }
  113. if (!CXTEdit::Create(dwStyle, m_rcOriginal, pParentWnd, nID))
  114. return FALSE;
  115. if (!CXTEdit::Initialize(m_pParentWnd))
  116. return FALSE;
  117. SetFont(m_pParentWnd->GetFontX());
  118. SetWindowText(m_strText);
  119. return TRUE;
  120. }
  121. void CXTOutBarEditItem::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
  122. {
  123. // if this is a folder, do default processing.
  124. if (m_bIsFolder)
  125. {
  126. CXTEdit::OnChar(nChar, nRepCnt, nFlags);
  127. return;
  128. }
  129. // if the escape or return key has been pressed, abort edit.
  130. else if (nChar == VK_ESCAPE || nChar == VK_RETURN)
  131. {
  132. if (nChar == VK_ESCAPE)
  133. {
  134. m_bEscapeKey = TRUE;
  135. }
  136. m_pParentWnd->SetFocus();
  137. return;
  138. }
  139. CXTEdit::OnChar(nChar, nRepCnt, nFlags);
  140. // get the inside size of the outlook bar.
  141. CRect rcInside;
  142. m_pParentWnd->GetInsideRect(rcInside);
  143. rcInside.DeflateRect(5, 5);
  144. // get the edit box text
  145. CString strText;
  146. GetWindowText(strText);
  147. // get the font used by the outlook bar.
  148. CFont* pFont = m_pParentWnd->GetFontX();
  149. ASSERT(pFont && pFont->GetSafeHandle());
  150. // select the font used by the outlook bar.
  151. CWindowDC dc(NULL);
  152. // get the size of the text string.
  153. CFont* f = dc.SelectObject(pFont);
  154. CSize sz = dc.GetTextExtent(strText);
  155. dc.SelectObject(f);
  156. // construct the edit box size.
  157. CRect rcItem = m_rcOriginal;
  158. // calculate the width of the edit box.
  159. if (m_bSmallIcons == TRUE)
  160. {
  161. rcItem.right = rcItem.left + sz.cx + 9;
  162. if (rcItem.right < m_rcOriginal.right)
  163. {
  164. rcItem.right = m_rcOriginal.right;
  165. }
  166. else if (rcItem.right > rcInside.right)
  167. {
  168. rcItem.right = rcInside.right;
  169. }
  170. }
  171. else
  172. {
  173. int x = (rcInside.Width()-sz.cx)/2;
  174. rcItem.left = rcInside.left + x;
  175. rcItem.right = rcItem.left + sz.cx;
  176. rcItem.InflateRect(6, 0);
  177. if (rcItem.Width() < m_rcOriginal.Width())
  178. {
  179. rcItem = m_rcOriginal;
  180. }
  181. else if (rcItem.Width() > rcInside.Width())
  182. {
  183. rcItem.left = rcInside.left;
  184. rcItem.right = rcInside.right;
  185. }
  186. }
  187. // resize the edit box.
  188. MoveWindow(&rcItem);
  189. }
  190. BOOL CXTOutBarEditItem::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
  191. {
  192. HCURSOR hCursor = AfxGetApp()->LoadStandardCursor(IDC_IBEAM);
  193. if (hCursor != NULL)
  194. {
  195. ::SetCursor(hCursor);
  196. return TRUE;
  197. }
  198. return CXTEdit::OnSetCursor(pWnd, nHitTest, message);
  199. }
  200. /////////////////////////////////////////////////////////////////////////////
  201. // CXTOutBarCtrl
  202. IMPLEMENT_THEME_HOST(CXTOutBarCtrl)
  203. IMPLEMENT_THEME_REFRESH(CXTOutBarCtrl, CWnd)
  204. CXTOutBarCtrl::CXTOutBarCtrl()
  205. : CXTThemeManagerStyleHost(GetThemeFactoryClass())
  206. , m_penBlack(PS_SOLID, 1, RGB(0, 0, 0))
  207. {
  208. m_rcUpArrow.SetRectEmpty();
  209. m_rcDownArrow.SetRectEmpty();
  210. m_dwFlags = OBS_XT_DEFAULT;
  211. m_pLargeImageList = NULL;
  212. m_pSmallImageList = NULL;
  213. m_sizeOffset.cx = 4;
  214. m_sizeOffset.cy = 3;
  215. m_sizeMargin.cx = 5;
  216. m_sizeMargin.cy = 5;
  217. m_nFolderHeight = 22;
  218. m_nSelFolder = 0;
  219. m_nLastFolderSelected = -1;
  220. m_nFolderHilighted = -1;
  221. m_nLastItemSelected = -1;
  222. m_nItemHilighted = -1;
  223. m_nLastDragItemDraw = -1;
  224. m_nLastDragItemDrawType = -1;
  225. m_nFirstItem = 0;
  226. m_nIconSpacingSmall = 10;
  227. m_nIconSpacingLarge = 8;
  228. m_nAnimationTickCount = 10;
  229. m_nSelAnimCount = 0;
  230. m_nSelAnimTiming = 0;
  231. m_nHitInternal1 = 0;
  232. m_nHitInternal2 = 0;
  233. m_bUpArrow = FALSE;
  234. m_bDownArrow = FALSE;
  235. m_bUpPressed = FALSE;
  236. m_bDownPressed = FALSE;
  237. m_bLooping = FALSE;
  238. m_bPressedHighlight = FALSE;
  239. m_bIconPressed = FALSE;
  240. SetFontX(CFont::FromHandle((HFONT)GetStockObject(DEFAULT_GUI_FONT)));
  241. }
  242. CXTOutBarCtrl::~CXTOutBarCtrl()
  243. {
  244. int iFolder;
  245. for (iFolder = 0; iFolder < m_arFolder.GetSize(); iFolder++)
  246. {
  247. m_nSelFolder = iFolder;
  248. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  249. SAFE_DELETE(pBarFolder);
  250. }
  251. m_arFolder.RemoveAll();
  252. m_penBlack.DeleteObject();
  253. m_font.DeleteObject();
  254. }
  255. IMPLEMENT_DYNCREATE(CXTOutBarCtrl, CWnd)
  256. BEGIN_MESSAGE_MAP(CXTOutBarCtrl, CWnd)
  257. //{{AFX_MSG_MAP(CXTOutBarCtrl)
  258. ON_WM_TIMER()
  259. ON_WM_PAINT()
  260. ON_MESSAGE(WM_PRINTCLIENT, OnPrintClient)
  261. ON_WM_ERASEBKGND()
  262. ON_WM_MOUSEMOVE()
  263. ON_MESSAGE_VOID(WM_MOUSELEAVE, OnMouseLeave)
  264. ON_WM_LBUTTONDOWN()
  265. ON_WM_SETCURSOR()
  266. ON_WM_SIZE()
  267. ON_COMMAND(XT_IDC_LARGEICON, OnLargeIcon)
  268. ON_UPDATE_COMMAND_UI(XT_IDC_LARGEICON, OnUpdateLargeIcon)
  269. ON_COMMAND(XT_IDC_SMALLICON, OnSmallIcon)
  270. ON_UPDATE_COMMAND_UI(XT_IDC_SMALLICON, OnUpdateSmallIcon)
  271. ON_COMMAND(XT_IDC_REMOVEITEM, OnRemoveItem)
  272. ON_UPDATE_COMMAND_UI(XT_IDC_REMOVEITEM, OnUpdateRemoveItem)
  273. ON_COMMAND(XT_IDC_RENAMEITEM, OnRenameItem)
  274. ON_UPDATE_COMMAND_UI(XT_IDC_RENAMEITEM, OnUpdateRenameItem)
  275. ON_WM_RBUTTONUP()
  276. ON_WM_DESTROY()
  277. //}}AFX_MSG_MAP
  278. END_MESSAGE_MAP()
  279. void CXTOutBarCtrl::DrawItemIcon(CDC* pDC, CPoint pt, CXTOutBarItem* pBarItem, CImageList* pImageList)
  280. {
  281. GetTheme()->DrawItemIcon(pDC, pt, pBarItem, pImageList);
  282. }
  283. void CXTOutBarCtrl::DrawItemText(CDC* pDC, CRect rc, CXTOutBarItem* pBarItem, UINT nFormat)
  284. {
  285. GetTheme()->DrawItemText(pDC, rc, pBarItem, nFormat);
  286. }
  287. LRESULT CXTOutBarCtrl::NotifyOwner(UINT nCode, XT_OUTBAR_INFO* pobi)
  288. {
  289. CWnd* pOwner = (CWnd*)GetOwner();
  290. ASSERT_VALID(pOwner);
  291. pobi->hWnd = m_hWnd;
  292. return pOwner->SendMessage(XTWM_OUTBAR_NOTIFY,
  293. (WPARAM)nCode, (LPARAM)pobi);
  294. }
  295. void CXTOutBarCtrl::EndLabelEdit(CXTOutBarEditItem* pEdit, bool bIsFolder)
  296. {
  297. ASSERT_VALID(pEdit); // must be valid.
  298. XT_OUTBAR_INFO obi;
  299. obi.bFolder = bIsFolder;
  300. obi.nIndex = pEdit->GetIndex();
  301. obi.lpszText = pEdit->GetText();
  302. obi.nDragFrom = -1;
  303. obi.nDragTo = -1;
  304. LRESULT lResult = 0;
  305. // give the owner a chance to process the message first.
  306. if (bIsFolder)
  307. {
  308. lResult = NotifyOwner(OBN_XT_ONGROUPENDEDIT, &obi);
  309. }
  310. else
  311. {
  312. lResult = NotifyOwner(OBN_XT_ONLABELENDEDIT, &obi);
  313. }
  314. // If the return value is "0" then the owner has processed
  315. // the message, so just return.
  316. if (lResult == 0)
  317. {
  318. return;
  319. }
  320. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  321. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  322. if (pBarFolder == NULL)
  323. {
  324. return;
  325. }
  326. CRect rc;
  327. bool  bUpdate = false;
  328. if (bIsFolder)
  329. {
  330. bUpdate = (pBarFolder->GetName() != obi.lpszText);
  331. if (bUpdate)
  332. {
  333. GetFolderRect(obi.nIndex, rc);
  334. SetFolderText(obi.nIndex, obi.lpszText);
  335. }
  336. }
  337. else
  338. {
  339. ASSERT(obi.nIndex >= 0 && obi.nIndex < pBarFolder->GetItemCount());
  340. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(obi.nIndex);
  341. if (pBarItem == NULL)
  342. {
  343. return;
  344. }
  345. bUpdate = (pBarItem->GetName() != obi.lpszText);
  346. if (bUpdate)
  347. {
  348. GetInsideRect(rc);
  349. SetItemText(obi.nIndex, obi.lpszText);
  350. }
  351. }
  352. if (bUpdate)
  353. {
  354. OnLabelChanged(&obi); // handle in derived class.
  355. InvalidateRect(rc);
  356. }
  357. }
  358. void CXTOutBarCtrl::OnLabelChanged(const XT_OUTBAR_INFO* /*pObi*/)
  359. {
  360. }
  361. void CXTOutBarCtrl::SetFolderText(const int iIndex, LPCTSTR lpszFolderName)
  362. {
  363. if (m_arFolder.GetSize() != 0)
  364. {
  365. ASSERT((iIndex >= 0) && (iIndex < GetFolderCount()));
  366. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  367. pBarFolder->SetName(lpszFolderName);
  368. }
  369. }
  370. void CXTOutBarCtrl::SetItemText(const int iIndex, LPCTSTR lpszItemName)
  371. {
  372. if (m_arFolder.GetSize() != 0)
  373. {
  374. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  375. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  376. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  377. pBarItem->SetName(lpszItemName);
  378. }
  379. }
  380. BOOL CXTOutBarCtrl::Create(DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, const DWORD dwFlags)
  381. {
  382. if (!CreateEx(NULL, dwStyle, rect, pParentWnd, nID, dwFlags))
  383. {
  384. return FALSE;
  385. }
  386. return TRUE;
  387. }
  388. BOOL CXTOutBarCtrl::CreateEx(DWORD dwExStyle, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, const DWORD dwFlags)
  389. {
  390. ASSERT(dwStyle & WS_CHILD);
  391. ASSERT_VALID(pParentWnd);
  392. if (!CWnd::CreateEx(dwExStyle, NULL, NULL, dwStyle, rect, pParentWnd, nID))
  393. {
  394. return FALSE;
  395. }
  396. m_dwFlags = dwFlags;
  397. return TRUE;
  398. }
  399. BOOL CXTOutBarCtrl::IsSmallIconView(const int iFolder/*= -1*/) const
  400. {
  401. if (GetFolderCount() <= 0 || iFolder == -1)
  402. return ((m_dwFlags & OBS_XT_SMALLICON) == OBS_XT_SMALLICON);
  403. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  404. return pBarFolder->m_bSmallIcons;
  405. }
  406. void CXTOutBarCtrl::SetSmallIconView(const BOOL bSet, const int iFolder/*= -1*/)
  407. {
  408. m_nFirstItem = 0;
  409. if (iFolder != -1)
  410. {
  411. if (iFolder >= 0 && iFolder < GetFolderCount())
  412. {
  413. CXTOutBarFolder* pbf = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  414. pbf->m_bSmallIcons = bSet;
  415. }
  416. }
  417. else
  418. {
  419. // do all current folders, and set flag so new folders
  420. // have the chosen style
  421. int nFolders = GetFolderCount();
  422. int i;
  423. for (i = 0; i < nFolders; i++)
  424. {
  425. CXTOutBarFolder* pbf = (CXTOutBarFolder*)m_arFolder.GetAt(i);
  426. pbf->m_bSmallIcons = bSet;
  427. }
  428. if (bSet && !IsSmallIconView())
  429. m_dwFlags |= OBS_XT_SMALLICON;
  430. else if (!bSet && IsSmallIconView())
  431. m_dwFlags &= ~OBS_XT_SMALLICON;
  432. }
  433. CRect rc;
  434. GetInsideRect(rc);
  435. InvalidateRect(rc, false);
  436. }
  437. DWORD CXTOutBarCtrl::GetFlag() const
  438. {
  439. return m_dwFlags;
  440. }
  441. void CXTOutBarCtrl::ModifyFlag(const DWORD& dwRemove, const DWORD& dwAdd, const bool bRedraw)
  442. {
  443. if (dwRemove)
  444. {
  445. m_dwFlags &= ~dwRemove;
  446. }
  447. if (dwAdd)
  448. {
  449. m_dwFlags |= dwAdd;
  450. }
  451. if (GetSafeHwnd())
  452. {
  453. if (bRedraw == true)
  454. {
  455. Invalidate();
  456. UpdateWindow();
  457. }
  458. }
  459. }
  460. void CXTOutBarCtrl::OnTimer(UINT_PTR nIDEvent)
  461. {
  462. if (nIDEvent == 3 && m_nLastItemSelected >= 0)
  463. {
  464. m_nSelAnimCount++;
  465. if (m_nSelAnimCount > 10) m_nSelAnimCount = -1;
  466. if (m_nSelAnimCount == 0)
  467. DrawAnimItem(-1, 1, m_nLastItemSelected);
  468. if (m_nSelAnimCount == 1)
  469. DrawAnimItem(0, 0, m_nLastItemSelected);
  470. if (m_nSelAnimCount == 2)
  471. DrawAnimItem(1, 1, m_nLastItemSelected);
  472. if (m_nSelAnimCount == 3)
  473. DrawAnimItem(0, 0, m_nLastItemSelected);
  474. }
  475. CWnd::OnTimer(nIDEvent);
  476. }
  477. void CXTOutBarCtrl::DrawAnimItem(const int iOffsetX, const int iOffsetY, const int iIndex)
  478. {
  479. CImageList* pImageList = GetFolderImageList(m_nSelFolder, IsSmallIconView(m_nSelFolder));
  480. CRect rc, irc;
  481. GetInsideRect(irc);
  482. GetItemRect(m_nSelFolder, m_nLastItemSelected, rc);
  483. if (m_arFolder.GetSize() != 0)
  484. {
  485. if (m_nLastItemSelected >= m_nFirstItem && irc.bottom > rc.bottom && irc.top < rc.top)
  486. {
  487. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  488. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  489. ASSERT(iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  490. CXTOutBarItem* pBarItem = (CXTOutBarItem*)pBarFolder->GetItemAt(iIndex);
  491. ASSERT(pBarItem && pImageList);
  492. CClientDC dc(this);
  493. if (IsSmallIconView(m_nSelFolder))
  494. {
  495. if (pImageList)
  496. {
  497. IMAGEINFO ii;
  498. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  499. CSize szImage = CRect(ii.rcImage).Size();
  500. CPoint pt;
  501. pt.x = rc.left + 2;
  502. pt.y = rc.top + (rc.Height() - szImage.cy) / 2;
  503. CRect rcBck(pt.x-1, pt.y-1, pt.x + szImage.cx + 2, pt.y + szImage.cy + 2);
  504. dc.FillSolidRect(rcBck, GetXtremeColor(COLOR_3DSHADOW));
  505. pt.x += iOffsetX;
  506. pt.y += iOffsetY;
  507. DrawItemIcon(&dc, pt, pBarItem, pImageList);
  508. }
  509. }
  510. else
  511. {
  512. if (pImageList)
  513. {
  514. IMAGEINFO ii;
  515. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  516. CSize szImage = CRect(ii.rcImage).Size();
  517. CPoint pt;
  518. pt.x = rc.left + (rc.Width() - szImage.cx) / 2;
  519. pt.y = rc.top;
  520. CRect rcBck(pt.x-1, pt.y-1, pt.x + szImage.cx + 2, pt.y + szImage.cy + 2);
  521. dc.FillSolidRect(rcBck, GetXtremeColor(COLOR_3DSHADOW));
  522. pt.x += iOffsetX;
  523. pt.y += iOffsetY;
  524. DrawItemIcon(&dc, pt, pBarItem, pImageList);
  525. }
  526. }
  527. }
  528. }
  529. }
  530. void CXTOutBarCtrl::OnPaint()
  531. {
  532. // background is already filled in gray
  533. CPaintDC dc(this);
  534. // Paint to a memory device context to help
  535. // eliminate screen flicker.
  536. CXTPBufferDC memDC(dc);
  537. OnDraw(&memDC);
  538. }
  539. LRESULT CXTOutBarCtrl::OnPrintClient(WPARAM wParam, LPARAM /*lParam*/)
  540. {
  541. CDC* pDC = CDC::FromHandle((HDC)wParam);
  542. if (pDC)
  543. {
  544. OnDraw(pDC);
  545. }
  546. return TRUE;
  547. }
  548. void CXTOutBarCtrl::OnDraw(CDC* pDC)
  549. {
  550. // Get the client rect.
  551. CXTPClientRect rectClient(this);
  552. CRect rc;
  553. GetInsideRect(rc);
  554. if (!GetFolderChild())
  555. {
  556. //pDC->FillSolidRect(rc, m_clrBack);
  557. GetTheme()->FillInsideRect(pDC, rc);
  558. }
  559. int max = (int)m_arFolder.GetSize();
  560. CRect frc;
  561. int t;
  562. for (t = 0; t < max; t++)
  563. {
  564. GetFolderRect(t, frc);
  565. DrawFolder(pDC, t, frc, xtMouseNormal);
  566. }
  567. if (!GetFolderChild())
  568. {
  569. int f, l;
  570. GetVisibleRange(m_nSelFolder, f, l);
  571. m_rcUpArrow.SetRect(0, 0, GetSystemMetrics(SM_CXVSCROLL), GetSystemMetrics(SM_CXVSCROLL));
  572. m_rcDownArrow = m_rcUpArrow;
  573. m_rcUpArrow.OffsetRect(rc.right - 5 - GetSystemMetrics(SM_CXVSCROLL), rc.top +5);
  574. m_rcDownArrow.OffsetRect(rc.right - 5 - GetSystemMetrics(SM_CXVSCROLL), rc.bottom - 5 - GetSystemMetrics(SM_CXVSCROLL));
  575. if (f > 0 && m_rcUpArrow.IntersectRect(rc, m_rcUpArrow))
  576. {
  577. if (m_bUpPressed)
  578. {
  579. DrawScrollButton(pDC, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP | DFCS_PUSHED);
  580. }
  581. else
  582. {
  583. DrawScrollButton(pDC, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP);
  584. }
  585. m_bUpArrow = TRUE;
  586. pDC->ExcludeClipRect(m_rcUpArrow);
  587. }
  588. else m_bUpArrow = FALSE;
  589. if (l < GetItemCount() - 1 && m_rcDownArrow.IntersectRect(rc, m_rcDownArrow))
  590. {
  591. if (m_bDownPressed)
  592. {
  593. DrawScrollButton(pDC, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN | DFCS_PUSHED);
  594. }
  595. else
  596. {
  597. DrawScrollButton(pDC, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN);
  598. }
  599. m_bDownArrow = TRUE;
  600. pDC->ExcludeClipRect(m_rcDownArrow);
  601. }
  602. else m_bDownArrow = FALSE;
  603. PaintItems(pDC, m_nSelFolder, rc);
  604. }
  605. //dc.BitBlt(rectClient.left, rectClient.top, rectClient.Width(), rectClient.Height(), pDC, 0, 0, SRCCOPY);
  606. //pDC->RestoreDC(nSavedDC);
  607. if (m_nFolderHilighted >= 0)
  608. {
  609. int i = m_nFolderHilighted;
  610. m_nFolderHilighted = -1;
  611. HighlightFolder(pDC, i);
  612. }
  613. if (m_nItemHilighted >= 0)
  614. {
  615. int i = m_nItemHilighted;
  616. m_nItemHilighted = -1;
  617. HighlightItem(pDC, i, m_bPressedHighlight);
  618. }
  619. }
  620. BOOL CXTOutBarCtrl::OnEraseBkgnd(CDC* /*pDC*/)
  621. {
  622. return TRUE;
  623. }
  624. BOOL CXTOutBarCtrl::GetFolderRect(const int iIndex, CRect& rect) const
  625. {
  626. if (m_arFolder.GetSize() != 0)
  627. {
  628. int max = (int)m_arFolder.GetSize();
  629. ASSERT(iIndex >= 0 && iIndex < max);
  630. if (iIndex >= 0 && iIndex < max)
  631. {
  632. CRect rc;
  633. GetClientRect(rc);
  634. if (iIndex > m_nSelFolder)
  635. rect.SetRect(rc.left, rc.bottom - ((max - iIndex))* m_nFolderHeight, rc.right,
  636. rc.bottom - (max - iIndex - 1)* m_nFolderHeight);
  637. else
  638. rect.SetRect(rc.left, rc.top + iIndex* m_nFolderHeight, rc.right,
  639. rc.top + (1 + iIndex)* m_nFolderHeight);
  640. return TRUE;
  641. }
  642. }
  643. return FALSE;
  644. }
  645. void CXTOutBarCtrl::GetItemRect(const int iFolder, const int iIndex, CRect& rect, CRect* pInsideRect /*= NULL*/)
  646. {
  647. CRect rc;
  648. if (pInsideRect)
  649. rc = *pInsideRect;
  650. else
  651. GetInsideRect(rc);
  652. int top = rc.top;
  653. CSize sz(0, 0);
  654. int y = 0;
  655. int t;
  656. for (t = 0; t < iIndex; t++)
  657. {
  658. sz = GetItemSize(iFolder, t, rectItemBoth);
  659. top += sz.cy;
  660. if (IsSmallIconView(iFolder))
  661. {
  662. top += m_nIconSpacingSmall;
  663. }
  664. else
  665. {
  666. top += m_nIconSpacingLarge;
  667. }
  668. if (t == m_nFirstItem - 1) y = top - rc.top;
  669. }
  670. sz = GetItemSize(iFolder, iIndex, rectItemBoth);
  671. rect.SetRect(rc.left, top, rc.left + sz.cx, top + sz.cy);
  672. rect.top -= y;
  673. rect.bottom -= y;
  674. rect.left += m_sizeMargin.cx;
  675. rect.top += m_sizeMargin.cy;
  676. rect.bottom += m_sizeMargin.cy;
  677. if (!IsSmallIconView(iFolder))
  678. {
  679. rect.left = rc.left;
  680. rect.right = rc.right;
  681. }
  682. }
  683. void CXTOutBarCtrl::DrawFolder(CDC* pDC, const int iIndex, CRect rect, const XTMouseState eHilight)
  684. {
  685. if (iIndex < m_arFolder.GetSize() && (iIndex >= 0))
  686. {
  687. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  688. GetTheme()->DrawFolder(pDC, rect, pBarFolder, eHilight);
  689. }
  690. }
  691. int CXTOutBarCtrl::AddFolder(LPCTSTR lpszFolderName, const DWORD dwData)
  692. {
  693. CXTOutBarFolder* pBarFolder = new CXTOutBarFolder(lpszFolderName, dwData, this);
  694. ASSERT(pBarFolder);
  695. pBarFolder->m_bSmallIcons = (m_dwFlags & OBS_XT_SMALLICON);
  696. return (int)m_arFolder.Add((void*)pBarFolder);
  697. }
  698. void CXTOutBarCtrl::GetInsideRect(CRect& rect) const
  699. {
  700. GetClientRect(rect);
  701. if (m_arFolder.GetSize() > 0)
  702. {
  703. int max = (int)m_arFolder.GetSize();
  704. rect.top += m_nFolderHeight* (m_nSelFolder + 1);
  705. rect.bottom -= (max - m_nSelFolder - 1) * m_nFolderHeight;
  706. return;
  707. }
  708. }
  709. void CXTOutBarCtrl::OnMouseLeave()
  710. {
  711. OnMouseMove(0, CPoint(-1, -1));
  712. }
  713. void CXTOutBarCtrl::OnMouseMove(UINT nFlags, CPoint point)
  714. {
  715. int iIndex;
  716. int ht = HitTestEx(point, iIndex);
  717. int nFolderHilighted = -1, nItemHilighted = -1;
  718. if (ht == hitFolder) nFolderHilighted = iIndex;
  719. if (ht == hitItem) nItemHilighted = iIndex;
  720. if ((m_nFolderHilighted != nFolderHilighted) || (m_nItemHilighted != nItemHilighted))
  721. {
  722. m_nFolderHilighted = nFolderHilighted;
  723. m_nItemHilighted = nItemHilighted;
  724. Invalidate(FALSE);
  725. if (nFolderHilighted != -1 || nItemHilighted != -1)
  726. {
  727. TRACKMOUSEEVENT tme =
  728. {
  729. sizeof(TRACKMOUSEEVENT), TME_LEAVE, m_hWnd
  730. };
  731. _TrackMouseEvent(&tme);
  732. }
  733. }
  734. CWnd::OnMouseMove(nFlags, point);
  735. }
  736. CXTOutBarCtrl::HitTestCode CXTOutBarCtrl::HitTestEx(const CPoint& point, int& iIndex)
  737. {
  738. if (m_arFolder.GetSize() != 0)
  739. {
  740. if (m_bUpArrow && m_rcUpArrow.PtInRect(point)) return hitUpScroll;
  741. if (m_bDownArrow && m_rcDownArrow.PtInRect(point)) return hitDnScroll;
  742. int max = (int)m_arFolder.GetSize(), t;
  743. CRect rc;
  744. for (t = 0; t < max; t++)
  745. {
  746. GetFolderRect(t, rc);
  747. if (rc.PtInRect(point))
  748. {
  749. iIndex = t;
  750. return hitFolder;
  751. }
  752. }
  753. GetInsideRect(rc);
  754. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  755. max = pBarFolder->GetItemCount();
  756. for (t = m_nFirstItem; t < max; t++)
  757. {
  758. CRect rcItem;
  759. if (!IsSmallIconView(m_nSelFolder))
  760. {
  761. GetIconRect(m_nSelFolder, t, rcItem);
  762. if (rcItem.PtInRect(point))
  763. {
  764. iIndex = t;
  765. return hitItem;
  766. }
  767. else if (rcItem.top > rc.bottom) break;
  768. GetLabelRect(m_nSelFolder, t, rcItem);
  769. rcItem.top -= m_sizeOffset.cy;
  770. if (rcItem.PtInRect(point))
  771. {
  772. iIndex = t;
  773. return hitItem;
  774. }
  775. else if (rcItem.top > rc.bottom) break;
  776. }
  777. else
  778. {
  779. GetItemRect(m_nSelFolder, t, rcItem);
  780. if (rcItem.PtInRect(point))
  781. {
  782. iIndex = t;
  783. return hitItem;
  784. }
  785. else if (rcItem.top > rc.bottom) break;
  786. }
  787. }
  788. }
  789. return hitNone;
  790. }
  791. void CXTOutBarCtrl::HighlightFolder(CDC* pDC, const int iIndex)
  792. {
  793. CWnd* pFocus = GetFocus();
  794. if (pFocus != NULL && pFocus != this && IsChild(pFocus))
  795. {
  796. return;
  797. }
  798. if (m_nFolderHilighted == iIndex)
  799. return;
  800. CDC dcClient;
  801. if (!pDC)
  802. {
  803. dcClient.Attach(::GetDC(m_hWnd));
  804. pDC = &dcClient;
  805. }
  806. if (m_nFolderHilighted >= 0)
  807. {
  808. CRect rc;
  809. if (GetFolderRect(m_nFolderHilighted, rc))
  810. {
  811. DrawFolder(pDC, m_nFolderHilighted, rc, xtMouseNormal);
  812. }
  813. }
  814. if (iIndex >= 0)
  815. {
  816. CRect rc;
  817. if (GetFolderRect(iIndex, rc))
  818. {
  819. DrawFolder(pDC, iIndex, rc, xtMouseHover);
  820. }
  821. }
  822. if (dcClient.GetSafeHdc())
  823. {
  824. ::ReleaseDC(m_hWnd, dcClient.Detach());
  825. }
  826. m_nFolderHilighted = iIndex;
  827. }
  828. void CXTOutBarCtrl::OnLButtonDown(UINT nFlags, CPoint point)
  829. {
  830. if (GetFocus() != this) SetFocus();
  831. int iIndex, ht = HitTestEx(point, iIndex);
  832. m_nLastDragItemDrawType = -1;
  833. CRect inRc;
  834. GetInsideRect(inRc);
  835. if (ht == hitFolder)
  836. {
  837. BOOL bHigh = TRUE;
  838. CRect rc;
  839. GetFolderRect(iIndex, rc);
  840. if (::GetCapture() == NULL)
  841. {
  842. SetCapture();
  843. ASSERT(this == GetCapture());
  844. CClientDC dc(this);
  845. DrawFolder(&dc, iIndex, rc, xtMouseSelect);
  846. AfxLockTempMaps();
  847. for (;;)
  848. {
  849. MSG msg;
  850. VERIFY(::GetMessage(&msg, NULL, 0, 0));
  851. if (CWnd::GetCapture() != this) break;
  852. switch (msg.message)
  853. {
  854. case WM_MOUSEMOVE:
  855. {
  856. CPoint pt(msg.lParam);
  857. int idx, ht1 = HitTestEx(pt, idx);
  858. if (ht1 == hitFolder && idx == iIndex)
  859. {
  860. if (!bHigh)
  861. {
  862. DrawFolder(&dc, iIndex, rc, xtMouseSelect);
  863. bHigh = TRUE;
  864. }
  865. }
  866. else
  867. {
  868. if (bHigh)
  869. {
  870. DrawFolder(&dc, iIndex, rc, xtMouseNormal);
  871. bHigh = FALSE;
  872. }
  873. }
  874. }
  875. break;
  876. case WM_LBUTTONUP:
  877. {
  878. ReleaseCapture();
  879. if (bHigh)
  880. {
  881. DrawFolder(&dc, iIndex, rc, xtMouseNormal);
  882. bHigh = FALSE;
  883. }
  884. CPoint pt(msg.lParam);
  885. int idx, ht1 = HitTestEx(pt, idx);
  886. if (ht1 == hitFolder && idx != m_nSelFolder)
  887. SetSelFolder(idx);
  888. }
  889. goto ExitLoop2;
  890. case WM_KEYDOWN:
  891. if (msg.wParam != VK_ESCAPE)
  892. break;
  893. default:
  894. DispatchMessage(&msg);
  895. break;
  896. }
  897. }
  898. ExitLoop2:
  899. ReleaseCapture();
  900. AfxUnlockTempMaps(FALSE);
  901. }
  902. if (bHigh) InvalidateRect(rc, FALSE);
  903. }
  904. else m_nLastFolderSelected = -1;
  905. if (ht == hitItem)
  906. {
  907. m_nLastDragItemDraw = -1;
  908. BOOL bHigh = TRUE, bDragging = FALSE;
  909. CRect rc;
  910. GetItemRect(m_nSelFolder, iIndex, rc);
  911. HCURSOR hCur = GetCursor();
  912. if (::GetCapture() == NULL)
  913. {
  914. SetCapture();
  915. ASSERT(this == GetCapture());
  916. CClientDC dc(this);
  917. HighlightItem(NULL, iIndex, TRUE);
  918. AfxLockTempMaps();
  919. for (;;)
  920. {
  921. MSG msg;
  922. VERIFY(::GetMessage(&msg, NULL, 0, 0));
  923. if (CWnd::GetCapture() != this) break;
  924. switch (msg.message)
  925. {
  926. case WM_MOUSEMOVE:
  927. {
  928. CPoint pt(msg.lParam);
  929. int idx, ht1 = HitTestEx(pt, idx);
  930. if (bDragging)
  931. {
  932. if (ht1 == hitItem)
  933. {
  934. DrawDragArrow(&dc, iIndex, idx);
  935. ::SetCursor(XTAuxData().hcurDragMove);
  936. hCur = XTAuxData().hcurDragMove;
  937. }
  938. else
  939. {
  940. CRect rcItem;
  941. GetItemRect(m_nSelFolder, GetItemCount() - 1, rcItem);
  942. if (pt.y > rcItem.bottom && pt.y < inRc.bottom)
  943. {
  944. DrawDragArrow(&dc, iIndex, GetItemCount());
  945. ::SetCursor(XTAuxData().hcurDragMove);
  946. hCur = XTAuxData().hcurDragMove;
  947. }
  948. else
  949. {
  950. DrawDragArrow(&dc, iIndex, -1);
  951. ::SetCursor(XTAuxData().hcurDragNone);
  952. hCur = XTAuxData().hcurDragNone;
  953. }
  954. }
  955. }
  956. else
  957. {
  958. if (ht1 == hitItem && idx == iIndex)
  959. {
  960. if (!bHigh)
  961. {
  962. HighlightItem(NULL, iIndex, TRUE);
  963. bHigh = TRUE;
  964. m_bPressedHighlight = TRUE;
  965. }
  966. }
  967. else
  968. {
  969. if (ht1 == hitItem)
  970. {
  971. if (bHigh)
  972. {
  973. HighlightItem(NULL, iIndex, FALSE);
  974. bHigh = FALSE;
  975. m_bPressedHighlight = FALSE;
  976. }
  977. }
  978. else
  979. {
  980. if (m_dwFlags & OBS_XT_DRAGITEMS)
  981. {
  982. HighlightItem(NULL, iIndex, TRUE);
  983. bHigh = TRUE;
  984. bDragging = TRUE;
  985. XT_OUTBAR_INFO obi;
  986. obi.nIndex = iIndex;
  987. obi.nDragFrom = -1;
  988. obi.nDragTo = -1;
  989. obi.lpszText = GetItemText(iIndex);
  990. obi.bFolder = false;
  991. NotifyOwner(OBN_XT_BEGINDRAG, &obi);
  992. ::SetCursor(XTAuxData().hcurDragMove);
  993. hCur = XTAuxData().hcurDragMove;
  994. m_bPressedHighlight = TRUE;
  995. }
  996. }
  997. }
  998. }
  999. }
  1000. break;
  1001. case WM_SETCURSOR:
  1002. ::SetCursor(hCur);
  1003. break;
  1004. case WM_LBUTTONUP:
  1005. {
  1006. ReleaseCapture();
  1007. if (bHigh)
  1008. {
  1009. HighlightItem(NULL, -1);
  1010. bHigh = FALSE;
  1011. }
  1012. CPoint pt(msg.lParam);
  1013. int idx, ht1 = HitTestEx(pt, idx);
  1014. if (!bDragging)
  1015. {
  1016. if (ht1 == hitItem && idx == iIndex)
  1017. {
  1018. CXTOutBarItem* pBarItem = GetBarFolderItem(m_nSelFolder, idx);
  1019. if (!pBarItem->IsEnabled())
  1020. {
  1021. return;
  1022. }
  1023. XT_OUTBAR_INFO obi;
  1024. obi.nIndex = idx;
  1025. obi.nDragFrom = -1;
  1026. obi.nDragTo = -1;
  1027. obi.lpszText = GetItemText(idx);
  1028. obi.bFolder = false;
  1029. int nSelFolder = m_nSelFolder;
  1030. LRESULT lResult = NotifyOwner(OBN_XT_ITEMCLICK, &obi);
  1031. if (lResult == TRUE && nSelFolder == GetSelFolder())
  1032. {
  1033. CXTOutBarFolder* pBarFolder = GetBarFolder(GetSelFolder());
  1034. pBarFolder->SetSelItem(idx);
  1035. if (m_nSelAnimTiming > 0 && iIndex != m_nLastItemSelected && m_nLastItemSelected >= 0)
  1036. {
  1037. DrawAnimItem(0, 0, m_nLastItemSelected);
  1038. }
  1039. BOOL bInvalidate = FALSE;
  1040. if ((m_dwFlags & OBS_XT_SELHIGHLIGHT) && (m_nLastItemSelected >= 0))
  1041. {
  1042. bInvalidate = TRUE;
  1043. }
  1044. m_nLastItemSelected = iIndex;
  1045. if ((m_dwFlags & OBS_XT_SELHIGHLIGHT) && (m_nLastItemSelected >= 0))
  1046. {
  1047. bInvalidate = TRUE;
  1048. }
  1049. if (bInvalidate)
  1050. {
  1051. Invalidate(FALSE);
  1052. }
  1053. }
  1054. }
  1055. }
  1056. else
  1057. {
  1058. if (ht1 == hitItem)
  1059. {
  1060. if (idx != iIndex)
  1061. {
  1062. int iItemSelected = m_nLastItemSelected;
  1063. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  1064. POSITION pos1 = pBarFolder->m_barItems.FindIndex(iIndex);
  1065. CXTOutBarItem* pItemFrom = (CXTOutBarItem*)pBarFolder->m_barItems.GetAt(pos1);
  1066. POSITION pos2 = pBarFolder->m_barItems.FindIndex(idx);
  1067. XT_OUTBAR_INFO obi;
  1068. obi.nIndex = iIndex;
  1069. obi.nDragFrom = iIndex;
  1070. obi.nDragTo = idx;
  1071. obi.lpszText = GetItemText(iIndex);
  1072. obi.bFolder = false;
  1073. LRESULT lResult = NotifyOwner(OBN_XT_DRAGITEM, &obi);
  1074. if (lResult == TRUE)
  1075. {
  1076. if (m_arFolder.GetSize() != 0)
  1077. {
  1078. pBarFolder->m_barItems.RemoveAt(pos1);
  1079. pBarFolder->m_barItems.InsertBefore(pos2, pItemFrom);
  1080. }
  1081. int i = -1;
  1082. if (iItemSelected == iIndex)
  1083. {
  1084. i = idx;
  1085. if (idx > iIndex)
  1086. {
  1087. i--;
  1088. }
  1089. }
  1090. else if (((iIndex <= m_nLastItemSelected) && (idx <= m_nLastItemSelected)) ||
  1091. ((iIndex > m_nLastItemSelected) && (idx > m_nLastItemSelected)))
  1092. {
  1093. // do nothing...
  1094. }
  1095. else if (m_nLastItemSelected != -1)
  1096. {
  1097. i = m_nLastItemSelected;
  1098. if (idx > m_nLastItemSelected)
  1099. {
  1100. i--;
  1101. }
  1102. else
  1103. {
  1104. i++;
  1105. }
  1106. }
  1107. if (i >= 0)
  1108. {
  1109. pBarFolder->SetSelItem(i);
  1110. m_nLastItemSelected = i;
  1111. }
  1112. }
  1113. }
  1114. }
  1115. else
  1116. {
  1117. CRect rcItem;
  1118. GetItemRect(m_nSelFolder, GetItemCount() - 1, rcItem);
  1119. if (pt.y > rcItem.bottom && pt.y < inRc.bottom)
  1120. {
  1121. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  1122. POSITION pos1 = pBarFolder->m_barItems.FindIndex(iIndex);
  1123. CXTOutBarItem* pItemFrom = (CXTOutBarItem*)pBarFolder->m_barItems.GetAt(pos1);
  1124. POSITION pos2 = pBarFolder->m_barItems.FindIndex(pBarFolder->m_barItems.GetCount());
  1125. XT_OUTBAR_INFO obi;
  1126. obi.nIndex = (int)pBarFolder->m_barItems.GetCount();
  1127. obi.nDragFrom = iIndex;
  1128. obi.nDragTo = obi.nIndex;
  1129. obi.lpszText = GetItemText(iIndex);
  1130. obi.bFolder = false;
  1131. LRESULT lResult = NotifyOwner(OBN_XT_DRAGITEM, &obi);
  1132. if (lResult == TRUE)
  1133. {
  1134. pBarFolder->m_barItems.RemoveAt(pos1);
  1135. pBarFolder->m_barItems.InsertAfter(pos2, pItemFrom);
  1136. }
  1137. }
  1138. }
  1139. }
  1140. }
  1141. goto ExitLoop4;
  1142. default:
  1143. DispatchMessage(&msg);
  1144. break;
  1145. }
  1146. }
  1147. ExitLoop4:
  1148. ReleaseCapture();
  1149. AfxUnlockTempMaps(FALSE);
  1150. if (bDragging)
  1151. {
  1152. Invalidate();
  1153. }
  1154. }
  1155. m_bPressedHighlight = FALSE;
  1156. if (bHigh) InvalidateRect(rc, FALSE);
  1157. }
  1158. else m_nItemHilighted = -1;
  1159. if (ht == hitDnScroll)
  1160. {
  1161. m_bLooping = TRUE;
  1162. BOOL bHigh = TRUE;
  1163. if (::GetCapture() == NULL)
  1164. {
  1165. SetCapture();
  1166. ASSERT(this == GetCapture());
  1167. CClientDC dc(this);
  1168. DrawScrollButton(&dc, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN | DFCS_PUSHED);
  1169. SetTimer(2, 300, NULL);
  1170. AfxLockTempMaps();
  1171. for (;;)
  1172. {
  1173. MSG msg;
  1174. VERIFY(::GetMessage(&msg, NULL, 0, 0));
  1175. if (CWnd::GetCapture() != this) break;
  1176. switch (msg.message)
  1177. {
  1178. case WM_MOUSEMOVE:
  1179. {
  1180. CPoint pt(msg.lParam);
  1181. if (m_rcDownArrow.PtInRect(pt))
  1182. {
  1183. if (bHigh == FALSE)
  1184. {
  1185. DrawScrollButton(&dc, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN | DFCS_PUSHED);
  1186. bHigh = TRUE;
  1187. m_bDownArrow = TRUE;
  1188. m_bDownPressed = TRUE;
  1189. }
  1190. }
  1191. else
  1192. {
  1193. if (bHigh == TRUE)
  1194. {
  1195. DrawScrollButton(&dc, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN);
  1196. bHigh = FALSE;
  1197. m_bDownArrow = FALSE;
  1198. m_bDownPressed = FALSE;
  1199. }
  1200. }
  1201. }
  1202. break;
  1203. case WM_LBUTTONUP:
  1204. {
  1205. ReleaseCapture();
  1206. if (bHigh)
  1207. {
  1208. DrawScrollButton(&dc, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN);
  1209. bHigh = FALSE;
  1210. }
  1211. m_bDownArrow = FALSE;
  1212. CPoint pt(msg.lParam);
  1213. if (m_rcDownArrow.PtInRect(pt))
  1214. {
  1215. CRect itrc;
  1216. GetItemRect(m_nSelFolder, GetItemCount() - 1, itrc);
  1217. CRect crc;
  1218. GetInsideRect(crc);
  1219. if (itrc.bottom > crc.bottom)
  1220. {
  1221. m_nFirstItem++;
  1222. InvalidateRect(crc, TRUE);
  1223. }
  1224. }
  1225. }
  1226. goto ExitLoop3;
  1227. case WM_TIMER:
  1228. {
  1229. if (msg.wParam == 2)
  1230. {
  1231. if (bHigh)
  1232. {
  1233. CPoint pt(msg.pt);
  1234. ScreenToClient(&pt);
  1235. if (m_rcDownArrow.PtInRect(pt))
  1236. {
  1237. m_bDownPressed = TRUE;
  1238. CRect itrc;
  1239. GetItemRect(m_nSelFolder, GetItemCount() - 1, itrc);
  1240. CRect crc;
  1241. GetInsideRect(crc);
  1242. if (itrc.bottom > crc.bottom)
  1243. {
  1244. m_nFirstItem++;
  1245. InvalidateRect(crc, TRUE);
  1246. }
  1247. else goto ExitLoop3;
  1248. }
  1249. else m_bDownPressed = FALSE;
  1250. }
  1251. }
  1252. break;
  1253. }
  1254. case WM_KEYDOWN:
  1255. if (msg.wParam != VK_ESCAPE)
  1256. break;
  1257. default:
  1258. DispatchMessage(&msg);
  1259. break;
  1260. }
  1261. }
  1262. ExitLoop3:
  1263. KillTimer(2);
  1264. ReleaseCapture();
  1265. AfxUnlockTempMaps(FALSE);
  1266. m_bLooping = FALSE;
  1267. m_bDownPressed = FALSE;
  1268. m_bDownArrow = FALSE;
  1269. CRect crc;
  1270. GetInsideRect(crc);
  1271. InvalidateRect(crc, TRUE);
  1272. }
  1273. }
  1274. if (ht == hitUpScroll)
  1275. {
  1276. m_bLooping = TRUE;
  1277. BOOL bHigh = TRUE;
  1278. if (::GetCapture() == NULL)
  1279. {
  1280. SetCapture();
  1281. ASSERT(this == GetCapture());
  1282. CClientDC dc(this);
  1283. DrawScrollButton(&dc, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP | DFCS_PUSHED);
  1284. SetTimer(2, 300, NULL);
  1285. AfxLockTempMaps();
  1286. for (;;)
  1287. {
  1288. MSG msg;
  1289. VERIFY(::GetMessage(&msg, NULL, 0, 0));
  1290. if (CWnd::GetCapture() != this) break;
  1291. switch (msg.message)
  1292. {
  1293. case WM_MOUSEMOVE:
  1294. {
  1295. CPoint pt(msg.lParam);
  1296. if (m_rcUpArrow.PtInRect(pt))
  1297. {
  1298. if (bHigh == FALSE)
  1299. {
  1300. DrawScrollButton(&dc, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP | DFCS_PUSHED);
  1301. bHigh = TRUE;
  1302. m_bUpArrow = TRUE;
  1303. m_bUpPressed = TRUE;
  1304. }
  1305. }
  1306. else
  1307. {
  1308. if (bHigh == TRUE)
  1309. {
  1310. DrawScrollButton(&dc, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP);
  1311. bHigh = FALSE;
  1312. m_bUpArrow = FALSE;
  1313. m_bUpPressed = FALSE;
  1314. }
  1315. }
  1316. }
  1317. break;
  1318. case WM_LBUTTONUP:
  1319. {
  1320. ReleaseCapture();
  1321. if (bHigh)
  1322. {
  1323. DrawScrollButton(&dc, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP);
  1324. bHigh = FALSE;
  1325. }
  1326. m_bUpArrow = FALSE;
  1327. CPoint pt(msg.lParam);
  1328. if (m_rcUpArrow.PtInRect(pt))
  1329. {
  1330. if (m_nFirstItem > 0)
  1331. {
  1332. m_nFirstItem--;
  1333. CRect crc;
  1334. GetInsideRect(crc);
  1335. InvalidateRect(crc, TRUE);
  1336. }
  1337. }
  1338. }
  1339. goto ExitLoop;
  1340. case WM_TIMER:
  1341. {
  1342. if (msg.wParam == 2)
  1343. {
  1344. if (bHigh)
  1345. {
  1346. CPoint pt(msg.pt);
  1347. ScreenToClient(&pt);
  1348. if (m_rcUpArrow.PtInRect(pt))
  1349. {
  1350. m_bUpPressed = TRUE;
  1351. if (m_nFirstItem > 0)
  1352. {
  1353. m_nFirstItem--;
  1354. CRect crc;
  1355. GetInsideRect(crc);
  1356. InvalidateRect(crc, TRUE);
  1357. }
  1358. else goto ExitLoop;
  1359. }
  1360. else m_bUpPressed = FALSE;
  1361. }
  1362. }
  1363. break;
  1364. }
  1365. case WM_KEYDOWN:
  1366. if (msg.wParam != VK_ESCAPE) break;
  1367. default:
  1368. DispatchMessage(&msg);
  1369. break;
  1370. }
  1371. }
  1372. ExitLoop:
  1373. KillTimer(2);
  1374. ReleaseCapture();
  1375. AfxUnlockTempMaps(FALSE);
  1376. m_bLooping = FALSE;
  1377. m_bUpPressed = FALSE;
  1378. m_bUpArrow = FALSE;
  1379. CRect crc;
  1380. GetInsideRect(crc);
  1381. InvalidateRect(crc, TRUE);
  1382. }
  1383. }
  1384. CWnd::OnLButtonDown(nFlags, point);
  1385. }
  1386. int CXTOutBarCtrl::InsertItem(const int iFolder, const int iIndex, LPCTSTR lpszItemName, const int iImage, const DWORD dwData)
  1387. {
  1388. if (m_arFolder.GetSize() != 0)
  1389. {
  1390. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  1391. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1392. return pBarFolder->InsertItem(iIndex, lpszItemName, iImage, dwData);
  1393. }
  1394. return -1;
  1395. }
  1396. int CXTOutBarCtrl::GetItemCount() const
  1397. {
  1398. if (m_arFolder.GetSize() != 0)
  1399. {
  1400. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  1401. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  1402. return  pBarFolder->GetItemCount();
  1403. }
  1404. return 0;
  1405. }
  1406. void CXTOutBarCtrl::SetSelFolder(const int iIndex)
  1407. {
  1408. if (m_arFolder.GetSize() != 0)
  1409. {
  1410. ASSERT(iIndex >= 0 && iIndex < GetFolderCount());
  1411. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  1412. XT_OUTBAR_INFO obi;
  1413. obi.nIndex = iIndex;
  1414. obi.nDragFrom = -1;
  1415. obi.nDragTo = -1;
  1416. obi.lpszText = (LPCTSTR)pBarFolder->GetName();
  1417. obi.bFolder = true;
  1418. LRESULT lResult = NotifyOwner(OBN_XT_FOLDERCHANGE, &obi);
  1419. if (lResult == TRUE)
  1420. {
  1421. CWnd* pWnd = GetFolderChild();
  1422. if (pWnd)
  1423. {
  1424. pWnd->ShowWindow(SW_HIDE);
  1425. }
  1426. m_nLastItemSelected = -1;
  1427. if (iIndex != m_nSelFolder && m_nSelFolder >= 0)
  1428. {
  1429. if ((m_dwFlags & OBS_XT_ANIMATION) && (m_nAnimationTickCount >= 0))
  1430. {
  1431. AnimateFolderScroll(m_nSelFolder, iIndex);
  1432. }
  1433. }
  1434. m_nSelFolder = iIndex;
  1435. m_nFirstItem = 0;
  1436. pWnd = GetFolderChild();
  1437. if (pWnd)
  1438. {
  1439. CRect rc;
  1440. GetInsideRect(rc);
  1441. pWnd->MoveWindow(rc);
  1442. pWnd->ShowWindow(SW_SHOW);
  1443. }
  1444. InvalidateRect(NULL);
  1445. }
  1446. }
  1447. }
  1448. int CXTOutBarCtrl::GetFolderCount() const
  1449. {
  1450. return (int)m_arFolder.GetSize();
  1451. }
  1452. int CXTOutBarCtrl::GetSelFolder() const
  1453. {
  1454. return m_nSelFolder;
  1455. }
  1456. void CXTOutBarCtrl::RemoveFolder(const int iIndex)
  1457. {
  1458. if (m_arFolder.GetSize() != 0)
  1459. {
  1460. ASSERT(iIndex >= 0 && iIndex < GetFolderCount());
  1461. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  1462. XT_OUTBAR_INFO obi;
  1463. obi.nIndex = iIndex;
  1464. obi.nDragFrom = -1;
  1465. obi.nDragTo = -1;
  1466. obi.lpszText = (LPCTSTR)pBarFolder->GetName();
  1467. obi.bFolder = true;
  1468. LRESULT lResult = NotifyOwner(OBN_XT_DELETEFOLDER, &obi);
  1469. if (lResult == TRUE)
  1470. {
  1471. if (m_nSelFolder >= iIndex)
  1472. {
  1473. m_nSelFolder = iIndex - 1;
  1474. }
  1475. if (m_nSelFolder < 0 && GetFolderCount() > 0)
  1476. {
  1477. m_nSelFolder = 0;
  1478. }
  1479. m_arFolder.RemoveAt(iIndex);
  1480. SAFE_DELETE(pBarFolder);
  1481. InvalidateRect(NULL);
  1482. }
  1483. }
  1484. }
  1485. CImageList* CXTOutBarCtrl::SetImageList(CImageList* pImageList, DWORD dwImageList)
  1486. {
  1487. CImageList* pOldImageList = NULL;
  1488. if (dwImageList & OBS_XT_SMALLICON)
  1489. {
  1490. pOldImageList = m_pSmallImageList;
  1491. m_pSmallImageList = pImageList;
  1492. }
  1493. else if (dwImageList & OBS_XT_LARGEICON)
  1494. {
  1495. pOldImageList = m_pLargeImageList;
  1496. m_pLargeImageList = pImageList;
  1497. }
  1498. return pOldImageList;
  1499. }
  1500. CImageList* CXTOutBarCtrl::GetImageList(DWORD dwImageList)
  1501. {
  1502. if (dwImageList & OBS_XT_SMALLICON)
  1503. {
  1504. return m_pSmallImageList;
  1505. }
  1506. else if (dwImageList & OBS_XT_LARGEICON)
  1507. {
  1508. return m_pLargeImageList;
  1509. }
  1510. return NULL;
  1511. }
  1512. CImageList* CXTOutBarCtrl::SetFolderImageList(const int iFolder, CImageList* pImageList, DWORD dwImageList)
  1513. {
  1514. if (m_arFolder.GetSize() != 0)
  1515. {
  1516. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  1517. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1518. CImageList* pOldImageList = NULL;
  1519. if (dwImageList & OBS_XT_SMALLICON)
  1520. {
  1521. pOldImageList = pBarFolder->GetSmallImageList();
  1522. pBarFolder->SetSmallImageList(pImageList);
  1523. }
  1524. else if (dwImageList & OBS_XT_LARGEICON)
  1525. {
  1526. pOldImageList = pBarFolder->GetLargeImageList();
  1527. pBarFolder->SetLargeImageList(pImageList);
  1528. }
  1529. return pOldImageList;
  1530. }
  1531. return NULL;
  1532. }
  1533. void CXTOutBarCtrl::PaintItems(CDC* pDC, const int iFolder, CRect rc)
  1534. {
  1535. if (m_arFolder.GetSize() != 0)
  1536. {
  1537. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  1538. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1539. int nItemCount = pBarFolder->GetItemCount();
  1540. CRgn rgn;
  1541. rgn.CreateRectRgnIndirect(&rc);
  1542. pDC->SelectClipRgn(&rgn);
  1543. for (int i = m_nFirstItem; i < nItemCount; i++)
  1544. {
  1545. CRect rcItem;
  1546. GetItemRect(iFolder, i, rcItem, &rc);
  1547. if (rcItem.top > rc.bottom)
  1548. break;
  1549. DrawItem(pDC, iFolder, rcItem, i);
  1550. }
  1551. pDC->SelectClipRgn(NULL);
  1552. //rgn.DeleteObject();
  1553. }
  1554. }
  1555. CSize CXTOutBarCtrl::GetItemSize(const int iFolder, const int iIndex, const RectItem iType)
  1556. {
  1557. if (m_arFolder.GetSize() != 0)
  1558. {
  1559. if (iFolder < 0 || iFolder >= GetFolderCount())
  1560. return CSize(0, 0);
  1561. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1562. if (iIndex < 0 || iIndex >= pBarFolder->GetItemCount())
  1563. return CSize(0, 0);
  1564. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  1565. CSize szImage(0, 0);
  1566. CSize szLabel(0, 0);
  1567. CSize szAll(0, 0);
  1568. if (pBarItem->GetIndex() >= 0)
  1569. {
  1570. if (iType != rectItemLabel)
  1571. {
  1572. CImageList* il = GetFolderImageList(iFolder, IsSmallIconView(iFolder));
  1573. ASSERT(il);
  1574. if (il)
  1575. {
  1576. IMAGEINFO ii;
  1577. il->GetImageInfo(pBarItem->GetIndex(), &ii);
  1578. szImage = CRect(ii.rcImage).Size();
  1579. }
  1580. }
  1581. }
  1582. if (pBarItem->GetName().GetLength())
  1583. {
  1584. if (iType != rectItemIcon)
  1585. {
  1586. CClientDC dc(this);
  1587. CFont* oft = (CFont*)dc.SelectObject(GetFontX());
  1588. if (IsSmallIconView(iFolder))
  1589. {
  1590. szLabel = dc.GetTextExtent(pBarItem->GetName(), pBarItem->GetName().GetLength());
  1591. }
  1592. else
  1593. {
  1594. CRect rc;
  1595. GetInsideRect(rc);
  1596. rc.bottom = rc.top;
  1597. dc.DrawText(pBarItem->GetName(), pBarItem->GetName().GetLength(), rc,
  1598. DT_CALCRECT | DT_CENTER | DT_VCENTER | DT_WORDBREAK | DT_END_ELLIPSIS);
  1599. szLabel = rc.Size();
  1600. }
  1601. dc.SelectObject(oft);
  1602. }
  1603. }
  1604. if (IsSmallIconView(iFolder))
  1605. {
  1606. if (iType == rectItemIcon) szAll = szImage;
  1607. else if (iType == rectItemLabel) szAll = szLabel;
  1608. else if (iType == rectItemBoth) szAll = CSize(szImage.cx + szLabel.cx + m_sizeOffset.cx, szImage.cy > szLabel.cy ? szImage.cy : szLabel.cy);
  1609. }
  1610. else
  1611. {
  1612. if (iType == rectItemIcon) szAll = szImage;
  1613. else if (iType == rectItemLabel) szAll = szLabel;
  1614. else if (iType == rectItemBoth)
  1615. {
  1616. szAll = CSize(szImage.cx > szLabel.cx ? szImage.cx : szLabel.cx, szLabel.cy + szImage.cy + m_sizeOffset.cy + m_nIconSpacingLarge);
  1617. }
  1618. }
  1619. return szAll;
  1620. }
  1621. return CSize(0, 0);
  1622. }
  1623. CImageList* CXTOutBarCtrl::GetFolderImageList(const int iIndex, const BOOL bSmall) const
  1624. {
  1625. if (m_arFolder.GetSize() != 0)
  1626. {
  1627. ASSERT(iIndex >= 0 && iIndex < GetFolderCount());
  1628. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  1629. if (bSmall)
  1630. {
  1631. if (pBarFolder->GetSmallImageList())
  1632. return pBarFolder->GetSmallImageList();
  1633. else
  1634. return m_pSmallImageList;
  1635. }
  1636. if (pBarFolder->GetLargeImageList())
  1637. return pBarFolder->GetLargeImageList();
  1638. else
  1639. return m_pLargeImageList;
  1640. }
  1641. return NULL;
  1642. }
  1643. void CXTOutBarCtrl::DrawScrollButton(CDC* pDC, CRect rect, UINT uType, UINT uState)
  1644. {
  1645. pDC->DrawFrameControl(&rect, uType, uState);
  1646. }
  1647. void CXTOutBarCtrl::DrawItem(CDC* pDC, const int iFolder, CRect rc, const int iIndex, const BOOL bOnlyImage)
  1648. {
  1649. if (m_arFolder.GetSize() != 0)
  1650. {
  1651. CImageList* pImageList = GetFolderImageList(iFolder, IsSmallIconView(iFolder));
  1652. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  1653. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1654. ASSERT(iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  1655. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  1656. ASSERT(pBarItem && pImageList);
  1657. if (!pBarItem)
  1658. return;
  1659. CFont* oft = (CFont*)pDC->SelectObject(GetFontX());
  1660. pDC->SetBkMode(TRANSPARENT);
  1661. CRect rcFolder;
  1662. GetFolderRect(iFolder, rcFolder);
  1663. rc.left = rcFolder.left +4;
  1664. rc.right = rcFolder.right-4;
  1665. // Exclude the up and down arrows from painting.
  1666. if (m_bUpArrow)
  1667. pDC->ExcludeClipRect(&m_rcUpArrow);
  1668. if (m_bDownArrow)
  1669. pDC->ExcludeClipRect(&m_rcDownArrow);
  1670. // set the items text color.
  1671. pDC->SetTextColor(GetTheme()->m_clrText);
  1672. if ((m_dwFlags & OBS_XT_SELHIGHLIGHT) && (m_nLastItemSelected == iIndex) && (m_nLastItemSelected >= 0))
  1673. {
  1674. CRect rcIcon;
  1675. GetIconRect(m_nSelFolder, m_nLastItemSelected, rcIcon);
  1676. rcIcon.InflateRect(1, 1);
  1677. GetTheme()->DrawIconFrame(pDC, rcIcon, TRUE, FALSE);
  1678. }
  1679. if (pImageList && pBarItem)
  1680. {
  1681. IMAGEINFO ii;
  1682. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  1683. CSize szImage = CRect(ii.rcImage).Size();
  1684. CPoint pt;
  1685. if (IsSmallIconView(iFolder))
  1686. {
  1687. pt.x = rc.left + 2;
  1688. pt.y = rc.top + (rc.Height() - szImage.cy) / 2;
  1689. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  1690. if (!bOnlyImage)
  1691. {
  1692. rc.left += (szImage.cx + m_sizeOffset.cx);
  1693. DrawItemText(pDC, rc, pBarItem, DT_VCENTER | DT_SINGLELINE | DT_END_ELLIPSIS);
  1694. }
  1695. }
  1696. else
  1697. {
  1698. pt.x = rc.left + (rc.Width() - szImage.cx) / 2;
  1699. pt.y = rc.top;
  1700. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  1701. if (!bOnlyImage)
  1702. {
  1703. rc.top += szImage.cy + m_sizeOffset.cy;
  1704. rc.bottom += 4;
  1705. DrawItemText(pDC, rc, pBarItem, DT_CENTER | DT_VCENTER | DT_WORDBREAK | DT_END_ELLIPSIS);
  1706. }
  1707. }
  1708. }
  1709. pDC->SelectObject(oft);
  1710. }
  1711. }
  1712. BOOL CXTOutBarCtrl::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
  1713. {
  1714. CPoint pt(GetMessagePos());
  1715. ScreenToClient(&pt);
  1716. int iIndex, ht = HitTestEx(pt, iIndex);
  1717. if (ht == hitFolder)
  1718. {
  1719. ::SetCursor(XTAuxData().hcurHand);
  1720. return TRUE;
  1721. }
  1722. return CWnd::OnSetCursor(pWnd, nHitTest, message);
  1723. }
  1724. void CXTOutBarCtrl::GetVisibleRange(const int iFolder, int& iFirst, int& iLast)
  1725. {
  1726. iFirst = m_nFirstItem;
  1727. CRect rcInside;
  1728. GetInsideRect(rcInside);
  1729. if (m_arFolder.GetSize() != 0)
  1730. {
  1731. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1732. if (pBarFolder == NULL)
  1733. return;
  1734. int iCount = pBarFolder->GetItemCount();
  1735. if (iCount <= 0)
  1736. {
  1737. iFirst = iLast = -1;
  1738. return;
  1739. }
  1740. int iItem;
  1741. for (iItem = m_nFirstItem; iItem < iCount; iItem++)
  1742. {
  1743. CRect rcItem;
  1744. GetItemRect(iFolder, iItem, rcItem);
  1745. if (rcItem.bottom > rcInside.bottom)
  1746. {
  1747. iLast = iItem - 1;
  1748. break;
  1749. }
  1750. else
  1751. {
  1752. iLast = iItem;
  1753. }
  1754. }
  1755. }
  1756. }
  1757. void CXTOutBarCtrl::OnSize(UINT nType, int cx, int cy)
  1758. {
  1759. m_bUpArrow = m_bDownArrow = FALSE;
  1760. CWnd::OnSize(nType, cx, cy);
  1761. int t, max = GetFolderCount();
  1762. CRect rc;
  1763. GetInsideRect(rc);
  1764. for (t = 0; t < max; t++)
  1765. {
  1766. CWnd* pWnd = GetFolderChild(t);
  1767. if (pWnd) pWnd->SetWindowPos(0, rc.left, rc.top, rc.Width(), rc.Height(), SWP_NOZORDER);
  1768. }
  1769. }
  1770. void CXTOutBarCtrl::DrawIcon(CDC* pDC, int iIcon, int iFolder, bool bHilight)
  1771. {
  1772. // Clear the icon background
  1773. CRect rcIcon;
  1774. GetIconRect(iFolder, iIcon, rcIcon);
  1775. if (!IsSmallIconView(iFolder))
  1776. {
  1777. rcIcon.InflateRect(1, 1);
  1778. }
  1779. else
  1780. {
  1781. rcIcon.left--;
  1782. rcIcon.top--;
  1783. }
  1784. GetTheme()->FillInsideRect(pDC, rcIcon);
  1785. // Should we highlight the icon ?
  1786. if (bHilight)
  1787. {
  1788. GetTheme()->DrawIconFrame(pDC, rcIcon, FALSE, m_bIconPressed);
  1789. }
  1790. // Draw the icon using the image list.
  1791. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  1792. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  1793. ASSERT(iIcon >= 0 && iIcon < pBarFolder->GetItemCount());
  1794. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIcon);
  1795. CImageList* pImageList = GetFolderImageList(iFolder, IsSmallIconView(iFolder));
  1796. ASSERT(pBarItem && pImageList);
  1797. CRect rc;
  1798. GetItemRect(iFolder, iIcon, rc);
  1799. CRect rcFolder;
  1800. GetFolderRect(iFolder, rcFolder);
  1801. rc.left = rcFolder.left +4;
  1802. rc.right = rcFolder.right-4;
  1803. if (IsSmallIconView(iFolder))
  1804. {
  1805. if (pImageList && pBarItem)
  1806. {
  1807. IMAGEINFO ii;
  1808. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  1809. CSize szImage = CRect(ii.rcImage).Size();
  1810. CPoint pt;
  1811. pt.x = rc.left + 2;
  1812. pt.y = rc.top + (rc.Height() - szImage.cy) / 2;
  1813. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  1814. }
  1815. }
  1816. else
  1817. {
  1818. if (pImageList && pBarItem)
  1819. {
  1820. IMAGEINFO ii;
  1821. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  1822. CSize szImage = CRect(ii.rcImage).Size();
  1823. CPoint pt;
  1824. pt.x = rc.left + (rc.Width() - szImage.cx) / 2;
  1825. pt.y = rc.top;// + (rc.Height() - szImage.cy) / 2;
  1826. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  1827. }
  1828. }
  1829. }
  1830. void CXTOutBarCtrl::HighlightItem(CDC* pDC, const int iIndex, const BOOL bPressed)
  1831. {
  1832. static BOOL bOldPressed = FALSE;
  1833. CWnd* pFocus = GetFocus();
  1834. if (pFocus != NULL && pFocus != this && IsChild(pFocus))
  1835. {
  1836. return;
  1837. }
  1838. if (m_nItemHilighted == iIndex && bOldPressed == bPressed)
  1839. {
  1840. return;
  1841. }
  1842. if (iIndex != -1)
  1843. {
  1844. CXTOutBarItem* pBarItem = GetBarFolderItem(m_nSelFolder, iIndex);
  1845. if (!pBarItem->IsEnabled())
  1846. {
  1847. return;
  1848. }
  1849. }
  1850. m_bIconPressed = FALSE;
  1851. bOldPressed = bPressed;
  1852. XT_OUTBAR_INFO obi;
  1853. obi.nIndex = iIndex;
  1854. obi.nDragFrom = -1;
  1855. obi.nDragTo = -1;
  1856. obi.lpszText = GetItemText(iIndex);
  1857. obi.bFolder = false;
  1858. LRESULT lResult = NotifyOwner(OBN_XT_ITEMHOVER, &obi);
  1859. if (!lResult)
  1860. return;
  1861. CDC dcClient;
  1862. if (!pDC)
  1863. {
  1864. dcClient.Attach(::GetDC(m_hWnd));
  1865. pDC = &dcClient;
  1866. }
  1867. CRect irc;
  1868. GetInsideRect(irc);
  1869. CRgn rgn;
  1870. rgn.CreateRectRgnIndirect(&irc);
  1871. pDC->SelectClipRgn(&rgn);
  1872. // Exclude the up and down arrows from painting.
  1873. if (m_bUpArrow)
  1874. pDC->ExcludeClipRect(&m_rcUpArrow);
  1875. if (m_bDownArrow)
  1876. pDC->ExcludeClipRect(&m_rcDownArrow);
  1877. if (m_nItemHilighted >= 0 && IsValidItem(m_nItemHilighted))
  1878. {
  1879. DrawIcon(pDC, m_nItemHilighted, m_nSelFolder, false);
  1880. }
  1881. if ((m_nSelAnimTiming > 0 && iIndex == m_nLastItemSelected) ||
  1882. ((m_dwFlags & OBS_XT_SELHIGHLIGHT) && (m_nLastItemSelected == iIndex)))
  1883. {
  1884. m_nItemHilighted = -1;
  1885. }
  1886. else
  1887. {
  1888. m_nItemHilighted = iIndex;
  1889. m_bIconPressed = bPressed;
  1890. if (m_nItemHilighted >= 0 && IsValidItem(m_nItemHilighted))
  1891. {
  1892. DrawIcon(pDC, m_nItemHilighted, m_nSelFolder, true);
  1893. }
  1894. }
  1895. pDC->SelectClipRgn(NULL);
  1896. rgn.DeleteObject();
  1897. if (dcClient.GetSafeHdc())
  1898. {
  1899. ::ReleaseDC(m_hWnd, dcClient.Detach());
  1900. }
  1901. }
  1902. void CXTOutBarCtrl::GetIconRect(const int iFolder, const int iIndex, CRect& rect, CRect* pInsideRect /*= NULL*/)
  1903. {
  1904. CRect rc;
  1905. if (pInsideRect)
  1906. rc = *pInsideRect;
  1907. else
  1908. GetInsideRect(rc);
  1909. int top = rc.top;
  1910. CSize sz(0, 0);
  1911. int y = 0;
  1912. int spacing = IsSmallIconView(iFolder) ? m_nIconSpacingSmall : m_nIconSpacingLarge;
  1913. int t;
  1914. for (t = 0; t < iIndex; t++)
  1915. {
  1916. sz = GetItemSize(iFolder, t, rectItemBoth);
  1917. top += sz.cy;
  1918. top += spacing;
  1919. if (t == m_nFirstItem - 1) y = top - rc.top;
  1920. }
  1921. top += spacing;
  1922. sz = GetItemSize(iFolder, iIndex, rectItemIcon);
  1923. if (IsSmallIconView(iFolder))
  1924. {
  1925. rect.SetRect(rc.left, top, rc.left + sz.cx, top + sz.cy);
  1926. rect.left += m_sizeMargin.cx + 2;
  1927. rect.right += m_sizeMargin.cx + 2;
  1928. rect.top -= m_sizeMargin.cy;
  1929. rect.bottom -= m_sizeMargin.cy;
  1930. rect.top -= y;
  1931. rect.bottom -= y;
  1932. }
  1933. else
  1934. {
  1935. rect.SetRect(rc.left + (rc.Width() - sz.cx)/2, top, rc.left + (rc.Width() - sz.cx)/2 + sz.cx, top + sz.cy);
  1936. rect.top -= y + 3;
  1937. rect.bottom -= y + 2;
  1938. }
  1939. }
  1940. void CXTOutBarCtrl::GetLabelRect(const int iFolder, const int iIndex, CRect& rect)
  1941. {
  1942. CRect rc;
  1943. GetInsideRect(rc);
  1944. int top = rc.top;
  1945. CSize sz(0, 0);
  1946. int y = 0;
  1947. int spacing = IsSmallIconView(iFolder) ? m_nIconSpacingSmall : m_nIconSpacingLarge;
  1948. int t;
  1949. for (t = 0; t < iIndex; t++)
  1950. {
  1951. sz = GetItemSize(iFolder, t, rectItemBoth);
  1952. top += sz.cy;
  1953. top += spacing;
  1954. if (t == m_nFirstItem - 1) y = top - rc.top;
  1955. }
  1956. top += spacing;
  1957. sz = GetItemSize(iFolder, iIndex, rectItemBoth);
  1958. CSize szLabel = GetItemSize(iFolder, iIndex, rectItemLabel);
  1959. if (IsSmallIconView(iFolder))
  1960. {
  1961. rect.SetRect(rc.left, top, rc.left + sz.cx, top + sz.cy);
  1962. rect.left += m_sizeMargin.cx + 2;
  1963. rect.right += m_sizeMargin.cx + 2;
  1964. rect.top += m_sizeMargin.cy - 5;
  1965. rect.bottom += m_sizeMargin.cy - 5;
  1966. }
  1967. else
  1968. {
  1969. rect.SetRect(rc.left + (rc.Width() - sz.cx)/2, top, rc.left + (rc.Width() - sz.cx)/2 + sz.cx, top + sz.cy);
  1970. rect.top -= y + 3;
  1971. rect.bottom -= y + 2;
  1972. rect.bottom -= m_nIconSpacingLarge;
  1973. rect.top = rect.bottom - szLabel.cy;
  1974. }
  1975. }
  1976. void CXTOutBarCtrl::StartGroupEdit(const int iIndex)
  1977. {
  1978. CRect rc;
  1979. GetFolderRect(iIndex, rc);
  1980. rc.InflateRect(-2, -2);
  1981. if (m_arFolder.GetSize() != 0)
  1982. {
  1983. ASSERT(iIndex >= 0 && iIndex < GetFolderCount());
  1984. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  1985. CXTOutBarEditItem* pEdit = new CXTOutBarEditItem;
  1986. if (pEdit->Create(pBarFolder->GetName(), WS_CHILD | WS_VISIBLE | ES_CENTER | ES_AUTOHSCROLL,
  1987. rc, this, 0xFF, iIndex, true, false))
  1988. {
  1989. pEdit->Initialize(this);
  1990. pEdit->ModifyStyleEx(0, WS_EX_CLIENTEDGE, SWP_FRAMECHANGED);
  1991. pEdit->SetFocus();
  1992. pEdit->SetSel(0, -1);
  1993. }
  1994. }
  1995. }
  1996. void CXTOutBarCtrl::StartItemEdit(const int iIndex)
  1997. {
  1998. if (m_arFolder.GetSize() != 0)
  1999. {
  2000. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2001. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2002. ASSERT(iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  2003. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  2004. if (pBarItem == NULL)
  2005. {
  2006. return;
  2007. }
  2008. CRect rcLabel;
  2009. GetLabelRect(m_nSelFolder, iIndex, rcLabel);
  2010. DWORD dwStyle = WS_CHILD | WS_VISIBLE | WS_BORDER | ES_AUTOHSCROLL;
  2011. bool bNoDown = false;
  2012. if (IsSmallIconView(m_nSelFolder))
  2013. {
  2014. bNoDown = true;
  2015. CRect rcIcon;
  2016. GetIconRect(m_nSelFolder, iIndex, rcIcon);
  2017. rcLabel.OffsetRect(rcIcon.Width()+1, -6);
  2018. rcLabel.InflateRect(0, 1);
  2019. }
  2020. else
  2021. {
  2022. CRect rcInside;
  2023. GetInsideRect(rcInside);
  2024. CWindowDC dc(NULL);
  2025. CFont* f = dc.SelectObject(GetFontX());
  2026. CSize sz = dc.GetTextExtent(pBarItem->GetName());
  2027. dc.SelectObject(f);
  2028. int x = (rcInside.Width()-sz.cx)/2;
  2029. rcLabel.left = rcInside.left + x;
  2030. rcLabel.right = rcLabel.left + sz.cx;
  2031. rcLabel.InflateRect(6, 2);
  2032. rcLabel.OffsetRect(0, -1);
  2033. dwStyle |= ES_CENTER;
  2034. }
  2035. CXTOutBarEditItem* pEdit = new CXTOutBarEditItem;
  2036. if (pEdit->Create(pBarItem->GetName(), dwStyle, rcLabel, this,
  2037. 0xFF, iIndex, false, bNoDown))
  2038. {
  2039. pEdit->Initialize(this);
  2040. pEdit->SetFocus();
  2041. pEdit->SetSel(0, -1);
  2042. Invalidate();
  2043. }
  2044. }
  2045. }
  2046. void CXTOutBarCtrl::OnLargeIcon()
  2047. {
  2048. SetSmallIconView(FALSE, m_nSelFolder);
  2049. Invalidate();
  2050. }
  2051. void CXTOutBarCtrl::OnUpdateLargeIcon(CCmdUI* pCmdUI)
  2052. {
  2053. pCmdUI->Enable(IsSmallIconView(m_nSelFolder));
  2054. }
  2055. void CXTOutBarCtrl::OnSmallIcon()
  2056. {
  2057. SetSmallIconView(TRUE, m_nSelFolder);
  2058. Invalidate();
  2059. }
  2060. void CXTOutBarCtrl::OnUpdateSmallIcon(CCmdUI* pCmdUI)
  2061. {
  2062. pCmdUI->Enable(!IsSmallIconView(m_nSelFolder));
  2063. }
  2064. void CXTOutBarCtrl::OnRemoveItem()
  2065. {
  2066. if (m_nHitInternal1 == hitFolder)
  2067. {
  2068. RemoveFolder(m_nHitInternal2);
  2069. }
  2070. else if (m_nHitInternal1 == hitItem)
  2071. {
  2072. RemoveItem(m_nHitInternal2);
  2073. }
  2074. }
  2075. void CXTOutBarCtrl::OnUpdateRemoveItem(CCmdUI* /*pCmdUI*/)
  2076. {
  2077. }
  2078. void CXTOutBarCtrl::OnRenameItem()
  2079. {
  2080. if (m_nHitInternal1 == hitFolder)
  2081. {
  2082. StartGroupEdit(m_nHitInternal2);
  2083. }
  2084. else if (m_nHitInternal1 == hitItem)
  2085. {
  2086. StartItemEdit(m_nHitInternal2);
  2087. }
  2088. }
  2089. void CXTOutBarCtrl::OnUpdateRenameItem(CCmdUI* /*pCmdUI*/)
  2090. {
  2091. }
  2092. void CXTOutBarCtrl::RemoveAllItems(int iFolder, bool bNofify/*= false*/)
  2093. {
  2094. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  2095. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2096. if (pBarFolder != NULL)
  2097. {
  2098. int iFolderCount = pBarFolder->GetItemCount();
  2099. int iIndex;
  2100. for (iIndex = iFolderCount-1; iIndex >= 0; --iIndex)
  2101. {
  2102. CXTOutBarItem* pBarItem = pBarFolder->RemoveItemAt(iIndex);
  2103. if (pBarItem == NULL)
  2104. {
  2105. continue;
  2106. }
  2107. if (bNofify)
  2108. {
  2109. XT_OUTBAR_INFO obi;
  2110. obi.nIndex = iIndex;
  2111. obi.nDragFrom = -1;
  2112. obi.nDragTo = -1;
  2113. obi.lpszText = pBarItem->GetName();
  2114. obi.bFolder = false;
  2115. NotifyOwner(OBN_XT_DELETEITEM, &obi);
  2116. }
  2117. SAFE_DELETE(pBarItem);
  2118. }
  2119. Invalidate();
  2120. UpdateWindow();
  2121. }
  2122. }
  2123. void CXTOutBarCtrl::RemoveItem(const int iIndex)
  2124. {
  2125. if (m_arFolder.GetSize() != 0)
  2126. {
  2127. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2128. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2129. ASSERT(iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  2130. if (IsValidItem(iIndex))
  2131. {
  2132. XT_OUTBAR_INFO obi;
  2133. obi.nIndex = iIndex;
  2134. obi.nDragFrom = -1;
  2135. obi.nDragTo = -1;
  2136. obi.lpszText = GetItemText(iIndex);
  2137. obi.bFolder = false;
  2138. LRESULT lResult = NotifyOwner(OBN_XT_DELETEITEM, &obi);
  2139. if (lResult == TRUE)
  2140. {
  2141. CXTOutBarItem* pBarItem = pBarFolder->RemoveItemAt(iIndex);
  2142. SAFE_DELETE(pBarItem);
  2143. if (m_nLastItemSelected == iIndex)
  2144. m_nLastItemSelected = -1;
  2145. InvalidateRect(NULL);
  2146. }
  2147. }
  2148. }
  2149. }
  2150. BOOL CXTOutBarCtrl::IsValidItem(const int iIndex) const
  2151. {
  2152. if (m_arFolder.GetSize() != 0)
  2153. {
  2154. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2155. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2156. return (iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  2157. }
  2158. return FALSE;
  2159. }
  2160. DWORD CXTOutBarCtrl::GetItemData(const int iIndex) const
  2161. {
  2162. if (m_arFolder.GetSize() != 0)
  2163. {
  2164. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2165. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2166. if (IsValidItem(iIndex))
  2167. {
  2168. CXTOutBarItem* i = pBarFolder->GetItemAt(iIndex);
  2169. return i->GetData();
  2170. }
  2171. }
  2172. return 0;
  2173. }
  2174. int CXTOutBarCtrl::GetItemImage(const int iIndex) const
  2175. {
  2176. if (m_arFolder.GetSize() != 0)
  2177. {
  2178. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2179. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2180. if (IsValidItem(iIndex))
  2181. {
  2182. CXTOutBarItem* i = pBarFolder->GetItemAt(iIndex);
  2183. return i->GetIndex();
  2184. }
  2185. }
  2186. return 0;
  2187. }
  2188. void CXTOutBarCtrl::SetItemData(const int iIndex, const DWORD dwData)
  2189. {
  2190. if (m_arFolder.GetSize() != 0)
  2191. {
  2192. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2193. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2194. if (IsValidItem(iIndex))
  2195. {
  2196. CXTOutBarItem* i = pBarFolder->GetItemAt(iIndex);
  2197. i->SetData(dwData);
  2198. }
  2199. }
  2200. }
  2201. void CXTOutBarCtrl::SetItemImage(const int iIndex, const int iImage)
  2202. {
  2203. if (m_arFolder.GetSize() != 0)
  2204. {
  2205. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2206. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2207. if (IsValidItem(iIndex))
  2208. {
  2209. CXTOutBarItem* i = pBarFolder->GetItemAt(iIndex);
  2210. i->SetIndex(iImage);
  2211. }
  2212. }
  2213. }
  2214. void CXTOutBarCtrl::DrawDragArrow(CDC* pDC, const int /*iFrom*/, const int iTo)
  2215. {
  2216. if (iTo == m_nLastDragItemDraw)
  2217. return;
  2218. CRect rc1;
  2219. if (m_nLastDragItemDraw >= 0)
  2220. {
  2221. GetDragItemRect(m_nLastDragItemDraw, rc1);
  2222. if (m_nLastDragItemDrawType == 0)
  2223. {
  2224. rc1.bottom = rc1.top + 5;
  2225. }
  2226. else if (m_nLastDragItemDrawType == 1)
  2227. {
  2228. rc1.top -= 4;
  2229. rc1.bottom = rc1.top + 9;
  2230. }
  2231. else if (m_nLastDragItemDrawType == 2)
  2232. {
  2233. rc1.top -= 4;
  2234. rc1.bottom = rc1.top + 5;
  2235. }
  2236. InvalidateRect(rc1, TRUE);
  2237. UpdateWindow();
  2238. }
  2239. m_nLastDragItemDraw = iTo;
  2240. if (m_nLastDragItemDraw >= 0)
  2241. {
  2242. m_nLastDragItemDrawType = GetDragItemRect(m_nLastDragItemDraw, rc1);
  2243. CPen* op = pDC->SelectObject(&m_penBlack);
  2244. pDC->MoveTo(rc1.left, rc1.top);
  2245. pDC->LineTo(rc1.right, rc1.top);
  2246. if (m_nLastDragItemDrawType != 2)
  2247. {
  2248. pDC->MoveTo(rc1.left, rc1.top + 1);
  2249. pDC->LineTo(rc1.left + 7, rc1.top + 1);
  2250. pDC->MoveTo(rc1.left, rc1.top + 2);
  2251. pDC->LineTo(rc1.left + 5, rc1.top + 2);
  2252. pDC->MoveTo(rc1.left, rc1.top + 3);
  2253. pDC->LineTo(rc1.left + 3, rc1.top + 3);
  2254. pDC->MoveTo(rc1.left, rc1.top + 4);
  2255. pDC->LineTo(rc1.left + 1, rc1.top + 4);
  2256. pDC->MoveTo(rc1.right-1, rc1.top + 1);
  2257. pDC->LineTo(rc1.right - 7, rc1.top + 1);
  2258. pDC->MoveTo(rc1.right-1, rc1.top + 2);
  2259. pDC->LineTo(rc1.right - 5, rc1.top + 2);
  2260. pDC->MoveTo(rc1.right-1, rc1.top + 3);
  2261. pDC->LineTo(rc1.right - 3, rc1.top + 3);
  2262. }
  2263. if (m_nLastDragItemDrawType != 0)
  2264. {
  2265. pDC->MoveTo(rc1.left, rc1.top-1);
  2266. pDC->LineTo(rc1.left + 7, rc1.top-1);
  2267. pDC->MoveTo(rc1.left, rc1.top-2);
  2268. pDC->LineTo(rc1.left + 5, rc1.top-2);
  2269. pDC->MoveTo(rc1.left, rc1.top-3);
  2270. pDC->LineTo(rc1.left + 3, rc1.top-3);
  2271. pDC->MoveTo(rc1.left, rc1.top-4);
  2272. pDC->LineTo(rc1.left + 1, rc1.top-4);
  2273. pDC->MoveTo(rc1.right-1, rc1.top-1);
  2274. pDC->LineTo(rc1.right - 7, rc1.top-1);
  2275. pDC->MoveTo(rc1.right-1, rc1.top-2);
  2276. pDC->LineTo(rc1.right - 5, rc1.top-2);
  2277. pDC->MoveTo(rc1.right-1, rc1.top-3);
  2278. pDC->LineTo(rc1.right - 3, rc1.top-3);
  2279. pDC->MoveTo(rc1.right-1, rc1.top-4);
  2280. pDC->LineTo(rc1.right - 1, rc1.top-4);
  2281. }
  2282. pDC->SelectObject(op);
  2283. }
  2284. }
  2285. int CXTOutBarCtrl::GetDragItemRect(const int iIndex, CRect& rect)
  2286. {
  2287. CRect rc, crc;
  2288. GetInsideRect(crc);
  2289. crc.InflateRect(-2, 0);
  2290. GetItemRect(m_nSelFolder, iIndex < GetItemCount() ? iIndex : iIndex - 1, rc);
  2291. int line = 0;
  2292. if (iIndex < GetItemCount())
  2293. {
  2294. line = rc.top - 4;
  2295. }
  2296. else
  2297. {
  2298. line = rc.bottom + 6;
  2299. }
  2300. int tpe;
  2301. if (iIndex == 0)
  2302. {
  2303. rect.SetRect(crc.left, line + 2, crc.right, line + 7);
  2304. tpe = 0;
  2305. }
  2306. else if (iIndex < GetItemCount())
  2307. {
  2308. rect.SetRect(crc.left, line - 9, crc.right, line);
  2309. tpe = 1;
  2310. if (IsSmallIconView(m_nSelFolder))
  2311. {
  2312. rect.top += 8;
  2313. rect.bottom += 8;
  2314. }
  2315. }
  2316. else
  2317. {
  2318. rect.SetRect(crc.left, line, crc.right, line + 5);
  2319. tpe = 2;
  2320. }
  2321. return tpe;
  2322. }
  2323. void CXTOutBarCtrl::DrawPreviewBitmap(CWnd* pWnd, CDC* pDC, int nOffset)
  2324. {
  2325. if (!pWnd || !::IsWindow(pWnd->m_hWnd))
  2326. return;
  2327. CPoint ovpt = pDC->SetViewportOrg(0, nOffset);
  2328. BOOL bPrev = pWnd->ShowWindow(SW_SHOW);
  2329. CXTPWindowRect rectWindow(pWnd);
  2330. CClientDC dc(pWnd);
  2331. CBitmap bitmapNonClient;
  2332. VERIFY(bitmapNonClient.CreateCompatibleBitmap(&dc, rectWindow.Width(), rectWindow.Height()));
  2333. CXTPCompatibleDC dcMemSourceNonClient(&dc, &bitmapNonClient);
  2334. dcMemSourceNonClient.FillSolidRect(0, 0, rectWindow.Width(), rectWindow.Height(), GetXtremeColor(COLOR_WINDOW));
  2335. pWnd->Print(&dcMemSourceNonClient, PRF_NONCLIENT);
  2336. pDC->BitBlt(0, 0, rectWindow.Width(), rectWindow.Height(), &dcMemSourceNonClient, 0, 0, SRCCOPY);
  2337. CPoint pLT(0, 0);
  2338. pWnd->ClientToScreen(&pLT);
  2339. CPoint pt = pDC->GetViewportOrg();
  2340. pDC->SetViewportOrg(pt.x + pLT.x - rectWindow.left, pt.y + pLT.y - rectWindow.top);
  2341. pWnd->Print(pDC, PRF_CHILDREN | PRF_CLIENT | PRF_ERASEBKGND);
  2342. if (!bPrev) pWnd->ShowWindow(SW_HIDE);
  2343. pDC->SetViewportOrg(ovpt);
  2344. }
  2345. void CXTOutBarCtrl::AnimateFolderScroll(const int iFrom, const int iTo)
  2346. {
  2347. if (m_arFolder.GetSize() == 0)
  2348. return;
  2349. ASSERT(iFrom >= 0 && iFrom < GetFolderCount());
  2350. ASSERT(iTo >= 0 && iTo < GetFolderCount());
  2351. CRect rc, frc;
  2352. GetInsideRect(rc);
  2353. GetFolderRect(iTo, frc);
  2354. int nFolderHeight = frc.Height();
  2355. CClientDC dc(this);
  2356. CDC memDC;
  2357. memDC.CreateCompatibleDC(&dc);
  2358. int nTopItem = iTo > iFrom ? iFrom : iTo;
  2359. int nBottomItem = iTo > iFrom ? iTo : iFrom;
  2360. CBitmap bmpFrom, bmpTo;
  2361. bmpFrom.CreateCompatibleBitmap(&dc, rc.Width(), rc.Height() + nFolderHeight * (nBottomItem - nTopItem));
  2362. bmpTo.CreateCompatibleBitmap(&dc, rc.Width(), rc.Height() + nFolderHeight * (nBottomItem - nTopItem));
  2363. CBitmap* pOldBitmap = (CBitmap*)memDC.SelectObject(&bmpFrom);
  2364. CRect rcFrom(0, 0, rc.Width(), rc.Height());
  2365. GetTheme()->FillInsideRect(&memDC, rcFrom);
  2366. CWnd* pWnd1 = GetFolderChild(nTopItem);
  2367. CWnd* pWnd2 = GetFolderChild(nBottomItem);
  2368. if (pWnd1)
  2369. {
  2370. DrawPreviewBitmap(pWnd1, &memDC, 0);
  2371. }
  2372. else PaintItems(&memDC, nTopItem, rcFrom);
  2373. memDC.SelectObject(&bmpTo);
  2374. CRect rcTo(rcFrom);
  2375. for (int i = nTopItem + 1; i <= nBottomItem; i++)
  2376. {
  2377. CRect rcFolder(frc.left, rcTo.top, frc.right, rcTo.top + nFolderHeight);
  2378. DrawFolder(&memDC, i, rcFolder, xtMouseNormal);
  2379. rcTo.OffsetRect(0, nFolderHeight);
  2380. }
  2381. GetTheme()->FillInsideRect(&memDC, rcTo);
  2382. if (pWnd2)
  2383. {
  2384. DrawPreviewBitmap(pWnd2, &memDC, rcTo.top);
  2385. }
  2386. else PaintItems(&memDC, nBottomItem, rcTo);
  2387. if (iTo > iFrom)
  2388. {
  2389. GetFolderRect(iFrom, rcFrom);
  2390. GetFolderRect(iTo, rcTo);
  2391. for (int y = rcTo.top - nFolderHeight; y > rcFrom.bottom; y -= nFolderHeight)
  2392. {
  2393. memDC.SelectObject(&bmpFrom);
  2394. dc.BitBlt(rc.left, rcFrom.bottom, rc.Width(), y - rcFrom.bottom, &memDC, 0, 0, SRCCOPY);
  2395. memDC.SelectObject(&bmpTo);
  2396. dc.BitBlt(rc.left, y, rc.Width(), rc.bottom - y + (iTo - iFrom) * nFolderHeight, &memDC, 0, 0, SRCCOPY);
  2397. Sleep(m_nAnimationTickCount);
  2398. }
  2399. }
  2400. else
  2401. {
  2402. nTopItem = iTo;
  2403. nBottomItem = iFrom;
  2404. GetFolderRect(nTopItem, rcFrom);
  2405. GetFolderRect(nBottomItem, rcTo);
  2406. rcTo.OffsetRect(0, rc.Height());
  2407. for (int y = rcFrom.bottom; y < rcTo.top - nFolderHeight ; y += nFolderHeight)
  2408. {
  2409. memDC.SelectObject(&bmpFrom);
  2410. dc.BitBlt(rc.left, rcFrom.bottom, rc.Width(), y - rcFrom.bottom, &memDC, 0, 0, SRCCOPY);
  2411. memDC.SelectObject(&bmpTo);
  2412. dc.BitBlt(rc.left, y, rc.Width(), rc.bottom - y , &memDC, 0, 0, SRCCOPY);
  2413. Sleep(m_nAnimationTickCount);
  2414. }
  2415. }
  2416. memDC.SelectObject(pOldBitmap);
  2417. }
  2418. CString CXTOutBarCtrl::GetItemText(const int iIndex)
  2419. {
  2420. CString item = _T("");
  2421. if (m_arFolder.GetSize() != 0)
  2422. {
  2423. ASSERT(m_nSelFolder >= 0 && m_nSelFolder < GetFolderCount());
  2424. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(m_nSelFolder);
  2425. if (IsValidItem(iIndex))
  2426. {
  2427. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  2428. if (pBarItem->GetName().GetLength())
  2429. item = pBarItem->GetName();
  2430. }
  2431. }
  2432. return item;
  2433. }
  2434. int CXTOutBarCtrl::AddFolderBar(LPCTSTR lpszFolderName, CWnd* pWndChild, const DWORD dwData)
  2435. {
  2436. CXTOutBarFolder* pBarFolder = new CXTOutBarFolder(lpszFolderName, dwData, this);
  2437. ASSERT(pBarFolder);
  2438. pBarFolder->SetChild(pWndChild);
  2439. return  (int)m_arFolder.Add((void*)pBarFolder);
  2440. }
  2441. CWnd* CXTOutBarCtrl::GetFolderChild(int iFolder)
  2442. {
  2443. if (GetFolderCount())
  2444. {
  2445. if (iFolder < 0 && m_nSelFolder < 0)
  2446. return NULL;
  2447. if (iFolder < 0)
  2448. iFolder = m_nSelFolder;
  2449. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2450. return pBarFolder->GetChild();
  2451. }
  2452. return NULL;
  2453. }
  2454. DWORD CXTOutBarCtrl::GetFolderData(int iFolder)
  2455. {
  2456. if (iFolder < 0) iFolder = m_nSelFolder;
  2457. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2458. return pBarFolder->GetData();
  2459. }
  2460. void CXTOutBarCtrl::SetAnimSelHighlight(const int iTime)
  2461. {
  2462. if (iTime <= 0) KillTimer(3);
  2463. else SetTimer(3, iTime, NULL);
  2464. m_nSelAnimTiming = iTime;
  2465. }
  2466. void CXTOutBarCtrl::OnRButtonUp(UINT nFlags, CPoint point)
  2467. {
  2468. XT_OUTBAR_INFO obi = { 0 };
  2469. int ht = HitTestEx(point, obi.nIndex);
  2470. if (ht != hitItem)
  2471. {
  2472. obi.bFolder = true;
  2473. if (ht != hitFolder)
  2474. {
  2475. obi.nIndex = -1;
  2476. }
  2477. }
  2478. // allow users to add their own right click handlers.
  2479. LRESULT lResult = NotifyOwner(OBN_XT_ITEMRCLICK, &obi);
  2480. if (lResult == TRUE)
  2481. {
  2482. m_nHitInternal1 = HitTestEx(point, m_nHitInternal2);
  2483. CPoint pt = point;
  2484. ClientToScreen(&pt);
  2485. CMenu menu;
  2486. VERIFY(XTPResourceManager()->LoadMenu(&menu, XT_IDM_POPUP));
  2487. CMenu* pPopup = menu.GetSubMenu(0);
  2488. ASSERT(pPopup != NULL);
  2489. if (!pPopup)
  2490. return;
  2491. CWnd* pWndPopupOwner = this;
  2492. CString string;
  2493. if (m_nHitInternal1 == hitItem)
  2494. {
  2495. if ((m_dwFlags & OBS_XT_REMOVEITEMS) || (m_dwFlags & OBS_XT_EDITITEMS))
  2496. {
  2497. pPopup->AppendMenu(MF_SEPARATOR);
  2498. if (m_dwFlags & OBS_XT_REMOVEITEMS)
  2499. {
  2500. VERIFY(XTPResourceManager()->LoadString(&string, XT_IDC_REMOVEITEM));
  2501. pPopup->AppendMenu(MF_STRING, XT_IDC_REMOVEITEM, string);
  2502. }
  2503. if (m_dwFlags & OBS_XT_EDITITEMS)
  2504. {
  2505. VERIFY(XTPResourceManager()->LoadString(&string, XT_IDC_RENAMEITEM));
  2506. pPopup->AppendMenu(MF_STRING, XT_IDC_RENAMEITEM, string);
  2507. }
  2508. }
  2509. }
  2510. else if (m_nHitInternal1 == hitFolder)
  2511. {
  2512. if ((m_dwFlags & OBS_XT_REMOVEGROUPS) || (m_dwFlags & OBS_XT_EDITGROUPS))
  2513. {
  2514. pPopup->AppendMenu(MF_SEPARATOR);
  2515. if (m_dwFlags & OBS_XT_REMOVEGROUPS)
  2516. {
  2517. VERIFY(XTPResourceManager()->LoadString(&string, XT_IDC_REMOVEGROUP));
  2518. pPopup->AppendMenu(MF_STRING, XT_IDC_REMOVEITEM, string);
  2519. }
  2520. if (m_dwFlags & OBS_XT_EDITGROUPS)
  2521. {
  2522. VERIFY(XTPResourceManager()->LoadString(&string, XT_IDC_RENAMEGROUP));
  2523. pPopup->AppendMenu(MF_STRING, XT_IDC_RENAMEITEM, string);
  2524. }
  2525. }
  2526. }
  2527. pPopup->CheckMenuItem(IsSmallIconView(m_nSelFolder) ?
  2528. XT_IDC_SMALLICON : XT_IDC_LARGEICON, MF_CHECKED | MF_BYCOMMAND);
  2529. XTFuncContextMenu(pPopup, TPM_LEFTALIGN | TPM_RIGHTBUTTON, pt.x, pt.y, pWndPopupOwner, XT_IDR_TBAR_OUT);
  2530. }
  2531. CWnd::OnRButtonUp(nFlags, point);
  2532. }
  2533. void CXTOutBarCtrl::OnDestroy()
  2534. {
  2535. // send notification for each folder / item that is deleted.
  2536. int iFolder;
  2537. for (iFolder = 0; iFolder < m_arFolder.GetSize(); iFolder++)
  2538. {
  2539. m_nSelFolder = iFolder;
  2540. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2541. XT_OUTBAR_INFO obif;
  2542. obif.nIndex = iFolder;
  2543. obif.nDragFrom = -1;
  2544. obif.nDragTo = -1;
  2545. obif.lpszText = pBarFolder->GetName();
  2546. obif.bFolder = true;
  2547. if (pBarFolder != NULL)
  2548. {
  2549. int iCount = (int)pBarFolder->m_barItems.GetCount();
  2550. int iItem;
  2551. for (iItem = 0; iItem < iCount; iItem++)
  2552. {
  2553. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iItem);
  2554. XT_OUTBAR_INFO obi;
  2555. obi.nIndex = iItem;
  2556. obi.nDragFrom = -1;
  2557. obi.nDragTo = -1;
  2558. obi.lpszText = pBarItem->GetName();
  2559. obi.bFolder = false;
  2560. NotifyOwner(OBN_XT_DELETEITEM, &obi);
  2561. }
  2562. NotifyOwner(OBN_XT_DELETEFOLDER, &obif);
  2563. }
  2564. }
  2565. CWnd::OnDestroy();
  2566. }
  2567. void CXTOutBarCtrl::EnableItem(int iFolder, int iItem, bool bEnable)
  2568. {
  2569. ASSERT(iFolder < m_arFolder.GetSize());
  2570. CXTOutBarItem* pBarItem = GetBarFolderItem(iFolder, iItem);
  2571. if (pBarItem)
  2572. {
  2573. pBarItem->EnableItem(bEnable);
  2574. }
  2575. }
  2576. void CXTOutBarCtrl::SetSelItem(int iFolder, int iItem, BOOL bEnsureVisible/*= TRUE*/)
  2577. {
  2578. if (iFolder >= 0 && iFolder < GetFolderCount())
  2579. {
  2580. // select the specified folder.
  2581. SetSelFolder(iFolder);
  2582. // get a pointer to the folder object.
  2583. CXTOutBarFolder* pBarFolder = GetBarFolder(iFolder);
  2584. int iItemCount = pBarFolder->GetItemCount();
  2585. if (pBarFolder && iItem >= 0 && iItem < iItemCount)
  2586. {
  2587. // select the specified item.
  2588. pBarFolder->SetSelItem(iItem);
  2589. m_nLastItemSelected = iItem;
  2590. if (bEnsureVisible)
  2591. {
  2592. int iFirst, iLast;
  2593. GetVisibleRange(iFolder, iFirst, iLast);
  2594. if (iLast != -1 && (iItem < iFirst || iItem > iLast))
  2595. {
  2596. m_nFirstItem = iItem-(iLast-iFirst);
  2597. }
  2598. }
  2599. }
  2600. }
  2601. }
  2602. /////////////////////////////////////////////////////////////////////////////
  2603. // CXTOutBarItem
  2604. /////////////////////////////////////////////////////////////////////////////
  2605. CXTOutBarItem::CXTOutBarItem(LPCTSTR lpszName, const int nImageIndex, DWORD dwData)
  2606. : m_strName(lpszName)
  2607. , m_nIndex(nImageIndex)
  2608. , m_dwData(dwData)
  2609. , m_bSelected(false)
  2610. , m_bEnabled(true)
  2611. {
  2612. }
  2613. CXTOutBarItem::~CXTOutBarItem()
  2614. {
  2615. }
  2616. /////////////////////////////////////////////////////////////////////////////
  2617. // CXTOutBarFolder
  2618. /////////////////////////////////////////////////////////////////////////////
  2619. CXTOutBarFolder::CXTOutBarFolder(LPCTSTR lpszName, DWORD dwData, CXTOutBarCtrl* pControl)
  2620. : m_strName(lpszName)
  2621. , m_dwData(dwData)
  2622. , m_pLargeList(NULL)
  2623. , m_pSmallList(NULL)
  2624. , m_pChild(NULL)
  2625. , m_pControl(pControl)
  2626. {
  2627. m_bSmallIcons = TRUE;
  2628. }
  2629. CXTOutBarFolder::~CXTOutBarFolder()
  2630. {
  2631. while (!m_barItems.IsEmpty())
  2632. {
  2633. CXTOutBarItem* pBarItem = m_barItems.RemoveTail();
  2634. SAFE_DELETE(pBarItem);
  2635. }
  2636. }
  2637. int CXTOutBarFolder::InsertItem(int iIndex, LPCTSTR lpszName, const int nImage, const DWORD dwData)
  2638. {
  2639. // if the iIndex is out of range, insert the item at the end
  2640. // of the list.
  2641. if ((iIndex < 0) || (iIndex > GetItemCount()))
  2642. {
  2643. iIndex = GetItemCount();
  2644. }
  2645. // allocate a new bar item.
  2646. CXTOutBarItem* pBarItem = new CXTOutBarItem(lpszName, nImage, dwData);
  2647. ASSERT(pBarItem);
  2648. // insert the new item into the item array.
  2649. InsertItemAt(iIndex, pBarItem);
  2650. return iIndex;
  2651. }
  2652. CXTOutBarItem* CXTOutBarFolder::GetItemAt(int iIndex)
  2653. {
  2654. POSITION pos = m_barItems.FindIndex(iIndex);
  2655. return m_barItems.GetAt(pos);
  2656. }
  2657. void CXTOutBarFolder::InsertItemAt(int iIndex, CXTOutBarItem* pBarItem)
  2658. {
  2659. POSITION pos = m_barItems.FindIndex(iIndex);
  2660. m_barItems.InsertAfter(pos, pBarItem);
  2661. }
  2662. CXTOutBarItem* CXTOutBarFolder::RemoveItemAt(int iIndex)
  2663. {
  2664. POSITION pos = m_barItems.FindIndex(iIndex);
  2665. if (pos != NULL)
  2666. {
  2667. CXTOutBarItem* pBarItem = m_barItems.GetAt(pos);
  2668. if (pBarItem != NULL)
  2669. {
  2670. m_barItems.RemoveAt(pos);
  2671. return pBarItem;
  2672. }
  2673. }
  2674. return NULL;
  2675. }
  2676. void CXTOutBarFolder::SetSelItem(int iItem)
  2677. {
  2678. POSITION pos;
  2679. for (pos = m_barItems.GetHeadPosition(); pos; m_barItems.GetNext(pos))
  2680. {
  2681. CXTOutBarItem* pBarItem = m_barItems.GetAt(pos);
  2682. ASSERT(pBarItem != NULL);
  2683. if (pBarItem)
  2684. {
  2685. pBarItem->SelectItem(false);
  2686. }
  2687. }
  2688. CXTOutBarItem* pBarItem = GetItemAt(iItem);
  2689. if (pBarItem != NULL)
  2690. {
  2691. pBarItem->SelectItem(true);
  2692. }
  2693. }
  2694. CXTOutBarItem* CXTOutBarFolder::GetSelItem()
  2695. {
  2696. POSITION pos;
  2697. for (pos = m_barItems.GetHeadPosition(); pos; m_barItems.GetNext(pos))
  2698. {
  2699. CXTOutBarItem* pBarItem = m_barItems.GetAt(pos);
  2700. if (pBarItem == NULL)
  2701. {
  2702. return NULL;
  2703. }
  2704. if (pBarItem && pBarItem->IsSelected())
  2705. {
  2706. return pBarItem;
  2707. }
  2708. }
  2709. return NULL;
  2710. }
  2711. int CXTOutBarFolder::GetSelIndex()
  2712. {
  2713. int i;
  2714. for (i = 0; i < m_barItems.GetCount(); ++i)
  2715. {
  2716. CXTOutBarItem* pBarItem = GetItemAt(i);
  2717. if (pBarItem == NULL)
  2718. {
  2719. return -1;
  2720. }
  2721. if (pBarItem && pBarItem->IsSelected())
  2722. {
  2723. return i;
  2724. }
  2725. }
  2726. return -1;
  2727. }
  2728. void CXTOutBarCtrl::SetBackColor(COLORREF clrBack)
  2729. {
  2730. GetTheme()->m_clrBack.SetCustomValue(clrBack);
  2731. }
  2732. void CXTOutBarCtrl::SetTextColor(COLORREF clrText)
  2733. {
  2734. GetTheme()->m_clrText.SetCustomValue(clrText);
  2735. }
  2736. /////////////////////////////////////////////////////////////////////////////
  2737. // CXTToolBox
  2738. /////////////////////////////////////////////////////////////////////////////
  2739. CXTToolBox::CXTToolBox()
  2740. {
  2741. m_bAnimating = false;
  2742. m_iFirst = 0;
  2743. m_iLast = 0;
  2744. }
  2745. CXTToolBox::~CXTToolBox()
  2746. {
  2747. }
  2748. IMPLEMENT_DYNCREATE(CXTToolBox, CXTOutBarCtrl)
  2749. BEGIN_MESSAGE_MAP(CXTToolBox, CXTOutBarCtrl)
  2750. //{{AFX_MSG_MAP(CXTToolBox)
  2751. ON_WM_RBUTTONUP()
  2752. //}}AFX_MSG_MAP
  2753. END_MESSAGE_MAP()
  2754. /////////////////////////////////////////////////////////////////////////////
  2755. // CXTToolBox message handlers
  2756. void CXTToolBox::GetItemRect(const int iFolder, const int iIndex, CRect& rect, CRect* pInsideRect /*= NULL*/)
  2757. {
  2758. GetIconRect(iFolder, iIndex, rect, pInsideRect);
  2759. }
  2760. void CXTToolBox::GetIconRect(const int iFolder, const int iIndex, CRect& rcIcon, CRect* pInsideRect /*= NULL*/)
  2761. {
  2762. CXTOutBarCtrl::GetIconRect(iFolder, iIndex, rcIcon, pInsideRect);
  2763. CRect rcInside;
  2764. if (pInsideRect)
  2765. rcInside = *pInsideRect;
  2766. else
  2767. GetInsideRect(rcInside);
  2768. rcIcon.left = rcInside.left;
  2769. rcIcon.right = rcInside.right;
  2770. rcIcon.InflateRect(-2, 4);
  2771. if ((m_bUpArrow && iIndex == m_iFirst) ||
  2772. (m_bDownArrow && iIndex == m_iLast) ||
  2773. (m_bDownArrow && iIndex == m_iLast-1))
  2774. {
  2775. rcIcon.right -= XTAuxData().cxHThumb + 6;
  2776. }
  2777. }
  2778. void CXTToolBox::DrawIcon(CDC* pDC, int iIcon, int iFolder, bool bHilight)
  2779. {
  2780. CRect rcIcon;
  2781. GetIconRect(iFolder, iIcon, rcIcon);
  2782. pDC->Draw3dRect(rcIcon, GetTheme()->m_clrBack, GetTheme()->m_clrBack);
  2783. // Should we highlight the icon ?
  2784. if (bHilight)
  2785. {
  2786. if (m_bIconPressed)
  2787. {
  2788. pDC->Draw3dRect(rcIcon,
  2789. GetXtremeColor(COLOR_3DSHADOW), GetXtremeColor(COLOR_3DHILIGHT));
  2790. }
  2791. else
  2792. {
  2793. pDC->Draw3dRect(rcIcon,
  2794. GetXtremeColor(COLOR_3DHILIGHT), GetXtremeColor(COLOR_3DSHADOW));
  2795. }
  2796. }
  2797. // Draw the icon using the image list.
  2798. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  2799. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2800. ASSERT(iIcon >= 0 && iIcon < pBarFolder->GetItemCount());
  2801. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIcon);
  2802. CImageList* pImageList = GetFolderImageList(iFolder, IsSmallIconView(iFolder));
  2803. ASSERT(pBarItem && pImageList);
  2804. CRect rc;
  2805. GetItemRect(iFolder, iIcon, rc);
  2806. CRect rcFolder;
  2807. GetFolderRect(iFolder, rcFolder);
  2808. rc.left = rcFolder.left +4;
  2809. rc.right = rcFolder.right-4;
  2810. if (pImageList && pBarItem)
  2811. {
  2812. IMAGEINFO ii;
  2813. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  2814. CSize szImage = CRect(ii.rcImage).Size();
  2815. CPoint pt;
  2816. pt.x = rc.left + 2;
  2817. pt.y = rc.top + (rc.Height() - szImage.cy) / 2;
  2818. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  2819. }
  2820. }
  2821. void CXTToolBox::DrawItem(CDC* pDC, const int iFolder, CRect rc, const int iIndex, const BOOL bOnlyImage)
  2822. {
  2823. if (m_arFolder.GetSize() != 0)
  2824. {
  2825. CImageList* pImageList = GetFolderImageList(iFolder, IsSmallIconView(iFolder));
  2826. ASSERT(iFolder >= 0 && iFolder < GetFolderCount());
  2827. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iFolder);
  2828. ASSERT(iIndex >= 0 && iIndex < pBarFolder->GetItemCount());
  2829. CXTOutBarItem* pBarItem = pBarFolder->GetItemAt(iIndex);
  2830. ASSERT(pBarItem && pImageList);
  2831. if (!pBarItem)
  2832. return;
  2833. CFont* oft = (CFont*)pDC->SelectObject(GetFontX());
  2834. COLORREF ocr = pDC->SetTextColor(GetXtremeColor(COLOR_WINDOW));
  2835. int obk = pDC->SetBkMode(TRANSPARENT);
  2836. CRect rcFolder;
  2837. GetFolderRect(iFolder, rcFolder);
  2838. rc.left = rcFolder.left +4;
  2839. rc.right = rcFolder.right-4;
  2840. // Exclude the up and down arrows from painting.
  2841. if (m_bUpArrow)
  2842. pDC->ExcludeClipRect(&m_rcUpArrow);
  2843. if (m_bDownArrow)
  2844. pDC->ExcludeClipRect(&m_rcDownArrow);
  2845. // set the items text color.
  2846. pDC->SetTextColor(GetTheme()->m_clrText);
  2847. if (pBarItem && pBarItem->IsSelected() && !m_bAnimating && m_nLastItemSelected >= 0)
  2848. {
  2849. CRect rcIcon;
  2850. GetIconRect(m_nSelFolder, m_nLastItemSelected, rcIcon);
  2851. pDC->FillSolidRect(rcIcon, GetXtremeColor(COLOR_3DLIGHT));
  2852. pDC->Draw3dRect(rcIcon, GetXtremeColor(COLOR_3DSHADOW), GetXtremeColor(COLOR_3DHILIGHT));
  2853. }
  2854. if (pImageList)
  2855. {
  2856. IMAGEINFO ii;
  2857. pImageList->GetImageInfo(pBarItem->GetIndex(), &ii);
  2858. CSize szImage = CRect(ii.rcImage).Size();
  2859. CPoint pt;
  2860. pt.x = rc.left + 2;
  2861. pt.y = rc.top + (rc.Height() - szImage.cy) / 2;
  2862. DrawItemIcon(pDC, pt, pBarItem, pImageList);
  2863. if (!bOnlyImage)
  2864. {
  2865. rc.left += (szImage.cx + m_sizeOffset.cx);
  2866. DrawItemText(pDC, rc, pBarItem, DT_VCENTER | DT_SINGLELINE | DT_END_ELLIPSIS);
  2867. }
  2868. }
  2869. pDC->SetTextColor(ocr);
  2870. pDC->SelectObject(oft);
  2871. pDC->SetBkMode(obk);
  2872. }
  2873. }
  2874. void CXTToolBox::DrawScrollButton(CDC* pDC, CRect rect, UINT uType, UINT uState)
  2875. {
  2876. rect.InflateRect(1, 2, 1, 2);
  2877. CRect rcInside;
  2878. GetInsideRect(rcInside);
  2879. rcInside.DeflateRect(2, 2);
  2880. rect.right = rcInside.right;
  2881. rect.left = rect.right - rect.Width();
  2882. rect.DeflateRect(1, 1, 1, 1);
  2883. CXTOutBarCtrl::DrawScrollButton(pDC, rect, uType, uState);
  2884. rect.DeflateRect(1, 1, 1, 1);
  2885. pDC->Draw3dRect(&rect, GetXtremeColor(COLOR_3DFACE), GetXtremeColor(COLOR_3DFACE));
  2886. rect.InflateRect(1, 1, 1, 1);
  2887. if ((uState & DFCS_PUSHED) == DFCS_PUSHED)
  2888. {
  2889. pDC->Draw3dRect(&rect, GetXtremeColor(COLOR_3DSHADOW), GetXtremeColor(COLOR_3DHILIGHT));
  2890. }
  2891. else
  2892. {
  2893. pDC->Draw3dRect(&rect, GetXtremeColor(COLOR_3DHILIGHT), GetXtremeColor(COLOR_3DSHADOW));
  2894. }
  2895. }
  2896. void CXTToolBox::OnDraw(CDC* pDC)
  2897. {
  2898. m_iLast = m_nFirstItem;
  2899. GetTheme()->m_clrBack.SetStandardValue(GetXtremeColor(COLOR_3DFACE));
  2900. GetTheme()->m_clrText.SetStandardValue(GetXtremeColor(COLOR_BTNTEXT));
  2901. // Get the client rect.
  2902. CXTPClientRect rectClient(this);
  2903. CRect rc;
  2904. GetInsideRect(rc);
  2905. if (!GetFolderChild())
  2906. {
  2907. pDC->FillSolidRect(rc, GetTheme()->m_clrBack);
  2908. }
  2909. int max = (int)m_arFolder.GetSize();
  2910. CRect frc;
  2911. int t;
  2912. for (t = 0; t < max; t++)
  2913. {
  2914. GetFolderRect(t, frc);
  2915. DrawFolder(pDC, t, frc, xtMouseNormal);
  2916. }
  2917. if (!GetFolderChild())
  2918. {
  2919. GetVisibleRange(m_nSelFolder, m_iFirst, m_iLast);
  2920. m_rcUpArrow.SetRect(0, 0, GetSystemMetrics(SM_CXVSCROLL), GetSystemMetrics(SM_CXVSCROLL));
  2921. m_rcDownArrow = m_rcUpArrow;
  2922. m_rcUpArrow.OffsetRect(rc.right - 5 - GetSystemMetrics(SM_CXVSCROLL), rc.top +5);
  2923. m_rcDownArrow.OffsetRect(rc.right - 5 - GetSystemMetrics(SM_CXVSCROLL), rc.bottom - 5 - GetSystemMetrics(SM_CXVSCROLL));
  2924. if (m_iFirst > 0 && m_rcUpArrow.IntersectRect(rc, m_rcUpArrow))
  2925. {
  2926. if (m_bUpPressed)
  2927. {
  2928. DrawScrollButton(pDC, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP | DFCS_PUSHED);
  2929. }
  2930. else
  2931. {
  2932. DrawScrollButton(pDC, m_rcUpArrow, DFC_SCROLL, DFCS_SCROLLUP);
  2933. }
  2934. m_bUpArrow = TRUE;
  2935. pDC->ExcludeClipRect(m_rcUpArrow);
  2936. }
  2937. else
  2938. {
  2939. m_bUpArrow = FALSE;
  2940. }
  2941. if (m_bUpArrow && m_rcUpArrow.bottom >= m_rcDownArrow.top)
  2942. {
  2943. m_rcDownArrow.OffsetRect(0, m_rcUpArrow.bottom - m_rcDownArrow.top + 1);
  2944. }
  2945. if (GetItemCount() > 0 && m_iLast < GetItemCount() - 1 && m_rcDownArrow.IntersectRect(rc, m_rcDownArrow))
  2946. {
  2947. if (m_bDownPressed)
  2948. {
  2949. DrawScrollButton(pDC, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN | DFCS_PUSHED);
  2950. }
  2951. else
  2952. {
  2953. DrawScrollButton(pDC, m_rcDownArrow, DFC_SCROLL, DFCS_SCROLLDOWN);
  2954. }
  2955. m_bDownArrow = TRUE;
  2956. pDC->ExcludeClipRect(m_rcDownArrow);
  2957. }
  2958. else m_bDownArrow = FALSE;
  2959. PaintItems(pDC, m_nSelFolder, rc);
  2960. }
  2961. //dc.BitBlt(rectClient.left, rectClient.top, rectClient.Width(), rectClient.Height(), pDC, 0, 0, SRCCOPY);
  2962. //pDC->RestoreDC(nSavedDC);
  2963. if (m_nFolderHilighted >= 0)
  2964. {
  2965. int i = m_nFolderHilighted;
  2966. m_nFolderHilighted = -1;
  2967. HighlightFolder(pDC, i);
  2968. }
  2969. if (m_nItemHilighted >= 0)
  2970. {
  2971. int i = m_nItemHilighted;
  2972. m_nItemHilighted = -1;
  2973. HighlightItem(pDC, i, m_bPressedHighlight);
  2974. }
  2975. }
  2976. void CXTToolBox::DrawFolder(CDC* pDC, const int iIndex, CRect rect, const XTMouseState eHilight)
  2977. {
  2978. if (m_arFolder.GetSize() != 0)
  2979. {
  2980. pDC->FillSolidRect(rect, GetTheme()->m_clrBack);
  2981. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  2982. int nSavedDC = pDC->SaveDC();
  2983. CPen pen(PS_SOLID, 1, GetXtremeColor(COLOR_3DSHADOW));
  2984. CRect rcInside;
  2985. GetInsideRect(rcInside);
  2986. CRect rcFolder(rect);
  2987. rcFolder.left = rcInside.left + 2;
  2988. rcFolder.right = rcInside.right - 2;
  2989. rcFolder.DeflateRect(0, 1);
  2990. switch (eHilight)
  2991. {
  2992. case xtMouseNormal:
  2993. case xtMouseHover:
  2994. {
  2995. pDC->FillSolidRect(rcFolder, GetXtremeColor(COLOR_3DFACE));
  2996. pDC->Draw3dRect(rcFolder, GetXtremeColor(COLOR_3DHILIGHT), GetXtremeColor(COLOR_3DSHADOW));
  2997. pDC->SetBkMode(TRANSPARENT);
  2998. pDC->SelectObject(GetFontX());
  2999. rcFolder.DeflateRect(4, 0);
  3000. pDC->DrawText(pBarFolder->GetName(), rcFolder,
  3001. DT_VCENTER | DT_SINGLELINE | DT_END_ELLIPSIS);
  3002. }
  3003. break;
  3004. case xtMouseSelect:
  3005. {
  3006. pDC->FillSolidRect(rcFolder, GetXtremeColor(COLOR_3DFACE));
  3007. pDC->Draw3dRect(rcFolder, GetXtremeColor(COLOR_3DSHADOW), GetXtremeColor(COLOR_3DHILIGHT));
  3008. pDC->SetBkMode(TRANSPARENT);
  3009. pDC->SelectObject(GetFontX());
  3010. rcFolder.DeflateRect(4, 0);
  3011. pDC->DrawText(pBarFolder->GetName(), rcFolder,
  3012. DT_VCENTER | DT_SINGLELINE | DT_END_ELLIPSIS);
  3013. }
  3014. break;
  3015. }
  3016. pDC->RestoreDC(nSavedDC);
  3017. pen.DeleteObject();
  3018. }
  3019. }
  3020. void CXTToolBox::SetSelFolder(const int iIndex)
  3021. {
  3022. if (m_arFolder.GetSize() != 0)
  3023. {
  3024. ASSERT(iIndex >= 0 && iIndex < GetFolderCount());
  3025. CXTOutBarFolder* pBarFolder = (CXTOutBarFolder*)m_arFolder.GetAt(iIndex);
  3026. XT_OUTBAR_INFO obi;
  3027. obi.nIndex = iIndex;
  3028. obi.nDragFrom = -1;
  3029. obi.nDragTo = -1;
  3030. obi.lpszText = (LPCTSTR)pBarFolder->GetName();
  3031. obi.bFolder = true;
  3032. LRESULT lResult = NotifyOwner(OBN_XT_FOLDERCHANGE, &obi);
  3033. if (lResult == TRUE)
  3034. {
  3035. CWnd* pWnd = GetFolderChild();
  3036. if (pWnd)
  3037. {
  3038. pWnd->ShowWindow(SW_HIDE);
  3039. }
  3040. if (iIndex != m_nSelFolder && m_nSelFolder >= 0)
  3041. {
  3042. if ((m_dwFlags & OBS_XT_ANIMATION) && (m_nAnimationTickCount >= 0))
  3043. {
  3044. m_bAnimating = true;
  3045. AnimateFolderScroll(m_nSelFolder, iIndex);
  3046. m_bAnimating = false;
  3047. }
  3048. }
  3049. m_nSelFolder = iIndex;
  3050. m_nFirstItem = 0;
  3051. m_nLastItemSelected = pBarFolder->GetSelIndex();
  3052. if (m_nLastItemSelected != -1)
  3053. {
  3054. pBarFolder->SetSelItem(m_nLastItemSelected);
  3055. }
  3056. pWnd = GetFolderChild();
  3057. if (pWnd)
  3058. {
  3059. CRect rc;
  3060. GetInsideRect(rc);
  3061. pWnd->MoveWindow(rc);
  3062. pWnd->ShowWindow(SW_SHOW);
  3063. }
  3064. InvalidateRect(NULL);
  3065. }
  3066. }
  3067. }
  3068. BOOL CXTToolBox::CreateEx(DWORD dwExStyle, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, const DWORD dwFlags)
  3069. {
  3070. if (!CXTOutBarCtrl::CreateEx(dwExStyle, dwStyle, rect, pParentWnd, nID,
  3071. dwFlags | OBS_XT_SELHIGHLIGHT | OBS_XT_ANIMATION))
  3072. {
  3073. return FALSE;
  3074. }
  3075. SetSmallIconView(TRUE);
  3076. return TRUE;
  3077. }
  3078. void CXTToolBox::OnRButtonUp(UINT nFlags, CPoint point)
  3079. {
  3080. // bypass base class.
  3081. CWnd::OnRButtonUp(nFlags, point);
  3082. }