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

HarmonyOS APP<玩转React>开源教程二十:收藏功能实现

第20次收藏功能实现收藏功能让用户可以标记感兴趣的课程方便后续快速访问。本次课程将完整实现收藏功能包括服务层、状态管理和收藏页面。项目效果学习目标掌握 BookmarkService 设计学会收藏状态管理实现收藏列表持久化完成收藏页面开发实现收藏功能全流程20.1 收藏数据模型Bookmark 结构interfaceBookmark{lessonId:string;// 课程 IDmoduleId:string;// 所属模块 IDaddedAt:string;// 添加时间ISO 格式}存储键// Constants.etsexportclassStorageKeys{staticreadonlyBOOKMARKS:stringbookmarks;}20.2 BookmarkService 设计服务结构exportclassBookmarkService{// 内存缓存privatestaticcachedBookmarks:Bookmark[][];// 加载收藏staticasyncloadBookmarks():PromiseBookmark[];// 添加收藏staticasyncaddBookmark(lessonId:string,moduleId:string):Promisevoid;// 移除收藏staticasyncremoveBookmark(lessonId:string):Promisevoid;// 切换收藏状态staticasynctoggleBookmark(lessonId:string,moduleId:string):Promiseboolean;// 检查是否已收藏staticisBookmarked(lessonId:string):boolean;}加载收藏staticasyncloadBookmarks():PromiseBookmark[]{try{constbookmarksawaitStorageUtil.getObjectBookmark[](StorageKeys.BOOKMARKS,[]);BookmarkService.cachedBookmarksbookmarks;returnbookmarks;}catch(error){console.error([BookmarkService] Failed to load bookmarks:,error);return[];}}保存收藏privatestaticasyncsaveBookmarks(bookmarks:Bookmark[]):Promisevoid{try{awaitStorageUtil.setObject(StorageKeys.BOOKMARKS,bookmarks);BookmarkService.cachedBookmarksbookmarks;}catch(error){console.error([BookmarkService] Failed to save bookmarks:,error);}}20.3 收藏操作实现添加收藏staticasyncaddBookmark(lessonId:string,moduleId:string):Promisevoid{constbookmarksawaitBookmarkService.loadBookmarks();// 检查是否已收藏避免重复if(bookmarks.some(bb.lessonIdlessonId)){return;}// 创建新收藏constnewBookmark:Bookmark{lessonId,moduleId,addedAt:newDate().toISOString()};bookmarks.push(newBookmark);awaitBookmarkService.saveBookmarks(bookmarks);}移除收藏staticasyncremoveBookmark(lessonId:string):Promisevoid{constbookmarksawaitBookmarkService.loadBookmarks();constfilteredbookmarks.filter(bb.lessonId!lessonId);awaitBookmarkService.saveBookmarks(filtered);}切换收藏状态staticasynctoggleBookmark(lessonId:string,moduleId:string):Promiseboolean{constisCurrentlyBookmarkedBookmarkService.isBookmarked(lessonId);if(isCurrentlyBookmarked){awaitBookmarkService.removeBookmark(lessonId);returnfalse;// 返回新状态未收藏}else{awaitBookmarkService.addBookmark(lessonId,moduleId);returntrue;// 返回新状态已收藏}}检查收藏状态// 同步方法使用缓存staticisBookmarked(lessonId:string):boolean{returnBookmarkService.cachedBookmarks.some(bb.lessonIdlessonId);}20.4 收藏分组功能按模块分组staticgetBookmarksByModule():Mapstring,Bookmark[]{constgroupednewMapstring,Bookmark[]();for(constbookmarkofBookmarkService.cachedBookmarks){constmoduleBookmarksgrouped.get(bookmark.moduleId)??[];moduleBookmarks.push(bookmark);grouped.set(bookmark.moduleId,moduleBookmarks);}returngrouped;}获取收藏数量staticgetBookmarkCount():number{returnBookmarkService.cachedBookmarks.length;}清空收藏staticasyncclearBookmarks():Promisevoid{awaitBookmarkService.saveBookmarks([]);}20.5 收藏页面实现页面结构EntryComponentstruct BookmarkPage{Statebookmarks:Bookmark[][];StateisLoading:booleantrue;StorageLink(isDarkMode)isDarkMode:booleanfalse;aboutToAppear():void{this.loadBookmarks();}privateasyncloadBookmarks():Promisevoid{this.bookmarksawaitBookmarkService.loadBookmarks();this.isLoadingfalse;}}空状态展示BuilderEmptyState(){Column(){Text().fontSize(64)Text(暂无收藏).fontSize(18).fontColor(this.isDarkMode?#d1d5db:#495057).margin({top:16})Text(点击课程旁的 ☆ 添加收藏).fontSize(14).fontColor(this.isDarkMode?#9ca3af:#6c757d).margin({top:8})}.width(100%).height(100%).justifyContent(FlexAlign.Center)}收藏列表BuilderBookmarkList(){List(){ForEach(this.bookmarks,(bookmark:Bookmark){ListItem(){this.BookmarkItem(bookmark)}.margin({bottom:8})},(bookmark:Bookmark)bookmark.lessonId)}.width(100%).layoutWeight(1).padding({left:16,right:16,top:12}).scrollBar(BarState.Off)}收藏项组件BuilderBookmarkItem(bookmark:Bookmark){Row(){// 获取课程信息Column(){Text(this.getLessonTitle(bookmark)).fontSize(15).fontWeight(FontWeight.Medium).fontColor(this.isDarkMode?#ffffff:#1a1a2e)Text(this.getModuleTitle(bookmark)).fontSize(12).fontColor(this.isDarkMode?#d1d5db:#495057).margin({top:4})}.alignItems(HorizontalAlign.Start).layoutWeight(1)// 取消收藏按钮Text(★).fontSize(20).fontColor(#fcc419).onClick(()this.removeBookmark(bookmark))// 箭头Text(›).fontSize(18).fontColor(this.isDarkMode?#9ca3af:#6c757d).margin({left:8})}.width(100%).padding(16).backgroundColor(this.isDarkMode?#282c34:#ffffff).borderRadius(12).onClick((){router.pushUrl({url:pages/LessonDetail,params:{moduleId:bookmark.moduleId,lessonId:bookmark.lessonId}});})}// 获取课程标题privategetLessonTitle(bookmark:Bookmark):string{constlessonTutorialService.getLessonById(bookmark.moduleId,bookmark.lessonId);returnlesson?.title??未知课程;}// 获取模块标题privategetModuleTitle(bookmark:Bookmark):string{constmoduleTutorialService.getModuleById(bookmark.moduleId);returnmodule?.title??未知模块;}// 移除收藏privateasyncremoveBookmark(bookmark:Bookmark):Promisevoid{awaitBookmarkService.removeBookmark(bookmark.lessonId);this.bookmarksthis.bookmarks.filter(bb.lessonId!bookmark.lessonId);}20.6 完整服务代码/** * 收藏管理服务 */import{StorageUtil}from../common/StorageUtil;import{StorageKeys}from../common/Constants;import{Bookmark}from../models/Models;exportclassBookmarkService{privatestaticcachedBookmarks:Bookmark[][];staticasyncloadBookmarks():PromiseBookmark[]{try{constbookmarksawaitStorageUtil.getObjectBookmark[](StorageKeys.BOOKMARKS,[]);BookmarkService.cachedBookmarksbookmarks;returnbookmarks;}catch(error){console.error([BookmarkService] Failed to load bookmarks:,error);return[];}}privatestaticasyncsaveBookmarks(bookmarks:Bookmark[]):Promisevoid{try{awaitStorageUtil.setObject(StorageKeys.BOOKMARKS,bookmarks);BookmarkService.cachedBookmarksbookmarks;}catch(error){console.error([BookmarkService] Failed to save bookmarks:,error);}}staticgetCachedBookmarks():Bookmark[]{returnBookmarkService.cachedBookmarks;}staticasyncaddBookmark(lessonId:string,moduleId:string):Promisevoid{constbookmarksawaitBookmarkService.loadBookmarks();if(bookmarks.some(bb.lessonIdlessonId)){return;}constnewBookmark:Bookmark{lessonId,moduleId,addedAt:newDate().toISOString()};bookmarks.push(newBookmark);awaitBookmarkService.saveBookmarks(bookmarks);}staticasyncremoveBookmark(lessonId:string):Promisevoid{constbookmarksawaitBookmarkService.loadBookmarks();constfilteredbookmarks.filter(bb.lessonId!lessonId);awaitBookmarkService.saveBookmarks(filtered);}staticasynctoggleBookmark(lessonId:string,moduleId:string):Promiseboolean{constisCurrentlyBookmarkedBookmarkService.isBookmarked(lessonId);if(isCurrentlyBookmarked){awaitBookmarkService.removeBookmark(lessonId);returnfalse;}else{awaitBookmarkService.addBookmark(lessonId,moduleId);returntrue;}}staticisBookmarked(lessonId:string):boolean{returnBookmarkService.cachedBookmarks.some(bb.lessonIdlessonId);}staticgetBookmarksByModule():Mapstring,Bookmark[]{constgroupednewMapstring,Bookmark[]();for(constbookmarkofBookmarkService.cachedBookmarks){constmoduleBookmarksgrouped.get(bookmark.moduleId)??[];moduleBookmarks.push(bookmark);grouped.set(bookmark.moduleId,moduleBookmarks);}returngrouped;}staticgetBookmarkCount():number{returnBookmarkService.cachedBookmarks.length;}staticasyncclearBookmarks():Promisevoid{awaitBookmarkService.saveBookmarks([]);}}20.7 在其他页面集成课程详情页// LessonDetail.etsStateisBookmarked:booleanfalse;aboutToAppear():void{this.isBookmarkedBookmarkService.isBookmarked(this.lessonId);}// 收藏按钮Text(this.isBookmarked?★:☆).fontSize(24).fontColor(this.isBookmarked?#fcc419:#9ca3af).onClick(async(){this.isBookmarkedawaitBookmarkService.toggleBookmark(this.lessonId,this.moduleId);})模块详情页// ModuleDetail.etsLessonItem({lesson:lesson,isBookmarked:BookmarkService.isBookmarked(lesson.id),onBookmarkTap:async(){awaitBookmarkService.toggleBookmark(lesson.id,module.id);this.bookmarkVersion;// 触发刷新}})本次课程小结通过本次课程你已经✅ 掌握了 BookmarkService 设计✅ 学会了收藏状态管理✅ 实现了收藏列表持久化✅ 完成了收藏页面开发✅ 实现了收藏功能全流程课后练习添加排序功能支持按添加时间排序添加批量操作支持批量删除收藏添加导出功能导出收藏列表下次预告第21次测验服务层实现我们将开发测验功能QuizService 设计测验数据结构答案验证逻辑分数计算进入测验功能开发

相关文章:

HarmonyOS APP<玩转React>开源教程二十:收藏功能实现

第20次:收藏功能实现收藏功能让用户可以标记感兴趣的课程,方便后续快速访问。本次课程将完整实现收藏功能,包括服务层、状态管理和收藏页面。项目效果学习目标 掌握 BookmarkService 设计学会收藏状态管理实现收藏列表持久化完成收藏页面开发…...

nnUNetV2实战:从零构建医学影像2D分割数据集全流程解析

1. 环境准备与框架安装 第一次接触nnUNetV2时,我被它繁琐的环境依赖搞得焦头烂额。现在回想起来,其实只要掌握几个关键点就能避开大部分坑。建议使用Python 3.9的环境,我实测Python 3.10的兼容性最好。先通过conda创建隔离环境: c…...

从吾爱论坛到开源神器:EternalBlaze作者的技术初心与硬链接工具诞生记

在国产软件生态中,有这样一群开发者——他们不为商业利益,只为解决实际问题; 他们不求声名远播,只愿作品惠及他人。 EternalBlaze的创作者Henglie正是这样一位典型的技术实践者。 这款源于吾爱破解论坛的硬链接工具&#xff0c…...

大容量硬盘空间管理实战:用EternalBlaze硬链接技术优化TB级存储资源

在数据爆炸式增长的时代,个人用户拥有数TB存储空间已不罕见。 从4K视频素材到高分辨率照片,从虚拟机镜像到开发环境快照,大容量硬盘承载着日益庞大的数字资产。 然而,存储容量的扩张往往伴随着效率的下降——重复文件在庞大的数…...

python cosyVoice实现tts文本转语音、音频(未完成)

文章目录步骤chatTts需要连外网不好用,想着本地弄个。阿里开源项目cosyVoice(舒适语音)(是cosy,不是cos)评级比较不错。步骤 1、pycharm工作区新建文件夹cosyVoice-demo,进入文件夹clone代码。 https://github.com/FunAudioLLM/CosyVoice.gi…...

深搜算法 6300:Grid Path Construction(2418)

6300:Grid Path Construction(2418)时间限制: 1000 ms 内存限制: 524288 KB 提交数: 0 通过数: 0 Special Judge【题目描述】Given an nm grid and two squares a(y1,x1) and b(y2,x2), create a path from a to b that visits each square exactly…...

别再乱用jet了!Matplotlib中5个最值得推荐的科学可视化colormap及使用场景

科学可视化中的色彩艺术:Matplotlib最佳colormap实践指南 在科研论文和工程报告中,数据可视化是传递复杂信息的核心手段。然而,许多研究者至今仍在无意识地使用已被科学可视化领域淘汰的jet色标——这种彩虹色标不仅会造成数据特征的误读&…...

3DTiles白膜性能优化指南:如何让SHP建筑模型在Cesium中流畅加载

3DTiles白膜性能优化实战:从SHP到Cesium的高效加载策略 当我们将城市级建筑SHP数据转换为3DTiles白膜时,最令人头疼的莫过于浏览器中缓慢的加载速度和卡顿的交互体验。我曾在一个智慧园区项目中处理过包含2万多栋建筑的SHP数据集,初始转换后的…...

从YouTube到国内大厂,VPU(视频处理单元)如何重塑视频云的技术栈?

VPU技术革命:解码下一代视频云架构的三大范式转移 当你在深夜用手机观看4K直播时,是否想过每秒数GB的视频数据如何穿越千里依然清晰流畅?当城市每个角落的摄像头都在实时上传画面时,这些海量视频流又如何在云端被高效处理&#xf…...

NRF52系列选型终极指南:从52810到52840,5个关键指标帮你省下30%成本

NRF52系列选型终极指南:从52810到52840,5个关键指标帮你省下30%成本 在物联网设备开发中,芯片选型往往决定了项目60%以上的成本结构。作为Nordic Semiconductor的明星产品线,NRF52系列凭借其出色的低功耗性能和丰富的外设资源&…...

联发科MTK Sensor Bring Up避坑指南:以STK3321为例的常见问题解析

联发科MTK Sensor Bring Up深度实战:STK3321典型问题与系统化解决方案 在联发科(MTK)平台进行传感器(Sensor) Bring Up是智能设备开发中的关键环节,却也是最容易踩坑的技术难点之一。作为MTK生态中广泛使用的环境光传感器,STK3321的集成过程看…...

ARM64服务器上Docker跑Redis总崩溃?3种配置文件调试方案实测

ARM64服务器上Docker跑Redis总崩溃?3种配置文件调试方案实测 最近在ARM64架构服务器上部署Redis时,不少开发者都遇到了容器启动后立即退出的问题。这并非简单的配置错误,而是ARM架构与x86环境的差异导致的兼容性问题。本文将分享三种经过实测…...

Ubuntu20.04下微信中文输入终极解决方案:修改deepin-wine配置全记录

Ubuntu 20.04下微信中文输入问题的深度解决方案 作为一名长期使用Linux系统的开发者,我深知在Ubuntu上使用微信时遇到中文输入问题的痛苦。特别是在需要频繁与同事、客户沟通时,这个问题会严重影响工作效率。本文将分享一套经过实战验证的解决方案&#…...

锂离子电池恒流恒压充电Simulink仿真模型(CC-CV)及其电路结构与充电过程说明

锂离子电池恒流恒压充电Simulink仿真模型(CC-CV) 电路结构包括:直流电压源、DC/DC变换器、锂离子电池、CCCV控制系统 [hot]赠送2000多字的说明文档和参考文献,帮助您更快理解 恒流恒压充电过程: [1]在CC阶段对电池施加…...

利用Mermaid在Markdown中高效构建数据库ER图

1. 为什么选择Mermaid画ER图 第一次接触数据库设计时,我用Visio画了三天ER图,结果产品经理说要改两个字段,所有连线都得重新调整。直到发现Markdown里用Mermaid画ER图的玩法,才明白什么叫"降维打击"。这个组合有多香&am…...

OpenHarmony开发避坑指南:手把手教你写对BUILD.gn,解决90%的编译问题

OpenHarmony开发避坑指南:手把手教你写对BUILD.gn,解决90%的编译问题 在OpenHarmony开发中,BUILD.gn文件是构建系统的核心配置文件,它决定了代码如何被编译、链接和打包。然而,许多开发者在编写BUILD.gn时常常陷入各种…...

8、C语言指针专题:指针与字符串

在C语言中,字符串本质是“以空字符\0结尾的字符序列”,而指针是操作字符串最灵活、高效的工具。字符串的存储、访问、修改、排序及各类处理,都可以通过指针实现,且指针操作相比数组下标操作,更节省内存、执行效率更高。…...

从Maya到Max:如何完美转换Bone骨骼并优化飘带动画效果

从Maya到Max:专业级骨骼转换与飘带动画优化全流程 在3D动画制作中,角色服装、头发等飘动元素的自然表现往往决定了作品的真实感与视觉冲击力。作为资深动画师,我经常需要在Maya和3ds Max这两个行业标准软件之间切换工作流程。本文将分享一套…...

7、C语言指针专题:多级指针

在C语言中,指针的核心是“指向内存地址”,而多级指针则是“指向指针的指针”——二级指针指向一级指针的地址,三级指针指向二级指针的地址,以此类推。多级指针看似复杂,实则是一级指针逻辑的延伸,其核心用途…...

MAC和PHY到底在搞什么?用大白话拆解网卡工作原理

MAC和PHY到底在搞什么?用大白话拆解网卡工作原理 作为硬件工程师,调试网卡时最常遇到的灵魂拷问就是:"为什么ping不通?"这时候如果连MAC和PHY在搞什么都不清楚,那真是两眼一抹黑。今天我们就用修车师傅看发动…...

LLM 大语言模型 训练的时候 batchsize 调整大导致梯度爆炸问题解决

LLM 大语言模型 训练的时候 batchsize 调整大导致梯度爆炸问题解决 优化器AdamW 确实比 SGD 更容易在大 batch 下梯度爆炸,因为自适应学习率会放大稀疏梯度的更新步长。 针对 AdamW 大 batch,给你几个立竿见影的修复方案: 1. 优化器参数调整…...

第8章 时序数据的洞察:从构建到分析的全链路实践

第8章 时序数据的洞察:从构建到分析的全链路实践 时间序列数据是数据分析领域中最具挑战性也最具价值的类型之一。与普通的横截面数据不同,时间序列数据带有一个天然的顺序维度——时间。股票价格、气温变化、网站流量、销售额趋势,这些数据都随着时间推移而产生,前后观测…...

第7章 时间维度的雕琢:日期时间数据的清洗与计算艺术

第7章 时间维度的雕琢:日期时间数据的清洗与计算艺术 在数据分析的世界里,时间维度是最常见的分析轴线之一。无论是销售趋势分析、用户行为轨迹追踪,还是项目进度监控,日期时间数据都扮演着核心角色。然而,原始的日期时…...

第5章 数据融合之道:多源文件的聚合与分发艺术

第5章 数据融合之道:多源文件的聚合与分发艺术 在数据分析的实战过程中,单一数据文件往往无法满足复杂业务需求。真实的商业场景中,数据可能分散在数十个甚至上百个Excel工作簿中,每个工作簿可能包含多个工作表。例如,连锁企业的各门店每日上传销售报表,财务系统每月导出…...

UnityShader实战指南:从ShaderLab到Surface Shader的进阶之路

1. ShaderLab基础语法入门 第一次接触UnityShader时,我完全被ShaderLab的语法搞懵了。记得当时为了修改一个简单的颜色参数,花了整整一下午研究Properties块的写法。现在回头看,ShaderLab其实就像乐高积木的说明书,只要掌握几个核…...

从LangChain到Dify:手把手构建生产级AI工作流

摘要:本文深入解析从LangChain到Dify的技术演进,通过真实业务场景演示如何构建生产级AI工作流。涵盖从"胶水代码"到"企业级平台"的架构变迁,提供完整的代码实现、踩坑经验、性能优化策略,助你快速从原型走向生…...

养狗管理拟参照道路交通法个人观点:计分、吊证、入刑,这些行为将被终身禁养

近年来,犬只伤人事件频发,每年全国被猫狗咬伤抓伤人数高达4000万。这一数字远超交通事故发生量,但长期以来,养犬管理始终停留在“办个证、罚点款”的层面。如果养犬管理能够参照道路交通安全法的逻辑,建立“记分制”“…...

保姆级教程:用FFmpeg+Nginx把监控摄像头RTSP流转成HLS网页播放

从RTSP到HLS:零基础构建浏览器兼容的监控视频流系统 在智能安防和物联网应用场景中,监控摄像头产生的视频流通常采用RTSP协议传输,但现代浏览器却无法直接播放这种流媒体格式。本文将手把手带您实现RTSP到HLS的完整转换方案,通过F…...

PyAV实战:如何用TCP协议稳定拉取RTSP视频流(附超时解决方案)

PyAV实战:TCP协议拉取RTSP视频流的工程化解决方案 引言 在视频处理项目中,稳定获取RTSP流是许多开发者面临的共同挑战。不同于简单的本地文件读取,网络视频流传输涉及复杂的协议交互和实时性要求。PyAV作为FFmpeg的Python绑定,提供…...

OpenCV CSRT目标跟踪实战:从摄像头到无人机,5步搞定复杂场景跟踪

OpenCV CSRT目标跟踪实战:从摄像头到无人机,5步搞定复杂场景跟踪 计算机视觉领域的目标跟踪技术正在经历一场从实验室到真实场景的落地革命。想象一下,当你的无人机能够自主锁定并跟随拍摄目标,或者在工业质检线上精准追踪移动零件…...