SideShow Gadget本地存储实现

简介: 标准的Sideshow设备连接PC后,由PC上Sideshow管理中心负责向Sideshow设备添加Gadget应用,该Gadget实例被Sideshow设备缓存到内存中。当Sideshow设备未连接PC,掉电重启后,所有原先添加的Gadget信息将全部丢失。

说明
标准的Sideshow设备连接PC后,由PC上Sideshow管理中心负责向Sideshow设备添加Gadget应用,该Gadget实例被Sideshow设备缓存到内存中。当Sideshow设备未连接PC,掉电重启后,所有原先添加的Gadget信息将全部丢失。

本功能的目的就是,Sideshow设备在上述情况下重启,可以自动恢复原先的Gadget应用。

2、Sideshow功能模块关系图
image.png

 

SideShow设备启动后,主界面优先启动,显示一些本地Gadget和系统基本信息(日期时间等)。于此同时Message组件也开始工作,全权接管和PC Sideshow相关的通信功能,新的Gadget也就是通过该通道添加到Sideshow设备中来,新的画面、信息、状态也源源不断由PC向SideShow设备而来,此时SideShow设备也会根据Gadget的操作情况,主动向PC Sideshow管理中心回传gadget的状态信息(包括是否获得焦点、按键信息、菜单选择信息等等)。

当Gadget相关信息由PC过来时,会先存储到Cache中去,并不直接显示相关画面,UI引擎根据需要,会随时从Cache中获取相关界面元素。

3、Gadget本地存储实现
3.1、实现思路
虽然所有的UI信息都缓存到Cache中,但是由于显示Gadget是一个动态过程,是显示引擎根据需要,动态从Cache中构建Gadget。而该需要就是Message组件通知并促成的。如果仅仅把Cache的内存持久化,而不启动消息循环引擎,Gadget是无法加载并应用的。

一切的根源来至于Message组件中的各种不同的消息,所以最直接的思路就是截获并持久化相关消息, 并在设备启动时(要先判断是否联机,如果已联机,则不执行任何代码),读取并投送相关消息,模拟Sideshow设备联机时的相关信息交互,从而实现了Gadget非联机状态情况下的动态加载,由于是通过模拟消息方式实现了Gadget的加载,所以此时Gadget的功能和联机时无异(注意:由于没有联机,所以Gadget显示的信息和内容是非即时的)。

3.2、具体实现
3.2.1 消息结构简介
PC和Sideshow设备相关的消息共74种,消息结构分两部分,一是消息头(包含消息指令),二是Payload(负荷数据),我们要缓存的数据也主要是这两部分。

 

详情可参见《SideShow通信协议》相关文档。

3.2.2 需要持久化的消息指令
1.png
注:考虑到性能,第5项和第6项消息没有做相关的处理。

3.2.3 本地存储格式
首先创建Gadget目录(/ROOT/CADGET_CACHE)

每一个Gadget应用对应一个由其GUID为命名的目录,目录中存放两类信息,一是:index.App,该文件存放了Gadget的名字和图标相关信息;二是Gadget内容信息,命令格式 {内容ID}.Content。

3.2.4 Gadget持久化类代码

  public class GadgetCache

    {

        public static  bool OnlineFlag = false;

        public static bool RunFlag = false;

        public static int CheckCachedDirectory();

        public static string[] EnumApps();

        public static string[] EnumContentItems(string AppId);

        public static int LoadApp(string AppId, out byte[] bytData);

        public static int SaveApp(string AppId, byte[] bytData);

        public static int DeleteApp(string AppId);

        public static int DeleteAllApps();

       public static int LoadContentItem(string AppId, string ContentId, out byte[] bytData);

        public static int SaveContentItem(string AppId, string ContentId, byte[] bytData);

        public static int DeleteContentItem(string AppId, string ContentId);

        public static int DeleteAllContentItems(string AppId);

     }

详细实现代码 略

代码所在的文件:PcMessageHandler.cs

3.2.5 Gadget持久化执行代码

case PacketType.AddApp:

    {

        ClientInfo info = new ClientInfo();

        Guid appId      = command.ReadGuid();

        Guid endptId    = command.ReadGuid();

        info.Name       = command.ReadString();

        info.Policy     = (CachePolicy)command.ReadInt();

        info.OnLineOnly = command.ReadUint() != 0;

        info.Icon48Data = command.ReadByteArray();

        info.Icon32Data = command.ReadByteArray();

        info.Icon16Data = command.ReadByteArray();

 

       shell.ConnectClient(appId, endptId, info);

       if (!GadgetCache.RunFlag) GadgetCache.SaveApp(appId.ToString(), msg.Payload);

        break;

    }

 

case PacketType.AddContentItem:

    {

        Guid appId      = command.ReadGuid();

        Guid endptId    = command.ReadGuid();

        uint contentId  = command.ReadUint();

        byte[] itemData = command.ReadByteArray();

       

        cacheMgr.AddItem(appId, endptId, contentId, itemData);

        if (!GadgetCache.RunFlag) GadgetCache.SaveContentItem(appId.ToString(), contentId.ToString(), msg.Payload);

        break;

    }

case PacketType.DeleteContentItem:

case PacketType.DeleteAllContentItems:

(其它代码 略)

代码所在的文件:PcMessageHandler.cs 中的ProcessMessage函数

 

3.2.6 Gadget本地数据读取及恢复
1、启动

new Thread(new ThreadStart(Load_Gadget)).Start();

代码所在的文件:PcMessageHandler.cs 中的Initialize函数

 

2、数据读取及恢复

void Load_Gadget()

{

    Thread.Sleep(3000);  //wait 3s

    try

    {

        if (!GadgetCache.OnlineFlag)

        {

            Globals.YFShowInfo("Load ...", "Load_Gadget", Globals.YFInfoType.Level_8);

            GadgetCache.RunFlag = true;

            string[] AppIds = GadgetCache.EnumApps();

            byte[] bytData = null;

            foreach (string AppId in AppIds)

            {

                if (GadgetCache.OnlineFlag)

                {

                    GadgetCache.RunFlag = false;

                    return;

                }

                GadgetCache.LoadApp(AppId, out bytData);

                ProcessMessage(GadgetCache.ToAuxMessage(269, bytData));

                string[] ContentIds = GadgetCache.EnumContentItems(AppId);

                foreach (string ContentId in ContentIds)

                {

                    if (GadgetCache.OnlineFlag)

                    {

                        GadgetCache.RunFlag = false;

                        return;

                    }

                    GadgetCache.LoadContentItem(AppId, ContentId, out bytData);

                    ProcessMessage(GadgetCache.ToAuxMessage(276, bytData));

                }

            }

            GadgetCache.RunFlag = false;

            Globals.YFShowInfo("End", "Load_Gadget", Globals.YFInfoType.Level_8);

        }

    }

    catch

    {

        GadgetCache.RunFlag = false;

        Globals.YFShowInfo("Load ...Error", "Load_Gadget", Globals.YFInfoType.Level_8);

    }

}

代码所在的文件:PcMessageHandler.cs 中的PcMessageHandler类

4、其它说明
Sideshow SDK自带的Sideshow模拟器竟然不支持文件系统,所以为了便于调试(在硬件平台上测试是很费时间的),我又扩展了我的MF模拟器,让它也可以支持Sideshow了,不过唯一不爽的是,添加该功能后,以后所有的MF应用程序必须要求VS2008具有管理员权限。

好处是我的MF模拟器可以模拟Sideshow设备连线和掉线,这一点官方的Sideshow模拟器是不支持的,有了这个功能,我才方便测试我的Gadget本地恢复。

 
image.png

 

这是非联机模式,注意Sideshow界面的右上角没有系统联机的状态信息,此外,仔细看,一些非本地Gadget也出现在界面中的了,这时它们的操作和联机一样。

 
image.png

 

这是联机模式,此时出现的Gadget是Sideshow控制面板选中的Gadget。注意界面的右上角出现了联机状态。

 
说明:Sideshow连接中的COM仅仅是一个通道借用,目的是采用串口的操作函数进行数据读写,其实物理串口并没有数据。标准Sideshow模拟器连接的是COM33,我的模拟器和实际设备一致,选用的是COM3。

相关文章
|
消息中间件 存储 大数据
一文读懂 kafka 的事务机制 1
一文读懂 kafka 的事务机制
|
JSON 程序员 定位技术
使用echarts+echarts-gl绘制3d地图,实现地图轮播效果
记录一下大屏开发中使用到的echarts-gl 大屏的页面根据需求前前后后改了几个版本了,地图的样式也改了又改 这里记录一下,因为echarts属性用到的比较多也比较杂,防止以后需要用到忘记了
|
Windows
解决xp下无法通过windows installer服务安装此安装程序包。您必须安装带有更新版本Wi
今天装 tortoisegit 的时候发现安装包不能使用。报错无法通过windows installer服务安装此安装程序包。您必须安装带有更新版本Windows Installer服务的Windows Service Pack解决方案: http://www.microsoft.com/zh-cn/download/confirmation.aspx?id=8483 下载新的更新补丁然后重启之后再次安装tortoisegit 就ok了。
2053 0
|
C语言
keil5如何创建工程
首先双击点击打开我们的keil5的一个软件 新建一个工程,对于单片机的程序来说,每个功能程序都必须要
745 0
keil5如何创建工程
|
4天前
|
人工智能 自然语言处理 JavaScript
2026年Windows+Ollama本地部署OpenClaw保姆级教程:本地AI Agent+阿里云上快速搭建
2026年OpenClaw凭借本地部署、私有化运行的特性,成为打造个人智能体的核心工具,而Ollama作为轻量级本地大模型管理工具,能让OpenClaw摆脱对云端大模型的依赖,实现**本地推理、数据不泄露、全流程私有化**的智能体验。本文基于Windows 11系统,从硬件环境准备、Ollama安装与模型定制、OpenClaw部署配置、技能扩展到常见问题排查,打造保姆级本地部署教程,同时补充阿里云OpenClaw(Clawdbot)快速部署步骤,兼顾本地私有化需求与云端7×24小时运行需求,文中所有代码命令均可直接复制执行,确保零基础用户也能快速搭建属于自己的本地智能体。
5677 32
|
4天前
|
人工智能 JSON JavaScript
手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人
手把手教你用 OpenClaw(v2026.2.22-2)+ 飞书,10分钟零代码搭建专属AI机器人!内置飞书插件,无需额外安装;支持Claude等主流模型,命令行一键配置。告别复杂开发,像聊同事一样自然对话。
2426 7
手把手教你用 OpenClaw + 飞书,打造专属 AI 机器人
|
10天前
|
存储 人工智能 负载均衡
阿里云OpenClaw多Agent实战宝典:从极速部署到AI团队搭建,一个人=一支高效军团
在AI自动化时代,单一Agent的“全能模式”早已无法满足复杂任务需求——记忆臃肿导致响应迟缓、上下文污染引发逻辑冲突、无关信息加载造成Token浪费,这些痛点让OpenClaw的潜力大打折扣。而多Agent架构的出现,彻底改变了这一现状:通过“单Gateway+多分身”模式,让一个Bot在不同场景下切换独立“大脑”,如同组建一支分工明确的AI团队,实现创意、写作、编码、数据分析等任务的高效协同。
4492 29

热门文章

最新文章