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

Cocos Creator 3.x 高维护性打字机对话系统设计与实现

在 Cocos Creator 项目中对话系统是 RPG、冒险、视觉小说等类型游戏的核心功能之一。如何设计一个维护性高、可扩展、策划友好、支持存档的打字机Typewriter系统是许多开发者面临的挑战。该系统采用组件化 配置化 JSON 数据驱动 事件解耦 状态机 存档集成的设计理念代码清晰、参数集中、业务与逻辑分离极大降低了后期维护和迭代成本。1. 系统整体架构系统由以下核心模块组成每个模块职责单一便于独立维护和测试模块名称主要职责设计优势TypewriterComponent核心打字逻辑逐字符显示、计时、跳过独立、可复用、事件驱动TypewriterConfig打字参数配置速度、音效、标签等数据与逻辑分离支持 JSON 热更新DialogueEntry单条对话数据结构支持富文本、头像、分支等DialogueManager对话序列管理、UI 更新、存档集成业务层核心JSON 驱动解耦彻底SaveData存档数据结构轻量、稳定支持多槽位推荐节点层级DialoguePanel根节点PortraitSprite—— 角色头像SpeakerNameLabel—— 说话者名字DialogueTextLabel 或 RichText—— 挂载 TypewriterComponentDialogueManager脚本挂载2. TypewriterConfig配置类// TypewriterConfig.ts import { _decorator, CCString, CCFloat, CCBoolean, CCInteger } from cc; const { ccclass, property } _decorator; ccclass(TypewriterConfig) export class TypewriterConfig { property(CCFloat) public speed: number 0.05; property(CCBoolean) public canSkip: boolean true; property(CCBoolean) public autoNext: boolean false; property(CCString) public typeSound: string ; property(CCString) public completeSound: string ; property(CCInteger) public defaultPauseMs: number 300; property(CCBoolean) public enableRichParse: boolean true; constructor(data?: PartialTypewriterConfig) { if (data) Object.assign(this, data); } public static fromJSON(json: any): TypewriterConfig { return new TypewriterConfig(json); } public toJSON(): any { /* ... */ } public clone(): TypewriterConfig { /* ... */ } }3. TypewriterComponent核心打字组件核心逻辑封装在此组件中支持 Label/RichText、自动开始、跳过、每字符/完成事件。import { _decorator, Component, Label, RichText, CCString, CCFloat, CCBoolean, EventHandler } from cc; const { ccclass, property, menu } _decorator; ccclass(TypewriterComponent) menu(UI/TypewriterComponent) // 在组件菜单中一键添加 export class TypewriterComponent extends Component { property(Label) public label: Label | null null; property(RichText) public richText: RichText | null null; property(CCString) public fullText: string ; property(CCFloat) public speed: number 0.05; // 秒/字符 property(CCBoolean) public autoStart: boolean true; property(CCBoolean) public canSkip: boolean true; property(EventHandler) public onCharacterTyped: EventHandler new EventHandler(); property(EventHandler) public onComplete: EventHandler new EventHandler(); private _currentIndex: number 0; private _isTyping: boolean false; private _timer: number 0; private _currentText: string ; onLoad() { if (!this.label !this.richText) { console.warn(【TypewriterComponent】必须绑定 Label 或 RichText 组件); } } start() { if (this.autoStart this.fullText) { this.startTyping(); } } /** 开始打字支持外部传入新文本 */ public startTyping(text?: string): void { if (text ! undefined) this.fullText text; this._currentIndex 0; this._currentText ; this._isTyping true; this._timer 0; this.updateDisplay(); this.node.emit(typing-start); } /** 玩家点击跳过 */ public skipTyping(): void { if (!this.canSkip || !this._isTyping) return; this._currentIndex this.fullText.length; this._currentText this.fullText; this.updateDisplay(); this._isTyping false; this.node.emit(typing-complete); EventHandler.emitEvents(this.onComplete); } update(dt: number) { if (!this._isTyping) return; this._timer dt; if (this._timer this.speed) { this._timer - this.speed; // 支持掉帧补偿 this.typeNextCharacter(); } } private typeNextCharacter(): void { if (this._currentIndex this.fullText.length) { const nextChar this.fullText[this._currentIndex]; this._currentText nextChar; this._currentIndex; this.updateDisplay(); // 每字符事件可播放打字音效 this.node.emit(character-typed, nextChar); EventHandler.emitEvents(this.onCharacterTyped, nextChar); } else { this._isTyping false; this.node.emit(typing-complete); EventHandler.emitEvents(this.onComplete); } } private updateDisplay(): void { if (this.label) this.label.string this._currentText; if (this.richText) this.richText.string this._currentText; // 基础版支持标签 } }富文本高级扩展提示维护性高若 fullText 包含color#ff0000红色文字/color基础 append 即可工作。若需更精确不打断标签可在 parseTokens 中将文本拆成“可见字符 标签”队列逐个处理可见字符即可。4. DialogueEntry对话条目数据// DialogueEntry.ts ccclass(DialogueEntry) export class DialogueEntry { property(CCString) public id: string ; property(CCString) public speaker: string ; property(CCString) public text: string ; property(CCString) public portrait: string ; property(TypewriterConfig) public config: TypewriterConfig new TypewriterConfig(); property(CCBoolean) public autoNext: boolean false; property(CCString) public nextId: string ; public static fromJSON(json: any): DialogueEntry { /* ... */ } }5. DialogueManager对话管理器 存档集成这是系统的业务核心负责加载 JSON 对话表、切换对话、更新 UI、自动保存进度。// DialogueManager.ts 关键存档部分已集成 import { sys } from cc; import { SaveData } from ./SaveData; ccclass(DialogueManager) export class DialogueManager extends Component { // ... 属性绑定typewriter、portraitSprite、speakerLabel 等 private _dialogues: DialogueEntry[] []; private _currentIndex: number -1; // 加载对话 JSON public loadDialoguesFromJSON(jsonPath: string dialogues/main) { /* ... */ } public startDialogue(index: number) { /* 更新 UI、应用 config、开始打字 */ } public nextDialogue() { /* ... */ } // 存档系统 private getSaveKey(slot: number 1): string { return dialogue_save_${this.defaultDialogueGroup}_slot${slot}; } public saveProgress(slot: number 1): void { if (this._currentIndex 0) return; const entry this._dialogues[this._currentIndex]; const saveData new SaveData(this.defaultDialogueGroup); saveData.currentDialogueId entry.id; saveData.currentIndex this._currentIndex; sys.localStorage.setItem(this.getSaveKey(slot), JSON.stringify(saveData.toJSON())); console.log(存档成功槽位 ${slot}对话ID ${entry.id}); } public loadProgress(slot: number 1): boolean { const jsonStr sys.localStorage.getItem(this.getSaveKey(slot)); if (!jsonStr) return false; try { const saveData SaveData.fromJSON(JSON.parse(jsonStr)); let index this._dialogues.findIndex(d d.id saveData.currentDialogueId); if (index -1) index saveData.currentIndex; if (index 0) { this.startDialogue(index); return true; } } catch (e) { console.error(读档失败, e); } return false; } public deleteSave(slot: number 1) { sys.localStorage.removeItem(this.getSaveKey(slot)); } // 打字完成时自动保存 private _onTypingComplete() { this.saveProgress(); // 关键自动保存进度 // ... 处理 autoNext 等 } }SaveData.ts轻量存档结构仅保存dialogueGroup、currentDialogueId、currentIndex、timestamp和extraData避免存储大量冗余文本。6. JSON 数据驱动示例dialogues/main.json[{id:d001,speaker:村长,text:勇者你终于醒了[pause500] 魔王又在作乱了,portrait:portraits/village-chief,config:{speed:0.04,canSkip:true},autoNext:false}]7. 使用与集成示例在场景控制器中start() { this.dialogueManager.loadDialoguesFromJSON(dialogues/main); const loaded this.dialogueManager.loadProgress(1); // 启动时读档 if (!loaded) { // 从头开始 } } onClickContinue() { this.dialogueManager.nextDialogue(); } onClickSave() { this.dialogueManager.saveProgress(1); }

相关文章:

Cocos Creator 3.x 高维护性打字机对话系统设计与实现

在 Cocos Creator 项目中,对话系统是 RPG、冒险、视觉小说等类型游戏的核心功能之一。如何设计一个维护性高、可扩展、策划友好、支持存档的打字机(Typewriter)系统,是许多开发者面临的挑战。 该系统采用组件化 配置化 JSON 数…...

SEATA分布式事务——AT模式一

简介 AI Agent 不仅仅是一个能聊天的机器人(如普通的 ChatGPT),而是一个能够感知环境、进行推理、自主决策并调用工具来完成特定任务的智能系统,更够完成更为复杂的AI场景需求。 AI Agent 功能 根据查阅的资料,agent的…...

从数据采集到回放验证:ADTF 适配 ROS 的 ADAS 测试实践谒

一、简化查询 1. 先看一下查询的例子 /// /// 账户获取服务 /// /// /// public class AccountGetService(AccountTable table, IShadowBuilder builder) {private readonly SqlSource _source new(builder.DataSource);private readonly IParamQuery _accountQuery build…...

MLX9062x红外热成像传感器驱动开发与温度解算详解

1. MLX9062x 红外热成像阵列传感器驱动深度解析MLX9062x 系列是比利时 Melexis 公司推出的非接触式红外温度传感芯片家族,包含 MLX90620(164 像素)与 MLX90621(164 像素,但支持更高帧率与增强校准)两款核心…...

全志科技Linux驱动开发面试经验与Cache一致性解析

1. 全志科技Linux驱动开发工程师面试全解析作为一名在嵌入式Linux领域摸爬滚打多年的老司机,最近刚经历了全志科技的社招面试。这家国产芯片大厂的面试风格相当有特色,特别是对Cache一致性和驱动开发细节的考察,堪称"灵魂拷问"级别…...

2024版:从零到一,手把手教你完成UniApp支付宝支付功能配置

1. 为什么需要UniApp支付宝支付功能? 移动应用开发中,支付功能几乎是必备模块。作为国内主流支付方式之一,支付宝支付覆盖了超过10亿用户,接入支付宝意味着你的应用可以触达绝大多数国内用户。UniApp作为跨平台开发框架&#xff0…...

Qt键盘控制按钮实战:用WASD键玩转UI交互(附完整代码)

Qt键盘控制按钮实战:用WASD键玩转UI交互(附完整代码) 想象一下,当你正在开发一款自助点餐系统时,突然发现触摸屏失灵了——这种场景下,键盘控制的UI交互能力就成了救命稻草。Qt框架提供的键盘事件处理机制&…...

Oracle 18c新特性实战:5分钟搞定DataGuard备库修复(附常见错误排查)

Oracle 18c DataGuard备库修复实战:从归档缺失到坏块处理的完整指南 凌晨三点,当手机铃声刺破夜空时,我知道又一个不眠之夜开始了。监控系统显示生产备库出现了47-55号归档缺失,而主库的归档日志早已被清理。传统解决方案需要手动…...

企业官网源码_公司网站模板_自适应手机端

一、源码下载平台:企业建站的“数字工具箱” 1. 开源生态驱动创新 GitHub、Gitee等全球开源代码托管平台,汇聚了数百万企业级项目。以GitHub为例,其企业官网源码库涵盖电商、教育、金融等20余个行业,包含完整的前端框架&#xf…...

一文学习 工作流开发 BPMN、 Flowable牌

一、什么是requests? requests 是一个用于发送HTTP请求的 Python 库。 它可以帮助你: 轻松发送GET、POST、PUT、DELETE等请求 处理Cookie、会话等复杂性 自动解压缩内容 处理国际化域名和URL 二、应用场景 requests 广泛应用于以下实际场景: …...

2026届学术党必备的五大AI学术工具解析与推荐

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 想要降低文本的AI生成特征,就得从措辞、逻辑以及情感这三方面去调整指令。在词汇…...

2026届最火的十大AI科研工具实际效果

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 为了把文本被判定成是由AI生成内容(AIGC)的可能性给降低下来&#xf…...

STM32duino双VL53L1X激光测距库详解

1. 项目概述STM32duino X-NUCLEO-53L1A1 是一个面向 Arduino 兼容生态的 STM32 平台专用驱动库,专为意法半导体(STMicroelectronics)官方扩展板 X-NUCLEO-53L1A1 设计。该扩展板搭载两颗 VL53L1X 飞行时间(Time-of-Flight, ToF&am…...

从‘抓不住’到‘抓得稳’:手把手教你用Gazebo Grasp_fix插件搞定Robotiq夹爪仿真(含物体建模技巧)

从‘抓不住’到‘抓得稳’:Gazebo Grasp_fix插件与Robotiq夹爪仿真实战指南 在机器人仿真领域,Gazebo作为一款强大的物理仿真工具,为开发者提供了测试和验证机器人算法的虚拟环境。然而,当涉及到机械臂抓取任务时,许多…...

2024版IntelliJ IDEA中文设置保姆级教程(附社区版/专业版差异)

2024版IntelliJ IDEA中文设置全攻略:从安装到疑难排错 刚接触IntelliJ IDEA的开发者常被其强大的功能所震撼,但英文界面却成了第一道门槛。作为JetBrains家族的旗舰IDE,2024版本在本地化支持上有了显著改进,但专业版与社区版的汉化…...

JBoltAI Agent OS 管数字帮手的企业大管家

如果公司里每个人都有了智能体这个数字帮手,几十上百个帮手一起干活,没人管肯定乱套:有的可能乱翻公司数据,有的学会的好技能就自己用,管理者也不知道大家的数字帮手都在干些啥。而 JBoltAI Agent OS,就是管…...

前端加密的隐秘陷阱:Crypto-JS与JSEncrypt常见误用与解决方案

前端加密的隐秘陷阱:Crypto-JS与JSEncrypt常见误用与解决方案 1. 密钥管理的致命疏忽 在项目评审中,我经常发现开发者将加密密钥直接硬编码在JavaScript文件里。这种看似方便的做法实际上让加密形同虚设——攻击者只需查看源代码就能获取密钥&#xff0c…...

C语言字符串与指针操作技巧解析

1. 字符串的本质与指针操作在嵌入式开发中,字符串处理是最基础也是最重要的技能之一。很多人虽然每天都在使用字符串,但对它的本质理解却不够深入。实际上,C语言中的字符串本质上就是一个字符指针,它指向内存中连续存储的字符序列…...

LPC11U24单总线DHT22/RHT03轻量驱动实现

1. RHT03传感器驱动库深度解析:面向LPC11U24平台的轻量级DHT22/RHT03固件实现1.1 项目背景与工程定位RHT03是DHT22温湿度传感器的兼容型号,采用单总线数字通信协议,具备0.5℃温度精度与2%RH湿度精度,工作电压范围3.3–5.5V&#x…...

【typst-rs】info.rs文件

Typst Info 命令代码解析 这段代码是 Typst 编译器 typst info 命令的实现,用于显示 Typst 环境信息,包括版本、构建配置、运行时特性、字体配置、包配置和相关环境变量。 主要数据结构 Info 结构体 struct Info {version: &static str, // Typst…...

ID12RFID库详解:嵌入式125kHz RFID读卡实践指南

1. ID12RFID 库深度解析:面向嵌入式系统的 125kHz RFID 标签读取实践指南ID12RFID 是一个专为嵌入式平台设计的轻量级 C/C 库,用于驱动 ID-12(及兼容型号如 ID-20、RDM6300)系列 125kHz 低频 RFID 模块。该模块采用 ASK 调制方式&…...

【算法日记】Day 9 动态规划专题——最长递增子序列问题及扩展

Abstract:#动态规划 #最长递增子序列 #二分查找 #排序 1. 题目 题目:LeetCode 354. 俄罗斯套娃信封核心思路:先将信封按宽度升序排序,若宽度相同则按高度降序排序。然后对排序后的高度序列求最长递增子序列(LIS&…...

STM32总线架构解析与性能优化实战

1. STM32单片机内部总线架构概述作为嵌入式开发者,理解STM32单片机的内部总线结构是优化代码性能的关键。在Cortex-M3架构的STM32F1系列中,总线系统就像一座精心设计的立交桥网络,各司其职又相互配合。我第一次调试DMA传输卡顿时,…...

【typst-rs】greet.rs文件

以下是对greet.rs的详细解析。 use std::io::{self, Read};/// This is shown to users who just type typst the first time. #[rustfmt::skip] const GREETING: &str color_print::cstr!("\ <s>Welcome to Typst, we are glad to have you here!</> ❤…...

嵌入式系统软件抗干扰技术实战解析

1. 嵌入式系统抗干扰技术概述在工业控制、智能家居和物联网设备等嵌入式应用场景中&#xff0c;电磁干扰、电源波动等环境因素常常导致系统运行异常。作为一名有十年嵌入式开发经验的工程师&#xff0c;我处理过数十起由干扰引起的系统故障案例。硬件抗干扰措施如屏蔽、滤波固然…...

从《节奏医生》到你的游戏:拆解Koreographer Pro版如何实现高级音频集成(Wwise/FMOD)

从《节奏医生》到你的游戏&#xff1a;Koreographer Pro版如何实现高级音频集成&#xff08;Wwise/FMOD&#xff09; 在《节奏医生》这类音游中&#xff0c;玩家按键与音乐节拍的完美同步是游戏体验的核心。这种精准的音频同步背后&#xff0c;往往需要复杂的音频中间件集成。对…...

I2C总线原理与应用实战指南

1. I2C总线基础概念解析I2C&#xff08;Inter-Integrated Circuit&#xff09;总线是飞利浦半导体&#xff08;现NXP&#xff09;在1980年代开发的一种同步、多主从架构的串行通信总线。作为一名嵌入式工程师&#xff0c;我几乎在每个项目中都会用到这个看似简单却功能强大的两…...

从零开始:在RK3588上运行RKNN版YOLOv5目标检测(保姆级教程)

从零开始&#xff1a;在RK3588上运行RKNN版YOLOv5目标检测&#xff08;保姆级教程&#xff09; RK3588作为Rockchip新一代旗舰级SoC&#xff0c;其内置的NPU模块为边缘计算场景提供了强大的AI推理能力。本教程将手把手带您完成YOLOv5目标检测模型在RK3588开发板上的完整部署流程…...

显示器EDID数据解析全攻略:从制造商ID到色彩特性的秘密

显示器EDID数据解析全攻略&#xff1a;从制造商ID到色彩特性的秘密 当你连接一台新显示器时&#xff0c;操作系统是如何知道它的最佳分辨率和刷新率的&#xff1f;答案就藏在EDID&#xff08;Extended Display Identification Data&#xff09;这个小小的数据块中。EDID是显示器…...

ESP32伺服与PWM控制库:硬件自适应资源管理

1. 项目概述ESP32ServoController 是一款专为 ESP32 系列微控制器设计的高性能 PWM 与伺服控制库。它并非对 Espressif 官方 LEDC&#xff08;LED Control&#xff09;外设驱动的简单封装&#xff0c;而是基于其硬件架构进行深度抽象与工程化重构的底层控制框架。该库的核心设计…...