f_00000a 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. import { queenApi } from "/node_modules/.vite/deps/queenjs.js?v=aff0ab7e";
  2. import { Controller } from "/src/comm/core/controller.ts";
  3. import { RxValue } from "/src/comm/core/rxValue.ts";
  4. import { useCtx } from "/src/comm/ctx/index.ts?t=1702975066869";
  5. import { message } from "/node_modules/.vite/deps/ant-design-vue.js?v=f7d2fa05";
  6. class App {
  7. name = "";
  8. version = "0.0.0";
  9. lastUpdateDate = "";
  10. icon = "";
  11. revAssets = [];
  12. size = "0MB";
  13. tags = "";
  14. desc = "";
  15. entry = "";
  16. type = "";
  17. key = "";
  18. instCount = 1;
  19. Version = "";
  20. ExePath = "";
  21. }
  22. export class InstallController extends Controller {
  23. state = RxValue.create({
  24. apps: [],
  25. installing: false,
  26. downProgress: "",
  27. installProgress: ""
  28. });
  29. async onReady() {
  30. return await this.parseInstalledApp();
  31. }
  32. async parseInstalledApp() {
  33. const { deviceCtrl, prjCtrl } = useCtx();
  34. const records = [];
  35. try {
  36. const appconfig = await deviceCtrl.ReadFileText(
  37. `${prjCtrl.RootDir}/app.json`
  38. );
  39. if (!appconfig.error && appconfig.text) {
  40. const app = JSON.parse(appconfig.text);
  41. records.push(app);
  42. }
  43. } catch (error) {
  44. console.error(error);
  45. }
  46. this.state.setApps(records);
  47. }
  48. async install(item) {
  49. console.log("install=>", item);
  50. const { deviceCtrl, prjCtrl } = useCtx();
  51. const localpath = prjCtrl.createPath(`${item._id}.zip`);
  52. this.state.installing = true;
  53. const ok = await deviceCtrl.DownloadFile(
  54. item.url,
  55. localpath,
  56. 60 * 10,
  57. (evt, msg) => {
  58. console.log("down", evt, msg.fva11);
  59. if (evt == "progress") {
  60. this.state.downProgress = ((msg.fva11 || 0) * 100).toFixed(1);
  61. }
  62. if (evt == "succ") {
  63. this.state.downProgress = "";
  64. message.success("下载成功,开始安装...");
  65. }
  66. if (evt == "error") {
  67. this.state.installing = false;
  68. deviceCtrl.TrayNotice({
  69. context: `软件下载失败,请检查你的网络设置`
  70. });
  71. }
  72. }
  73. );
  74. if (ok) {
  75. const unzipOk = await deviceCtrl.Unzip(
  76. localpath,
  77. prjCtrl.RootDir,
  78. (evt, msg) => {
  79. console.log("install", evt, msg.fva11);
  80. if (evt == "progress") {
  81. this.state.installProgress = ((msg.fva11 || 0) * 10).toFixed(1);
  82. }
  83. if (evt == "succ") {
  84. this.state.installProgress = "";
  85. this.state.installing = false;
  86. deviceCtrl.TrayNotice({
  87. context: `软件安装成功`
  88. });
  89. }
  90. if (evt == "error") {
  91. this.state.installing = false;
  92. this.state.installProgress = "";
  93. deviceCtrl.TrayNotice({
  94. context: `软件安装失败,请重新安装`
  95. });
  96. }
  97. }
  98. );
  99. if (unzipOk) {
  100. await deviceCtrl.UpdateConfig(item.version);
  101. await this.parseInstalledApp();
  102. }
  103. }
  104. queenApi.hideLoading();
  105. }
  106. async update(appRecord) {
  107. const { deviceCtrl, prjCtrl } = useCtx();
  108. const runnings = await deviceCtrl.GetRunningApp();
  109. let appRunning = runnings.find((item) => item.Key == appRecord.name);
  110. console.log("running", runnings);
  111. if (appRunning) {
  112. queenApi.messageWarn("请先关闭应用程序,再进行更新!");
  113. return;
  114. }
  115. const fpath = prjCtrl.getAppInstallDir() + `/${appRecord.name}`;
  116. const ok = await deviceCtrl.RemoveDir(fpath);
  117. if (ok) {
  118. this.install(appRecord);
  119. }
  120. }
  121. async uninstall(appRecord) {
  122. const { deviceCtrl, prjCtrl } = useCtx();
  123. const runnings = await deviceCtrl.GetRunningApp();
  124. let appRunning = runnings.find((item) => item.Key == appRecord.key);
  125. if (appRunning) {
  126. queenApi.messageWarn("请先关闭应用程序,再进行卸载!");
  127. return;
  128. }
  129. const fpath = prjCtrl.getAppInstallDir() + `/${appRecord.key}`;
  130. const ok = await deviceCtrl.RemoveDir(fpath);
  131. if (ok) {
  132. deviceCtrl.DeleteShortcut(appRecord.key);
  133. deviceCtrl.TrayNotice({ context: `${appRecord.name}卸载成功!` });
  134. await this.parseInstalledApp();
  135. }
  136. }
  137. async CreateShortcut(appRecord) {
  138. const { deviceCtrl, prjCtrl } = useCtx();
  139. const iconPath = prjCtrl.getAppInstallDir() + `/${appRecord.key}/${appRecord.icon}`;
  140. const ok = await deviceCtrl.CreateShortcut(
  141. appRecord.key,
  142. deviceCtrl.profile.exePath,
  143. iconPath
  144. );
  145. if (ok) {
  146. deviceCtrl.TrayNotice({ context: `${appRecord.name}快捷方式创建成功!` });
  147. }
  148. }
  149. }
  150. //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["installCtrl.ts"],"sourcesContent":["import { queenApi } from \"queenjs\";\r\nimport { Controller } from \"../core/controller\";\r\nimport { RxValue } from \"../core/rxValue\";\r\nimport { useCtx } from \"../ctx\";\r\nimport { message } from \"ant-design-vue\";\r\n\r\n/**\r\n *  app安装管理\r\n *  1. 读取安装的apps.manifest文件解析当前已安装的应用\r\n *  2. 所有app安装在目录${AppInstallDir}/下面, 安装后被 apps.manifest配置文件索引\r\n *  3. 安装的app升级后，更新apps.manifest索引文件，删除旧的${AppInstallDir}/下的文件\r\n *  4. 一个app在 apps.manifest索引文件中格式大致如下\r\n *  5. 安装在目录${AppInstallDir}/下面都对应有个一个app.profile文件\r\n *\r\n *    {\r\n *       name: \"spu3d\",\r\n *       version: \"1.0.0\",\r\n *       lastUpdateDate: \"2023-10-10\",\r\n *       icon: \"${AppInstallDir}/spu3d/icon.png\",\r\n *       revAssets: \".pack,.mat, matgroup, spu3d image\",\r\n *       size: \"123MB\",\r\n *       tags: \"xx\",\r\n *       desc: \"\",\r\n *       entry: \"${AppInstallDir}/spu3d/spu3d.exe\"\r\n *       type: \"nativeapp\" //nativeapp  webapp\r\n *       key: \"spu3d\" //应用id\r\n *       instCount: 1, //0 不限制实列,  1-单实例  m-m个实列\r\n *    }\r\n */\r\n\r\nclass App {\r\n  name = \"\";\r\n  version = \"0.0.0\";\r\n  lastUpdateDate = \"\";\r\n  icon = \"\";\r\n  revAssets = [];\r\n  size = \"0MB\";\r\n  tags = \"\";\r\n  desc = \"\";\r\n  entry = \"\";\r\n  type = \"\";\r\n  key = \"\";\r\n  instCount = 1;\r\n  Version = \"\";\r\n  ExePath = \"\";\r\n}\r\n\r\nexport class InstallController extends Controller {\r\n  state = RxValue.create({\r\n    apps: [] as App[],\r\n    installing: false,\r\n    downProgress: \"\",\r\n    installProgress: \"\",\r\n  });\r\n\r\n  async onReady() {\r\n    return await this.parseInstalledApp();\r\n  }\r\n\r\n  async parseInstalledApp() {\r\n    const { deviceCtrl, prjCtrl } = useCtx();\r\n\r\n    // const installdir = prjCtrl.getAppInstallDir();\r\n\r\n    // const ret = await deviceCtrl.GetSubDirNames(installdir);\r\n\r\n    // console.log(\"list dirs=>\", ret);\r\n\r\n    //解析已安装的目录\r\n    const records: App[] = [];\r\n\r\n    try {\r\n      const appconfig = await deviceCtrl.ReadFileText(\r\n        `${prjCtrl.RootDir}/app.json`\r\n      );\r\n\r\n      if (!appconfig.error && appconfig.text) {\r\n        const app = JSON.parse(appconfig.text);\r\n        records.push(app);\r\n      }\r\n    } catch (error) {\r\n      console.error(error);\r\n    }\r\n    this.state.setApps(records);\r\n  }\r\n\r\n  async install(item: any) {\r\n    console.log(\"install=>\", item);\r\n    const { deviceCtrl, prjCtrl } = useCtx();\r\n    const localpath = prjCtrl.createPath(`${item._id}.zip`);\r\n    this.state.installing = true;\r\n    const ok = await deviceCtrl.DownloadFile(\r\n      item.url,\r\n      localpath,\r\n      60 * 10,\r\n      (evt, msg) => {\r\n        console.log(\"down\", evt, msg.fva11);\r\n\r\n        if (evt == \"progress\") {\r\n          this.state.downProgress = ((msg.fva11 || 0) * 100).toFixed(1);\r\n        }\r\n        if (evt == \"succ\") {\r\n          this.state.downProgress = \"\";\r\n          message.success(\"下载成功，开始安装...\");\r\n        }\r\n        if (evt == \"error\") {\r\n          this.state.installing = false;\r\n          deviceCtrl.TrayNotice({          \r\n            context: `软件下载失败，请检查你的网络设置`,\r\n          });\r\n        }\r\n      }\r\n    );\r\n    if (ok) {\r\n      const unzipOk = await deviceCtrl.Unzip(\r\n        localpath,\r\n        prjCtrl.RootDir,\r\n        (evt, msg) => {\r\n          console.log(\"install\", evt, msg.fva11);\r\n          if (evt == \"progress\") {\r\n            this.state.installProgress = ((msg.fva11 || 0) * 10).toFixed(1);\r\n          }\r\n          if (evt == \"succ\") {\r\n            this.state.installProgress = \"\";\r\n            this.state.installing = false;\r\n            deviceCtrl.TrayNotice({             \r\n              context: `软件安装成功`,\r\n            });\r\n          }\r\n          if (evt == \"error\") {\r\n            this.state.installing = false;\r\n            this.state.installProgress = \"\";\r\n            deviceCtrl.TrayNotice({            \r\n              context: `软件安装失败，请重新安装`,\r\n            });\r\n          }\r\n        }\r\n      );\r\n      if (unzipOk) {\r\n        await deviceCtrl.UpdateConfig(item.version);\r\n        await this.parseInstalledApp();\r\n      }\r\n    }\r\n    queenApi.hideLoading();\r\n  }\r\n  async update(appRecord: any) {\r\n    const {  deviceCtrl, prjCtrl } = useCtx();\r\n    //提示停止当前应用\r\n\r\n    const runnings = await deviceCtrl.GetRunningApp();\r\n    let appRunning = runnings.find((item) => item.Key == appRecord.name);\r\n    console.log(\"running\", runnings);\r\n    if (appRunning) {\r\n      queenApi.messageWarn(\"请先关闭应用程序，再进行更新！\");\r\n      return;\r\n    }\r\n    const fpath = prjCtrl.getAppInstallDir() + `/${appRecord.name}`;\r\n    const ok = await deviceCtrl.RemoveDir(fpath);\r\n    if (ok) {\r\n      this.install(appRecord);\r\n    }\r\n  }\r\n  async uninstall(appRecord: any) {\r\n    const {  deviceCtrl, prjCtrl } = useCtx();\r\n    //提示停止当前应用\r\n    const runnings = await deviceCtrl.GetRunningApp();\r\n    let appRunning = runnings.find((item) => item.Key == appRecord.key);\r\n    if (appRunning) {\r\n      queenApi.messageWarn(\"请先关闭应用程序，再进行卸载！\");\r\n      return;\r\n    }\r\n    //删除安装文件\r\n    const fpath = prjCtrl.getAppInstallDir() + `/${appRecord.key}`;\r\n    const ok = await deviceCtrl.RemoveDir(fpath);\r\n    if (ok) {\r\n      deviceCtrl.DeleteShortcut(appRecord.key);\r\n      deviceCtrl.TrayNotice({ context: `${appRecord.name}卸载成功!` });\r\n      //重新解析已安装文件\r\n      await this.parseInstalledApp();\r\n    }\r\n  }\r\n  async CreateShortcut(appRecord: any) {\r\n    const { deviceCtrl, prjCtrl } = useCtx();\r\n\r\n    const iconPath =\r\n      prjCtrl.getAppInstallDir() + `/${appRecord.key}/${appRecord.icon}`;\r\n    const ok = await deviceCtrl.CreateShortcut(\r\n      appRecord.key,\r\n      deviceCtrl.profile.exePath,\r\n      iconPath\r\n    );\r\n    if (ok) {\r\n      deviceCtrl.TrayNotice({ context: `${appRecord.name}快捷方式创建成功!` });\r\n    }\r\n  }\r\n}\r\n"],"mappings":"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","names":[]}