大师学SwiftUI第16章 - UIKit框架集成
其它相关内容请见虚拟现实(VR)/增强现实(AR)&visionOS开发学习笔记
SwiftUI是一套新框架,因此并没有包含我们构建专业应用所需的所有工具。这意味着我们会需要求助于UIKit(移动设备)和AppKit(Mac电脑)等原生框架所提供的工具。
我们已介绍过UIKit。它是一套SwiftUI在后台实现用于创建大部分视图和控件的框架。有些UIKit类用于运行应用(UIApplication
)、加载图片(UIImage
)、管理设备(UIDevice
)及窗口(UIWindow
),而另一些定义用于设置应用和场景(UiApplicationDelegate
和UIWindowSceneDelegate
)的代理。当然,该框架提供了我们创建界面所需的所有工具,包括两个创建和管理视图的基本类,称为UIView
和UIViewController
。
最后面这两个类,UIView
和UIViewController
,是在我们需要对SwiftUI界面添加UIKit特性时需要实现的类。UIView的子类用于在屏幕上显示信息,比如标签和图像,以及创建控件,比如按钮、滑块和开关。而UIViewController
类的子类用于展示视图及包含处理它们的值和与用户交互的必要功能。为了将这些工具集成到SwiftUI界面中,SwiftUI框架定义了两个协议:UIViewRepresentable
和UIViewControllerRepresentable
。
Representable视图
UIViewRepresentable
协议定义了一结构体,作为UIView
类或其子类所创建对象的一层封装(wrapper)。实现这一协议的结构体可表示SwiftUI界面中的一个UIKit视图。创建及管理UIKit视图,结构体必须要实现如下的方法。
-
makeUIView(context: Context)
:该方法创建一个UIKit视图并返回。context
参数是对提供视图状态信息的UIViewRepresentableContext
类型结构体的引用。 -
updateUIView(UIViewType, context: Context)
:该方法通过一个绑定属性使用SwiftUI界面所提供的信息更新UIKit视图。第一个参数是对UIKit视图的引用,context
参数是对提供视图状态信息的UIViewRepresentableContext
类型结构体的引用。 -
dismantleUIView(UIViewType, coordinator: Coordinator)
:该类型方法为释放视图做准备工作。第一个参数是UIKit视图的指针,coordinator
参数是将值发回SwiftUI界面的对象。 -
makeCoordinator()
:该方法创建的对将UIKit视图的信息传回给SwiftUI界面。
要在SwiftUI接口中包含一个UIView
对象(或其子类所创建对象),我们需要定义一个实现UIViewRepresentable
协议的结构体,并实现上述的方法。makeUIView()
和updateUIView()
方法是必选的。在makeUIView()
方法中,我们必须创建并返回一个UIKit视图实例,updateUIView()
法用于使用SwiftUI界面中所获取的值来更新视图。
下例中创建了一个蓝色背景的UIKit视图,并在SwiftUI视图中包含它。只需要用makeUIView()
方法创建UIKit视图,但同时也要实现updateUIView()
方法,因为协议有相应要求。
示例16-1:准备一个供SwiftUI使用的UIKit视图
struct MyCustomView: UIViewRepresentable {func makeUIView(context: Context) -> some UIView {let view = UIView()view.backgroundColor = UIColor(.blue)return view}func updateUIView(_ uiView: UIViewType, context: Context) {}
}
在每次创建新的MyCustomView
结构体实例时调用makeUIView()
方法。在该方法中,我们创建一个UIView
视图,将背景色设置为蓝色并返回。因此,每次我们创建MyCustomView
结构体实例的时候,会创建一个UIView
并添加到SwiftUI实例之中,如下例所示。
示例16-2:在SwiftUI视图中显示UIKit视图
struct ContentView: View {var body: some View {VStack {MyCustomView().frame(width: 200, height: 150).padding()Spacer()}}
}
representable
视图默认为弹性大小,但可以通过SwiftUI修饰符进行修改。本例中,我们使用frame()修饰符赋了固定的宽和高。
图16-1:SwiftUI界面中的UIKit视图
UIView
对象创建一个空视图,但我们也可以实现接收其它用户输入的视图,如文本框、开关等。我们使用updateUIView()
方法将SwiftUI视图中的值传递给UIKit视图。但如果想将UIKit视图中的值传递给SwiftUI界面,则需要实现makeCoordinator()
方法。在这个方法中,我们必须创建一个coordinator
对象实例并返回。coordinator
是一个可将UIKit视图中的信息回发给Swift界面的对象,通常通过修改绑定属性实现。如何处理这些值取决于我们操作的UIkit视图的类型。例如,UITexView
视图创建一个用户可输入多行文本的输入框,和SwiftUI中的TextEditor
视图一样。这个视图通过调用代理方法上报改变。因此,要获取用户在UITexView
中插入的文本并在SwiftUI中进行处理,我们需要创建一个实现UITextViewDelegate
协议的coordinator
类,并实现其方法。下例演示了如何创建一个UIViewRepresentable
结构体来操作这个类。
示例16-3:对SwiftUI视图发送和接收值
import SwiftUIstruct TextView: UIViewRepresentable {@Binding var input: Stringfunc makeUIView(context: Context) -> UITextView {let view = UITextView()view.backgroundColor = UIColor.yellowview.font = UIFont.systemFont(ofSize: 17)view.delegate = context.coordinatorreturn view}func updateUIView(_ uiView: UITextView, context: Context) {uiView.text = input}func makeCoordinator() -> CoordinatorTextView {return CoordinatorTextView(input: $input)}
}class CoordinatorTextView: NSObject, UITextViewDelegate {@Binding var inputCoordinator: Stringinit(input: Binding<String>) {self._inputCoordinator = input}func textViewDidChange(_ textView: UITextView) {inputCoordinator = textView.text}
}
本示例创建了一个名为TexView
的UIViewRepresentable
结构体。结构体中有一个名为input
的@Binding
属性,接收值并传递给SwiftUI视图。在TexView
结构体定义的下面,我们定义了一类名为CoordinatorTextView
的类。这是我们视图的coordinator
,任务是将值发回给SwiftUI视图。为此,我们通过一个关联TexView
结构体中定义的input属性的@Binding
属性对其初始化,然后实现一个代理方法,在用户插入或删了字符时执行。该方法获取输入框中的当前文本,因此我们将其赋值给那个@Binding
属性,发回到SwiftUI视图。UIViewRepresentable
结构体初始化时。CoordinatorTextView
对象通过makeCoordinator()
方法创建,因而该结构体在一开始就可接收和发送值了。
SwiftUI界面中的视图实现和之前相同。只需对存储添加一个@State
属性,将值传递给input
属性。
示例16-4:对UIKit视图接收和发送值
struct ContentView: View {@State private var inputText: String = "Initial text"var body: some View {VStack {HStack {Text(inputText)Spacer()Button("Clear") {inputText = ""}}TextView(input: $inputText)}.padding()}
}
以上视图定义了一个名为inputText
的@State
属性、一个显示其值的Text
视图、一个用空字符串替换当前值的按钮以及自定义TextInput
视图的一个实例。该视图获取到inputText
属性的指针,与representable
视图中的输入属性相关联,这样便可以来回传递值 。
用户在编辑器中输入或删除字符时,representable
视图调用coordinator
中的textViewDidChange()
方法,该方法将文本视图中的当前值赋给inputCoordinator
属性,也就赋给了input
属性。这意味着值在@State
属性中可用 ,并且Text
视图可将其显示到屏幕上。而在用户点击Clear
按钮时,我们将空字符串赋值给@State
属性,系统执行representable
视图中的updateUIView()
方法,关联@State
属性的@Binding
属性的值被赋给了视图,文本视图被清除。
图16-2:SwiftUI界面中的UITexView
视图
✍️跟我一起做:创建一个多平台项目。根据示例16-3中的代码创建一个名为TextView.swift
的文件。根据示例16-4中的代码更新ContextView
视图。插入一段文本。应该会看到顶部的文本发生了变化。点击Clear
按钮。会看到文本视图及顶部的视图中的文本被删除。
注意:UIViewRepresentable
结构体可表示任意UIKit视图。实现会取决于想使用的视图。我们会在后续的章节中看到更多的示例。要学习UIKit及UiKit视图,可参阅UIKit for MasterMinds等书。
Representable视图控制器
UIViewControllerRepresentable
协议定义一个结构体,作为对UIViewController
类及子类所创建对象的封装。该类展示一个可包含其它视图的视图,为窗口或整个屏幕定义界面,类似SwiftUI文件中所定义的视图。实现UIViewControllerRepresentable
协议的结构体可在一个SwiftUI界面中展示这些视图控制器。创建及管理这个视图控制器,结构体必须实现如下方法。
-
makeUIViewController(context: Context)
:该方法创建一个UIKit视图控制器并返回。context
参数是UIViewControllerRepresentableContext
类型结构体的一个指针,提供视图控制器状态的相关信息。 -
updateUIViewController(UIViewControllerType, context: Context)
:该方法使用SwiftUI界面所提供的信息更新UIKit视图控制器。第一个参数是UIKit视图控制器的指针,context
参数是UIViewControllerRepresentableContext
类型结构体的一个指针,提供视图控制器状态的相关信息。 -
dismantleUiViewController(UIViewControllerType, coordinator: Coordinator)
:该类型方法为释放视图控制器做准备。第一个参数是UIKit视图控制器的指针,coordinator
参数是是将值发回SwiftUI界面的对象。 -
makeCoordinator()
:该方法创建将UIKit视图控制器信息传回SwiftUI界面的对象。
UIKit视图控制器由UIViewController
的子类创建。文件通过File菜单创建 ,但选取UIKit子类的选项称为Cocoa Touch Class。选择该选项后,Xcode会显示一个可输入文件名称的窗口,并可选择所创建子类相应的父类。本例中我们创建了一个继承UIViewController
类的DetailViewController
类。
示例16-5:创建一个UIKit视图控制器
import UIKitclass DetailViewController: UIViewController {override func viewDidLoad() {super.viewDidLoad()let label = UILabel()label.frame = CGRect(x: 20, y: 16, width: 250, height: 30)label.font = UIFont.systemFont(ofSize: 30)label.text = "Hello World!"view.addSubview(label)}
}
在初始化视图控制器类时,它创建一个表示界面的视图,将其赋值给view
属性,并调用viewDidLoad()
方法来告诉我们的代码视图已就绪。在这个方法中,我们可以执行所有必要的任务来初始化视图。在本例中,我们创建一个UILabel
对象来在屏幕上显示文本。该对象类似SwiftUI的Text
视图,但需要做一些配置。在我们的例子中,通过将CGRect
值赋给frame
属性来为其设置位置和大小,字体大小定义为30像素,赋值给要显示的文本,通过addSubview()
方法将其添加给视图控制器的视图。
现在有了视图控制器,我们需要创建一个representable
视图将其转为SwiftUI视图,如下例所示:
示例16-6:为UIKit视图控制器创建representable
视图
import SwiftUIstruct MyViewController: UIViewControllerRepresentable {func makeUIViewController(context: Context) -> DetailViewController {let controller = DetailViewController()return controller}func updateUIViewController(_ uiViewController: UIViewControllerType, context: Context) {}
}
UIViewControllerRepresentable
协议类似UIViewRepresentable
。我们必须定义一个实现此协议的结构体,然后添加创建和更新视图所需的方法。在本例中,我们仅定义了makeUIViewController()
,因为只需要创建视图控制器的实例,来在屏幕上进行显示。以下的SwiftUI视图在用户点击按钮时在NavigationStack
视图中加载该视图控制器。
图16-7:通过SwiftUI视图加载一个UIKit视图控制器
struct ContentView: View {var body: some View {NavigationStack {VStack {NavigationLink("Open UIKit View", destination: {MyViewController()}).buttonStyle(.borderedProminent)Spacer()}.padding()}}
}
以上视图包含一个加载MyViewController
结构体实例的NavigationLink
视图,因而可以从初始视图导航至结构体所创建的视图控制器,和常规SwiftUI视图的操作一样。
图16-3:SwiftUI界面中的UIKit视图控制器
✍️跟我一起做:选择屏幕顶部菜单中的File选项,新建一个文件。点击iOS版块中的Cocoa Touch Class图标,创建一个UIKit文件。在SubClass选项中选择UIViewController
。插入名称DetailViewController
并点击Next进行保存。使用示例16-5中的代码更新DetailViewController
。用示例16-6中的代码创建一个名为MyViewController.swift
的文件。使用示例16-7中的代码更新ContentView
视图。运行应用,点击按钮打开UIKit视图控制器。
代码请见:GitHub仓库
相关文章:

大师学SwiftUI第16章 - UIKit框架集成
其它相关内容请见虚拟现实(VR)/增强现实(AR)&visionOS开发学习笔记 SwiftUI是一套新框架,因此并没有包含我们构建专业应用所需的所有工具。这意味着我们会需要求助于UIKit(移动设备)和AppKit(Mac电脑)等原…...

7.docker运行redis容器
1.准备redis的配置文件 从上一篇运行MySQL容器我们知道,需要给容器挂载数据卷,来持久化数据和配置,相应的redis也不例外。这里我们以redis6.0.8为例来实际说明下。 1.1 查找redis的配置文件redis.conf 下面这个网址有各种版本的配置文件供…...

unity教程
前言 伴随游戏行业的兴起,unity引擎的使用越来越普遍,本文章主要记录博主本人入门unity的相关记录大部分依赖siki学院进行整理。12 一、认识unity引擎? 1、Unity相关信息: Unity的诞生:https://www.jianshu.com/p/550…...

未定义与 ‘double‘ 类型的输入参数相对应的函数 ‘Link‘
报错 检查对函数"Link"得调用中是否缺失参数或参数数据类型不正确。 未定义与"double"类型的输入参数相对应的函数"Link"。 问题描述 网上搜了搜一般说是toolbox没有下载导致的,相当于调用的包本地没有。 但是我看看了 Robotics…...

为什么Transformer模型中使用Layer Normalization(Layer Norm)而不是Batch Normalization(BN)
❤️觉得内容不错的话,欢迎点赞收藏加关注😊😊😊,后续会继续输入更多优质内容❤️ 👉有问题欢迎大家加关注私戳或者评论(包括但不限于NLP算法相关,linux学习相关,读研读博…...

Vite - 配置 - 文件路径别名的配置
为什么要配置别名 别名的配置,主要作用是为了缩短代码中的导入路径。例如有如下的项目目录: project-name| -- src| -- a| --b| --c| --d| --e| -- abc.png| -- index.html| -- main.js如果想在 main.js 文件中使用 abc.png ,则使用的路径是 ࿱…...

phpStorm Xdebug调试 加FireFox浏览器
步骤1: [Xdebug] zend_extension“D:\phpstudy_pro\Extensions\php\php5.4.45nts\ext\php_xdebug.dll” xdebug.collect_params1 xdebug.collect_return1 xdebug.remote_enableOn xdebug.remote_hostlocalhost xdebug.remote_port9001 xdebug.remote_handlerdbgp ;…...

多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测
多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测 目录 多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 …...

linux配置固定ip(两种方法)
首先刚下载的vm,刚创建的虚拟机,肯定是需要配置ip的 其次以前我的每次都是设置自动ip,这样每次登录都会自动获取ip地址,并且每次的ip都不相同。 ~方法: 开机登陆后 1)Cd /etc/sysconfig/network-scripts 2)Vi ifcf…...

什么是缓存雪崩、击穿、穿透?
背景 数据一般是存储于数据库中,数据库中的数据都是存在磁盘上的,磁盘读写的速度相较于内存或者CPU中的寄存器来说是非常慢的了。 如果用户的请求都直接访问数据库的话,请求数量一上来,数据库很容易就崩溃了,所以为了…...

可以通过电脑远程控制安卓设备的软件
有些时候,我们需要用电脑远程控制安卓设备,比如远程维护门店设备、安卓系统的户外广告牌等等。我们来探索和比较几款允许用户通过电脑远程控制安卓设备的软件。 1、Splashtop Business Splashtop 是一种多功能远程访问解决方案,以其高性能流…...

HP惠普暗影精灵9笔记本电脑OMEN by HP Transcend 16英寸游戏本16-u0000原厂Windows11系统
惠普暗影9恢复出厂开箱状态,原装出厂Win11-22H2系统ISO镜像 下载链接:https://pan.baidu.com/s/17ftbBHEMFSEOw22tnYvPog?pwd91p1 提取码:91p1 适用型号:16-u0006TX、16-u0007TX、16-u0008TX、16-u0009TX、16-u0017TX 原厂系…...

vue2+elementUI 仿照SPC开发CPK分析工具
源码地址请访问 Vue CPK分析工具页面设计源码(支持左右可拖拽和表格可编辑、复制粘贴)仿照SPC开发-CSDN博客...

云ES使用集群限流插件(aliyun-qos)
aliyun-qos插件是阿里云Elasticsearch团队自研的插件,能够提高集群的稳定性。该插件能够实现集群级别的读写限流,在关键时刻对指定索引降级,将流量控制在合适范围内。例如当上游业务无法进行流量控制时,尤其对于读请求业务,可根据aliyun-qos插件设置的规则,按照业务的优先…...

2023.11.17 hadoop之HDFS进阶
目录 HDFS的机制 元数据简介 元数据存储流程:namenode 生成了多个edits文件和一个fsimage文件 edits和fsimage文件 SecondaryNameNode辅助NameNode的方式: HDFS的存储原理 写入数据原理: 发送写入请求,获取主节点同意,开始写入,写入完成 读取数据原理:发送读取请求,获取…...

如何在el-tree懒加载并且包含下级的情况下进行数据回显-01
在项目中做需求,遇到一个比较棘手的问题,el-tree懒加载在包含下级的时候,需要做回显,将选中的数据再次勾选上,在处理这个需求的时候有两点是比较困难的: el-tree是懒加载的,包含下级需要一层一…...

系列六、JVM的内存结构【栈】
一、产生背景 由于跨平台性的设计,Java的指令都是根据栈来设计的,不同平台的CPU架构不同,所以不能设计为基于寄存器的。 二、概述 栈也叫栈内存,主管Java程序的运行,是在线程创建时创建,线程销毁时销毁&…...

技巧篇:在Pycharm中配置集成Git
一、在Pycharm中配置集成Git 我们使用git需要先安装git工具,这里给出下载地址,下载后一路直接安装即可: https://git-for-windows.github.io/ 0. git中的一些常用词释义 Repository name: 仓库名称 Description(可选):…...

Yolov5
Yolov5 Anchor 1.Anchor是啥? anchor字面意思是锚,是个把船固定的东东(上图),anchor在计算机视觉中有锚点或锚框,目标检测中常出现的anchor box是锚框,表示固定的参考框…...

36、Flink 的 Formats 之Parquet 和 Orc Format
Flink 系列文章 1、Flink 部署、概念介绍、source、transformation、sink使用示例、四大基石介绍和示例等系列综合文章链接 13、Flink 的table api与sql的基本概念、通用api介绍及入门示例 14、Flink 的table api与sql之数据类型: 内置数据类型以及它们的属性 15、Flink 的ta…...

Docker 笔记(一)--安装
Docker 笔记(一)–安装 记录Docker 安装操作记录,便于查询。 参考 链接: Docker 入门到实战教程(二)安装Docker链接: docker入门(利用docker部署web应用)链接: 阿里云容器镜像服务/镜像加速器/操作文档链接: 网易镜像中心链接: 阿里云镜像…...

endnote20如何导入已经下载好的ris和pdf文件
查看此链接 1 文献导入 1.1 PDF导入 (1)方法一 打开:菜单栏–>Import–>FIle或folder 单个导入PDF或导入一个文件夹的PDF或通过拖曳多个PDF进入空白处完成导入 1.3 导入已经整理好的文献资料 已有的ris文件 打开:菜单栏–…...

x程无忧sign逆向分析
x程无忧sign逆向分析: 详情页sign: 详情页网站: import base64 # 解码 result base64.b64decode(aHR0cHM6Ly9qb2JzLjUxam9iLmNvbS9ndWFuZ3pob3UvMTUxODU1MTYyLmh0bWw/cz1zb3Vfc291X3NvdWxiJnQ9MF8wJnJlcT0zODQ4NGQxMzc2Zjc4MDY2M2Y1MGY2Y…...

Rust8.1 Smart Pointers
Rust学习笔记 Rust编程语言入门教程课程笔记 参考教材: The Rust Programming Language (by Steve Klabnik and Carol Nichols, with contributions from the Rust Community) Lecture 15: Smart Pointers src/main.rs use crate::List::{Cons, Nil}; use std::ops::Deref…...

MATLAB与Excel的数据交互
准备阶段 clear all % 添加Excel函数 try Excel=actxGetRunningServer(Excel.Application); catch Excel=actxserver(Excel.application); end % 设置Excel可见 Excel.visible=1; 插入数据 % % 激活eSheet1 % eSheet1.Activate; % 或者 % Activate(eSheet1); % % 打开…...

使用.NET 4.0、3.5时,UnmanagedFunctionPointer导致堆栈溢出
本文介绍了使用.NET 4.0、3.5时,UnmanagedFunctionPointer导致堆栈溢出的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧! 问题描述 我在带有try catch块的点击处理程序中有一个简单的函数。…...

jbase打印导出实现
上一篇实现了虚拟M层,这篇基于虚拟M实现打印导出。 首先对接打印层 using Newtonsoft.Json; using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Text; using System.Threading.Tasks; using System.Xml;namesp…...

特征缩放和转换以及自定义Transformers(Machine Learning 研习之九)
特征缩放和转换 您需要应用于数据的最重要的转换之一是功能扩展。除了少数例外,机器学习算法在输入数值属性具有非常不同的尺度时表现不佳。住房数据就是这种情况:房间总数约为6至39320间,而收入中位数仅为0至15间。如果没有任何缩放,大多数…...

前端算法面试之堆排序-每日一练
如果对前端八股文感兴趣,可以留意公重号:码农补给站,总有你要的干货。 今天分享一个非常热门的算法--堆排序。堆的运用非常的广泛,例如,Python中的heapq模块提供了堆排序算法,可以用于实现优先队列…...

C++之set/multise容器
C之set/multise容器 set基本概念 set构造和赋值 #include <iostream> #include<set> using namespace std;void PrintfSet(set<int>&s) {for(set<int>::iterator it s.begin();it ! s.end();it){cout<<*it<<" ";}cout&l…...