Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

796 lines
17 KiB

  1. //+-------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 1996 - 1999.
  5. //
  6. // File: lview.cxx
  7. //
  8. // Contents:
  9. //
  10. // History: 15 Aug 1996 DLee Created
  11. //
  12. //--------------------------------------------------------------------------
  13. #include "pch.cxx"
  14. #pragma hdrstop
  15. //
  16. // Window procedure for ListView
  17. //
  18. LRESULT WINAPI ListViewWndProc(
  19. HWND hwnd,
  20. UINT msg,
  21. WPARAM wParam,
  22. LPARAM lParam)
  23. {
  24. CListView *pControl = (CListView *) GetWindowLongPtr(hwnd, 0);
  25. LRESULT lRet = 0;
  26. switch (msg)
  27. {
  28. case WM_CREATE :
  29. pControl = new CListView;
  30. pControl->Create (GetParent(hwnd), hwnd);
  31. SetWindowLongPtr (hwnd, 0, (LONG_PTR) pControl);
  32. break;
  33. case WM_DESTROY :
  34. delete pControl;
  35. lRet = DefWindowProc(hwnd, msg, wParam, lParam);
  36. break;
  37. case WM_SETFONT:
  38. pControl->SetFont ((HFONT)wParam);
  39. break;
  40. case WM_SETFOCUS:
  41. pControl->SetFocus();
  42. lRet = DefWindowProc(hwnd, msg, wParam, lParam);
  43. break;
  44. case wmInsertItem:
  45. pControl->InsertItem ((int)lParam);
  46. break;
  47. case wmDeleteItem:
  48. pControl->DeleteItem ((int)lParam);
  49. break;
  50. case wmUpdateItem:
  51. pControl->InvalidateItem ((int)lParam);
  52. break;
  53. case wmSetCountBefore:
  54. pControl->SetCountBefore ((int)lParam);
  55. break;
  56. case wmSetCount:
  57. pControl->SetTotalCount ((int)lParam);
  58. break;
  59. case wmResetContents:
  60. pControl->ResetContents();
  61. break;
  62. case WM_SIZE:
  63. pControl->Size (wParam, LOWORD(lParam), HIWORD(lParam));
  64. break;
  65. case WM_PAINT:
  66. {
  67. PAINTSTRUCT paint;
  68. BeginPaint ( hwnd, &paint );
  69. pControl->Paint (paint);
  70. EndPaint(hwnd, &paint );
  71. }
  72. break;
  73. case WM_LBUTTONUP:
  74. pControl->ButtonUp(HIWORD(lParam));
  75. break;
  76. case WM_LBUTTONDOWN:
  77. pControl->ButtonDown(HIWORD(lParam));
  78. break;
  79. case WM_LBUTTONDBLCLK:
  80. SendMessage (pControl->Parent(),
  81. WM_COMMAND,
  82. MAKEWPARAM(idListChild, LBN_DBLCLK),
  83. (LPARAM) hwnd);
  84. break;
  85. case WM_KEYDOWN:
  86. pControl->KeyDown ((int)wParam);
  87. break;
  88. case WM_VSCROLL:
  89. pControl->Vscroll ((int)LOWORD(wParam), (int)HIWORD(wParam));
  90. break;
  91. case WM_MOUSEWHEEL :
  92. lRet = pControl->MouseWheel( hwnd, wParam, lParam );
  93. break;
  94. case wmContextMenuHitTest:
  95. lRet = pControl->ContextMenuHitTest( wParam, lParam );
  96. break;
  97. default :
  98. lRet = DefWindowProc(hwnd, msg, wParam, lParam);
  99. break;
  100. }
  101. return lRet;
  102. } //ListViewWndProc
  103. CListView::CListView ()
  104. : _hwndParent(0),
  105. _hwnd(0),
  106. _cBefore(0),
  107. _cTotal (0),
  108. _cx(0),
  109. _cy(0),
  110. _cyLine(1),
  111. _cLines(0),
  112. _hfont(0),
  113. _iWheelRemainder(0)
  114. {}
  115. LRESULT CListView::MouseWheel(
  116. HWND hwnd,
  117. WPARAM wParam,
  118. LPARAM lParam )
  119. {
  120. // forward what we don't process
  121. if ( wParam & ( MK_SHIFT | MK_CONTROL ) )
  122. return DefWindowProc( hwnd, WM_MOUSEWHEEL, wParam, lParam );
  123. // add the current scroll to the remainder from last time
  124. int iDelta = (int) (short) HIWORD( wParam );
  125. iDelta += _iWheelRemainder;
  126. // if there isn't enough to process this time, just return
  127. if ( abs( iDelta ) < WHEEL_DELTA )
  128. {
  129. _iWheelRemainder = iDelta;
  130. return 0;
  131. }
  132. // compute the remainder and amount to scroll
  133. _iWheelRemainder = ( iDelta % WHEEL_DELTA );
  134. iDelta /= WHEEL_DELTA;
  135. BOOL fDown;
  136. if ( iDelta < 0 )
  137. {
  138. fDown = TRUE;
  139. iDelta = -iDelta;
  140. }
  141. else
  142. fDown = FALSE;
  143. // get the # of lines to scroll per WHEEL_DELTA
  144. int cLines;
  145. SystemParametersInfo( SPI_GETWHEELSCROLLLINES, 0, &cLines, 0 );
  146. if ( 0 == cLines )
  147. return 0;
  148. int cVisibleLines = _cLines;
  149. // if scrolling a page, do so. don't scroll more than one page
  150. if ( WHEEL_PAGESCROLL == cLines )
  151. iDelta = __max( 1, (cVisibleLines - 1) );
  152. else
  153. {
  154. iDelta *= cLines;
  155. if ( iDelta >= cVisibleLines )
  156. iDelta = __max( 1, (cVisibleLines - 1) );
  157. }
  158. // phew. do the scroll
  159. if ( 0 != iDelta )
  160. {
  161. if ( fDown )
  162. _GoDown( iDelta );
  163. else
  164. _GoUp( iDelta );
  165. }
  166. return iDelta;
  167. } //MouseWheel
  168. LRESULT CListView::ContextMenuHitTest(
  169. WPARAM wParam,
  170. LPARAM lParam )
  171. {
  172. POINT pt;
  173. // cast required to sign extend [multimon bug]
  174. pt.x = (LONG)(short)LOWORD( lParam );
  175. pt.y = (LONG)(short)HIWORD( lParam );
  176. RECT rc;
  177. GetWindowRect( _hwnd, &rc );
  178. // did they click in the window?
  179. if ( !PtInRect( &rc, pt ) )
  180. return -1;
  181. // convert y to window view coordinates
  182. int vy = pt.y - rc.top;
  183. // did they click on a line in the window?
  184. int line = vy / _cyLine;
  185. int newLine = line;
  186. if ( line >= _cLines || line >= _cTotal )
  187. return -1;
  188. // make this line the current selection
  189. ButtonDown( vy );
  190. return line;
  191. } //ContextMenuHitTest
  192. //
  193. // Create
  194. //
  195. void CListView::Create (HWND hwndParent, HWND hwnd)
  196. {
  197. _hwndParent = hwndParent;
  198. _hwnd = hwnd;
  199. MEASUREITEMSTRUCT measure;
  200. measure.CtlType = odtListView;
  201. //
  202. // Owner: Measure item!
  203. //
  204. SendMessage (_hwndParent, wmMeasureItem, 0, (LPARAM) &measure);
  205. _cyLine = measure.itemHeight;
  206. } //Create
  207. //
  208. // Key Down
  209. //
  210. void CListView::KeyDown (int nKey)
  211. {
  212. switch (nKey)
  213. {
  214. case ' ' :
  215. ButtonDown( 0 );
  216. break;
  217. case 11:
  218. case 13:
  219. // treat ENTER as a double-click
  220. //
  221. // Owner: Double click!
  222. //
  223. SendMessage (_hwndParent, WM_COMMAND, MAKEWPARAM(idListChild, LBN_DBLCLK), (LPARAM) _hwnd);
  224. break;
  225. //
  226. // Translate keystrokes into scrolling actions
  227. //
  228. case VK_HOME:
  229. SendMessage (_hwnd, WM_VSCROLL, SB_TOP, 0L);
  230. break;
  231. case VK_END:
  232. SendMessage (_hwnd, WM_VSCROLL, SB_BOTTOM, 0L);
  233. break;
  234. case VK_PRIOR:
  235. SendMessage (_hwnd, WM_VSCROLL, SB_PAGEUP, 0L);
  236. break;
  237. case VK_NEXT:
  238. SendMessage (_hwnd, WM_VSCROLL, SB_PAGEDOWN, 0L);
  239. break;
  240. case VK_UP:
  241. SelectUp ();
  242. break;
  243. case VK_DOWN:
  244. SelectDown ();
  245. break;
  246. }
  247. } //KeyDown
  248. void CListView::UpdateHighlight(
  249. int oldLine,
  250. int newLine )
  251. {
  252. // unhighlight
  253. if ( -1 != oldLine )
  254. RefreshRow( oldLine );
  255. // highlight
  256. if ( oldLine != newLine )
  257. RefreshRow( newLine );
  258. UpdateWindow (_hwnd);
  259. } //UpdateHighlight
  260. void CListView::SelectUp ()
  261. {
  262. int newLine;
  263. if ( SendMessage( _hwndParent, wmListNotify, listSelectUp, (LPARAM)&newLine ))
  264. UpdateHighlight( newLine + 1, newLine );
  265. } //SelectUp
  266. void CListView::SelectDown ()
  267. {
  268. int newLine;
  269. if ( SendMessage( _hwndParent, wmListNotify, listSelectDown, (LPARAM)&newLine ))
  270. UpdateHighlight( newLine - 1, newLine );
  271. } //SelectDown
  272. //
  273. // Button up (select)
  274. //
  275. void CListView::ButtonUp (int y)
  276. {
  277. }
  278. void CListView::ButtonDown (int y)
  279. {
  280. int line = y / _cyLine;
  281. int newLine = line;
  282. if (line >= _cLines)
  283. return;
  284. //
  285. // Owner: Selection made!
  286. //
  287. if (SendMessage (_hwndParent, wmListNotify, listSelect, (LPARAM)&line ))
  288. UpdateHighlight( line, newLine );
  289. ::SetFocus (_hwnd);
  290. } //ButtonDown
  291. void CListView::SetFocus()
  292. {
  293. //
  294. // Owner: Focus!
  295. //
  296. SendMessage (_hwndParent, WM_COMMAND, MAKEWPARAM(idListChild, LBN_SETFOCUS), (LPARAM) _hwnd);
  297. } //SetFocus
  298. //
  299. // Size
  300. //
  301. void CListView::Size (WPARAM flags, int cx, int cy)
  302. {
  303. int cxOld = _cx;
  304. int cyOld = _cy;
  305. _cx = cx;
  306. _cy = cy;
  307. BOOL fInvalidate = FALSE;
  308. if (cy != cyOld)
  309. {
  310. _cLines = cy / _cyLine;
  311. //
  312. // Owner: Size!
  313. //
  314. long cRows = _cLines;
  315. fInvalidate = (BOOL)SendMessage(_hwndParent, wmListNotify, listSize, (LPARAM) &cRows);
  316. }
  317. // Don't repaint the common area
  318. RECT rect;
  319. rect.top = 0;
  320. rect.left = 0;
  321. rect.bottom = min (cy, cyOld);
  322. rect.right = min (cx, cxOld);
  323. // no need -- user does this for free, and it causes repaint bugs
  324. // ValidateRect (_hwnd, &rect );
  325. if (cy != cyOld)
  326. {
  327. if ( fInvalidate )
  328. InvalidateAndUpdateScroll();
  329. else
  330. UpdateScroll();
  331. }
  332. } //Size
  333. //
  334. // Paint
  335. //
  336. void CListView::Paint (PAINTSTRUCT& paint)
  337. {
  338. RECT& rect = paint.rcPaint;
  339. int lineStart = rect.top / _cyLine;
  340. int lineEnd = (rect.bottom + _cyLine - 1) / _cyLine;
  341. DRAWITEMSTRUCT draw;
  342. draw.hwndItem = _hwnd;
  343. draw.itemAction = ODA_DRAWENTIRE;
  344. HDC hdc = paint.hdc;
  345. draw.hDC = hdc;
  346. HFONT hfontOld = (HFONT) SelectObject (hdc, _hfont);
  347. for (int i = lineStart; i < lineEnd; i++)
  348. {
  349. draw.itemState = 0;
  350. if ( GetFocus() == _hwnd )
  351. draw.itemState |= ODS_FOCUS;
  352. draw.itemID = i;
  353. draw.rcItem.top = 0;
  354. draw.rcItem.left = 0;
  355. draw.rcItem.bottom = _cyLine;
  356. draw.rcItem.right = _cx;
  357. SetViewportOrgEx( hdc, 0, i * _cyLine, 0 );
  358. //
  359. // Owner: Draw item!
  360. //
  361. SendMessage (_hwndParent, wmDrawItem, 0, (LPARAM)&draw);
  362. }
  363. SelectObject (hdc, hfontOld);
  364. } //Paint
  365. //
  366. // Set Font
  367. //
  368. void CListView::SetFont (HFONT hfontNew)
  369. {
  370. _hfont = hfontNew;
  371. MEASUREITEMSTRUCT measure;
  372. measure.CtlType = odtListView;
  373. //
  374. // Owner: Measure item
  375. //
  376. SendMessage (_hwndParent, wmMeasureItem, 0, (LPARAM) &measure);
  377. _cyLine = measure.itemHeight;
  378. long cRows = (_cy + _cyLine - 1) / _cyLine;
  379. _cLines = cRows;
  380. //
  381. // Owner: Size
  382. //
  383. SendMessage(_hwndParent, wmListNotify, listSize, (LPARAM) &cRows);
  384. InvalidateAndUpdateScroll();
  385. } //SetFont
  386. //
  387. // Scrolling
  388. //
  389. void CListView::Vscroll ( int action, int nPos)
  390. {
  391. switch (action)
  392. {
  393. case SB_LINEUP:
  394. LineUp ();
  395. break;
  396. case SB_LINEDOWN:
  397. LineDown ();
  398. break;
  399. case SB_THUMBTRACK:
  400. // don't refresh when thumb dragging
  401. // over many hits (too expensive)
  402. break;
  403. case SB_THUMBPOSITION:
  404. ScrollPos (nPos);
  405. break;
  406. case SB_PAGEDOWN:
  407. PageDown ();
  408. break;
  409. case SB_PAGEUP:
  410. PageUp ();
  411. break;
  412. case SB_TOP:
  413. Top ();
  414. break;
  415. case SB_BOTTOM:
  416. Bottom ();
  417. break;
  418. }
  419. } //VScroll
  420. void CListView::LineUp ()
  421. {
  422. long cLine = 1;
  423. //
  424. // Owner: Line up!
  425. //
  426. SendMessage(_hwndParent, wmListNotify, listScrollLineUp, (LPARAM) &cLine);
  427. if (cLine == 1)
  428. {
  429. if (_cBefore != 0)
  430. _cBefore--;
  431. // Force scroll and redraw
  432. RECT rect;
  433. GetClientRect (_hwnd, &rect);
  434. MyScrollWindow (_hwnd, 0, _cyLine, &rect, &rect);
  435. UpdateScroll();
  436. }
  437. } //LineUp
  438. void CListView::LineDown ()
  439. {
  440. long cLine = 1;
  441. //
  442. // Owner: Line down!
  443. //
  444. SendMessage(_hwndParent, wmListNotify, listScrollLineDn, (LPARAM) &cLine);
  445. if (cLine == 1)
  446. {
  447. RECT rect;
  448. GetClientRect (_hwnd, &rect);
  449. MyScrollWindow (_hwnd, 0, -_cyLine, &rect, &rect);
  450. _cBefore++;
  451. UpdateScroll();
  452. }
  453. } //LineDown
  454. void CListView::_GoUp(
  455. long cToGo )
  456. {
  457. CWaitCursor wait;
  458. long count = cToGo;
  459. count = __min( count, _cBefore );
  460. //
  461. // Owner: Page up!
  462. //
  463. SendMessage(_hwndParent, wmListNotify, listScrollPageUp, (LPARAM) &count);
  464. // _cBefore is approximate; don't give up if it is too big
  465. if ( 0 == count )
  466. {
  467. if ( _cBefore > 0 )
  468. count = _cBefore - 1;
  469. else
  470. count = 1; // worst case; scroll up one line
  471. SendMessage( _hwndParent,
  472. wmListNotify,
  473. listScrollPageUp,
  474. (LPARAM) &count );
  475. }
  476. // gee, we're having a bad hair day
  477. if ( 0 == count )
  478. {
  479. count = 1; // worst case; scroll up one line
  480. SendMessage( _hwndParent,
  481. wmListNotify,
  482. listScrollPageUp,
  483. (LPARAM) &count );
  484. }
  485. if ( 0 != count )
  486. {
  487. // count == number of lines open at the top
  488. _cBefore -= count;
  489. if (_cBefore < 0)
  490. _cBefore = 0;
  491. InvalidateAndUpdateScroll();
  492. }
  493. } //_GoUp
  494. void CListView::PageUp ()
  495. {
  496. _GoUp( _cLines - 1 );
  497. } //PageUp
  498. void CListView::_GoDown(
  499. long cToGo )
  500. {
  501. CWaitCursor wait;
  502. long count = cToGo;
  503. //
  504. // Owner: Page Down!
  505. //
  506. SendMessage(_hwndParent, wmListNotify, listScrollPageDn, (LPARAM) &count);
  507. // count == number of lines open at the bottom
  508. if ( 0 != count )
  509. {
  510. _cBefore += count;
  511. if (_cBefore >= ( _cTotal - _cLines ) )
  512. _cBefore = ( _cTotal - _cLines );
  513. InvalidateAndUpdateScroll();
  514. }
  515. } //_GoDown
  516. void CListView::PageDown ()
  517. {
  518. _GoDown( _cLines - 1 );
  519. } //PageDown
  520. void CListView::Top ()
  521. {
  522. long count = _cLines;
  523. //
  524. // Owner: Top!
  525. //
  526. SendMessage(_hwndParent, wmListNotify, listScrollTop, (LPARAM) &count );
  527. _cBefore = 0;
  528. InvalidateAndUpdateScroll();
  529. } //Top
  530. void CListView::Bottom ()
  531. {
  532. long count = _cLines;
  533. //
  534. // Owner: Bottom!
  535. //
  536. SendMessage(_hwndParent, wmListNotify, listScrollBottom, (LPARAM) &count);
  537. // count == number of lines visible
  538. _cBefore = _cTotal - count;
  539. if (_cBefore < 0)
  540. _cBefore = 0;
  541. InvalidateAndUpdateScroll();
  542. } //Bottom
  543. void CListView::ScrollPos (int pos)
  544. {
  545. long iRow = pos;
  546. //
  547. // Owner: Scroll Position!
  548. //
  549. SendMessage(_hwndParent, wmListNotify, listScrollPos, (LPARAM) &iRow);
  550. if (iRow != -1)
  551. {
  552. _cBefore = iRow;
  553. InvalidateAndUpdateScroll();
  554. }
  555. } //ScrollPos
  556. //
  557. // Message: Reset Contents
  558. //
  559. void CListView::ResetContents()
  560. {
  561. _cBefore = 0;
  562. _cTotal = 0;
  563. UpdateScroll();
  564. RECT rect;
  565. GetClientRect (_hwnd, &rect);
  566. InvalidateRect (_hwnd, &rect, TRUE );
  567. UpdateWindow (_hwnd);
  568. } //ResetContents
  569. void CListView::InvalidateAndUpdateScroll()
  570. {
  571. RECT rect;
  572. GetClientRect (_hwnd, &rect);
  573. InvalidateRect (_hwnd, &rect, TRUE );
  574. UpdateScroll();
  575. } //InvalidateAndUpdateScroll
  576. //
  577. // Message: Insert item after iRow
  578. //
  579. void CListView::InsertItem (int iRow)
  580. {
  581. Win4Assert (iRow < _cLines );
  582. RECT rect;
  583. GetClientRect (_hwnd, &rect);
  584. rect.top = (iRow + 1) * _cyLine;
  585. MyScrollWindow( _hwnd, 0, _cyLine, &rect, &rect, FALSE );
  586. _cTotal++;
  587. UpdateWindow (_hwnd);
  588. UpdateScroll();
  589. } //InsertItem
  590. //
  591. // Message: Delete item
  592. //
  593. void CListView::DeleteItem (int iRow)
  594. {
  595. Win4Assert (iRow < _cLines );
  596. RECT rect;
  597. GetClientRect (_hwnd, &rect);
  598. rect.top = (iRow + 1) * _cyLine;
  599. MyScrollWindow( _hwnd, 0, -_cyLine, &rect, &rect, FALSE );
  600. _cTotal--;
  601. if (_cTotal < 0)
  602. _cTotal = 0;
  603. // Invalidate the area which was
  604. // scrolled up (the last row before scrolling), if visible
  605. if ( _cTotal && _cTotal < _cLines )
  606. {
  607. RefreshRow( _cTotal );
  608. }
  609. UpdateScroll();
  610. } //DeleteItem
  611. //
  612. // Message: Invalidate item
  613. //
  614. void CListView::InvalidateItem (int iRow)
  615. {
  616. Win4Assert (iRow <= _cLines );
  617. RefreshRow (iRow);
  618. UpdateWindow (_hwnd);
  619. } //InvalidateItem
  620. //
  621. // Message: Set count before
  622. //
  623. void CListView::SetCountBefore (int cBefore)
  624. {
  625. _cBefore = cBefore;
  626. SetScrollPos (_hwnd, SB_VERT, _cBefore, TRUE);
  627. } //SetCountBefore
  628. //
  629. // Message: Set total count
  630. //
  631. void CListView::SetTotalCount (int cTotal)
  632. {
  633. _cTotal = cTotal;
  634. UpdateScroll ();
  635. } //SetTotalCount
  636. //
  637. // Internal methods
  638. //
  639. void CListView::RefreshRow (int iRow)
  640. {
  641. Win4Assert ( iRow < _cLines );
  642. RECT rect;
  643. rect.top = iRow * _cyLine;
  644. rect.left = 0;
  645. rect.bottom = rect.top + _cyLine;
  646. rect.right = _cx;
  647. InvalidateRect (_hwnd, &rect, TRUE );
  648. } //RefreshRow
  649. void CListView::UpdateScroll()
  650. {
  651. if (_cTotal - _cLines >= 0)
  652. {
  653. ShowScrollBar( _hwnd, SB_VERT, TRUE );
  654. SetScrollRange (_hwnd, SB_VERT, 0, _cTotal - 1, FALSE);
  655. SetScrollPos (_hwnd, SB_VERT, _cBefore, TRUE);
  656. // proportional scroll box
  657. SCROLLINFO si;
  658. si.cbSize = sizeof(si);
  659. si.fMask = SIF_PAGE;
  660. si.nPage = _cLines;
  661. SetScrollInfo( _hwnd, SB_VERT, &si, TRUE );
  662. EnableScrollBar (_hwnd, SB_VERT, ESB_ENABLE_BOTH );
  663. }
  664. else
  665. {
  666. _cBefore = 0;
  667. ShowScrollBar( _hwnd, SB_VERT, FALSE );
  668. EnableScrollBar (_hwnd, SB_VERT, ESB_DISABLE_BOTH );
  669. }
  670. } //UpdateScroll