TreeView.cs 41 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571
  1. // Permission is hereby granted, free of charge, to any person obtaining
  2. // a copy of this software and associated documentation files (the
  3. // "Software"), to deal in the Software without restriction, including
  4. // without limitation the rights to use, copy, modify, merge, publish,
  5. // distribute, sublicense, and/or sell copies of the Software, and to
  6. // permit persons to whom the Software is furnished to do so, subject to
  7. // the following conditions:
  8. //
  9. // The above copyright notice and this permission notice shall be
  10. // included in all copies or substantial portions of the Software.
  11. //
  12. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  13. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  14. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  15. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  16. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  17. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  18. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  19. //
  20. // Copyright (c) 2004-2005 Novell, Inc.
  21. //
  22. // Authors:
  23. // Jackson Harper ([email protected])
  24. // Kazuki Oikawa ([email protected])
  25. using System;
  26. using System.Collections;
  27. using System.ComponentModel;
  28. using System.ComponentModel.Design;
  29. using System.Drawing;
  30. using System.Drawing.Drawing2D;
  31. using System.Runtime.InteropServices;
  32. namespace System.Windows.Forms {
  33. [DefaultProperty("Nodes")]
  34. [DefaultEvent("AfterSelect")]
  35. [Designer("System.Windows.Forms.Design.TreeViewDesigner, " + Consts.AssemblySystem_Design, "System.ComponentModel.Design.IDesigner")]
  36. public class TreeView : Control {
  37. #region Fields
  38. private string path_separator = "\\";
  39. private int item_height = -1;
  40. private bool sorted;
  41. internal TreeNode top_node;
  42. internal TreeNode root_node;
  43. internal bool nodes_added;
  44. private TreeNodeCollection nodes;
  45. private TreeNode selected_node = null;
  46. private TreeNode focused_node = null;
  47. private bool select_mmove = false;
  48. private ImageList image_list;
  49. private int image_index = -1;
  50. private int selected_image_index = -1;
  51. private bool full_row_select;
  52. private bool hot_tracking;
  53. private int indent = 19;
  54. private TextBox edit_text_box;
  55. private TreeNode edit_node;
  56. private bool checkboxes;
  57. private bool label_edit;
  58. private bool scrollable;
  59. private bool show_lines = true;
  60. private bool show_root_lines = true;
  61. private bool show_plus_minus = true;
  62. private bool hide_selection = true;
  63. private bool add_hscroll;
  64. private bool add_vscroll;
  65. private int max_node_width;
  66. private VScrollBar vbar;
  67. private int skipped_nodes;
  68. private HScrollBar hbar;
  69. private int hbar_offset;
  70. private int used_height;
  71. private bool update_node_bounds;
  72. private int update_stack;
  73. private bool update_needed;
  74. private TreeViewEventHandler on_after_check;
  75. private TreeViewEventHandler on_after_collapse;
  76. private TreeViewEventHandler on_after_expand;
  77. private NodeLabelEditEventHandler on_after_label_edit;
  78. private TreeViewEventHandler on_after_select;
  79. private TreeViewCancelEventHandler on_before_check;
  80. private TreeViewCancelEventHandler on_before_collapse;
  81. private TreeViewCancelEventHandler on_before_expand;
  82. private NodeLabelEditEventHandler on_before_label_edit;
  83. private TreeViewCancelEventHandler on_before_select;
  84. private Pen dash;
  85. private StringFormat string_format;
  86. private int open_node_count = -1;
  87. private int drag_begin_x = 0;
  88. private int drag_begin_y = 0;
  89. private long handle_count = 1;
  90. #endregion // Fields
  91. #region Public Constructors
  92. public TreeView ()
  93. {
  94. base.background_color = ThemeEngine.Current.ColorWindow;
  95. base.foreground_color = ThemeEngine.Current.ColorWindowText;
  96. root_node = new TreeNode (this);
  97. root_node.Text = "ROOT NODE";
  98. nodes = new TreeNodeCollection (root_node);
  99. root_node.SetNodes (nodes);
  100. MouseDown += new MouseEventHandler (MouseDownHandler);
  101. MouseUp += new MouseEventHandler(MouseUpHandler);
  102. MouseMove += new MouseEventHandler(MouseMoveHandler);
  103. SizeChanged += new EventHandler (SizeChangedHandler);
  104. FontChanged += new EventHandler (FontChangedHandler);
  105. LostFocus += new EventHandler (FocusChangedHandler);
  106. GotFocus += new EventHandler (FocusChangedHandler);
  107. MouseWheel += new MouseEventHandler(MouseWheelHandler);
  108. SetStyle (ControlStyles.UserPaint | ControlStyles.StandardClick, false);
  109. string_format = new StringFormat ();
  110. string_format.LineAlignment = StringAlignment.Center;
  111. string_format.Alignment = StringAlignment.Center;
  112. }
  113. #endregion // Public Constructors
  114. #region Public Instance Properties
  115. public override Color BackColor {
  116. get { return base.BackColor;}
  117. set { base.BackColor = value; }
  118. }
  119. [Browsable(false)]
  120. [EditorBrowsable(EditorBrowsableState.Never)]
  121. public override Image BackgroundImage {
  122. get { return base.BackgroundImage; }
  123. set { base.BackgroundImage = value; }
  124. }
  125. [DefaultValue(BorderStyle.Fixed3D)]
  126. [DispId(-504)]
  127. public BorderStyle BorderStyle {
  128. get { return InternalBorderStyle; }
  129. set { InternalBorderStyle = value; }
  130. }
  131. [DefaultValue(false)]
  132. public bool CheckBoxes {
  133. get { return checkboxes; }
  134. set {
  135. if (value == checkboxes)
  136. return;
  137. checkboxes = value;
  138. // Match a "bug" in the MS implementation where disabling checkboxes
  139. // collapses the entire tree, but enabling them does not affect the
  140. // state of the tree.
  141. if (!checkboxes)
  142. root_node.CollapseAllUncheck ();
  143. Refresh ();
  144. }
  145. }
  146. public override Color ForeColor {
  147. get { return base.ForeColor; }
  148. set { base.ForeColor = value; }
  149. }
  150. [DefaultValue(false)]
  151. public bool FullRowSelect {
  152. get { return full_row_select; }
  153. set {
  154. if (value == full_row_select)
  155. return;
  156. full_row_select = value;
  157. Refresh ();
  158. }
  159. }
  160. [DefaultValue(true)]
  161. public bool HideSelection {
  162. get { return hide_selection; }
  163. set {
  164. if (hide_selection == value)
  165. return;
  166. hide_selection = value;
  167. this.Refresh ();
  168. }
  169. }
  170. [DefaultValue(false)]
  171. public bool HotTracking {
  172. get { return hot_tracking; }
  173. set { hot_tracking = value; }
  174. }
  175. [DefaultValue(0)]
  176. [Editor("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof(System.Drawing.Design.UITypeEditor))]
  177. [Localizable(true)]
  178. [TypeConverter(typeof(TreeViewImageIndexConverter))]
  179. public int ImageIndex {
  180. get { return image_index; }
  181. set {
  182. if (value < -1) {
  183. throw new ArgumentException ("'" + value + "' is not a valid value for 'value'. " +
  184. "'value' must be greater than or equal to 0.");
  185. }
  186. if (image_index == value)
  187. return;
  188. image_index = value;
  189. Refresh ();
  190. }
  191. }
  192. [DefaultValue(null)]
  193. public ImageList ImageList {
  194. get { return image_list; }
  195. set {
  196. image_list = value;
  197. Refresh ();
  198. }
  199. }
  200. [Localizable(true)]
  201. public int Indent {
  202. get { return indent; }
  203. set {
  204. if (indent == value)
  205. return;
  206. if (value > 32000) {
  207. throw new ArgumentException ("'" + value + "' is not a valid value for 'Indent'. " +
  208. "'Indent' must be less than or equal to 32000");
  209. }
  210. if (value < 0) {
  211. throw new ArgumentException ("'" + value + "' is not a valid value for 'Indent'. " +
  212. "'Indent' must be greater than or equal to 0.");
  213. }
  214. indent = value;
  215. Refresh ();
  216. }
  217. }
  218. [Localizable(true)]
  219. public int ItemHeight {
  220. get {
  221. if (item_height == -1)
  222. return FontHeight + 3;
  223. return item_height;
  224. }
  225. set {
  226. if (value == item_height)
  227. return;
  228. item_height = value;
  229. Refresh ();
  230. }
  231. }
  232. [DefaultValue(false)]
  233. public bool LabelEdit {
  234. get { return label_edit; }
  235. set { label_edit = value; }
  236. }
  237. [DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
  238. [MergableProperty(false)]
  239. [Localizable(true)]
  240. public TreeNodeCollection Nodes {
  241. get { return nodes; }
  242. }
  243. [DefaultValue("\\")]
  244. public string PathSeparator {
  245. get { return path_separator; }
  246. set { path_separator = value; }
  247. }
  248. [DefaultValue(true)]
  249. public bool Scrollable {
  250. get { return scrollable; }
  251. set {
  252. if (scrollable == value)
  253. return;
  254. scrollable = value;
  255. }
  256. }
  257. [Editor("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof(System.Drawing.Design.UITypeEditor))]
  258. [TypeConverter(typeof(TreeViewImageIndexConverter))]
  259. [Localizable(true)]
  260. [DefaultValue(0)]
  261. public int SelectedImageIndex {
  262. get { return selected_image_index; }
  263. set {
  264. if (value < -1) {
  265. throw new ArgumentException ("'" + value + "' is not a valid value for 'value'. " +
  266. "'value' must be greater than or equal to 0.");
  267. }
  268. }
  269. }
  270. [Browsable(false)]
  271. [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  272. public TreeNode SelectedNode {
  273. get { return selected_node; }
  274. set {
  275. if (selected_node == value)
  276. return;
  277. TreeViewCancelEventArgs e = new TreeViewCancelEventArgs (value, false, TreeViewAction.Unknown);
  278. OnBeforeSelect (e);
  279. if (e.Cancel)
  280. return;
  281. Rectangle invalid = Rectangle.Empty;
  282. if (selected_node != null) {
  283. invalid = Bloat (selected_node.Bounds);
  284. }
  285. if (focused_node != null) {
  286. invalid = Rectangle.Union (invalid,
  287. Bloat (focused_node.Bounds));
  288. }
  289. if (value != null)
  290. invalid = Rectangle.Union (invalid, Bloat (value.Bounds));
  291. selected_node = value;
  292. focused_node = value;
  293. if (full_row_select) {
  294. invalid.X = 0;
  295. invalid.Width = ViewportRectangle.Width;
  296. }
  297. if (invalid != Rectangle.Empty)
  298. Invalidate (invalid);
  299. // We ensure its visible after we update because
  300. // scrolling is used for insure visible
  301. if (selected_node != null)
  302. selected_node.EnsureVisible ();
  303. OnAfterSelect (new TreeViewEventArgs (value, TreeViewAction.Unknown));
  304. }
  305. }
  306. private Rectangle Bloat (Rectangle rect)
  307. {
  308. rect.Y--;
  309. rect.X--;
  310. rect.Height += 2;
  311. rect.Width += 2;
  312. return rect;
  313. }
  314. [DefaultValue(true)]
  315. public bool ShowLines {
  316. get { return show_lines; }
  317. set {
  318. if (show_lines == value)
  319. return;
  320. show_lines = value;
  321. Refresh ();
  322. }
  323. }
  324. [DefaultValue(true)]
  325. public bool ShowPlusMinus {
  326. get { return show_plus_minus; }
  327. set {
  328. if (show_plus_minus == value)
  329. return;
  330. show_plus_minus = value;
  331. Refresh ();
  332. }
  333. }
  334. [DefaultValue(true)]
  335. public bool ShowRootLines {
  336. get { return show_root_lines; }
  337. set {
  338. if (show_root_lines == value)
  339. return;
  340. show_root_lines = value;
  341. Refresh ();
  342. }
  343. }
  344. [DefaultValue(false)]
  345. public bool Sorted {
  346. get { return sorted; }
  347. set {
  348. if (sorted != value)
  349. sorted = value;
  350. if (sorted) {
  351. Nodes.Sort ();
  352. top_node = null;
  353. Refresh ();
  354. }
  355. }
  356. }
  357. [Browsable(false)]
  358. [EditorBrowsable(EditorBrowsableState.Never)]
  359. [Bindable(false)]
  360. public override string Text {
  361. get { return base.Text; }
  362. set { base.Text = value; }
  363. }
  364. [Browsable(false)]
  365. [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  366. public TreeNode TopNode {
  367. get { return top_node; }
  368. }
  369. [Browsable(false)]
  370. [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  371. public int VisibleCount {
  372. get {
  373. return ClientRectangle.Height / ItemHeight;
  374. }
  375. }
  376. #endregion // Public Instance Properties
  377. #region Protected Instance Properties
  378. protected override CreateParams CreateParams {
  379. get {
  380. CreateParams cp = base.CreateParams;
  381. return cp;
  382. }
  383. }
  384. protected override Size DefaultSize {
  385. get { return new Size (121, 97); }
  386. }
  387. #endregion // Protected Instance Properties
  388. #region Public Instance Methods
  389. public void BeginUpdate () {
  390. update_stack++;
  391. }
  392. public void CollapseAll ()
  393. {
  394. root_node.CollapseAll ();
  395. }
  396. public void EndUpdate ()
  397. {
  398. if (update_stack > 1) {
  399. update_stack--;
  400. } else {
  401. update_stack = 0;
  402. if (update_needed) {
  403. Invalidate (ViewportRectangle);
  404. update_needed = false;
  405. }
  406. }
  407. }
  408. public void ExpandAll () {
  409. root_node.ExpandAll ();
  410. }
  411. public TreeNode GetNodeAt (Point pt) {
  412. return GetNodeAt (pt.Y);
  413. }
  414. public TreeNode GetNodeAt (int x, int y)
  415. {
  416. return GetNodeAt (y);
  417. }
  418. private TreeNode GetNodeAtUseX (int x, int y) {
  419. TreeNode node = GetNodeAt (y);
  420. if (node == null || !(IsTextArea (node, x) || full_row_select))
  421. return null;
  422. return node;
  423. }
  424. public int GetNodeCount (bool include_subtrees) {
  425. return root_node.GetNodeCount (include_subtrees);
  426. }
  427. public override string ToString () {
  428. int count = Nodes.Count;
  429. if (count <= 0)
  430. return String.Concat (base.ToString (), "Node Count: 0");
  431. return String.Concat (base.ToString (), "Node Count: ", count, " Nodes[0]: ", Nodes [0]);
  432. }
  433. #endregion // Public Instance Methods
  434. #region Protected Instance Methods
  435. protected override void CreateHandle () {
  436. base.CreateHandle ();
  437. }
  438. protected override void Dispose (bool disposing) {
  439. if (disposing)
  440. image_list = null;
  441. base.Dispose (disposing);
  442. }
  443. protected OwnerDrawPropertyBag GetItemRenderStyles (TreeNode node, int state) {
  444. return node.prop_bag;
  445. }
  446. protected override bool IsInputKey (Keys key_data) {
  447. if ((key_data & Keys.Alt) == 0) {
  448. switch (key_data & Keys.KeyCode) {
  449. case Keys.Enter:
  450. case Keys.Escape:
  451. case Keys.Prior:
  452. case Keys.Next:
  453. case Keys.End:
  454. case Keys.Home:
  455. case Keys.Left:
  456. case Keys.Up:
  457. case Keys.Right:
  458. case Keys.Down:
  459. return true;
  460. }
  461. }
  462. return base.IsInputKey (key_data);
  463. }
  464. protected override void OnKeyDown (KeyEventArgs e)
  465. {
  466. OpenTreeNodeEnumerator ne;
  467. switch (e.KeyData & Keys.KeyCode) {
  468. case Keys.Add:
  469. if (selected_node != null && selected_node.IsExpanded)
  470. selected_node.Expand ();
  471. break;
  472. case Keys.Subtract:
  473. if (selected_node != null && selected_node.IsExpanded)
  474. selected_node.Collapse ();
  475. break;
  476. case Keys.Left:
  477. if (selected_node != null) {
  478. if (selected_node.IsExpanded)
  479. selected_node.Collapse ();
  480. else {
  481. ne = new OpenTreeNodeEnumerator (selected_node);
  482. if (ne.MovePrevious () && ne.MovePrevious ())
  483. SelectedNode = ne.CurrentNode;
  484. }
  485. }
  486. break;
  487. case Keys.Right:
  488. if (selected_node != null) {
  489. if (!selected_node.IsExpanded)
  490. selected_node.Expand ();
  491. else {
  492. ne = new OpenTreeNodeEnumerator (selected_node);
  493. if (ne.MoveNext () && ne.MoveNext ())
  494. SelectedNode = ne.CurrentNode;
  495. }
  496. }
  497. break;
  498. case Keys.Up:
  499. if (selected_node != null) {
  500. ne = new OpenTreeNodeEnumerator (selected_node);
  501. if (ne.MovePrevious () && ne.MovePrevious ())
  502. SelectedNode = ne.CurrentNode;
  503. }
  504. break;
  505. case Keys.Down:
  506. if (selected_node != null) {
  507. ne = new OpenTreeNodeEnumerator (selected_node);
  508. if (ne.MoveNext () && ne.MoveNext ())
  509. SelectedNode = ne.CurrentNode;
  510. }
  511. break;
  512. case Keys.Home:
  513. if (root_node.Nodes.Count > 0) {
  514. ne = new OpenTreeNodeEnumerator (root_node.Nodes [0]);
  515. if (ne.MoveNext ())
  516. SelectedNode = ne.CurrentNode;
  517. }
  518. break;
  519. case Keys.End:
  520. if (root_node.Nodes.Count > 0) {
  521. ne = new OpenTreeNodeEnumerator (root_node.Nodes [0]);
  522. while (ne.MoveNext ())
  523. { }
  524. SelectedNode = ne.CurrentNode;
  525. }
  526. break;
  527. case Keys.PageDown:
  528. if (selected_node != null) {
  529. ne = new OpenTreeNodeEnumerator (selected_node);
  530. int move = ViewportRectangle.Height / ItemHeight;
  531. for (int i = 0; i < move && ne.MoveNext (); i++) {
  532. }
  533. SelectedNode = ne.CurrentNode;
  534. }
  535. break;
  536. case Keys.PageUp:
  537. if (selected_node != null) {
  538. ne = new OpenTreeNodeEnumerator (selected_node);
  539. int move = ViewportRectangle.Height / ItemHeight;
  540. for (int i = 0; i < move && ne.MovePrevious (); i++)
  541. { }
  542. SelectedNode = ne.CurrentNode;
  543. }
  544. break;
  545. case Keys.Multiply:
  546. if (selected_node != null)
  547. selected_node.ExpandAll ();
  548. break;
  549. }
  550. base.OnKeyDown (e);
  551. if (!e.Handled && checkboxes &&
  552. selected_node != null &&
  553. (e.KeyData & Keys.KeyCode) == Keys.Space) {
  554. selected_node.check_reason = TreeViewAction.ByKeyboard;
  555. selected_node.Checked = !selected_node.Checked;
  556. e.Handled = true;
  557. }
  558. }
  559. protected override void OnKeyPress (KeyPressEventArgs e)
  560. {
  561. base.OnKeyPress (e);
  562. if (e.KeyChar == ' ')
  563. e.Handled = true;
  564. }
  565. protected override void OnKeyUp (KeyEventArgs e)
  566. {
  567. base.OnKeyUp (e);
  568. if ((e.KeyData & Keys.KeyCode) == Keys.Space)
  569. e.Handled = true;
  570. }
  571. protected virtual void OnItemDrag (ItemDragEventArgs e)
  572. {
  573. if (ItemDrag != null)
  574. ItemDrag (this, e);
  575. }
  576. protected internal virtual void OnAfterCheck (TreeViewEventArgs e) {
  577. if (on_after_check != null)
  578. on_after_check (this, e);
  579. }
  580. protected internal virtual void OnAfterCollapse (TreeViewEventArgs e) {
  581. if (on_after_collapse != null)
  582. on_after_collapse (this, e);
  583. }
  584. protected internal virtual void OnAfterExpand (TreeViewEventArgs e) {
  585. if (on_after_expand != null)
  586. on_after_expand (this, e);
  587. }
  588. protected virtual void OnAfterLabelEdit (NodeLabelEditEventArgs e) {
  589. if (on_after_label_edit != null)
  590. on_after_label_edit (this, e);
  591. }
  592. protected virtual void OnAfterSelect (TreeViewEventArgs e) {
  593. if (on_after_select != null)
  594. on_after_select (this, e);
  595. }
  596. protected internal virtual void OnBeforeCheck (TreeViewCancelEventArgs e) {
  597. if (on_before_check != null)
  598. on_before_check (this, e);
  599. }
  600. protected internal virtual void OnBeforeCollapse (TreeViewCancelEventArgs e) {
  601. if (on_before_collapse != null)
  602. on_before_collapse (this, e);
  603. }
  604. protected internal virtual void OnBeforeExpand (TreeViewCancelEventArgs e) {
  605. if (on_before_expand != null)
  606. on_before_expand (this, e);
  607. }
  608. protected virtual void OnBeforeLabelEdit (NodeLabelEditEventArgs e) {
  609. if (on_before_label_edit != null)
  610. on_before_label_edit (this, e);
  611. }
  612. protected virtual void OnBeforeSelect (TreeViewCancelEventArgs e) {
  613. if (on_before_select != null)
  614. on_before_select (this, e);
  615. }
  616. protected override void OnHandleCreated (EventArgs e) {
  617. base.OnHandleCreated (e);
  618. vbar = new ImplicitVScrollBar ();
  619. hbar = new ImplicitHScrollBar ();
  620. vbar.Visible = false;
  621. hbar.Visible = false;
  622. vbar.ValueChanged += new EventHandler (VScrollBarValueChanged);
  623. hbar.ValueChanged += new EventHandler (HScrollBarValueChanged);
  624. SuspendLayout ();
  625. Controls.AddImplicit (vbar);
  626. Controls.AddImplicit (hbar);
  627. ResumeLayout ();
  628. }
  629. protected override void OnHandleDestroyed (EventArgs e) {
  630. base.OnHandleDestroyed (e);
  631. }
  632. protected override void WndProc(ref Message m) {
  633. switch ((Msg) m.Msg) {
  634. case Msg.WM_PAINT: {
  635. PaintEventArgs paint_event;
  636. paint_event = XplatUI.PaintEventStart (Handle, true);
  637. DoPaint (paint_event);
  638. XplatUI.PaintEventEnd (Handle, true);
  639. return;
  640. }
  641. case Msg.WM_LBUTTONDBLCLK:
  642. int val = m.LParam.ToInt32();
  643. DoubleClickHandler (null, new
  644. MouseEventArgs (MouseButtons.Left,
  645. 2, val & 0xffff,
  646. (val>>16) & 0xffff, 0));
  647. break;
  648. }
  649. base.WndProc (ref m);
  650. }
  651. #endregion // Protected Instance Methods
  652. #region Internal & Private Methods and Properties
  653. internal string LabelEditText {
  654. get {
  655. if (edit_text_box == null)
  656. return String.Empty;
  657. return edit_text_box.Text;
  658. }
  659. }
  660. internal IntPtr CreateNodeHandle ()
  661. {
  662. return (IntPtr) handle_count++;
  663. }
  664. internal TreeNode NodeFromHandle (IntPtr handle)
  665. {
  666. // This method is called rarely, so instead of maintaining a table
  667. // we just walk the tree nodes to find the matching handle
  668. return NodeFromHandleRecursive (root_node, handle);
  669. }
  670. private TreeNode NodeFromHandleRecursive (TreeNode node, IntPtr handle)
  671. {
  672. if (node.handle == handle)
  673. return node;
  674. foreach (TreeNode child in node.Nodes) {
  675. TreeNode match = NodeFromHandleRecursive (child, handle);
  676. if (match != null)
  677. return match;
  678. }
  679. return null;
  680. }
  681. // TODO: we shouldn't have to compute this on the fly
  682. internal Rectangle ViewportRectangle {
  683. get {
  684. Rectangle res = ClientRectangle;
  685. if (vbar != null && vbar.Visible)
  686. res.Width -= vbar.Width;
  687. if (hbar != null && hbar.Visible)
  688. res.Height -= hbar.Height;
  689. return res;
  690. }
  691. }
  692. private TreeNode GetNodeAt (int y)
  693. {
  694. if (nodes.Count <= 0)
  695. return null;
  696. if (top_node == null)
  697. top_node = nodes [0];
  698. OpenTreeNodeEnumerator o = new OpenTreeNodeEnumerator (TopNode);
  699. int move = y / ItemHeight;
  700. for (int i = -1; i < move; i++) {
  701. if (!o.MoveNext ())
  702. return null;
  703. }
  704. return o.CurrentNode;
  705. }
  706. private bool IsTextArea (TreeNode node, int x)
  707. {
  708. return node != null && node.Bounds.Left <= x && node.Bounds.Right >= x;
  709. }
  710. private bool IsSelectableArea (TreeNode node, int x)
  711. {
  712. if (node == null)
  713. return false;
  714. int l = node.Bounds.Left;
  715. if (ImageList != null)
  716. l -= ImageList.ImageSize.Width;
  717. return l <= x && node.Bounds.Right >= x;
  718. }
  719. private bool IsPlusMinusArea (TreeNode node, int x)
  720. {
  721. if (node.Nodes.Count == 0 || (node.parent == root_node && !show_root_lines))
  722. return false;
  723. int l = node.Bounds.Left + 5;
  724. if (show_root_lines || node.Parent != null)
  725. l -= indent;
  726. if (ImageList != null)
  727. l -= ImageList.ImageSize.Width + 3;
  728. if (checkboxes)
  729. l -= 19;
  730. return (x > l && x < l + 8);
  731. }
  732. private bool IsCheckboxArea (TreeNode node, int x)
  733. {
  734. int l = node.Bounds.Left + 5;
  735. if (show_root_lines || node.Parent != null)
  736. l -= indent;
  737. if (ImageList != null)
  738. l -= ImageList.ImageSize.Width + 3;
  739. return (x > l && x < l + 10);
  740. }
  741. internal void SetTop (TreeNode node)
  742. {
  743. OpenTreeNodeEnumerator walk = new OpenTreeNodeEnumerator (root_node);
  744. int offset = 0;
  745. while (walk.CurrentNode != node && walk.MoveNext ())
  746. offset++;
  747. vbar.Value = offset;
  748. }
  749. internal void SetBottom (TreeNode node)
  750. {
  751. OpenTreeNodeEnumerator walk = new OpenTreeNodeEnumerator (node);
  752. int bottom = ViewportRectangle.Bottom;
  753. int offset = 0;
  754. while (walk.MovePrevious ()) {
  755. if (walk.CurrentNode.Bounds.Bottom <= bottom)
  756. break;
  757. offset++;
  758. }
  759. vbar.Value += offset;
  760. }
  761. internal void UpdateBelow (TreeNode node)
  762. {
  763. if (update_stack > 0) {
  764. update_needed = true;
  765. return;
  766. }
  767. if (node == root_node) {
  768. Refresh ();
  769. return;
  770. }
  771. // We need to update the current node so the plus/minus block gets update too
  772. Rectangle invalid = new Rectangle (0, node.Bounds.Top - 1,
  773. Width, Height - node.Bounds.Top + 1);
  774. Invalidate (invalid);
  775. }
  776. internal void UpdateNode (TreeNode node)
  777. {
  778. if (update_stack > 0) {
  779. update_needed = true;
  780. return;
  781. }
  782. if (node == root_node) {
  783. Refresh ();
  784. return;
  785. }
  786. Rectangle invalid = new Rectangle (0, node.Bounds.Top - 1, Width,
  787. node.Bounds.Height + 1);
  788. Invalidate (invalid);
  789. }
  790. internal void UpdateNodePlusMinus (TreeNode node)
  791. {
  792. if (update_stack > 0) {
  793. update_needed = true;
  794. return;
  795. }
  796. int l = node.Bounds.Left + 5;
  797. if (show_root_lines || node.Parent != null)
  798. l -= indent;
  799. if (ImageList != null)
  800. l -= ImageList.ImageSize.Width + 3;
  801. if (checkboxes)
  802. l -= 19;
  803. Invalidate (new Rectangle (l, node.Bounds.Top, 8, node.Bounds.Height));
  804. }
  805. private void DoPaint (PaintEventArgs pe)
  806. {
  807. if (Width <= 0 || Height <= 0 || Visible == false)
  808. return;
  809. Draw (pe.ClipRectangle, pe.Graphics);
  810. }
  811. private void Draw (Rectangle clip, Graphics dc)
  812. {
  813. if (top_node == null && Nodes.Count > 0)
  814. top_node = nodes [0];
  815. if (selected_node == null && Nodes.Count > 0)
  816. SelectedNode = nodes [0];
  817. // Decide if we need a scrollbar
  818. int old_open_node_count = open_node_count;
  819. //Rectangle fill = ClientRectangle;
  820. add_vscroll = false;
  821. add_hscroll = false;
  822. dc.FillRectangle (ThemeEngine.Current.ResPool.GetSolidBrush (BackColor), clip);
  823. Color dash_color = ControlPaint.Dark (BackColor);
  824. if (dash_color == BackColor)
  825. dash_color = ControlPaint.Light (BackColor);
  826. dash = new Pen (dash_color, 1);
  827. dash.DashStyle = DashStyle.Dot;
  828. int depth = 0;
  829. int item_height = ItemHeight;
  830. int height = ClientRectangle.Height;
  831. open_node_count = 0;
  832. used_height = 0;
  833. foreach (TreeNode node in nodes) {
  834. DrawNode (node, dc, clip, ref depth, item_height, height);
  835. depth = 0;
  836. }
  837. add_vscroll = (open_node_count * ItemHeight) > ClientRectangle.Height;
  838. if (max_node_width > ClientRectangle.Width)
  839. add_hscroll = true;
  840. if (add_vscroll)
  841. add_hscroll = max_node_width > ClientRectangle.Width - ThemeEngine.Current.VScrollBarDefaultSize.Width;
  842. if (add_hscroll)
  843. add_vscroll = (open_node_count * ItemHeight) > ClientRectangle.Height - ThemeEngine.Current.HScrollBarDefaultSize.Height;
  844. if (add_hscroll) {
  845. AddHorizontalScrollBar ();
  846. } else if (hbar != null) {
  847. hbar_offset = 0;
  848. hbar.Visible = false;
  849. }
  850. if (add_vscroll) {
  851. AddVerticalScrollBar (open_node_count, old_open_node_count != open_node_count);
  852. } else if (vbar != null) {
  853. vbar.Visible = false;
  854. skipped_nodes = 0;
  855. }
  856. if (add_hscroll && add_vscroll) {
  857. Rectangle corner = new Rectangle (hbar.Right, vbar.Bottom, vbar.Width, hbar.Height);
  858. if (clip.IntersectsWith (corner))
  859. dc.FillRectangle (ThemeEngine.Current.ResPool.GetSolidBrush (ThemeEngine.Current.ColorControl), corner);
  860. }
  861. }
  862. private void DrawNodePlusMinus (TreeNode node, Graphics dc, int x, int middle)
  863. {
  864. dc.DrawRectangle (SystemPens.ControlDark, x, middle - 4, 8, 8);
  865. if (node.IsExpanded) {
  866. dc.DrawLine (SystemPens.ControlDarkDark, x + 2, middle, x + 6, middle);
  867. } else {
  868. dc.DrawLine (SystemPens.ControlDarkDark, x + 2, middle, x + 6, middle);
  869. dc.DrawLine (SystemPens.ControlDarkDark, x + 4, middle - 2, x + 4, middle + 2);
  870. }
  871. }
  872. private void DrawNodeCheckBox (TreeNode node, Graphics dc, int x, int middle)
  873. {
  874. using(Pen pen = new Pen (Color.Black, 2) )
  875. dc.DrawRectangle (pen, x+ 3, middle - 4, 11, 11);
  876. if (node.Checked) {
  877. using(Pen check_pen = new Pen (Color.Black, 1)) {
  878. dc.DrawLine (check_pen, x + 6, middle + 0, x + 8, middle + 3);
  879. dc.DrawLine (check_pen, x + 6, middle + 1, x + 8, middle + 4);
  880. dc.DrawLine (check_pen, x + 7, middle + 3, x + 13, middle - 2);
  881. dc.DrawLine (check_pen, x + 7, middle + 4, x + 13, middle - 1);
  882. }
  883. }
  884. }
  885. private void DrawNodeLines (TreeNode node, Graphics dc, bool visible, Pen dash, int x, int y,
  886. int middle, int item_height, int node_count)
  887. {
  888. int ladjust = 9; // left adjust
  889. int radjust = 0; // right adjust
  890. if (node_count > 0 && show_plus_minus)
  891. ladjust = 13;
  892. if (checkboxes)
  893. radjust = 3;
  894. dc.DrawLine (dash, x - indent + ladjust, middle, x + radjust, middle);
  895. //if (!visible)
  896. // return;
  897. int ly = 0;
  898. if (node.PrevNode != null) {
  899. int prevadjust = (node.Nodes.Count > 0 && show_plus_minus ? (node.PrevNode.Nodes.Count == 0 ? 0 : 4) :
  900. (node.PrevNode.Nodes.Count == 0 ? 0 : 4));
  901. int myadjust = (node.Nodes.Count > 0 && show_plus_minus ? 4 : 0);
  902. ly = node.PrevNode.Bounds.Bottom - (item_height / 2) + prevadjust;
  903. dc.DrawLine (dash, x - indent + 9, middle - myadjust, x - indent + 9, ly);
  904. } else if (node.Parent != null) {
  905. int myadjust = (node.Nodes.Count > 0 && show_plus_minus ? 4 : 0);
  906. ly = node.Parent.Bounds.Bottom + 1;
  907. dc.DrawLine (dash, x - indent + 9, middle - myadjust, x - indent + 9, ly);
  908. }
  909. }
  910. private void DrawNodeImage (TreeNode node, Graphics dc, Rectangle clip, int x, int y)
  911. {
  912. // Rectangle r = new Rectangle (x, y + 2, ImageList.ImageSize.Width, ImageList.ImageSize.Height);
  913. if (!RectsIntersect (clip, x, y + 2, ImageList.ImageSize.Width, ImageList.ImageSize.Height))
  914. return;
  915. if (node.ImageIndex > -1 && ImageList != null && node.ImageIndex < ImageList.Images.Count) {
  916. ImageList.Draw (dc, x, y + 2, ImageList.ImageSize.Width,
  917. ImageList.ImageSize.Height, node.ImageIndex);
  918. } else if (ImageIndex > -1 && ImageList != null && ImageIndex < ImageList.Images.Count) {
  919. ImageList.Draw (dc, x, y + 2, ImageList.ImageSize.Width,
  920. ImageList.ImageSize.Height, ImageIndex);
  921. }
  922. }
  923. private void DrawEditNode (TreeNode node)
  924. {
  925. SuspendLayout ();
  926. if (edit_text_box == null) {
  927. edit_text_box = new TextBox ();
  928. edit_text_box.BorderStyle = BorderStyle.FixedSingle;
  929. edit_text_box.KeyUp += new KeyEventHandler (EditTextBoxKeyDown);
  930. edit_text_box.Leave += new EventHandler (EditTextBoxLeave);
  931. Controls.AddImplicit (edit_text_box);
  932. }
  933. edit_text_box.Bounds = node.Bounds;
  934. edit_text_box.Width += 4;
  935. edit_text_box.Text = node.Text;
  936. edit_text_box.Visible = true;
  937. edit_text_box.Focus ();
  938. edit_text_box.SelectAll ();
  939. ResumeLayout ();
  940. }
  941. private void EditTextBoxKeyDown (object sender, KeyEventArgs e)
  942. {
  943. if (e.KeyCode == Keys.Return)
  944. EndEdit ();
  945. }
  946. private void EditTextBoxLeave (object sender, EventArgs e)
  947. {
  948. EndEdit ();
  949. }
  950. private void EndEdit ()
  951. {
  952. edit_text_box.Visible = false;
  953. edit_node.EndEdit (false);
  954. UpdateNode(edit_node);
  955. }
  956. private void UpdateNodeBounds (TreeNode node, int x, int y, int item_height, Graphics dc)
  957. {
  958. Font font = node.NodeFont;
  959. if (node.NodeFont == null)
  960. font = Font;
  961. if (node.NeedsWidth || update_node_bounds)
  962. node.SetWidth ((int) dc.MeasureString (node.Text, font, 0,
  963. string_format).Width + 3);
  964. node.SetHeight (item_height);
  965. node.SetPosition (x, y);
  966. }
  967. private void DrawSelectionAndFocus(TreeNode node, Graphics dc, Rectangle r)
  968. {
  969. if (Focused && focused_node == node) {
  970. ControlPaint.DrawFocusRectangle (dc, r, ForeColor, BackColor);
  971. }
  972. r.Inflate(-1, -1);
  973. if ((!HideSelection || Focused) && SelectedNode == node)
  974. dc.FillRectangle (ThemeEngine.Current.ResPool.GetSolidBrush (ThemeEngine.Current.ColorHighlight), r);
  975. else
  976. dc.FillRectangle (ThemeEngine.Current.ResPool.GetSolidBrush (node.BackColor), r);
  977. }
  978. private void DrawStaticNode (TreeNode node, Graphics dc)
  979. {
  980. if (!full_row_select)
  981. DrawSelectionAndFocus(node, dc, node.Bounds);
  982. Font font = node.NodeFont;
  983. if (node.NodeFont == null)
  984. font = Font;
  985. Color text_color = ((Focused || !HideSelection) && SelectedNode == node ?
  986. ThemeEngine.Current.ColorHighlightText : node.ForeColor);
  987. dc.DrawString (node.Text, font,
  988. ThemeEngine.Current.ResPool.GetSolidBrush (text_color),
  989. node.Bounds, string_format);
  990. }
  991. private void DrawNode (TreeNode node, Graphics dc, Rectangle clip, ref int depth, int item_height, int max_height)
  992. {
  993. open_node_count++;
  994. int x = (!show_root_lines && node.Parent != null ? depth - 1 : depth) * indent - hbar_offset;
  995. int y = item_height * (open_node_count - skipped_nodes - 1);
  996. bool visible = (y >= 0 && y < max_height);
  997. int _n_count = node.nodes.Count;
  998. int middle = y + (item_height / 2);
  999. // The thing is totally out of the clipping rectangle
  1000. if (clip.Top > y + ItemHeight || clip.Bottom < y)
  1001. visible = false;
  1002. if (visible && full_row_select) {
  1003. Rectangle r = new Rectangle(1, y+2, ViewportRectangle.Width-2, item_height);
  1004. DrawSelectionAndFocus(node, dc, r);
  1005. }
  1006. if (show_root_lines || node.Parent != null) {
  1007. x += 5;
  1008. if (_n_count > 0) {
  1009. if (show_plus_minus && visible) {
  1010. DrawNodePlusMinus (node, dc, x, middle);
  1011. }
  1012. }
  1013. x += indent - 5;
  1014. }
  1015. int ox = x;
  1016. if (checkboxes) {
  1017. if (visible)
  1018. DrawNodeCheckBox (node, dc, ox, middle);
  1019. ox += 19;
  1020. }
  1021. if (show_lines)
  1022. DrawNodeLines (node, dc, visible, dash, x, y, middle, item_height, _n_count);
  1023. if (ImageList != null) {
  1024. if (visible)
  1025. DrawNodeImage (node, dc, clip, ox, y);
  1026. // MS leaves the space for the image if the ImageList is
  1027. // non null regardless of whether or not an image is drawn
  1028. ox += ImageList.ImageSize.Width + 3; // leave a little space so the text isn't against the image
  1029. }
  1030. UpdateNodeBounds (node, ox, y, item_height, dc);
  1031. bool bounds_in_clip = clip.IntersectsWith (node.Bounds) || full_row_select;
  1032. if (visible && bounds_in_clip) {
  1033. if (node.IsEditing)
  1034. DrawEditNode (node);
  1035. else
  1036. DrawStaticNode (node, dc);
  1037. }
  1038. if (node.Bounds.Right > max_node_width) {
  1039. max_node_width = node.Bounds.Right;
  1040. if (max_node_width > ClientRectangle.Width && !add_hscroll) {
  1041. max_height -= ItemHeight;
  1042. add_hscroll = true;
  1043. }
  1044. }
  1045. if (node.Bounds.Bottom > used_height)
  1046. used_height = node.Bounds.Bottom;
  1047. depth++;
  1048. if (node.IsExpanded) {
  1049. for (int i = 0; i < _n_count; i++) {
  1050. int tdepth = depth;
  1051. DrawNode (node.nodes [i], dc, clip, ref tdepth, item_height, max_height);
  1052. }
  1053. }
  1054. }
  1055. private void AddVerticalScrollBar (int total_nodes, bool count_changed)
  1056. {
  1057. vbar.Bounds = new Rectangle (ClientRectangle.Width - vbar.Width,
  1058. 0, vbar.Width, (add_hscroll ? Height - ThemeEngine.Current.HScrollBarDefaultSize.Height : Height));
  1059. if (count_changed) {
  1060. vbar.Maximum = total_nodes;
  1061. int height = ClientRectangle.Height;
  1062. vbar.LargeChange = height / ItemHeight;
  1063. }
  1064. vbar.Visible = true;
  1065. }
  1066. private void AddHorizontalScrollBar ()
  1067. {
  1068. hbar.Bounds = new Rectangle (ClientRectangle.Left,
  1069. ClientRectangle.Bottom - hbar.Height,
  1070. (add_vscroll ? Width - ThemeEngine.Current.VScrollBarDefaultSize.Width : Width), hbar.Height);
  1071. int width = ClientRectangle.Width;
  1072. if (vbar.Visible)
  1073. width -= vbar.Width;
  1074. hbar.SmallChange = 15;
  1075. hbar.LargeChange = 50;
  1076. int num_pixels = max_node_width - width;
  1077. hbar.Maximum = num_pixels + 50;
  1078. hbar.Visible = true;
  1079. }
  1080. private void SizeChangedHandler (object sender, EventArgs e)
  1081. {
  1082. SuspendLayout ();
  1083. if (max_node_width > ClientRectangle.Width && ClientRectangle.Width >= 0) {
  1084. add_hscroll = true;
  1085. AddHorizontalScrollBar ();
  1086. return;
  1087. }
  1088. if (used_height > ClientRectangle.Height && ClientRectangle.Height >= 0) {
  1089. add_vscroll = true;
  1090. AddVerticalScrollBar (open_node_count, true);
  1091. return;
  1092. }
  1093. if (vbar != null && vbar.Visible) {
  1094. int height = (hbar != null && hbar.Visible ? Height - hbar.Height : Height);
  1095. vbar.SetBounds (Right - vbar.Width, 0, 0, height, BoundsSpecified.X | BoundsSpecified.Height);
  1096. }
  1097. if (hbar != null && hbar.Visible) {
  1098. int width = (vbar != null && vbar.Visible ? Width - vbar.Width : Width);
  1099. hbar.SetBounds (0, Bottom - hbar.Height, width, 0, BoundsSpecified.Y | BoundsSpecified.Width);
  1100. }
  1101. ResumeLayout ();
  1102. }
  1103. private void VScrollBarValueChanged (object sender, EventArgs e)
  1104. {
  1105. SetVScrollPos (vbar.Value, null);
  1106. }
  1107. private void SetVScrollPos (int pos, TreeNode new_top)
  1108. {
  1109. if (pos < 0)
  1110. pos = 0;
  1111. if (skipped_nodes == pos)
  1112. return;
  1113. int old_skip = skipped_nodes;
  1114. skipped_nodes = pos;
  1115. int diff = old_skip - skipped_nodes;
  1116. // Determine the new top node if we have to
  1117. if (new_top == null) {
  1118. if (top_node == null)
  1119. top_node = nodes [0];
  1120. OpenTreeNodeEnumerator walk = new OpenTreeNodeEnumerator (TopNode);
  1121. if (diff < 0) {
  1122. for (int i = diff; i <= 0; i++)
  1123. walk.MoveNext ();
  1124. new_top = walk.CurrentNode;
  1125. } else {
  1126. for (int i = 0; i <= diff; i++)
  1127. walk.MovePrevious ();
  1128. new_top = walk.CurrentNode;
  1129. }
  1130. }
  1131. top_node = new_top;
  1132. int y_move = diff * ItemHeight;
  1133. XplatUI.ScrollWindow (Handle, ViewportRectangle, 0, y_move, false);
  1134. }
  1135. private void HScrollBarValueChanged(object sender, EventArgs e)
  1136. {
  1137. int old_offset = hbar_offset;
  1138. hbar_offset = hbar.Value;
  1139. if (hbar_offset < 0)
  1140. hbar_offset = 0;
  1141. XplatUI.ScrollWindow (Handle, ViewportRectangle, old_offset - hbar_offset, 0, false);
  1142. }
  1143. private void MouseWheelHandler(object sender, MouseEventArgs e) {
  1144. if (vbar == null || !vbar.Visible) {
  1145. return;
  1146. }
  1147. if (e.Delta < 0) {
  1148. vbar.Value = Math.Min(vbar.Value + SystemInformation.MouseWheelScrollLines, vbar.Maximum);
  1149. } else {
  1150. vbar.Value = Math.Max(0, vbar.Value - SystemInformation.MouseWheelScrollLines);
  1151. }
  1152. }
  1153. private void FontChangedHandler (object sender, EventArgs e)
  1154. {
  1155. update_node_bounds = true;
  1156. }
  1157. private void FocusChangedHandler (object sender, EventArgs e)
  1158. {
  1159. if (selected_node != null)
  1160. UpdateNode (selected_node);
  1161. }
  1162. private void MouseDownHandler (object sender, MouseEventArgs e)
  1163. {
  1164. TreeNode node = GetNodeAt (e.Y);
  1165. if (node == null)
  1166. return;
  1167. if (show_plus_minus && IsPlusMinusArea (node, e.X)) {
  1168. node.Toggle ();
  1169. return;
  1170. } else if (checkboxes && IsCheckboxArea (node, e.X)) {
  1171. node.check_reason = TreeViewAction.ByMouse;
  1172. node.Checked = !node.Checked;
  1173. UpdateNode(node);
  1174. return;
  1175. } else if (IsSelectableArea (node, e.X) || full_row_select) {
  1176. TreeNode old_selected = selected_node;
  1177. SelectedNode = node;
  1178. if (label_edit && e.Clicks == 1 && selected_node == old_selected) {
  1179. node.BeginEdit ();
  1180. if (edit_node != null) {
  1181. edit_node.EndEdit (false);
  1182. UpdateNode (edit_node);
  1183. }
  1184. edit_node = node;
  1185. UpdateNode (edit_node);
  1186. } else if (selected_node != focused_node) {
  1187. select_mmove = true;
  1188. }
  1189. }
  1190. }
  1191. private void MouseUpHandler (object sender, MouseEventArgs e) {
  1192. drag_begin_x = -1;
  1193. drag_begin_y = -1;
  1194. OnClick (EventArgs.Empty);
  1195. if (!select_mmove)
  1196. return;
  1197. select_mmove = false;
  1198. TreeViewCancelEventArgs ce = new TreeViewCancelEventArgs (selected_node, false, TreeViewAction.ByMouse);
  1199. OnBeforeSelect (ce);
  1200. Rectangle invalid;
  1201. if (!ce.Cancel) {
  1202. if (focused_node != null) {
  1203. invalid = Rectangle.Union (Bloat (focused_node.Bounds),
  1204. Bloat (selected_node.Bounds));
  1205. } else {
  1206. invalid = Bloat (selected_node.Bounds);
  1207. }
  1208. focused_node = selected_node;
  1209. OnAfterSelect (new TreeViewEventArgs (selected_node, TreeViewAction.ByMouse));
  1210. Invalidate (invalid);
  1211. } else {
  1212. selected_node = focused_node;
  1213. }
  1214. }
  1215. private void MouseMoveHandler (object sender, MouseEventArgs e) {
  1216. if (e.Button == MouseButtons.Left || e.Button == MouseButtons.Right) {
  1217. if (drag_begin_x == -1 && drag_begin_y == -1) {
  1218. drag_begin_x = e.X;
  1219. drag_begin_y = e.Y;
  1220. } else {
  1221. double rise = Math.Pow (drag_begin_x - e.X, 2);
  1222. double run = Math.Pow (drag_begin_y - e.Y, 2);
  1223. double move = Math.Sqrt (rise + run);
  1224. if (move > 3) {
  1225. TreeNode drag = GetNodeAtUseX (e.X, e.Y);
  1226. if (drag != null) {
  1227. OnItemDrag (new ItemDragEventArgs (e.Button, drag));
  1228. }
  1229. drag_begin_x = -1;
  1230. drag_begin_y = -1;
  1231. }
  1232. }
  1233. }
  1234. if(!select_mmove)
  1235. return;
  1236. TreeNode node = GetNodeAtUseX (e.X,e.Y);
  1237. if(node == selected_node)
  1238. return;
  1239. selected_node = focused_node;
  1240. select_mmove = false;
  1241. Refresh();
  1242. }
  1243. private void DoubleClickHandler (object sender, MouseEventArgs e) {
  1244. TreeNode node = GetNodeAtUseX (e.X,e.Y);
  1245. if(node != null) {
  1246. node.Toggle();
  1247. }
  1248. }
  1249. private bool RectsIntersect (Rectangle r, int left, int top, int width, int height)
  1250. {
  1251. return !((r.Left > left + width) || (r.Right < left) ||
  1252. (r.Top > top + height) || (r.Bottom < top));
  1253. }
  1254. #endregion // Internal & Private Methods and Properties
  1255. #region Events
  1256. public event ItemDragEventHandler ItemDrag;
  1257. public event TreeViewEventHandler AfterCheck {
  1258. add { on_after_check += value; }
  1259. remove { on_after_check -= value; }
  1260. }
  1261. public event TreeViewEventHandler AfterCollapse {
  1262. add { on_after_collapse += value; }
  1263. remove { on_after_collapse -= value; }
  1264. }
  1265. public event TreeViewEventHandler AfterExpand {
  1266. add { on_after_expand += value; }
  1267. remove { on_after_expand -= value; }
  1268. }
  1269. public event NodeLabelEditEventHandler AfterLabelEdit {
  1270. add { on_after_label_edit += value; }
  1271. remove { on_after_label_edit -= value; }
  1272. }
  1273. public event TreeViewEventHandler AfterSelect {
  1274. add { on_after_select += value; }
  1275. remove { on_after_select -= value; }
  1276. }
  1277. [Browsable (false)]
  1278. [EditorBrowsable (EditorBrowsableState.Never)]
  1279. public event EventHandler BackgroundImageChanged {
  1280. add { base.BackgroundImageChanged += value; }
  1281. remove { base.BackgroundImageChanged -= value; }
  1282. }
  1283. public event TreeViewCancelEventHandler BeforeCheck {
  1284. add { on_before_check += value; }
  1285. remove { on_before_check -= value; }
  1286. }
  1287. public event TreeViewCancelEventHandler BeforeCollapse {
  1288. add { on_before_collapse += value; }
  1289. remove { on_before_collapse -= value; }
  1290. }
  1291. public event TreeViewCancelEventHandler BeforeExpand {
  1292. add { on_before_expand += value; }
  1293. remove { on_before_expand -= value; }
  1294. }
  1295. public event NodeLabelEditEventHandler BeforeLabelEdit {
  1296. add { on_before_label_edit += value; }
  1297. remove { on_before_label_edit -= value; }
  1298. }
  1299. public event TreeViewCancelEventHandler BeforeSelect {
  1300. add { on_before_select += value; }
  1301. remove { on_before_select -= value; }
  1302. }
  1303. [EditorBrowsable (EditorBrowsableState.Never)]
  1304. [Browsable (false)]
  1305. public new event PaintEventHandler Paint {
  1306. add { base.Paint += value; }
  1307. remove { base.Paint -= value; }
  1308. }
  1309. [EditorBrowsable (EditorBrowsableState.Never)]
  1310. [Browsable (false)]
  1311. public new event EventHandler TextChanged {
  1312. add { base.TextChanged += value; }
  1313. remove { base.TextChanged -= value; }
  1314. }
  1315. #endregion // Events
  1316. }
  1317. }