在我的初始视图控制器中,我有一个
UITabbarController
作为子视图控制器。
我想让
UITabbarController
显示其 UITabbar
,并使用具有 Compact HorizonSizeClass 的 TraitCollection,以便在选项卡栏中,图像和标题显示为垂直对齐而不是并排。
iOS13现在不支持重写UITabbarController的traitCollection getter, Xcode 给出以下警告。
override var traitCollection: UITraitCollection{
let current = super.traitCollection
let compact = UITraitCollection(horizontalSizeClass: .compact)
return UITraitCollection(traitsFrom: [current, compact])
}
MyTabbarController 类覆盖了 -traitCollection getter,这是不支持的。如果您尝试覆盖特征,则必须使用适当的 API。
在研究了合适的API之后,我发现了
open func setOverrideTraitCollection(_ collection: UITraitCollection?, forChild childViewController: UIViewController)
实现此功能后,我可以覆盖 myTabbarController 的特征集合,但只有在视图更改方向之后。仅当我将 viewWillTransition 重写为方法时,此 API 才有效。
override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
super.viewWillTransition(to: size, with: coordinator)
let currentTC = traitCollection
let compactTC = UITraitCollection(horizontalSizeClass: .compact)
let custom = UITraitCollection(traitsFrom: [currentTC, compactTC])
print("ovverride trait collections before transition")
setOverrideTraitCollection(custom, forChild: tabController)
}
我只能在设备旋转时覆盖这些特征。如果我尝试覆盖任何其他视图控制器生命周期方法中的特征集合,则此 API 将不起作用。最初加载视图时如何覆盖 TraitCollection? 我尝试在初始视图控制器的
viewDidLoad()
方法中使用相同的代码,但没有效果。
我不确定OP是否可以正常工作,但我最近遇到了同样的问题。就我而言,我需要将 iPhone 和 iPad 的设备方向视为相同,特别是在纵向方向上将 HorizontalSizeClass 设置为 .compact。
因为 setOverrideTraitCollection() 仅适用于子视图控制器,所以我必须将我的“主”视图控制器嵌入到另一个视图控制器(我称之为“根”视图控制器)中,并在根视图控制器中处理特征覆盖。 正如OP提到的,这需要在应用程序启动时和方向改变时发生。就我而言,我可以在prepareForSegue 中执行启动代码。不知道为什么将代码放在 viewDidLoad() 中对 OP 不起作用——也许是因为他没有为子视图控制器的视图调用 setNeedsLayout() 。
这是我的根视图控制器代码:
class RootViewController: UIViewController {
var masterViewController: MasterViewController?
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if segue.identifier == "MasterViewSegue" {
masterViewController = segue.destination as? MasterViewController
updateMasterViewTraits(for: CGSize(width: view.bounds.width, height: view.bounds.height))
}
}
override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
super.viewWillTransition(to: size, with: coordinator)
updateMasterViewTraits(for: size)
}
func updateMasterViewTraits(for size: CGSize) {
var orientationTraits: UITraitCollection
if size.width < size.height {
orientationTraits = UITraitCollection(traitsFrom:[UITraitCollection(horizontalSizeClass: .compact), UITraitCollection(verticalSizeClass: .regular)])
} else {
orientationTraits = UITraitCollection(traitsFrom:[UITraitCollection(horizontalSizeClass: .regular), UITraitCollection(verticalSizeClass: .compact)])
}
let traits = UITraitCollection(traitsFrom: [traitCollection, orientationTraits])
setOverrideTraitCollection(traits, forChild: masterViewController!)
masterViewController!.view.setNeedsLayout()
}
}
使用 iOS 17 或更高版本时,可以通过设置视图控制器的
traitOverrides
属性来直接覆盖特征集合属性。示例:
override func viewDidLoad() {
super.viewDidLoad()
traitOverrides.horizontalSizeClass = .compact
}