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

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

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

Q&A

解決済

1回答

1198閲覧

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

emthy

総合スコア17

0グッド

0クリップ

投稿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

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

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の取り出し方が間違っているのでしょうか?
guest

回答1

0

自己解決

classにおける変数の取り扱いかたについての理解がうまくできていなかったようで、日にちを置いて冷静になってみたらおかしいことに気づきました。
コメントいただいた方ありがとうございました。
ファイル
・ContentView
省略あり
@StateObject var viewModel = ReceiverViewModel()
Text("(viewModel.records)")

・HeartRateReceiver
import Foundation
import WatchConnectivity

final class ReceiverViewModel: NSObject, ObservableObject {
@Published var records = 0

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) { }

// WCSessionDelegateのデリゲートメソッドの実装 // 受信時に呼ばれる関数でuserInfoのデータから受信時と逆の手順でRecord型を取得します。取得できたらrecordsに追加する // 追加時にデータバインディングによってリストが更新される func session(_ session: WCSession, didReceiveUserInfo userInfo: [String : Any] = [:]) { guard let data = userInfo["record"] as? Data, let record = try? JSONDecoder().decode(Int.self, from: data) else { // Error handring if need return //処理の中断 } self.records = record }

}

投稿2022/09/29 11:40

emthy

総合スコア17

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.41%

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

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

質問する

関連した質問