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

前提

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

コメントを投稿

0 コメント