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

《漫步华尔街(第12版)》读书笔记

《漫步华尔街(第12版)》读书笔记

股票分析 基本面分析 * 基本面分析的四个基本决定因素 * 预期增长率 * 复合增长(复利)对投资决策有很重要的意义。 * 一只股票的股利增长和盈利增长率越高,理性投资者应愿意为其支付越高的价格。 * 推论:一只股票的超常增长率持续时间越长,理性投资者应愿意为其支付越高的价格。 * 预期股利支付率 * 对于预期增长率相同的两只股票来说,持有股利支付率越高的股票,较之股利支付率低的股票,会使你的财务状况更好。 * 在其他条件相同的情况下,一家公司发放的现金股利占其盈利的比例越高,理性投资者应愿意为其股票支付越高的价格。 * 特例,很多处于强劲增长阶段的公司,往往不支付任何股利。这时候不满足「在其他条件相同的情况下」。 * 风险程度 * 在其他条件相同的情况下,一家公司的股票风险越低,理性投资者(以及厌恶风险的投资者)应愿意为其股票支付越高的价格。 * 市场利率水平 * 在其他条件相同的情况下,市场利率越低,理性投资者应愿意为股票支付越高的价格。 * 举例,银行存款利率

By Gray
2025 端午日本九日游

2025 端午日本九日游

从日本回来后就一直忙个不停,忙着搬家和工作。这周末终于有时间回顾和记录一下日本的旅游行程。 这次出国游是年初就规划好的。端午节假期三天再加上节后请假四天,以及周末,总共能休 9 天。5 月 31 号出发,6 月 9 号凌晨的航班飞回北京。 出发前的准备 机票和酒店 越临近出发日期,机票和酒店就越贵。所以我们早早地就把机票和酒店定了。 去程机票订的山航,青岛转机,5 月 31 号从北京出发抵达青岛,在青岛玩一天,翌日早上从青岛飞往关西机场。回程机票订的海南航空,从东京羽田机场直飞北京,是凌晨两三点的红眼航班。 本次行程要去关西(京都、大阪、奈良)、关东(东京、富士山)。关西三个城市很近,一直住在京都即可,从京都往返大阪和奈良。关东就住在东京。京都的酒店订在了京都站附近,出站走几步就能到,交通非常便利。东京的酒店订在了马喰町附近,附近有很多地铁线路,包括浅草线、

By Gray
2025 关税危机中学到的投资经验

2025 关税危机中学到的投资经验

充足的现金流很重要 好的买入机会不会每天都出现,但当它出现的时候,你最好还有筹码可以投入。 有些人手里握不住钱,一有闲钱就赶紧买入基金、股票,生怕错过了机会,让钱白搭手里。市场是疯狂的、充满变数的,尤其是在特朗普上台后,一句话就可能让股市涨停或跌停。那些专业的理财投资机构尚不能预测市场,何况我们这些散户呢。在不稳定的市场中,我们要学习巴菲特,备好现金,耐心等待买入(抄底)机会。 不要提前打光子弹 美股标普 500 指数从 2 月中旬到 3 月中旬累计跌了约 10%。如果这时候你觉得已经跌了很多,可以 all in 抄底了,那么你就会错过 4 月上旬的那次狂跌——一周跌了约 10%。没有人能预测市场,除了此刻的股市指挥家特朗普。散户们能学到的经验就是「永远不要提前打光子弹」,你以为的谷底其实只是个半山腰。 相信自己,保持耐心 在美股大跌的时段里,小红书、v2ex

By Gray