我的编程空间,编程开发者的网络收藏夹
学习永远不晚

怎么在c#项目中自定义MarkupExtension

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

怎么在c#项目中自定义MarkupExtension

这篇文章给大家介绍怎么在c#项目中自定义MarkupExtension,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

Markup Extension,顾名思义,就是对xaml的扩展,在XAML中,规定如果属性以{}开始及结束,就是Markup Extension,Markup Extension指的是继承于MarkupExtension的类,首先我们通过一张图来看看WPF中有哪些已知的Markup Extension。

怎么在c#项目中自定义MarkupExtension

  看了这张图片之后是不是对这个MarkupExtension有一个常规的认识,你会发现这个在WPF中实在是太重要了,通过这个MarkupExtension我们能够实现绑定、资源等等一系列的操作,在介绍完这个之后,我们来看看,这个抽象的MarkupExtension基类到底是什么?里面包含些什么?怎么去使用它?

#region 程序集 WindowsBase.dll, v3.0.0.0// C:\Program Files\Reference Assemblies\Microsoft\Framework\v3.0\WindowsBase.dll#endregionusing System;namespace System.Windows.Markup{ // 摘要: //  为所有 XAML 标记扩展提供基类。 public abstract class MarkupExtension {  // 摘要:  //  初始化从 System.Windows.Markup.MarkupExtension 派生的类的新实例。  protected MarkupExtension();  // 摘要:  //  在派生类中实现时,返回一个对象,此对象被设置为此标记扩展的目标属性的值。  //  // 参数:  // serviceProvider:  //  可以为标记扩展提供服务的对象。  //  // 返回结果:  //  将在扩展应用到的属性上设置的对象值。  public abstract object ProvideValue(IServiceProvider serviceProvider); }}

   其实看看里面的内容,仅仅提供了一个抽象的方法ProvideValue,我们在继承这个抽象类后需要去重载这个抽象方法,然后来实现自己的逻辑。

  在对整个MarkupExtension介绍之后,我们可以对它进行一个总结,那就是:

  XAML标记扩展语法格式:

  <元素对象 对象属性=”{扩展标记 扩展标记属性 = 扩展属性值}” />
      这个是不是很熟悉,如果还是不够直观的话,我们可以通过代码来进行说明:      

<TextBox Text=”{Binding Path=ProductName}”/>

  再来一个复杂一些的例子吧

<Popup IsOpen="{Binding Path=IsSubmenuOpen, RelativeSource={RelativeSource TemplatedParent}}" Placement="Right" x:Name="SubMenuPopup" Focusable="false" AllowsTransparency="true" PopupAnimation="{DynamicResource {x:Static SystemParameters.MenuPopupAnimationKey}}"/>

  类似的这种我们在WPF中见到的是在是太多了,那么既然基类是一个抽象方法那么我们是不是可以通过重载这种方式来写自己的MarkupExtension呢?这个当然是可以的,我们可以通过下面的几个例子来进行相应的说明。

  示例1:通过MarkupExtension绑定MenuItem的Icon属性。

  我们知道,MenuItem的Icon属性可以通过下面的方式进行设置:

<MenuItem Header="New">  <MenuItem.Icon>    <Image Source="data/cat.png"/>  </MenuItem.Icon></MenuItem>

  这个是MSDN介绍的常规方式,在这里我们可以通过三种不同的方式来达到这个目的,具体来看看是怎么实现的吧?

<Menu Grid.Column="0">           <MenuItem Header="文本">               <MenuItem Header="重做">                   <MenuItem.Icon>                       <Image Stretch="Uniform" Source="{extension:ImageBinding Redo}"></Image>                   </MenuItem.Icon>               </MenuItem>               <MenuItem Header="撤销">                   <MenuItem.Icon>                       <Image Stretch="Uniform" Source="{extension:ImageBinding Undo}"></Image>                   </MenuItem.Icon>               </MenuItem>               <MenuItem Header="保存所有">                   <MenuItem.Icon>                       <Image Stretch="Uniform" Source="{Binding SaveAll,Converter={StaticResource SourceConverter}}"></Image>                   </MenuItem.Icon>               </MenuItem>               <MenuItem Header="测试">                   <MenuItem.Icon>                       <Image Stretch="Uniform" Source="Resources/Images/Redo.png"></Image>                   </MenuItem.Icon>               </MenuItem>           </MenuItem>           <MenuItem Header="编辑"></MenuItem>           <MenuItem Header="视图"></MenuItem>           <MenuItem Header="插件"></MenuItem>       </Menu>

  第一种方式就是我们今天重点介绍的通过继承MarkupExtension来实现同样的效果,我们来具体分析一下这个ImageBinding

public class ImageBindingExtension : System.Windows.Markup.MarkupExtension   {       public ImageBindingExtension(string path)           : this()       {           Path = path;       }        public ImageBindingExtension()       {       }        [ConstructorArgument("path")]       public string Path       {           get;           set;       }         public override object ProvideValue(IServiceProvider serviceProvider)       {           IProvideValueTarget target = serviceProvider.GetService(typeof(IProvideValueTarget)) as IProvideValueTarget;            if (target.TargetObject is Setter)           {               return new Binding(Path) { Converter = ImgaeSourceConverter.Default };           }           else           {               Binding binding = new Binding(Path) { Converter = ImgaeSourceConverter.Default };               return binding.ProvideValue(serviceProvider);           }                  }   }

  这里面我们定义的Path属性就是绑定到ViewModel中的一个特定的属性,这里我们通过重写ProvideValue方法,最终调用BindingBase的ProvideValue返回ImageSource对象,这里是通过一个转换器来实现源属性(字符串)到目标属性ImageSource的转换的,我们会发现,其实这种方法和直接绑定并设置转换器其实效果是一样的,只不过第一种方式更为直观,将所有的转换过程都放在了重写ProvideValue函数的过程中了,这个读者在后面可以对照demo去认真思考然后加以总结。

  示例2:通过MarkupExtension绑定到ListBox的ItemsSource属性

  这个稍微复杂一些,我们在Reflection这个MarkupExtension中加入了一些自定义的属性,这些属性能够控制后面返回的数据源的最终内容,其实这个也是非常好理解的,我们在定义RelativeSource这个MarkupExtension的时候,也是通过定义Mode、AncestorType、AncestorLevel等属性组合起来最终实现在视觉树上找到最终的元素。在代码里面也不复杂主要是通过反射来获取Button的属性、方法、事件、字段等等,这个具体的实现过程可以参考后面的代码。

public class ReflectionExtension : System.Windows.Markup.MarkupExtension    {        public Type CurrentType { get; set; }        public bool IncludeMethods { get; set; }        public bool IncludeFields { get; set; }        public bool IncludeEvents { get; set; }         public ReflectionExtension(Type currentType)        {            this.CurrentType = currentType;        }         public override object ProvideValue(IServiceProvider serviceProvider)        {            if (this.CurrentType == null)            {                throw new ArgumentException("Type argument is not specified");            }             ObservableCollection<string> collection = new ObservableCollection<string>();            foreach (PropertyInfo p in this.CurrentType.GetProperties())            {                collection.Add(string.Format("属性 : {0}", p.Name));            }             if (this.IncludeMethods)            {                foreach (MethodInfo m in this.CurrentType.GetMethods())                {                    collection.Add(string.Format("方法 : {0} with {1} argument(s)", m.Name, m.GetParameters().Count()));                }            }            if (this.IncludeFields)            {                foreach (FieldInfo f in this.CurrentType.GetFields())                {                    collection.Add(string.Format("字段 : {0}", f.Name));                }            }            if (this.IncludeEvents)            {                foreach (EventInfo e in this.CurrentType.GetEvents())                {                    collection.Add(string.Format("事件 : {0}", e.Name));                }            }            return collection;        }     }

关于怎么在c#项目中自定义MarkupExtension就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

怎么在c#项目中自定义MarkupExtension

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

怎么在c#项目中自定义MarkupExtension

这篇文章给大家介绍怎么在c#项目中自定义MarkupExtension,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Markup Extension,顾名思义,就是对xaml的扩展,在XAML中,规定如果属性以{}开始
2023-06-06

怎么在C++项目中利用priority_queue自定义排序

这篇文章给大家介绍怎么在C++项目中利用priority_queue自定义排序,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先,无论 priority_queue 中存储的是基础数据类型(int、double 等),
2023-06-06

JAVA在项目中怎么自定义异常

这篇文章主要讲解了“JAVA在项目中怎么自定义异常”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JAVA在项目中怎么自定义异常”吧!JAVA项目中自定义异常1.数据返回处理类@Datapub
2023-06-20

怎么在java项目中对注解自定义

这期内容当中小编将会给大家带来有关怎么在java项目中对注解自定义,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。java 自定义注解的实例详解Java的Annotation是在5.0版本之后引入的,可以
2023-05-31

怎么在java项目中自定义线程池

怎么在java项目中自定义线程池?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。使用线程池时,可以使用 newCachedThreadPool()或者 newFi
2023-05-31

怎么在VS2019中自定义一个项目模板

怎么在VS2019中自定义一个项目模板?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、自定义项目模板创建 本次我们来创建一个实现了基础功能的webapi模板 创建自定义的项目
2023-06-14

在Android项目中怎么自定义阴影效果

在Android项目中怎么自定义阴影效果?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。Android 自定义阴影效果详解及实例Android5.X中,Google为其增加了
2023-05-31

如何在C#项目中使用WPF自定义按钮

今天就跟大家聊聊有关如何在C#项目中使用WPF自定义按钮,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。实现效果使用图片做按钮背景;自定义鼠标进入时效果;自定义按压效果;自定义禁用效果
2023-06-06

如何在Android项目中自定义title

这篇文章给大家介绍如何在Android项目中自定义title,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先编写title的布局文件,title.xml:
2023-05-31

如何在java 8项目中自定义collector

本篇文章给大家分享的是有关如何在java 8项目中自定义collector,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。需求:将 一个容器List 按照一定的字段进
2023-05-31

在iview+vue项目中怎么使用自定义icon图标

这篇文章主要讲解了“在iview+vue项目中怎么使用自定义icon图标”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“在iview+vue项目中怎么使用自定义icon图标”吧!1. UI设计
2023-06-29

如何在Android项目中自定义view函数

如何在Android项目中自定义view函数 ?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。首先 往Canvas上面draw需要一个Paint。 Paint p = new
2023-05-31

Visual Studio怎么自定义项目模版

这篇“Visual Studio怎么自定义项目模版”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Visual Studio怎
2023-07-02

怎么在C语言中自定义类型

本篇文章为大家展示了怎么在C语言中自定义类型,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、初始结构体结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量。下面举一个例子:s
2023-06-08

C++中怎么自定义类

这篇文章将为大家详细讲解有关C++中怎么自定义类,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。/////////////TestClass.h/////////////////// cla
2023-06-17

怎么在java项目中定义一个数组

这篇文章给大家介绍怎么在java项目中定义一个数组,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;3. 客户端开发;4. 网页开发;5
2023-06-14

在spring-boot项目中如何实现自定义filter

在spring-boot项目中如何实现自定义filter?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。传统的javaEE增加Filter是在web.xml中配置
2023-05-31

在Spring Boot项目中如何实现自定义PropertySourceLoader

今天就跟大家聊聊有关在Spring Boot项目中如何实现自定义PropertySourceLoader,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。SpringBoot 的配置文件
2023-05-31

如何在Android项目中自定义侧滑菜单栏

本篇文章给大家分享的是有关如何在Android项目中自定义侧滑菜单栏,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。自定义侧滑菜单的简单实现不少APP中都有这种侧滑菜单,例如QQ
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录