App Extension allows us to extend the functionality of our app while users are using other apps.
Today Extension is also called widget. It allows important messages to reach your users faster. For example, users can check the weather, or stock prices, check calendars, and so on. According to apple’s official documentation, a widget should have the following features.
- Make sure content is up to date
- User events to respond to
- Good performance (takes up a lot of memory on iOS and the system may kill the widget)
Create Today the Extension
Xcode -> File -> New -> Target -> TodayExtension
Just like creating a new project, once the Settings are created, there will be a Target in the project. Modify Scheme to run the Extension you just created, and you will see the widget in the Today notification center. It says “Hello World.”
Plus Xcode creates default template files for you.
- TodayViewController. Swift (if the OC will be
.h
ε.m
File) - MainInterface.storyboard
- Info.plist
Note: The default is to use the storyboard as the entry point to the widget. If you don’t want to use the storyboard, you can remove the storyboard and put it in the info.plist
NSExtensionMainStoryboard
toNSExtensionPrincipalClass
MainInterface
toTodayViewController
Set the interface
After completing the steps above, either you choose to use stroyBoard as the entry point to your widget or you choose to do it in code. It’s all the same.
For some unknown reason, all the articles I read on the Internet use code to do this. So in this article and the sample code that follows, you will use Xcode’s default storyboard for the layout of the widget.
I will solve the problem
- Open the main app in the widget and pass the parameters
- Widgets share data with the main app
- Widgets share resources with the main app
- Widget opening and collapsing
I met the pit
There’s nothing wrong with Today Extension, after all.
- When testing, since the Widget and the main app are two different targets, printing the corresponding value in the AppDelegate when passing parameters has no effect. At first I thought the changes in the main app were invalid because the scheme was a widget. But that’s not the case. Display the parameter as alert, and you can see that the main app is actually running.
Let’s start with the preparation I did
In order not to pull so many useless things. Let’s start with something THAT I did that’s not really relevant to today’s topic.
Write the main app
In the main app I wrote a UITableView and used Userdefault to save the data I wanted to persist. And then I add and delete to the Todo list.
widget
I have the same UITableView in the widget that only has the view function.
Things to do
Widgets share resources with the main app
Widgets share code and resources with the main app. As an engineer, we have to think of the unchanging truth of high clustering and low coupling in everything we do. As much as possible, widgets share code with the main app.
There are two main schemes:
- framework
- Direct sharing
In the framework case, cocoapods is a new target, so you only need to add the corresponding code to your podfile to use the widget.
The other is direct sharing, and that’s easy. In my example, I have the main app and widget share an image and a TodoCell class (including xiB files). The only thing I did was to select the file in Xcode and then check the corresponding target in TargetMenberShip to the right of Xcode.
Widgets share data with the main app
Widgets and apps are technically two different apps, and the only way to share data between them is to use App Groups.
First up in the main app
target -> capabilities -> app groups
Open the app Groups function and click + to set the ID. Change one if it’s repeated.
widget app
target -> capabilities -> app groups
Then the group list can see the corresponding group. Check it.
At this point you have completed the prerequisites for the widget to share data with the main app.
The next thing we need to do is to adjust the Userdefault code in our preparation.
Will UserDefaults. Standard change
UserDefaults(suiteName: "your group id")
Copy the code
This can then be used in widgets
let userdefault = UserDefaults(suiteName: "group.com.sunny.group")
Copy the code
Got the data persisted in the main app. Other uses of app Groups can be further explored.
Collapse and expand widgets
Apple’s official documentation clearly states that the widget’s interface cannot be swiped. After all, widgets and notifications center swipes don’t conflict.
So sometimes we need to fold up the widgets because they are really annoying when they are too long.
So let’s just talk a little bit about what we did on iOS10, because there are no devices below iOS10.
In the TodayViewController’s didLoad
// Add collapse button in iOS10
if #available(iOSApplicationExtension 10.0, *) { extensionContext? .widgetLargestAvailableDisplayMode = .expanded }else {
// You need to add your own collapse button on iOS8 and iOS9
}
Copy the code
Then implement the methods in the NCWidgetProviding protocol
func widgetActiveDisplayModeDidChange(_ activeDisplayMode: NCWidgetDisplayMode, withMaximumSize maxSize: CGSize) {
// There is no such proxy on iOS8 and iOS9. You need to set target-Action for the buttons you add and then modify them
switch activeDisplayMode {
case .compact:
preferredContentSize = maxSize
case .expanded:
preferredContentSize = CGSize(width: 0.0, height: 60 * CGFloat(dataSource.count))}}Copy the code
In iOS8 and iOS9, you don’t have this feature. We’ll just have to write a button and then do it.
Widget opens the main app
The widget opens the main app in the same way that openURL does, and then adds the required parameters to the URL.
The preparatory work
Main app -> Target -> Info -> UrlTypes
Add a URlType and set the URL Scheme to your custom string. Such as “sunny”.
Write this code in the widget where you want to jump
self.extensionContext? .open(NSURL(string: "sunny://action=\(dataSource[indexPath.row])")
Copy the code
Parameter passing is the same as above, concatenated in the URL. As mentioned earlier, widgets and apps can share data. It could also be a way of passing parameters.
At this time open the main app is directly into the main interface. What if we need to do something else?
Think back to wechat or Alipay, you had to write some code in the AppDelegate.
func application(_ app: UIApplication, open url: URL, options: [UIApplicationOpenURLOptionsKey : Any] = [:]) -> Bool {
let prefix = "sunny://"// Check whether it is from a reliable place
if url.absoluteString.hasPrefix(prefix) {
// The parameter is coming! Do the opposite
let a = UIAlertController(title: url.absoluteString, message: nil, preferredStyle: .alert)
a.addAction(UIAlertAction(title: "Cancel", style: .cancel, handler: nil))
self.window? .rootViewController? .present(a, animated:true, completion: nil)
return true
}
return false
}
Copy the code
others
highly
The default height of widgets is limited.
Compact:
- max = 110
- mim = 110
Expanded under:
- min = 110
- Max = Varies according to different models.
No matter how you set it, it doesn’t go beyond this range, okay
widgetPerformUpdate
func widgetPerformUpdate(completionHandler: (@escaping (NCUpdateResult) -> Void)) {
// Perform any setup necessary in order to update the view.
// If an error is encountered, use NCUpdateResult.Failed
// If there's no update required, use NCUpdateResult.NoData // If there's an update, use NCUpdateResult.NewData
completionHandler(NCUpdateResult.newData)
}
Copy the code
This method is used to select whether the widget will refresh when it appears again.
notice
inNSExtensionContext
Several notifications seen in TodayExtension do not appear to be for TodayExtension.
NSExtensionContext you can see several notifications that they’re all listening for the state of the host app. So for widgets, the host app is Today.
The last
This article uses Today Extension to do a very simple function. Of course, we can do more with him than that. This is where we need to unleash our ingenuity.
Sample code download link Because it was written using Swift, you find that it doesn’t compile well for well-known reasons. You can contact me, I will do the adaptation.