iOS 横竖屏切换

所有 ViewController 都要继承自中间类,包括 TabBarController 和 NavigationController,当有这两者时,其 ChildViewController 的方向跟随 TabBarController 和 NavigationController。重写的方法直接 return topViewController 的对应值

Navigation

class BaseNavigationController: UINavigationController {
  
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return topViewController?.supportedInterfaceOrientations ?? self.supportedInterfaceOrientations
    }
    
    override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
        return topViewController?.preferredInterfaceOrientationForPresentation ?? self.preferredInterfaceOrientationForPresentation
    }
}

普通界面的基类返回当前 APP 的主要方向(一般为竖屏)

class BaseViewController: UIViewController {

    override var shouldAutorotate: Bool {
        return true
    }
    
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .portrait
    }
    
    override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
        return .portrait
    }
}

TabBar 返回自己的方向,避免在 TabBar 旋转

class HomeTabBarController: RAMAnimatedTabBarController {
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .portrait
    }
    
    override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
        return .portrait
    }
}

第一个 ViewController,不用加关于旋转屏的代码

class ViewController: BaseViewController {
    
    @objc func pushTUI() {
        self.navigationController?.pushViewController(ToViewController(), animated: true)
    }
    
    @objc func presentTUI() {
        let vc = ToViewController()
        vc.modalPresentationStyle = .overFullScreen
        self.present(vc, animated: true, completion: nil)
    }
    // MARK: - Life Cycle
    override func viewDidLoad() {
        super.viewDidLoad()
        commonLoad()
    }
    
    private func commonLoad() {
        view.addSubview(pushBtn)
        view.addSubview(presentBtn)
    }

    private lazy var pushBtn: UIButton = {
        let btn = UIButton(frame: CGRect(x: 60, y: 200, width: 80, height: 80))
        btn.backgroundColor = UIColor.blue
        btn.setTitle("Push", for: .normal)
        btn.addTarget(self, action: #selector(pushTUI), for: .touchUpInside)
        return btn
    }()
        
    private lazy var presentBtn: UIButton = {
        let btn = UIButton(frame: CGRect(x: 60, y: 300, width: 80, height: 80))
        btn.backgroundColor = UIColor.red
        btn.setTitle("Present", for: .normal)
        btn.addTarget(self, action: #selector(presentTUI), for: .touchUpInside)
        return btn
    }()
}

需要强制横屏后进入的 ViewController

class ToViewController: BaseViewController {
    
    // MARK: - 旋转屏
    override var prefersStatusBarHidden: Bool {
        // 隐藏状态栏
        return false
    }
    
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .landscapeRight
    }
    
    override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
        return .landscapeRight
    }
    
    // MARK: - 侧滑手势(Push 的需要加)
    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        navigationController?.interactivePopGestureRecognizer?.isEnabled = false
    }
    
    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        navigationController?.interactivePopGestureRecognizer?.isEnabled = false
    }
    
    override func dismiss(animated flag: Bool, completion: (() -> Void)? = nil) {
        // Push 的需要加
        UIDevice.current.setValue(NSNumber(value: UIInterfaceOrientation.portrait.rawValue), forKey: "orientation")
        super.dismiss(animated: flag, completion: completion)
    }
    
    @objc func dismissTUI() {
        self.dismiss(animated: true, completion: nil)
    }
    
    // MARK: - Life Cycle
    override func viewDidLoad() {
        super.viewDidLoad()
        commonLoad()
    }
    
    private func commonLoad() {
        // Push 的需要加
        UIDevice.current.setValue(NSNumber(value: UIInterfaceOrientation.landscapeRight.rawValue), forKey: "orientation")
        if presentingViewController != nil {
            view.backgroundColor = .red
        } else {
            view.backgroundColor = .blue

        }
        view.addSubview(dismissBtn)
    }
    
    private lazy var dismissBtn: UIButton = {
        let btn = UIButton(frame: CGRect(x: 60, y: 200, width: 80, height: 80))
        btn.backgroundColor = .black
        btn.setTitle("dismiss", for: .normal)
        btn.addTarget(self, action: #selector(dismissTUI), for: .touchUpInside)
        return btn
    }()
}

Tips

iOS 13 开始 present ViewController 时需要设置一下 modalPresentationStyle, 要在创建的时候设置,不要在viewDidLoad里设置

@objc func presentTUI() {
    let vc = ToViewController()
    vc.modalPresentationStyle = .overFullScreen
    self.present(vc, animated: true, completion: nil)
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,734评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,931评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,133评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,532评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,585评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,462评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,262评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,153评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,587评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,792评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,919评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,635评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,237评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,855评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,983评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,048评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,864评论 2 354

推荐阅读更多精彩内容

  • iOS 中横竖屏切换的功能,在开发iOS app中总能遇到。以前看过几次,感觉简单,但是没有敲过代码实现,最近又碰...
    零度_不结冰阅读 2,183评论 0 0
  • iOS 横竖屏切换解决方案 前言 在大多数项目中,App 的 UI 方向都是竖屏的,所以一般会在 target 中...
    yizhaorong阅读 15,359评论 3 68
  • 横竖屏切换,控制每一个Controller的横竖屏状态 前言: 在网上看了看,发现大多数博客都写过,但是都会有些小...
    张梓辰阅读 15,682评论 6 15
  • 网上关于横竖屏切换的资料很多,但是很容易踩到坑,不是屏幕不旋转,就是屏幕旋转后没有状态栏等,在写的小demo里屏幕...
    凌云01阅读 546评论 0 0
  • 导航控制器PUSH 需求:当前竖屏下情况下,导航栏直接PUSH出一个横屏的控制器,并在POP后返回竖屏状态。效果图...
    毅个天亮阅读 975评论 0 1