2018年1月19日 星期五

還原TreeView 的展開狀態

例:
 
如果在 A 下的 C 下加入 D,如果只是在 TreeView 上改變顯示,那麼要遍歷整個 Tree,找出所有的 C (B 下有一個 C)之後,改變畫面,雖然可以這樣做,但程式碼上似乎複雜了點
所以改變思路,在 C 下加入 D 之後,重新綁定 TreeView 的資料。

但是重新綁定 TreeView 之後,TreeView 會收合到剩下第一層,使用者就要重新展開之後,再往下加入其他子項目,這點其實很麻煩。

所以在重新綁定前先記錄下目前 TreeView 的展開狀態,綁定之後再還原展開狀態,程式碼加下

設定兩個全域變數
private Dictionary<string, bool> NodesStatus = new Dictionary<string, bool>();  // 記錄展開狀態
private string SelectNodeFullPath = ""; // 記錄選擇到的節點

記錄展開狀態
/// <summary>
/// 記錄展開狀態
/// </summary>
/// <param name="nodes"></param>
private void GetTreeNodesStatus(TreeNodeCollection nodes)
{
 foreach (TreeNode node in nodes)
 {
  if (node.IsExpanded)
  {
   NodesStatus[node.FullPath] = true;
  }
  else
  {
   NodesStatus.Remove(node.FullPath);
  }

  if (node.IsSelected)
  {
   SelectNodeFullPath = node.FullPath;
  }
  GetTreeNodesStatus(node.Nodes);
 }
}

還原展開狀態
/// <summary>
/// 還原展開狀態
/// </summary>
/// <param name="nodes"></param>
private void SetTreeNodesStatus(TreeNodeCollection nodes)
{
 foreach (TreeNode node in nodes)
 {
  if (NodesStatus.ContainsKey(node.FullPath))
  {
   node.Expand();
  }

  if (node.FullPath == SelectNodeFullPath)
  {
   treeView1.SelectedNode = node;
  }

  SetTreeNodesStatus(node.Nodes);
 }
}

使用方式
// 對treeView1做了某些操作之後(例如加入子選項)
GetTreeNodesStatus(treeView1.Nodes);
BindTreeViewData(); // 綁定TreeView
SetTreeNodesStatus(treeView1.Nodes);

2018年1月17日 星期三

C# Window Form - 動態指定Form尺寸

如果想要動態控制 Form 的尺寸,做法會先指定以某個控制項(或原點)做為基準,指定控制項的位置後,確定最右下角的那個控制項之後,才能決定 Form 的尺寸。

比方右下角有一個 Button(button1),指定的 Form 的語法會是
this.Height = button1.Top + button1.Height;
this.Width = button1.Left + button1.Width;
但這樣加完的結果是看不到,因為 Form 的最上方有一條標題列,要再加上這標題列的高度才行,所以改成
this.Height = button1.Top + button1.Height + iControlSpacer + (this.Height-this.ClientSize.Height);
this.Width = button1.Left + button1.Width + iControlSpacer + (this.Width - this.ClientSize.Width);
其中 this.Height-this.ClientSize.Height 是標題列高度的計算方式,
iControlSpacer  是控制項之間的間隔