supergridctrl.cpp
上传用户:meggie0806
上传日期:2007-01-02
资源大小:87k
文件大小:62k
源码类别:

ListView/ListBox

开发平台:

Visual C++

  1. // SuperGridCtrl.cpp : implementation file
  2. //
  3. #include "stdafx.h"
  4. #include "SuperGrid.h"
  5. #include "SuperGridCtrl.h"
  6. #include "ListEditCtrl.h"
  7. #ifdef _DEBUG
  8. #define new DEBUG_NEW
  9. #undef THIS_FILE
  10. static char THIS_FILE[] = __FILE__;
  11. #endif
  12. /*
  13.     SuperGrid, YALC (yet another listview control) 
  14. Written by me(Allan Nielsen), that is .. if the code works :)
  15. mailto:allan@object-oriented.com
  16. Copyright (c) 1999.
  17. If you find bugs, please notify me and I'll contact the author who wrote the darn thing :)
  18.     You may use it, abuse it, redistribute it, in any way you desire, but
  19. if you choose to use this control in any commercial or non commercial application then please 
  20. send me an email letting me know, makes me :) . 
  21. HISTORY 
  22. what's new since last update: 
  23. 4 dec. 1998.
  24. - MeasureItem now uses current font
  25. - Combobox support in CItemInfo
  26.   see the ShowList in MySuperGrid.h/cpp, PreTranslateMessage functions
  27.   and the file ComboInListView.cpp/h
  28.                 - function to indicate the precens of a combobox it draws a down arrow ..see DrawComboBox
  29.   the idea came from Robert Bouwens. :)
  30. - in file TestTreeDlg.cpp:  due to the MeasureItem message-reflection
  31.   the listctrl is now created in the CDialog::OnCreate function and later on resized
  32. - LVIS_DROPHILITED support when dragging items
  33. - Support for LVS_EX_CHECKBOXES   
  34. - Added some examples on how to print preview selected item(or at least to get some information from of the grid)
  35.   the sample is in the CMySuperGrid.h/cpp which is derived from CSuperGridCtrl it shows you
  36.   how to initalize the grid, sort items, search for items and subitems, select and delete thems
  37. - added virtual GetIcon function to set your default icon handler for a listview index.
  38. - added virtual GetCellRGB() to set the color for current selected cell
  39. - renamed CTreeNode to CTreeItem, sorry ;-(
  40. - the nested class CTreeItem has been stripped down to nothing but simple (struct)data members
  41.   only servering as a linked list of lists now :-)
  42. - well.. moved all the importent stuff from CTreeItem to CSuperGridCtrl e.g
  43.   all operations on a CTreeItem now resides in the CSuperGridCtrl, much more OO-like ;-)
  44. - added Quicksort function
  45. - got rid of TheOpenCloseThing function.
  46. - added virtual function OnUpdateListViewItem..called when ever an Item is about to be updated
  47. - added virtual function OnControlLButtonDown...called when ever LButtondown in a cell
  48. - added virtual function CreateDragImageEx...
  49. 11 jan 1999:
  50. - added SetFocus in OnDblclk and in OnLButtonDown, big thank you to Dieter Gollwitzer 
  51. - fixed a minor hittest error in the HitTestOnSign(), again thanks to Dieter Gollwitzer 
  52. 16 jan 1999: 
  53. - added virtual OnItemExpanding(....
  54. - added virtual OnItemExpanded(...
  55. - added virtual OnCollapsing(....
  56. - added virtual OnItemCollapsed(...
  57. - added virtual OnDeleteItem(....
  58. - added virtual OnVKMultiply(...
  59. - added virtual OnVkSubTract(..
  60. - added virtual OnVKAdd(....
  61. - added SetChildrenFlag();
  62. 1 feb 1999:
  63. - CItemInfo now supports individual cell-color
  64. - fixed some bugs in regards to listdata associated with in each cell
  65. - added virtual BOOL OnVkReturn(........
  66. - added virtual BOOL OnItemLButtonDown(.....
  67. 4 mai 1999  - moved to Switzerland to work as a systemdeveloper for 2-3 years :).
  68. - new email :)
  69. - added new class for encapsulating drawing + - buttons.
  70.    22.june 1999 - Added GetCount()
  71. - removed support for VC++ 5.0.
  72. - added GetTreeItem(.....) 
  73. - GetSelectedItem(.....)
  74. - Howto disable Drag/drop in CMySuperGrid
  75. - level 4 compiled
  76. - Clean up some redudant code crap
  77. 30.june 99  - fixed minor error in DrawItem, a drawing error occurred when 1.st column was hidden, 
  78.   it was previous fixed, but some how it was introduced again :).
  79. 9. juli 99  - CRAIG SCHMIDT "the beachboy":)
  80.   Craig did the implementation of the Multiple Root request
  81.   his email are "craig_a_schmidt@yahoo.com" if you have any Questions
  82.   in regards to multiple roots implementation, although I think I can help too :).
  83.   he added the following functions :
  84.   - CTreeItem*  InsertRootItem(.....
  85.   - void DeleteRootItem(....
  86.   - BOOL IsRoot(...
  87.   - CTreeItem* GetRootItem(..
  88.   - int GetRootIndex(..
  89.   - int GetRootCount(...
  90.   - and a member variable, CPtrList m_RootItems which of course are a collection of ..ta da RootItems a.k.a CTreeItem.
  91.   see the file CMySuperGrid.cpp for usage of the InsertRootItem.
  92.     10.juli 99    added simple wrappers for the rootitems collection
  93.   - Added GetRootHeadPosition(...)
  94.   - Added GetRootTailPosition();
  95.   - Added CTreeItem* GetNextRoot(...
  96.   - Added CTreeItem* GetPrevRoot(...
  97. 12. juli 99  - fixed minor error in CreateDragImageEx, an error occurred when 1st column was hidden and a drag operation was initiated
  98. 13. juli 99  - Added void DeleteAll()...deletes all items in the grid, uhh
  99.  - Removed GetNextSiblingItem(...), GetPrevSiblingItem(...) due to support for multiple roots. 
  100.  - Added CTreeItem *GetNext(...) and CTreeItem* GetPrev(..) due to support for multiple roots.
  101. 14. juli 99  - Fixed a memory leak in CMySuperGrid::HowToInsertItemsAfterTheGridHasBeenInitialized....
  102. WHAT'S UP :
  103. - Vacation
  104. - ATL VERSION COMMING UP.
  105. - Better documentation :)
  106. - Peace on earth.
  107. - service pack 4 for VC++ 6.0 :=|
  108. - Windows 3000 :)
  109. */
  110. /////////////////////////////////////////////////////////////////////////////
  111. // CSuperGridCtrl
  112. CSuperGridCtrl::CSuperGridCtrl()
  113. {
  114. m_cxImage = m_cyImage = m_bIsDragging = m_CurSubItem = 0;
  115.     m_nDragTarget=m_nDragItem = -1;
  116. m_psTreeLine.CreatePen(PS_SOLID, 1, RGB(192,192,192));
  117. m_psRectangle.CreatePen(PS_SOLID, 1, RGB(198,198,198));
  118. m_psPlusMinus.CreatePen(PS_SOLID, 1, RGB(0,0,0));
  119. m_brushErase.CreateSolidBrush(RGB(255,255,255));
  120. m_himl = NULL;
  121. }
  122. CSuperGridCtrl::~CSuperGridCtrl()
  123. {
  124. m_psPlusMinus.DeleteObject();
  125. m_psRectangle.DeleteObject();
  126. m_psTreeLine.DeleteObject();
  127. m_brushErase.DeleteObject();
  128. while(m_RootItems.GetCount())
  129. {
  130. CTreeItem * root = (CTreeItem*)m_RootItems.RemoveHead();
  131. if(root!=NULL && GetData(root) != NULL)
  132. delete GetData(root);
  133. delete root;
  134. }
  135. m_RootItems.RemoveAll();
  136. }
  137. BEGIN_MESSAGE_MAP(CSuperGridCtrl, CListCtrl)
  138. //{{AFX_MSG_MAP(CSuperGridCtrl)
  139. ON_NOTIFY_REFLECT(NM_DBLCLK, OnDblclk)
  140. ON_NOTIFY_REFLECT(LVN_ENDLABELEDIT, OnEndlabeledit)
  141. ON_WM_CREATE()
  142. ON_WM_HSCROLL()
  143. ON_WM_LBUTTONDOWN()
  144. ON_WM_MOUSEMOVE()
  145. ON_WM_TIMER()
  146. ON_WM_VSCROLL()
  147. ON_WM_LBUTTONUP()
  148. ON_NOTIFY_REFLECT(LVN_KEYDOWN, OnKeydown)
  149. ON_NOTIFY_REFLECT(LVN_BEGINDRAG, OnBegindrag)
  150. ON_WM_MEASUREITEM_REFLECT()
  151. ON_WM_DRAWITEM_REFLECT()
  152. ON_WM_SYSCOLORCHANGE()
  153. //}}AFX_MSG_MAP
  154. END_MESSAGE_MAP()
  155. /////////////////////////////////////////////////////////////////////////////
  156. // CSuperGridCtrl message handlers
  157. BOOL CSuperGridCtrl::PreCreateWindow(CREATESTRUCT& cs) 
  158. {
  159. cs.style |= LVS_REPORT | LVS_SINGLESEL | LVS_SHAREIMAGELISTS | LVS_OWNERDRAWFIXED | LVS_SHOWSELALWAYS;
  160. return CListCtrl::PreCreateWindow(cs);
  161. }
  162. int CSuperGridCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct) 
  163. {
  164. if (CListCtrl::OnCreate(lpCreateStruct) == -1)
  165. return -1;
  166. return 0;
  167. }
  168. #define OFFSET_FIRST 2 
  169. #define OFFSET_OTHER 6
  170. void CSuperGridCtrl::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct) 
  171. {
  172. if (lpDrawItemStruct->CtlType != ODT_LISTVIEW)
  173.         return;
  174. if(lpDrawItemStruct->itemAction == ODA_DRAWENTIRE)
  175. {
  176. if(m_himl==NULL)
  177. {
  178. m_himl = (HIMAGELIST)::SendMessage(m_hWnd, LVM_GETIMAGELIST, (WPARAM)(int)(LVSIL_SMALL), 0L);
  179. if(m_himl==NULL)
  180. return;
  181. }
  182. LV_ITEM lvi;
  183. static _TCHAR szBuff[MAX_PATH];
  184. LPCTSTR pszText;
  185. int nItem = lpDrawItemStruct->itemID;
  186. CRect rcItem(lpDrawItemStruct->rcItem);
  187. lvi.mask = LVIF_TEXT | LVIF_IMAGE | LVIF_STATE | LVIF_INDENT | LVIF_PARAM; 
  188. lvi.iItem = nItem;
  189. lvi.iSubItem =0;
  190. lvi.pszText = szBuff;
  191. lvi.cchTextMax = sizeof(szBuff);
  192. lvi.stateMask = 0xFFFF;
  193. GetItem(&lvi);
  194. CTreeItem *pSelItem = (CTreeItem*)lpDrawItemStruct->itemData;
  195. CRect rcLabel;
  196. GetItemRect(nItem, rcLabel, LVIR_LABEL);
  197. CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC);
  198. ASSERT(pDC);
  199. CRect rcClipBox;
  200. pDC->GetClipBox(rcClipBox);
  201. COLORREF crBackground, crText;
  202. if (lpDrawItemStruct->itemState & ODS_SELECTED)
  203. {
  204. // Set the text background and foreground colors
  205. crBackground = GetSysColor (COLOR_HIGHLIGHT);
  206. crText = GetSysColor (COLOR_HIGHLIGHTTEXT);
  207. }
  208. else
  209. {
  210. // Set the text background and foreground colors to the standard window
  211. // colors
  212. crBackground = GetSysColor (COLOR_WINDOW);
  213. crText = GetSysColor (COLOR_WINDOWTEXT);
  214. }
  215. //Get current Icon, you have overridden this I hope :-)
  216. int iImage = GetIcon(pSelItem);
  217. if(iImage!=-1)
  218. {
  219. if(lvi.iImage!=iImage)
  220. {
  221. LV_ITEM lvItem;
  222. lvItem.mask =  LVIF_IMAGE;
  223. lvItem.iImage = iImage;
  224. lvItem.iItem = nItem;
  225. lvItem.iSubItem = 0;
  226. SetItem(&lvItem);
  227. lvi.iImage = iImage; 
  228. }
  229. }
  230. CRect rc;
  231. GetItemRect(nItem, rc, LVIR_BOUNDS);
  232. CRect rcIcon;
  233. GetItemRect(nItem, rcIcon, LVIR_ICON);
  234. //Draw Current image
  235. int nOffset = (rcItem.Height() - m_cyImage)/2;
  236. int nY = rcItem.bottom - m_cyImage - nOffset;
  237. int nWidth = m_cxImage;//def icon size
  238. //do not draw icon out side 1.st column.
  239. int iTryIndent = GetIndent(pSelItem) * m_cxImage + m_cxImage;
  240. if(rc.left + iTryIndent  > GetColumnWidth(0)-2/*looks better -2*/)
  241. nWidth = (rc.left + iTryIndent) - GetColumnWidth(0);
  242. UINT uiFlags = ILD_TRANSPARENT;
  243. if( GetItemState(nItem, LVIF_STATE) & LVIS_DROPHILITED)//if dragging show a SelectDropTarget alike effect :)
  244. uiFlags |= ILD_BLEND50;
  245. if((nWidth=m_cxImage-nWidth) >-1)//calc width of icon
  246. {
  247. ImageList_DrawEx(m_himl, lvi.iImage, pDC->m_hDC,  
  248.  rc.left + (GetIndent(pSelItem) * m_cxImage), 
  249.  nY, 
  250.  nWidth,
  251.  m_cyImage,
  252.  CLR_DEFAULT, 
  253.  CLR_DEFAULT, 
  254.  uiFlags);
  255. DrawTreeItem(pDC, pSelItem, nItem, rc);
  256. }
  257. //Draw selection bar (erase old selection too)
  258. pDC->SetBkColor(crBackground);
  259. CRect rcClip = lpDrawItemStruct->rcItem;
  260. rcClip.left += GetIndent(pSelItem) * m_cxImage + m_cxImage + 2;
  261. if(rcClip.left > GetColumnWidth(0))
  262. rcClip.left = GetColumnWidth(0);
  263. //fill background color
  264. ExtTextOut(pDC->GetSafeHdc(), 0, 0, ETO_OPAQUE, rcClip, NULL, 0, NULL);
  265. //draw color in first col if any
  266. rcClip.right = rcLabel.right;
  267. CItemInfo *pItemInfo = GetData(pSelItem);
  268. COLORREF clf = pItemInfo->GetItemClr();
  269. if(clf!=-1)
  270. {
  271. CBrush br(clf);
  272. pDC->FillRect(rcClip, &br);
  273. }
  274. //draw selection rect in 1.st col if selected
  275. if ((lpDrawItemStruct->itemState & ODS_SELECTED) && (m_CurSubItem==0))
  276. {
  277. CBrush br(GetCellRGB());
  278. pDC->FillRect(rcClip,&br);
  279. pDC->DrawFocusRect(rcClip);
  280. }
  281. //if checkbox style
  282. UINT nStateImageMask = lvi.state & LVIS_STATEIMAGEMASK;
  283. if (nStateImageMask)
  284. {
  285. int nImage = (nStateImageMask>>12) - 1;
  286. CImageList *pImageList = GetImageList(LVSIL_STATE);
  287. if (pImageList)
  288. {
  289. int cxIcon,cyIcon=0;
  290. ImageList_GetIconSize(pImageList->m_hImageList, &cxIcon, &cyIcon);
  291. if(rc.left + (GetIndent(pSelItem) * m_cxImage) + m_cxImage + cxIcon < GetColumnWidth(0))
  292. pImageList->Draw(pDC, nImage,CPoint(rc.left + (GetIndent(pSelItem) * m_cxImage) + cxIcon, nY), ILD_TRANSPARENT);
  293. }
  294. }
  295. //draw 1. item
  296. GetItemRect(nItem, rcItem, LVIR_LABEL);
  297. pszText = MakeShortString(pDC, szBuff, rcItem.right - rcItem.left, 2*OFFSET_FIRST);
  298. rcLabel = rcItem;
  299. rcLabel.left+=OFFSET_FIRST;
  300. rcLabel.right-=OFFSET_FIRST;
  301. pDC->SetBkColor (crBackground);
  302. pDC->SetTextColor (crText);
  303. pDC->DrawText(pszText,-1, rcLabel,DT_LEFT | DT_SINGLELINE | DT_NOPREFIX | DT_VCENTER | DT_EXTERNALLEADING);
  304. //draw subitems..
  305. LV_COLUMN lvc;
  306. lvc.mask = LVCF_FMT | LVCF_WIDTH;
  307. for(int nColumn=1; GetColumn(nColumn, &lvc); nColumn++)
  308. {
  309. rcItem.left=rcItem.right;
  310. rcItem.right+=lvc.cx;
  311. if (rcItem.left < rcClipBox.right && rcItem.right > rcClipBox.left && rcItem.right > rcItem.left)
  312. {
  313. //support for colors in each cell
  314. COLORREF clf = pItemInfo->GetBkColor(nColumn-1);
  315. if(clf!=-1)
  316. {
  317. CBrush br(clf);
  318. pDC->FillRect(rcItem, &br);
  319. }
  320. int nRetLen = GetItemText(nItem, nColumn, szBuff, sizeof(szBuff));
  321. if(nRetLen==0)
  322. pszText=NULL;
  323. else
  324. pszText=MakeShortString(pDC,szBuff,rcItem.right-rcItem.left,2*OFFSET_OTHER);
  325. UINT nJustify=DT_LEFT;
  326. if(pszText==szBuff)
  327. {
  328. switch(lvc.fmt & LVCFMT_JUSTIFYMASK)
  329. {
  330. case LVCFMT_RIGHT:
  331. nJustify=DT_RIGHT;
  332. break;
  333. case LVCFMT_CENTER:
  334. nJustify=DT_CENTER;
  335. break;
  336. default:
  337. break;
  338. }
  339. }
  340. rcLabel=rcItem;
  341. rcLabel.left+=OFFSET_OTHER;
  342. rcLabel.right-=OFFSET_OTHER;
  343. if (lpDrawItemStruct->itemState & ODS_SELECTED && !m_bIsDragging)
  344. DrawFocusCell(pDC, lpDrawItemStruct->itemID, nColumn);
  345. if(pszText!=NULL)
  346. pDC->DrawText(pszText,-1,rcLabel, nJustify | DT_SINGLELINE | DT_NOPREFIX | DT_VCENTER | DT_EXTERNALLEADING);
  347. }//if
  348. //draw down arrow if combobox regardless of rcClipBox rgn
  349. if (lpDrawItemStruct->itemState & ODS_SELECTED)
  350. DrawComboBox(pDC, pSelItem, nItem, nColumn);
  351. }//for
  352. }//ODA_DRAWENTIRE
  353. }
  354. void CSuperGridCtrl::DrawComboBox(CDC* pDC, CTreeItem *pSelItem, int nItem, int nColumn)
  355. {
  356. CItemInfo* pInfo = GetData(pSelItem);
  357. CItemInfo::CONTROLTYPE ctrlType;
  358. if(pInfo->GetControlType(nColumn-1, ctrlType))
  359. {
  360. if(ctrlType == pInfo->CONTROLTYPE::combobox) 
  361. {
  362. CRect rect;
  363. GetSubItemRect(nItem, nColumn, LVIR_BOUNDS, rect);
  364. rect.left=rect.right - GetSystemMetrics(SM_CYVSCROLL);
  365. pDC->DrawFrameControl(rect, DFC_SCROLL, DFCS_SCROLLDOWN);
  366. }
  367. }
  368. }
  369. //this piece of code is borrowed from the wndproc.c file in the odlistvw.exe example from MSDN and was converted to mfc-style
  370. void CSuperGridCtrl::MeasureItem(LPMEASUREITEMSTRUCT lpMeasureItemStruct)
  371. {
  372.     if (lpMeasureItemStruct->CtlType != ODT_LISTVIEW)
  373.         return;
  374. TEXTMETRIC tm;
  375. CClientDC dc(this);
  376. CFont* pFont = GetFont();
  377. CFont* pOldFont = dc.SelectObject(pFont);
  378. dc.GetTextMetrics(&tm);
  379. int nItemHeight = tm.tmHeight + tm.tmExternalLeading;
  380. lpMeasureItemStruct->itemHeight = nItemHeight + 4; //or should I go for max(nItemheight+4, m_cxImage+2);
  381. dc.SelectObject(pOldFont);
  382. }
  383. //the basic rutine making the ... thing snatched it from some tedious code example some where in MSDN call odlistvw.exe
  384. LPCTSTR CSuperGridCtrl::MakeShortString(CDC* pDC, LPCTSTR lpszLong, int nColumnLen, int nOffset)
  385. {
  386. static const _TCHAR szThreeDots[]=_T("...");
  387. int nStringLen=lstrlen(lpszLong);
  388. if(nStringLen==0 || pDC->GetTextExtent(lpszLong,nStringLen).cx + nOffset < nColumnLen)
  389. return(lpszLong);
  390. static _TCHAR szShort[MAX_PATH];
  391. lstrcpy(szShort,lpszLong);
  392. int nAddLen = pDC->GetTextExtent(szThreeDots,sizeof(szThreeDots)).cx;
  393. for(int i=nStringLen-1; i > 0; i--)
  394. {
  395. szShort[i]=0;
  396. if(pDC->GetTextExtent(szShort,i).cx + nOffset + nAddLen < nColumnLen)
  397. break;
  398. }
  399. lstrcat(szShort,szThreeDots);
  400. return(szShort);
  401. }
  402. void CSuperGridCtrl::OnKeydown(NMHDR* pNMHDR, LRESULT* pResult) 
  403. {
  404. LV_KEYDOWN* pLVKeyDow = (LV_KEYDOWN*)pNMHDR;
  405. switch(pLVKeyDow->wVKey)
  406. {
  407. case VK_SPACE: 
  408. {
  409. if(GetExtendedStyle() & LVS_EX_CHECKBOXES)
  410. {
  411. int nIndex = GetSelectedItem();
  412. if(nIndex !=-1)
  413. {
  414. CTreeItem* pItem = GetTreeItem(nIndex);
  415. if(pItem!=NULL)
  416. {
  417. CItemInfo *pInfo = GetData(pItem);
  418. pInfo->SetCheck(!pInfo->GetCheck());
  419. }
  420. }
  421. }
  422. }break;
  423. case VK_DELETE: 
  424. {
  425. int nItem = GetSelectedItem();
  426. if(nItem!=-1)
  427. {
  428. CTreeItem* pSelItem = GetTreeItem(nItem);
  429. if(pSelItem != NULL)
  430. {
  431. if(OnDeleteItem(pSelItem, nItem))
  432. DeleteItemEx(pSelItem, nItem);
  433. }
  434. }
  435. } break;
  436. case VK_MULTIPLY:
  437. {  
  438. int nIndex = GetSelectedItem();
  439. if(nIndex != -1)
  440. {
  441. CWaitCursor wait;
  442. SetRedraw(0);
  443. CTreeItem *pParent = GetTreeItem(nIndex);
  444. int nScroll=0;
  445. if(OnVKMultiply(pParent, nIndex))
  446. {
  447. ExpandAll(pParent, nScroll);
  448. }
  449. SetRedraw(1);
  450. RedrawItems(nIndex, nScroll);
  451. EnsureVisible(nScroll, TRUE);
  452.  }
  453.  }break;
  454. case VK_ADD:
  455. {
  456. int nIndex = GetSelectedItem();
  457. if(nIndex!=-1)
  458. {
  459. CWaitCursor wait;
  460. CTreeItem *pSelItem = GetTreeItem(nIndex);
  461. int nScrollIndex = 0;
  462. if(OnVKAdd(pSelItem, nIndex))
  463. {
  464.  nScrollIndex = Expand(pSelItem, nIndex);
  465. }
  466. CRect rc;
  467. GetItemRect(nIndex, rc, LVIR_BOUNDS);
  468. InvalidateRect(rc);
  469. UpdateWindow();
  470. EnsureVisible(nScrollIndex, 1);
  471. }
  472. }break;
  473. case VK_SUBTRACT:
  474. {
  475. int nIndex = GetSelectedItem();
  476. if(nIndex!=-1)
  477. {
  478. CWaitCursor wait;
  479. CTreeItem *pSelItem = GetTreeItem(nIndex);
  480. if(OnVkSubTract(pSelItem, nIndex))
  481. {
  482. Collapse(pSelItem);
  483. }
  484. CRect rc;
  485. GetItemRect(nIndex, rc, LVIR_BOUNDS);
  486. InvalidateRect(rc);
  487. UpdateWindow();
  488. }
  489. }break;
  490. default :break;
  491. }
  492. *pResult = 0;
  493. }
  494. BOOL CSuperGridCtrl::HitTestOnSign(CPoint point, LVHITTESTINFO& ht)
  495. {
  496. ht.pt = point;
  497. // Test which subitem was clicked.
  498. SubItemHitTest(&ht);
  499. if(ht.iItem!=-1)
  500. {
  501. //first hittest on checkbox
  502. BOOL bHit = FALSE;
  503. if(GetExtendedStyle() & LVS_EX_CHECKBOXES)
  504. {
  505. if (ht.flags == LVHT_ONITEM && (GetStyle() & LVS_OWNERDRAWFIXED))//isn't this allways ownerdrawfixed :-)
  506. {
  507. CRect rcIcon,rcLabel;
  508. GetItemRect(ht.iItem, rcIcon, LVIR_ICON);//has to be between these two ....right :)
  509. GetItemRect(ht.iItem, rcLabel, LVIR_LABEL);
  510. // check if hit was on a state icon 
  511. if (point.x > rcIcon.left && point.x < rcLabel.left)
  512. bHit = TRUE;
  513. }
  514. else if (ht.flags & LVHT_ONITEMSTATEICON)
  515. bHit = TRUE;
  516. }
  517. CTreeItem* pItem = GetTreeItem(ht.iItem);
  518. if(pItem!=NULL)
  519. {
  520. if(bHit)//if checkbox
  521. {
  522. //yes I know..have to maintain to sets of checkstates here...
  523. //one for listview statemask and one for CTreeItem..but its located here so no harm done
  524. SetCheck(ht.iItem,!GetCheck(ht.iItem));
  525. CItemInfo *pInfo = GetData(pItem);
  526. pInfo->SetCheck(!pInfo->GetCheck());
  527. }
  528. //if haschildren and clicked on + or - then expand/collapse
  529. if(ItemHasChildren(pItem))
  530. {
  531. //hittest on the plus/sign "button" 
  532. CRect rcBounds;
  533. GetItemRect(ht.iItem, rcBounds, LVIR_BOUNDS);
  534. CRectangle rect(this, NULL, GetIndent(pItem), rcBounds);
  535. BOOL bRedraw=0;//if OnItemExpanding or OnCollapsing returns false, dont redraw
  536. if(rect.HitTest(point))
  537. {
  538. SetRedraw(0);
  539. int nScrollIndex=0;
  540. if(IsCollapsed(pItem))
  541. {
  542. if(OnItemExpanding(pItem, ht.iItem))
  543. {
  544. nScrollIndex = Expand(pItem, ht.iItem);
  545. OnItemExpanded(pItem, ht.iItem);
  546. bRedraw=1;
  547. }
  548. }
  549. else {
  550.    if(OnCollapsing(pItem))
  551.    {
  552. Collapse(pItem);
  553. OnItemCollapsed(pItem);
  554. bRedraw=1;
  555.    }
  556. }
  557. SetRedraw(1);
  558. if(bRedraw)
  559. {
  560. CRect rc;
  561. GetItemRect(ht.iItem, rc, LVIR_BOUNDS);
  562. InvalidateRect(rc);
  563. UpdateWindow();
  564. EnsureVisible(nScrollIndex, 1);
  565. return 0;
  566. }
  567. }
  568. }//has kids
  569. }//pItem!=NULL
  570. }
  571. return 1;
  572. }
  573. void CSuperGridCtrl::OnDblclk(NMHDR* pNMHDR, LRESULT* pResult) 
  574. {
  575. if( GetFocus() != this) 
  576. SetFocus();
  577. NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
  578. LVHITTESTINFO ht;
  579. ht.pt = pNMListView->ptAction;
  580. SubItemHitTest(&ht);
  581. if(OnItemLButtonDown(ht))
  582. {
  583. BOOL bSelect=1;
  584. bSelect = HitTestOnSign(pNMListView->ptAction, ht);
  585. //normal selection
  586. if(bSelect && ht.iItem !=-1)
  587. {
  588. int nIndex = GetSelectedItem();
  589. if(nIndex!=-1)
  590. {
  591. CTreeItem *pSelItem = GetTreeItem(nIndex);
  592. if (pSelItem != NULL)
  593. {
  594. BOOL bRedraw=0;
  595. if(ItemHasChildren(pSelItem))
  596. {
  597. SetRedraw(0);
  598. int nScrollIndex=0;
  599. if(IsCollapsed(pSelItem))
  600. {
  601. if(OnItemExpanding(pSelItem, nIndex))
  602. {
  603. nScrollIndex = Expand(pSelItem, nIndex);
  604. OnItemExpanded(pSelItem, nIndex);
  605. bRedraw=1;
  606. }
  607. }
  608. else 
  609. {
  610.    if(OnCollapsing(pSelItem))
  611.    {
  612. Collapse(pSelItem);
  613. OnItemCollapsed(pSelItem);
  614. bRedraw=1;
  615.    }
  616. }
  617. SetRedraw(1);
  618. if(bRedraw)
  619. {
  620. CRect rc;
  621. GetItemRect(nIndex,rc,LVIR_BOUNDS);
  622. InvalidateRect(rc);
  623. UpdateWindow();
  624. EnsureVisible(nScrollIndex,1);
  625. }
  626. }//ItemHasChildren
  627. }//!=NULL
  628. }
  629. }
  630. }
  631. *pResult = 0;
  632. }
  633. void CSuperGridCtrl::OnBegindrag(NMHDR* pNMHDR, LRESULT* pResult) 
  634. {
  635. NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
  636. if(pNMListView->iItem != -1)
  637. {
  638. m_nDragItem = pNMListView->iItem;
  639. CImageList* pDragImage=NULL;
  640. pDragImage = CreateDragImageEx(m_nDragItem);//override this if you want another dragimage or none at all.
  641. if(pDragImage)
  642. {
  643. pDragImage->BeginDrag(0, CPoint(0,0));
  644. pDragImage->DragEnter(this, pNMListView->ptAction);
  645. SetCapture();
  646. m_bIsDragging = TRUE;
  647. }
  648. delete pDragImage;
  649. }
  650. *pResult = 0;
  651. }
  652. //Create dragimage : Icon + the itemtext
  653. CImageList *CSuperGridCtrl::CreateDragImageEx(int nItem)
  654. {
  655.     CImageList *pList = new CImageList;          
  656. //get image index
  657. LV_ITEM lvItem;
  658. lvItem.mask =  LVIF_IMAGE;
  659. lvItem.iItem = nItem;
  660. lvItem.iSubItem = 0;
  661. GetItem(&lvItem);
  662. CRect rc;
  663. GetItemRect(nItem, &rc, LVIR_BOUNDS);         
  664. CString str;
  665. str = GetItemText(nItem, 0);
  666. CFont *pFont = GetFont();
  667. rc.OffsetRect(-rc.left, -rc.top);            
  668. rc.right = GetColumnWidth(0);                
  669. pList->Create(rc.Width(), rc.Height(),ILC_COLOR24| ILC_MASK , 1, 1);
  670. CDC *pDC = GetDC();                          
  671. if(pDC) 
  672. {
  673. CDC dc;       
  674. dc.CreateCompatibleDC(pDC);      
  675. CBitmap bmpMap;
  676. bmpMap.CreateCompatibleBitmap(pDC, rc.Width(), rc.Height());
  677. CBitmap *pOldBmp = dc.SelectObject(&bmpMap);
  678. CFont *pOldFont = dc.SelectObject(pFont);
  679. dc.FillSolidRect(rc, GetSysColor(COLOR_WINDOW));
  680. CImageList *pImgList = GetImageList(LVSIL_SMALL);
  681. if(pImgList)
  682. pImgList->Draw(&dc, lvItem.iImage, CPoint(0,0), ILD_TRANSPARENT);
  683. dc.TextOut(m_cxImage + 4, 0, str);
  684. dc.SelectObject(pOldFont);
  685. dc.SelectObject(pOldBmp);                 
  686. //causes an error if the 1st column is hidden so must check the imagelist
  687. if(pList->m_hImageList != NULL)
  688. pList->Add(&bmpMap, RGB(255,255,255));
  689. else { 
  690. delete pList;
  691. pList=NULL;
  692. }
  693. ReleaseDC(pDC);   
  694. }   
  695. return pList;
  696. }
  697. void CSuperGridCtrl::OnMouseMove(UINT nFlags, CPoint point) 
  698. {
  699.     if(m_bIsDragging)
  700.     {
  701. KillTimer(1);
  702. if (CWnd::GetCapture() != this)
  703. m_bIsDragging=0;
  704. if(nFlags==MK_RBUTTON || nFlags==MK_MBUTTON)
  705. m_bIsDragging=0;
  706. if(GetKeyState(VK_ESCAPE) < 0)
  707. m_bIsDragging=0;
  708. if(!m_bIsDragging)//why not put this in a funtion :)
  709. {
  710. SetItemState (m_nDragTarget, 0, LVIS_DROPHILITED);
  711. CImageList::DragLeave(this);
  712. CImageList::EndDrag();
  713. ReleaseCapture();
  714. InvalidateRect(NULL);
  715. UpdateWindow();
  716. }
  717. else
  718. {
  719. CPoint ptList(point);
  720. MapWindowPoints(this, &ptList, 1);
  721. CImageList::DragMove(ptList);
  722. UINT uHitTest = LVHT_ONITEM;
  723. m_nDragTarget = HitTest(ptList, &uHitTest);
  724. // try turn off hilight for previous DROPHILITED state
  725. int nPrev = GetNextItem(-1,LVNI_DROPHILITED);
  726. if(nPrev != m_nDragTarget)//prevents flicker 
  727. SetItemState(nPrev, 0, LVIS_DROPHILITED);
  728. CRect rect;
  729. GetClientRect (rect);
  730. int cy = rect.Height();
  731. if(m_nDragTarget!=-1)
  732. {
  733. SetItemState(m_nDragTarget, LVIS_DROPHILITED, LVIS_DROPHILITED);
  734. CTreeItem* pTarget = GetTreeItem(m_nDragTarget);
  735. if ((point.y >= 0 && point.y <= m_cyImage) || (point.y >= cy - m_cyImage && point.y <= cy) || 
  736. ( pTarget!=NULL && ItemHasChildren(pTarget) && IsCollapsed(pTarget)))
  737. {
  738. SetTimer(1,300,NULL);
  739. }
  740. }
  741. }
  742.     }
  743. CListCtrl::OnMouseMove(nFlags, point);
  744. }
  745. void CSuperGridCtrl::OnTimer(UINT nIDEvent) 
  746. {
  747. CListCtrl::OnTimer(nIDEvent);
  748. if(nIDEvent==1)
  749. {
  750. if(CWnd::GetCapture()!=this)
  751. {
  752. SetItemState(m_nDragTarget, 0, LVIS_DROPHILITED);
  753. m_bIsDragging=0;
  754. CImageList::DragLeave(this);
  755. CImageList::EndDrag();
  756. ReleaseCapture();
  757. InvalidateRect(NULL);
  758. UpdateWindow();
  759. KillTimer(1);
  760. return;
  761. }
  762. SetTimer(1,300,NULL);//reset timer
  763. DWORD dwPos = ::GetMessagePos();
  764. CPoint ptList(LOWORD(dwPos),HIWORD(dwPos));
  765. ScreenToClient(&ptList);
  766. CRect rect;
  767. GetClientRect(rect);
  768. int cy = rect.Height();
  769. //
  770. // perform autoscroll if the cursor is near the top or bottom.
  771. //
  772. if (ptList.y >= 0 && ptList.y <= m_cyImage) 
  773. {
  774. int n = GetTopIndex(); 
  775. CImageList::DragShowNolock(0);
  776. SendMessage(WM_VSCROLL, MAKEWPARAM(SB_LINEUP, 0), NULL);
  777. CImageList::DragShowNolock(1);
  778. if (GetTopIndex()== n)
  779. KillTimer (1);
  780. else {
  781. CImageList::DragShowNolock(0);
  782. CImageList::DragMove(ptList);
  783. CImageList::DragShowNolock(1);
  784. return;
  785. }
  786. }
  787. else if (ptList.y >= cy - m_cyImage && ptList.y <= cy) 
  788. {
  789. int n = GetTopIndex(); 
  790. CImageList::DragShowNolock(0);
  791. SendMessage(WM_VSCROLL, MAKEWPARAM(SB_LINEDOWN, 0), NULL);
  792. CImageList::DragShowNolock(1);
  793. if (GetTopIndex()== n)
  794. KillTimer (1);
  795. else {
  796. CImageList::DragShowNolock(0);
  797. CImageList::DragMove(ptList);
  798. CImageList::DragShowNolock(1);
  799. return;
  800. }
  801. }
  802. //Hover test 
  803. CImageList::DragMove(ptList);
  804. UINT uHitTest = LVHT_ONITEM;
  805. m_nDragTarget = HitTest(ptList, &uHitTest);
  806. if(m_nDragTarget!=-1)
  807. {
  808. //if the target has children
  809. //expand them
  810. CTreeItem* pTarget=GetTreeItem(m_nDragTarget);
  811. if(pTarget != NULL && ItemHasChildren(pTarget) && IsCollapsed(pTarget) && (m_nDragItem!=-1))
  812. {
  813. CImageList::DragShowNolock(0);
  814. CTreeItem* pSource = GetTreeItem(m_nDragItem);
  815. SetRedraw(0);
  816. int nScrollIndex=0;
  817. if(ItemHasChildren(pTarget) && IsCollapsed(pTarget))
  818. {
  819. if(OnItemExpanding(pTarget, m_nDragTarget))
  820. {
  821. nScrollIndex = Expand(pTarget, m_nDragTarget);
  822. OnItemExpanded(pTarget, m_nDragTarget);
  823. }
  824. }
  825. m_nDragItem = NodeToIndex(pSource);
  826. SetRedraw(1);
  827. EnsureVisible(nScrollIndex, 1);
  828. InvalidateRect(NULL);
  829. UpdateWindow();
  830. CImageList::DragShowNolock(1);
  831. KillTimer(1);
  832. return;
  833. }
  834. }
  835. KillTimer(1);
  836. }
  837. }
  838. void CSuperGridCtrl::OnLButtonUp(UINT nFlags, CPoint point) 
  839. {
  840. if(m_bIsDragging == TRUE)
  841.     {
  842. KillTimer(1);
  843.         CImageList::DragLeave(this);
  844.         CImageList::EndDrag();
  845.         ReleaseCapture();
  846.         m_bIsDragging = FALSE;
  847. SetItemState(m_nDragTarget, 0, LVIS_DROPHILITED);
  848.         if((m_nDragTarget != -1) && (m_nDragTarget != m_nDragItem) && (m_nDragItem!=-1))//no drop on me self
  849.         {
  850. CTreeItem* pSource = GetTreeItem(m_nDragItem);
  851. CTreeItem* pTarget = GetTreeItem(m_nDragTarget);
  852. if(IsRoot(pSource))
  853. return;
  854. CTreeItem* pParent = GetParentItem(pSource);
  855. if(pParent==pTarget) //can't drag child to parent
  856. return;
  857. if(!IsChildOf(pSource, pTarget))//can't drag parent to child
  858. {
  859. CWaitCursor wait;
  860. SetRedraw(0);
  861. if(DoDragDrop(pTarget, pSource))
  862. {
  863. UINT uflag = LVIS_SELECTED | LVIS_FOCUSED;
  864. SetItemState(m_nDragTarget, uflag, uflag);
  865. m_nDragItem=-1;
  866. //delete source
  867. int nIndex = NodeToIndex(pSource);
  868. DeleteItem(nIndex);
  869. HideChildren(pSource, TRUE, nIndex);
  870. Delete(pSource);
  871. InternaleUpdateTree();
  872. SetRedraw(1);
  873. InvalidateRect(NULL);
  874. UpdateWindow();
  875. }else
  876. SetRedraw(1);
  877. }
  878.     }
  879.     }
  880.     else
  881. CListCtrl::OnLButtonUp(nFlags, point);
  882. }
  883. //used with the drag/drop operation
  884. void CSuperGridCtrl::CopyChildren(CTreeItem* pDest, CTreeItem* pSrc)
  885. {
  886. if (ItemHasChildren(pSrc))
  887. {
  888. POSITION pos = pSrc->m_listChild.GetHeadPosition();
  889. while (pos != NULL)
  890. {
  891. CTreeItem* pItem = (CTreeItem *)pSrc->m_listChild.GetNext(pos);
  892. CItemInfo* lp = CopyData(GetData(pItem));
  893. CTreeItem* pNewItem = InsertItem(pDest, lp);
  894. CopyChildren(pNewItem, pItem);
  895. }
  896. }
  897. }
  898. //hmmm 
  899. BOOL CSuperGridCtrl::DoDragDrop(CTreeItem* pTarget, CTreeItem* pSource)
  900. {
  901. if(pTarget==NULL)
  902. return 0;
  903. BOOL bUpdate=FALSE;
  904. if(!IsCollapsed(pTarget))
  905. bUpdate=TRUE; //children are expanded, want to see update right away
  906. //make a copy of the source data
  907. CItemInfo* lp = CopyData(GetData(pSource));
  908. //create new node with the source data and make pTarget the parent
  909. CTreeItem* pNewParent = InsertItem(pTarget, lp, bUpdate);
  910. //if the source has children copy all source data and make the newly create item the parent
  911. if(ItemHasChildren(pSource))
  912. CopyChildren(pNewParent, pSource);
  913. return 1;
  914. }
  915. void CSuperGridCtrl::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar) 
  916. {
  917. //its not meself
  918. if( GetFocus() != this) 
  919. SetFocus();
  920. CListCtrl::OnHScroll(nSBCode, nPos, pScrollBar);
  921. }
  922. void CSuperGridCtrl::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar) 
  923. {
  924. if( GetFocus() != this) 
  925. SetFocus();
  926. CListCtrl::OnVScroll(nSBCode, nPos, pScrollBar);
  927. }
  928. BOOL CSuperGridCtrl::PreTranslateMessage(MSG* pMsg) 
  929. {
  930. if(pMsg->message == WM_KEYDOWN)
  931. {
  932. if(GetFocus()==this)
  933. {
  934. switch( pMsg->wParam )
  935. {
  936. case VK_LEFT:
  937. {
  938. // Decrement the order number.
  939. m_CurSubItem--;
  940. if(m_CurSubItem < 0) 
  941. m_CurSubItem = 0;
  942. else{
  943. CHeaderCtrl* pHeader = GetHeaderCtrl();
  944. // Make the column visible.
  945. // We have to take into account that the header may be reordered.
  946. MakeColumnVisible( Header_OrderToIndex( pHeader->m_hWnd, m_CurSubItem));
  947. // Invalidate the item.
  948. int iItem = GetSelectedItem();
  949. if( iItem != -1 )
  950. {
  951. CRect rcBounds;
  952. GetItemRect(iItem, rcBounds, LVIR_BOUNDS);
  953. InvalidateRect(&rcBounds);
  954. UpdateWindow();
  955. }
  956. }
  957. }
  958. return TRUE;
  959. case VK_RIGHT:
  960. {
  961. // Increment the order number.
  962. m_CurSubItem++;
  963. CHeaderCtrl* pHeader = GetHeaderCtrl();
  964. int nColumnCount = pHeader->GetItemCount();
  965. // Don't go beyond the last column.
  966. if( m_CurSubItem > nColumnCount -1 ) 
  967. m_CurSubItem = nColumnCount-1;
  968. else
  969. {
  970. MakeColumnVisible(Header_OrderToIndex( pHeader->m_hWnd, m_CurSubItem));
  971.  
  972. int iItem = GetSelectedItem();
  973. // Invalidate the item.
  974. if( iItem != -1 )
  975. {
  976. CRect rcBounds;
  977. GetItemRect(iItem, rcBounds, LVIR_BOUNDS);
  978. InvalidateRect(&rcBounds);
  979. UpdateWindow();
  980. }
  981. }
  982. }
  983. return TRUE;
  984. case VK_RETURN://edit itemdata
  985. {
  986. BOOL bResult = OnVkReturn();
  987. if(!bResult)
  988. {
  989. int iItem = GetSelectedItem();
  990. if( m_CurSubItem != -1 && iItem != -1)
  991. {
  992. CHeaderCtrl* pHeader = GetHeaderCtrl();
  993. int iSubItem = Header_OrderToIndex(pHeader->m_hWnd, m_CurSubItem);
  994. if(iSubItem==0)//that's just me saying all nodes in col 0 are edit-controls, you may modify this
  995. {
  996. CRect rcItem;
  997. GetItemRect(iItem, rcItem, LVIR_LABEL);
  998. DWORD dwStyle = WS_BORDER | WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL|ES_LEFT;
  999. CEdit *pEdit = new CListEditCtrl(iItem, iSubItem, GetItemText(iItem, iSubItem));
  1000. pEdit->Create(dwStyle, rcItem, this, 0x1233);
  1001. }
  1002. else
  1003. EditLabelEx(iItem, iSubItem);
  1004. return 1;
  1005. }
  1006. }
  1007. }
  1008. break;
  1009. default:
  1010. break;
  1011. }
  1012. }
  1013. }
  1014. return CListCtrl::PreTranslateMessage(pMsg);
  1015. }
  1016. #define IDC_EDITCTRL 0x1234
  1017. CEdit* CSuperGridCtrl::EditLabelEx(int nItem, int nCol)
  1018. {
  1019. CRect rect;
  1020. int offset = 0;
  1021. if(!EnsureVisible(nItem, TRUE)) 
  1022. return NULL;
  1023. GetSubItemRect(nItem, nCol, LVIR_BOUNDS, rect);
  1024. // Now scroll if we need to expose the column
  1025. CRect rcClient;
  1026. GetClientRect(rcClient);
  1027. if( offset + rect.left < 0 || offset + rect.left > rcClient.right )
  1028. {
  1029. CSize size(offset + rect.left,0);
  1030. Scroll(size);
  1031. rect.left -= size.cx;
  1032. }
  1033. rect.left += offset;
  1034. rect.right = rect.left + GetColumnWidth(nCol);
  1035. if(rect.right > rcClient.right) 
  1036.    rect.right = rcClient.right;
  1037. // Get Column alignment
  1038. LV_COLUMN lvcol;
  1039. lvcol.mask = LVCF_FMT;
  1040. GetColumn(nCol, &lvcol);
  1041. DWORD dwStyle;
  1042. if((lvcol.fmt & LVCFMT_JUSTIFYMASK) == LVCFMT_LEFT)
  1043. dwStyle = ES_LEFT;
  1044. else if((lvcol.fmt & LVCFMT_JUSTIFYMASK) == LVCFMT_RIGHT)
  1045. dwStyle = ES_RIGHT;
  1046. else 
  1047. dwStyle = ES_CENTER;
  1048. dwStyle |=WS_BORDER|WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL;
  1049. CEdit *pEdit = new CListEditCtrl(nItem, nCol, GetItemText(nItem, nCol));
  1050. pEdit->Create(dwStyle, rect, this, IDC_EDITCTRL);
  1051. //pEdit->ModifyStyleEx(0,WS_EX_CLIENTEDGE); //funny thing happend here, uncomment this, 
  1052. //and then edit an item, 
  1053. //enter a long text so that the ES_AUTOHSCROLL comes to rescue
  1054. //yes that's look funny, ???.
  1055. return pEdit;
  1056. }
  1057. void CSuperGridCtrl::OnEndlabeledit(NMHDR* pNMHDR, LRESULT* pResult) 
  1058. {
  1059. LV_DISPINFO *plvDispInfo = (LV_DISPINFO*)pNMHDR;
  1060.   LV_ITEM *plvItem = &plvDispInfo->item;
  1061. if (plvItem->pszText != NULL)//valid text
  1062. {
  1063. if(plvItem->iItem != -1) //valid item
  1064. {
  1065. CTreeItem*pSelItem = GetTreeItem(plvItem->iItem);
  1066. if(pSelItem != NULL)
  1067. {
  1068. OnUpdateListViewItem(pSelItem, plvItem);
  1069. }
  1070. }
  1071. }
  1072. *pResult = 0;
  1073. }
  1074. int CSuperGridCtrl::GetNumCol()
  1075. {
  1076. CHeaderCtrl* pHeader = GetHeaderCtrl();
  1077. return pHeader ? pHeader->GetItemCount() : 0;
  1078. }
  1079. //Think Rex Myer is spooking here
  1080. void CSuperGridCtrl::MakeColumnVisible(int nCol)
  1081. {
  1082. if(nCol < 0)
  1083. return;
  1084. // Get the order array to total the column offset.
  1085. CHeaderCtrl* pHeader = GetHeaderCtrl();
  1086. int nColCount = pHeader->GetItemCount();
  1087. ASSERT( nCol < nColCount);
  1088. int *pOrderarray = new int[nColCount];
  1089. Header_GetOrderArray(pHeader->m_hWnd, nColCount, pOrderarray);
  1090. // Get the column offset
  1091. int offset = 0;
  1092. for(int i = 0; pOrderarray[i] != nCol; i++)
  1093. offset += GetColumnWidth(pOrderarray[i]);
  1094. int colwidth = GetColumnWidth(nCol);
  1095. delete[] pOrderarray;
  1096. CRect rect;
  1097. GetItemRect(0, &rect, LVIR_BOUNDS);
  1098. // Now scroll if we need to show the column
  1099. CRect rcClient;
  1100. GetClientRect(&rcClient);
  1101. if(offset + rect.left < 0 || offset + colwidth + rect.left > rcClient.right)
  1102. {
  1103. CSize size(offset + rect.left,0);
  1104. Scroll(size);
  1105. InvalidateRect(NULL);
  1106. UpdateWindow();
  1107. }
  1108. }
  1109. //Think Rex Myer is spooking here
  1110. int CSuperGridCtrl::IndexToOrder( int iIndex )
  1111. {
  1112. // This translates a column index value to a column order value.
  1113. CHeaderCtrl* pHeader = GetHeaderCtrl();
  1114. int nColCount = pHeader->GetItemCount();
  1115. int *pOrderarray = new int[nColCount];
  1116. Header_GetOrderArray(pHeader->m_hWnd, nColCount, pOrderarray);
  1117. for(int i=0; i<nColCount; i++)
  1118. {
  1119. if(pOrderarray[i] == iIndex )
  1120. {
  1121. delete[] pOrderarray;
  1122. return i;
  1123. }
  1124. }
  1125. delete[] pOrderarray;
  1126. return -1;
  1127. }
  1128. void CSuperGridCtrl::DrawFocusCell(CDC *pDC, int nItem, int iSubItem)
  1129. {
  1130. if(iSubItem==m_CurSubItem)
  1131. {
  1132. CRect rect;
  1133. GetSubItemRect(nItem, iSubItem, LVIR_BOUNDS, rect);
  1134. CBrush br(GetCellRGB());
  1135. if(iSubItem==0)
  1136. GetItemRect(iSubItem, rect, LVIR_LABEL);
  1137. pDC->FillRect(rect, &br);
  1138. pDC->DrawFocusRect(rect);
  1139. }
  1140. }
  1141. //insert item and return new parent pointer.
  1142. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::InsertItem(CTreeItem *pParent, CItemInfo* lpInfo,  BOOL bUpdate)
  1143. {
  1144. if(pParent==NULL)
  1145. return NULL;
  1146. CTreeItem *pItem = NULL;
  1147. pItem =  new CTreeItem();
  1148. if(lpInfo==NULL)
  1149. lpInfo = new CItemInfo;
  1150. UpdateData(pItem, lpInfo);
  1151. SetIndent(pItem, GetIndent(pParent)+1);
  1152. SetParentItem(pItem, pParent);
  1153. //add as the last child 
  1154. pParent->m_listChild.AddTail(pItem);
  1155. if(!bUpdate)
  1156.  Hide(pParent, TRUE);
  1157. else
  1158. {
  1159. //calc listview index for the new node
  1160. int nIndex = NodeToIndex(pItem);
  1161. CString str = GetData(pItem)->GetItemText();
  1162. LV_ITEM     lvItem;
  1163. lvItem.mask = LVIF_TEXT | LVIF_INDENT | LVIF_PARAM;
  1164. lvItem.pszText = str.GetBuffer(1); 
  1165. //insert item
  1166. lvItem.iItem = nIndex;
  1167. lvItem.iSubItem = 0;
  1168. lvItem.lParam = (LPARAM)pItem;
  1169. lvItem.iIndent = GetIndent(pItem);
  1170. CListCtrl::InsertItem(&lvItem);
  1171. if(lpInfo->GetCheck())
  1172. SetCheck(nIndex);
  1173. //Get subitems
  1174. int nSize = GetData(pItem)->GetItemCount();
  1175. for(int i=0; i < nSize;i++)
  1176. {
  1177.    CString str = GetData(pItem)->GetSubItem(i);
  1178.    lvItem.mask = LVIF_TEXT;
  1179.    lvItem.iSubItem = i+1;
  1180.    lvItem.pszText = str.GetBuffer(1);
  1181.    SetItem(&lvItem);
  1182. }
  1183. InternaleUpdateTree();//better do this
  1184. }
  1185. return pItem;
  1186. }
  1187. void CSuperGridCtrl::InternaleUpdateTree()
  1188. {
  1189. int nItems = GetItemCount();
  1190. for(int nItem=0; nItem < nItems; nItem++)
  1191. {
  1192. CTreeItem* pItem = GetTreeItem(nItem);
  1193. SetCurIndex(pItem, nItem);
  1194. }
  1195. }
  1196. int CSuperGridCtrl::NodeToIndex(CTreeItem *pNode)
  1197. {
  1198. int nStartIndex=0;
  1199. POSITION pos = m_RootItems.GetHeadPosition();
  1200. while(pos!=NULL)
  1201. {
  1202. CTreeItem * root = (CTreeItem*)m_RootItems.GetNext(pos);
  1203. int ret = _NodeToIndex(root, pNode, nStartIndex);
  1204. if(ret != -1)
  1205. return ret;
  1206. }
  1207. return -1;
  1208. }
  1209. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetRootItem(int nIndex)
  1210. {
  1211. POSITION pos = m_RootItems.FindIndex(nIndex);
  1212. if(pos==NULL)
  1213. return NULL;
  1214. return (CTreeItem*)m_RootItems.GetAt(pos);
  1215. }
  1216. int CSuperGridCtrl::GetRootIndex(CTreeItem * root)
  1217. {
  1218. int nIndex = 0;
  1219. POSITION pos = m_RootItems.GetHeadPosition();
  1220. while(pos != NULL)
  1221. {
  1222. CTreeItem * pItem = (CTreeItem*)m_RootItems.GetNext(pos);
  1223. if(pItem== root)
  1224. return nIndex;
  1225. nIndex++;
  1226. }
  1227. return -1;
  1228. }
  1229. BOOL CSuperGridCtrl::IsRoot(CTreeItem * lpItem)
  1230. {
  1231. return m_RootItems.Find(lpItem) != NULL;
  1232. }
  1233. void CSuperGridCtrl::DeleteRootItem(CTreeItem * root)
  1234. {
  1235. POSITION pos = m_RootItems.Find(root);
  1236. if(pos!=NULL)
  1237. {
  1238. CTreeItem* pRoot=(CTreeItem*)m_RootItems.GetAt(pos);
  1239. if(pRoot->m_lpNodeInfo!=NULL)
  1240. delete pRoot->m_lpNodeInfo;
  1241. delete pRoot;
  1242. m_RootItems.RemoveAt(pos);
  1243. }
  1244. }
  1245. CSuperGridCtrl::CTreeItem*  CSuperGridCtrl::InsertRootItem(CItemInfo * lpInfo)
  1246. {
  1247. if(lpInfo==NULL)
  1248. lpInfo = new CItemInfo;
  1249. CTreeItem* pRoot = NULL;
  1250. pRoot =  new CTreeItem();
  1251. CleanMe(pRoot);
  1252. UpdateData(pRoot, lpInfo);
  1253. SetIndent(pRoot, 1);
  1254. SetCurIndex(pRoot, GetItemCount());
  1255. SetParentItem(pRoot, NULL);
  1256. CItemInfo* lp = GetData(pRoot);
  1257. LV_ITEM lvItem;
  1258. lvItem.mask = LVIF_TEXT | LVIF_INDENT | LVIF_PARAM;
  1259. CString strItem = lp->GetItemText();
  1260. lvItem.pszText = strItem.GetBuffer(1); 
  1261. lvItem.iItem = GetItemCount();
  1262. lvItem.lParam = (LPARAM)pRoot;
  1263. lvItem.iIndent = 1;
  1264. lvItem.iSubItem = 0;
  1265. CListCtrl::InsertItem(&lvItem);
  1266. int nSize = lp->GetItemCount();
  1267. for(int i=0; i < nSize;i++)
  1268. {
  1269.    CString str = lp->GetSubItem(i);
  1270.    lvItem.mask = LVIF_TEXT;
  1271.    lvItem.iSubItem = i+1;
  1272.    lvItem.pszText = str.GetBuffer(1);
  1273.    SetItem(&lvItem);
  1274. }
  1275. m_RootItems.AddTail(pRoot);
  1276. return pRoot;
  1277. }
  1278. void CSuperGridCtrl::DrawTreeItem(CDC* pDC, CTreeItem* pSelItem, int nListItem, const CRect& rcBounds)
  1279. {
  1280. int nColWidth = GetColumnWidth(0);
  1281. int yDown = rcBounds.top;
  1282.     CPen* pPenTreeLine = pDC->SelectObject(&m_psTreeLine);
  1283. int iIndent = GetIndent(pSelItem);
  1284. int nHalfImage = (m_cxImage >> 1);
  1285. int nBottomDown = yDown + nHalfImage + ((rcBounds.Height() - m_cyImage) >> 1);
  1286. //
  1287. BOOL bChild = ItemHasChildren(pSelItem);
  1288. BOOL bCollapsed = IsCollapsed(pSelItem);
  1289. //draw outline
  1290. while(1)
  1291. {
  1292. CTreeItem* pParent = GetParentItem(pSelItem);
  1293. if(pParent==NULL)//no more parents, stop
  1294. break;
  1295. POSITION pos = pParent->m_listChild.GetTailPosition();
  1296. while(pos!=NULL)
  1297. {
  1298. CTreeItem *pLastChild = (CTreeItem*)pParent->m_listChild.GetPrev(pos);
  1299. int nIndex = GetCurIndex(pLastChild);
  1300. int nCurIndent = GetIndent(pLastChild);
  1301. if(nListItem > nIndex && iIndent > nCurIndent)//no need to go further in this loop
  1302. break;
  1303. //no drawing outside the 1st columns right
  1304. int xLine =  rcBounds.left + nCurIndent * m_cxImage - nHalfImage;
  1305. if(nIndex == nListItem && nCurIndent==iIndent)
  1306. {
  1307. //draw '-
  1308. int x;
  1309. pDC->MoveTo(xLine, yDown);
  1310. pDC->LineTo(xLine, nBottomDown);
  1311. // -
  1312. xLine + nHalfImage > nColWidth ? x = nColWidth: x = xLine + nHalfImage;
  1313. pDC->MoveTo(xLine, nBottomDown);
  1314. pDC->LineTo(x, nBottomDown);
  1315. break;
  1316. }
  1317. else
  1318. if(nIndex > nListItem && nCurIndent==iIndent)
  1319. {
  1320. //draw |-
  1321. int x;
  1322. xLine + nHalfImage > nColWidth ? x = nColWidth : x = xLine + nHalfImage;
  1323. pDC->MoveTo(xLine, nBottomDown);
  1324. pDC->LineTo(x, nBottomDown);
  1325. //-
  1326. pDC->MoveTo(xLine, yDown);
  1327. pDC->LineTo(xLine, rcBounds.bottom);
  1328. break;
  1329. }
  1330. else
  1331. if(nIndex > nListItem && nCurIndent < iIndent)
  1332. {
  1333. //draw |
  1334. pDC->MoveTo(xLine, yDown);
  1335. pDC->LineTo(xLine, rcBounds.bottom);
  1336. break;
  1337. }
  1338. }
  1339. pSelItem = pParent;//next
  1340. }
  1341. //draw plus/minus sign
  1342. if(bChild)
  1343. {
  1344. CRectangle rect(this, pDC, iIndent, rcBounds);
  1345. rect.DrawRectangle(this);
  1346. CPen* pPenPlusMinus = pDC->SelectObject(&m_psPlusMinus);
  1347. if(bCollapsed)
  1348. rect.DrawPlus();
  1349. else {
  1350. rect.DrawMinus();
  1351. //draw line up to parent folder
  1352. CPen* pLine = pDC->SelectObject(&m_psTreeLine);
  1353. int nOffset = (rcBounds.Height() - m_cyImage)/2;
  1354. pDC->MoveTo(rect.GetLeft()+ m_cxImage, rcBounds.top + m_cyImage+nOffset);
  1355. pDC->LineTo(rect.GetLeft() + m_cxImage, rcBounds.bottom);
  1356. pDC->SelectObject(pLine);
  1357. }
  1358. pDC->SelectObject(pPenPlusMinus);
  1359. }
  1360. pDC->SelectObject(pPenTreeLine);
  1361. }
  1362. //walk all over the place setting the hide/show flag of the nodes.
  1363. //it also deletes items from the listviewctrl.
  1364. void CSuperGridCtrl::HideChildren(CTreeItem *pItem, BOOL bHide,int nItem)
  1365. {
  1366. if(!IsCollapsed(pItem))
  1367. if(ItemHasChildren(pItem))
  1368. {
  1369. Hide(pItem, bHide);
  1370. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1371. while (pos != NULL)
  1372. {
  1373. HideChildren((CTreeItem *)pItem->m_listChild.GetNext(pos),bHide,nItem+1);
  1374. DeleteItem(nItem);
  1375. }
  1376. }
  1377. }
  1378. void CSuperGridCtrl::Collapse(CTreeItem *pItem)
  1379. {
  1380. if(pItem != NULL && ItemHasChildren(pItem))
  1381. {
  1382. SetRedraw(0);
  1383. int nIndex = NodeToIndex(pItem);
  1384. HideChildren(pItem, TRUE, nIndex+1);
  1385. InternaleUpdateTree();
  1386. SetRedraw(1);
  1387. }
  1388. }
  1389. void CSuperGridCtrl::ExpandAll(CTreeItem *pItem, int& nScroll)
  1390. {
  1391. const int nChildren = pItem->m_listChild.GetCount();
  1392. if (nChildren > 0)
  1393. {
  1394. int nIndex = NodeToIndex(pItem);
  1395. nScroll = Expand(pItem, nIndex);
  1396. }
  1397. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1398. while (pos)
  1399. {
  1400. CTreeItem *pChild = (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1401. ExpandAll(pChild, nScroll);
  1402. }
  1403. }
  1404. int CSuperGridCtrl::Expand(CTreeItem* pSelItem, int nIndex)
  1405. {
  1406. if(ItemHasChildren(pSelItem) && IsCollapsed(pSelItem))
  1407. {
  1408. LV_ITEM lvItem;
  1409. lvItem.mask = LVIF_INDENT;
  1410. lvItem.iItem = nIndex;
  1411. lvItem.iSubItem = 0;
  1412. lvItem.lParam=(LPARAM)pSelItem;
  1413. lvItem.iIndent = GetIndent(pSelItem);
  1414. SetItem(&lvItem);
  1415. Hide(pSelItem, FALSE);
  1416. //expand children
  1417. POSITION pos = pSelItem->m_listChild.GetHeadPosition();
  1418. while(pos != NULL)
  1419. {
  1420. CTreeItem* pNextNode = (CTreeItem*)pSelItem->m_listChild.GetNext(pos);
  1421. CString str = GetData(pNextNode)->GetItemText();
  1422. LV_ITEM lvItem;
  1423. lvItem.mask = LVIF_TEXT | LVIF_INDENT | LVIF_PARAM;
  1424. lvItem.pszText =str.GetBuffer(1); 
  1425. lvItem.iItem = nIndex + 1;
  1426. lvItem.iSubItem = 0;
  1427. lvItem.lParam=(LPARAM)pNextNode;
  1428. lvItem.iIndent = GetIndent(pSelItem)+1;
  1429. CListCtrl::InsertItem(&lvItem);
  1430. if(GetData(pNextNode)->GetCheck())
  1431. SetCheck(nIndex + 1);
  1432. //get subitems
  1433. int nSize = GetData(pNextNode)->GetItemCount();
  1434. for(int i=0; i< nSize;i++)
  1435. {
  1436.    CString str=GetData(pNextNode)->GetSubItem(i);
  1437.    lvItem.mask = LVIF_TEXT;
  1438.    lvItem.iSubItem = i+1;
  1439.    lvItem.pszText=str.GetBuffer(1);
  1440.    SetItem(&lvItem);
  1441. }
  1442. nIndex++;
  1443. }
  1444. }
  1445. InternaleUpdateTree();
  1446. return nIndex;
  1447. }
  1448. int CSuperGridCtrl::SelectNode(CTreeItem *pLocateNode)
  1449. {
  1450. if(IsRoot(pLocateNode))
  1451. {
  1452. UINT uflag = LVIS_SELECTED | LVIS_FOCUSED;
  1453. SetItemState(0, uflag, uflag);
  1454. return 0;
  1455. }
  1456. int nSelectedItem=-1;
  1457. CTreeItem* pNode = pLocateNode;
  1458. CTreeItem* pTopLevelParent=NULL;
  1459. //Get top parent
  1460. while(1)
  1461. {
  1462. CTreeItem *pParent = GetParentItem(pLocateNode);
  1463. if(IsRoot(pParent))
  1464. break;
  1465. pLocateNode = pParent;
  1466. }
  1467. pTopLevelParent = pLocateNode;//on top of all
  1468. //Expand the folder
  1469. if(pTopLevelParent != NULL)
  1470. {
  1471. SetRedraw(0);
  1472. CWaitCursor wait;
  1473. CTreeItem *pRoot = GetParentItem(pTopLevelParent);
  1474. if(IsCollapsed(pRoot))
  1475. Expand(pRoot,0);
  1476. ExpandUntil(pTopLevelParent, pNode);
  1477. UINT uflag = LVIS_SELECTED | LVIS_FOCUSED;
  1478. nSelectedItem = NodeToIndex(pNode);
  1479. SetItemState(nSelectedItem, uflag, uflag);
  1480. SetRedraw(1);
  1481. EnsureVisible(nSelectedItem, TRUE);
  1482. }
  1483. return nSelectedItem;
  1484. }
  1485. void CSuperGridCtrl::ExpandUntil(CTreeItem *pItem, CTreeItem* pStopAt)
  1486. {
  1487. const int nChildren = pItem->m_listChild.GetCount();
  1488. if (nChildren > 0)
  1489. {
  1490. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1491. while (pos)
  1492. {
  1493. CTreeItem *pChild = (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1494. if(pChild == pStopAt)
  1495. {
  1496. int nSize = GetIndent(pChild);
  1497. CTreeItem** ppParentArray = new CTreeItem*[nSize];
  1498. int i=0;
  1499. while(1)
  1500. {
  1501. CTreeItem *pParent = GetParentItem(pChild);
  1502. if(IsRoot(pParent))
  1503. break;
  1504. pChild = pParent;
  1505. ppParentArray[i] = pChild;
  1506. i++;
  1507. }
  1508. for(int x=i; x > 0; x--)
  1509. {
  1510. CTreeItem *pParent = ppParentArray[x-1];
  1511. Expand(pParent, NodeToIndex(pParent));
  1512. }
  1513. delete [] ppParentArray;
  1514. return;
  1515. }
  1516. }
  1517. }
  1518. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1519. while (pos)
  1520. {
  1521. CTreeItem *pChild = (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1522. ExpandUntil(pChild, pStopAt);
  1523. }
  1524. }
  1525. void CSuperGridCtrl::DeleteItemEx(CTreeItem *pSelItem, int nItem)
  1526. {
  1527. SetRedraw(0);
  1528. DeleteItem(nItem);//delete cur item in listview
  1529. //delete/hide all children in pSelItem
  1530. HideChildren(pSelItem, TRUE, nItem);
  1531. //delete all internal nodes
  1532. // If root, must delete from m_rootData
  1533. if(GetParentItem(pSelItem) == NULL )
  1534. {
  1535. DeleteRootItem(pSelItem);
  1536. }
  1537. else
  1538. Delete(pSelItem);
  1539. InternaleUpdateTree();
  1540. if(nItem-1<0)//no more items in list
  1541. {
  1542. SetRedraw(1); 
  1543. InvalidateRect(NULL);
  1544. UpdateWindow();
  1545. return;
  1546. }
  1547. UINT uflag = LVIS_SELECTED | LVIS_FOCUSED;
  1548. CRect rcTestIfItemIsValidToSelectOtherWiseSubtrackOneFromItem;//just to get the documention right :)
  1549. GetItemRect(nItem, rcTestIfItemIsValidToSelectOtherWiseSubtrackOneFromItem ,LVIR_LABEL) ? SetItemState(nItem, uflag, uflag) : SetItemState(nItem-1, uflag, uflag);
  1550. SetRedraw(1);
  1551. InvalidateRect(NULL);
  1552. UpdateWindow();
  1553. }
  1554. void CSuperGridCtrl::CleanMe(CTreeItem *pItem)
  1555. {
  1556. // delete child nodes
  1557. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1558. while (pos != NULL)
  1559. {
  1560. CTreeItem* pItemData = (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1561. if(pItemData!=NULL)
  1562. {
  1563. if(pItemData->m_lpNodeInfo!=NULL)
  1564. delete pItemData->m_lpNodeInfo;
  1565. pItemData->m_listChild.RemoveAll();
  1566. delete pItemData;
  1567. }
  1568. }
  1569. pItem->m_listChild.RemoveAll();
  1570. }
  1571. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetNext(CTreeItem* pStartAt, CTreeItem* pNode, BOOL bInit, BOOL bDontIncludeHidden)
  1572. {
  1573. static BOOL bFound;
  1574. if (bInit)
  1575. bFound = FALSE;
  1576. if (pNode == pStartAt)
  1577. bFound = TRUE;
  1578. if(bDontIncludeHidden)
  1579. {
  1580. if (!IsCollapsed(pStartAt))
  1581. {
  1582. POSITION pos = pStartAt->m_listChild.GetHeadPosition();
  1583. while (pos != NULL)
  1584. {
  1585. CTreeItem* pChild = (CTreeItem*)pStartAt->m_listChild.GetNext(pos);
  1586. if (bFound)
  1587. return pChild;
  1588. pChild = GetNext(pChild, pNode, FALSE, TRUE);
  1589. if (pChild != NULL)
  1590. return pChild;
  1591. }
  1592. }
  1593. }
  1594. else {
  1595. POSITION pos = pStartAt->m_listChild.GetHeadPosition();
  1596. while (pos != NULL)
  1597. {
  1598. CTreeItem* pChild = (CTreeItem*)pStartAt->m_listChild.GetNext(pos);
  1599. if (bFound)
  1600. return pChild;
  1601. pChild = GetNext(pChild, pNode, FALSE,FALSE);
  1602. if (pChild != NULL)
  1603. return pChild;
  1604. }
  1605. }
  1606. // if reached top and last level return original
  1607. if (bInit)
  1608. return pNode;
  1609. else
  1610. return NULL;
  1611. }
  1612. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetPrev(CTreeItem* pStartAt, CTreeItem* pNode, BOOL bInit, BOOL bDontIncludeHidden)
  1613. {
  1614. static CTreeItem* pPrev;
  1615. if (bInit)
  1616. pPrev = pStartAt;
  1617. if (pNode == pStartAt)
  1618. return pPrev;
  1619. pPrev = pStartAt;
  1620. if(bDontIncludeHidden)
  1621. {
  1622. if (!IsCollapsed(pStartAt))
  1623. {
  1624. POSITION pos = pStartAt->m_listChild.GetHeadPosition();
  1625. while (pos != NULL)
  1626. {
  1627. CTreeItem* pCur = (CTreeItem*)pStartAt->m_listChild.GetNext(pos);
  1628. CTreeItem* pChild = GetPrev(pCur,pNode, FALSE,TRUE);
  1629. if (pChild != NULL)
  1630. return pChild;
  1631. }
  1632. }
  1633. }
  1634. else {
  1635. POSITION pos = pStartAt->m_listChild.GetHeadPosition();
  1636. while (pos != NULL)
  1637. {
  1638. CTreeItem* pCur = (CTreeItem*)pStartAt->m_listChild.GetNext(pos);
  1639. CTreeItem* pChild = GetPrev(pCur,pNode, FALSE,FALSE);
  1640. if (pChild != NULL)
  1641. return pChild;
  1642. }
  1643. }
  1644. if (bInit)
  1645. return pPrev;
  1646. else
  1647. return NULL;
  1648. }
  1649. int CSuperGridCtrl::_NodeToIndex(CTreeItem *pStartpos, CTreeItem* pNode, int& nIndex, BOOL binit)
  1650. {
  1651. static BOOL bFound;
  1652. // Account for other root nodes
  1653. if(GetParentItem(pStartpos) == NULL && GetRootIndex(pStartpos) !=0)
  1654. nIndex++;
  1655. if(binit)
  1656. bFound=FALSE;
  1657. if(pStartpos==pNode)
  1658. bFound=TRUE;
  1659. if(!IsCollapsed(pStartpos))
  1660. {
  1661. POSITION pos = GetHeadPosition(pStartpos);
  1662. while (pos)
  1663. {
  1664. CTreeItem *pChild = GetNextChild(pStartpos, pos);
  1665. if(bFound)
  1666. return nIndex;
  1667. // Craig Schmidt: Cannot set nIndex as return value.  Worked find with single root but
  1668. //    the calling function get confused since the return value may indicate
  1669. //    that the next root needs to be searched.  Didn'd spend much time on
  1670. //    this so there is probably a better way of doing this.
  1671. // nIndex = _NodeToIndex(pChild, pNode, nIndex, binit);
  1672. _NodeToIndex(pChild, pNode, nIndex, binit);
  1673. nIndex++;
  1674. }
  1675. }
  1676. if(binit && bFound)
  1677. return nIndex;
  1678. else
  1679. return -1;
  1680. }
  1681. BOOL CSuperGridCtrl::Delete(CTreeItem* pNode, BOOL bClean)
  1682. {
  1683. POSITION pos = m_RootItems.GetHeadPosition();
  1684. while(pos!=NULL)
  1685. {
  1686. CTreeItem * pRoot = (CTreeItem*)m_RootItems.GetNext(pos);
  1687. if(_Delete(pRoot, pNode, bClean))
  1688. return TRUE;
  1689. }
  1690. return FALSE;
  1691. }
  1692. BOOL CSuperGridCtrl::_Delete(CTreeItem* pStartAt, CTreeItem* pNode, BOOL bClean)
  1693. {
  1694. POSITION pos = pStartAt->m_listChild.GetHeadPosition();
  1695. while (pos != NULL)
  1696. {
  1697. POSITION posPrev = pos;
  1698. CTreeItem *pChild = (CTreeItem*)pStartAt->m_listChild.GetNext(pos);
  1699. if (pChild == pNode)
  1700. {
  1701. pStartAt->m_listChild.RemoveAt(posPrev);
  1702. if(bClean)
  1703. {
  1704. if(GetData(pNode)!=NULL)
  1705. delete GetData(pNode);
  1706. delete pNode;
  1707. }
  1708. return TRUE;
  1709. }
  1710. if (_Delete(pChild, pNode) == TRUE)
  1711. return TRUE;
  1712. }
  1713. return FALSE;
  1714. }
  1715. BOOL CSuperGridCtrl::IsChildOf(const CTreeItem* pParent, const CTreeItem* pChild) const
  1716. {
  1717. if (pChild == pParent)
  1718. return TRUE;
  1719. POSITION pos = pParent->m_listChild.GetHeadPosition();
  1720. while (pos != NULL)
  1721. {
  1722. CTreeItem* pNode = (CTreeItem*)pParent->m_listChild.GetNext(pos);
  1723. if (IsChildOf(pNode, pChild))
  1724. return TRUE;
  1725. }
  1726. return FALSE;
  1727. }
  1728. void CSuperGridCtrl::OnLButtonDown(UINT nFlags, CPoint point) 
  1729. {
  1730. if( GetFocus() != this) 
  1731. SetFocus();
  1732. LVHITTESTINFO ht;
  1733. ht.pt = point;
  1734. SubItemHitTest(&ht);
  1735. if(OnItemLButtonDown(ht))
  1736. {
  1737. BOOL bSelect=1;
  1738. bSelect = HitTestOnSign(point, ht);
  1739. if(bSelect && ht.iItem!=-1)
  1740. {
  1741. m_CurSubItem = IndexToOrder(ht.iSubItem);
  1742. CHeaderCtrl* pHeader = GetHeaderCtrl();
  1743. // Make the column fully visible.
  1744. MakeColumnVisible(Header_OrderToIndex(pHeader->m_hWnd, m_CurSubItem));
  1745. CListCtrl::OnLButtonDown(nFlags, point);
  1746. OnControlLButtonDown(nFlags, point, ht);
  1747. //update row anyway for selection bar
  1748. CRect rc;
  1749. GetItemRect(ht.iItem, rc, LVIR_BOUNDS);
  1750. InvalidateRect(rc);
  1751. UpdateWindow();
  1752. }
  1753. }
  1754. }
  1755. void CSuperGridCtrl::OnUpdateListViewItem(CTreeItem* lpItem, LV_ITEM *plvItem)
  1756. {
  1757. //default implementation you would go for this 9 out of 10 times
  1758. CItemInfo *lp = GetData(lpItem);
  1759. CString str = (CString)plvItem->pszText;
  1760. if(lp!=NULL)
  1761. {
  1762. if(plvItem->iSubItem==0)
  1763. lp->SetItemText(str);
  1764. else //subitem data 
  1765. lp->SetSubItemText(plvItem->iSubItem-1, str);
  1766.    UpdateData(lpItem, lp);
  1767. }
  1768. SetItemText(plvItem->iItem, plvItem->iSubItem, plvItem->pszText);
  1769. }
  1770. void CSuperGridCtrl::DeleteAll()
  1771. {
  1772. POSITION pos = m_RootItems.GetHeadPosition();
  1773. while(pos!=NULL)
  1774. {
  1775. CTreeItem * pRoot = (CTreeItem*)m_RootItems.GetNext(pos);
  1776. if(pRoot!=NULL)
  1777. DeleteItemEx(pRoot, 0);
  1778. }
  1779. }
  1780. POSITION CSuperGridCtrl::GetRootHeadPosition() const
  1781. {
  1782. return m_RootItems.GetHeadPosition();
  1783. }
  1784. POSITION CSuperGridCtrl::GetRootTailPosition() const
  1785. {
  1786. return m_RootItems.GetTailPosition();
  1787. }
  1788. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetNextRoot(POSITION& pos) const
  1789. {
  1790. return (CTreeItem*)m_RootItems.GetNext(pos);
  1791. }
  1792. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetPrevRoot(POSITION& pos) const
  1793. {
  1794. return (CTreeItem*)m_RootItems.GetNext(pos);
  1795. }
  1796. POSITION CSuperGridCtrl::GetHeadPosition(CTreeItem* pItem) const
  1797. {
  1798. return pItem->m_listChild.GetHeadPosition();
  1799. }
  1800. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetNextChild(CTreeItem *pItem, POSITION& pos) const
  1801. {
  1802. return (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1803. }
  1804. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetPrevChild(CTreeItem *pItem, POSITION& pos) const
  1805. {
  1806. return (CTreeItem*)pItem->m_listChild.GetPrev(pos);
  1807. }
  1808. POSITION CSuperGridCtrl::GetTailPosition(CTreeItem *pItem) const
  1809. {
  1810. return pItem->m_listChild.GetTailPosition();
  1811. }
  1812. void CSuperGridCtrl::AddTail(CTreeItem *pParent, CTreeItem *pChild)
  1813. {
  1814. pParent->m_listChild.AddTail(pChild);
  1815. }
  1816. inline int StrComp(const CString* pElement1, const CString* pElement2)
  1817. {
  1818. return pElement1->Compare(*pElement2);
  1819. }
  1820. int CSuperGridCtrl::CompareChildren(const void* p1, const void* p2)
  1821. {
  1822. CTreeItem * pChild1 = *(CTreeItem**)p1;
  1823. CTreeItem * pChild2 = *((CTreeItem**)p2);
  1824. CItemInfo *pItem1=(*pChild1).m_lpNodeInfo;
  1825. CItemInfo *pItem2=(*pChild2).m_lpNodeInfo;
  1826. return StrComp(&(pItem1->GetItemText()), &(pItem2->GetItemText()));
  1827. }
  1828. void CSuperGridCtrl::Sort(CTreeItem* pParent, BOOL bSortChildren)
  1829. {
  1830. const int nChildren = NumChildren(pParent);
  1831. if (nChildren > 1)
  1832. {
  1833. CTreeItem** ppSortArray = new CTreeItem*[nChildren];
  1834. // Fill in array with pointers to our children.
  1835. POSITION pos = pParent->m_listChild.GetHeadPosition();
  1836. for (int i=0; pos; i++)
  1837. {
  1838. ASSERT(i < nChildren);
  1839. ppSortArray[i] = (CTreeItem*)pParent->m_listChild.GetAt(pos);
  1840. pParent->m_listChild.GetNext(pos);
  1841. }
  1842. qsort(ppSortArray, nChildren, sizeof(CTreeItem*), CompareChildren);
  1843. // reorg children with new sorted list
  1844. pos = pParent->m_listChild.GetHeadPosition();
  1845. for (i=0; pos; i++)
  1846. {
  1847. ASSERT(i < nChildren);
  1848. pParent->m_listChild.SetAt(pos, ppSortArray[i]);
  1849. pParent->m_listChild.GetNext(pos);
  1850. }
  1851. delete [] ppSortArray;
  1852. }
  1853. if(bSortChildren)
  1854. {
  1855. POSITION pos = pParent->m_listChild.GetHeadPosition();
  1856. while (pos)
  1857. {
  1858. CTreeItem *pChild = (CTreeItem*)pParent->m_listChild.GetNext(pos);
  1859. Sort(pChild, TRUE);
  1860. }
  1861. }
  1862. }
  1863. int CSuperGridCtrl::NumChildren(const CTreeItem *pItem) const
  1864. {
  1865. return pItem->m_listChild.GetCount();
  1866. }
  1867. BOOL CSuperGridCtrl::ItemHasChildren(const CTreeItem* pItem) const
  1868. BOOL bChildren = pItem->m_listChild.GetCount() != 0;
  1869. //see if we have a flag
  1870. int nFlag = pItem->m_bSetChildFlag;
  1871. if(nFlag!=-1)
  1872. return 1;
  1873. else
  1874. return bChildren;
  1875. }
  1876. void CSuperGridCtrl::SetChildrenFlag(CTreeItem *pItem, int nFlag) const
  1877. {
  1878. pItem->m_bSetChildFlag = nFlag;
  1879. }
  1880. BOOL CSuperGridCtrl::IsCollapsed(const CTreeItem* pItem) const
  1881. {
  1882. return pItem->m_bHideChildren;//e.g not visible
  1883. }
  1884. void CSuperGridCtrl::Hide(CTreeItem* pItem, BOOL bFlag)
  1885. {
  1886. pItem->m_bHideChildren=bFlag;
  1887. }
  1888. int CSuperGridCtrl::GetIndent(const CTreeItem* pItem) const
  1889. {
  1890. return pItem->m_nIndent;
  1891. }
  1892. void CSuperGridCtrl::SetIndent(CTreeItem *pItem, int iIndent)
  1893. {
  1894. pItem->m_nIndent = iIndent;
  1895. }
  1896. int CSuperGridCtrl::GetCurIndex(const CTreeItem *pItem) const
  1897. {
  1898. return pItem->m_nIndex;
  1899. }
  1900. void CSuperGridCtrl::SetCurIndex(CTreeItem* pItem, int nIndex) 
  1901. {
  1902. pItem->m_nIndex = nIndex;
  1903. }
  1904. void CSuperGridCtrl::SetParentItem(CTreeItem*pItem, CTreeItem* pParent)
  1905. {
  1906. pItem->m_pParent=pParent;
  1907. }
  1908. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetParentItem(const CTreeItem* pItem) 
  1909. {
  1910. return pItem->m_pParent;
  1911. };
  1912. CItemInfo* CSuperGridCtrl::GetData(const CTreeItem* pItem) 
  1913. {
  1914. return pItem->m_lpNodeInfo;
  1915. }
  1916. void CSuperGridCtrl::UpdateData(CTreeItem* pItem, CItemInfo* lpInfo)
  1917. {
  1918. pItem->m_lpNodeInfo = lpInfo;
  1919. }
  1920. //overrides
  1921. CItemInfo* CSuperGridCtrl::CopyData(CItemInfo* lpSrc)
  1922. {
  1923. ASSERT(FALSE);//debug
  1924. return NULL;  //release
  1925. }
  1926. //default implementation for setting icons
  1927. int CSuperGridCtrl::GetIcon(const CTreeItem* pItem)
  1928. {
  1929. return 0;//just take the first item in CImageList ..what ever that is
  1930. }
  1931. void CSuperGridCtrl::OnControlLButtonDown(UINT nFlags, CPoint point, LVHITTESTINFO& ht)
  1932. {
  1933. if(ht.iSubItem==0)
  1934. {
  1935. CRect rcItem;
  1936. GetItemRect(ht.iItem, rcItem, LVIR_LABEL);
  1937. DWORD dwStyle = WS_BORDER | WS_CHILD | WS_VISIBLE | ES_AUTOHSCROLL|ES_LEFT;
  1938. CEdit *pEdit = new CListEditCtrl(ht.iItem, ht.iSubItem, GetItemText(ht.iItem, ht.iSubItem));
  1939. pEdit->Create(dwStyle, rcItem, this, 0x1233);
  1940. }
  1941. else
  1942. EditLabelEx(ht.iItem, ht.iSubItem);
  1943. }
  1944. COLORREF CSuperGridCtrl::GetCellRGB()
  1945. {
  1946. return RGB(192,0,0);
  1947. }
  1948. BOOL CSuperGridCtrl::OnVKMultiply(CTreeItem *pItem, int nIndex)
  1949. {
  1950. return 1;
  1951. }
  1952. BOOL CSuperGridCtrl::OnVkSubTract(CTreeItem *pItem, int nIndex)
  1953. {
  1954. return 1;
  1955. }
  1956. BOOL CSuperGridCtrl::OnVKAdd(CTreeItem *pItem, int nIndex)
  1957. {
  1958. return 1;
  1959. }
  1960. BOOL CSuperGridCtrl::OnDeleteItem(CTreeItem* pItem, int nIndex)
  1961. {
  1962. return 1;
  1963. }
  1964. BOOL CSuperGridCtrl::OnItemExpanding(CTreeItem *pItem, int iItem)
  1965. {
  1966. return 1;
  1967. }
  1968. BOOL CSuperGridCtrl::OnItemExpanded(CTreeItem* pItem, int iItem)
  1969. {
  1970. return 1;
  1971. }
  1972. BOOL CSuperGridCtrl::OnCollapsing(CTreeItem *pItem)
  1973. {
  1974. return 1;
  1975. }
  1976. BOOL CSuperGridCtrl::OnItemCollapsed(CTreeItem *pItem)
  1977. {
  1978. return 1;
  1979. }
  1980. BOOL CSuperGridCtrl::OnItemLButtonDown(LVHITTESTINFO& ht)
  1981. {
  1982. return 1;
  1983. }
  1984. BOOL CSuperGridCtrl::OnVkReturn(void)
  1985. {
  1986. return 0;
  1987. }
  1988. void CSuperGridCtrl::OnSysColorChange() 
  1989. {
  1990. CListCtrl::OnSysColorChange();
  1991. //nop nothing yet
  1992. }
  1993. UINT CSuperGridCtrl::_GetCount(CTreeItem* pItem, UINT& nCount)
  1994. {
  1995. POSITION pos = pItem->m_listChild.GetHeadPosition();
  1996. while (pos)
  1997. {
  1998. CTreeItem *pChild = (CTreeItem*)pItem->m_listChild.GetNext(pos);
  1999. nCount = _GetCount(pChild, nCount);
  2000. nCount++;
  2001. }
  2002. return nCount;
  2003. }
  2004. UINT CSuperGridCtrl::GetCount(void) 
  2005. {
  2006. UINT nCount=0;
  2007. UINT _nCount=0;
  2008. POSITION pos = m_RootItems.GetHeadPosition();
  2009. while(pos!=NULL)
  2010. {
  2011. CTreeItem * pRoot = (CTreeItem*)m_RootItems.GetNext(pos);
  2012. nCount += _GetCount(pRoot, _nCount) + 1;
  2013. }
  2014. return nCount;
  2015. }
  2016. CSuperGridCtrl::CTreeItem* CSuperGridCtrl::GetTreeItem(int nIndex /*nIndex must be valid of course*/ ) 
  2017. {
  2018. return reinterpret_cast<CTreeItem*>(GetItemData(nIndex));
  2019. }
  2020. int CSuperGridCtrl::GetSelectedItem(void) const
  2021. {
  2022. return GetNextItem(-1, LVNI_ALL | LVNI_SELECTED); 
  2023. }
  2024. //////////////////////////////////////////////////////////////////////////
  2025. //
  2026. // not much but ... 
  2027. //
  2028. //////////////////////////////////////////////////////////////////////////
  2029. CSuperGridCtrl::CTreeItem::~CTreeItem()
  2030. {
  2031. // delete child nodes
  2032. POSITION pos = m_listChild.GetHeadPosition();
  2033. while (pos != NULL)
  2034. {
  2035. CTreeItem* pItem = (CTreeItem*)m_listChild.GetNext(pos);
  2036. if(pItem!=NULL)
  2037. {
  2038. if(pItem->m_lpNodeInfo!=NULL)
  2039. delete pItem->m_lpNodeInfo;
  2040. delete pItem;
  2041. }
  2042. }
  2043. m_listChild.RemoveAll();
  2044. }
  2045. //////////////////////////////////////////////////////////////////////////
  2046. //
  2047. // Simple class CRectangle for the + - sign, 
  2048. //
  2049. //////////////////////////////////////////////////////////////////////////
  2050. CRectangle::CRectangle(CSuperGridCtrl* pCtrl, CDC* pDC, int iIndent, const CRect& rcBounds)
  2051. {
  2052. m_pDC=pDC;
  2053. int nHalfImage = (pCtrl->m_cxImage >> 1);
  2054. int nBottomDown = rcBounds.top + nHalfImage + ((rcBounds.Height() - pCtrl->m_cyImage) >> 1);
  2055. m_right_bottom.cx = (pCtrl->m_cxImage>>1)+2+1;
  2056. m_right_bottom.cy = (pCtrl->m_cyImage>>1)+2+1;
  2057. m_left = rcBounds.left  + iIndent * pCtrl->m_cxImage - nHalfImage;
  2058. m_top = nBottomDown - (m_right_bottom.cy >> 1);
  2059. m_left_top.x = m_left -  (m_right_bottom.cx >> 1);
  2060. m_left_top.y = m_top;
  2061. m_topdown = nBottomDown;
  2062. }
  2063. void CRectangle::DrawRectangle(CSuperGridCtrl* pCtrl)
  2064. {
  2065. //erase bkgrnd
  2066. CRect rc(m_left_top, m_right_bottom);
  2067. m_pDC->FillRect(rc, &pCtrl->m_brushErase);
  2068. //draw rectangle
  2069. CPen* pPenRectangle = m_pDC->SelectObject(&pCtrl->m_psRectangle);
  2070. m_pDC->Rectangle(rc);
  2071. m_pDC->SelectObject(pPenRectangle);
  2072. }
  2073. CRectangle::~CRectangle()
  2074. {
  2075. }
  2076. BOOL CRectangle::HitTest(CPoint pt)
  2077. {
  2078. CRect rc = GetHitTestRect();
  2079. return rc.PtInRect(pt);
  2080. }
  2081. void CRectangle::DrawPlus(void)
  2082. {
  2083. m_pDC->MoveTo(m_left, m_topdown-2);
  2084. m_pDC->LineTo(m_left, m_topdown+3);
  2085. m_pDC->MoveTo(m_left-2, m_topdown);
  2086. m_pDC->LineTo(m_left+3, m_topdown);
  2087. }
  2088. void CRectangle::DrawMinus(void)
  2089. {
  2090. m_pDC->MoveTo(m_left-2, m_topdown);
  2091. m_pDC->LineTo(m_left+3, m_topdown);
  2092. }