C#實現的下拉多選框,下拉多選樹,多級節點

来源:https://www.cnblogs.com/zhenzaizai/p/18120526
-Advertisement-
Play Games

今天給大家上個硬貨,下拉多選框,同時也是下拉多選樹,支持父節點跟子節點!該控制項是基於Telerik控制項封裝實現的,所以大家在使用的過程中需要引用Telerik.WinControls.dll、Telerik.WinControls.UI.dll,還有一些相關的類庫,大家有需要的可以去網上自己找,另外 ...


  今天給大家上個硬貨,下拉多選框,同時也是下拉多選樹,支持父節點跟子節點!該控制項是基於Telerik控制項封裝實現的,所以大家在使用的過程中需要引用Telerik.WinControls.dll、Telerik.WinControls.UI.dll,還有一些相關的類庫,大家有需要的可以去網上自己找,另外我也會把一些動態庫放到CSDN上面,大家需要可以去下載。

[ToolboxItem(true)]
    public partial class DropDownTreeViewControl : RadControl
    {
        public DropDownTreeViewElement TreeViewElement { get; private set; }

        public RadTreeView TreeViewControl
        {
            get
            {
                return this.TreeViewElement.TreeViewControl;
            }
        }

        protected override void CreateChildItems(RadElement parent)
        {
            this.AllowShowFocusCues = true;

            base.CreateChildItems(parent);

            this.TreeViewElement = new DropDownTreeViewElement();

            parent.Children.Add(TreeViewElement);

        }

        protected override void OnEnter(EventArgs e)
        {
            base.OnEnter(e);
            this.TreeViewElement.Focus();
        }

        protected override void OnBindingContextChanged(EventArgs e)
        {
            base.OnBindingContextChanged(e);

            this.TreeViewControl.BindingContext = this.BindingContext;
        }

        public class DropDownTreeViewElement : LightVisualElement
        {
            private readonly Color BG_COLOR = Color.White;
            private readonly Color BORDER_COLOR = Color.LightBlue;
            private readonly Color ARROW_BORDER_COLOR = Color.LightGray;
            private readonly Color ARROW_NORMAL_BG_COLOR = Color.White;
            private readonly Color ARROW_MOUSE_OVER_BG_COLOR = Color.LightYellow;
            private readonly Color ARROW_PRESSED_BG_COLOR = Color.DarkOrange;

            private readonly int NORMAL_BORDER_WIDTH = 1;
            private readonly int FOCUS_BORDER_WIDTH = 2;

            private RadArrowButtonElement arrow;
            private PopupForm popup;
            private bool isInnerCallHide;

            public bool IsPopupOpen { get; private set; }
            public RadTreeView TreeViewControl
            {
                get
                {
                    return this.popup.TreeView;
                }
            }

            protected override void InitializeFields()
            {
                base.InitializeFields();

                // style
                this.DrawBorder = true;
                this.BorderBoxStyle = BorderBoxStyle.SingleBorder;
                this.BorderGradientStyle = GradientStyles.Solid;
                this.BorderColor = BORDER_COLOR;
                this.DrawFill = true;
                this.NumberOfColors = 1;
                this.GradientStyle = GradientStyles.Solid;
                this.BackColor = BG_COLOR;
                this.StretchHorizontally = true;
                this.StretchVertically = true;
            }

            protected override void CreateChildElements()
            {
                base.CreateChildElements();

                // arrow
                this.CreateArrow();

                // popup
                this.CreatePopup();

                this.Children.Add(arrow);
            }

            private void CreatePopup()
            {
                this.popup = new PopupForm(this);
                this.popup.PopupClosing += Popup_PopupClosing;
                this.popup.PopupClosed += Popup_PopupClosed;
            }

            private void Popup_PopupClosing(object sender, RadPopupClosingEventArgs args)
            {
                // mouse postion in control-bounds prevent
                if (args.CloseReason == RadPopupCloseReason.Mouse)
                {
                    var boundsSc = RectangleToScreen(this.Bounds);
                    if (boundsSc.Contains(MousePosition))
                    {
                        args.Cancel = true;
                    }
                }
            }

            private void Popup_PopupClosed(object sender, RadPopupClosedEventArgs args)
            {
                if (isInnerCallHide)
                {
                    return;
                }
                this.IsPopupOpen = false;
                this.SwitchArrowState(false);
            }

            private void CreateArrow()
            {
                this.arrow = new RadArrowButtonElement()
                {
                    ClickMode = ClickMode.Press,
                    MinSize = new Size(SystemInformation.VerticalScrollBarWidth,
                                    RadArrowButtonElement.RadArrowButtonDefaultSize.Height),
                    StretchHorizontally = false,
                    StretchVertically = true,
                    Margin = new System.Windows.Forms.Padding(2),
                };

                arrow.Fill.NumberOfColors = 1;
                arrow.Fill.BackColor = ARROW_NORMAL_BG_COLOR;
                arrow.Border.BoxStyle = BorderBoxStyle.SingleBorder;
                arrow.Border.GradientStyle = GradientStyles.Solid;
                arrow.Border.ForeColor = ARROW_BORDER_COLOR;

                arrow.RadPropertyChanged += Arrow_RadPropertyChanged;
                arrow.Click += Arrow_Click;
            }

            private void Arrow_Click(object sender, EventArgs e)
            {
                if (this.IsPopupOpen)
                {
                    this.IsPopupOpen = false;
                    this.SwitchArrowState(false);
                    this.HidePopup();
                }
                else
                {
                    this.IsPopupOpen = true;
                    this.SwitchArrowState(true);
                    this.ShowPopup();
                }
            }

            private void HidePopup()
            {
                this.isInnerCallHide = true;
                this.popup.Hide();
                this.isInnerCallHide = false;
            }

            private void ShowPopup()
            {
                this.popup.Width = this.Bounds.Width;
                this.popup.Height = 250;
                this.popup.ShowPopup(this.RectangleToScreen(this.ControlBoundingRectangle));
            }

            private void SwitchArrowState(bool isPressed)
            {
                this.arrow.Fill.BackColor = isPressed ? ARROW_PRESSED_BG_COLOR :
                    (arrow.IsMouseOver ? ARROW_MOUSE_OVER_BG_COLOR : ARROW_NORMAL_BG_COLOR);
            }

            protected override void OnPropertyChanged(RadPropertyChangedEventArgs e)
            {
                if (e.Property == ContainsFocusProperty)
                {
                    var isFocus = (bool)e.NewValue;
                    this.BorderWidth = isFocus ? FOCUS_BORDER_WIDTH : NORMAL_BORDER_WIDTH;
                }

                base.OnPropertyChanged(e);
            }

            protected override SizeF ArrangeOverride(SizeF finalSize)
            {
                base.ArrangeOverride(finalSize);

                // arrow on right side
                float width = this.arrow.DesiredSize.Width;
                float x = this.RightToLeft ? 0f : (finalSize.Width - width);
                RectangleF finalRect = new RectangleF(x, 0f, width, finalSize.Height);
                this.arrow.Arrange(finalRect);

                return finalSize;
            }

            private void Arrow_RadPropertyChanged(object sender, RadPropertyChangedEventArgs e)
            {
                if (e.Property == RadArrowButtonElement.IsMouseOverProperty)
                {
                    if (this.IsPopupOpen)
                    {
                        return;
                    }

                    var arrow = sender as RadArrowButtonElement;
                    var isMouseOver = (bool)e.NewValue;

                    arrow.Fill.BackColor = isMouseOver ? ARROW_MOUSE_OVER_BG_COLOR : ARROW_NORMAL_BG_COLOR;
                }
            }

        }

        public class PopupForm : RadSizablePopupControl
        {
            private HostTreeView tv;

            public PopupForm(RadItem owner)
                : base(owner)
            {
                this.SizingMode = SizingMode.UpDownAndRightBottom;
                this.VerticalAlignmentCorrectionMode = AlignmentCorrectionMode.SnapToOuterEdges;
            }

            public RadTreeView TreeView
            {
                get
                {
                    return this.tv.TreeView;
                }
            }

            protected override void CreateChildItems(RadElement parent)
            {
                base.CreateChildItems(parent);

                this.tv = new HostTreeView();
                this.tv.TreeView.Focusable = false;
                this.tv.TreeView.CheckBoxes = true;
                this.SizingGripDockLayout.Children.Add(tv);
            }


            public override bool OnMouseWheel(Control target, int delta)
            {
                if (delta < 0)
                {
                    this.tv.TreeView.VScrollBar.PerformSmallIncrement(1);
                }
                else
                {
                    this.tv.TreeView.VScrollBar.PerformSmallDecrement(1);
                }

                return true;
            }
        }

        public class HostTreeView : Telerik.WinControls.RadHostItem
        {
            public HostTreeView()
                : base(new RadTreeView())
            {

            }

            public RadTreeView TreeView
            {
                get
                {
                    return this.HostedControl as RadTreeView;
                }
            }
        }
    }

  最後說明一點吧,這次封裝對於我自己來說還有一個不滿意的地方,那就是選擇一些項目以後,界面上不顯示已經選擇的項,希望有人能夠完善一下,給出改造後的代碼。

版權聲明:本文為博主原創文章,若要轉載,請註明原處。
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C++ Break 和 Continue break 語句還可以用來跳出迴圈。 在以下示例中,當 i 等於 4 時跳出迴圈: for (int i = 0; i < 10; i++) { if (i == 4) { break; } cout << i << "\n"; } C++ Continue ...
  • C++ 20 的 std::format 是一個很神奇、很實用的工具,最神奇的地方在於它能在編譯期檢查字元串的格式是否正確,而且不需要什麼特殊的使用方法,只需要像使用普通函數那樣傳參即可。 #include <format> int a = 1; std::string s1 = std::form ...
  • 官網:Vue Router | Vue.js 的官方路由 (vuejs.org) 安裝命令:npm install vue-router@4 1.添加兩個頁面\vuedemo\src\views\index.vue、\vuedemo\src\views\content.vue 2.添加\vuedem ...
  • 首先插件添加:Live Server、Vue - Official、Vue VSCode Snippets、別名路徑跳轉 官網:Vite | 下一代的前端工具鏈 (vitejs.dev) 1.創建一個文件夾VueApp,運行cmd轉到該目錄下,執行命令:npm create vite@latest ...
  • 1.安裝node.js : Node.js — Run JavaScript Everywhere (nodejs.org) 2.查看安裝版本命令:node -v 、 npm -v, npm是Node.js包管理器, 用來安裝各種庫、框架和工具。 3.查看當前的鏡像源: npm get regist ...
  • 在Avalonia的UI框架中,TemplatedControl是一個核心組件,它提供了一種強大的方式來創建可重用且高度可定製的控制項。 本文將深入探討TemplatedControl的概念、其帶來的優勢以及它在實際開發中的應用場景,並通過一個示例代碼來展示其用法。 什麼是TemplatedContr ...
  • 本文探討了.NET Emit 入門教程的第六部分,聚焦於ILGenerator中的方法調用指令。通過詳細分析 ILGenerator 的使用方法和方法調用指令,讀者可以更深入地瞭解.NET平臺下動態生成代碼的實現機制。通過本文的閱讀,讀者可以更加熟練地使用 ILGenerator 來動態生成高效、靈... ...
  • 自上一個系列文章寫完之後,最近的兩三個月時間,一直在寫WPF相關技術文章,斷斷續續地寫了近二十幾篇文章,為了方便大家閱讀,現將本系列文章分類整理,彙總如下。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...