選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

628 行
44KB

  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <title>WPS加载项配置</title>
  6. <link rel="icon" type="image/png" sizes="32x32"
  7. href="">
  8. <style>
  9. body {
  10. margin: 30px;
  11. }
  12. .addonList {
  13. max-width: 80%;
  14. flex-direction: column;
  15. padding: 18px;
  16. border-radius: 4px;
  17. border: 1px solid silver;
  18. }
  19. .addonItem {
  20. font-size: 16px;
  21. line-height: 36px;
  22. margin-bottom: 4px;
  23. }
  24. .addonItem:hover {
  25. border-radius: 2px;
  26. border: 1px dashed silver;
  27. }
  28. .addonItemName1 {
  29. display: inline-block;
  30. width: 15%;
  31. text-align: left;
  32. vertical-align: middle;
  33. word-wrap: break-word;
  34. }
  35. .addonItemName2 {
  36. display: inline-block;
  37. width: 10%;
  38. vertical-align: middle;
  39. text-align: left;
  40. word-wrap: break-word;
  41. }
  42. .addonItemName3 {
  43. display: inline-block;
  44. width: 10%;
  45. vertical-align: middle;
  46. text-align: left;
  47. word-wrap: break-word;
  48. }
  49. .addonItemName4 {
  50. display: inline-block;
  51. width: 35%;
  52. text-align: left;
  53. vertical-align: middle;
  54. word-wrap: break-word;
  55. }
  56. .addonItemName5 {
  57. display: inline-block;
  58. width: 15%;
  59. text-align: left;
  60. vertical-align: middle;
  61. word-wrap: break-word;
  62. }
  63. .addonItemName6 {
  64. display: inline-block;
  65. width: 10%;
  66. text-align: left;
  67. vertical-align: middle;
  68. word-wrap: break-word;
  69. }
  70. .addonItemButton {
  71. padding: 4px 8px;
  72. background-color: #417ff9;
  73. display: inline-block;
  74. cursor: pointer;
  75. box-sizing: border-box;
  76. border-radius: 4px;
  77. text-align: center;
  78. color: #fff;
  79. }
  80. .addonItemButton:hover {
  81. background-color: #5696ff;
  82. }
  83. .addonItemTitle {
  84. padding: 0px;
  85. border-bottom: 1px solid silver;
  86. }
  87. .addonItemTitle:hover {
  88. border-radius: 0px;
  89. border: 0px;
  90. }
  91. .ClearAll {
  92. max-width: 80%;
  93. margin-top: 20px;
  94. font-size: 16px;
  95. line-height: 36px;
  96. text-align: center;
  97. cursor: pointer;
  98. border: 1px dashed silver;
  99. padding: 0px 18px;
  100. }
  101. .ClearAll:hover {
  102. border-radius: 2px;
  103. background-color: silver;
  104. }
  105. .divTitle {
  106. font-size: 30px;
  107. font-weight: bolder;
  108. margin-bottom: 20px;
  109. }
  110. </style>
  111. <script>
  112. function getHttpObj() {
  113. var httpobj = null;
  114. if (IEVersion() < 10) {
  115. try {
  116. httpobj = new XDomainRequest();
  117. } catch (e1) {
  118. httpobj = new createXHR();
  119. }
  120. } else {
  121. httpobj = new createXHR();
  122. }
  123. return httpobj;
  124. }
  125. //兼容IE低版本的创建xmlhttprequest对象的方法
  126. function createXHR() {
  127. if (typeof XMLHttpRequest != 'undefined') { //兼容高版本浏览器
  128. return new XMLHttpRequest();
  129. } else if (typeof ActiveXObject != 'undefined') { //IE6 采用 ActiveXObject, 兼容IE6
  130. var versions = [ //由于MSXML库有3个版本,因此都要考虑
  131. 'MSXML2.XMLHttp.6.0',
  132. 'MSXML2.XMLHttp.3.0',
  133. 'MSXML2.XMLHttp'
  134. ];
  135. for (var i = 0; i < versions.length; i++) {
  136. try {
  137. return new ActiveXObject(versions[i]);
  138. } catch (e) {
  139. //跳过
  140. }
  141. }
  142. } else {
  143. throw new Error('您的浏览器不支持XHR对象');
  144. }
  145. }
  146. var fromCharCode = String.fromCharCode;
  147. // encoder stuff
  148. var cb_utob = function (c) {
  149. if (c.length < 2) {
  150. var cc = c.charCodeAt(0);
  151. return cc < 0x80 ? c :
  152. cc < 0x800 ? (fromCharCode(0xc0 | (cc >>> 6)) +
  153. fromCharCode(0x80 | (cc & 0x3f))) :
  154. (fromCharCode(0xe0 | ((cc >>> 12) & 0x0f)) +
  155. fromCharCode(0x80 | ((cc >>> 6) & 0x3f)) +
  156. fromCharCode(0x80 | (cc & 0x3f)));
  157. } else {
  158. var cc = 0x10000 +
  159. (c.charCodeAt(0) - 0xD800) * 0x400 +
  160. (c.charCodeAt(1) - 0xDC00);
  161. return (fromCharCode(0xf0 | ((cc >>> 18) & 0x07)) +
  162. fromCharCode(0x80 | ((cc >>> 12) & 0x3f)) +
  163. fromCharCode(0x80 | ((cc >>> 6) & 0x3f)) +
  164. fromCharCode(0x80 | (cc & 0x3f)));
  165. }
  166. };
  167. var re_utob = /[\uD800-\uDBFF][\uDC00-\uDFFFF]|[^\x00-\x7F]/g;
  168. var utob = function (u) {
  169. return u.replace(re_utob, cb_utob);
  170. };
  171. var _encode = function (u) {
  172. var isUint8Array = Object.prototype.toString.call(u) === '[object Uint8Array]';
  173. if (isUint8Array)
  174. return u.toString('base64')
  175. else
  176. return btoa(utob(String(u)));
  177. }
  178. if (typeof btoa !== 'function') btoa = func_btoa;
  179. function func_btoa(input) {
  180. var str = String(input);
  181. var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
  182. for (
  183. // initialize result and counter
  184. var block, charCode, idx = 0, map = chars, output = '';
  185. // if the next str index does not exist:
  186. // change the mapping table to "="
  187. // check if d has no fractional digits
  188. str.charAt(idx | 0) || (map = '=', idx % 1);
  189. // "8 - idx % 1 * 8" generates the sequence 2, 4, 6, 8
  190. output += map.charAt(63 & block >> 8 - idx % 1 * 8)
  191. ) {
  192. charCode = str.charCodeAt(idx += 3 / 4);
  193. if (charCode > 0xFF) {
  194. throw new InvalidCharacterError("'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.");
  195. }
  196. block = block << 8 | charCode;
  197. }
  198. return output;
  199. }
  200. function encode(u, urisafe) {
  201. return !urisafe ?
  202. _encode(u) :
  203. _encode(String(u)).replace(/[+\/]/g, function (m0) {
  204. return m0 == '+' ? '-' : '_';
  205. }).replace(/=/g, '');
  206. }
  207. function IEVersion() {
  208. var userAgent = navigator.userAgent; //取得浏览器的userAgent字符串
  209. var isIE = userAgent.indexOf("compatible") > -1 && userAgent.indexOf("MSIE") > -1; //判断是否IE<11浏览器
  210. var isEdge = userAgent.indexOf("Edge") > -1 && !isIE; //判断是否IE的Edge浏览器
  211. var isIE11 = userAgent.indexOf('Trident') > -1 && userAgent.indexOf("rv:11.0") > -1;
  212. if (isIE) {
  213. var reIE = new RegExp("MSIE (\\d+\\.\\d+);");
  214. reIE.test(userAgent);
  215. var fIEVersion = parseFloat(RegExp["$1"]);
  216. if (fIEVersion == 7) {
  217. return 7;
  218. } else if (fIEVersion == 8) {
  219. return 8;
  220. } else if (fIEVersion == 9) {
  221. return 9;
  222. } else if (fIEVersion == 10) {
  223. return 10;
  224. } else {
  225. return 6; //IE版本<=7
  226. }
  227. } else if (isEdge) {
  228. return 20; //edge
  229. } else if (isIE11) {
  230. return 11; //IE11
  231. } else {
  232. return 30; //不是ie浏览器
  233. }
  234. }
  235. /**
  236. * 生成guid的接口
  237. * @returns guid
  238. */
  239. function guid() {
  240. return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
  241. var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
  242. return v.toString(16);
  243. });
  244. }
  245. /**
  246. * 自定义协议启动服务端
  247. * 默认不带参数serverId,linux未升级之前不要使用多用户
  248. */
  249. function InitWpsCloudSvr() {
  250. if (serverId == undefined)
  251. window.location.href = "ksoWPSCloudSvr://start=RelayHttpServer"//是否启动wps弹框
  252. else
  253. window.location.href = "ksoWPSCloudSvr://start=RelayHttpServer" + "&serverId=" + serverId //是否启动wps弹框
  254. }
  255. var serverId = undefined;
  256. /**
  257. * 获取serverId的接口
  258. * @returns serverId
  259. */
  260. function getServerId() {
  261. if (window.localStorage) {
  262. if (localStorage.getItem("serverId")) {
  263. //
  264. }
  265. else {
  266. localStorage.setItem("serverId", guid());
  267. }
  268. return localStorage.getItem("serverId");
  269. }
  270. else {
  271. return guid();
  272. }
  273. }
  274. function startWps(req, t, callback) {
  275. function startWpsInnder(reqInner, tryCount, bPop) {
  276. if (tryCount < 1) {
  277. if (callback)
  278. callback({
  279. status: 2,
  280. message: "请允许浏览器打开WPS Office"
  281. });
  282. return;
  283. }
  284. var bRetry = true;
  285. var xmlReq = getHttpObj();
  286. //WPS客户端提供的接收参数的本地服务,HTTP服务端口为58890,HTTPS服务端口为58891
  287. //这俩配置,取一即可,不可同时启用
  288. xmlReq.open(reqInner.type, reqInner.url);
  289. xmlReq.onload = function (res) {
  290. if (res.target.status != 200) {
  291. var responseStr = IEVersion() < 10 ? xmlReq.responseText : res.target.response;
  292. var errorMessage = JSON.parse(responseStr)
  293. if (errorMessage.data == "Subserver not available." && tryCount == 4 && bPop) {
  294. InitWpsCloudSvr();
  295. setTimeout(function () {
  296. if (bRetry) {
  297. bRetry = false;
  298. startWpsInnder(reqInner, --tryCount, false);
  299. }
  300. }, 3000);
  301. }
  302. }
  303. if (callback)
  304. callback({
  305. status: 0,
  306. res: res
  307. });
  308. }
  309. xmlReq.ontimeout = xmlReq.onerror = function (res) {
  310. xmlReq.bTimeout = true;
  311. if (bPop) { //打开wps并传参
  312. InitWpsCloudSvr()
  313. }
  314. setTimeout(function () {
  315. if (bRetry) {
  316. bRetry = false;
  317. startWpsInnder(reqInner, --tryCount, false);
  318. }
  319. }, 1000);
  320. }
  321. if (IEVersion() < 10) {
  322. xmlReq.onreadystatechange = function () {
  323. if (xmlReq.readyState != 4)
  324. return;
  325. if (xmlReq.bTimeout) {
  326. return;
  327. }
  328. if (xmlReq.status === 200)
  329. xmlReq.onload();
  330. else
  331. xmlReq.onerror();
  332. }
  333. }
  334. xmlReq.timeout = 3000;
  335. xmlReq.send(t)
  336. }
  337. startWpsInnder(req, 4, true);
  338. return;
  339. }
  340. function CheckPlugin(element) {
  341. var id = GetAddonId(element);
  342. var ele = document.getElementById(id + "_status");
  343. var xmlReq = getHttpObj();
  344. var offline = element.online === "false";
  345. var url = offline ? element.url : element.url + "ribbon.xml";
  346. xmlReq.open("POST", "http://localhost:58890/redirect/runParams");
  347. xmlReq.onload = function (res) {
  348. if ((offline && res.target.response.startsWith("7z"))
  349. || !offline && res.target.response.startsWith("<customUI")) {
  350. ele.style.color = "green";
  351. ele.style.textAlign = "center";
  352. ele.innerHTML = "正常";
  353. } else {
  354. ele.style.color = "white";
  355. ele.style.backgroundColor = "gray";
  356. ele.style.textAlign = "center";
  357. ele.innerHTML = "无效";
  358. ele.title = offline ? ("不是有效的7z格式" + url) : ("不是有效的ribbon.xml," + url);
  359. }
  360. }
  361. xmlReq.onerror = function (res) {
  362. xmlReq.bTimeout = true;
  363. ele.style.color = "white";
  364. ele.style.backgroundColor = "gray";
  365. ele.style.textAlign = "center";
  366. ele.innerHTML = "无效";
  367. ele.title = "网页路径不可访问,如果是跨域问题,不影响使用:" + url;
  368. }
  369. xmlReq.ontimeout = function (res) {
  370. xmlReq.bTimeout = true;
  371. ele.style.color = "white";
  372. ele.style.backgroundColor = "gray";
  373. ele.style.textAlign = "center";
  374. ele.innerHTML = "异常";
  375. ele.title = "访问超时," + url;
  376. }
  377. if (IEVersion() < 10) {
  378. xmlReq.onreadystatechange = function () {
  379. if (xmlReq.readyState != 4)
  380. return;
  381. if (xmlReq.bTimeout) {
  382. return;
  383. }
  384. if (xmlReq.status === 200)
  385. xmlReq.onload();
  386. else
  387. xmlReq.onerror();
  388. }
  389. }
  390. xmlReq.timeout = 5000;
  391. var data = {
  392. method: "get",
  393. url: url,
  394. data: ""
  395. }
  396. var sendData = FormatSendData(data)
  397. xmlReq.send(sendData);
  398. }
  399. function GetAddonId(element) {
  400. return element.name + "/" + element.addonType;
  401. }
  402. function UpdateElement(element, cmd) {
  403. if (typeof element.name === 'undefined')
  404. return
  405. var id = GetAddonId(element);
  406. var addonList = document.getElementById("addonList");
  407. //var param = JSON.stringify(element).replace(/"/g, "\'");
  408. var buttonLabel = cmd === 'enable' ? "安装" : "卸载";
  409. var des = "文字";
  410. if (element.addonType == "et")
  411. des = "电子表格";
  412. else if (element.addonType == "wpp")
  413. des = "演示";
  414. var loadType = "在线";
  415. if (element.online == "false")
  416. loadType = "离线";
  417. var old = document.getElementById(id);
  418. if (old !== null) {
  419. var oldOnline = old.wpsaddon.online === "false";
  420. var newOnline = element.online === "false";
  421. if (cmd === 'disable'
  422. && (oldOnline !== newOnline
  423. || old.wpsaddon.url !== element.url
  424. || (oldOnline && old.wpsaddon.version !== element.version))) {
  425. buttonLabel = "更新/卸载";
  426. cmd = "choose";
  427. }
  428. old.wpsaddoncmd = cmd;
  429. document.getElementById(id + '_button').innerHTML = buttonLabel;
  430. } else {
  431. var ele = document.createElement("div");
  432. ele.className = "addonItem";
  433. ele.id = id;
  434. ele.wpsaddon = element;
  435. ele.wpsaddoncmd = cmd;
  436. ele.innerHTML =
  437. '<div class="addonItemName1">' + element.name + '</div>\n' +
  438. '<div class="addonItemName2">' + des + '</div>\n' +
  439. '<div class="addonItemName3">' + loadType + '</div>\n' +
  440. '<div class="addonItemName4">' + element.url + '</div>\n' +
  441. '<div class="addonItemName5"><div class="addonItemButton" id="' + id + '_button' + '" onclick="WpsAddonHandle(\'' + id + '\')">' + buttonLabel + '</div></div>\n' +
  442. '<div class="addonItemName6" id="' + id + '_status' + '">验证中...</div>\n';
  443. addonList.appendChild(ele);
  444. CheckPlugin(element);
  445. }
  446. }
  447. function WpsAddonHandle(id) {
  448. var ele = document.getElementById(id);
  449. var element = ele.wpsaddon;
  450. var cmd = ele.wpsaddoncmd;
  451. WpsAddonHandleEx(element, cmd)
  452. }
  453. function WpsAddonHandleEx(element, cmd) {
  454. if (cmd === "choose") {
  455. if (confirm("点击确定将更新 WPS 加载项,或点击取消完成卸载")) {
  456. cmd = "enable";
  457. } else {
  458. cmd = "disable";
  459. }
  460. }
  461. var data = FormartData(element, cmd);
  462. var req = { url: "http://localhost:58890/deployaddons/runParams", type: "POST" };
  463. startWps(req, data, function (res) {
  464. if (res.status == 0) {
  465. if (cmd === "disableall") {
  466. window.location.reload();
  467. } else {
  468. if (res.res.target.response == "OK"
  469. || (res.res.target.response == "" && res.res.target.status == 200)) {
  470. var newCmd = 'disable';
  471. if (cmd === 'disable')
  472. newCmd = 'enable';
  473. UpdateElement(element, newCmd)
  474. alert("配置成功!");
  475. }
  476. else {
  477. alert("配置失败!");
  478. }
  479. }
  480. } else {
  481. alert(res.message);
  482. }
  483. });
  484. }
  485. function FormartData(element, cmd) {
  486. var data = {
  487. "cmd": cmd, //"enable", 启用, "disable", 禁用, "disableall", 禁用所有
  488. "name": element.name,
  489. "url": element.url,
  490. "addonType": element.addonType,
  491. "online": element.online,
  492. "version": element.version
  493. }
  494. return FormatSendData(data);
  495. }
  496. function FormatSendData(data) {
  497. var strData = JSON.stringify(data);
  498. if (IEVersion() < 10)
  499. eval("strData = '" + JSON.stringify(strData) + "';");
  500. if (serverVersion >= "1.0.2" && serverId != undefined) {
  501. var base64Data = encode(strData);
  502. return JSON.stringify({
  503. serverId: serverId,
  504. data: base64Data
  505. })
  506. }
  507. else {
  508. return encode(strData);
  509. }
  510. }
  511. function LoadLocalAddons() {
  512. var baseData
  513. if (serverVersion >= "1.0.2" && serverId != undefined)
  514. baseData = JSON.stringify({ serverId: serverId });
  515. var req = { url: "http://127.0.0.1:58890/publishlist", type: "POST" };
  516. startWps(req, baseData, function (res) {
  517. if (res.status == 0) {
  518. var addonList = document.getElementById("addonList");
  519. var curList = JSON.parse(res.res.target.response);
  520. curList.forEach(function (element) {
  521. if (element.enable === "false")
  522. return;
  523. UpdateElement(element, 'disable')
  524. });
  525. } else {
  526. alert(res.message);
  527. }
  528. });
  529. }
  530. function LoadPublishAddons() {
  531. var addonList = document.getElementById("addonList");
  532. var curList = [{"name":"WpsOAAssist","addonType":"wps","online":"true","multiUser":"false","url":"http://webtest.ningdatech.com/oaassist/wps/"}];
  533. curList.forEach(function (element) {
  534. var param = JSON.stringify(element).replace("\"", "\'");
  535. UpdateElement(element, 'enable')
  536. });
  537. }
  538. var serverVersion = "wait";
  539. function InitSdk() {
  540. var req = { url: "http://127.0.0.1:58890/version", type: "POST" };
  541. startWps(
  542. req,
  543. JSON.stringify({ serverId: serverId }),
  544. function (res) {
  545. if (res.status !== 0) {
  546. return;
  547. }
  548. if (serverVersion == "wait") {
  549. serverVersion = res.res.target.response;
  550. LoadPublishAddons();
  551. LoadLocalAddons();
  552. }
  553. },
  554. );
  555. }
  556. function LoadAddons() {
  557. var addonList = document.getElementById("addonList");
  558. addonList.style.maxWidth = 800 * window.devicePixelRatio + "px";
  559. var ClearAll = document.getElementById("ClearAll");
  560. ClearAll.style.maxWidth = 800 * window.devicePixelRatio + "px";
  561. InitSdk();
  562. }
  563. function ClearAll() {
  564. if (confirm('确定要禁用所有WPS加载项吗?')) {
  565. var element = {};
  566. WpsAddonHandleEx(element, 'disableall');
  567. }
  568. }
  569. </script>
  570. </head>
  571. <body onload="LoadAddons()">
  572. <div class="divTitle">WPS加载项配置</div>
  573. <div class="addonList" id="addonList">
  574. <div class="addonItem addonItemTitle">
  575. <div class="addonItemName1">加载项名称</div>
  576. <div class="addonItemName2">类型</div>
  577. <div class="addonItemName3">加载方式</div>
  578. <div class="addonItemName4">URL</div>
  579. <div class="addonItemName5">管理</div>
  580. <div class="addonItemName6">状态</div>
  581. </div>
  582. </div>
  583. <div class="ClearAll" onclick="ClearAll()" id="ClearAll">禁用所有 WPS 加载项</div>
  584. </body>
  585. </html>