【我們一起寫框架】MVVM的WPF框架(三)—數據控制項

来源:https://www.cnblogs.com/kiba/archive/2018/09/20/9648808.html
-Advertisement-
Play Games

這世上,沒人能一次性寫出完美無缺的框架;因為,任何一個框架都需要項目的淬煉,然後才能升華,趨近完美。 所以,框架是個反覆修改的東西,最終形成的東西。 如果你學了一點技術,覺得自己可以寫出框架了,覺得自己有架構師的能力,然而自己總是懷才不遇——那一定是你的錯覺。 因為,你框架沒有經過項目淬煉;而淬煉過 ...


這世上,沒人能一次性寫出完美無缺的框架;因為,任何一個框架都需要項目的淬煉,然後才能升華,趨近完美。

所以,框架是個反覆修改的東西,最終形成的東西。

如果你學了一點技術,覺得自己可以寫出框架了,覺得自己有架構師的能力,然而自己總是懷才不遇——那一定是你的錯覺。

因為,你框架沒有經過項目淬煉;而淬煉過框架的人都瞭解,設計的再好的框架,最終會被業務需求打的細碎,然後被開發人員攪和再一起。

所以細節決定成敗,沒有細節的框架就是扯淡。

DataControl—數據控制項

上文我們已經編寫出來了WPF的MVVM基礎框架,但為了讓他更加強壯,為了讓他多堅持一陣子再粉碎,我們要讓ViewModel更強壯,所以我們要編寫[數據控制項]。

數據控制項其實很好理解,它就是把UI控制項中存儲的數據提取出來,好讓ViewModel可以通過修改數據來控制UI變化;當然,為了更好的控制UI變化,數據控制項里還得包含一點管理UI的屬性。

因為WPF里的控制項大多繼承自Control,所以我們先創建Control的數據控制項。

public class Control<T> : INotifyPropertyChanged
{ 
    public event PropertyChangedEventHandler PropertyChanged;

    public T _DataContent ;
    public T DataContent { get { return _DataContent; } set { _DataContent = value; OnPropertyChanged(); } }

    public Visibility _Visibility;
    public Visibility Visibility { get { return _Visibility; } set { _Visibility = value; OnPropertyChanged(); } }

    public bool _IsReadOnly;
    public bool IsReadOnly { get { return _IsReadOnly; } set { _IsReadOnly = value; OnPropertyChanged(); } }

    public bool _IsEnabled;
    public bool IsEnabled { get { return _IsEnabled; } set { _IsEnabled = value; OnPropertyChanged(); } }

   

    protected void OnPropertyChanged([CallerMemberName]string propertyName = "")
    {
        if (PropertyChanged != null)
        {
            PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
        }
    }
}

如上代碼所示,我們創建了Control的數據控制項。

可以看到,處理存貯數據的DataContent屬性之外,還創建了一些管理UI的屬性IsEnabled、IsReadOnly、Visibility。

父類數據控制項創建完成後,我們開始創建子類的數據控制項。[如果子類要管理的UI屬性不在父類內,我們就需要額外創建一些]

TextBlock和TextBox

我們先創建最基礎的,最常用的TextBlock和TextBox。

TextBlock代碼如下:

public class TextBlock<T> : Control<T>
{ 
    public T _Text;
    public T Text
    {
        get { return _Text; }
        set
        {
            _Text = value; 
            OnPropertyChanged();
        }
    } 
}

TextBox代碼如下:

 public class TextBox<T> : Control<T>
 {
     public Action<T> TextChangeCallBack = null;

     public T _Text;
     public T Text {
         get { return _Text; }
         set
         {
             _Text = value;
             if (TextChangeCallBack != null)
             {
                 TextChangeCallBack(_Text);
             }
             OnPropertyChanged();
         }
     } 
 }

可以看到TextBlock和TextBox都繼承了Control,而他們的區別隻是TextBox多了一個TextChangeCallBack。

有人會想到,那完全可以用TextBox替代TextBlock。

理論上,TextBlock是可以被替換,但為了程式清晰,還是區別開來更好。

控制項定義好了,我們現在看一下如何應用。

TextBox應用

xaml頁面代碼
<TextBox Text="{Binding ChangeTextBox.Text,Mode=TwoWay}" Margin="5"  FontSize="12"></TextBox>
----------------------------------
ViewModel頁面代碼 public TextBox<string> ChangeTextBox { get; set; } public VM_PageTextBox() {
   ChangeTextBox = new TextBox<string>(); ChangeTextBox.TextChangeCallBack = (text) => { MessageBox(text); };//聲明TextChange }

如代碼所示,我們在ViewModel中定義了ChangeTextBox屬性,然後再Xaml中綁定了ChangeTextBox屬性的Text到UI控制項TextBox的Text屬性上,這樣我們就實現了數據聯動。

並且代碼中實例化了TextChangeCallBack委托,那麼當Text數據變化時,該委托就會觸發。

註意:TextChangeCallBack委托與TextChanged事件不同,並不是每次修改文字都會觸發,而是當TextBox的Text內容真正被修改時,才會觸發;我們可以簡單的理解為TextBox失去焦點時才會觸發。

這裡只介紹TextBox應用,TextBlock應用就不介紹了,因為使用方式和TextBox一樣。

如果想瞭解更多數據控制項的應用,請去GitHub下載源碼。

ComboBox

ComboBox稍微複雜一點,因為他多了一個ItemSource屬性。

我們先看ComboBox的數據控制項代碼:

public class ComboBox<T> : Control<T>
{
    public Action<T> SelectCallBack = null;
    public ComboBox()
    {

    }
    public ObservableCollection<T> _ItemsSource;
    public ObservableCollection<T> ItemsSource
    {
        get
        {
            return _ItemsSource;
        }
        set
        {
            _ItemsSource = value;
            if (_ItemsSource != null && _ItemsSource.Count > 0 && SelectedItem == null)
            {
                SelectedItem = _ItemsSource.First();
            }
            OnPropertyChanged();
        }
    }
    public T _SelectedItem;
    public T SelectedItem
    {
        get { return _SelectedItem; }
        set
        {
            _SelectedItem = value;
            if (SelectCallBack != null)
            {
                SelectCallBack(_SelectedItem);
            }
            OnPropertyChanged();
        }
    }
    private ICollectionView _ItemsSourceView;
    public ICollectionView ItemsSourceView
    {
        get
        {
            _ItemsSourceView = CollectionViewSource.GetDefaultView(_ItemsSource);
            return _ItemsSourceView;
        }
        set
        {
            _ItemsSourceView = value;
            OnPropertyChanged();
        }
    }
    public void SetItemsSource(List<T> itemSource)
    {
        ItemsSource = new ObservableCollection<T>(itemSource);
    }
}

代碼相對簡單,SelectedItem和ItemsSource用來綁定UI控制項ComboBox的同名屬性。

ItemsSourceView:ItemsSourceView屬性可能有些難理解,這裡我們簡單介紹一下。

因為WPF的UI控制項被創建以後,要被添加到視覺樹中,所以最終會被顯示在屏幕上的是包裹著控制項的視覺樹;其中視覺樹與控制項是可以分離的;比如控制項中綁定的數據是10行,而視覺樹可以顯示3行。

為了管理視覺樹,我們創建了ItemsSourceView屬性。

因為ItemsSourceView是ICollectionView類型,所以ItemsSourceView可以處理排序、篩選和分組。[有興趣的同學可以自行瞭解下ICollectionView類型]

感覺這樣描述還是很難理解,讓我們一起在應用中慢慢理解吧。

ObservableCollection:我們可以看到ItemsSource是類型是ObservableCollection,而不是List。為什麼要用ObservableCollection呢?

很簡單,因為ObservableCollection繼承了INotifyCollectionChanged,即,數據控制項進行[行]的增刪,也會讓UI進行[行]的增刪。

ComboBox應用

在應用之前,我們先在Proxy建立一個獲取數據是代理。

創建獲取數據的方法如下:

public List<User> GetComboBoxData()
{ 
     List<User> userList = new List<User>();
     User user1 = new User() { Id = 1, Name = "張三", Age = 11 };
     userList.Add(user1);
    return userList;
}

Xaml頁面代碼如下:

 <ComboBox  Margin="5" Width="200" FontSize="12" ItemsSource="{Binding TestComboBox.ItemsSource}" DisplayMemberPath="Name"  SelectedValuePath="Id" SelectedItem="{Binding TestComboBox.SelectedItem}"       ></ComboBox> 

ViewModel代碼如下:

public ComboBox<User> TestComboBox { get; set; }
TestDataProxy proxy = new TestDataProxy();
public VM_PageComboBox()
{
    TestComboBox = new ComboBox<User>();
    TestComboBox.SetItemsSource(proxy.GetComboBoxData());
    TestComboBox.SelectCallBack = (user) => {
        MessageBox(user.Name);
    };
}  

如上所示,我們已經實行了在ViewModel中管理ComboBox。

----------------------------------------------------------------------------------------------------

本篇文章就先講到這了,下一篇文章我們將一起為框架編寫DataGrid數據控制項。

因為DataGrid數據控制項是所有數據控制項中最複雜的,而且代碼量特別多;所以,我決定,單拿出一篇來介紹DataGrid。

框架代碼已經傳到Github上了,並且會持續更新。

相關文章:

【我們一起寫框架】MVVM的WPF框架(一)—序篇

【我們一起寫框架】MVVM的WPF框架(二)—綁定

To be continued——DataGrid

Github地址:https://github.com/kiba518/KibaFramework

----------------------------------------------------------------------------------------------------

註:此文章為原創,歡迎轉載,請在文章頁面明顯位置給出此文鏈接!
若您覺得這篇文章還不錯,請點擊下右下角的推薦】,非常感謝!

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.其一 在運行速度方面:StringBuilder > StringBuffer > String 上實例 上述兩個例子只是for迴圈 從0 到99 博主筆記本有點卡 當數據龐大時優勢就展示出了 鐵證如山 ,就是有圖有真相的那種 那麼, 為什麼會出現這種運行速度的差別,引出其二區別 2.其二 數據 ...
  • 概念 枚舉是一組命名整型常量 枚舉類型是使用 enum 關鍵字聲明的 例如定義一個變數,該變數的值表示一周中的一天: enum Days { Sunday, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday}; 聲明枚舉變數 一般語法: e ...
  • 註解:最近做了一個小工具,在Winform中對Picture控制項有一個需求,可以通過滑鼠從外部拖拽圖片到控制項的上,釋放滑鼠,顯示圖片! //函數從動態鏈接庫中倒入(模擬滑鼠事件) 4.在來設置PictureBox的事件 5.以上就可以完成推拽圖片顯示圖片(無論是本地還是QQ消息框中的圖片都可以實現) ...
  • 本文轉自: https://blog.csdn.net/Cooldiok/article/details/7831351 2017年10月22日 21:31:22 Cooldiok 微軟作為ASP.NET的創造者,它對於官網的結構設計肯定有值得我們借鑒和參考的地方 本項目是基於VS2017 pro開 ...
  • 前言:監控web網站方法有很多種,這篇文章說一下對windows伺服器 asp.net網站的監控 採用的方案,Powershell + Influxdb + Grafana 1、PowerShell + Influxdb PowerShell用來收集IIS指標,10秒採集一次,然後寫入Influxd ...
  • 這一節介紹EF CodeFirst模式中的1對0/1,1對多,多對多關係的配置,只有梳理清楚實體間的關係,才能進行愉快的開發,因此這節雖然很簡單但是還是記錄了一下。 1. 1對0/1關係配置 1. 通過數據註釋屬性配置1對0/1關係 我們將要實現一個Student和StudentAddress實體的 ...
  • 使用特性(Attribute)可以將描述程式集的信息和描述程式集中任何類型和成員的信息添加到程式集的元數據和IL代碼中,程式可以在運行時通過反射獲取到這些信息; 一、通過直接或間接的繼承自抽象類System.Attribute可以創建自定義的特性類,自定義的特性類必須聲明為公共類,命名一般使用Att ...
  • ADO.NET進行參數化時會自動將參數值包含在單引號中,除了特殊需求,最好不要自己手動添加單引號。ADO.NET中識別參數標識是使用符號@,如果在SQL語句中將參數標識放在單引號中,單引號中的參數標識只會被當成字元串! 所以要對LIKE語句進行參數化查詢時,就要先對參數值進行格式化,在傳參之前就設置 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...