当前位置: 首页 > article >正文

SwiftUI DatePicker实战:打造一个旅行计划App(含完整代码)

SwiftUI DatePicker实战构建旅行计划App的进阶技巧每次规划旅行时最让人头疼的莫过于安排行程日期。作为iOS开发者我们可以用SwiftUI的DatePicker组件为用户打造流畅的日期选择体验。不同于基础教程本文将带你深入实战通过构建一个功能完善的旅行计划App掌握DatePicker的高级应用技巧。1. 旅行App的日期选择架构设计在开始编码前合理的架构设计能避免后期大量重构。旅行App通常需要处理三类日期时间数据出发日期用户旅程开始的日期返回日期旅程结束的日期必须晚于出发日期出发时间具体出发时刻struct TravelPlan: Identifiable { let id UUID() var departureDate: Date var returnDate: Date var departureTime: Date var destination: String }状态管理方案对比方案优点缺点适用场景State简单直接难以跨视图共享单一视图简单状态ObservedObject可复用需要额外创建类中等复杂度组件EnvironmentObject全局访问过度使用会导致依赖混乱应用级共享状态对于我们的旅行计划App采用State与Binding的组合最为合适struct TravelPlanView: View { State private var departureDate Date() State private var returnDate Calendar.current.date(byAdding: .day, value: 3, to: Date())! State private var departureTime Date() // 日期范围限制 private let minDate Date() private let maxDate Calendar.current.date(byAdding: .month, value: 6, to: Date())! }2. 智能日期范围限制实现基础日期范围限制使用in参数即可实现但旅行场景需要更智能的逻辑DatePicker(出发日期, selection: $departureDate, in: minDate...maxDate, displayedComponents: .date) .onChange(of: departureDate) { newDate in // 自动调整返回日期确保不早于出发日 if returnDate newDate { returnDate newDate } }进阶技巧节假日自动标记func isHoliday(_ date: Date) - Bool { let calendar Calendar.current let components calendar.dateComponents([.month, .day], from: date) // 示例节假日判断实际应使用更完整的节假日库 return (components.month 1 components.day 1) || // 元旦 (components.month 10 components.day 1) // 国庆 } DatePicker(出发日期, selection: $departureDate) .datePickerStyle(.graphical) .environment(\.calendar, Calendar.current) .background { if isHoliday(departureDate) { Color.orange.opacity(0.2) } }3. 多样式DatePicker的混合应用不同场景适合不同的DatePicker样式我们的旅行App可以这样组合出发日期选择- 图形日历式Section(header: Text(行程日期)) { DatePicker(出发日期, selection: $departureDate, displayedComponents: .date) .datePickerStyle(.graphical) DatePicker(返回日期, selection: $returnDate, in: departureDate...maxDate, displayedComponents: .date) .datePickerStyle(.graphical) }出发时间选择- 滚轮式Section(header: Text(出发时间)) { DatePicker(选择时间, selection: $departureTime, displayedComponents: .hourAndMinute) .datePickerStyle(.wheel) .labelsHidden() }快速选择预设- 紧凑式HStack { Text(行程时长) Spacer() DatePicker(, selection: $returnDate, in: departureDate...maxDate, displayedComponents: .date) .datePickerStyle(.compact) }4. 实时响应与数据联动优秀的日期选择体验应该实时反馈用户选择// 计算行程天数 var tripDuration: Int { Calendar.current.dateComponents([.day], from: departureDate, to: returnDate).day! 1 } // 在界面中显示 Section(header: Text(行程摘要)) { LabeledContent(出发日期, value: departureDate.formatted(.dateTime.day().month().year())) LabeledContent(返回日期, value: returnDate.formatted(.dateTime.day().month().year())) LabeledContent(出发时间, value: departureTime.formatted(.dateTime.hour().minute())) LabeledContent(行程天数, value: \(tripDuration)天) }日期变化时的副作用处理.onChange(of: departureDate) { [departureDate] newDate in let daysAdded Calendar.current .dateComponents([.day], from: departureDate, to: newDate) .day! // 保持返回日期与出发日期的相对间隔 returnDate Calendar.current .date(byAdding: .day, value: daysAdded, to: returnDate)! // 自动调整到合理时间范围 if returnDate maxDate { returnDate maxDate } }5. 性能优化与用户体验提升当日期选择器频繁更新时需要注意性能问题防抖处理State private var debounceTimer: Timer? DatePicker(出发日期, selection: $departureDate) .onChange(of: departureDate) { newDate in debounceTimer?.invalidate() debounceTimer Timer.scheduledTimer(withTimeInterval: 0.5, repeats: false) { _ in // 实际处理逻辑 updateRelatedData(for: newDate) } }大日期范围优化// 对于长期行程规划使用月份导航更高效 DatePicker(出发日期, selection: $departureDate, in: minDate...maxDate, displayedComponents: .date) .datePickerStyle(.graphical) .environment(\.calendar, Calendar(identifier: .gregorian))无障碍支持DatePicker(出发日期, selection: $departureDate) .accessibilityLabel(选择出发日期) .accessibilityHint(当前选择: \(departureDate.formatted())) .accessibilityAction(named: 今天) { departureDate Date() }6. 完整实现与扩展功能将上述功能整合成完整视图struct TravelPlanView: View { State private var departureDate Date() State private var returnDate Calendar.current.date(byAdding: .day, value: 3, to: Date())! State private var departureTime Date() State private var destination private let minDate Date() private let maxDate Calendar.current.date(byAdding: .month, value: 6, to: Date())! var body: some View { NavigationStack { Form { destinationSection dateSelectionSection timeSelectionSection summarySection } .navigationTitle(旅行计划) .toolbar { Button(保存) { savePlan() } } } } private var destinationSection: some View { Section(header: Text(目的地)) { TextField(输入旅行目的地, text: $destination) } } private var dateSelectionSection: some View { Section(header: Text(行程日期)) { DatePicker(出发日期, selection: $departureDate, in: minDate...maxDate, displayedComponents: .date) .datePickerStyle(.graphical) DatePicker(返回日期, selection: $returnDate, in: departureDate...maxDate, displayedComponents: .date) .datePickerStyle(.graphical) } } private var timeSelectionSection: some View { Section(header: Text(出发时间)) { DatePicker(选择时间, selection: $departureTime, displayedComponents: .hourAndMinute) .datePickerStyle(.wheel) } } private var summarySection: some View { Section(header: Text(行程摘要)) { LabeledContent(目的地, value: destination.isEmpty ? 未设置 : destination) LabeledContent(出发日期, value: departureDate.formatted(.dateTime.day().month().year())) LabeledContent(返回日期, value: returnDate.formatted(.dateTime.day().month().year())) LabeledContent(出发时间, value: departureTime.formatted(.dateTime.hour().minute())) LabeledContent(行程天数, value: \(tripDuration)天) } } private var tripDuration: Int { Calendar.current.dateComponents([.day], from: departureDate, to: returnDate).day! 1 } private func savePlan() { let newPlan TravelPlan( departureDate: departureDate, returnDate: returnDate, departureTime: departureTime, destination: destination ) // 保存逻辑... } }扩展功能建议添加重复行程功能每周/每月固定出行集成天气API显示选定日期的天气预报实现多城市行程规划每个城市可设置不同日期添加分享功能生成行程图片分享给同伴

相关文章:

SwiftUI DatePicker实战:打造一个旅行计划App(含完整代码)

SwiftUI DatePicker实战:构建旅行计划App的进阶技巧 每次规划旅行时,最让人头疼的莫过于安排行程日期。作为iOS开发者,我们可以用SwiftUI的DatePicker组件为用户打造流畅的日期选择体验。不同于基础教程,本文将带你深入实战&…...

Path of Building:流放之路玩家的终极离线Build规划指南

Path of Building:流放之路玩家的终极离线Build规划指南 【免费下载链接】PathOfBuilding Offline build planner for Path of Exile. 项目地址: https://gitcode.com/GitHub_Trending/pa/PathOfBuilding 你是否曾经在《流放之路》中花费数小时计算天赋点、装…...

GitHub 高效使用指南【实战篇】

1. GitHub 入门:从注册到创建第一个仓库 刚接触GitHub时,很多人会被各种专业术语吓到。其实GitHub就像是一个代码版的社交网络,你可以在这里分享代码、学习他人项目、参与开源贡献。我刚开始用GitHub时也犯过不少错误,比如把私钥文…...

终极指南:5分钟快速上手BiliTools哔哩哔哩工具箱

终极指南:5分钟快速上手BiliTools哔哩哔哩工具箱 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱,支持下载视频、番剧等等各类资源 项目地址: https://gitcode.com/GitHub_Trending/bilit/BiliTools 你是否…...

SpringCloud进阶--Seata与分布式事务畔

起因是我想在搞一些操作windows进程的事情时,老是需要右键以管理员身份运行,感觉很麻烦。就研究了一下怎么提权,顺手瞄了一眼Windows下用户态权限分配,然后也是感谢《深入解析Windows操作系统》这本书给我偷令牌的灵感吧&#xff…...

别再数据线了!用FastAPI 分钟搭个局域网文件+剪贴板神器茄

为 HagiCode 添加 GitHub Pages 自动部署支持 本项目早期代号为 PCode,现已正式更名为 HagiCode。本文记录了如何为项目引入自动化静态站点部署能力,让内容发布像喝水一样简单。 背景/引言 在 HagiCode 的开发过程中,我们遇到了一个很现实的问…...

Turbo Boost Switcher:终极Mac性能管理工具,轻松掌控CPU性能

Turbo Boost Switcher:终极Mac性能管理工具,轻松掌控CPU性能 【免费下载链接】Turbo-Boost-Switcher Turbo Boost disabler / enable app for Mac OS X 项目地址: https://gitcode.com/gh_mirrors/tu/Turbo-Boost-Switcher 你是否曾经在Mac上遇到…...

分布式电源优化配置的二阶锥模型编程方法

分布式电源优化配置 二阶锥 编程方法:采用matlabyalmip编程,cplex或gurobi作为求解器。 主要内容:考虑配电网二阶锥模型,运行主体包括光伏、微燃机以及负荷,创新性考虑敏感负荷及加权电压支撑能力指标,约束…...

大模型热更新失效的5个隐性陷阱(GPU显存泄漏、KV Cache错位、Tokenizer版本漂移全解析)

第一章:大模型工程化中的模型热更新机制 2026奇点智能技术大会(https://ml-summit.org) 模型热更新是支撑大模型服务持续可用与敏捷演进的核心能力,它允许在不中断推理请求的前提下动态加载新版本权重、替换推理图结构或切换Tokenizer配置。该机制显著降…...

3分钟快速上手!MaaYuan代号鸢如鸢自动化辅助工具终极指南

3分钟快速上手!MaaYuan代号鸢如鸢自动化辅助工具终极指南 【免费下载链接】MaaYuan 代号鸢 / 如鸢 一键长草小助手 项目地址: https://gitcode.com/gh_mirrors/ma/MaaYuan 还在为《代号鸢》和《如鸢》的日常任务重复操作而烦恼吗?MaaYuan作为一款…...

Springboot 实现多数据源(PostgreSQL 和 SQL Server)连接淤

一、环境准备 Free Spire.Doc for Python 是免费 Python 文档处理库,无需依赖 Microsoft Word,支持 Word 文档的创建、编辑、转换等操作,其中内置的 Markdown 解析能力,能高效实现 Markdown 到 Doc/Docx 格式的转换,且…...

AI 时代:祛魅、适应与重新定义蓖

指令替换 项目需求:将加法指令替换为减法 项目目录如下 /MyProject ├── CMakeLists.txt # CMake 配置文件 ├── build/ #构建目录 │ └── test.c #测试编译代码 └── mypass2.cpp # pass 项目代码 一,测试代码示例 test.c // test.c #includ…...

chromeplugin虑

正文 异步/等待解决了什么问题? 在传统同步I/O操作中(如文件读取或Web API调用),调用线程会被阻塞直到操作完成。这在UI应用中会导致界面冻结,在服务器应用中则造成线程资源的浪费。async/await通过非阻塞的异步操作解…...

避坑指南:Win10安装PyTorch2.0时CUDA版本冲突的那些事儿

深度学习环境搭建实战:PyTorch 2.0与CUDA版本冲突的终极解决方案 刚接触深度学习的开发者往往会在环境搭建阶段遭遇"版本地狱"——尤其是当PyTorch要求的CUDA版本与本地安装的NVIDIA驱动不匹配时,系统弹出的错误提示足以让人抓狂。本文将带你…...

用Stata复现经典论文:Card Krueger (1994)最低工资DID分析全流程(含数据与代码)

用Stata复现经典论文:Card & Krueger (1994)最低工资DID分析全流程 1994年,经济学家David Card和Alan Krueger发表了一篇颠覆传统经济学认知的经典论文,他们通过对比新泽西州和宾夕法尼亚州快餐店就业数据,发现最低工资上涨并…...

【PCL-8】从PCA到OBB:点云最小包围盒的数学原理与PCL实战

1. 点云包围盒:从AABB到OBB的进化 当我们处理三维点云数据时,经常需要用一个简单的几何体来近似表示复杂的点云形状。这就是包围盒(Bounding Box)的概念。最常见的两种包围盒是AABB(轴对齐包围盒)和OBB&…...

ThinkPad T480黑苹果终极指南:如何用OpenCore配置实现完美macOS体验?

ThinkPad T480黑苹果终极指南:如何用OpenCore配置实现完美macOS体验? 【免费下载链接】t480-oc 💻 Lenovo ThinkPad T480 / T580 / X280 Hackintosh (macOS Monterey 12.x - Sequoia 15.x) - OpenCore 项目地址: https://gitcode.com/gh_mi…...

数字万用表的二极管档和电阻档,测LED到底该用哪个?实测对比给你看

数字万用表的二极管档和电阻档,测LED到底该用哪个?实测对比给你看 当你面对一个未知好坏的LED,手头只有一块数字万用表时,可能会纠结该选择哪个档位进行测量。是二极管档?电阻档?还是传统的电压档&#xff…...

操作系统层优化:为 Stable Yogi 模型部署调优 Linux 内核参数

操作系统层优化:为 Stable Yogi 模型部署调优 Linux 内核参数 如果你已经成功部署了 Stable Yogi 模型,但总觉得它的推理速度还能再快一点,或者在高并发请求下系统偶尔会卡顿、报错,那么问题可能不在模型本身,而在于它…...

从死元组到事务回卷:图解PostgreSQL的MVCC机制与VACUUM底层原理

从死元组到事务回卷:图解PostgreSQL的MVCC机制与VACUUM底层原理 当你在PostgreSQL中执行一条简单的UPDATE语句时,数据库内部究竟发生了什么?这个看似平常的操作背后,隐藏着一套精妙的多版本并发控制(MVCC)机…...

AutoGen Studio实战:用Qwen3-4B模型快速打造智能客服助手

AutoGen Studio实战:用Qwen3-4B模型快速打造智能客服助手 1. AutoGen Studio简介 AutoGen Studio是一个低代码AI代理开发平台,它让开发者能够快速构建、组合和部署AI代理应用。基于AutoGen AgentChat框架构建,提供了可视化界面来管理多代理…...

BGE Reranker-v2-m3性能优化:算法与工程实践

BGE Reranker-v2-m3性能优化:算法与工程实践 1. 引言 在信息检索和RAG(检索增强生成)应用中,重排序模型的质量和效率直接影响着最终的用户体验。BGE Reranker-v2-m3作为北京智源研究院推出的轻量级重排序模型,凭借其…...

【操作系统】CTFos Pro-专为CTF优化的高性能虚拟机正式版

1. CTFos Pro虚拟机:专为CTF优化的高性能解决方案 如果你经常参加CTF比赛或者进行安全研究,肯定遇到过这样的烦恼:每次搭建环境都要耗费大量时间,各种工具安装配置让人头疼,不同比赛需要的环境还不一样。CTFos Pro就是…...

Gemini-CLI 从零到精通的命令行AI开发指南

1. 认识Gemini-CLI:你的命令行AI助手 第一次听说Gemini-CLI时,我也觉得这不过又是一个AI玩具。直到在本地终端里用它5分钟写完一个Python爬虫脚本,才意识到这个命令行工具的强大。简单来说,Gemini-CLI就像把Google最先进的AI模型…...

【华为云CCE实战】内网环境下的Nacos集群容器化部署全流程

1. 内网环境下的Nacos集群部署挑战 在企业级微服务架构中,Nacos作为服务注册中心和配置中心扮演着关键角色。但在内网隔离环境下部署Nacos集群,就像在没有GPS信号的隧道里组车队——既需要确保每辆车(节点)都能互相定位&#xff0…...

DanmakuFactory:解决弹幕格式兼容性难题的专业转换工具

DanmakuFactory:解决弹幕格式兼容性难题的专业转换工具 【免费下载链接】DanmakuFactory 支持特殊弹幕的xml转ass格式转换工具 项目地址: https://gitcode.com/gh_mirrors/da/DanmakuFactory 在视频创作和弹幕文化日益普及的今天,不同平台间的弹幕…...

Graphormer与经典力学的结合:分子动力学模拟初始结构筛选

Graphormer与经典力学的结合:分子动力学模拟初始结构筛选 1. 引言:当AI遇上分子模拟 想象你是一位计算化学研究员,每天要花费数小时等待分子动力学模拟结果。传统方法需要从零开始计算每个分子构象的能量和稳定性,这个过程既耗时…...

MacOS上MPV播放器隐藏技巧:如何自定义画面旋转快捷键(附完整配置步骤)

MacOS上MPV播放器隐藏技巧:如何自定义画面旋转快捷键(附完整配置步骤) 在视频播放领域,MPV以其轻量级和高度的可定制性赢得了技术爱好者的青睐。不同于主流播放器的固定功能模式,MPV更像是一个开放的工具箱&#xff0c…...

告别环境冲突!用Anaconda虚拟环境搞定QGIS 3.18二次开发(附Pycharm代码补全修复)

告别环境冲突!用Anaconda虚拟环境搞定QGIS 3.18二次开发(附Pycharm代码补全修复) 当你在深夜调试QGIS插件时,突然发现昨天还能运行的脚本今天报了一堆依赖错误——这种场景对GIS开发者来说再熟悉不过了。环境冲突、版本不匹配、ID…...

如何在Mac上原生读写NTFS硬盘?终极指南与免费工具推荐

如何在Mac上原生读写NTFS硬盘?终极指南与免费工具推荐 【免费下载链接】Free-NTFS-for-Mac Nigate: An open-source NTFS utility for Mac. It supports all Mac models (Intel and Apple Silicon), providing full read-write access, mounting, and management fo…...