ScrollBarManager.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499
  1. using LYFZ.OtherExpansion.SkinClass;
  2. using LYFZ.OtherExpansion.Win32;
  3. using LYFZ.OtherExpansion.Win32.Struct;
  4. using System;
  5. using System.Drawing;
  6. using System.Runtime.InteropServices;
  7. using System.Windows.Forms;
  8. namespace LYFZ.OtherExpansion.SkinControl
  9. {
  10. internal class ScrollBarManager : NativeWindow, IDisposable
  11. {
  12. private class ScrollBarMaskControl : MaskControlBase
  13. {
  14. private ScrollBarManager _owner;
  15. public ScrollBarMaskControl(ScrollBarManager owner) : base(owner.OwnerHWnd)
  16. {
  17. this._owner = owner;
  18. }
  19. protected override void OnPaint(IntPtr hWnd)
  20. {
  21. this._owner.DrawScrollBar(this._owner.OwnerHWnd, hWnd);
  22. }
  23. protected override void Dispose(bool disposing)
  24. {
  25. if (disposing)
  26. {
  27. this._owner = null;
  28. }
  29. base.Dispose(disposing);
  30. }
  31. }
  32. private bool _bPainting;
  33. private ScrollBar _owner;
  34. private ScrollBarManager.ScrollBarMaskControl _maskControl;
  35. private ScrollBarHistTest _lastMouseDownHistTest;
  36. private bool _disposed;
  37. private IntPtr OwnerHWnd
  38. {
  39. get
  40. {
  41. return this._owner.Handle;
  42. }
  43. }
  44. private Orientation Direction
  45. {
  46. get
  47. {
  48. if (this._owner is HScrollBar)
  49. {
  50. return Orientation.Horizontal;
  51. }
  52. return Orientation.Vertical;
  53. }
  54. }
  55. private int ArrowCx
  56. {
  57. get
  58. {
  59. return SystemInformation.HorizontalScrollBarArrowWidth;
  60. }
  61. }
  62. private int ArrowCy
  63. {
  64. get
  65. {
  66. return SystemInformation.VerticalScrollBarArrowHeight;
  67. }
  68. }
  69. internal ScrollBarManager(ScrollBar owner)
  70. {
  71. this._owner = owner;
  72. this.CreateHandle();
  73. }
  74. ~ScrollBarManager()
  75. {
  76. this.Dispose(false);
  77. }
  78. protected override void WndProc(ref Message m)
  79. {
  80. try
  81. {
  82. int msg = m.Msg;
  83. if (msg <= 125)
  84. {
  85. if (msg != 15)
  86. {
  87. if (msg == 71)
  88. {
  89. WINDOWPOS pos = (WINDOWPOS)Marshal.PtrToStructure(m.LParam, typeof(WINDOWPOS));
  90. bool hide = ((long)pos.flags & 128L) != 0L;
  91. bool show = ((long)pos.flags & 64L) != 0L;
  92. if (hide)
  93. {
  94. this._maskControl.SetVisibale(false);
  95. }
  96. else
  97. {
  98. if (show)
  99. {
  100. this._maskControl.SetVisibale(true);
  101. }
  102. }
  103. this._maskControl.CheckBounds(m.HWnd);
  104. base.WndProc(ref m);
  105. goto IL_22D;
  106. }
  107. if (msg == 125)
  108. {
  109. this.DrawScrollBar(m.HWnd, this._maskControl.Handle, false, true);
  110. base.WndProc(ref m);
  111. goto IL_22D;
  112. }
  113. }
  114. else
  115. {
  116. if (!this._bPainting)
  117. {
  118. PAINTSTRUCT ps = default(PAINTSTRUCT);
  119. this._bPainting = true;
  120. NativeMethods.BeginPaint(m.HWnd, ref ps);
  121. this.DrawScrollBar(m.HWnd, this._maskControl.Handle);
  122. NativeMethods.ValidateRect(m.HWnd, ref ps.rcPaint);
  123. NativeMethods.EndPaint(m.HWnd, ref ps);
  124. this._bPainting = false;
  125. m.Result = Result.TRUE;
  126. goto IL_22D;
  127. }
  128. base.WndProc(ref m);
  129. goto IL_22D;
  130. }
  131. }
  132. else
  133. {
  134. if (msg == 233)
  135. {
  136. this.DrawScrollBar(m.HWnd, this._maskControl.Handle, true, false);
  137. base.WndProc(ref m);
  138. goto IL_22D;
  139. }
  140. switch (msg)
  141. {
  142. case 512:
  143. case 514:
  144. this.DrawScrollBar(m.HWnd, this._maskControl.Handle);
  145. base.WndProc(ref m);
  146. goto IL_22D;
  147. case 513:
  148. this._lastMouseDownHistTest = this.ScrollBarHitTest(m.HWnd);
  149. this.DrawScrollBar(m.HWnd, this._maskControl.Handle);
  150. base.WndProc(ref m);
  151. goto IL_22D;
  152. default:
  153. if (msg == 675)
  154. {
  155. this.DrawScrollBar(m.HWnd, this._maskControl.Handle);
  156. base.WndProc(ref m);
  157. goto IL_22D;
  158. }
  159. break;
  160. }
  161. }
  162. base.WndProc(ref m);
  163. IL_22D:;
  164. }
  165. catch
  166. {
  167. }
  168. }
  169. private void DrawScrollBar(IntPtr scrollBarHWnd, IntPtr maskHWnd)
  170. {
  171. this.DrawScrollBar(scrollBarHWnd, maskHWnd, false, false);
  172. }
  173. private void DrawScrollBar(IntPtr scrollBarHWnd, IntPtr maskHWnd, bool sbm, bool styleChanged)
  174. {
  175. Orientation direction = this.Direction;
  176. bool bHorizontal = direction == Orientation.Horizontal;
  177. Rectangle bounds;
  178. Rectangle trackRect;
  179. Rectangle topLeftArrowRect;
  180. Rectangle bottomRightArrowRect;
  181. Rectangle thumbRect;
  182. this.CalculateRect(scrollBarHWnd, bHorizontal, out bounds, out trackRect, out topLeftArrowRect, out bottomRightArrowRect, out thumbRect);
  183. ScrollBarHistTest histTest;
  184. ControlState topLeftArrowState;
  185. ControlState bottomRightArrowState;
  186. ControlState thumbState;
  187. this.GetState(scrollBarHWnd, bHorizontal, out histTest, out topLeftArrowState, out bottomRightArrowState, out thumbState);
  188. if (sbm)
  189. {
  190. if (histTest == ScrollBarHistTest.None)
  191. {
  192. thumbState = ControlState.Pressed;
  193. }
  194. else
  195. {
  196. if (this._lastMouseDownHistTest == ScrollBarHistTest.Track)
  197. {
  198. thumbState = ControlState.Normal;
  199. }
  200. }
  201. }
  202. if (styleChanged)
  203. {
  204. thumbState = ControlState.Normal;
  205. }
  206. this.DrawScrollBar(maskHWnd, bounds, trackRect, topLeftArrowRect, bottomRightArrowRect, thumbRect, topLeftArrowState, bottomRightArrowState, thumbState, direction);
  207. }
  208. private void DrawScrollBar(ControlState topLeftArrowState, ControlState bottomRightArrowState, ControlState thumbState)
  209. {
  210. Orientation direction = this.Direction;
  211. bool bHorizontal = direction == Orientation.Horizontal;
  212. Rectangle bounds;
  213. Rectangle trackRect;
  214. Rectangle topLeftArrowRect;
  215. Rectangle bottomRightArrowRect;
  216. Rectangle thumbRect;
  217. this.CalculateRect(this.OwnerHWnd, bHorizontal, out bounds, out trackRect, out topLeftArrowRect, out bottomRightArrowRect, out thumbRect);
  218. this.DrawScrollBar(this._maskControl.Handle, bounds, trackRect, topLeftArrowRect, bottomRightArrowRect, thumbRect, topLeftArrowState, bottomRightArrowState, thumbState, direction);
  219. }
  220. private void DrawScrollBar(IntPtr maskHWnd, Rectangle bounds, Rectangle trackRect, Rectangle topLeftArrowRect, Rectangle bottomRightArrowRect, Rectangle thumbRect, ControlState topLeftArrowState, ControlState bottomRightArrowState, ControlState thumbState, Orientation direction)
  221. {
  222. bool bHorizontal = direction == Orientation.Horizontal;
  223. bool bEnabled = this._owner.Enabled;
  224. IScrollBarPaint paint = this._owner as IScrollBarPaint;
  225. if (paint == null)
  226. {
  227. return;
  228. }
  229. ImageDc tempDc = new ImageDc(bounds.Width, bounds.Height);
  230. IntPtr hdc = NativeMethods.GetDC(maskHWnd);
  231. try
  232. {
  233. using (Graphics g = Graphics.FromHdc(tempDc.Hdc))
  234. {
  235. using (PaintScrollBarTrackEventArgs te = new PaintScrollBarTrackEventArgs(g, trackRect, direction, bEnabled))
  236. {
  237. paint.OnPaintScrollBarTrack(te);
  238. }
  239. ArrowDirection arrowDirection = bHorizontal ? ArrowDirection.Left : ArrowDirection.Up;
  240. using (PaintScrollBarArrowEventArgs te2 = new PaintScrollBarArrowEventArgs(g, topLeftArrowRect, topLeftArrowState, arrowDirection, direction, bEnabled))
  241. {
  242. paint.OnPaintScrollBarArrow(te2);
  243. }
  244. arrowDirection = (bHorizontal ? ArrowDirection.Right : ArrowDirection.Down);
  245. using (PaintScrollBarArrowEventArgs te3 = new PaintScrollBarArrowEventArgs(g, bottomRightArrowRect, bottomRightArrowState, arrowDirection, direction, bEnabled))
  246. {
  247. paint.OnPaintScrollBarArrow(te3);
  248. }
  249. using (PaintScrollBarThumbEventArgs te4 = new PaintScrollBarThumbEventArgs(g, thumbRect, thumbState, direction, bEnabled))
  250. {
  251. paint.OnPaintScrollBarThumb(te4);
  252. }
  253. }
  254. NativeMethods.BitBlt(hdc, 0, 0, bounds.Width, bounds.Height, tempDc.Hdc, 0, 0, 13369376);
  255. }
  256. finally
  257. {
  258. NativeMethods.ReleaseDC(maskHWnd, hdc);
  259. tempDc.Dispose();
  260. }
  261. }
  262. private void CalculateRect(IntPtr scrollBarHWnd, bool bHorizontal, out Rectangle bounds, out Rectangle trackRect, out Rectangle topLeftArrowRect, out Rectangle bottomRightArrowRect, out Rectangle thumbRect)
  263. {
  264. RECT rect = default(RECT);
  265. NativeMethods.GetWindowRect(scrollBarHWnd, ref rect);
  266. NativeMethods.OffsetRect(ref rect, -rect.Left, -rect.Top);
  267. int arrowWidth = bHorizontal ? this.ArrowCx : this.ArrowCy;
  268. bounds = rect.Rect;
  269. Point point = this.GetScrollBarThumb(bounds, bHorizontal, arrowWidth);
  270. trackRect = bounds;
  271. if (!bHorizontal)
  272. {
  273. topLeftArrowRect = new Rectangle(0, 0, bounds.Width, arrowWidth);
  274. bottomRightArrowRect = new Rectangle(0, bounds.Height - arrowWidth, bounds.Width, arrowWidth);
  275. thumbRect = new Rectangle(0, point.X, bounds.Width, point.Y - point.X);
  276. return;
  277. }
  278. topLeftArrowRect = new Rectangle(0, 0, arrowWidth, bounds.Height);
  279. bottomRightArrowRect = new Rectangle(bounds.Width - arrowWidth, 0, arrowWidth, bounds.Height);
  280. if (this._owner.RightToLeft == RightToLeft.Yes)
  281. {
  282. thumbRect = new Rectangle(point.Y, 0, point.X - point.Y, bounds.Height);
  283. return;
  284. }
  285. thumbRect = new Rectangle(point.X, 0, point.Y - point.X, bounds.Height);
  286. }
  287. private void GetState(IntPtr scrollBarHWnd, bool bHorizontal, out ScrollBarHistTest histTest, out ControlState topLeftArrowState, out ControlState bottomRightArrowState, out ControlState thumbState)
  288. {
  289. histTest = this.ScrollBarHitTest(scrollBarHWnd);
  290. bool bLButtonDown = Win32Helper.LeftKeyPressed();
  291. bool bEnabled = this._owner.Enabled;
  292. topLeftArrowState = ControlState.Normal;
  293. bottomRightArrowState = ControlState.Normal;
  294. thumbState = ControlState.Normal;
  295. switch (histTest)
  296. {
  297. case ScrollBarHistTest.TopArrow:
  298. case ScrollBarHistTest.LeftArrow:
  299. if (bEnabled)
  300. {
  301. topLeftArrowState = (bLButtonDown ? ControlState.Pressed : ControlState.Hover);
  302. return;
  303. }
  304. break;
  305. case ScrollBarHistTest.BottomArrow:
  306. case ScrollBarHistTest.RightArrow:
  307. if (bEnabled)
  308. {
  309. bottomRightArrowState = (bLButtonDown ? ControlState.Pressed : ControlState.Hover);
  310. return;
  311. }
  312. break;
  313. case ScrollBarHistTest.Thumb:
  314. if (bEnabled)
  315. {
  316. thumbState = (bLButtonDown ? ControlState.Pressed : ControlState.Hover);
  317. }
  318. break;
  319. default:
  320. return;
  321. }
  322. }
  323. protected void CreateHandle()
  324. {
  325. base.AssignHandle(this.OwnerHWnd);
  326. this._maskControl = new ScrollBarManager.ScrollBarMaskControl(this);
  327. this._maskControl.OnCreateHandle();
  328. }
  329. internal void ReleaseHandleInternal()
  330. {
  331. if (base.Handle != IntPtr.Zero)
  332. {
  333. base.ReleaseHandle();
  334. }
  335. }
  336. private SCROLLBARINFO GetScrollBarInfo(IntPtr hWnd)
  337. {
  338. SCROLLBARINFO sbi = default(SCROLLBARINFO);
  339. sbi.cbSize = Marshal.SizeOf(sbi);
  340. NativeMethods.SendMessage(hWnd, 235, 0, ref sbi);
  341. return sbi;
  342. }
  343. private SCROLLBARINFO GetScrollBarInfo(IntPtr hWnd, uint objid)
  344. {
  345. SCROLLBARINFO sbi = default(SCROLLBARINFO);
  346. sbi.cbSize = Marshal.SizeOf(sbi);
  347. NativeMethods.GetScrollBarInfo(hWnd, objid, ref sbi);
  348. return sbi;
  349. }
  350. private Point GetScrollBarThumb()
  351. {
  352. bool bHorizontal = this.Direction == Orientation.Horizontal;
  353. int arrowWidth = bHorizontal ? this.ArrowCx : this.ArrowCy;
  354. return this.GetScrollBarThumb(this._owner.ClientRectangle, bHorizontal, arrowWidth);
  355. }
  356. private Point GetScrollBarThumb(Rectangle rect, bool bHorizontal, int arrowWidth)
  357. {
  358. ScrollBar scrollBar = this._owner;
  359. Point point = default(Point);
  360. int width;
  361. if (bHorizontal)
  362. {
  363. width = rect.Width - arrowWidth * 2;
  364. }
  365. else
  366. {
  367. width = rect.Height - arrowWidth * 2;
  368. }
  369. int value = scrollBar.Maximum - scrollBar.Minimum - scrollBar.LargeChange + 1;
  370. float thumbWidth = (float)width / ((float)value / (float)scrollBar.LargeChange + 1f);
  371. if (thumbWidth < 8f)
  372. {
  373. thumbWidth = 8f;
  374. }
  375. if (value != 0)
  376. {
  377. int curValue = scrollBar.Value - scrollBar.Minimum;
  378. if (curValue > value)
  379. {
  380. curValue = value;
  381. }
  382. point.X = (int)((float)curValue * (((float)width - thumbWidth) / (float)value));
  383. }
  384. point.X += arrowWidth;
  385. point.Y = point.X + (int)Math.Ceiling((double)thumbWidth);
  386. if (bHorizontal && scrollBar.RightToLeft == RightToLeft.Yes)
  387. {
  388. point.X = scrollBar.Width - point.X;
  389. point.Y = scrollBar.Width - point.Y;
  390. }
  391. return point;
  392. }
  393. private ScrollBarHistTest ScrollBarHitTest(IntPtr hWnd)
  394. {
  395. NativeMethods.Point point = default(NativeMethods.Point);
  396. RECT rect = default(RECT);
  397. Point thumbPoint = this.GetScrollBarThumb();
  398. int arrowCx = this.ArrowCx;
  399. int arrowCy = this.ArrowCy;
  400. NativeMethods.GetWindowRect(hWnd, ref rect);
  401. NativeMethods.OffsetRect(ref rect, -rect.Left, -rect.Top);
  402. RECT tp = rect;
  403. NativeMethods.GetCursorPos(ref point);
  404. NativeMethods.ScreenToClient(hWnd, ref point);
  405. if (this.Direction == Orientation.Horizontal)
  406. {
  407. if (NativeMethods.PtInRect(ref rect, point))
  408. {
  409. tp.Right = arrowCx;
  410. if (NativeMethods.PtInRect(ref tp, point))
  411. {
  412. return ScrollBarHistTest.LeftArrow;
  413. }
  414. tp.Left = rect.Right - arrowCx;
  415. tp.Right = rect.Right;
  416. if (NativeMethods.PtInRect(ref tp, point))
  417. {
  418. return ScrollBarHistTest.RightArrow;
  419. }
  420. if (this._owner.RightToLeft == RightToLeft.Yes)
  421. {
  422. tp.Left = point.y;
  423. tp.Right = point.x;
  424. }
  425. else
  426. {
  427. tp.Left = thumbPoint.X;
  428. tp.Right = thumbPoint.Y;
  429. }
  430. if (NativeMethods.PtInRect(ref tp, point))
  431. {
  432. return ScrollBarHistTest.Thumb;
  433. }
  434. return ScrollBarHistTest.Track;
  435. }
  436. }
  437. else
  438. {
  439. if (NativeMethods.PtInRect(ref rect, point))
  440. {
  441. tp.Bottom = arrowCy;
  442. if (NativeMethods.PtInRect(ref tp, point))
  443. {
  444. return ScrollBarHistTest.TopArrow;
  445. }
  446. tp.Top = rect.Bottom - arrowCy;
  447. tp.Bottom = rect.Bottom;
  448. if (NativeMethods.PtInRect(ref tp, point))
  449. {
  450. return ScrollBarHistTest.BottomArrow;
  451. }
  452. tp.Top = thumbPoint.X;
  453. tp.Bottom = thumbPoint.Y;
  454. if (NativeMethods.PtInRect(ref tp, point))
  455. {
  456. return ScrollBarHistTest.Thumb;
  457. }
  458. return ScrollBarHistTest.Track;
  459. }
  460. }
  461. return ScrollBarHistTest.None;
  462. }
  463. private void InvalidateWindow(bool messaged)
  464. {
  465. this.InvalidateWindow(this.OwnerHWnd, messaged);
  466. }
  467. private void InvalidateWindow(IntPtr hWnd, bool messaged)
  468. {
  469. if (messaged)
  470. {
  471. NativeMethods.RedrawWindow(hWnd, IntPtr.Zero, IntPtr.Zero, 2);
  472. return;
  473. }
  474. NativeMethods.RedrawWindow(hWnd, IntPtr.Zero, IntPtr.Zero, 257);
  475. }
  476. public void Dispose()
  477. {
  478. this.Dispose(true);
  479. GC.SuppressFinalize(this);
  480. }
  481. protected virtual void Dispose(bool disposing)
  482. {
  483. if (!this._disposed)
  484. {
  485. if (disposing)
  486. {
  487. if (this._maskControl != null)
  488. {
  489. this._maskControl.Dispose();
  490. this._maskControl = null;
  491. }
  492. this._owner = null;
  493. }
  494. this.ReleaseHandleInternal();
  495. }
  496. this._disposed = true;
  497. }
  498. }
  499. }