在 iOS 13 中覆盖 TraitCollection

问题描述 投票:0回答:2

在我的初始视图控制器中,我有一个

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()
方法中使用相同的代码,但没有效果。

ios swift uitabbarcontroller uitraitcollection
2个回答
0
投票

我不确定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()
    }

}

0
投票

使用 iOS 17 或更高版本时,可以通过设置视图控制器的

traitOverrides
属性来直接覆盖特征集合属性。示例:

override func viewDidLoad() {
    super.viewDidLoad()
    
    traitOverrides.horizontalSizeClass = .compact
}
© www.soinside.com 2019 - 2024. All rights reserved.