Swift 圖表使用Foudation庫(kù)中測(cè)量類型詳解
前言
在這篇文章中,我們將建立一個(gè)條形圖,比較基督城地區(qū)自然散步的持續(xù)時(shí)間。我們將使用今年推出的新的Swift Charts 框架,并將看到如何繪制默認(rèn)不符合 Plottable 協(xié)議的類型的數(shù)據(jù),如 Measurement<UnitDuration>。
定義圖表的數(shù)據(jù)
讓我們先定義一下要在圖表中展現(xiàn)的數(shù)據(jù)。
我們聲明了一個(gè)包含標(biāo)題和步行時(shí)間(小時(shí))的 Walk 結(jié)構(gòu)體。我們使用 Foundation 框架中的測(cè)量類型Measurement和單位類型UnitDuration來(lái)表示每次步行的時(shí)間。
struct Walk {
let title: String
let duration: Measurement<UnitDuration>
}
我們?cè)跀?shù)組 works 中存儲(chǔ)要在圖表中顯示的數(shù)據(jù)。
let walks = [
Walk(
title: "Taylors Mistake to Sumner Beach Coastal Walk",
duration: Measurement(value: 3.1, unit: .hours)
),
Walk(
title: "Bottle Lake Forest",
duration: Measurement(value: 2, unit: .hours)
),
Walk(
title: "Old Halswell Quarry Loop",
duration: Measurement(value: 0.5, unit: .hours)
),
...
]
在圖表中使用測(cè)量值
嘗試直接在圖表中使用測(cè)量值
讓我們定義一個(gè) Chart,并將 walks 數(shù)組作為數(shù)據(jù)參數(shù)傳遞給它。因?yàn)槲覀冎牢覀兊?code>walk 標(biāo)題是唯一的,所以我們可以直接使用它們作為 id,但你也可以將你的數(shù)據(jù)模型改為 Identifiable。
Chart(walks, id: \.title) { walk in
BarMark(
x: .value("Duration", walk.duration),
y: .value("Walk", walk.title)
)
}
注意,因?yàn)?Measurement<UnitDuration> 沒(méi)有遵守 Plottable 協(xié)議,我們會(huì)得到一個(gè)錯(cuò)誤:「Initializer 'init(x:y:width:height:stacking:)' requires that 'Measurement' conform to 'Plottable'」
BarkMark 的初始化器期望收到一個(gè)用于 x 和 y 的 PlottableValue 參數(shù)。而且 PlottableValue 的值類型必須符合 Plottable 協(xié)議。
我們有幾個(gè)選擇來(lái)解決這個(gè)錯(cuò)誤。我們可以提取測(cè)量值的 value,它是一個(gè) Double 類型,它是默認(rèn)符合 Plottable 的,我們可以擴(kuò)展具有 Plottable 一致性的 Measurement<UnitDuration>,或者我們可以定義一個(gè)包裝了測(cè)量的類型并使其符合 Plottable 協(xié)議。
如果我們簡(jiǎn)單地從測(cè)量值中提取,我們就會(huì)失去上下文,不知道用什么單位來(lái)創(chuàng)建測(cè)量值。這意味著,我們將無(wú)法正確格式化圖表的標(biāo)簽來(lái)向用戶表示單位。雖然我們可以記住我們?cè)趧?chuàng)建測(cè)量時(shí)使用了小時(shí) hours,但這并不理想。例如,我們可以決定以后改變數(shù)據(jù)模型,以分鐘為單位存儲(chǔ)持續(xù)時(shí)間,或者數(shù)據(jù)可能來(lái)自其他地方,所以手動(dòng)重構(gòu)單位并不是一個(gè)完美的解決方案。
用 Plottable 的一致性來(lái)擴(kuò)展 Measurement<UnitDuration> 是可行的,但根據(jù) Swift 中關(guān)于外部類型的追溯一致性的警告 (Warning for Retroactive Conformances of External Types),如果 Swift Charts 在未來(lái)添加了這種一致性,它可能會(huì)被破壞。
我們將研究如何定義我們自己的類型來(lái)包裝 measurement,并為我們的自定義類型添加 Plottable 的一致性。
設(shè)計(jì)一個(gè)包裝器類型
設(shè)計(jì)一個(gè)符合 Plottable 標(biāo)準(zhǔn)的包裝器類型
我們將定義一個(gè)自定義的 PlottableMeasurement 類型,并使其成為通用的,所以它可以容納任何類型的單位的測(cè)量類型。
struct PlottableMeasurement<UnitType: Unit> {
var measurement: Measurement<UnitType>
}
然后,我們將為 PlottableMeasurement 添加 Plottable 的一致性,其單位為 UnitDuration 類型。我們可以在將來(lái)添加對(duì)其他單位的支持。
extension PlottableMeasurement: Plottable where UnitType == UnitDuration {
var primitivePlottable: Double {
self.measurement.converted(to: .minutes).value
}
init?(primitivePlottable: Double) {
self.init(
measurement: Measurement(
value: primitivePlottable,
unit: .minutes
)
)
}
}
Plottable 協(xié)議有兩個(gè)要求: primitivePlottable 屬性必須返回原始類型之一,如 Double、String 或 Date,以及一個(gè)可失敗的初始化器,從原始 plottable 類型創(chuàng)建一個(gè)值。
我決定將測(cè)量值轉(zhuǎn)換為分鐘,但你可以選擇適合你需要的任何其他單位。只是在與原始值轉(zhuǎn)換時(shí)要使用相同的單位,這一點(diǎn)很重要。
我們現(xiàn)在可以更新我們的圖表,以使用我們的自定義 Plottable 類型。
Chart(walks, id: \.title) { walk in
BarMark(
x: .value(
"Duration",
PlottableMeasurement(measurement: walk.duration)
),
y: .value("Walk", walk.title)
)
}
它可以工作,但X軸上的標(biāo)簽沒(méi)有格式化,沒(méi)有向用戶顯示測(cè)量單位。我們接下來(lái)要解決這個(gè)問(wèn)題。

顯示格式化標(biāo)簽
顯示帶有測(cè)量單位的格式化標(biāo)簽
為了定制X軸上的標(biāo)簽,我們將使用chartXAxis(content:)修改器,并用傳遞給我們的值重構(gòu)x軸的標(biāo)記。
Chart(walks, id: \.title) { ... }
.chartXAxis {
AxisMarks { value in
AxisGridLine()
AxisValueLabel("""
\(value.as(PlottableMeasurement.self)!
.measurement
.converted(to: .hours),
format: .measurement(
width: .narrow,
numberFormatStyle: .number.precision(
.fractionLength(0))
)
)
""")
}
}
我們首先添加網(wǎng)格線,然后重構(gòu)給定值的標(biāo)簽。
AxisValueLabel在初始化器中接受一個(gè)LocalizedStringKey,它可以通過(guò)插值測(cè)量和指定其格式風(fēng)格來(lái)構(gòu)建。
我們收到的值是使用我們?cè)?Plottable 一致性中定義的初始化器創(chuàng)建的,所以在我們的案例中,測(cè)量值是以分鐘為單位提供的。但我相信對(duì)于這個(gè)特定的圖表,使用小時(shí)會(huì)更好。我們可以很容易地將測(cè)量值轉(zhuǎn)換為插值內(nèi)部所需的單位。在這里,我們確定該值是 PlottableMeasurement 類型的,所以我們可以強(qiáng)制解包類型轉(zhuǎn)換。
我選擇了縮小的格式和小數(shù)點(diǎn)后零位數(shù)作為數(shù)字樣式,但你可以根據(jù)你的具體圖表調(diào)整這些設(shè)置。
最后的結(jié)果是在X軸上顯示以小時(shí)為單位的格式化持續(xù)時(shí)間。

你可以從我們的 GitHub repo 中獲得這篇文章中使用的項(xiàng)目的完整 示例代碼。
以上就是Swift 圖表使用Foudation庫(kù)中測(cè)量類型詳解的詳細(xì)內(nèi)容,更多關(guān)于Swift 圖表Foudation庫(kù)測(cè)量類型的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Swift 中如何使用 Option Pattern 改善可選項(xiàng)的 API 設(shè)計(jì)
這篇文章主要介紹了Swift 中如何使用 Option Pattern 改善可選項(xiàng)的 API 設(shè)計(jì),幫助大家更好的進(jìn)行ios開發(fā),感興趣的朋友可以了解下2020-10-10
淺談在Swift中關(guān)于函數(shù)指針的實(shí)現(xiàn)
這篇文章主要介紹了淺談在Swift中關(guān)于函數(shù)指針的實(shí)現(xiàn),是作者根據(jù)C語(yǔ)言的指針特性在Swifft中做出的一個(gè)實(shí)驗(yàn),需要的朋友可以參考下2015-07-07
swift中c風(fēng)格的for循環(huán)執(zhí)行效率
這篇文章主要介紹了swift中c風(fēng)格的for循環(huán)執(zhí)行效率 的相關(guān)資料,需要的朋友可以參考下2016-07-07
Swift TableView實(shí)現(xiàn)凍結(jié)窗格功能
這篇文章主要為大家詳細(xì)介紹了Swift TableView實(shí)現(xiàn)凍結(jié)窗格功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
Swift算法實(shí)現(xiàn)逐字翻轉(zhuǎn)字符串的方法示例
大家都知道翻轉(zhuǎn)字符串在字符串算法中算是比較常見的,下面這篇文章主要介紹了Swift算法實(shí)現(xiàn)逐字翻轉(zhuǎn)字符串的方法,文中給出了詳細(xì)的示例代碼,需要的朋友可以參考借鑒,下面來(lái)一起看看吧。2017-03-03
利用Swift實(shí)現(xiàn)各類的CATransition動(dòng)畫詳解
CATransition動(dòng)畫主要在過(guò)渡時(shí)使用,比如兩個(gè)頁(yè)面層級(jí)改變的時(shí)候添加一個(gè)轉(zhuǎn)場(chǎng)效果。CATransition分為兩類,一類是公開的動(dòng)畫效果,一類是非公開的動(dòng)畫效果。這篇文章主要給大家介紹了關(guān)于如何利用Swift實(shí)現(xiàn)各類CATransition動(dòng)畫的相關(guān)資料,需要的朋友可以參考下。2017-09-09

