質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
86.12%

解決済

SwiftUI 心拍数の送信処理などについて

emthy
emthy

総合スコア16

1回答

0リアクション

0クリップ

195閲覧

投稿2022/09/28 08:42

前提

SwiftUIでapplewatch,iphoneの連動アプリを作りたいと思っています。
プログラミング、アプリ作成初心者のため手探りで進めています。至らない点があると思いますが、ご指導いただけると嬉しいです。

実現したいこと

applewatchで取得した心拍数の情報をiPhoneに送信し、iphoneでもその心拍数を表示するというアプリケーションを作成するためにコードを書いています

・なぜこのエラーが起きてしまったのか
・自分が実現したいアプリケーションのために書くコードは私が書いているコードで大体合っているのか

がわからず、苦戦しているのでXcodeアプリケーション作成に精通している方教えていただけるとありがたいです。

該当のソースコード(エラーはコメントアウトで示したあります。)

iPhone側のファイル ・ContentView import SwiftUI import HealthKit struct ContentView: View { @StateObject var viewModel = ReceiverViewModel() var body: some View { VStack{ HStack{ Text("❤️") .font(.system(size: 50)) Spacer() } HStack{ Text("Rate") .fontWeight(.regular) .font(.system(size: 70)) Text("BPM") .font(.headline) .fontWeight(.bold) .foregroundColor(Color.red) .padding(.bottom, 28.0) Text(viewModel.record)//エラー1Initializer 'init(_:)' requires that 'Binding<Subject>' conform to 'StringProtocol' //エラー2Referencing subscript 'subscript(dynamicMember:)' requires wrapper 'ObservedObject<ReceiverViewModel>.Wrapper' //エラー3Value of type 'ReceiverViewModel' has no dynamic member 'record' using key path from root type 'ReceiverViewModel' .fontWeight(.regular) .font(.system(size: 70)) Spacer() } } .padding() } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView() } } ・HeartRateReceiver import Foundation import WatchConnectivity final class ReceiverViewModel: NSObject, ObservableObject { private let session: WCSession init(session: WCSession = .default) { self.session = session super.init() self.session.delegate = self session.activate() } } extension ReceiverViewModel: WCSessionDelegate { func session(_ session: WCSession, activationDidCompleteWith activationState: WCSessionActivationState, error: Error?) { if let error = error { print(error.localizedDescription) } else { print("The session has completed activation.") } } func sessionDidBecomeInactive(_ session: WCSession) { } func sessionDidDeactivate(_ session: WCSession) { } func session(_ session: WCSession, didReceiveUserInfo userInfo: [String : Any] = [:]) { guard let data = userInfo["record"] as? Data, var record = try? JSONDecoder().decode(Int.self, from: data) else { return } } } applewatch側のファイル ・ContentView import SwiftUI import HealthKit struct ContentView: View { private var healthStore = HKHealthStore() let heartRateQuantity = HKUnit(from: "count/min") @State private var value = 0 var viewModel = HeartRateTransfer() var body: some View { VStack{ HStack{ Text("❤️") .font(.system(size: 50)) Spacer() } HStack{ Text("\(value)") .fontWeight(.regular) .font(.system(size: 70)) Text("BPM") .font(.headline) .fontWeight(.bold) .foregroundColor(Color.red) .padding(.bottom, 28.0) Spacer() } } .padding() .onAppear(perform: start) } func start() { autorizeHealthKit() startHeartRateQuery(quantityTypeIdentifier: .heartRate) } func autorizeHealthKit() { let healthKitTypes: Set = [ HKObjectType.quantityType(forIdentifier: HKQuantityTypeIdentifier.heartRate)!] healthStore.requestAuthorization(toShare: healthKitTypes, read: healthKitTypes) { _, _ in } } private func startHeartRateQuery(quantityTypeIdentifier: HKQuantityTypeIdentifier) { let devicePredicate = HKQuery.predicateForObjects(from: [HKDevice.local()]) let updateHandler: (HKAnchoredObjectQuery, [HKSample]?, [HKDeletedObject]?, HKQueryAnchor?, Error?) -> Void = { query, samples, deletedObjects, queryAnchor, error in guard let samples = samples as? [HKQuantitySample] else { return } self.process(samples, type: quantityTypeIdentifier) } let query = HKAnchoredObjectQuery(type: HKObjectType.quantityType(forIdentifier: quantityTypeIdentifier)!, predicate: devicePredicate, anchor: nil, limit: HKObjectQueryNoLimit, resultsHandler: updateHandler) query.updateHandler = updateHandler healthStore.execute(query) } private func process(_ samples: [HKQuantitySample], type: HKQuantityTypeIdentifier) { var lastHeartRate = 0.0 for sample in samples { if type == .heartRate { lastHeartRate = sample.quantity.doubleValue(for: heartRateQuantity) } self.value = Int(lastHeartRate) viewModel.transfer(hrate: value) } } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView() } } ・HeartRateTransfer import Foundation import WatchConnectivity final class HeartRateTransfer: NSObject { private let session: WCSession init(session: WCSession = .default) { self.session = session super.init() self.session.delegate = self session.activate() } func transfer(hrate: Int) { guard let data = try? JSONEncoder().encode(hrate) else { return } let userInfo: [String: Any] = ["record": data] self.session.transferUserInfo(userInfo) } } extension HeartRateTransfer: WCSessionDelegate { func session(_ session: WCSession, activationDidCompleteWith activationState: WCSessionActivationState, error: Error?) { if let error = error { print(error.localizedDescription) } else { print("The session has completed activation.") } } }

試したこと

エラーについての検索をしましたが、状態変数ではないのに、状態変数のBindingが必要と出てたり、よくわかりませんでした、、、

アプリ作成にあたって参考にしたサイト

AppleWatchアプリを作ってみる(7)HealthKitを設定して心拍数を表示するアプリを実機で動かす
https://i-doctor.sakura.ne.jp/font/?p=47914

【SwiftUI】Watch Connectivity を利用して Apple Watch からバックグラウンドで iPhone にデータを送って画面更新する
https://qiita.com/MilanistaDev/items/0ce079b255034be84472

補足情報(FW/ツールのバージョンなど)

Xcode Version13.4.1

以下のような質問にはリアクションをつけましょう

  • 質問内容が明確
  • 自分も答えを知りたい
  • 質問者以外のユーザにも役立つ

リアクションが多い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

気になる質問をクリップする

クリップした質問は、後からいつでもマイページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

下記のような質問は推奨されていません。

  • 間違っている
  • 質問になっていない投稿
  • スパムや攻撃的な表現を用いた投稿

適切な質問に修正を依頼しましょう。

y_waiwai

2022/09/28 10:05

エラーが出たなら、エラーメッセージを提示しましょう エラーメッセージは、よけいな省略翻訳しないで出たそのママをコピペで提示してください
emthy

2022/09/28 13:44 編集

見づらいかもしれませんがエラーメッセージ全てコピペで提示してあります iPhone側のファイルContentViewの二つ目のHStackの中に出たエラー3つとも示してあります
hoshi-takanori

2022/09/28 20:18

ReceiverViewModel に record というプロパティがないせいでは。
emthy

2022/09/29 05:42 編集

extension ReceiverViewModelという拡張クラスの中にvar recordとして定義しています!
emthy

2022/09/29 05:52

recordの取り出し方が間違っているのでしょうか?

まだ回答がついていません

会員登録して回答してみよう

アカウントをお持ちの方は

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
86.12%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問

同じタグがついた質問を見る