Reputation: 28177
What is the best way to detect a device's orientation in an application extension? I have had mixed results with solutions I've found on here:
How to detect Orientation Change in Custom Keyboard Extension in iOS 8?
Get device current orientation (App Extension)
I have looked at size classes and UITraitCollection
and found that the device inaccurately reports that it is in portrait when it is in fact in landscape (not sure if this is OS bug, or I am not querying the right APIs the right way).
What is the best method for accomplishing:
Thank you,
Upvotes: 13
Views: 8027
Reputation: 1
My share extension targets iOS 14.0 or newer, and an answer provided by a different user suggests subscribing to the UIApplicationWillChangeStatusBarOrientationNotification
notification, which is deprecated as of iOS 13.
The deprecation note in UIApplication.h
suggests using viewWillTransitionToSize:withTransitionCoordinator:
instead, which is working well for me. I've defined this message on my share extension's root view controller. I do have to deduce the orientation from the size that's provided, like so:
- (void)viewWillTransitionToSize:(CGSize)size withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator
{
[super viewWillTransitionToSize:size withTransitionCoordinator:coordinator];
if (size.height > size.width) {
// setup your UI for portrait orientation
}
else {
// setup your UI for landscape orientation
}
}
Upvotes: 0
Reputation: 1698
You can get the device orientation in extension by adding a observer on UIApplicationWillChangeStatusBarOrientationNotification and then extracting the orientation as follows.
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(orientationWillChange:) name:UIApplicationWillChangeStatusBarOrientationNotification object:nil];
}
- (void)orientationWillChange:(NSNotification*)n
{
UIInterfaceOrientation orientation = (UIInterfaceOrientation)[[n.userInfo objectForKey:UIApplicationStatusBarOrientationUserInfoKey] intValue];
if (orientation == UIInterfaceOrientationLandscapeLeft)
//handle accordingly
else if (orientation == UIInterfaceOrientationLandscapeRight)
//handle accordingly
else if (orientation == UIInterfaceOrientationPortraitUpsideDown)
//handle accordingly
else if (orientation == UIInterfaceOrientationPortrait)
//handle accordingly
}
Thanks
Upvotes: 1
Reputation: 850
I faced with that problem and looked through your examples too, but there wasn't a good solution of it. How I resolved it: I created a class that does some calculations of the UIScreen values and returns the self defined device orientation.
Class Header:
typedef NS_ENUM(NSInteger, InterfaceOrientationType) {
InterfaceOrientationTypePortrait,
InterfaceOrientationTypeLandscape
};
@interface InterfaceOrientation : NSObject
+ (InterfaceOrientationType)orientation;
@end
Implementation:
@implementation InterfaceOrientation
+ (InterfaceOrientationType)orientation{
CGFloat scale = [UIScreen mainScreen].scale;
CGSize nativeSize = [UIScreen mainScreen].currentMode.size;
CGSize sizeInPoints = [UIScreen mainScreen].bounds.size;
InterfaceOrientationType result;
if(scale * sizeInPoints.width == nativeSize.width){
result = InterfaceOrientationTypePortrait;
}else{
result = InterfaceOrientationTypeLandscape;
}
return result;
}
@end
I put it to viewWillLayoutSubviews or viewDidLayoutSubviews methods to catch the orientation changes event.
if([InterfaceOrientation orientation] == InterfaceOrientationTypePortrait){
// portrait
}else{
// landscape
}
In case you want to get an exact side of the device orientation (left, right, upside down) this approach won't resolve your problem. It just returns either the portrait or landscape orientations.
Hope it will help you.
Upvotes: 20