Force landscape mode in one ViewController using Swift
Asked Answered
B

21

90

I am trying to force only one view in my application on landscape mode, I am calling:

override func shouldAutorotate() -> Bool {
    print("shouldAutorotate")
    return false
}

override func supportedInterfaceOrientations() -> Int {
    print("supportedInterfaceOrientations")
    return Int(UIInterfaceOrientationMask.LandscapeLeft.rawValue)
}

override func preferredInterfaceOrientationForPresentation() -> UIInterfaceOrientation {
    return UIInterfaceOrientation.LandscapeLeft
}

The view is launched in the portrait mode, and keep rotating when I change the device orientation. The shouldAutorotate() method is never called.

Any help would be appreciated.

Bihar answered 20/11, 2014 at 10:55 Comment(2)
B
90

It may be useful for others, I found a way to force the view to launch in landscape mode:

Put this in the viewDidLoad():

let value = UIInterfaceOrientation.landscapeLeft.rawValue
UIDevice.current.setValue(value, forKey: "orientation")

and,

override var shouldAutorotate: Bool {
    return true
}
Bihar answered 21/11, 2014 at 10:1 Comment(6)
I’ve just tried using this and it didn’t work; although changing shouldAutorotate() to return false worked. Just to be clear: I wanted it to stay in Portrait no matter what. (Also changed UIInterfaceOrientation.LandscapeLeft to UIInterfaceOrientation.Portrait of course)Rizas
it's because your view load in Portrait mode and you want it to stay like that, my view load in Portrait mode and I want it to show in Landscape mode. If I put shouldAutorotate to NO, it will not show in LandscapeBihar
Thanks!! This worked great for me on an iOS9 app with Swift 2.2 to force portrait. I started with a 'parent' view in landscape, put the above code in viewDidLoad of the ViewController which i present modally from the 'parent', and when the modal view was presented, the view rotated to portrait. When i dismissed the modal view, the 'parent' remained in portrait (though i bet you could store a 'previous orientation' value before showing the modal and set the 'parent' view back to the previous orientation, if you wanted to swanky about it!Descender
Could this cause Apple to reject your app? orientation is a readonly property, and you're manipulating the value. I would think Apple would frown upon this.Sybil
If the device was initially oriented LandscapeRight, would you really want everything to suddenly appear upside down?Incarnate
worked for me on iOS 14, Swift 5.3, only by overriding shouldAutorotate.Adrianneadriano
A
71

Swift 4

override func viewDidLoad() {
    super.viewDidLoad()
    let value = UIInterfaceOrientation.landscapeLeft.rawValue
    UIDevice.current.setValue(value, forKey: "orientation")
}

override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
    return .landscapeLeft
}

override var shouldAutorotate: Bool {
    return true
}

If your view is embedded in a navigation controller, the above alone won't work. You have to cascade up by the following extension after the class definition.

extension UINavigationController {

override open var shouldAutorotate: Bool {
    get {
        if let visibleVC = visibleViewController {
            return visibleVC.shouldAutorotate
        }
        return super.shouldAutorotate
    }
}

override open var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation{
    get {
        if let visibleVC = visibleViewController {
            return visibleVC.preferredInterfaceOrientationForPresentation
        }
        return super.preferredInterfaceOrientationForPresentation
    }
}

override open var supportedInterfaceOrientations: UIInterfaceOrientationMask{
    get {
        if let visibleVC = visibleViewController {
            return visibleVC.supportedInterfaceOrientations
        }
        return super.supportedInterfaceOrientations
    }
}}


Swift 3

override func viewDidLoad() {
    super.viewDidLoad()
    let value = UIInterfaceOrientation.landscapeLeft.rawValue
    UIDevice.current.setValue(value, forKey: "orientation")
}

private func supportedInterfaceOrientations() -> UIInterfaceOrientationMask {
    return UIInterfaceOrientationMask.landscapeLeft
}

private func shouldAutorotate() -> Bool {
    return true
}
Admix answered 29/11, 2016 at 6:38 Comment(5)
This code doesn't work on iOS 12. App gets crash and shows the error like "Supported orientations has no common orientation with the application and [PlayerViewController shouldAutorotate] is returning YES"Riggle
@MayurRathod did you find the fix to ur crash.Tolle
@Tolle Yes I implemented orientation change code in delegate fileRiggle
@MayurRathod If you have time, could you please post your answerOrjonikidze
The Solution works fine for the iPhone but in iPad the presenting controller and hence the entire App is oriented to landscape. Any idea where can I be wrong?Sweetbrier
H
40

Swift 4 , Tested in iOS 11

You can specify the orientation in projectTarget -> General -> DeploymentInfo(Device Orientation) -> Portrait (Landscapeleft and Landscaperight are optional)

AppDelegate

    var myOrientation: UIInterfaceOrientationMask = .portrait
    func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask {
        return myOrientation
    }

LandScpaeViewController

override func viewDidLoad() {
        super.viewDidLoad()
        let appDelegate = UIApplication.shared.delegate as! AppDelegate
        appDelegate.myOrientation = .landscape
}

OnDismissButtonTap

let appDelegate = UIApplication.shared.delegate as! AppDelegate
 appDelegate.myOrientation = .portrait

Thats it. :)

Hounding answered 27/10, 2017 at 4:37 Comment(3)
supportedInterfaceOrientationsFor is getting called only if device is rotated once.Eckart
This is what I've been looking for, thanks! For anyone wondering the difference, this will present the new view in the rotation without animation. Where as setting the orientation on UIDevice will add a default rotation animation.Fenn
omg yess!! this works pretty well, thank you!Haemachrome
I
23

Using Swift 2.2

Try:

let value = UIInterfaceOrientation.LandscapeLeft.rawValue
UIDevice.currentDevice().setValue(value, forKey: "orientation")

Followed By:

UIViewController.attemptRotationToDeviceOrientation()

From Apple's UIViewController Class Reference:

Some view controllers may want to use app-specific conditions to determine what interface orientations are supported. If your view controller does this, when those conditions change, your app should call this class method. The system immediately attempts to rotate to the new orientation.

Then, as others have suggested, override the following methods as appropriate:

override func supportedInterfaceOrientations() -> UIInterfaceOrientationMask {
    return UIInterfaceOrientationMask.LandscapeLeft
}

override func shouldAutorotate() -> Bool {
    return true
}

I was having a similar issue with a signature view and this solved it for me.

Inconstant answered 6/6, 2016 at 14:42 Comment(1)
Thanks for this answer man, This comment > UIViewController.attemptRotationToDeviceOrientation() Is very important because sometimes controller does not know if the rotation is to be performed or not. But this tells it to perform the rotation accordingly.Sapsago
E
23

In AppDelegate add this:

//Orientation Variables
var myOrientation: UIInterfaceOrientationMask = .portrait

func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask {
    return myOrientation  
}

Add this in viewController, that want to change orientation:

override func viewDidLoad() {
        super.viewDidLoad()
        self.rotateToLandsScapeDevice()
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        self.rotateToPotraitScapeDevice()
    }

    func rotateToLandsScapeDevice(){
        let appDelegate = UIApplication.shared.delegate as! AppDelegate
        appDelegate.myOrientation = .landscapeLeft
        UIDevice.current.setValue(UIInterfaceOrientation.landscapeLeft.rawValue, forKey: "orientation")
        UIView.setAnimationsEnabled(true)
    }

    func rotateToPotraitScapeDevice(){
        let appDelegate = UIApplication.shared.delegate as! AppDelegate
        appDelegate.myOrientation = .portrait
        UIDevice.current.setValue(UIInterfaceOrientation.portrait.rawValue, forKey: "orientation")
        UIView.setAnimationsEnabled(true)
    }
Eiten answered 17/4, 2019 at 8:25 Comment(8)
Worked like a charm even after two and a half years later. Thanks.Dalesman
Perfect solution !!Animosity
anybody experiencing a bug with this solution? If I rotate to landscape then select a different app on my device using App Task Manager then navigate back to my app, then click back. The orientation stays in landscape.Heinz
@grantespo, I'm a bit late to the party, but I ran into that bug and fixed it by making changes to the calling ViewController. I overrode supportedInterfaceOrientation to .portrait and overrode shouldAutorotate to true.Longdrawnout
working for UIInterfaceOrientation.landscapeLeft not for UIInterfaceOrientation.landscapeRightLau
this should be the accepted answer, thanks a lot 🙏Croupier
@Heinz any solution of your bug?Folly
@Heinz Yup. supportedInterfaceOrientationsFor is broken. Completely broken on the first screen if the user has the device in the opposite orientation of the locked orientation. It freaks out and draws crazily. Then fixes itself if you move to other screens and come back, or try to rotate it.Bony
K
18

For me, the best results came from combining Zeesha's answer and sazz's answer.

Add the following lines to AppDelegate.swift:

var orientationLock = UIInterfaceOrientationMask.portrait
var myOrientation: UIInterfaceOrientationMask = .portrait
func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask {
    return myOrientation
}  

Add the following line to your view controller class:

let appDel = UIApplication.shared.delegate as! AppDelegate

Add the following lines to your view controller's viewDidLoad():

appDel.myOrientation = .landscape
UIDevice.current.setValue(UIInterfaceOrientation.landscapeLeft.rawValue, forKey: "orientation")

(optional) Add this line to your dismiss function:

appDel.myOrientation = .portrait
UIDevice.current.setValue(UIInterfaceOrientation.portrait.rawValue, forKey: "orientation")

What these lines of code do is set the default orientation to portrait, rotate it landscape when the view controller loads, and then finally reset it back to portrait once the view controller closes.

Knocker answered 28/1, 2018 at 19:15 Comment(2)
This works for me too, because in my Project -> Deployment info I just have Portrait mode selected. ThanksIambus
I tried many methods over the years. This is by far the easiest solution, no iterating over all stacked controllers etc. Hope it will last :)Diathesis
O
12

Overwrite (in ViewController):

override public var shouldAutorotate: Bool {
    return false
} 

override public var supportedInterfaceOrientations: UIInterfaceOrientationMask {
    return .landscapeRight
}

override public var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
    return .landscapeRight
}

Hint for ios 13. As of ios 13, VC has different modalPresentationStyle as .automatic and device present modal view instead of Full-Screen VC. To fix this one must set modalPresentationStyle to .fullScreen. Example:

let viewController = YourViewController()
viewController.modalPresentationStyle = .fullScreen
Oops answered 31/3, 2020 at 10:33 Comment(3)
viewController.modalPresentationStyle = .fullScreen helped me. Thanks!Teodor
This only worked for me when the viewController is presented using .present(viewController, animated: false, completion: nil).. NOT when pushing viewController using .pushViewController(viewController, animated: true)Oestrin
This is partially work. BUT my case I've one navigation controller where I've presented landscape screen and when user switch to other app I need to present appLockScreen on another navigation controller (which works fine). Once user done with appLockScreen new navigation controller is dismissed and old landscaped screen is auto potraitted. :(Folly
C
9

I needed to force one controller into portrait orientation. Adding this worked for me.

swift 4 with iOS 11

override var   supportedInterfaceOrientations : UIInterfaceOrientationMask{

    return  .portrait

}
Cytosine answered 12/11, 2017 at 14:31 Comment(0)
D
3

I faced a similar issue in my project. It only has support for portrait. The ViewController structure is that, Navigation contained a controller (I called A), and a long Scrollview in A controller. I need A(portrait) present to B(landscape right).

In the beginning I tried the method below and it seemed to work but eventually I found a bug in it.

Swift 5 & iOS12

// In B controller just override three properties

override var shouldAutorotate: Bool {
    return false
}

override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
    return UIInterfaceOrientationMask.landscapeRight
}

override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
    return .landscapeRight
}

And then something become strange. When controller B dismiss to controller A. The ScrollView in controller A has been slid some point.

So I used another method, so I rotate the screen when viewWillAppear. You can see the code for that below.

// In controller B
// not need override shouldAutorotate , supportedInterfaceOrientations , preferredInterfaceOrientationForPresentation

override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)
    let appDel = UIApplication.shared.delegate as! AppDelegate
    appDel.currentOrientation = .landscapeRight
    UIDevice.current.setValue( UIInterfaceOrientation.landscapeRight.rawValue, forKey: "orientation")
    UIViewController.attemptRotationToDeviceOrientation()
}

//in viewWillDisappear rotate to portrait can not fix the bug


override func dismiss(animated flag: Bool, completion: (() -> Void)? = nil) {
    let appDel = UIApplication.shared.delegate as! AppDelegate
    appDel.currentOrientation = .portrait
    UIDevice.current.setValue( UIInterfaceOrientation.portrait.rawValue, forKey: "orientation")
    UIViewController.attemptRotationToDeviceOrientation() //must call 
    super.dismiss(animated: true, completion: nil)
}
// in AppDelegate
// the info plist is only supported portrait also, No need to change it

var currentOrientation : UIInterfaceOrientationMask = .portrait


func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask {
    return self.currentOrientation
}
Dejecta answered 17/10, 2019 at 3:40 Comment(1)
can you create any example/demo on change orientation at button click?Hoofer
G
2

Works in Swift 2.2

 func application(application: UIApplication, supportedInterfaceOrientationsForWindow window: UIWindow?) -> UIInterfaceOrientationMask {
    if self.window?.rootViewController?.presentedViewController is SignatureViewController {

        let secondController = self.window!.rootViewController!.presentedViewController as! SignatureViewController

        if secondController.isPresented {

            return UIInterfaceOrientationMask.LandscapeLeft;

        } else {

            return UIInterfaceOrientationMask.Portrait;
        }

    } else {

        return UIInterfaceOrientationMask.Portrait;
    }
}
Ginglymus answered 12/5, 2016 at 13:46 Comment(1)
Does the code above go in AppDelegate or just in the ViewController that you're trying to force to portrait?Descender
N
2

Swift 3. This locks the orientation each time the user re-opens the app.

class MyViewController: UIViewController {
    ...
    override func viewDidLoad() {
        super.viewDidLoad()

        // Receive notification when app is brought to foreground
        NotificationCenter.default.addObserver(self, selector: #selector(self.onDidBecomeActive), name: NSNotification.Name.UIApplicationDidBecomeActive, object: nil)
    }

    // Handle notification
    func onDidBecomeActive() {
        setOrientationLandscape()
    }

    // Change orientation to landscape
    private func setOrientationLandscape() {
        if !UIDevice.current.orientation.isLandscape {
            let value = UIInterfaceOrientation.landscapeLeft.rawValue
            UIDevice.current.setValue(value, forKey:"orientation")
            UIViewController.attemptRotationToDeviceOrientation()
        }
    }

    // Only allow landscape left
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return UIInterfaceOrientationMask.landscapeLeft
    }

    /*
    // Allow rotation - this seems unnecessary
    private func shouldAutoRotate() -> Bool {
        return true
    }
    */
    ...
}
Nancinancie answered 5/5, 2017 at 21:18 Comment(0)
P
2

Swift 4

Trying to keep the orientation nothing worked but this for me:

...        
override func viewDidLoad() {
       super.viewDidLoad()
       forcelandscapeRight()
       let notificationCenter = NotificationCenter.default
       notificationCenter.addObserver(self, selector: #selector(forcelandscapeRight), name: Notification.Name.UIDeviceOrientationDidChange, object: nil)
    }

    @objc func forcelandscapeRight() {
        let value = UIInterfaceOrientation.landscapeRight.rawValue
        UIDevice.current.setValue(value, forKey: "orientation")
    }
....
Paraffin answered 29/9, 2017 at 21:15 Comment(0)
R
2

In ViewController in viewDidLoad Method call below function

func rotateDevice(){
    UIDevice.current.setValue(UIInterfaceOrientation.landscapeLeft.rawValue, forKey: "orientation")
    UIView.setAnimationsEnabled(true) // while rotating device it will perform the rotation animation
}`

App Delegate File Add Below Function & Variables

//Orientation Variables
var orientationLock = UIInterfaceOrientationMask.portrait
var myOrientation: UIInterfaceOrientationMask = .portrait

func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask { return .landscape }
Riggle answered 7/4, 2019 at 18:7 Comment(0)
H
1

According to the documentation of supportedInterfaceOrientations the shouldAutorotate method should return true or YES in Objective-C so that the supportedInterfaceOrientations are considered.

Hoekstra answered 20/11, 2014 at 11:58 Comment(0)
S
1

iOS 16+: requestGeometryUpdate(_:errorHandler:) API

As noted by @simonbs on Twitter, iOS 16 introduces a new API to update the current interface orientation. While in most cases, other, conventional methods will do the job, in some edge cases, they don't work (forcing the use of private APIs like suggested in this answer). Here, the new public API comes to the rescue.

The API works as follows:

windowScene.requestGeometryUpdate(.iOS(interfaceOrientations: .portrait))

You can optionally also pass a closure to handle errors (though I have no experience under which circumstances errors may occur):

windowScene.requestGeometryUpdate(.iOS(interfaceOrientations: .portrait)) { error in
   // Handle error...
}
Sweetie answered 19/7, 2022 at 13:17 Comment(2)
iOS 16 Error encountered: Error Domain=UISceneErrorDomain Code=101 "None of the requested orientations are supported by the view controller. Requested: portrait; Supported: landscapeLeft, landscapeRight" UserInfo={NSLocalizedDescription=None of the requested orientations are supported by the view controller. Requested: portrait; Supported: landscapeLeft, landscapeRight}Marianamariand
@Sweetie thanks you for sharing. Will you please share how can I achieve this for iPhone single screen in landscape for UIWindow usage?Folly
S
1

Tested on IOS 13 to 16.4

First, create a protocol for your rotatable view controller:

protocol Rotatable:UIViewController{}

Then inside your add below code in your AppDelegate:

 func application(_ application: UIApplication, supportedInterfaceOrientationsFor window: UIWindow?) -> UIInterfaceOrientationMask {
    guard let topVC = UIApplication.getTopViewController(base: window?.rootViewController) as? Rotatable else {
        return .portrait
    }
    guard !topVC.isModal else {
        //Use isBeingDismissed to prevent rotation glitches when changing the top view controller.
        return topVC.isBeingDismissed ? .portrait : topVC.supportedInterfaceOrientations
    }

    return topVC.supportedInterfaceOrientations
}

You also need the below extensions:

extension UIApplication {

    static func getTopViewController(base: UIViewController? = rootViewController) -> UIViewController? {
        
        if let nav = base as? UINavigationController {
            return getTopViewController(base: nav.visibleViewController)
        }

        if let tab = base as? UITabBarController {
            if let selected = tab.selectedViewController {
                return getTopViewController(base: selected)
            }
        }

        if let presented = base?.presentedViewController {
            return getTopViewController(base: presented)
        }

        return base
    }
    
    static var rootViewController:UIViewController?{
      return UIApplication
            .shared
            .connectedScenes
            .flatMap { ($0 as? UIWindowScene)?.windows ?? [] }
            .last { $0.isKeyWindow }?.rootViewController
    }

}


extension UIViewController{
    
    var isModal: Bool {

        let presentingIsModal = presentingViewController != nil
        let presentingIsNavigation = navigationController?.presentingViewController?.presentedViewController == navigationController
        let presentingIsTabBar = tabBarController?.presentingViewController is UITabBarController

        return presentingIsModal || presentingIsNavigation || presentingIsTabBar
    }
    
}

After that you just need to override the below variables in your view controllers:

class MyViewController: UIViewController,Rotatable {

     override var shouldAutorotate: Bool {
            return true
        }
        
     override public var supportedInterfaceOrientations: UIInterfaceOrientationMask {
            return .landscape
        }

}
Sore answered 17/6, 2023 at 5:59 Comment(5)
Currently, this is the only answer working for me. All else is not working.Bethannbethanne
This approach works nice on presentation. But when trying to dismiss ViewController and the VC below is in the different orientation, then the dismiss happens, the VC below shows up in the wrong orientation and only then it rotates to the correct orientation with animation. Do you have any suggestion how to fix it?Felisafelise
Nothing fixes it. This is completely broken on iOS.Bony
@Bony What error do you encounter?Sore
@AliKhajehpour It's not an error. Basically it doesn't update fast enough in the lifecycle. Even if I do this, or use the AppDelegate supportedInterface method, it will still create the new VC in the "real" orientation, break a bunch of constraints, and then try to rotate it to the "locked" orientation. I need it to initially build with only the locked orientation so the view doesnt get messed up.Bony
W
0

My solution is

just added below codes in AppDelegate

enum PossibleOrientations {
  case portrait
    case landscape

    func o() -> UIInterfaceOrientationMask {
      switch self {
      case .portrait:
        return .portrait
      case .landscape:
        return .landscapeRight
      }
    }
}
var orientation: UIInterfaceOrientationMask = .portrait

func switchOrientation(to: PossibleOrientations) {
    let keyOrientation = "orientation"

    if to == .portrait && UIDevice.current.orientation.isPortrait {
        return
    } else if to == .landscape && UIDevice.current.orientation.isLandscape {
        return
    }

    switch to {
    case .portrait:
        orientation = .portrait
        UIDevice.current.setValue(UIInterfaceOrientation.portrait.rawValue, forKey: keyOrientation)
    case .landscape:
        orientation = .landscapeRight
        UIDevice.current.setValue(UIInterfaceOrientation.landscapeRight.rawValue, forKey: keyOrientation)
    }
}

And call below codes to change

override func viewDidLoad() {
    super.viewDidLoad()

    if let appDelegate = UIApplication.shared.delegate as? AppDelegate {
        appDelegate.switchOrientation(to: .landscape)
    }
}

or like below

@IBAction func actBack() {
    if let appDelegate = UIApplication.shared.delegate as? AppDelegate {
        appDelegate.switchOrientation(to: .portrait)
    }
    self.navigationController?.popViewController(animated: true)
}
Windham answered 12/11, 2018 at 13:25 Comment(0)
D
0
// below code put in view controller
// you can change landscapeLeft or portrait

override func viewWillAppear(_ animated: Bool) {
        UIDevice.current.setValue(UIInterfaceOrientation.landscapeRight.rawValue, forKey: "orientation")
    }

override var shouldAutorotate: Bool {
        return true
    }
    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .landscapeRight
    }
    override var preferredInterfaceOrientationForPresentation: UIInterfaceOrientation {
        return .landscapeRight
    }
Dermal answered 5/2, 2019 at 9:33 Comment(0)
V
0

I tried many of the answers below but I'm afraid they didn't work. Especially if nav bars and tab bars are also implemented on the app. The solution that worked for me was provided by Sunny Lee on this post here: Sunny Lee, Medium post

Which in turn is an update of this post: Original solution

The only change I made when implementing the post's solution, was to change the part which references .allButUpsideDown to .landscapeleft

Voccola answered 14/11, 2019 at 22:9 Comment(0)
S
0

In Xcode 11 with Swift 5 I Implemented the following. But it only works when the device orientation for the project target does not include all orientations. I disabled the check for Upside Down. After this, the following code works. If all checkboxes are enabled, the code is not called;

class MyController : UINavigationController {

    override var shouldAutorotate: Bool {
        return true
    }

    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .landscape
    }


}
Siena answered 9/12, 2019 at 16:10 Comment(0)
M
-2
class CustomUIViewController : UIViewController{

    override var   supportedInterfaceOrientations : UIInterfaceOrientationMask{

        return  .landscapeLeft

    }

}


class ViewController: CustomUIViewController {
.
.
.
}
Mccormick answered 20/5, 2017 at 10:45 Comment(1)
Welcome to stack overflow, how to write answer properly check here.Aquiculture

© 2022 - 2024 — McMap. All rights reserved.