如何检查视图控制器是否属于 UIHostingController 类型
How to check if a view controller is of type UIHostingController
我有一个 UIHostingController,它托管一个名为 CatalogView 的 SwiftUI 视图。
显示它时,附加了一个环境对象,所以基本上从 UIKit 中显示如下:
let rootCatalogView = CatalogView()
let appState = AppState.get()
let catalogView = UIHostingController(rootView: rootCatalogView.environmentObject(appState))
navigationController.pushViewController(catalogView, animated: true)
稍后我需要检查此 UIHostingController 是否在 navigationController.viewControllers
的列表中
type(of:) 显示如下,哪种有意义:
UIHostingController<ModifiedContent<CatalogView, _EnvironmentKeyWritingModifier<Optional<AppState>>>>
像 vc.self 是 UIHostingController.Type 或 vc.self 是 UIHostingController< CatalogView >。两者都输入 return false(vc 是 navigationController.viewControllers
以下显然有效,return是正确的,但 UIHostingController 初始化中的任何更改都会更改其类型
vc.isKind(of: UIHostingController<ModifiedContent<CatalogView, _EnvironmentKeyWritingModifier<Optional<StoreManager>>>>.self)
如何检查视图控制器是否为 UIHostingController 类型?
或者至少我怎样才能将控制器投射到 UIHostingController 以便我可以检查它的根视图?
由于通用参数,我们无法在不知道完整约束的情况下转换 ViewController 来查找它是否是 UIHostingController
。
我应该指出,这不是一个理想的修复,它实际上只是一个解决方法。
UIHostingController
是 UIViewController
的子 class,因此我们可以执行以下操作。
在 UIViewController
上创建计算 属性,returns 用于创建 UIViewController
的 class 的名称。这使我们可以在 UINavigationController
中包含的 ViewControllers
列表中搜索一些内容
extension UIViewController {
var className: String {
String(describing: Self.self)
}
}
创建一些 UIViewController 子classes 和我们的 UIHostingController
class FirstViewController: UIViewController {}
class SecondViewController: UIViewController {}
class MyHostingController<Content>: UIHostingController<Content> where Content : View {}
let first = FirstViewController()
let second = SecondViewController()
let hosting = UIHostingController(rootView: Text("I'm in a hosting controller"))
let myHosting = MyHostingController(rootView: Text("My hosting vc"))
然后我们可以将这些添加到 UINavigationController
。
let nav = UINavigationController(rootViewController: first)
nav.pushViewController(second, animated: false)
nav.pushViewController(hosting, animated: false)
nav.pushViewController(myHosting, animated: false)
现在我们的 UINavigationController 中有一些 ViewControllers,我们现在可以遍历它们并找到一个 ViewController,它的 className 包含我们正在寻找的内容。
for vc in nav.viewControllers {
print(vc.className)
}
这会将以下内容打印到控制台:
FirstViewController
SecondViewController
UIHostingController<Text>
MyHostingController<Text>
然后您可以 for-where
在层次结构中找到 ViewController。
for vc in nav.viewControllers where vc.className.contains("UIHostingController") {
// code that should run if its class is UIHostingController
print(vc.className)
}
for vc in nav.viewControllers where vc.className.contains("MyHostingController") {
// code that should run if its class is MyHostingController
print(vc.className)
}
正如我上面所说,这不是一个理想的解决方案,但它可能会对您有所帮助,直到有更好的转换方式而无需了解通用约束。
我有一个 UIHostingController,它托管一个名为 CatalogView 的 SwiftUI 视图。 显示它时,附加了一个环境对象,所以基本上从 UIKit 中显示如下:
let rootCatalogView = CatalogView()
let appState = AppState.get()
let catalogView = UIHostingController(rootView: rootCatalogView.environmentObject(appState))
navigationController.pushViewController(catalogView, animated: true)
稍后我需要检查此 UIHostingController 是否在 navigationController.viewControllers
的列表中type(of:) 显示如下,哪种有意义:
UIHostingController<ModifiedContent<CatalogView, _EnvironmentKeyWritingModifier<Optional<AppState>>>>
像 vc.self 是 UIHostingController.Type 或 vc.self 是 UIHostingController< CatalogView >。两者都输入 return false(vc 是 navigationController.viewControllers
以下显然有效,return是正确的,但 UIHostingController 初始化中的任何更改都会更改其类型
vc.isKind(of: UIHostingController<ModifiedContent<CatalogView, _EnvironmentKeyWritingModifier<Optional<StoreManager>>>>.self)
如何检查视图控制器是否为 UIHostingController 类型? 或者至少我怎样才能将控制器投射到 UIHostingController 以便我可以检查它的根视图?
由于通用参数,我们无法在不知道完整约束的情况下转换 ViewController 来查找它是否是 UIHostingController
。
我应该指出,这不是一个理想的修复,它实际上只是一个解决方法。
UIHostingController
是 UIViewController
的子 class,因此我们可以执行以下操作。
在 UIViewController
上创建计算 属性,returns 用于创建 UIViewController
的 class 的名称。这使我们可以在 UINavigationController
ViewControllers
列表中搜索一些内容
extension UIViewController {
var className: String {
String(describing: Self.self)
}
}
创建一些 UIViewController 子classes 和我们的 UIHostingController
class FirstViewController: UIViewController {}
class SecondViewController: UIViewController {}
class MyHostingController<Content>: UIHostingController<Content> where Content : View {}
let first = FirstViewController()
let second = SecondViewController()
let hosting = UIHostingController(rootView: Text("I'm in a hosting controller"))
let myHosting = MyHostingController(rootView: Text("My hosting vc"))
然后我们可以将这些添加到 UINavigationController
。
let nav = UINavigationController(rootViewController: first)
nav.pushViewController(second, animated: false)
nav.pushViewController(hosting, animated: false)
nav.pushViewController(myHosting, animated: false)
现在我们的 UINavigationController 中有一些 ViewControllers,我们现在可以遍历它们并找到一个 ViewController,它的 className 包含我们正在寻找的内容。
for vc in nav.viewControllers {
print(vc.className)
}
这会将以下内容打印到控制台:
FirstViewController
SecondViewController
UIHostingController<Text>
MyHostingController<Text>
然后您可以 for-where
在层次结构中找到 ViewController。
for vc in nav.viewControllers where vc.className.contains("UIHostingController") {
// code that should run if its class is UIHostingController
print(vc.className)
}
for vc in nav.viewControllers where vc.className.contains("MyHostingController") {
// code that should run if its class is MyHostingController
print(vc.className)
}
正如我上面所说,这不是一个理想的解决方案,但它可能会对您有所帮助,直到有更好的转换方式而无需了解通用约束。