HarmonyOS 5.0 PC应用开发实战:构建跨设备协同的桌面生产力工具

HarmonyOS 5.0 PC应用开发实战:构建跨设备协同的桌面生产力工具

文章目录

在这里插入图片描述

每日一句正能量

当你感到压力大,觉得不顺心的时候,就去逛逛菜市场……当看到年迈的老人,严寒酷暑,一小堆菜,一小堆水果,只为挣那几块几十块钱的家用,你所有的矫情和懒惰都会掉在地上碎成渣!

前言

摘要: 本文基于HarmonyOS 5.0.0版本,详细介绍如何开发一款具备跨设备协同能力的PC端生产力应用。通过实战案例,深入讲解ArkUI-X在PC端的适配、分布式软总线技术、以及多窗口管理等核心能力,为开发者提供完整的PC应用开发解决方案。


一、HarmonyOS PC应用开发背景与机遇

1.1 生态发展现状

随着HarmonyOS NEXT的正式发布,鸿蒙生态正式进入"纯血"时代。华为在2024年开发者大会上宣布,HarmonyOS PC版将于2025年全面商用,这意味着PC端将成为鸿蒙生态的重要拼图。对于开发者而言,这是一个巨大的蓝海市场——目前Windows桌面应用市场饱和,而鸿蒙PC应用尚处于起步阶段,先发优势明显。

1.2 技术架构特点

HarmonyOS PC应用并非简单的移动端移植,而是基于统一生态的重新设计:

  • 统一内核:采用与移动端相同的OpenHarmony内核,确保API一致性
  • 多窗口架构:支持自由窗口、分屏、多开等PC典型交互模式
  • 键鼠优化:原生支持键盘快捷键、鼠标右键菜单、滚轮缩放等操作
  • 跨端协同:通过分布式技术实现手机、平板、PC间的无缝流转

二、实战项目:跨设备Markdown编辑器

2.1 项目需求分析

我们将开发一款名为**“HarmonyMark”**的Markdown编辑器,核心功能包括:

  1. 基础编辑:支持Markdown语法高亮、实时预览、文件管理
  2. PC特性:多标签页、快捷键支持、拖拽打开文件
  3. 跨端协同:手机拍照→PC插入、平板手绘→PC同步、文件跨设备流转

2.2 技术选型

模块技术方案说明
UI框架ArkUI-X支持PC端响应式布局
状态管理AppStorage + LocalStorage跨Ability数据共享
分布式能力DistributedObject + 软总线跨设备数据同步
文件处理@ohos.file.fsPC端文件系统访问
窗口管理@ohos.window多窗口生命周期管理

三、核心代码实现

3.1 工程架构搭建

首先创建Stage模型工程,配置PC设备支持:

// entry/src/main/module.json5{"module":{"name":"entry","type":"entry","deviceTypes":["default","tablet","2in1"// 支持PC/二合一设备],"abilities":[{"name":"EntryAbility","srcEntry":"./ets/entryability/EntryAbility.ets","description":"$string:EntryAbility_desc","icon":"$media:layered_image","label":"$string:EntryAbility_label","startWindowIcon":"$media:startIcon","startWindowBackground":"$color:start_window_background","exported":true,"skills":[{"entities":["entity.system.home"],"actions":["action.system.home"]}],// PC端多窗口配置"windowMode":"multi_window","maxWindowRatio":"4:3","minWindowRatio":"1:2"}]}}

3.2 PC端响应式布局

HarmonyOS PC应用需要适配多种窗口尺寸,采用栅格系统实现响应式:

// MainPage.etsimport{ BreakpointSystem, BreakpointType }from'../utils/BreakpointSystem'@Entry@Component struct MainPage {@StorageProp('currentBreakpoint') currentBreakpoint:string='sm'private breakpointSystem: BreakpointSystem =newBreakpointSystem()// 编辑器状态@State currentFile: FileItem |null=null@State isPreviewMode:boolean=false@State editorContent:string=''aboutToAppear(){// 注册断点监听this.breakpointSystem.register()// 初始化分布式数据this.initDistributedData()}aboutToDisappear(){this.breakpointSystem.unregister()}build(){GridRow({ columns:{ sm:4, md:8, lg:12},// 响应式列数 gutter:{ x:12, y:12}, breakpoints:{ value:['320vp','600vp','840vp'], reference: BreakpointsReference.WindowSize }}){// 左侧文件栏:lg显示,sm/md隐藏GridCol({ span:{ sm:0, md:2, lg:3}, offset:{ sm:0, md:0, lg:0}}){FileSidebar({onFileSelect:(file: FileItem)=>this.handleFileSelect(file)})}.backgroundColor('#f5f5f5').height('100%')// 中间编辑区GridCol({ span:{ sm:4, md:6, lg:this.isPreviewMode ?5:9}}){EditorPanel({ content: $editorContent,onContentChange:(val:string)=>this.handleContentChange(val)})}.padding(16)// 右侧预览区:仅lg且预览模式显示GridCol({ span:{ sm:0, md:0, lg:4}}){if(this.currentBreakpoint ==='lg'&&this.isPreviewMode){PreviewPanel({ markdown:this.editorContent })}}.backgroundColor('#fafafa')}.width('100%').height('100%').onBreakpointChange((breakpoint)=>{ AppStorage.setOrCreate('currentBreakpoint', breakpoint)})}// 处理文件选择privatehandleFileSelect(file: FileItem){this.currentFile = file // 读取文件内容 fs.readText(file.uri).then((content)=>{this.editorContent = content // 同步到分布式数据this.syncToDistributed(file.uri, content)})}// 内容变更自动保存privatehandleContentChange(content:string){this.editorContent = content if(this.currentFile){this.autoSave(this.currentFile.uri, content)}}}

3.3 分布式数据同步实现

核心功能:实现PC与手机间的实时内容同步:

// DistributedEditorManager.etsimport distributedObject from'@ohos.data.distributedDataObject'import distributedDeviceManager from'@ohos.distributedDeviceManager'classEditorData{ uri:string='' content:string='' lastModified:number=0 deviceId:string=''}exportclassDistributedEditorManager{private distributedObject: distributedObject.DistributedObject |null=nullprivate sessionId:string='harmonymark_editor_session'private deviceManager: distributedDeviceManager.DeviceManager |null=null// 创建分布式数据对象asynccreateDistributedObject(initialData: EditorData){try{this.distributedObject = distributedObject.create(getContext(this),this.sessionId, initialData )// 监听数据变更this.distributedObject.on('change',(sessionId, fields)=>{console.info(`Data changed from ${sessionId}: ${JSON.stringify(fields)}`)this.handleRemoteChange(fields)})// 绑定到本地awaitthis.distributedObject.setSessionId(this.sessionId)console.info('Distributed object created successfully')}catch(err){console.error('Failed to create distributed object:', err)}}// 同步数据到所有设备asyncsyncContent(uri:string, content:string){if(!this.distributedObject)returnconst updateData: EditorData ={ uri: uri, content: content, lastModified: Date.now(), deviceId:this.getLocalDeviceId()}// 更新分布式对象this.distributedObject.uri = updateData.uri this.distributedObject.content = updateData.content this.distributedObject.lastModified = updateData.lastModified this.distributedObject.deviceId = updateData.deviceId console.info('Content synced to distributed object')}// 处理远程数据变更privatehandleRemoteChange(fields:Array<string>){if(!this.distributedObject)return// 检查是否是其他设备的更新if(fields.includes('content')&&this.distributedObject.deviceId !==this.getLocalDeviceId()){const remoteContent =this.distributedObject.content const remoteUri =this.distributedObject.uri // 触发UI更新 AppStorage.setOrCreate('remoteContent', remoteContent) AppStorage.setOrCreate('remoteUri', remoteUri)// 显示协同提示this.showCollaborationNotification(remoteContent)}}// 获取在线设备列表asyncgetAvailableDevices():Promise<Array<distributedDeviceManager.DeviceBasicInfo>>{try{this.deviceManager = distributedDeviceManager.createDeviceManager(getContext(this).bundleName)returnthis.deviceManager.getAvailableDeviceListSync()}catch(err){console.error('Failed to get devices:', err)return[]}}privategetLocalDeviceId():string{returnthis.deviceManager?.getLocalDeviceNetworkId()||''}privateshowCollaborationNotification(content:string){// 实现协同提示UI promptAction.showToast({ message:'其他设备已更新内容', duration:2000})}}

3.4 PC端多窗口管理

实现类似VS Code的多窗口编辑体验:

// MultiWindowManager.etsimport window from'@ohos.window'exportclassMultiWindowManager{privatestatic instance: MultiWindowManager private windowMap: Map<string, window.Window>=newMap()private mainWindow: window.Window |null=nullstaticgetInstance(): MultiWindowManager {if(!MultiWindowManager.instance){ MultiWindowManager.instance =newMultiWindowManager()}return MultiWindowManager.instance }// 初始化主窗口asyncinitMainWindow(){this.mainWindow =await window.getLastWindow(getContext(this))awaitthis.setupWindowConfig(this.mainWindow,'main')}// 创建新窗口打开文件asyncopenNewWindow(fileUri:string, fileName:string):Promise<void>{try{// 创建子窗口const subWindow =await window.createSubWindow(getContext(this),`editor_${Date.now()}`)const windowId = subWindow.getWindowProperties().id.toString()// 配置窗口属性awaitthis.setupWindowConfig(subWindow,'sub')// 设置窗口内容await subWindow.setUIContent('pages/EditorWindow',(data)=>{// 传递参数 AppStorage.setOrCreate('windowFileUri', fileUri) AppStorage.setOrCreate('windowFileName', fileName)})// 显示窗口await subWindow.showWindow()// 移动到合适位置(级联窗口效果)awaitthis.cascadeWindow(subWindow)// 保存引用this.windowMap.set(windowId, subWindow)// 监听窗口关闭 subWindow.on('windowStageDestroy',()=>{this.windowMap.delete(windowId)})}catch(err){console.error('Failed to create sub window:', err)}}// 配置窗口属性privateasyncsetupWindowConfig(win: window.Window, type:'main'|'sub'){// 设置窗口大小范围await win.setWindowLimits({ minWidth: type ==='main'?800:600, minHeight: type ==='main'?600:400, maxWidth:3840, maxHeight:2160})if(type ==='sub'){// 子窗口默认大小await win.resize(1000,700)// 启用窗口拖拽调整大小await win.setWindowTouchable(true)}// PC端特定优化await win.setWindowDecorVisible(true)// 显示系统标题栏await win.setWindowBackgroundColor('#ffffff')}// 级联窗口布局privateasynccascadeWindow(win: window.Window){const offset =this.windowMap.size *30const display =await window.getLastWindow(getContext(this)).getWindowProperties().displayId // 基于主窗口位置偏移await win.moveWindowTo(100+ offset,100+ offset)}// 分屏模式支持asyncenterSplitScreenMode(){if(!this.mainWindow)returnawaitthis.mainWindow.setWindowMode(window.WindowMode.SPLIT_PRIMARY)}// 获取所有打开的窗口getAllWindows():Array<window.Window>{returnArray.from(this.windowMap.values())}// 关闭所有子窗口asynccloseAllSubWindows(){for(const[id, win]ofthis.windowMap){await win.destroyWindow()}this.windowMap.clear()}}

3.5 键盘快捷键系统

PC应用的核心体验,实现专业编辑器级快捷键:

// KeyboardShortcutManager.etsimport{ KeyCode }from'@kit.InputKit'interfaceShortcutConfig{ key: KeyCode modifiers:Array<'ctrl'|'shift'|'alt'>action:()=>void description:string}exportclassKeyboardShortcutManager{private shortcuts: Map<string, ShortcutConfig>=newMap()private isListening:boolean=false// 注册默认快捷键registerDefaultShortcuts(){this.register({ key: KeyCode.KEY_S, modifiers:['ctrl'],action:()=>this.saveFile(), description:'保存文件'})this.register({ key: KeyCode.KEY_N, modifiers:['ctrl'],action:()=>this.newFile(), description:'新建文件'})this.register({ key: KeyCode.KEY_O, modifiers:['ctrl'],action:()=>this.openFile(), description:'打开文件'})this.register({ key: KeyCode.KEY_Z, modifiers:['ctrl'],action:()=>this.undo(), description:'撤销'})this.register({ key: KeyCode.KEY_Z, modifiers:['ctrl','shift'],action:()=>this.redo(), description:'重做'})this.register({ key: KeyCode.KEY_B, modifiers:['ctrl'],action:()=>this.insertBold(), description:'粗体'})this.register({ key: KeyCode.KEY_P, modifiers:['ctrl','shift'],action:()=>this.togglePreview(), description:'切换预览'})// 开始监听this.startListening()}register(config: ShortcutConfig){const key =this.getShortcutKey(config)this.shortcuts.set(key, config)}privatestartListening(){if(this.isListening)return// 使用InputKit监听键盘事件 inputMonitor.on('key',(event)=>{if(event.type !=='keyDown')returnconst pressedKey =this.getShortcutKey({ key: event.keyCode, modifiers:this.getActiveModifiers(event)}as ShortcutConfig)const shortcut =this.shortcuts.get(pressedKey)if(shortcut){ event.stopPropagation() shortcut.action()console.info(`Shortcut triggered: ${shortcut.description}`)}})this.isListening =true}privategetShortcutKey(config: ShortcutConfig):string{const mods = config.modifiers.sort().join('+')return`${mods}+${config.key}`}privategetActiveModifiers(event: KeyEvent):Array<string>{const mods:Array<string>=[]if(event.ctrlKey) mods.push('ctrl')if(event.shiftKey) mods.push('shift')if(event.altKey) mods.push('alt')return mods }// 快捷键动作实现privatesaveFile(){const content = AppStorage.get<string>('currentContent')||''const uri = AppStorage.get<string>('currentUri')if(uri){ fs.writeText(uri, content) promptAction.showToast({ message:'保存成功'})}}privatenewFile(){// 创建新文件逻辑 router.pushUrl({ url:'pages/Editor', params:{ newFile:true}})}privateopenFile(){// 打开文件选择器let documentPicker =newpicker.DocumentViewPicker(getContext(this)) documentPicker.select().then((result)=>{if(result.length >0){ AppStorage.setOrCreate('selectedFileUri', result[0])}})}privateundo(){// 调用编辑器撤销 AppStorage.setOrCreate('editorAction','undo')}privateredo(){ AppStorage.setOrCreate('editorAction','redo')}privateinsertBold(){ AppStorage.setOrCreate('editorInsert','****')}privatetogglePreview(){const current = AppStorage.get<boolean>('isPreviewMode')||false AppStorage.setOrCreate('isPreviewMode',!current)}}

四、跨设备协同场景实战

4.1 手机拍照插入PC文档

利用分布式文件系统实现:

// PhotoTransferManager.etsimport distributedFile from'@ohos.file.distributedFile'exportclassPhotoTransferManager{// 发起拍照请求到手机asyncrequestPhotoFromPhone():Promise<string>{// 查找在线手机设备const devices =awaitthis.getPhoneDevices()if(devices.length ===0){thrownewError('No phone device found')}const targetDevice = devices[0]// 通过分布式软总线发送拍照指令const session =awaitthis.createSession(targetDevice.networkId)await session.sendMessage({ action:'TAKE_PHOTO'})// 等待照片传输完成returnnewPromise((resolve, reject)=>{ session.onMessage((msg)=>{if(msg.type ==='PHOTO_READY'){// 获取分布式文件路径const distributedPath = msg.data.path // 复制到本地this.copyToLocal(distributedPath).then(resolve).catch(reject)}})setTimeout(()=>reject(newError('Photo transfer timeout')),30000)})}privateasynccopyToLocal(distributedPath:string):Promise<string>{const fileName =`photo_${Date.now()}.jpg`const localPath =getContext(this).filesDir +'/'+ fileName // 使用分布式文件API复制await distributedFile.copyFile(distributedPath, localPath)return localPath }}

4.2 平板手绘同步到PC

利用分布式数据对象实时同步手绘数据:

// DrawingSyncManager.etsinterfaceDrawingPoint{ x:number y:number pressure:number timestamp:number}interfaceDrawingStroke{ points:Array<DrawingPoint> color:string width:number}exportclassDrawingSyncManager{private distributedObj:any=nullasyncinit(){this.distributedObj = distributedObject.create(getContext(this),'drawing_session',{ strokes:[]asArray<DrawingStroke>})// 监听笔画数据this.distributedObj.on('change',(sessionId, fields)=>{if(fields.includes('strokes')){const strokes =this.distributedObj.strokes this.renderStrokes(strokes)}})}// 平板端调用:添加笔画asyncaddStroke(stroke: DrawingStroke){const currentStrokes =this.distributedObj.strokes ||[] currentStrokes.push(stroke)this.distributedObj.strokes = currentStrokes }// PC端调用:渲染笔画到CanvasprivaterenderStrokes(strokes:Array<DrawingStroke>){const canvas = AppStorage.get<CanvasRenderingContext2D>('drawingCanvas')if(!canvas)return canvas.clearRect(0,0, canvas.width, canvas.height) strokes.forEach(stroke =>{ canvas.beginPath() canvas.strokeStyle = stroke.color canvas.lineWidth = stroke.width canvas.lineCap ='round' canvas.lineJoin ='round' stroke.points.forEach((point, index)=>{if(index ===0){ canvas.moveTo(point.x, point.y)}else{ canvas.lineTo(point.x, point.y)}}) canvas.stroke()})}}

五、性能优化与最佳实践

5.1 大文件处理优化

Markdown文件可能很大,需要虚拟列表优化:

// VirtualListController.etsclassVirtualListController{private itemHeight:number=40private visibleCount:number=50private bufferCount:number=10// 计算可见区域getVisibleRange(scrollOffset:number):{ start:number, end:number}{const start = Math.floor(scrollOffset /this.itemHeight)-this.bufferCount const end = start +this.visibleCount +this.bufferCount *2return{ start: Math.max(0, start), end: Math.min(this.totalItems, end)}}// 渲染优化buildVirtualList(items:Array<string>){List({ space:0}){LazyForEach(this.dataSource,(item:string, index:number)=>{ListItem(){MarkdownLine({ content: item, lineNumber: index +1})}.height(this.itemHeight).recycle(true)// 启用回收复用},(item:string, index:number)=> index.toString())}.cachedCount(this.bufferCount)// 缓存缓冲区.onScroll((scrollOffset)=>{this.updateVisibleRange(scrollOffset)})}}

5.2 内存管理

PC应用可能长时间运行,需要注意内存泄漏:

// MemoryManager.etsexportclassMemoryManager{privatestatic intervals:Array<number>=[]privatestatic listeners:Array<()=>void>=[]// 安全设置定时器staticsetSafeInterval(callback:()=>void, delay:number):number{const id =setInterval(callback, delay)this.intervals.push(id)return id }// 安全注册事件staticaddSafeListener(event:string,handler:()=>void){ emitter.on(event, handler)this.listeners.push(()=> emitter.off(event, handler))}// 页面销毁时清理staticcleanup(){this.intervals.forEach(id =>clearInterval(id))this.intervals =[]this.listeners.forEach(off =>off())this.listeners =[]// 释放大对象 AppStorage.delete('largeDataCache')}}

六、调试与发布

6.1 PC端调试技巧

# 连接PC设备(需开启开发者模式) hdc list targets hdc shell # 实时查看日志 hdc hilog |grep HarmonyMark # 性能分析 hdc shell hiprofiler -c /data/local/tmp/config.json 

6.2 发布配置

// 配置PC应用图标和分类{"app":{"icon":"$media:pc_icon","label":"HarmonyMark","category":"productivity","pcConfig":{"supportWindowMode":["fullscreen","split","float"],"defaultWindowSize":[1200,800],"minWindowSize":[800,600]}}}

七、总结与展望

本文完整演示了HarmonyOS 5.0 PC应用的核心开发流程,涵盖:

  1. 响应式布局:通过GridRow/GridCol实现PC端自适应
  2. 分布式能力:利用DistributedObject实现跨设备协同
  3. 多窗口管理:支持专业级多文档编辑体验
  4. 键鼠交互:完整的快捷键系统提升效率

未来优化方向:

  • 接入AI能力实现智能Markdown补全
  • 支持插件系统扩展编辑器功能
  • 实现WebDAV云同步

HarmonyOS PC生态正处于快速发展期,开发者应抓住窗口期,提前布局PC应用市场。随着2025年鸿蒙PC的全面商用,早期投入将获得显著的先发优势。


参考资源:


转载自:https://blog.ZEEKLOG.net/u014727709/article/details/158931841
欢迎 👍点赞✍评论⭐收藏,欢迎指正

Read more

Mac Mini M4 跑 AI 模型全攻略:从 Ollama 到 Stable Diffusion 的保姆级配置指南

Mac Mini M4 本地AI模型实战:从零构建你的个人智能工作站 最近身边不少朋友都在讨论,能不能用一台小巧的Mac Mini M4,搭建一个属于自己的AI开发环境。毕竟,不是每个人都有预算去租用云端的高性能GPU,也不是所有项目都适合把数据传到云端处理。我折腾了大概两周,从Ollama到Stable Diffusion,把整个流程走了一遍,发现M4芯片的潜力远超预期。这篇文章,就是把我踩过的坑、验证过的有效配置,以及一些提升效率的小技巧,毫无保留地分享给你。无论你是想本地运行大语言模型进行对话和创作,还是想离线生成高质量的AI图像,这篇指南都能帮你把Mac Mini M4变成一个得力的AI伙伴。 1. 环境准备与基础配置 在开始安装任何AI工具之前,确保你的系统环境是干净且高效的,这能避免后续无数莫名其妙的依赖冲突。Mac Mini M4出厂预装的是较新的macOS版本,但这还不够。 首先,打开“系统设置” -> “通用” -> “软件更新”,确保你的macOS已经更新到可用的最新版本。苹果对Metal图形API和神经网络引擎的优化通常会随着系统更新而提升,这对于后续运

By Ne0inhk
知识库问答机器人:基于SpringAI+RAG的完整实现

知识库问答机器人:基于SpringAI+RAG的完整实现

一、引言 随着大语言模型的快速发展,RAG(Retrieval-Augmented Generation)技术已成为构建知识库问答系统的核心技术之一。本文将带领大家从零开始,使用Spring AI框架构建一个支持文档上传的知识库问答机器人,帮助大家深入理解RAG技术的核心原理和实践应用。 1.1 什么是RAG? RAG(检索增强生成)是一种结合了信息检索和文本生成的技术。它的基本工作流程是: 用户提出问题 系统从知识库中检索相关信息 大语言模型基于检索到的信息生成答案 从系统设计角度触发,RAG 的核心作用可以被描述为: 在LLM调用生成响应之前,由系统动态构造一个“最小且相关的知识上下文”。 请注意两个关键词: 动态 :每次问题都不同,检索的知识也不同(比如用户问 A 产品时找 A 的文档,问 B 产品时找 B 的文档) 最小 :只注入必要信息(比如用户问 “A 产品的定价”,就只塞定价相关的片段,而非整份产品手册) RAG可以有效的弥补上下文窗口的先天不足:不再需要把所有知识塞进窗口,

By Ne0inhk
Stable Diffusion 秋叶大神2025最新整合一键安装包

Stable Diffusion 秋叶大神2025最新整合一键安装包

这段时间我在折腾 Stable Diffusion,期间试过很多安装方式。有手动安装的,也有别人做好的整合包。手动安装的方式对环境要求高,步骤也多,系统要装 Python,要装依赖,还要配好运行库,哪一步出错都要重新查资料,挺消耗时间。后来了解到秋叶大神做的整合一键安装包,这个版本省掉了很多折腾,对新手比较友好。 我自己把安装流程整理了一遍,又结合网上的信息,把一些需要注意的地方写下来,希望能帮到想尝试 Stable Diffusion 的人。 这里完整下载链接 秋叶整合包是什么 这个整合包属于别人已经帮你配好的版本,里面把 Stable Diffusion WebUI、模型管理、插件、运行环境都准备好了。下载之后按照提示解压,点一下启动脚本就能跑起来,不需要另外去折腾环境。 整合包里放的 WebUI 是常见的 AUTOMATIC1111 版本,所以大部分教程都能直接用。适合想直接出图、想先体验一下模型效果的人。 系统环境方面 我现在用的是 Windows 电脑,所以下面写的内容主要基于

By Ne0inhk
DAY4 基于 OpenClaw + 飞书开放平台实现 AI 新闻推送机器人

DAY4 基于 OpenClaw + 飞书开放平台实现 AI 新闻推送机器人

DAY4 基于 OpenClaw + 飞书开放平台实现 AI 新闻推送机器人 目录 DAY4 基于 OpenClaw + 飞书开放平台实现 AI 新闻推送机器人 前  言 1 环境准备 1.1 华为云开发环境 1.2 ModelArts 代金券与模型服务 1.3 启动 OpenClaw 网关 2 飞书开放平台配置 2.1 创建企业自建应用 2.2 添加机器人能力 2.3 配置应用权限 2.4 发布应用版本 3 OpenClaw 与飞书集成 3.1 配置 OpenClaw

By Ne0inhk