在上一篇博文《[UWP]使用Popup构建UWP Picker》中我们简单讲述了一下使用Popup构建适用于MVVM框架下的弹窗层组件Picker的过程。但是没有应用实例的话可能体现不出Picker相对于ContentDialog的优点在哪里,毕竟Linus大神说过:

Talk is cheap, show me the code!

我们假定要实现这样一个颜色选择器:当用户需要选择一个颜色时,应用弹出颜色选择器,用户选择完成后,点击“确定”按钮关闭弹窗,并且向调用方代码返回用户选择的颜色值。

它的调用界面是这样的:

编写ColorPicker弹窗的业务逻辑代码

上篇博文里我们讲到要实现Picker功能,其ViewModel必须实现IObjectPicker接口。我在HHChaosToolkit库中已经定义了ObjectPickerBase作为Picker的公共基类,我们的ViewModel直接继承它就可以了。

IObjectPicker的接口定义:

  1. public interface IObjectPicker<T>
  2. {
  3. event EventHandler<ObjectPickedEventArgs<T>> ObjectPicked;
  4. event EventHandler Canceled;
  5. }

ObjectPickerBase的定义:

  1. public abstract class ObjectPickerBase<T> : ViewModelBase, IObjectPicker<T>
  2. {
  3. public event EventHandler<ObjectPickedEventArgs<T>> ObjectPicked;
  4. public event EventHandler Canceled;
  5. /// <summary>
  6. /// 设置选择的对象
  7. /// </summary>
  8. /// <param name="result"></param>
  9. public void SetResult(T result)
  10. {
  11. ObjectPicked?.Invoke(this, new ObjectPickedEventArgs<T>(result));
  12. }
  13. /// <summary>
  14. /// 取消Pick操作
  15. /// </summary>
  16. public void Exit()
  17. {
  18. Canceled?.Invoke(this, EventArgs.Empty);
  19. }
  20. public RelayCommand ExitCommand => new RelayCommand(Exit);
  21. }

这里我们编写一个TestColorPickerViewModel作为ColorPicker弹窗界面的ViewModel,其代码如下:

  1. public class TestColorPickerViewModel: ObjectPickerBase<Color>
  2. {
  3. private Color _pickedColor;
  4. public Color PickedColor
  5. {
  6. get => _pickedColor;
  7. set => Set(ref _pickedColor, value);
  8. }
  9. public override void OnNavigatedTo(NavigationEventArgs e)
  10. {
  11. if (e.Parameter is Color color)
  12. {
  13. PickedColor = color;
  14. }
  15. base.OnNavigatedTo(e);
  16. }
  17. public ICommand PickColorCommand => new RelayCommand(() =>
  18. {
  19. SetResult(PickedColor);
  20. });
  21. }

其中有一个重载的方法OnNavigatedTo,这个用于接受打开弹窗时给传递给Picker的参数,这个属于HHChaosToolkit类库中MVVM导航服务的一部分功能,以后的博客我可能会拿出来单独讲一下。

我们看到,TestColorPickerViewModel的代码逻辑非常简单,在执行PickColorCommand后返回PickedColor作为结果。

编写ColorPicker的UI层代码

View层交互不多,我们新建一个Page,然后添加一个ColorPicker控件,Color属性绑定ViewModel的PickedColor,添加一个“确定”按钮绑定PickColorCommand,xaml.cs文件中无需添加任何代码,xaml代码如下:

  1. <Page
  2. x:Class="HHChaosToolkit.Sample.Views.TestPages.TestColorPickerPage"
  3. xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
  4. xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
  5. xmlns:local="using:HHChaosToolkit.Sample.Views.TestPages"
  6. xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
  7. xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
  8. mc:Ignorable="d"
  9. DataContext="{Binding TestColorPickerViewModel, Source={StaticResource Locator}}"
  10. Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
  11. <Grid>
  12. <Grid Background="White" BorderBrush="#d9ddea" BorderThickness="1">
  13. <Grid.RowDefinitions>
  14. <RowDefinition Height="Auto" />
  15. <RowDefinition />
  16. <RowDefinition Height="Auto" />
  17. </Grid.RowDefinitions>
  18. <Grid Height="40" Background="#d9ddea">
  19. <Grid.ColumnDefinitions>
  20. <ColumnDefinition />
  21. <ColumnDefinition Width="auto" />
  22. </Grid.ColumnDefinitions>
  23. <TextBlock
  24. Margin="15,0"
  25. VerticalAlignment="Center"
  26. FontSize="14px"
  27. Foreground="#474261"
  28. Text="ColorPicker" />
  29. <Button
  30. Grid.Column="1"
  31. Command="{Binding ExitCommand}"
  32. Style="{StaticResource PickerCloseButtonStyle}" />
  33. </Grid>
  34. <Grid Grid.Row="1" Padding="20,10">
  35. <ColorPicker x:Name="ColorPicker" Color="{Binding PickedColor,Mode=TwoWay}"/>
  36. </Grid>
  37. <Grid Grid.Row="2" Padding="20">
  38. <Button HorizontalAlignment="Center" Content="确定" Command="{Binding PickColorCommand}"/>
  39. </Grid>
  40. </Grid>
  41. </Grid>
  42. </Page>

注册调用过程

注册ColorPicker弹窗

我们首先要在ViewModelLocator中注册TestColorPickerViewModel为可选取Color类型的Picker对象,代码如下:

  1. RegisterObjectPicker<Color, TestColorPickerViewModel, TestColorPickerPage>();

其中RegisterObjectPicker方法的源码如下:

  1. public void RegisterObjectPicker<T, VM, V>()
  2. where VM : ObjectPickerBase<T>
  3. {
  4. SimpleIoc.Default.Register<VM>();
  5. ObjectPickerService.Configure(typeof(T).FullName, typeof(VM).FullName, typeof(V));
  6. }

这段代码目的是在ObjectPickerService中注册TestColorPickerViewModel为可选取Color类型的Picker对象,这样我们之后的调用可以直接通过ObjectPickerService来进行。

必须要说明的是ObjectPickerService可以为同一类型注册多个Picker对象,类似于Windows系统中可安装多个视频播放器,调用时指定使用哪个播放器即可。

调用ColorPicker弹窗

在ObjectPickerService中注册过后,我们即可在任意需要选取颜色的地方使用我们的ColorPicker弹窗,最简单的调用方法时这样的:

  1. var pickerService = ServiceLocator.Current.GetInstance<ObjectPickerService>();
  2. var ret = await pickerService.PickSingleObjectAsync<Color>(typeof(TestColorPickerViewModel)
  3. .FullName, PickedColor);
  4. if (!ret.Canceled)
  5. {
  6. PickedColor = ret.Result;
  7. var toast = new Toast($"You picked a new color!({ret.Result})");
  8. toast.Show();
  9. }

当然我们也可以自定义弹出界面的位置、背景、动画及点击空白区域退出等选项。如果需要这样自定义的话,我们要用到PickerOpenOption这个类,这个类用来设置Picker弹出时的自定义配置项,例如:

  1. var pickerService = ServiceLocator.Current.GetInstance<ObjectPickerService>();
  2. var openOption = new PickerOpenOption
  3. {
  4. EnableTapBlackAreaExit = true,
  5. VerticalAlignment = VerticalAlignment.Stretch,
  6. HorizontalAlignment = HorizontalAlignment.Right,
  7. Background = new AcrylicBrush
  8. {
  9. TintOpacity = 0.1
  10. },
  11. Transitions = new TransitionCollection
  12. {
  13. new EdgeUIThemeTransition
  14. {
  15. Edge = EdgeTransitionLocation.Right
  16. }
  17. }
  18. };
  19. var ret = await pickerService.PickSingleObjectAsync<Color>(typeof(TestColorPickerViewModel)
  20. .FullName, PickedColor, openOption);
  21. if (!ret.Canceled)
  22. {
  23. PickedColor = ret.Result;
  24. var toast = new Toast($"You picked a new color!({ret.Result})");
  25. toast.Show();
  26. }

它的呈现效果是这样的:

结尾

这篇博文里我给大家讲解了如何使用Picker来构建一个颜色选择器弹窗,这只是一个小例子,Picker有非常多的使用场景,例如:

  • 文本输入弹窗(注册类型为string);
  • 普通自定义Dialog界面(统一注册类型为bool即可);
  • 图片编辑弹窗(注册类型为文件或者图片);
  • ...

最后,完整项目代码链接在这里:GitHub链接点这里,欢迎大家使用,或者提出意见!

本篇博客到此结束!谢谢大家!

[UWP]使用Picker实现一个简单的ColorPicker弹窗的更多相关文章

  1. 为WPF, UWP 及 Xamarin实现一个简单的消息组件

    原文地址:Implementing a simple messenger component for WPF, UWP and Xamarin 欢迎大家关注我的公众号:程序员在新西兰了解新西兰IT行业 ...

  2. 一个简单的页面弹窗插件 jquery.pageMsgFrame.js

    页面弹窗是网站中常用的交互效果,它可以强提示网站的某些信息给用户,或者作用于某些信息的修改等等功能. 这几天在做一个项目的时候,就顺捎把这个插件写一下,栽棵树,自己乘凉吧. 原创博文,转载请注明出处: ...

  3. 【UWP开发】一个简单的Toast实现

    Toast简介 在安卓里Toast是内置原生支持,它是Android中用来显示显示信息的一种机制.它主要用于向用户显示提示消息,没有焦点,显示的时间有限,过一定的时间就会自动消失.在UWP中虽然没有原 ...

  4. iOS开发UI篇—使用picker View控件完成一个简单的选餐应用

    iOS开发UI篇—使用picker View控件完成一个简单的选餐应用 一.实现效果 说明:点击随机按钮,能够自动选取,下方数据自动刷新. 二.实现思路 1.picker view的有默认高度为162 ...

  5. [UWP 开发] 一个简单的Toast实现

    Toast简介 在安卓里Toast是内置原生支持,它是Android中用来显示显示信息的一种机制.它主要用于向用户显示提示消息,没有焦点,显示的时间有限,过一定的时间就会自动消失.在UWP中虽然没有原 ...

  6. [UWP]实现Picker控件

    1. 前言 在WPF中,很多打开下拉框(Popup或Flyout)选择一个结果值的控件,除了ComboBox等少数例外,这种控件都以-Picker做名称后缀.因为要打开关闭下拉框和计算下拉框的弹出位置 ...

  7. IOS开发之小实例--创建一个简单的用于视频录制和回放的应用程序

    前言:还是看了一下国外的入门IOS文章:<Create a Simple App for Video Recording and Playback>,主要涉及视频录制和回放的功能的基本实现 ...

  8. IOS开发之小实例--使用UIImagePickerController创建一个简单的相机应用程序

    前言:本篇博文是本人阅读国外的IOS Programming Tutorial的一篇入门文章的学习过程总结,难度不大,因为是入门.主要是入门UIImagePickerController这个控制器,那 ...

  9. WPF 一个简单的颜色选择器

    原文:WPF 一个简单的颜色选择器 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/BYH371256/article/details/8340999 ...

随机推荐

  1. getColor问题

    getResource.getColor(R.color.color_name) 过时了,可以使用新加入的方法 ContextCompat.getColor(context, R.color.colo ...

  2. win10 KMS命令激活步骤<转>

    三.win10 KMS命令激活步骤如下: 1.右键点击开始图标,弹出这个菜单,选择[windows powershell(管理员)],或者命令提示符管理员: 2.打开命令窗口,复制这个命令slmgr ...

  3. Oracle 事务和异常处理

    Oracle 的异常和回滚 DECLARE dept_no ) :; BEGIN --开始事务 INSERT INTO dept VALUES (dept_no, '市场部', '北京'); --插入 ...

  4. kubectl-常用命令

    出处https://cloud.tencent.com/developer/article/1140076 kubectl apply -f kubernetes-dashboard.yaml -n ...

  5. 用react编写一个可以编辑的表格

    这只一个雏形,但是可以用了.难点是如何点击每行后面的编辑按钮,让当前行的格子都变成input. import {Component} from 'react' const Action = props ...

  6. 解决bootstrap和easyUI部分css类冲突问题。

    今天发现bootstrap和easyui的css类重复用了一个很笨的办法解决了,这种小事网上都不好搜啊. 我先引用的bootstrap后引用的easy UI,bootstrap的会被覆盖,boot的样 ...

  7. ArcGIS自定义工具箱-字段合并

    ArcGIS自定义工具箱-字段合并 联系方式:谢老师,135-4855-4328,xiexiaokui#qq.com 目的:用指定字符合并两个字段 用例:湖南/长沙=>湖南省长沙市 数据源: 使 ...

  8. .Net 中读写Oracle数据库常用两种方式

    .net中连接Oracle 的两种方式:OracleClient,OleDb转载 2015年04月24日 00:00:24 10820.Net 中读写Oracle数据库常用两种方式:OracleCli ...

  9. Pandas.Series.dt.dayofweek相关命令

  10. Centos7安装部署Zabbix3.4

    1.关闭selinux和firewall 1.1检测selinux是否关闭 [root@localhost ~]# getenforce  Disabled                       ...