[MAUI程序设计]界面多态与实现 焦点观察
.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"> </frame>
BlogCardViewDesktop用于博客条目在桌面设备中的呈现,条目菜单右键菜单方式展开,我们配置FlyoutBase.ContextFlyout属性,作为卡片,用一个Frame框架包裹其内容。设置卡片的阴影,圆角,以及内边距。代码如下:
<frame HasShadow="True" Margin="0,10,0,10" CornerRadius="5" Padding="8"> </frame>
.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
,给数据源初始化一些数据,用于测试。
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开发知识!
关键词:
上一篇:学科特长怎么写综合素质评价(学科特长怎么写)|环球即时
下一篇:最后一页
广告
X 关闭
- [MAUI程序设计]界面多态与实现 焦点观察
- 学科特长怎么写综合素质评价(学科特长怎么写)|环球即时
- 【环球时快讯】爱柯迪(600933.SH)2022年度权益分派10派2.3元 股权登记日5月18日
- 当前要闻:湘潭县二中怎么样是在哪里教学质量
- 龙湾瑶南片5亩文体及停车场地块出租,年租16万元
- 世界通讯!完美世界竞技平台csgo_go die
- 焦点热议:小学一年级都需学哪些古诗 一年级语文学过的古诗有哪些?
- 上海进一步增强产业竞争力 推进无人驾驶车创新应用 全球热讯
- 世界滚动:李荣浩梦幻联动辽宁男篮 李荣浩沈阳演唱会人气旺
- 连长崎美柚都喊着夺冠,陈梦世乒赛竞争对手多,平野美宇威胁很大
- 环球讯息:爱妈妈不止在今天,有人一次订走三四束花,给妈妈们的惊喜都准备好了吗?
X 关闭
-
-
京张高铁每日开行17对冬奥列车
京张高铁每日开行17对冬奥列车 预计冬奥服务保障期运送运动员、技术官员、持票观众等20万人次 2月6日,2022北京新闻中心举行“北
-
-
北京冬奥会开幕式上 小学生朱德恩深情演绎《我和我的祖国》
北京冬奥会开幕式上 小学生朱德恩深情演绎《我和我的祖国》 9岁小号手苦练悬臂吹响颂歌 2月4日晚,在北京冬奥会开幕式上,9岁的
-
-
2022北京冬奥会开幕式这19首乐曲串烧不简单
多名指挥家列曲目单 再由作曲家重新编曲 本报专访冬奥开幕式音乐总监赵麟 开幕式这19首乐曲串烧不简单 “二十四节气”倒计时、
-
-
“一墩难求” 冰墩墩引爆购买潮
设计师:没想到冰墩墩成爆款一墩难求冰墩墩引爆购买潮 北京冬奥组委:会源源不断供货北京冬奥会吉祥物冰墩墩近日引爆购买潮,导致一墩难求