在Avalonia的UI框架中,TemplatedControl是一个核心组件,它提供了一种强大的方式来创建可重用且高度可定制的控件。

本文将深入探讨TemplatedControl的概念、其带来的优势以及它在实际开发中的应用场景,并通过一个示例代码来展示其用法。

什么是TemplatedControl

TemplatedControl是Avalonia中一个特殊的控件类型,它允许开发者定义控件的模板结构。

这个模板可以包含其他控件、布局、数据绑定等,从而定义控件的外观和行为。

通过将控件的逻辑和外观分离,TemplatedControl提供了一种更加灵活和可维护的方式来创建控件。

在TemplatedControl中,开发者可以定义一些模板绑定点,这些绑定点允许在实例化控件时,将特定的子控件或数据绑定到模板中的对应位置。这

种机制使得控件具有极大的灵活性,可以适应各种不同的使用场景。

TemplatedControl的优势

  1. 高度可定制:TemplatedControl允许开发者通过修改模板来定制控件的外观和行为,从而满足不同的设计需求。

  2. 逻辑与外观分离:通过将控件的逻辑和外观分离,TemplatedControl使得代码更加清晰、易于维护。开发者可以专注于实现控件的功能逻辑,而不需要关心其外观的呈现。

  3. 提高复用性:通过定义通用的TemplatedControl,并在不同的地方使用不同的模板来实例化它,可以大大提高代码的复用性,减少重复劳动。

  4. 易于扩展:TemplatedControl的设计使得它很容易进行扩展。开发者可以继承现有的TemplatedControl并添加自定义的逻辑和模板,从而创建出具有特定功能的控件。

TemplatedControl的应用场景

TemplatedControl在Avalonia UI开发中有着广泛的应用场景。以下是一些常见的应用场景:

  1. 自定义控件:开发者可以使用TemplatedControl来创建具有独特外观和行为的自定义控件,如自定义按钮、自定义列表框等。

  2. 数据展示控件:对于需要展示数据的场景,如列表、表格、树形控件等,TemplatedControl可以提供一个灵活的模板来定义数据的展示方式。

  3. 主题和样式:通过修改TemplatedControl的模板,可以轻松实现应用程序的主题切换和样式定制。

示例代码

下面是一个简单的TemplatedControl示例,展示如何创建一个自定义的控件:

首先,我们定义模板让其包含一个Button和ContentPresenter。

其中Button使用TemplateBinding绑定Content属性。ContentPresenter展示调用时的子控件。

TemplatedControl1.axaml

<Styles xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:controls="using:AvaloniaApplication1">
<Design.PreviewWith>
<controls:TemplatedControl1 />
</Design.PreviewWith> <Style Selector="controls|TemplatedControl1">
<!-- Set Defaults -->
<Setter Property="Template">
<ControlTemplate>
<StackPanel>
<Button Name="PART_Button" Content="{TemplateBinding Content}" />
<ContentPresenter ContentTemplate="{TemplateBinding ContentTemplate}" Name="contentPresenter" HorizontalAlignment="Stretch" VerticalAlignment="Stretch"/>
</StackPanel>
</ControlTemplate>
</Setter>
</Style>
</Styles>

然后,在C#代码中实现类,需要定义Button的Content属性,点击事件,和ContentPresenter的子内容

TemplatedControl1.axaml.cs

using Avalonia;
using Avalonia.Controls;
using Avalonia.Controls.Presenters;
using Avalonia.Controls.Primitives;
using Avalonia.Controls.Templates;
using Avalonia.Interactivity;
using Avalonia.Markup.Xaml.Templates;
using Avalonia.Metadata;
using System;
using System.Linq; namespace AvaloniaApplication1
{
public class TemplatedControl1 : TemplatedControl
{
public static readonly StyledProperty<string> ContentProperty =
AvaloniaProperty.Register<TemplatedControl1, string>(nameof(Content)); public string Content
{
get { return GetValue(ContentProperty); }
set { SetValue(ContentProperty, value); }
} public static readonly RoutedEvent<RoutedEventArgs> ClickEvent =
RoutedEvent.Register<TemplatedControl1, RoutedEventArgs>(nameof(Click), RoutingStrategies.Bubble); public event EventHandler<RoutedEventArgs> Click
{
add => AddHandler(ClickEvent, value);
remove => RemoveHandler(ClickEvent, value);
} private void OnClick(object sender, RoutedEventArgs e)
{
RaiseEvent(new RoutedEventArgs(ClickEvent));
} public static readonly StyledProperty<DataTemplate> ContentTemplateProperty =
AvaloniaProperty.Register<TemplatedControl1, DataTemplate>(nameof(ContentTemplate)); [Content]
public IDataTemplate ContentTemplate
{
get => GetValue(ContentTemplateProperty);
set => SetValue(ContentTemplateProperty, value);
} public override void EndInit()
{
base.EndInit();
ApplyTemplate(); var childs = this.GetTemplateChildren().ToList();
var button = childs.FirstOrDefault(e => e.Name == "PART_Button");
if (button != null)
{
((Button)button).Click += OnClick;
} // Apply the content template to the ContentPresenter
//var contentPresenter = childs.FirstOrDefault(e => e.Name == "contentPresenter");
//((ContentPresenter)contentPresenter).ContentTemplate = ContentTemplate;
}
}
}

并在App.axaml中使用StyleInclude声明此控件

App.axaml

<Application xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
x:Class="AvaloniaApplication1.App"
xmlns:local="using:AvaloniaApplication1"
RequestedThemeVariant="Default">
<!-- "Default" ThemeVariant follows system theme variant. "Dark" or "Light" are other available options. --> <Application.DataTemplates>
<local:ViewLocator/>
</Application.DataTemplates> <Application.Styles>
<FluentTheme />
<StyleInclude Source="CControls/TemplatedControl1.axaml"/>
</Application.Styles>
</Application>

最后在MainWindow.axaml中使用此控件,并为此控件传递Content,Click属性,和DataTemplate的子内容

MainWindow.axaml

<Window xmlns="https://github.com/avaloniaui"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:vm="using:AvaloniaApplication1.ViewModels"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local="clr-namespace:AvaloniaApplication1"
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
x:Class="AvaloniaApplication1.Views.MainWindow"
x:DataType="vm:MainWindowViewModel"
Icon="/Assets/avalonia-logo.ico"
Title="AvaloniaApplication1"> <Design.DataContext>
<vm:MainWindowViewModel/>
</Design.DataContext> <local:TemplatedControl1 Content="test control" Click="HandleButtonClick">
<DataTemplate>
<Button Content="Click Me"/>
</DataTemplate>
</local:TemplatedControl1>
</Window>

MainWindow.axaml.cs中定义HandleButtonClick

MainWindow.axaml.cs

public void HandleButtonClick(object sender, RoutedEventArgs e)
{
Debug.WriteLine("click");
}

运行即可查看到效果

Avalonia的模板控件(Templated Controls)的更多相关文章

  1. ASP.NET自定义控件组件开发 第五章 模板控件开发

    原文:ASP.NET自定义控件组件开发 第五章 模板控件开发 第五章 模板控件开发 系列文章链接: ASP.NET自定义控件组件开发 第一章 待续 ASP.NET自定义控件组件开发 第一章 第二篇 接 ...

  2. asp.net模板控件示例

    原文:asp.net模板控件示例 模板控件允许将控件数据与其表示形式相分离,模板化控件不提供用户界面. 编写它则是为了实现一个命名容器以及包含属性和方法可由宿主页访问的类,MSDN是这样解释的. 下面 ...

  3. 【笔记】WPF之模板控件应用

    最近在捣鼓WPF的动画,想自定义一个控件模型来实现动画. 目标功能是这样:在WPF项目文件中创建一个自定义用户控件模型,该模型最外层是一个Grid,Grid布局为3行1列,第一列是一个图片按钮,第二列 ...

  4. WPF DataGrid控件中某一列根据另一个文本列的值显示相应的模板控件

    之前做项目的时候需要实现这样一个功能.WPF DataGrid有两列,一列为"更新状态”列,一列为"值"列,如果"更新状态"列的值为“固定值更新”,则 ...

  5. WPF--Blend制作Button控件模板--问题补充

    补充记录Button控件模板 控件模板制作过程中出现下图问题:动画对象不能用于动画属性"Fill” 并且这类问题Blend4中包括VS2010中仍然可以运行,但是只有VS2010中会报错:如 ...

  6. 《Programming WPF》翻译 第5章 7.控件模板

    原文:<Programming WPF>翻译 第5章 7.控件模板 如果仔细的看我们当前的TTT游戏,会发现Button对象并没有完全为我们工作.哪些TTT面板有内圆角? 图5-14 这里 ...

  7. Expression Blend实例中文教程(13) - 控件模板快速入门ControlTemplates

    上篇,介绍了控件样式(Style)和模板(Template)的基础概念,并且演示了使用Blend设计控件样式.本篇将继续介绍使用Blend设计自定义控件模板 - ControlTemplate.Con ...

  8. Xamarin XAML语言教程构建ControlTemplate控件模板

    Xamarin XAML语言教程构建ControlTemplate控件模板 控件模板ControlTemplate ControlTemplate是从Xamarin.Forms 2.1.0开始被引入的 ...

  9. 【Win10】【Win2D】实现控件阴影效果

    学过 WPF 的都知道,在 WPF 中,为控件添加一个阴影效果是相当容易的. <Border Width="100" Height="100" Backg ...

  10. 初识Windows窗体(包括各种控件,属性,方法)

    什么是Wind ows窗体? 顾名思义,win dows窗体就是将一些所必须的信息通过窗体的形式展示给客户看.例如:我们经常玩的QQ登陆界面,微信登陆界面,等等,都是以窗体的形式将信息展示给我们看的. ...

随机推荐

  1. 在本地运行 LLMs 的 6 种方法

    商业人工智能和大型语言模型 (LLM) 有一个很大的缺点:隐私.在处理敏感数据或专有数据时,我们无法从这些工具中获益.因此,我们需要了解如何在本地运行私人 LLM.开源模型提供了一种解决方案,但它们也 ...

  2. 逆向实战32——某东最新h5st4.4算法分析

    前言 本文章中所有内容仅供学习交流,抓包内容.敏感网址.数据接口均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权,请联系我立即删除! 目标网站 aHR0cHM6 ...

  3. linux复习基础命令

    Linux基础命令 学习目标 了解vm备份的两种方式 了解快照和克隆的区别 了解linux系统文件 掌握基础命令 指定vmtools的作用 1 vm两种备份方式 为什么要进行备份 保证centos操作 ...

  4. uni-app实现公众号登陆实现

    公众号实现登陆流程思路: 1. 创建一个页面用于登陆,页面上需要有输入账号和密码的表单,以及登陆按钮.2. 在登陆按钮的点击事件中,调用后端接口进行账号密码校验.如果校验通过,则将后端返回的用户信息保 ...

  5. 获取input[type="checkbox"]:checked 所在tr中特定元素

    1.要求如下  2.html源码 <div class="btn"> <button type="button" onclick=" ...

  6. react build 后,打包后自动将index.html copy 404.html - create-react-app 创建的项目

    起因:build上传gitee,启用路由需要404.html自动跳转 当前环境 create-react-app 搭建的架子 解决方案 由于默认的时候把build.js打包,无法查看,只好eject ...

  7. 2.4G无线音频传输方案市场调研分析

    基本概念   所谓的2.4G音频传输,不是使用标准的wifi和蓝牙协议传输,而是使用私有的2.4G通信协议来实现点对点,一对多和多对一的音频传输.2.4G私有协议音频传输,有这低延时,距离远,声音高清 ...

  8. Markdown表情参考

    emoji-github 文章内容来源 https://github.com/hoangdqvn/emoji-github/blob/master/README.md ️ Emoji-GIT Peop ...

  9. struts1之global-forwards

    当你的某个转发要经常用,并且要携带某些数据(request)的时候用全局转发,也就是global-forwards,例如我们在分页的时候,或者得到数据列表的时候.. ForwardAction呢,是为 ...

  10. jquery之获取某个元素上的事件

    jquery的给元素绑定的事件可以用data方法取出来 通过$(element).data("events")来获取 // 比如给一个button绑定两个click事件 $(&qu ...