在 Swift 圖表中使用 Foudation 庫中的測量類型
前言
在這篇文章中,我們將建立一個條形圖,比較基督城地區(qū)自然散步的持續(xù)時間。我們將使用今年推出的新的Swift Charts 框架,并將看到如何繪制默認(rèn)不符合 Plottable 協(xié)議的類型的數(shù)據(jù),如 Measurement<UnitDuration>。
定義圖表的數(shù)據(jù)
讓我們先定義一下要在圖表中展現(xiàn)的數(shù)據(jù)。
我們聲明了一個包含標(biāo)題和步行時間(小時)的 ??Walk?
?? 結(jié)構(gòu)體。我們使用 ??Foundation?
? 框架中的測量類型Measurement[1]和單位類型UnitDuration[2]來表示每次步行的時間。
我們在數(shù)組 works 中存儲要在圖表中顯示的數(shù)據(jù)。
在圖表中使用測量值
嘗試直接在圖表中使用測量值。
讓我們定義一個 Chart,并將 walks 數(shù)組作為數(shù)據(jù)參數(shù)傳遞給它。因為我們知道我們的walk 標(biāo)題是唯一的,所以我們可以直接使用它們作為 id,但你也可以將你的數(shù)據(jù)模型改為 Identifiable。
注意,因為 Measurement<UnitDuration>? 沒有遵守 Plottable 協(xié)議,我們會得到一個錯誤:「Initializer 'init(x:y:width:height:stacking:)' requires that 'Measurement' conform to 'Plottable'」
BarkMark? 的初始化器期望收到一個用于 x 和 y 的 PlottableValue? 參數(shù)。而且 PlottableValue? 的值類型必須符合 Plottable 協(xié)議。
我們有幾個選擇來解決這個錯誤。我們可以提取測量值的 value?,它是一個 Double? 類型,它是默認(rèn)符合 Plottable? 的,我們可以擴展具有 Plottable? 一致性的 Measurement<UnitDuration>?,或者我們可以定義一個包裝了測量的類型并使其符合 Plottable 協(xié)議。
如果我們簡單地從測量值中提取,我們就會失去上下文,不知道用什么單位來創(chuàng)建測量值。這意味著,我們將無法正確格式化圖表的標(biāo)簽來向用戶表示單位。雖然我們可以記住我們在創(chuàng)建測量時使用了小時 hours,但這并不理想。例如,我們可以決定以后改變數(shù)據(jù)模型,以分鐘為單位存儲持續(xù)時間,或者數(shù)據(jù)可能來自其他地方,所以手動重構(gòu)單位并不是一個完美的解決方案。
用 Plottable? 的一致性來擴展 Measurement<UnitDuration> 是可行的,但根據(jù) Swift 中關(guān)于外部類型的追溯一致性的警告 (Warning for Retroactive Conformances of External Types[3]),如果 Swift Charts 在未來添加了這種一致性,它可能會被破壞。
我們將研究如何定義我們自己的類型來包裝 measurement?,并為我們的自定義類型添加 Plottable 的一致性。
設(shè)計一個包裝器類型
設(shè)計一個符合 Plottable 標(biāo)準(zhǔn)的包裝器類型。
我們將定義一個自定義的 PlottableMeasurement 類型,并使其成為通用的,所以它可以容納任何類型的單位的測量類型。
然后,我們將為 PlottableMeasurement 添加 Plottable 的一致性,其單位為 UnitDuration 類型。我們可以在將來添加對其他單位的支持。
Plottable 協(xié)議有兩個要求:primitivePlottable 屬性必須返回原始類型之一,如 Double、String 或 Date,以及一個可失敗的初始化器,從原始 plottable 類型創(chuàng)建一個值。
我決定將測量值轉(zhuǎn)換為分鐘,但你可以選擇適合你需要的任何其他單位。只是在與原始值轉(zhuǎn)換時要使用相同的單位,這一點很重要。
我們現(xiàn)在可以更新我們的圖表,以使用我們的自定義 Plottable 類型。
它可以工作,但X軸上的標(biāo)簽沒有格式化,沒有向用戶顯示測量單位。我們接下來要解決這個問題。
顯示格式化標(biāo)簽
顯示帶有測量單位的格式化標(biāo)簽。
為了定制X軸上的標(biāo)簽,我們將使用chartXAxis(content:)修改器,并用傳遞給我們的值重構(gòu)x軸的標(biāo)記。
我們首先添加網(wǎng)格線,然后重構(gòu)給定值的標(biāo)簽。
AxisValueLabel在初始化器中接受一個LocalizedStringKey,它可以通過插值測量和指定其格式風(fēng)格來構(gòu)建。
我們收到的值是使用我們在 Plottable 一致性中定義的初始化器創(chuàng)建的,所以在我們的案例中,測量值是以分鐘為單位提供的。但我相信對于這個特定的圖表,使用小時會更好。我們可以很容易地將測量值轉(zhuǎn)換為插值內(nèi)部所需的單位。在這里,我們確定該值是 PlottableMeasurement 類型的,所以我們可以強制解包類型轉(zhuǎn)換。
我選擇了縮小的格式和小數(shù)點后零位數(shù)作為數(shù)字樣式,但你可以根據(jù)你的具體圖表調(diào)整這些設(shè)置。
最后的結(jié)果是在X軸上顯示以小時為單位的格式化持續(xù)時間。
你可以從我們的 GitHub repo 中獲得這篇文章中使用的項目的完整 示例代碼[4]。
參考資料
[1]Measurement: https://developer.apple.com/documentation/foundation/measurement?。
[2]UnitDuration: https://developer.apple.com/documentation/foundation/unitduration?。
[3]Warning for Retroactive Conformances of External Types: https://github.com/apple/swift-evolution/blob/main/proposals/0364-retroactive-conformance-warning.md?。
[4]示例代碼: https://github.com/SwiftCommunityRes/SwiftUI-Code-Examples/blob/main/Using-Measurements-from-Foundation-as-values-in-Swift-Charts/Using-Measurements-from-Foundation-as-values-in-Swift-Charts.swift?。