当前位置: 首页 > news >正文

【Halcon】WPF 自定义Halcon显示控件完整流程与 `OnApplyTemplate` 未触发的根本原因解析!

🛠️WPF 自定义Halcon显示控件完整流程与 OnApplyTemplate 未触发的根本原因解析!


本片文章最后给出自定义alcon显示控件源码,可以实现图片绑定!


WPF 中封装控件是非常常见的需求,而“自定义控件”是一种高级的控件复用方式。很多人在第一次尝试自定义控件时会遇到一个常见问题:

控件已经显示到界面了,但 OnApplyTemplate() 却从未被调用!

本文将带你完整梳理 WPF 自定义控件的定义流程,并重点分析 OnApplyTemplate() 没有触发的真正原因(并不是大家常说的“忘记设置 DefaultStyleKey”!),最后解释为什么必须在 App.xaml 中引入样式资源


🧱 什么是 WPF 自定义控件?

WPF 中有三种控件封装方式:

封装方式特点
UserControl最简单,直接嵌套已有控件组合
CustomControl(继承自 Control推荐方式,支持样式模板、主题切换
TemplatedControl(高级控件)Control 基础上进一步抽象和通用性封装

本文关注的是 自定义控件(即继承自 Control 的控件),其优势包括:

  • 可复用性强
  • 样式外置,界面逻辑和结构分离
  • 支持模板定制和视觉状态管理

✍️ 自定义控件的完整定义流程

1️⃣ 创建控件类(继承 Control

public class ImageView : Control
{static ImageView(){DefaultStyleKeyProperty.OverrideMetadata(typeof(ImageView),new FrameworkPropertyMetadata(typeof(ImageView)));}public override void OnApplyTemplate(){base.OnApplyTemplate();var part = GetTemplateChild("PART_Content") as Border;// 可访问模板内部元素}
}

OnApplyTemplate() 是你获取模板中元素的最佳时机。


2️⃣ 添加样式模板(例如 Views/ImageView.xaml

<ResourceDictionaryxmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:sw="clr-namespace:MyControlLib.Views"><Style TargetType="sw:ImageView"><Setter Property="Template"><Setter.Value><ControlTemplate TargetType="sw:ImageView"><Border x:Name="PART_Content" Background="LightGray"><TextBlock Text="模板加载成功" /></Border></ControlTemplate></Setter.Value></Setter></Style>
</ResourceDictionary>

控件模板中通过 PART_ 前缀命名可供控件类通过 GetTemplateChild 获取。


3️⃣ ✅ 最重要的一步:在 App.xaml 引入样式资源!

<Application.Resources><ResourceDictionary><ResourceDictionary.MergedDictionaries><ResourceDictionary Source="Views/ImageView.xaml" /></ResourceDictionary.MergedDictionaries></ResourceDictionary>
</Application.Resources>

OnApplyTemplate() 没有触发的原因!

唯一的原因是

样式资源没有添加到 App.xaml,导致 WPF 没有为控件应用样式!


❓为什么样式必须添加到 App.xaml?

WPF 控件模板来自于资源系统,而不是控件自己:

  • WPF 会在控件创建后,到资源系统中查找匹配该控件类型的样式
  • 如果找不到对应的样式(例如你没有在 App.xaml 添加),控件就不会被套用样式
  • 没有样式 => 没有模板 => 不调用 OnApplyTemplate()

⚠️ 也就是说:
不添加样式 = 没有模板 = OnApplyTemplate() 永远不会执行!


🧪 如何验证样式是否加载成功?

你可以在模板中放一个明显控件,比如:

<TextBlock Text="模板已应用!" Foreground="Red" />

如果程序运行后能看到这个控件,那就说明模板加载成功,OnApplyTemplate() 也会被调用。


✅ 小结

步骤是否必须
继承 Control✅ 是
设置 DefaultStyleKey✅ 是
定义样式模板✅ 是
将样式引入 App.xaml是!必须!
实现 OnApplyTemplate()可选,但用于访问模板子元素很常见

📎 最后提醒

如果你写的样式是在控件类库项目中,而不是主程序项目,那么:

🔗 引用样式路径应使用 Pack URI 方式:

<ResourceDictionary Source="/MyControlLib;component/Views/ImageView.xaml" />

⚔️ OnApplyTemplate() 与 Loaded 事件的区别

特性OnApplyTemplate()Loaded
调用时机模板刚刚被应用控件已加载进可视树
控件类型仅适用于自定义控件(Control所有控件
是否依赖模板✅ 依赖 ControlTemplate❌ 不依赖
常用于获取模板中的子元素访问可视化控件属性、执行初始化逻辑
是否能重复调用✔️ 可能多次(重设样式)❌ 通常仅一次(除非卸载重载)

Halcon自定义显示控件源码

自定义样式

<ResourceDictionaryxmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:h="clr-namespace:HalconDotNet;assembly=halcondotnet"xmlns:v="clr-namespace:ROIWindow.Views"><Style TargetType="v:ImageView"><Setter Property="Template"><Setter.Value><ControlTemplate TargetType="v:ImageView"><Grid><h:HSmartWindowControlWPFx:Name="PART_hSmart"HDoubleClickToFitContent="True"HDrawingObjectsModifier="None"HKeepAspectRatio="True"HMoveContent="False"HZoomContent="WheelForwardZoomsIn" /></Grid></ControlTemplate></Setter.Value></Setter></Style>
</ResourceDictionary>

自定义控件代码

using HalconDotNet;
using System;
using System.Collections.Generic;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;namespace ROIWindow.Views;public class ImageView : Control
{private HWindow window;private HSmartWindowControlWPF hSmart;static ImageView(){DefaultStyleKeyProperty.OverrideMetadata(typeof(ImageView), new FrameworkPropertyMetadata(typeof(ImageView)));}public HObject Image{get { return (HObject)GetValue(ImageProperty); }set { SetValue(ImageProperty, value); }}public static readonly DependencyProperty ImageProperty =DependencyProperty.Register("Image", typeof(HObject), typeof(ImageView), new PropertyMetadata(ImageChangedCallBack));public static void ImageChangedCallBack(DependencyObject d, DependencyPropertyChangedEventArgs e){if (d is ImageView view)view.Display();}public void Display(){if (window == null) return;if (Image != null && Image.IsInitialized()){window.ClearWindow();window.DispObj(Image);}}public override void OnApplyTemplate(){hSmart = (HSmartWindowControlWPF)GetTemplateChild("PART_hSmart");hSmart.Loaded += Hsmart_Loaded;base.OnApplyTemplate();}private void Hsmart_Loaded(object sender, RoutedEventArgs e){window = hSmart.HalconWindow;}
}

使用方法

<v:ImageView Image="{Binding Image}" />

最后别忘记在,App.xaml, 添加样式!!!!

<ResourceDictionary Source="pack://application:,,,/CameraXXXXX;component/Controls/ImageView.xaml" />

✍️ 作者:code_bean
📅 发布于:2025年7月
🔗 原创文章,转载请注明出处!


http://www.dtcms.com/a/267301.html

相关文章:

  • Web3 Study Log 003
  • 蓝牙墨水屏上位机学习(3)
  • Java 与 Vue 全栈开发:“一课一得“ 学习笔记系统实战
  • OneCode图表配置速查手册
  • CMake是什么
  • NV183NV185美光固态闪存NV196NV201
  • 供应链管理-采购管理:国际贸易及支付领域中常见的支持方式
  • FLUX.1-Kontext 高效训练 LoRA:释放大语言模型定制化潜能的完整指南
  • 软件版本FCCU(故障采集与控制单元)设计
  • 如何选择不会降低网站速度的WordPress主题
  • 动手实践OpenHands系列学习笔记11:现代开发流程
  • C#指针:解锁内存操作的底层密码
  • DVWA靶场通关笔记-验证码绕过reCAPTCHA(Medium级别)
  • 网安系列【6】之[特殊字符] SQL注入揭秘:从入门到防御实战指南
  • cloudflare配合github搭建免费开源影视LibreTV一个独享视频网站 详细教程
  • React Native 亲切的组件们(函数式组件/class组件)和陌生的样式
  • 百度开源文心一言4.5:论文解读和使用入门
  • 闲庭信步使用SV搭建图像测试平台:第三十二课——系列结篇语
  • 【学习笔记】MySQL技术内幕InnoDB存储引擎——第5章 索引与算法
  • MySQL(118)如何使用SSL进行加密连接?
  • 前端进阶之路-从传统前端到VUE-JS(第三期-VUE-JS配套UI组件的选择)(Element Plus的构建)
  • vscode remote-ssh 拓展免密访问 linux虚拟机
  • 二分查找,乘法口诀表,判断闰年,判断素数,使用函数实现数组操作
  • CSS02:四种CSS导入方式
  • 动手实践OpenHands系列学习笔记7:前端界面设计
  • Flyway 介绍以及与 Spring Boot 集成指南
  • CppCon 2018 学习:Surprises In Object Lifetime
  • Linux systemd 服务启动失败Main process exited, code=exited, status=203/EXEC
  • xformers--Transformer优化加速器使用
  • 暑假算法日记第一天