[MAUI程序设计]界面多态与实现 焦点观察

来源:2023-05-14 18:39:43    时间:博客园
目录需求一:在不同设备上使用不同 UI 外观定义控件 UI 外观基于平台自定义配置需求二:在不同数据类别中使用不同的 UI 外观定义视图 UI 外观创建数据模板创建选择器定义数据需求三:在不同数据状态中使用不同的 UI 外观使用绑定模型更改控件的外观使用视觉状态更改控件的外观项目地址.NET MAUI 实现界面多态有很多种方式,今天主要来说说在日常开发中常见的需求该如何应对。需求一:在不同设备上使用不同 UI 外观

.NET MAUI是一个跨平台的UI框架,可在一个项目中开发Android、iOS、Windows、MacOS等多个平台的应用。在不同设备上我们希望应用的界面或交互方式能够有所不同。

比如在本示例中,我们希望博客条目的菜单使用平台特性的交互方式以方便触屏或鼠标的操作:比如手机设备中博客条目的菜单使用侧滑方式呈现,而在桌面设备中使用右键菜单呈现。


【资料图】

要实现不同平台下的控件外观,我们可以定义一个ContentView控件,然后在不同平台上使用不同的控件模板(ControlTemplate)

控件模板(ControlTemplate)是我们的老朋友了,早在WPF时代就已经出现了,它可以完全改变一个控件的可视结构和外观,与使用Style改变控件外观样式和行为样式不同,使用Style只能改变控件已有的属性。

定义控件 UI 外观

首先用控件模板定义博客条目的外观,“博客条目”是包含博客标题,内容,以及发布时间等信息的卡片,视觉上呈现圆角矩形的白色不透明卡片效果。

博客条目控件是一个基于ContentView控件

                                               

在页面的资源中,添加如下两个ControlTemplate模板,分别用于手机设备和桌面设备。

BlogCardViewPhone用于博客条目在手机设备中的呈现,条目菜单侧滑栏方式展开,我们配置SwipeView控件,作为卡片,用一个Frame框架包裹其内容。设置卡片的阴影,圆角,以及内边距。

代码如下

                                                                                                                                            <frame  HasShadow="True"                    Margin="0,10,0,10"                    CornerRadius="5"                    Padding="8">                                                                            
BlogCardViewDesktop用于博客条目在桌面设备中的呈现,条目菜单右键菜单方式展开,我们配置FlyoutBase.ContextFlyout属性,作为卡片,用一个Frame框架包裹其内容。设置卡片的阴影,圆角,以及内边距。

代码如下:

    <frame HasShadow="True"            Margin="0,10,0,10"            CornerRadius="5"            Padding="8">                                                                                                                                    

.NET MAUI 提供了ContentPresenter作为模板控件中的内容占位符,用于标记模板化自定义控件或模板化页面要显示的内容将在何处显示。

各平台模板中的将显示控件的Content属性,也就是将TextBlogView中定义的内容,放到ContentPresenter处。

基于平台自定义配置

.NET MAUI 提供了 OnPlatform 标记扩展和 OnIdiom 标记扩展。以便在不同平台上使用不同的控件模板。

通过 OnPlatform 标记扩展可基于每个平台控件属性:

属性描述
Default平台的属性的默认值。
Android属性在 Android 上应用的值。
iOS属性在 iOS 上应用的值。
MacCatalyst设置为要在 Mac Catalyst 的值。
Tizen属性在 Tizen 平台的值。
WinUI属性在 WinUI 的值。

通过 OnIdiom 标记扩展可基于设备语义上的控件属性

属性描述
Default设备语义的属性的默认值。
Phone属性在手机上应用的值。
Tablet属性在平板电脑的值。
Desktop设置为要在桌面平台的值。
TV属性在电视平台的值。
Watch属性在可穿戴设备(手表)平台的值。

在本示例中,我们使用OnIdiom标记扩展,分别为手机和桌面设备配置不同的模板。

                        
需求二:在不同数据类别中使用不同的 UI 外观

数据模板(DataTemplate)可以在支持的控件上(如:CollectionView)定义数据表示形式可以使用数据模板选择器(DataTemplateSelector)来实现更加灵活的模板选择。

DataTemplateSelector 可用于在运行时根据数据绑定属性的值来选择 DataTemplate。 这样可将多个 DataTemplate 应用于同一类型的对象,以自定义特定对象的外观。

相对于ControlTemplate方式,DataTemplateSelector是从Xamarin.Forms 2.1引入的新特性。

定义视图 UI 外观

创建两种视图和模板选择器:

TextBlog: 文本博客PhotoBlog: 图片博客

编写文本博客条目展示标题和内容,创建TextBlogView.xaml,定义如下:

                                               

编写图片博客条目展示标题和博客中图片的缩略图,创建PhotoBlogView.xaml,定义如下:

                                                                                                            
创建数据模板

在页面的资源中,添加各个视图创建数据模板(DataTemplate)类型的资源。

                                                                                                                                                                  
创建选择器

创建BlogDataTemplateSelector,根据博客的类型,返回不同的数据模板(DataTemplate)对象。

public class BlogDataTemplateSelector : DataTemplateSelector{    public object ResourcesContainer { get; set; }    protected override DataTemplate OnSelectTemplate(object item, BindableObject container)    {        if (item == null)        {            return default;        }        if (item is Blog)        {            var dataTemplateName = (item as Blog).Type;            if (dataTemplateName == null) { return default; }            if (ResourcesContainer == null)            {                return Application.Current.Resources[dataTemplateName] as DataTemplate;            }            return (ResourcesContainer as VisualElement).Resources[dataTemplateName] as DataTemplate;        }        return default;    }}

DataTemplate将在页面资源字典中被创建。若没有绑定ResourcesContainer,则在App.xaml中寻找。

同样, 将BlogDataTemplateSelector添加到页面的资源中

注意,此时BlogDataTemplateSelector.ResourcesContainer指向MainContentPage,显式设置MainPage对象的名称:x:Name="MainContentPage"

定义数据

我们定义一个Blog类, 用于表示博客条目,包含标题,内容,发布时间,图片等属性。

public class Blog {    public Blog()    {        PostTime = DateTime.Now;        State = BlogState.Edit;    }        public Guid NoteId { get; set; }    public string Title { get; set; }    public string Type { get; set; }    public BlogState State { get; set; }    public string Content { get; set; }    public List Images { get; set; }    public DateTime PostTime { get; set; }    public bool IsHidden { get; set; }}

定义博客列表的绑定数据源 ObservableCollection Blogs,给数据源初始化一些数据,用于测试。

private async void CreateBlogAction(object obj){    var type = obj as string;    if (type == "TextBlog")    {        var blog = new Blog()        {            NoteId = Guid.NewGuid(),            Title = type + " Blog",            Type = type,            Content = type + " Blog Test, There are so many little details that a software developer must take care of before publishing an application. One of the most time-consuming is the task of adding icons to your toolbars, buttons, menus, headers, footers and so on.",            State = BlogState.PreView,            IsHidden = false,        };        this.Blogs.Add(blog);    }    else if (type == "PhotoBlog")    {        var blog = new Blog()        {            NoteId = Guid.NewGuid(),            Title = type + " Blog",            Type = type,            Content = type + " Blog Test",            Images = new List() { "p1.png", "p2.png", "p3.png", "p4.png" },            State = BlogState.PreView,            IsHidden = false,        };        this.Blogs.Add(blog);    }}

设置博客列表控件CollectionView绑定的数据源为Blogs,并设置数据模板选择器为BlogDataTemplateSelector

                                                                        

则列表中的每个博客条目将根据博客类型,使用不同的数据模板进行渲染。

效果如下:

需求三:在不同数据状态中使用不同的 UI 外观

此功能没有一个固定的解决方案,可以根据实际情况,选择合适的方式实现。

比如在本项目中,博客存在编辑和发布两个状态

public enum BlogState{    Edit,    PreView}
使用绑定模型更改控件的外观

最简单的方式是用IsVisible来控制控件中元素的显示和隐藏。

在本示例中,TextBlogView需要对编辑中的状态和预览中的状态进行区分。EnumToBoolConverter是枚举到bool值的转换器,它返回当前绑定对象的State属性与指定的BlogState枚举项是否一致,详情请查看 .NET MAUI 社区工具包

                                                                                                    

编辑状态:

发布状态:

使用视觉状态更改控件的外观

还可以使用定义自定义视觉状态对界面进行控制。

在本示例中,使用VisualStateManager定义了两个视觉状态,分别对应Label的编辑状态和发布状态,当State属性的值发生变化时,会触发对应的视觉状态。

编辑状态:

发布状态:

项目地址

Github:maui-learning

关注我,学习更多.NET MAUI开发知识!

关键词:

文章推荐

  • 锐龙7000系列首发阵容曝光:核心数不变 功耗翻倍

    不出意外的话,AMD将于本月底正式宣布Zen4架构的锐龙7000系列处理器,9月15日上市开卖,抢先Intel 13代酷睿一步。今天,锐龙7000系列的首

    快科技 2022-08-05
  • 赏传统年俗逛非遗庙会 铜官窑古镇重温传统民俗年

    中新网长沙2月6日电 (潘杏琼)在多地倡导就地过年的环境下,位于长沙市城北的铜官窑古镇景区,从1月24日至2月15日举行中国年·湖湘味·铜官

    中新网 2022-02-07
  • 哈尔滨铁路迎节后返程高峰 推出复工专列服务

    中新网哈尔滨2月6日电 (周晓舟 记者 史轶夫)中国铁路哈尔滨局有限公司6日发布消息,哈尔滨铁路迎来春节后返程客流高峰,6日至7日预

    中新网 2022-02-07
  • 冬奥动车组设5G超高清演播室 “瑞雪迎春”号智能化人性化结合

    中新网北京2月6日电 (记者 刘文曦)在时速350公里的高铁列车上首设5G超高清演播室,为北京冬奥会量身定制的新型奥运版智能复兴号动车组瑞

    中新网 2022-02-07
  • 中欧班列“签证官”:日行10公里 用锤子“听诊”

    (新春走基层)中欧班列“签证官”:日行10公里 用锤子“听诊”  中新网郑州2月6日电 题:中欧班列“签证官”:日行10公里,用锤子“

    中新网 2022-02-07
  • 广告

    X 关闭

  • 科技
  • 数码
  • more+

    X 关闭

  • 众测
  • more+

    京张高铁每日开行17对冬奥列车

      京张高铁每日开行17对冬奥列车  预计冬奥服务保障期运送运动员、技术官员、持票观众等20万人次  2月6日,2022北京新闻中心举行“北

    北京冬奥会开幕式上 小学生朱德恩深情演绎《我和我的祖国》

      北京冬奥会开幕式上 小学生朱德恩深情演绎《我和我的祖国》  9岁小号手苦练悬臂吹响颂歌  2月4日晚,在北京冬奥会开幕式上,9岁的

    2022北京冬奥会开幕式这19首乐曲串烧不简单

      多名指挥家列曲目单 再由作曲家重新编曲 本报专访冬奥开幕式音乐总监赵麟  开幕式这19首乐曲串烧不简单  “二十四节气”倒计时、

    “一墩难求” 冰墩墩引爆购买潮

    设计师:没想到冰墩墩成爆款一墩难求冰墩墩引爆购买潮 北京冬奥组委:会源源不断供货北京冬奥会吉祥物冰墩墩近日引爆购买潮,导致一墩难求