【wpf】WPF 自定義控件綁定數據對象的最佳實踐

WPF 自定義控件綁定數據對象的最佳實踐:以 ImageView 為例

在 WPF 中開發自定義控件時,如何優雅地綁定數據對象,是一個經常遇到的問題。最近在實現一個自定義的 ImageView 控件時,我遇到了一個典型場景:

  • 控件內部需要使用第三方控件 HSmartWindowControlWPF 來顯示圖像;
  • 控件需要和業務對象 GraphicInfo 綁定,并且要把自身引用回寫到 GraphicInfo.View
  • 控件還要支持在 ItemList(例如 ListBox)中批量使用。

本文就以這個案例為例,來總結下最佳實踐。在這里插入圖片描述

1. 直接使用 DataContext 的問題

最初的寫法是這樣的:

public override void OnApplyTemplate()
{base.OnApplyTemplate();var hSmart = (HSmartWindowControlWPF)GetTemplateChild("PART_hSmart");hSmart.Loaded += Hsmart_Loaded;hSmart.HMouseMove += HSmart_HMouseMove;if (DataContext is GraphicInfo info){info.View = this; // 子類控件回寫到數據對象}
}

XAML 里:

<local:ImageView DataContext="{Binding MyGraphic}" />

這種方式雖然能用,但有幾個問題:

  • 控件內部和外部公用了同一個 DataContext,如果控件內部還需要 MVVM 綁定,會和外部沖突。
  • ItemList 中使用時不夠直觀,容易出錯。

2. 定義依賴屬性(推薦)

更好的做法是為控件定義一個依賴屬性,例如 Graphic

public class ImageView : Control
{static ImageView(){DefaultStyleKeyProperty.OverrideMetadata(typeof(ImageView),new FrameworkPropertyMetadata(typeof(ImageView)));}// 定義依賴屬性public GraphicInfo Graphic{get => (GraphicInfo)GetValue(GraphicProperty);set => SetValue(GraphicProperty, value);}public static readonly DependencyProperty GraphicProperty =DependencyProperty.Register(nameof(Graphic), typeof(GraphicInfo), typeof(ImageView),new PropertyMetadata(null, OnGraphicChanged));private static void OnGraphicChanged(DependencyObject d, DependencyPropertyChangedEventArgs e){if (d is ImageView view && e.NewValue is GraphicInfo info){info.View = view; // 在綁定時回寫}}public override void OnApplyTemplate(){base.OnApplyTemplate();var hSmart = (HSmartWindowControlWPF)GetTemplateChild("PART_hSmart");if (hSmart != null){hSmart.Loaded += Hsmart_Loaded;hSmart.HMouseMove += HSmart_HMouseMove;}}private void Hsmart_Loaded(object sender, RoutedEventArgs e) { }private void HSmart_HMouseMove(object sender, HMouseEventArgs e) { }
}

這樣,控件就有了一個獨立的 Graphic 屬性,外部綁定時可以很清晰地寫:

<local:ImageView Graphic="{Binding MyGraphic}" />

3. 在 ItemList 中使用

ListBox 中批量展示多個 GraphicInfo 時,就非常自然:

<ListBox ItemsSource="{Binding Graphics}"><ListBox.ItemTemplate><DataTemplate><local:ImageView Graphic="{Binding}" /></DataTemplate></ListBox.ItemTemplate>
</ListBox>

這里的 Binding 就是每個 GraphicInfo,通過依賴屬性綁定到控件,不會和 DataContext 混用。


4. 對比與總結

使用 DataContext 的方式

? 簡單,少寫一個依賴屬性
? 容易和控件內部 MVVM 沖突
? 在 ItemList 中語義不清晰

使用依賴屬性的方式

? 更加清晰,控件的 DataContext 可以獨立使用
? 在 ItemList 中使用更自然
? 可以在依賴屬性的回調里處理邏輯(如回寫 View


最佳實踐

  • 如果控件只是簡單的 UI 展示,內部不需要額外綁定,可以用 DataContext。
  • 如果控件要在 ItemList 中使用,或者內部還需要用自己的 DataContext,推薦使用依賴屬性。

我個人在項目里最終采用了 依賴屬性模式,不僅解耦了數據和視圖,還能方便在列表場景下使用。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/95624.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/95624.shtml
英文地址,請注明出處:http://en.pswp.cn/web/95624.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

[Dify 專欄] 如何通過 Prompt 在 Dify 中模擬 Persona:即便沒有專屬配置,也能讓 AI 扮演角色

在 AI 應用開發中,“Persona(角色扮演)”常被視為塑造 AI 個性與專業邊界的重要手段。然而,許多開發者在使用 Dify 時會疑惑:為什么我在 Chat 應用 / Agent 應用 / Workflow 里都找不到所謂的 Persona 配置項? 答案是:Dify 平臺目前并沒有內建的 Persona 配置入口。角色…

解決雙向循環鏈表中對存儲數據進行奇偶重排輸出問題

1. 概念 對鏈表而言,雙向均可遍歷是最方便的,另外首尾相連循環遍歷也可大大增加鏈表操作的便捷性。因此,雙向循環鏈表,是在實際運用中是最常見的鏈表形態。 2. 基本操作 與普通的鏈表完全一致,雙向循環鏈表雖然指針較多,但邏輯是完全一樣。基本的操作包括: 節點設計 初…

Kubernetes集群升級與etcd備份恢復指南

目錄 Kubernetes etcd備份恢復 集群管理命令 環境變量 查看etcd版本 查看etcd集群節點信息 查看集群健康狀態 查看告警事件 添加成員(單節點部署的etcd無法直接擴容)&#xff08;不用做&#xff09; 更新成員 刪除成員 數據庫操作命令 增加(put) 查詢(get) 刪除(…

【LeetCode熱題100道筆記】旋轉圖像

題目描述 給定一個 n n 的二維矩陣 matrix 表示一個圖像。請你將圖像順時針旋轉 90 度。 你必須在 原地 旋轉圖像&#xff0c;這意味著你需要直接修改輸入的二維矩陣。請不要 使用另一個矩陣來旋轉圖像。 示例 1&#xff1a;輸入&#xff1a;matrix [[1,2,3],[4,5,6],[7,8,9]…

SpringBoot【集成p6spy】使用p6spy-spring-boot-starter集成p6spy監控數據庫(配置方法舉例)

使用p6spy-spring-boot-starter集成p6spy監控數據庫1.簡單說明2.核心依賴3.主要配置4.簡單測試5.其他配置1.簡單說明 p6spy 類似于 druid 可以攔截 SQL 可以用于項目調試&#xff0c;直接引入 p6spy 的博文已經很多了&#xff0c;這里主要是介紹一下 springboot 使用 p6spy-sp…

擴散模型的優化過程,主要的公式推導,主要是熟悉一下整體的理論框架

核心思想與定義 擴散模型的核心思想是&#xff1a;學習一個去噪過程&#xff0c;以逆轉一個固定的加噪過程。前向過程&#xff08;固定&#xff09;&#xff1a; 定義一個馬爾可夫鏈&#xff0c;逐步向數據 x0~q(x0)\mathbf{x}_0 \sim q(\mathbf{x}_0)x0?~q(x0?) 添加高斯噪…

數字簽名、數字證書、數字信封的概念與區別

要理解數字簽名、數字證書、數字信封&#xff0c;核心是抓住它們各自的核心目標 —— 分別解決 “身份真實性與內容完整性”“公鑰可信度”“數據機密性” 問題&#xff0c;且三者都基于 “非對稱加密”&#xff08;一對公鑰、私鑰&#xff0c;公鑰公開、私鑰保密&#xff0c;用…

Day35 網絡協議與數據封裝

day35 網絡協議與數據封裝 數據封裝與協議結構 以太網MAC幀格式數據封裝與傳輸流程 數據在傳輸過程中&#xff0c;從上層逐層封裝到底層&#xff0c;最終通過物理介質發送。封裝與傳輸的具體流程如下&#xff1a; 封裝過程&#xff08;從IP層到物理層&#xff09; IP層&#xf…

Deeplizard深度學習課程(七)—— 神經網絡實驗

前言我們正在利用pytorch實現CNN。主要分為四個小部分&#xff1a;數據預處理、神經網絡pytorch設計、訓練神經網絡 和 神經網絡實驗。在之前的章節中&#xff0c;我們已經完成了整個CNN框架的設計、訓練與簡單分析&#xff0c;本節將更進一步討論神經網絡處理過程中的細節問題…

STM32實踐項目(激光炮臺)

剛開始設想做一個上半部分可以上下180移動,下半部分底座360移動的激光炮臺。于是便開始了實踐。 所需材料清單: 序號 名稱 數量 備注說明 1 面包板(Breadboard) 2 用于電路搭建和模塊連接 2 杜邦線(公對公、公對母等) 若干 建議準備 30~50 根,方便連接 3 MB-102 電源模塊…

不止是夾住,更是“感知”:Contactile GAL2觸覺型夾爪實現自適應抓取

近日&#xff0c;專注于觸覺傳感與智能抓取技術的Contactile推出全新Contactile 觸覺型夾爪 GAL2&#xff0c;這款集成先進傳感技術的雙指夾爪&#xff0c;憑借實時觸覺反饋能力&#xff0c;為多行業智能抓取場景帶來突破性解決方案。 Contactile 觸覺型夾爪GAL2是一款多功能即…

Grafana - 監控磁盤使用率Variables使用

1 查詢prometheus2 編輯grafana dashboard 2.1 配置變量2.2 配置多選2.3 配置legend2.4 優化顯示 1 查詢prometheus 指標名稱描述node_filesystem_size_bytes文件系統總容量node_filesystem_avail_bytes用戶可用空間node_filesystem_files_free剩余inode數量比如我們想看/目…

WindowsAPI|每天了解幾個winAPI接口之網絡配置相關文檔Iphlpapi.h詳細分析10

上一篇&#xff1a;WindowsAPI|每天了解幾個winAPI接口之網絡配置相關文檔Iphlpapi.h詳細分析9 如果有錯誤歡迎指正批評&#xff0c;在此只作為科普和參考。 C:\Program Files (x86)\Windows Kits\10\Include\10.0.22621.0\um\iphlpapi.h 文章目錄GetNetworkParams&#xff1a…

算法 --- 分治(歸并)

分治&#xff08;歸并&#xff09; 分治&#xff08;特別是歸并&#xff09;算法適用于解決“整體求解依賴于子問題合并”且子問題相互獨立的題目&#xff0c;其典型特征是能將大規模數據分解、遞歸求解&#xff0c;然后通過合并操作&#xff08;這正是歸并排序中‘歸并’的精…

【程序人生】有夢想就能了不起,就怕你沒夢想

夢想不是遙不可及的星辰&#xff0c;而是需要我們用腳步丈量的路途兩年前的一個夏日&#xff0c;我在日記本上鄭重地寫下&#xff1a;"我要掌握Web開發&#xff0c;能夠獨立構建一個完整的Web應用。"那天是2023年6月8日&#xff0c;當時的我連Java和JavaScript都分不…

前端基礎(四十二):非固定高度的容器實現折疊面板效果

效果展示源碼 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title> </head>…

發票、收據合并 PDF 小程序,報銷上傳 3 秒搞定

每到報銷、報稅、財務整理時&#xff0c;手里是不是總有一堆格式不一的票據&#xff1a; 聊天記錄里的電子發票郵件附件中的 PDF 發票手機相冊里的報銷收據甚至還有零散的紙質票據掃描件 要上傳或交給財務前&#xff0c;還得一個個整理、轉換、排版&#xff0c;既耗時又容易出…

GitHub每日最火火火項目(9.4)

1. bytebot-ai / bytebot 項目名稱&#xff1a;bytebot項目介紹&#xff1a;基于 TypeScript 開發&#xff0c;是一款自托管的 AI 桌面智能體&#xff0c;能通過自然語言命令自動化執行計算機任務&#xff0c;運行在容器化的 Linux 桌面環境中。它借助自然語言處理和 AI 技術&a…

MMORPG 游戲戰斗系統架構

&#x1f30c; MMORPG 游戲戰斗系統架構 引用&#xff1a; 游戲服務器同步技術解析&#xff08;C&#xff09;MMORPG移動同步與反外掛 雖然我已離開游戲行業&#xff0c;轉而與幾位成功的商人共同創業&#xff0c;投身于商用機器人領域&#xff0c;但坦誠地說&#xff0c;游戲…

【數學建模學習筆記】啟發式算法:蒙特卡洛算法

蒙特卡洛模擬入門筆記&#xff1a;從原理到代碼實踐一、什么是蒙特卡洛模擬&#xff1f;蒙特卡洛模擬是一種通過大量隨機實驗來解決復雜問題的方法。簡單說&#xff0c;就是用電腦模擬成千上萬次隨機事件&#xff0c;然后統計結果&#xff0c;以此估算一個問題的答案。舉個生活…