HealthKitの心電図データを取得・表示する方法について記載します。
Xcodeで HealthKit の設定を行っていない場合は、以下の設定も行ってください。
HealthKitデータの読み込み・書き込みの設定方法について
目次
心電図データの取得方法
環境
・iOS14.4
・Xcode 12.4
デモ
最後に記載しているコードを実行すると以下のように表示されます。
ポイント1:権限のリクエスト
権限のリクエストには “HKCategoryType.electrocardiogramType()” を指定します。
1 2 3 4 5 6 7 8 9 10 |
let healthStore = HKHealthStore() let readTypes = Set([HKCategoryType.electrocardiogramType()]) healthStore.requestAuthorization(toShare: [], read: readTypes, completion: { success, error in if success == false { print("データにアクセスできません") return } }) |
ポイント2:心電図データの取得
権限のリクエストが成功した場合、データを取得することができます。
次のように、HKSampleQuery を生成し実行するとデータを取得できます。
1 2 3 4 5 6 7 8 9 10 11 12 13 |
// 心電図データを取得 let query = HKSampleQuery(sampleType: HKObjectType.electrocardiogramType(), predicate: HKQuery.predicateForSamples(withStart: fromDate, end: toDate, options: []), limit: HKObjectQueryNoLimit, sortDescriptors: [NSSortDescriptor(key: HKSampleSortIdentifierEndDate, ascending: true)]){ (query, results, error) in guard error == nil else { print("error"); return } if let tmpResults = results as? [HKElectrocardiogram] { // 取得したデータを格納 } } healthStore.execute(query) |
サンプルコード
サンプルコードは次の構成になっています。
ContentView.swift
取得データをリストで表示します。
ContentViewModel で格納される dataSource配列 の値を参照してリストを生成します。
ContentViewModel.swift
心電図データを取得します。
取得したデータは、1データずつ ListRowItem構造体に格納し、dataSource配列に追加します。
ListRowItem.swift
取得したデータを格納するための構造体です。
ContentView.swift
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 |
import SwiftUI struct ContentView: View { @ObservedObject var contentVM: ContentViewModel let dateformatter = DateFormatter() init(){ contentVM = ContentViewModel() dateformatter.timeZone = TimeZone(identifier: "Asia/Tokyo") dateformatter.locale = Locale(identifier: "ja_JP") dateformatter.dateFormat = "yyyy/MM/dd HH:mm:ss" // 心電図データを取得する関数を呼ぶ(引数は期間) contentVM.get( fromDate: dateformatter.date(from: "2021/05/16 00:00:00")!, toDate: dateformatter.date(from: "2021/05/16 23:59:59")! ) } var body: some View { NavigationView { List { if contentVM.dataSource.count == 0 { Text("データがありません。") } else { ForEach( contentVM.dataSource ){ item in VStack{ HStack{ Text("経過時間:\(item.timeSinceSampleStart)") Spacer() } HStack{ Text("電圧:\(item.mcV)") Spacer() } } } } }.navigationBarTitle(Text("心電図データ一覧"), displayMode: .inline) } } } struct ContentView_Previews: PreviewProvider { static var previews: some View { ContentView() } } |
ContentViewModel.swift
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 |
import SwiftUI import Combine import HealthKit class ContentViewModel: ObservableObject, Identifiable { @Published var dataSource:[ListRowItem] = [] func get( fromDate: Date, toDate: Date) { let healthStore = HKHealthStore() let readTypes = Set([HKCategoryType.electrocardiogramType()]) healthStore.requestAuthorization(toShare: [], read: readTypes, completion: { success, error in if success == false { print("データにアクセスできません") return } // 心電図データを取得 let query = HKSampleQuery(sampleType: HKObjectType.electrocardiogramType(), predicate: HKQuery.predicateForSamples(withStart: fromDate, end: toDate, options: []), limit: HKObjectQueryNoLimit, sortDescriptors: [NSSortDescriptor(key: HKSampleSortIdentifierEndDate, ascending: true)]){ (query, results, error) in guard error == nil else { print("error"); return } if let tmpResults = results as? [HKElectrocardiogram] { // 取得したデータを1件ずつ ListRowItem 構造体に格納 // ListRowItemは、dataSource配列に追加します。ViewのListでは、この dataSource配列を参照して心電図を表示します。 for item in tmpResults { var voltageQuantityCount = 0 let voltageQuery = HKElectrocardiogramQuery(item) { (query, result) in switch(result) { case .measurement(let measurement): if let voltageQuantity = measurement.quantity(for: .appleWatchSimilarToLeadI) { voltageQuantityCount += 1 let listItem = ListRowItem( id: voltageQuantityCount, timeSinceSampleStart: String(measurement.timeSinceSampleStart), mcV: String(voltageQuantity.doubleValue(for: HKUnit(from: "mcV"))) ) self.dataSource.append(listItem) } case .done: print("データ読み込み終了") case .error(let error): print("error:\(error)") @unknown default: print("エラー処理") } } healthStore.execute(voltageQuery) } } } healthStore.execute(query) }) } } |
ListRowItem.swift
1 2 3 4 5 6 7 |
import SwiftUI struct ListRowItem: Identifiable { var id: Int var timeSinceSampleStart: String var mcV: String } |