WPF MVVM 模式下的彈窗

来源:https://www.cnblogs.com/weiliuhong/archive/2020/07/15/wpf-mvvm-confirm-dialog.html
-Advertisement-
Play Games

WPF MVVM 模式下的彈窗,包括確認框和信息提示框,MVVM 模式的用戶控制項。 ...


WPF MVVM 模式下的彈窗

獨立觀察員 2020 年 7 月 15 日

一、總體展示

首先看看用戶控制項在設計頁面的大致效果:

 

中間部分自然就是確認彈框了,由標題、內容、確認按鈕、取消按鈕、倒計時、關閉按鈕組成,指定了大小範圍:

 

外層還有個 Grid,沒有指定大小,所以使用時會鋪滿容器,再配上帶透明度的背景色,可以當作蒙版,避免用戶繼續操作後面的界面,達到模態彈窗的效果:

 

確認彈框,手動關閉、點擊取消按鈕、超時關閉這三種情況下會輸出相關信息(需傳入記錄信息的委托方法),點擊確認按鈕則可以繼續執行業務方法。

 

還有一種是信息彈框,區別是不用於執行業務方法,也不輸出信息 (操作結果),只是用於提示用戶,且預設標題和預設超時時間不同(可修改):

 

二、用戶控制項前端

新建 WPF 用戶控制項後,貼入以下代碼:

<Grid Background="#905F9EA0">
    <Grid Background="Gainsboro" MinWidth="300" MinHeight="200" MaxWidth="400" MaxHeight="300">
        <Grid.RowDefinitions>
            <RowDefinition Height="28"></RowDefinition>
            <RowDefinition Height="28"></RowDefinition>
            <RowDefinition Height="*"></RowDefinition>
            <RowDefinition Height="Auto"></RowDefinition>
        </Grid.RowDefinitions>

        <DockPanel Height="28" Background="SteelBlue">
            <TextBox Text="{Binding DialogTitle, TargetNullValue=' 註意 ', FallbackValue=' 註意 '}" Height="26" Width="Auto" Background="SteelBlue" BorderThickness="0" HorizontalAlignment="Left" VerticalAlignment="Center" Foreground="White" FontSize="16" Focusable="False" IsHitTestVisible="False" IsTabStop="False" VerticalContentAlignment="Center" Padding="2,0,0,0">
            </TextBox>
            <Button x:Name="BtnClose" Command="{Binding CloseCommand}" Height="26" Width="26" HorizontalAlignment="Right" VerticalAlignment="Center" FontSize="16" Background="Transparent" Foreground="White" BorderThickness="0" >X</Button>
        </DockPanel>

        <StackPanel Grid.Row="1" Orientation="Horizontal" FlowDirection="RightToLeft">
            <TextBox VerticalContentAlignment="Center" Text="{Binding LeftTime, FallbackValue=20, TargetNullValue=20}" FontSize="16" Background="Transparent" Foreground="Coral" BorderThickness="0" Margin="5,0"></TextBox>
        </StackPanel>

        <TextBlock Grid.Row="2" FontSize="16" Text="{Binding DialogMessage, FallbackValue=' 是否確認操作?', TargetNullValue=' 是否確認操作?'}" VerticalAlignment="Center" HorizontalAlignment="Center"></TextBlock>

        <StackPanel Grid.Row="3" Orientation="Horizontal" FlowDirection="RightToLeft" VerticalAlignment="Center" Margin="0, 10">
            <Button x:Name="BtnConfirm" Command="{Binding ConfirmCommand}" Content="{Binding DialogConfirmBtnText, TargetNullValue=' 確認 ', FallbackValue=' 確認 '}" FontSize="16" Background="SteelBlue" Foreground="White" Margin="10, 0" Width="120"></Button>
            <Button x:Name="BtnCancel" Command="{Binding CancelCommand}" Content="{Binding DialogCancelBtnText, TargetNullValue=' 取消 ', FallbackValue=' 取消 '}" FontSize="16" Background="SteelBlue" Foreground="White" Margin="10, 0" Width="120"></Button>
        </StackPanel>
    </Grid>
</Grid>

 

就是簡單做了下佈局和樣式,然後做了數據綁定和命令綁定,我們移步到後臺來看。

 

三、用戶控制項後臺

由於使用了 MVVM 模式,所以頁面的後臺代碼中沒多少內容:

/// <summary>
/// [dlgcy] WPF MVVM 確認彈框;
/// </summary>
public partial class UC_ConfirmBox : UserControl
{
    public UC_ConfirmBox ()
    {
        InitializeComponent ();
    }

    /// <summary>
    /// 綁定 VM 中的 IsShowDialog
    /// </summary>
    public bool IsShowDialog
    {
        get { return (bool) GetValue (IsShowDialogProperty); }
        set { SetValue (IsShowDialogProperty, value); }
    }

    public static readonly DependencyProperty IsShowDialogProperty = DependencyProperty.Register ("IsShowDialog", typeof (bool), typeof (UC_ConfirmBox), new PropertyMetadata (false, (obj, args) =>
    {
        if (args.NewValue is bool newValue)
        {
            try
            {
                var control = obj as UC_ConfirmBox;
                control.Visibility = newValue ? Visibility.Visible : Visibility.Collapsed;
            }
            catch (Exception ex)
            {
                Console.WriteLine (ex.ToString ());
                MessageBox.Show ($"{ex.Message}");
            }
        }
    }));
}

 

建了個依賴屬性,用於使用用戶控制項時綁定。這個是綁定 ViewModel 中的同名屬性 IsShowDialog 的(是否顯示彈窗),實際上,不用這個依賴屬性而直接用 Visibility 綁定 IsShowDialog(ViewModel 中的),然後加上相關轉換器也可以,但那樣對用戶不太友好,所以這裡直接在依賴屬性中進行 Visibility 的判斷。(關於依賴屬性的使用可以看本人之前的文章《WPF 用戶控制項的自定義依賴屬性在 MVVM 模式下的使用備忘》)。

 

然後註意一點,這裡並沒有直接將 DataContext 關聯 ViewModel,而是要在使用用戶控制項時再綁定(大家覺得我做得對嗎),這個後面還會說到。

 

四、用戶控制項對應的 ViewModel

這裡代碼比較多,就不貼出來了,最後會給出代碼托管地址。ViewModel 整體結構如下:

 

ConfirmBoxViewModel 上有個特性 AddINotifyPropertyChangedInterface,這個是一個第三方的包 PropertyChanged.Fody 提供的,加上之後,類的公共自動屬性就具有了屬性變動通知功能。那麼為什麼還要繼承 BindableBase (實現了 InotifyPropertyChanged 的基類,參考《WPF 原生綁定和命令功能使用指南》)呢? 原因是,如果在屬性的 get/set 中做了一些操作的話,Fody 對該屬性好像就不起作用了,所以補救一下。

 

(1) 彈框時阻塞業務流程

先來看看 “成員” 部分:

 

有個線程同步對象 AutoResetEvent,預設設置為阻塞線程,由上圖可見,在彈框隱藏時會取消阻塞,那麼阻塞的時機自然就是彈框顯示後:

 

上圖顯示的確認框幫助類的 “彈出確認框” 方法中,由於是使用非同步調用,所以阻塞不會影響 UI 線程。阻塞方法可以指定超時時間,超時或者用戶沒有點擊確認按鈕則直接返回,否則,則執行傳入的委托方法,即實際的業務方法。

 

另一個 “彈出消息框” 方法則相較簡單,只是簡單阻塞了一段 “消息框超時時間”:

 

(2) 倒計時

上一小節開頭處給出的” 成員圖” 中,還有一個定時器類型對象 _timer,就是用於倒計時功能的。計時器在彈窗彈出時開始啟動,代碼位於 IsShowDialog 屬性的 Set 方法中(見” 成員圖”)。

計時器的執行方法在構造函數中綁定,執行方法內部,每隔一秒(聲明時設定)將剩餘時間減 1,減為 0 時停止,並執行關閉命令。此處和彈窗阻塞超時那裡可能有功能冗餘,當然,從另一方面來說,也可以看作是雙重保險。

 

(3) 其它

“Bindable” 區域中剩餘的屬性都沒有做特殊處理。

命令的使用可以參考前文提到的文章,命令的處理邏輯則比較簡單,就是設置是否顯示和是否確認:

 

五、使用

使用時,引入用戶控制項命名空間之後,將其與主界面平級擺放,實際就是覆蓋在主界面上方,然後設置其 Visibility 屬性為 “Collapsed”,不可見也不占用空間,避免影響主界面的開發:

 

IsShowDialog=”{Binding IsShowDialog}” 也是固定這樣設置即可,用於配合 DataContext 控制顯示隱藏,而 DataContext 則是綁定了主頁面 ViewModel 中相關的用戶控制項的 ViewModel 對象。

 

調用的時候要註意非同步的問題,使用輔助類 ConfirmBoxHelper 中的兩個方法即可:

 

給出文字版:

// 前臺;
<uc:UC_ConfirmBox DataContext="{Binding DialogVm}" Visibility="Collapsed" IsShowDialog="{Binding IsShowDialog}"></uc:UC_ConfirmBox>

// ViewModel;
public ConfirmBoxViewModel DialogVm { get; set; } = new ConfirmBoxViewModel ();

// 使用;
await ConfirmBoxHelper.ShowMessage (DialogVm, "操作前通知", 6);

await ConfirmBoxHelper.ShowConfirm (DialogVm, "您確定要進行此操作嗎?", () =>
{
    #region 業務方法

    //。。。   

  #endregion
}, ShowInfo); await ConfirmBoxHelper.ShowMessage (DialogVm, "操作後通知");

 

六、地址

這個是在 XMPP 通信 Demo 項目中寫的,項目地址為:

https://gitee.com/dlgcy/XmppPractice

 

同步首發:

http://dlgcy.com/wpf-mvvm-confirm-dialog/

微信訂閱號

 


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

-Advertisement-
Play Games
更多相關文章
  • # 二叉搜索樹的特點是左子樹小於根節點,右子樹大於根節點。# 因此當根節點為i的時候,左子樹的值為1:i-1,右子樹為i+1:n# 當節點為n的時候所有的能夠組成的樹為左子樹個數乘以右子樹個數。class Solution: def numTrees(self, n: int) -> int: dp ...
  • update 2020/7/15 優化了一下 \(markdown\) 的用法,增加了前面的題目描述。 題目: 題目描述 從加里敦大學城市規劃專業畢業的小明來到了一個地區城市規劃局工作。這個地區一共有 \(n\) 座城市,\(n-1\) 條高速公路,保證了任意兩運城市之間都可以通過高速公路相互可達, ...
  • 引言 現在各大技術社區 Spring Boot 的文章越來越多,Spring Boot 相關的圖文、視頻教程越來越多,使用 Spring Boot 的互聯網公司也越來越多; Java 程式員現在出去面試, Spring Boot 已經成了必問的內容。 一切都在證明,Spring Boot 已經成為了 ...
  • 《c++入門經典》筆記11 第十一章 開發高級指針 11.1在堆中創建對象 實際上,類就是對象的類型,對象也是一種變數,所以你可以在堆中創建int型變數,自然也就能創建自定義型變數。 Cat *pCat = new Cat; 這將調用預設構造函數(無參構造函數),每當在堆或棧中創建對象時,都將調用構 ...
  • 《c++入門經典》筆記10 第十章 創建指針 10.1理解指針及其用途 變數是可存儲一個值的對象:整型變數存儲一個數字,字元變數存儲一個字母,而指針是存儲記憶體地址的變數。 電腦記憶體是存儲變數值的地方。根據約定,電腦記憶體被劃分成按順序編號的記憶體單元,每個記憶體單元都有對應的地址。記憶體中,不管其類型是 ...
  • 一、文件基本處理1、什麼是文件 文件是操作系統提供給用戶/應用程式操作硬碟的一個虛擬單位ps:文件是操作系統的概念2、為何要用文件 存取硬碟必須使用文件ps:要永久保存數據就得用文件把數據放進硬碟3、如何用文件 f = open(文件路徑,打開模式) f.write(數據) f.close() 應用 ...
  • 協程使用註意事項 協程內部禁止使用全局變數,以免發生數據錯亂; 協程使用 use 關鍵字引入外部變數到當前作用域禁止使用引用,以免發生數據錯亂; 不能使用類靜態變數 Class::$array / 全局變數 $_array / 全局對象屬性 $object->array / 其他超全局變數 $GLO ...
  • 問題 :最近在項目論壇中,因為要上傳自定義圖片,在做好上傳的前端處理後,在後臺使用move_uploaded_file函數,移動文件的時候,一直返回的是 false; :因為對於這類新加的前端加後臺一起寫的功能,一般是先在本地建一個demo實現,然後在加入到項目中。就在這裡,居然出現一個問題,在前端 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...