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

2025 年度总结

2025 年度总结

今天是 2026 年 1 月 1 号,又是新的一年。这个元旦没有安排出行任务,就在家里休息休息,或者出门溜达溜达。昨天休了一天全薪病假,做了体检,写了年终绩效总结,晚上干了一顿烤肉,没有时间写个人的年度总结。今天起早写写总结。 以下「今年」指 2025 年。 职业发展 算起来,我已经毕业工作四年多了。职业发展整体上还算稳定,没有碰到过糟心事,遇到的领导们也都对我关怀有加。今年又晋升一次,薪资迈上新的台阶。越往上升,越觉得离职业生涯的终点越近,逼迫自己赶紧找个靠谱稳定的副业,到 35 岁没人要的时候能养活自己。 最近两年 AI 大模型的崛起,提高了许多行业的可替代性。码农虽然不是首当其冲的,但危机感已经弥漫在各个论坛博客公共平台上面。没有人能准确预测到未来发展,但做好两手准备是很有必要的。码农不能再只低着头守着自己的键盘和屏幕,也要往外看,接触社会上的各种信息,打破信息壁垒。掌握的信息越多,出路就越多。

By Gray
联通 FTTR 宽带从路由器设置自动重启和穿墙功率

联通 FTTR 宽带从路由器设置自动重启和穿墙功率

几个月前把家里宽带换成了联通的千兆 FTTR 宽带,包含一主一从两个点位。配套光猫设备是华为的星光 F50 尊享版。 主点位放置在客厅茶几上,方便连接电视。从点位放在卧室门口,那里恰好有一个不耽误过路的小拐角可以放路由器。平常我们基本不在客厅活动,其他区域最近的 Wi-Fi 信号源是从路由器,因此我们大多数的设备连接的都是从路由器。从路由器的工作负荷很大。 从路由器个头小主路由器很多,散热不咋地。工作时间久了发热就容易发生数据包堵塞,丢包延迟高。需要把它电源拔掉重启。从宽带开通到现在,数据包堵塞影响网络的情况每个月会发生一次。有一次还影响了居家办公的视频会议。宽带维修师傅也给不出有效的法子,建议就是定期插拔从路由器电源。 从路由器和书房之间隔了两堵墙。信号到我书桌那个位置时,千兆网速已经衰减到只有 400-500Mbps 了,折损将近一半。叠加路由器发热的 debuff,书桌位置的网速最差的时候几乎和百兆宽带差不多。 我尝试过在光猫后台管理将路由器功率设置到「穿墙」模式,但没有任何作用。今天在后台研究了一番发现,原来我之前设置的功率是仅对主路由器生效,从路由器还是标准功率。要修

By Gray