Table of Contents
Expo(React native)環境でapple watch アプリを実装する
前回
前回、下記の記事で、Expoプロジェクト内でApple Widgetを実装しました。
ExpoのContinuous Native Generationの考え方に基づき、ネイティブの機能を、/ios
ディレクトリ配下で実装せずに、expo-apple-targetを利用して/target
配下で実装することによって、Expoの開発体験を損なわずに、Apple Widgetの実装を行うことができました。
data:image/s3,"s3://crabby-images/08c3d/08c3dd2dc8cc687bff2ccfbca61a04521ff00ac5" alt="Expo(React native)環境でapple widgetを実装する"
今回作ったもの
今回も、引き続きexpo-apple-targetを利用して/target
配下Expoのプロジェクト内で、Apple Watchアプリの実装を行いました。
Note
Apple Watchアプリは、Swiftで実装した静的なビューを描画しているだけで、今回は、実際に温度データを取得して描画するようなことはしていません。
data:image/s3,"s3://crabby-images/60ab8/60ab862032c83d0dbb4c698809850f3e70b60891" alt="expo-applewatch-08 apple-watchシミュレータでアプリが起動しているキャプチャ"
開発環境
PC:Macbook M2 (14.3.1)
Xcode:15.3
Apple Watch:10.4
iOS:17.2
Apple Watch アプリを作る準備をする
Expoのプロジェクトを作成するところまでは、前回の記事同様ですので、今回はその先から解説していきます。
今回も下記のライブラリを利用します。
下記コマンドを実行します。
npm install @bacons/apple-targets
./app.jsonに設定をプラグイン設定を追加します。
"plugins": [
[
"@bacons/apple-targets",
{
"teamId": "XXXXXXXXX"
}
]
]
./targets/watch-app/expo-target.config.js に設定を追加します。
/** @type {import('@bacons/apple-targets').Config} */
module.exports = {
type: "watch",
icon: "https://upload.wikimedia.org/wikipedia/commons/thumb/e/e1/FullMoon2010.jpg/1200px-FullMoon2010.jpg",
colors: {
$accent: "steelblue",
},
deploymentTarget: "9.4",
};
設定が完了したら、prebuildを実施します。
$ npx expo prebuild
prebuildを実施すると、./ios 配下に、Xcodeプロジェクトファイルが生成されます。
これをXcodeで開きます。(workspaceでも良いです)
data:image/s3,"s3://crabby-images/c0fca/c0fca6d09b3dd166b49fc5ed847ae649b80b78a6" alt="expo-applewatch-01 xcodeのファイル"
開くと、expo:targets配下に、watch-app ディレクトリが表示され、このディレクトリでApple Watchアプリの開発を行うことができます。
data:image/s3,"s3://crabby-images/ee77b/ee77bf8c8518c2e971c4dba0be8d3608ffe2fa04" alt="expo-applewatch-02 xcodeにwatch-appディレクトリが表示されている画像"
これで、開発準備は完了です。
Apple Watchアプリ を実装する
“Hello, World”を表示
ここからは、SwiftでApple Widgetを実装していきます。
Apple Watchアプリ の開発は、Apple Developerドキュメントが参考になります。
data:image/s3,"s3://crabby-images/8f134/8f134d8448ff2b292c305be3e67f5175b5a67472" alt="Creating a watchOS app | Apple Developer Documentation"
ひとまずApple Watchで、”Hello, World”を表示します。(この記事では、Swiftの実装についての解説は省略します。)
targets/watch-app/index.swift
import SwiftUI
@main
struct watchEntry: App {
var body: some Scene {
WindowGroup {
ContentView()
}
}
}
targets/watch-app/content.swift
import SwiftUI
struct ContentView: View {
var body: some View {
VStack {
Image(systemName: "globe")
.imageScale(.large)
.foregroundColor(.accentColor)
Text("Hello, world!")
}
.padding()
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
プレビューすることができました。
data:image/s3,"s3://crabby-images/295de/295dea15c917cacac78d89a4dda2ee40aec2fbd9" alt="expo-applewatch-03 Xcode上でApple watchアプリがプレビューできた画像"
yarn start (npm run start)
を実行後、Xcodeから、iOSアプリ、watchOSアプリを起動することによって、シュミレーターでも動かすことができます。
data:image/s3,"s3://crabby-images/8e7c2/8e7c2b4bdab383b9ff76b8105e08f26283b31834" alt="expo-applewatch-04 iosシミュレーターでの画像"
data:image/s3,"s3://crabby-images/493e3/493e38c6a187c087484c99fa5f1b762544712e51" alt="expo-applewatch-05 apple watchシミュレータ上で表示できた画像"
これで、Expo、React Native環境内で、Apple Watchアプリが動くようになりました。
また、下記を参考にして、実機でも確認できるようになりました。
data:image/s3,"s3://crabby-images/8f134/8f134d8448ff2b292c305be3e67f5175b5a67472" alt="Devices and Simulator | Apple Developer Documentation"
Tip
実機のApple WatchがXcodeから認識されず、苦労しましたが、Xcodeを最新版(15.3)にアップデートすると、認識されるようになりました。
Figma でApple Watchアプリデザイン
今回は、いきなりApple Watch画面を実装していくのではなく、Figmaのデザインデータを利用して実装してみようと思います。
Figmaには、Apple Watchの便利なUIキットがいくつかあります。
今回は、下記のUIキットのデータをお借りします。
FigmaからSwiftのコードを生成するプラグインがあるので、利用してみます。
data:image/s3,"s3://crabby-images/2ae37/2ae377185ef34dcc7894c4d8aeae26f79d8bffcf" alt="expo-applewatch-06 Figmaプラグインでコードを生成したキャプチャ"
生成されたコードは下記です。
import SwiftUI
struct ContentView: View {
var body: some View {
VStack(spacing: 0) {
ZStack() {
Rectangle()
.foregroundColor(.clear)
.frame(width: 368, height: 448)
.background(.white)
.cornerRadius(60)
HStack(spacing: 0) {
ZStack() {
Rectangle()
.foregroundColor(.clear)
.frame(width: 39.90, height: 39.90)
.background(
AsyncImage(url: URL(string: "https://via.placeholder.com/40x40"))
)
.offset(x: -0, y: -0)
}
.frame(width: 71.91, height: 71.91)
}
.padding(EdgeInsets(top: 0, leading: 0, bottom: 0.09, trailing: 0.09))
.frame(width: 72, height: 72)
VStack(spacing: 5) {
Text("32°")
.font(Font.custom("Montserrat", size: 62).weight(.semibold))
.foregroundColor(.black)
Text("Bratislava")
.font(Font.custom("Montserrat", size: 28).weight(.medium))
.foregroundColor(.black)
.opacity(0.60)
}
.padding(EdgeInsets(top: 0, leading: 1, bottom: 0, trailing: 0))
HStack(spacing: 0) {
ZStack() {
Ellipse()
.foregroundColor(.clear)
.frame(width: 16, height: 16)
.background(.black)
.offset(x: -32, y: 0)
Ellipse()
.foregroundColor(.clear)
.frame(width: 16, height: 16)
.background(.black)
.offset(x: 0, y: 0)
.opacity(0.20)
Ellipse()
.foregroundColor(.clear)
.frame(width: 16, height: 16)
.background(.black)
.offset(x: 32, y: 0)
.opacity(0.20)
}
.frame(width: 80, height: 16)
}
.frame(width: 80, height: 16)
}
.frame(width: 368, height: 448)
}
.frame(width: 368, height: 448);
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
これをそのまま実行すると下記になりました。
data:image/s3,"s3://crabby-images/208d6/208d66f4d7c261cedfabdc58ef308d4bfc61e4ab" alt="expo-applewatch-07 Figmaプラグインでコード生成したものを実行した結果のキャプチャ"
表示が崩れたりフォントが違ったりしてますが、今回は、そこがメインではないので、簡単な調整に留めます。
調整したコードとスクリーンショットは下記になります。
import SwiftUI
struct ContentView: View {
var body: some View {
VStack(spacing: 0) {
ZStack() {
Rectangle()
.foregroundColor(.clear)
.frame(width: 368, height: 448)
.background(.white)
.cornerRadius(60)
VStack(spacing: 5) {
Text("32°")
.font(Font.custom("Montserrat", size: 62).weight(.semibold))
.foregroundColor(.black)
Text("Bratislava")
.font(Font.custom("Montserrat", size: 28).weight(.medium))
.foregroundColor(.black)
.opacity(0.60)
}
.padding(EdgeInsets(top: 0, leading: 1, bottom: 0, trailing: 0))
}
.frame(width: 368, height: 448)
}
.frame(width: 368, height: 448)
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
実機でもシミュレータでも動かすことが出来ました。
data:image/s3,"s3://crabby-images/f60b1/f60b14a29ca9520184e8b20b65e061fefbb31296" alt="expo-applewatch-09 applewatchの実機でアプリが起動しているキャプチャ"
data:image/s3,"s3://crabby-images/60ab8/60ab862032c83d0dbb4c698809850f3e70b60891" alt="expo-applewatch-08 apple-watchシミュレータでアプリが起動しているキャプチャ"
まとめ
Expo・React Nativeで実装しているアプリに、watchOSアプリを加えることできました。 また、おまけではありましたが、Figmaのデザインからコードを生成して実装も試してみることができました。
expo-apple-targets
のおかげで、App Extensionsを簡単に開発する土台ができ、Apple Widget、Apple Watch向けアプリも実装できたので、次回は、Apple Watchに通知を送ったり、他のApp ExtensionsやAndroid特有の機能、ウィジェットからディープリンクを利用したアプリ内の指定した画面へのキックなどをやってみたいなと思ってます。
参考記事
expo-apple-targets
data:image/s3,"s3://crabby-images/3e009/3e0095b4a9cd71141ce85e4ca4a1ffdf710263e2" alt="Apple Home Screen Widgets with Expo"
Apple Watch(watchOS)
data:image/s3,"s3://crabby-images/11c10/11c10842d5b2b0623d54342d5ff4bc78229e7c2f" alt="Planning your watchOS app"
data:image/s3,"s3://crabby-images/8f134/8f134d8448ff2b292c305be3e67f5175b5a67472" alt="Creating a watchOS app | Apple Developer Documentation"