Experience Introduction: Glow App Develops Apple Watch App

Experience Introduction: Glow App Develops Apple Watch App

Previously, I worked with two colleagues to develop an Apple Watch app for our Glow App in our spare time. This article is written to introduce the development of Apple Watch and list some pitfalls encountered during the development process. Although Watch OS 2 has been released and we developed it with WatchKit, many contents are also applicable to Watch OS 2. I hope this article is helpful to everyone.

Introduction

  • Design
  • WatchKit App Architecture
  • Data Communication
  • Provisioning Profiles and Entitlements
  • Tips

Design

Essentially, you can use Apple Watch as an extended screen of iPhone. You don't need to take out your phone. You can get information or do some simple operations by just raising your wrist. In fact, when you develop Watch App, you will find that the Watch simulator is implemented as an external display of iPhone simulator.

However, Apple Watch presents a new way of human-computer interaction, and the design interaction guidelines of iOS App are not applicable to Watch. Therefore, before designing and developing Watch App, it is necessary to understand its interaction and basic UI elements.

First, let’s talk about interaction. In addition to the familiar gesture interaction, Apple Watch provides 3 new ways of interaction:

  • Force Touch

    The Apple Watch display can sense pressure as well as user clicks, and a "hard press" can display a context menu with up to four actions.

  • The Digital Crown

    Like traditional watches, the crown is the most commonly used interaction. But on the Apple Watch, the crown is not used to adjust the time and date, or wind the watch. By turning the Digital Crown, you can accurately zoom, scroll, or select data without blocking your view. It also serves as a button to return. Press it to return to the home screen, and press it twice to return to the clock interface.

    It sounds great, but the crown API is not yet open, and the system automatically does the scrolling for you:[

  • Side Button

    A long button below the crown. Pressing it takes you to the Friends interface. Here, you can call or text the 12 contacts you've selected, or use the new ways the Watch communicates, such as tapping them, drawing a doodle, or sending a heartbeat.

Well, the relevant API is not open yet. Considering its contact function, it is estimated that it will not be opened in the future.

Apple Watch Human-Computer Interaction Guide

Watchkit App Architecture

When you add a new Watchkit App target, you will find that Xcode actually adds two new executables and compiles them together with your iOS App.

The dependency relationship between them is shown in the figure below. Watch App depends on Watchkit Extension, and Watchkit Extension depends on iOS App. As you can see from the two figures above and below, there are only Storyboard and ImageAssets in Watch App. That's right, in Watch OS 1, Watch App only contains some static things, and Extension is where the code is actually executed. Extension is responsible for obtaining data from iOS App and controlling what to display on the Watch App interface. The operation of Watch App is also initiated by Extension to iOS App. Watch App does not communicate directly with iOS App.

Each page of the Watch App needs to have a corresponding subclass of WKInterfaceController, such as InterfaceController and GlanceInterfaceController in the Extension folder in the figure above. In addition to init, WKInterfaceController has 3 methods related to the life cycle:

  1. // Called after init, you can configure the elements on the interface in this method  
  2. - ( void )awakeWithContext:(id)context;
  3. // Called when this page is about to be displayed to the user  
  4. - ( void )willActivate;
  5. // Called when this page is no longer displayed  
  6. - ( void )didDeactivate;

Data Communication

As mentioned earlier, the Watch App itself only contains some static content. It does not save data by itself and cannot send network requests. It can only interact with the iOS App through the Extension. Therefore, data transmission between the Watch App and the iOS App is the key and the main development work of most Watch Apps. There are five main methods of data transmission. The first is to use the openParentApplication:reply provided by WKInterfaceController, and then implement application:handleWatchKitExtensionRequest:reply on the iOS side to handle the request from the Watch Extension. The third is Wormhole, a third-party library that sends messages and data through the Dawrin notification center. The middle three methods all pass data in independent shared sandboxes through App Group.

  • WKInterfaceController openParentApplication:reply
  • NSUserDefaults
  • Core Data
  • NSFileManager
  • Dawrin notification center - MMWormhole

WKInterfaceController openParentApplication:reply

This method is very intuitive and is the most real-time and reliable of the several data transmission methods. You can use Enum to define several request types, and then pass the request type when sending the request, so that when the iOS App receives the request, it will know what to do. The iOS App uses the reply callback to pass back the request result.

In this way, the iOS App can be called up even in the background. However, the iOS App cannot actively call up the Watch Extension.

  1. NSDictionary *request = @{kRequestType: @(requestType)};
  2. [InterfaceController openParentApplication:request
  3. reply:^(NSDictionary *replyInfo, NSError *error) {
  4. }];

  1. - ( void )application:(UIApplication *)application
  2. handleWatchKitExtensionRequest:(NSDictionary *)userInfo
  3. reply:( void (^)(NSDictionary *))reply
  4. {
  5. RequestType requestType = userInfo[kRequestType];
  6. if (requestType == RequestTypeRefreshWatchData) {
  7. //  
  8. }
  9. }

The three methods in the middle are very similar. They all store data in an independent shared sandbox. The difference is how they store data. When the iOS App or Watch App needs data, it will look for it in the sandbox. It's like a secret mailbox. Only the two parties know where it is. So this is also an asynchronous transmission method. The two parties do not communicate directly. See the following code for specific usage.

NSUserDefaults

Using NSUserDefaults is the simplest, but there is a limit on the data size.

  1. NSString *appGroupName = @ "group.com.yourcompnay.shared" ;
  2. NSUserDefaults *defaults = [[NSUserDefaults alloc] initWithSuiteName:appGroupName];
  3. [defaults setObject:user.name forKey:@ "userName" ];

Core Data

If your iOS App has already put Core Data in a shared sandbox, you can consider this approach.

  1. NSString *appGroupName = @ "group.com.yourcompnay.shared" ;
  2. NSURL *storeURL = [[NSFileManager defaultManager]
  3. containerURLForSecurityApplicationGroupIdentifier:appGroupName];
  4. storeURL = [storeURL URLByAppendingPathComponent:@ "glow.sqlite" ];
  5. [self.persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType
  6. configuration: nil
  7. URL:storeURL
  8. options:nil
  9. error:&error]

NSFileManager && NSFileCoordinator

File reading and writing inevitably involve multi-threading issues, but don't worry, just use NSFileCoordinator.

  1. - coordinateReadingItemAtURL:options:error:byAccessor:
  2. - coordinateWritingItemAtURL:options:error:byAccessor:
  3. [coordinator coordinateWritingItemAtURL:fileURL
  4. options:nil
  5. error:nil
  6. byAccessor:^(NSURL* writingURL) {
  7. [dataToSave writeToURL:newURL atomically: true ];
  8. }];

NSFilePresenter

You can also listen for file changes by implementing the NSFilePresenter protocol, and get real-time updates for free without having to implement a refresh mechanism yourself.

  1. - presentedItemDidChange

Dawrin notification - MMWormhole

The first one is also very convenient to use. Watch Extension and iOS App send messages and listen for messages. Another big advantage is that Wormhole will save the data transmitted last time, so when Watch App wakes up, it can use the data in Wormhole first, and then update the interface when iOS App sends the latest data.

  1. // init  
  2. self.wormhole = [[MMWormhole alloc] initWithApplicationGroupIdentifier:kApplicationGroupIdentifier
  3. optionalDirectory:kWormholeDirectory];
  4. // iOS app  
  5. NSDictionary *message = [self makeWatchData];
  6. [self.wormhole passMessageObject:message identifier:kWormholeWatchData];
  7. // WatchKit Extension  
  8. NSDictionary *message = [self.wormhole messageWithIdentifier:kWormholeWatchData];
  9. // do something with message  
  10. [self.wormhole listenForMessageWithIdentifier:kWormholeWatchData
  11. listener:^(id messageObject) {
  12. NSLog(@ "Received data from wormhole." );
  13. }];

This is also the method I used at the beginning. But in the process of using it, I found that if the iOS App is in background mode, it cannot receive messages from the WatchKit Extension in real time. Therefore, we choose to mix openParentApplication:reply and Wormhole. When the Watch App wakes up, use the data in Wormhole to ensure the response speed of the Watch App, and use openParentApplication:reply to request the latest update from iOS.

Provisioning Profiles and Entitlements

At the beginning of development, the most troublesome things may be Code Signing, Provisioning and entitlements.

Each target must have its own App ID. So we need three in total:

  • yourAppID
  • yourAppID.watchkitextension
  • yourAppID.watchkitapp

You also need to create an associated Provisioning Profile for each App ID. If you use Xcode to automatically create a Provisioning Profile, it will only create the first two for you. You need to manually create them in the developer center.

In addition, you also need to make sure that your three Entitlements are correct. Version Number, Build Number, and App Groups (if used) must be the same, otherwise the compilation will not pass.

Tips

Debug

Sometimes, you may need to debug both iOS App and Watch App at the same time. However, Xcode only allows you to specify one target to run. You can either debug the iOS App code or the Watch App code. However, you can debug both at the same time by using Xcode's Attach to Process. The specific steps are as follows:

  • Run the WatchKit App
  • Open your iOS app in Simulator
  • In Xcode’s menu bar, select Debug -> Attach to Process, select your iOS App, and you can debug both iOS and WatchKit app at the same time.

App Icons and iTunes Connect

If you encounter an Invalid Binary error when uploading your app to iTunes Connect, it is likely because your Watch App icon has a transparent layer or alpha channel. A more convenient solution is to open the image with Preview, select Export, and then uncheck the Alpha option at the bottom and confirm.

End

<<:  Android Network--How I did it: Volley+OkHttp+Https

>>:  Microsoft is working on Android Launcher

Recommend

Where can I apply for a phone number starting with 400?

Where can I apply for a phone number starting wit...

Is your advertising 100% successful?

"Use XXX diapers to help mom have a good sle...

11 economic phenomena necessary for planning and promotion

Introduction: Starting from the essence and findi...

How to optimize iOS projects?

1. Structure and Architecture 1.1 Structure There...

Few Tik Tok fans? A practical tutorial on how to increase followers on Douyin!

What if you want to gain 1 million followers in 7...

How to recall lost users? Analysis from 3 aspects!

Some people say that whether or not one can succe...

The 4 levels of training for software engineers in Silicon Valley

I'm a young, educated, white guy. I'm ver...