SwiftUI|一种自定义 macOS TabView 的方案

介绍一种在 SwiftUI 里实现自定义 TabView 的方法(macOS)

SwiftUI|一种自定义 macOS TabView 的方案
Photo by Penfer / Unsplash

目前在 macOS 上,SwiftUI 的 TabView 是利用 AppKit 的 NSTabView 实现的。NSTabView 的默认样式带有边框和背景色,顶部 tab 切换使用 NSSegmentedControl 实现,无法满足特定设计需求。

NSTabView 图源:http://www.skyfox.org/cocoa-nstabview.html

网上有一些利用 SwiftUI 基础视图自己实现 tabbar 的方案。这些方案本质上不依赖 TabView 或 NSTabView 实现,只是做了一个模拟切换 tab 的视图,然后在 tab 切换时利用 SwiftUI 的渲染特性,在 body 里面返回对应 tab 的 View。这种方案有一个缺点——每次回到某个 tab 时,对应的 View 会被重新创建。如果页面结构简单的话,这种方案可取;但是对于复杂的页面,每次重新创建的性能开销还是很大的,就不能采用这种方案了。

所以自定义 TabView 最好还是基于原生的 TabView/NSTabView 组件实现。原生组件是不会在切到某个 tab 时重新创建对应的 View 或页面的。

在 AppKit/Cocoa 时代,大家也有这种自定义 TabView 的需求。当时有前辈给出了一种方案通过修改 NSTabView 的 tabViewType 隐藏它的 tabbar 和边框;然后在 NSTabView 上方添加一个自定义的切换 tab 的视图即可。这个方案在 SwiftUI 上也可以使用,也是本篇文章要介绍的方案。

将 NSTabView 的 tabViewType 设置为 noTabsNoBorder ,可以隐藏 NSTabView 的 tabbar 和边框。目前 SwiftUI 的 TabView 还不支持直接修改 tabViewType ,我们需要借助 SwiftUIIntrospect 来间接获取到 TabView 底层使用的 NSTabView。

TabView(selection: $selectedIndex,
        content:  {
    //...
})
.tabViewStyle(.automatic)
.backgroundStyle(.clear)
.introspect(.tabView, on: .macOS(.v14, .v15)) { tabView in
    tabView.tabViewType = .noTabsNoBorder
}

然后在 TabView 上方添加一个自定义的 tab 切换视图。这个自定义视图可以是利用 SwiftUI 基础视图自己实现的 tabbar,也可以图省事直接用 Picker(segmented 样式的)。

Picker("", selection: $selectedIndex) {
    ForEach(tabNames.indices, id: \.self) { nameIndex in
        Text(tabNames[nameIndex])
    }
}
.pickerStyle(.segmented)

定义一个 Int 类型的状态变量连接 TabView 和 Picker。

@State private var selectedIndex = 0

这样切换 Picker 选项时,TabView 也会切换到对应的页面。

Read more

SwiftUI|监测视图更新及更新原因

SwiftUI|监测视图更新及更新原因

SwiftUI 的优势之一就是可以自动让 UI 和数据状态保持同步。频繁的视图更新也会带来一定的性能问题,尽管 Apple 已经让 SwiftUI 在更新时尽量只渲染视图树中必须重绘的部分,但我们还是需要知道某种可以监测视图更新以及触发其更新的原因的手段。这样我们才能在业务层面做一些针对性的性能优化。 我们可以通过在 View 的 body 视图构造器中插入一条 print 语句来监测到哪些视图的 body 正在被执行。这条语句还必须是一个赋值语句,不能是仅一个 print。因为 print 返回的是一个 Void ,而不是一个 View ,视图构造器会报错。 struct ContentView: View { var body: some View { let _ = print("ContentView Changed!") VStack { Image(systemName: "globe"

By Gray
天津一日行

天津一日行

五月份周末抽了一天去了天津转转。说来惭愧,在北京待了那么多年,我竟然一次都没有去过天津。 我们买的高铁票是北京南到天津站,游玩也是选的天津站附近的景点。当天天气不咋滴,上午阴天,下午就下起了雨。 出地铁后,映入眼帘的就是海河以及解放桥。 经过解放桥之后,再往前走一段路,就能看到路边有一个卖煎饼果子的小摊。我们当时刚下高铁,肚子正饿,所幸就买了一个吃。额外加了鸡蛋和辣条。都说天津的煎饼果子是一绝,这吃起来确实挺香👍。 穿过金街,一座看着很有时代感的建筑就映入眼帘——天津劝业场。刚看到它的名字时,我还以为是我念错了,怎么会有这么古怪的名字。后来一查还真是叫这个名字。 “劝业”二字也寄托着国人实业图强的希望。所以在天津劝业场开业之初,其经营宗旨以“劝业商场”四字的字头写出四句警言。即:劝吾胞兴,业精于勤,商务发达,场益增新。– 看天津系列:传统核心商圈金街 可能是因为当天是周六且非节假日,这条步行街上的行人并不多。游客们忙着拍照,本地人闲逛,和北京商业街的繁忙是不可同日而语的。天津是出了名的慢节奏,我倒是很喜欢这样的氛围。 做天津攻略时,我就看到有文章说天津美食有三绝:狗不理

By Gray