1. <strong id="7actg"></strong>
    2. <table id="7actg"></table>

    3. <address id="7actg"></address>
      <address id="7actg"></address>
      1. <object id="7actg"><tt id="7actg"></tt></object>

        WPF MVVM 模式下的彈窗

        共 3559字,需瀏覽 8分鐘

         ·

        2020-07-25 00:25

        c96e2b70765dc144f3ffc8af3bd76be4.webp

        WPF?MVVM?模式下的彈窗

        獨(dú)立觀察員 2020 年 7 月 15 日

        一、總體展示

        首先看看用戶控件在設(shè)計頁面的大致效果:

        a6d7c9ca237480c45b3af1eb56c75a06.webp

        ?

        中間部分自然就是確認(rèn)彈框了,由標(biāo)題、內(nèi)容、確認(rèn)按鈕、取消按鈕、倒計時、關(guān)閉按鈕組成,指定了大小范圍:

        556b7fc5433aeb2df3b3b28fb8381a41.webp

        ?

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

        83bd613bd1c07a9dedd41effa5fd0e5f.webp

        ?

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

        ca3c680b95fa9aedcb469abb48f59a2a.webp

        ?

        還有一種是信息彈框,區(qū)別是不用于執(zhí)行業(yè)務(wù)方法,也不輸出信息 (操作結(jié)果),只是用于提示用戶,且默認(rèn)標(biāo)題和默認(rèn)超時時間不同(可修改):

        a1c1d6c3a2740c34a780764b5ad10b58.webp

        ?

        二、用戶控件前端

        新建 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" >XButton> 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=' 是否確認(rèn)操作?', TargetNullValue=' 是否確認(rèn)操作?'}" 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=' 確認(rèn) ', FallbackValue=' 確認(rèn) '}" 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>

        ?

        就是簡單做了下布局和樣式,然后做了數(shù)據(jù)綁定和命令綁定,我們移步到后臺來看。

        ?

        三、用戶控件后臺

        由于使用了?MVVM?模式,所以頁面的后臺代碼中沒多少內(nèi)容:

        ///?<summary>/// [dlgcy] WPF MVVM 確認(rèn)彈框;/// public partial class UC_ConfirmBox : UserControl{    public UC_ConfirmBox ()    {        InitializeComponent ();    }
        /// /// 綁定 VM 中的 IsShowDialog /// 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 的(是否顯示彈窗),實(shí)際上,不用這個依賴屬性而直接用 Visibility 綁定 IsShowDialog(ViewModel 中的),然后加上相關(guān)轉(zhuǎn)換器也可以,但那樣對用戶不太友好,所以這里直接在依賴屬性中進(jìn)行 Visibility 的判斷。(關(guān)于依賴屬性的使用可以看本人之前的文章《WPF?用戶控件的自定義依賴屬性在?MVVM?模式下的使用備忘》)。

        ?

        然后注意一點(diǎn),這里并沒有直接將 DataContext 關(guān)聯(lián) ViewModel,而是要在使用用戶控件時再綁定(大家覺得我做得對嗎),這個后面還會說到。

        ?

        四、用戶控件對應(yīng)的 ViewModel

        這里代碼比較多,就不貼出來了,最后會給出代碼托管地址。ViewModel 整體結(jié)構(gòu)如下:

        8fd3d4cd582003c5e35c609a10b790db.webp

        ?

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

        ?

        (1) 彈框時阻塞業(yè)務(wù)流程

        先來看看 “成員” 部分:

        8eeb8e14fe5c5002129be7f09dee74b3.webp

        ?

        有個線程同步對象 AutoResetEvent,缺省設(shè)置為阻塞線程,由上圖可見,在彈框隱藏時會取消阻塞,那么阻塞的時機(jī)自然就是彈框顯示后:

        057dfc026bfe7d95bf42bae65ddafa8c.webp

        ?

        上圖顯示的確認(rèn)框幫助類的 “彈出確認(rèn)框” 方法中,由于是使用異步調(diào)用,所以阻塞不會影響 UI 線程。阻塞方法可以指定超時時間,超時或者用戶沒有點(diǎn)擊確認(rèn)按鈕則直接返回,否則,則執(zhí)行傳入的委托方法,即實(shí)際的業(yè)務(wù)方法。

        ?

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

        afca909e738a42258544714aa15b39fa.webp

        ?

        (2) 倒計時

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

        計時器的執(zhí)行方法在構(gòu)造函數(shù)中綁定,執(zhí)行方法內(nèi)部,每隔一秒(聲明時設(shè)定)將剩余時間減 1,減為 0 時停止,并執(zhí)行關(guān)閉命令。此處和彈窗阻塞超時那里可能有功能冗余,當(dāng)然,從另一方面來說,也可以看作是雙重保險。

        2a042632afaaead1c9f3f79a7283458c.webp

        ?

        (3) 其它

        Bindable” 區(qū)域中剩余的屬性都沒有做特殊處理。

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

        8e1d383d4d3b643fad9e916ed8a1f6a9.webp

        ?

        五、使用

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

        ec3cb78286357f03f7aecc7c1f2f4497.webp

        ?

        IsShowDialog=”{Binding IsShowDialog}” 也是固定這樣設(shè)置即可,用于配合 DataContext 控制顯示隱藏,而 DataContext 則是綁定了主頁面 ViewModel 中相關(guān)的用戶控件的 ViewModel 對象。

        ?

        調(diào)用的時候要注意異步的問題,使用輔助類 ConfirmBoxHelper 中的兩個方法即可:

        16c6147d1deb95ced43ddc6da207f7b9.webp

        ?

        給出文字版:

        //?前臺;<uc:UC_ConfirmBox DataContext="{Binding DialogVm}" Visibility="Collapsed" IsShowDialog="{Binding IsShowDialog}">
        // ViewModel;public ConfirmBoxViewModel DialogVm { get; set; } = new ConfirmBoxViewModel ();
        //?使用;await?ConfirmBoxHelper.ShowMessage?(DialogVm,?"操作前通知",?6);
        await ConfirmBoxHelper.ShowConfirm (DialogVm, "您確定要進(jìn)行此操作嗎?", () =>{ #region 業(yè)務(wù)方法
        //。。。#endregion
        }, ShowInfo);
        await ConfirmBoxHelper.ShowMessage (DialogVm, "操作后通知");

        ?

        六、地址

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

        https://gitee.com/dlgcy/XmppPractice

        Dotnet9網(wǎng)站常駐編輯


        長按關(guān)注我,

        歡迎技術(shù)交流!

        -好東西要轉(zhuǎn)發(fā),設(shè)為"星標(biāo)"★搶先看-

        27ecd5578b8b98d6286681b587df68b5.webp點(diǎn)擊閱讀原文,查看Dotnet9站點(diǎn)更多關(guān)于WPF的博文。

        瀏覽 54
        點(diǎn)贊
        評論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報
        評論
        圖片
        表情
        推薦
        點(diǎn)贊
        評論
        收藏
        分享

        手機(jī)掃一掃分享

        分享
        舉報
        1. <strong id="7actg"></strong>
        2. <table id="7actg"></table>

        3. <address id="7actg"></address>
          <address id="7actg"></address>
          1. <object id="7actg"><tt id="7actg"></tt></object>
            久久一区二区三区在线 | 欧美性丝袜老师 | 中国女人野外做爰a正片 | 国产大片久久久 | 国产丰满果冻VⅰdeOSSeX | 嗯啊h水患者护士啪啪np文字 | 操逼视频高清无码 | 北条麻妃无码免费看 | 美女床上诱惑视频 | 欧美操逼的视频 |