主頁 > .NET開發 > WPF進階技巧和實戰07--自定義元素02

WPF進階技巧和實戰07--自定義元素02

2021-08-31 16:09:56 .NET開發

在01節中,研究了如何開發自定義控制元件,下節開始考慮更特殊的選擇:派生自定義面板以及構建自定義繪圖

創建自定義面板

創建自定義面板是一種比較常見的自定義控制元件開發子集,面板可以駐留一個或多個子元素,并且實作了特定的布局邏輯以恰當地安排子元素,常見的基本型別的面板:StackPanel、DockPanel、WrapPanel、Canvas,Grid,TabPanel,ToolBarPverflowPanel,VirtualizingPanel,

兩步布局程序

每個面板都有相同的功能:負責改變子元素尺寸和安排子元素的兩步布局程序,第一個階段是測量階段,這個階段決定其子元素希望具有多大的尺寸,第二個階段是排列階段,這個階段為每個控制元件指定邊界,

可以通過重寫函式MeasureOverride()和ArrangeOverride(),來添加自己的邏輯,

  1. MeasureOverride()方法

這個方法決定了每個子元素希望多大的空間,會遍歷子元素集合,并呼叫每個子元素的Measure()發放來控制子元素的最大可用空間,最后,面板回傳所有子元素所需的空間,

public static readonly DependencyProperty DiameterProperty = DependencyProperty.Register(
            "Diameter", typeof(double), typeof(FixLampCirclePanel), new FrameworkPropertyMetadata(170.0, FrameworkPropertyMetadataOptions.AffectsMeasure));

public double Diameter
{
   get => (double)GetValue(DiameterProperty);
   set => SetValue(DiameterProperty, value);
}

protected override Size MeasureOverride(Size availableSize)
{
    if (Children.Count == 0) return new Size(Diameter, Diameter);

    var newSize = new Size(Diameter, Diameter);

    foreach (UIElement element in Children)
    {
        element.Measure(newSize);
    }

    return newSize;
}

元素呼叫Measure()方法之后才會渲染自身,后續在子元素執行計算時,才會使用DesiredSize屬性來請求尺寸,

  1. ArrangeOverride()方法

測量完所有尺寸后,就需要排列所有子元素,Arrange()方法來實作這個程序,

public static readonly DependencyProperty KeepVerticalProperty = DependencyProperty.Register(
    "KeepVertical", typeof(bool), typeof(FixLampCirclePanel), new FrameworkPropertyMetadata(false, FrameworkPropertyMetadataOptions.AffectsMeasure));

public bool KeepVertical
{
    get => (bool)GetValue(KeepVerticalProperty);
    set => SetValue(KeepVerticalProperty, value);
}

public static readonly DependencyProperty OffsetAngleProperty = DependencyProperty.Register(
    "OffsetAngle", typeof(double), typeof(FixLampCirclePanel), new FrameworkPropertyMetadata(0.0, FrameworkPropertyMetadataOptions.AffectsMeasure));

public double OffsetAngle
{
    get => (double)GetValue(OffsetAngleProperty);
    set => SetValue(OffsetAngleProperty, value);
}



protected override Size ArrangeOverride(Size finalSize)
{
    if (base.Children.Count == 0) return finalSize;

    //第一個放在中間,第一個移動半徑為0即可,其余的均分布
    var perDeg = 360.0 / (Children.Count - 1);
    var radius = 0.0;
    for (int i = 0; i < Children.Count; i++)
    {
        if (i != 0) radius = Diameter / 2;

        UIElement element = base.Children[i];
        var centerX = element.DesiredSize.Width / 2.0;
        var centerY = element.DesiredSize.Height / 2.0;
        var angle = perDeg * i + OffsetAngle;
        var transform = new RotateTransform
        {
            CenterX = centerX,
            CenterY = centerY,
            Angle = KeepVertical ? 0 : angle
        };
        element.RenderTransform = transform;
        var r = Math.PI * angle / 180.0;
        var x = radius * Math.Cos(r);
        var y = radius * Math.Sin(r);
        var rectX = x + finalSize.Width / 2 - centerX;
        var rectY = y + finalSize.Height / 2 - centerY;
        element.Arrange(new Rect(rectX, rectY, element.DesiredSize.Width, element.DesiredSize.Height));
    }

    return finalSize;
}

Canvas面板的副本

Canvas面板在希望的位置放置子元素,并且為子元素設定他們希望的尺寸,所以不需要計算如何分割可用空間,所以為每個子元素提供無線的空間,同時,回傳值是空的Size物件,所以面板是不請求任何空間,而是由您明確地為Canvas面板指定尺寸,或者將其放置到布局容器中進行拉伸以填充整個容器可用的空間,

protected override Size MeasureOverride(Size constraint)
{
    Size availableSize = new Size(double.PositiveInfinity, double.PositiveInfinity);
    foreach (UIElement internalChild in base.InternalChildren)
    {
        internalChild?.Measure(availableSize);
    }

    return default(Size);
}

ArrangeOverride()方法通過附加屬性(Left,Right,Top,Bottom)來確定每個子元素的位置,

protected override Size ArrangeOverride(Size arrangeSize)
{
    foreach (UIElement internalChild in base.InternalChildren)
    {
        if (internalChild == null)
        {
            continue;
        }

        double x = 0.0;
        double y = 0.0;
        double left = Canvas.GetLeft(internalChild);
        if (!Double.IsNaN(left))
        {
            x = left;
        }
        else
        {
            double right = Canvas.GetRight(internalChild);
            if (!Double.IsNaN(right))
            {
                x = arrangeSize.Width - internalChild.DesiredSize.Width - right;
            }
        }

        double top = Canvas.GetTop(internalChild);
        if (!Double.IsNaN(top))
        {
            y = top;
        }
        else
        {
            double bottom = Canvas.GetBottom(internalChild);
            if (!Double.IsNaN(bottom))
            {
                y = arrangeSize.Height - internalChild.DesiredSize.Height - bottom;
            }
        }

        internalChild.Arrange(new Rect(new Point(x, y), internalChild.DesiredSize));
    }

    return arrangeSize;
}

更好的WrapPanel

在傳統的WrapPanel中添加強制換行的功能,可以通過自定義控制元件來實作,首先要添加強制換行附加屬性,沒有使用常規屬性封裝器封裝這個屬性,因為不在定義他們的同一個類中設定它,而是使用兩個靜態方法,

public static readonly DependencyProperty LineBreakBeforeProperty = 
    DependencyProperty.RegisterAttached("LineBreakBefore", typeof(bool), typeof(WrapBreakPanel), 
        new FrameworkPropertyMetadata() { AffectsArrange = true, AffectsMeasure = true });

public static void SetLineBreakBefore(UIElement element, bool value)
        {
            element.SetValue(LineBreakBeforeProperty, value);
        }
public static bool GetLineBreakBefore(UIElement element)
{
    return (bool)element.GetValue(LineBreakBeforeProperty);
}

自定義繪圖元素

在WPF中,這些類位于元素樹的最底層,通過單獨的文本、形狀、位圖來執行渲染,

OnRender()方法

需要執行自定義渲染,就必須重寫OnRender()方法,該方法繼承自UIElement基類,一些空間使用OnRender()方法繪制可視化細節并在其上疊加其他元素形成組合,Border類是OnRender()方法中繪制邊框,Panel類是在OnRender()方法中繪制背景,兩者都支持子內容,并且這些子內容在自定義的繪圖之上進行渲染,

OnRender()方法接收一個DrawingContext物件,使用這個物件進行繪制操作,OnRender()方法中不能顯示的創建和關閉DrawingContext物件,因為幾個不同的OnRender()方法使用相同的DrawingContext物件,在開始繪制時,WPF會自動創建DrawingContext物件,并且當不再需要時自動關閉該物件,

OnRender()方法實際上并沒有繪制在螢屏上,而是繪制在DrawingContext物件上,然后WPF快取這些資訊,WPF來決定何時需要重新繪制并使用DrawingContext物件創建內容,WPF無縫地管理繪制和重繪的程序,由用戶來定義內容,

自定義繪圖元素

下面的例子通過RadialGradientBrush畫刷繪制陰影背景,中心點跟隨滑鼠移動,

public class CustomDrawnElement : FrameworkElement
{
    public Color BackgroundColor { get => (Color)GetValue(BackgroundColorProperty); set => SetValue(BackgroundColorProperty, value); }
    public static readonly DependencyProperty BackgroundColorProperty =
        DependencyProperty.Register("BackgroundColor", typeof(Color), typeof(CustomDrawnElement),
            new FrameworkPropertyMetadata(Colors.Yellow) { AffectsRender = true });

    protected override void onm ouseMove(MouseEventArgs e)
    {
        base.OnMouseMove(e);
        this.InvalidateVisual();
    }

    protected override void onm ouseLeave(MouseEventArgs e)
    {
        base.OnMouseLeave(e);
        this.InvalidateVisual();
    }



    protected override void OnRender(DrawingContext drawingContext)
    {
        base.OnRender(drawingContext);

        Rect rect = new Rect(0, 0, base.ActualWidth, ActualHeight);
        drawingContext.DrawRectangle(GetForegroundBrush(), null, rect);
    }

    private Brush GetForegroundBrush()
    {
        if (!IsMouseOver)
        {
            return new SolidColorBrush(BackgroundColor);
        }
        else
        {
            RadialGradientBrush brush = new RadialGradientBrush(Colors.White, BackgroundColor);

            Point point = Mouse.GetPosition(this);
            Point newPoint = new Point(point.X / base.ActualWidth, point.Y / base.ActualHeight);

            brush.GradientOrigin = newPoint;
            brush.Center = newPoint;

            return brush;
        }
    }
}

創建自定義元素

在WPF中,切記不要再控制元件中進行自定義繪圖,會破壞WPF無外觀控制元件的原則,一旦使用了繪圖邏輯,就會使得控制元件的可視化外觀不能通過控制元件模板來定制,

更好的方法是設計單獨的繪制自定義內容的元素,然后再控制元件的默認模板內部使用自定義元素,

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/295840.html

標籤:WPF

上一篇:七、從GitHub瀏覽Prism示例代碼的方式入門WPF下的Prism之RegionContext

下一篇:WPF 如何獲取有哪些 VisualBrush 用了某個控制元件

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • C#多執行緒學習(二) 如何操縱一個執行緒

    <a href="https://www.cnblogs.com/x-zhi/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2943582/20220801082530.png" alt="" /></...

    uj5u.com 2023-04-19 09:17:20 more
  • C#多執行緒學習(二) 如何操縱一個執行緒

    C#多執行緒學習(二) 如何操縱一個執行緒 執行緒學習第一篇:C#多執行緒學習(一) 多執行緒的相關概念 下面我們就動手來創建一個執行緒,使用Thread類創建執行緒時,只需提供執行緒入口即可。(執行緒入口使程式知道該讓這個執行緒干什么事) 在C#中,執行緒入口是通過ThreadStart代理(delegate)來提供的 ......

    uj5u.com 2023-04-19 09:16:49 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    <a href="https://www.cnblogs.com/huangxincheng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/214741/20200614104537.png" alt="" /&g...

    uj5u.com 2023-04-18 08:39:04 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    一:背景 1. 講故事 前段時間協助訓練營里的一位朋友分析了一個程式卡死的問題,回過頭來看這個案例比較經典,這篇稍微整理一下供后來者少踩坑吧。 二:WinDbg 分析 1. 為什么會卡死 因為是表單程式,理所當然就是看主執行緒此時正在做什么? 可以用 ~0s ; k 看一下便知。 0:000> k # ......

    uj5u.com 2023-04-18 08:33:10 more
  • SignalR, No Connection with that ID,IIS

    <a href="https://www.cnblogs.com/smartstar/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/u36196.jpg" alt="" /></a>...

    uj5u.com 2023-03-30 17:21:52 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:15:33 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:13:31 more
  • C#遍歷指定檔案夾中所有檔案的3種方法

    <a href="https://www.cnblogs.com/xbhp/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/957602/20230310105611.png" alt="" /></a&...

    uj5u.com 2023-03-27 14:46:55 more
  • C#/VB.NET:如何將PDF轉為PDF/A

    <a href="https://www.cnblogs.com/Carina-baby/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2859233/20220427162558.png" alt="" />...

    uj5u.com 2023-03-27 14:46:35 more
  • 武裝你的WEBAPI-OData聚合查詢

    <a href="https://www.cnblogs.com/podolski/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/616093/20140323000327.png" alt="" /><...

    uj5u.com 2023-03-27 14:46:16 more