format-lib.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747
  1. /**
  2. * 日期格式化
  3. * @param {Object} pattern
  4. */
  5. Date.prototype.format = function (pattern) {
  6. let pad = function (source, length) {
  7. let pre = "",
  8. negative = (source < 0),
  9. string = String(Math.abs(source));
  10. if (string.length < length) {
  11. pre = (new Array(length - string.length + 1)).join('0');
  12. }
  13. return (negative ? "-" : "") + pre + string;
  14. };
  15. if ('string' !== typeof pattern) {
  16. return this.toString();
  17. }
  18. let replacer = function (patternPart, result) {
  19. pattern = pattern.replace(patternPart, result);
  20. };
  21. let year = this.getFullYear(),
  22. month = this.getMonth() + 1,
  23. date2 = this.getDate(),
  24. hours = this.getHours(),
  25. minutes = this.getMinutes(),
  26. seconds = this.getSeconds(),
  27. milliSec = this.getMilliseconds();
  28. replacer(/yyyy/g, pad(year, 4));
  29. replacer(/yy/g, pad(parseInt(year.toString().slice(2), 10), 2));
  30. replacer(/MM/g, pad(month, 2));
  31. replacer(/M/g, month);
  32. replacer(/dd/g, pad(date2, 2));
  33. replacer(/d/g, date2);
  34. replacer(/HH/g, pad(hours, 2));
  35. replacer(/H/g, hours);
  36. replacer(/hh/g, pad(hours % 12, 2));
  37. replacer(/h/g, hours % 12);
  38. replacer(/mm/g, pad(minutes, 2));
  39. replacer(/m/g, minutes);
  40. replacer(/ss/g, pad(seconds, 2));
  41. replacer(/s/g, seconds);
  42. replacer(/SSS/g, pad(milliSec, 3));
  43. replacer(/S/g, milliSec);
  44. return pattern;
  45. };
  46. /**
  47. * 自动消失的Alert弹窗
  48. * @param content
  49. */
  50. window.toast = function (content) {
  51. window.clearTimeout(window.feHelperAlertMsgTid);
  52. let elAlertMsg = document.querySelector("#fehelper_alertmsg");
  53. if (!elAlertMsg) {
  54. let elWrapper = document.createElement('div');
  55. elWrapper.innerHTML = '<div id="fehelper_alertmsg" style="position:fixed;bottom:25px;left:5px;z-index:1000000">' +
  56. '<p style="background:#000;display:inline-block;color:#fff;text-align:center;' +
  57. 'padding:10px 10px;margin:0 auto;font-size:14px;border-radius:4px;">' + content + '</p></div>';
  58. elAlertMsg = elWrapper.childNodes[0];
  59. document.body.appendChild(elAlertMsg);
  60. } else {
  61. elAlertMsg.querySelector('p').innerHTML = content;
  62. elAlertMsg.style.display = 'block';
  63. }
  64. window.feHelperAlertMsgTid = window.setTimeout(function () {
  65. elAlertMsg.style.display = 'none';
  66. }, 1000);
  67. };
  68. /**
  69. * FeHelper Json Format Lib,入口文件
  70. * @example
  71. * Formatter.format(jsonString)
  72. */
  73. window.Formatter = (function () {
  74. "use strict";
  75. let jfContent,
  76. jfPre,
  77. jfStyleEl,
  78. jfStatusBar,
  79. formattingMsg;
  80. let lastItemIdGiven = 0;
  81. let cachedJsonString = '';
  82. // 单例Worker实例
  83. let workerInstance = null;
  84. let _initElements = function () {
  85. jfContent = $('#jfContent');
  86. if (!jfContent[0]) {
  87. jfContent = $('<div id="jfContent" />').appendTo('body');
  88. }
  89. jfPre = $('#jfContent_pre');
  90. if (!jfPre[0]) {
  91. jfPre = $('<pre id="jfContent_pre" />').appendTo('body');
  92. }
  93. jfStyleEl = $('#jfStyleEl');
  94. if (!jfStyleEl[0]) {
  95. jfStyleEl = $('<style id="jfStyleEl" />').appendTo('head');
  96. }
  97. formattingMsg = $('#formattingMsg');
  98. if (!formattingMsg[0]) {
  99. formattingMsg = $('<div id="formattingMsg"><span class="x-loading"></span>格式化中...</div>').appendTo('body');
  100. }
  101. try {
  102. jfContent.html('').show();
  103. jfPre.html('').hide();
  104. jfStatusBar && jfStatusBar.hide();
  105. formattingMsg.hide();
  106. } catch (e) {
  107. }
  108. };
  109. /**
  110. * HTML特殊字符格式化
  111. * @param str
  112. * @returns {*}
  113. */
  114. let htmlspecialchars = function (str) {
  115. str = str.replace(/&/g, '&amp;');
  116. str = str.replace(/</g, '&lt;');
  117. str = str.replace(/>/g, '&gt;');
  118. str = str.replace(/"/g, '&quot;');
  119. str = str.replace(/'/g, '&#039;');
  120. str = str.replace(/\\/g, '&#92;');
  121. return str;
  122. };
  123. /**
  124. * 直接下载,能解决中文乱码
  125. * @param content
  126. * @private
  127. */
  128. let _downloadSupport = function (content) {
  129. // 下载链接
  130. let dt = (new Date()).format('yyyyMMddHHmmss');
  131. let blob = new Blob([content], {type: 'application/octet-stream'});
  132. let button = $('<button class="xjf-btn xjf-btn-right">下载JSON</button>').appendTo('#optionBar');
  133. if (typeof chrome === 'undefined' || !chrome.permissions) {
  134. button.click(function (e) {
  135. let aLink = $('#aLinkDownload');
  136. if (!aLink[0]) {
  137. aLink = $('<a id="aLinkDownload" target="_blank" title="保存到本地">下载JSON数据</a>').appendTo('body');
  138. aLink.attr('download', 'FeHelper-' + dt + '.json');
  139. aLink.attr('href', URL.createObjectURL(blob));
  140. }
  141. aLink[0].click();
  142. });
  143. } else {
  144. button.click(function (e) {
  145. // 请求权限
  146. chrome.permissions.request({
  147. permissions: ['downloads']
  148. }, (granted) => {
  149. if (granted) {
  150. chrome.downloads.download({
  151. url: URL.createObjectURL(blob),
  152. saveAs: true,
  153. conflictAction: 'overwrite',
  154. filename: 'FeHelper-' + dt + '.json'
  155. });
  156. } else {
  157. toast('必须接受授权,才能正常下载!');
  158. }
  159. });
  160. });
  161. }
  162. };
  163. /**
  164. * chrome 下复制到剪贴板
  165. * @param text
  166. */
  167. let _copyToClipboard = function (text) {
  168. let input = document.createElement('textarea');
  169. input.style.position = 'fixed';
  170. input.style.opacity = 0;
  171. input.value = text;
  172. document.body.appendChild(input);
  173. input.select();
  174. document.execCommand('Copy');
  175. document.body.removeChild(input);
  176. toast('Json片段复制成功,随处粘贴可用!')
  177. };
  178. /**
  179. * 从el中获取json文本
  180. * @param el
  181. * @returns {string}
  182. */
  183. let getJsonText = function (el) {
  184. let txt = el.text().replace(/复制\|下载\|删除/gm,'').replace(/":\s/gm, '":').replace(/,$/, '').trim();
  185. if (!(/^{/.test(txt) && /\}$/.test(txt)) && !(/^\[/.test(txt) && /\]$/.test(txt))) {
  186. txt = '{' + txt + '}';
  187. }
  188. try {
  189. txt = JSON.stringify(JSON.parse(txt), null, 4);
  190. } catch (err) {
  191. }
  192. return txt;
  193. };
  194. // 添加json路径
  195. let _showJsonPath = function (curEl) {
  196. let keys = [];
  197. do {
  198. if (curEl.hasClass('item-block')) {
  199. if (!curEl.hasClass('rootItem')) {
  200. keys.unshift('[' + curEl.prevAll('.item').length + ']');
  201. } else {
  202. break;
  203. }
  204. } else {
  205. keys.unshift(curEl.find('>.key').text());
  206. }
  207. if (curEl.parent().hasClass('rootItem') || curEl.parent().parent().hasClass('rootItem')) {
  208. break;
  209. }
  210. curEl = curEl.parent().parent();
  211. } while (curEl.length && !curEl.hasClass('rootItem'));
  212. // 过滤掉空值和无效的key,避免产生多余的点号
  213. let validKeys = keys.filter(key => key && key.trim() !== '');
  214. // 构建路径:正确处理对象属性和数组索引的连接
  215. let path = '';
  216. for (let i = 0; i < validKeys.length; i++) {
  217. let key = validKeys[i];
  218. if (key.startsWith('[') && key.endsWith(']')) {
  219. // 数组索引,直接拼接(前面永远不需要点号)
  220. path += key;
  221. } else {
  222. // 对象属性
  223. if (i > 0) {
  224. // 对象属性前面需要点号(数组索引后面的属性也需要点号)
  225. path += '.';
  226. }
  227. path += key;
  228. }
  229. }
  230. let jfPath = $('#jsonPath');
  231. if (!jfPath.length) {
  232. jfPath = $('<span id="jsonPath"/>').prependTo(jfStatusBar);
  233. }
  234. jfPath.html('当前节点:$.' + path);
  235. };
  236. // 给某个节点增加操作项
  237. let _addOptForItem = function (el, show) {
  238. // 下载json片段
  239. let fnDownload = function (event) {
  240. event.stopPropagation();
  241. let txt = getJsonText(el);
  242. // 下载片段
  243. let dt = (new Date()).format('yyyyMMddHHmmss');
  244. let blob = new Blob([txt], {type: 'application/octet-stream'});
  245. if (typeof chrome === 'undefined' || !chrome.permissions) {
  246. // 下载JSON的简单形式
  247. $(this).attr('download', 'FeHelper-' + dt + '.json').attr('href', URL.createObjectURL(blob));
  248. } else {
  249. // 请求权限
  250. chrome.permissions.request({
  251. permissions: ['downloads']
  252. }, (granted) => {
  253. if (granted) {
  254. chrome.downloads.download({
  255. url: URL.createObjectURL(blob),
  256. saveAs: true,
  257. conflictAction: 'overwrite',
  258. filename: 'FeHelper-' + dt + '.json'
  259. });
  260. } else {
  261. toast('必须接受授权,才能正常下载!');
  262. }
  263. });
  264. }
  265. };
  266. // 复制json片段
  267. let fnCopy = function (event) {
  268. event.stopPropagation();
  269. _copyToClipboard(getJsonText(el));
  270. };
  271. // 删除json片段
  272. let fnDel = function (event) {
  273. event.stopPropagation();
  274. if (el.parent().is('#formattedJson')) {
  275. toast('如果连最外层的Json也删掉的话,就没啥意义了哦!');
  276. return false;
  277. }
  278. toast('节点已删除成功!');
  279. el.remove();
  280. jfStatusBar && jfStatusBar.hide();
  281. };
  282. $('.boxOpt').hide();
  283. if (show) {
  284. let jfOptEl = el.children('.boxOpt');
  285. if (!jfOptEl.length) {
  286. jfOptEl = $('<b class="boxOpt">' +
  287. '<a class="opt-copy" title="复制当前选中节点的JSON数据">复制</a>|' +
  288. '<a class="opt-download" target="_blank" title="下载当前选中节点的JSON数据">下载</a>|' +
  289. '<a class="opt-del" title="删除当前选中节点的JSON数据">删除</a></b>').appendTo(el);
  290. } else {
  291. jfOptEl.show();
  292. }
  293. jfOptEl.find('a.opt-download').unbind('click').bind('click', fnDownload);
  294. jfOptEl.find('a.opt-copy').unbind('click').bind('click', fnCopy);
  295. jfOptEl.find('a.opt-del').unbind('click').bind('click', fnDel);
  296. }
  297. };
  298. // 显示当前节点的Key
  299. let _toogleStatusBar = function (curEl, show) {
  300. if (!jfStatusBar) {
  301. jfStatusBar = $('<div id="statusBar"/>').appendTo('body');
  302. }
  303. if (!show) {
  304. jfStatusBar.hide();
  305. return;
  306. } else {
  307. jfStatusBar.show();
  308. }
  309. _showJsonPath(curEl);
  310. };
  311. /**
  312. * 递归折叠所有层级的对象和数组节点
  313. * @param elements
  314. */
  315. function collapse(elements) {
  316. elements.each(function () {
  317. var el = $(this);
  318. if (el.children('.kv-list').length) {
  319. el.addClass('collapsed');
  320. // 只给没有id的节点分配唯一id,并生成注释
  321. if (!el.attr('id')) {
  322. el.attr('id', 'item' + (++lastItemIdGiven));
  323. let count = el.children('.kv-list').eq(0).children().length;
  324. let comment = count + (count === 1 ? ' item' : ' items');
  325. jfStyleEl[0].insertAdjacentHTML(
  326. 'beforeend',
  327. '\n#item' + lastItemIdGiven + '.collapsed:after{color: #aaa; content:" // ' + comment + '"}'
  328. );
  329. }
  330. // 递归对子节点继续折叠,确保所有嵌套层级都被处理
  331. collapse(el.children('.kv-list').children('.item-object, .item-block'));
  332. }
  333. });
  334. }
  335. /**
  336. * 创建几个全局操作的按钮,置于页面右上角即可
  337. * @private
  338. */
  339. let _buildOptionBar = function () {
  340. let optionBar = $('#optionBar');
  341. if (optionBar.length) {
  342. optionBar.html('');
  343. } else {
  344. optionBar = $('<span id="optionBar" />').appendTo(jfContent.parent());
  345. }
  346. $('<span class="x-split">|</span>').appendTo(optionBar);
  347. let buttonFormatted = $('<button class="xjf-btn xjf-btn-left">元数据</button>').appendTo(optionBar);
  348. let buttonCollapseAll = $('<button class="xjf-btn xjf-btn-mid">折叠所有</button>').appendTo(optionBar);
  349. let plainOn = false;
  350. buttonFormatted.bind('click', function (e) {
  351. if (plainOn) {
  352. plainOn = false;
  353. jfPre.hide();
  354. jfContent.show();
  355. buttonFormatted.text('元数据');
  356. } else {
  357. plainOn = true;
  358. jfPre.show();
  359. jfContent.hide();
  360. buttonFormatted.text('格式化');
  361. }
  362. jfStatusBar && jfStatusBar.hide();
  363. });
  364. buttonCollapseAll.bind('click', function (e) {
  365. // 如果内容还没有格式化过,需要再格式化一下
  366. if (plainOn) {
  367. buttonFormatted.trigger('click');
  368. }
  369. if (buttonCollapseAll.text() === '折叠所有') {
  370. buttonCollapseAll.text('展开所有');
  371. // 递归折叠所有层级的对象和数组,确保所有内容都被折叠
  372. collapse($('#jfContent .item-object, #jfContent .item-block'));
  373. } else {
  374. buttonCollapseAll.text('折叠所有');
  375. // 展开所有内容
  376. $('.item-object,.item-block').removeClass('collapsed');
  377. }
  378. jfStatusBar && jfStatusBar.hide();
  379. });
  380. };
  381. // 附加操作
  382. let _addEvents = function () {
  383. // 折叠、展开
  384. $('#jfContent span.expand').bind('click', function (ev) {
  385. ev.preventDefault();
  386. ev.stopPropagation();
  387. let parentEl = $(this).parent();
  388. parentEl.toggleClass('collapsed');
  389. if (parentEl.hasClass('collapsed')) {
  390. collapse(parentEl);
  391. }
  392. });
  393. // 点击选中:高亮
  394. $('#jfContent .item').bind('click', function (e) {
  395. let el = $(this);
  396. if (el.hasClass('x-selected')) {
  397. _toogleStatusBar(el, false);
  398. _addOptForItem(el, false);
  399. el.removeClass('x-selected');
  400. e.stopPropagation();
  401. return true;
  402. }
  403. $('.x-selected').removeClass('x-selected');
  404. el.addClass('x-selected');
  405. // 显示底部状态栏
  406. _toogleStatusBar(el, true);
  407. _addOptForItem(el, true);
  408. if (!$(e.target).is('.item .expand')) {
  409. e.stopPropagation();
  410. } else {
  411. $(e.target).parent().trigger('click');
  412. }
  413. // 触发钩子
  414. if (typeof window._OnJsonItemClickByFH === 'function') {
  415. window._OnJsonItemClickByFH(getJsonText(el));
  416. }
  417. });
  418. // 行悬停效果:只高亮当前直接悬停的item,避免嵌套冒泡
  419. let currentHoverElement = null;
  420. $('#jfContent .item').bind('mouseenter', function (e) {
  421. // 只处理视觉效果,不触发任何其他逻辑
  422. // 清除之前的悬停样式
  423. if (currentHoverElement) {
  424. currentHoverElement.removeClass('fh-hover');
  425. }
  426. // 添加当前悬停样式
  427. let el = $(this);
  428. el.addClass('fh-hover');
  429. currentHoverElement = el;
  430. // 严格阻止事件冒泡和默认行为
  431. e.stopPropagation();
  432. e.stopImmediatePropagation();
  433. e.preventDefault();
  434. });
  435. $('#jfContent .item').bind('mouseleave', function (e) {
  436. // 只处理视觉效果,不触发任何其他逻辑
  437. let el = $(this);
  438. el.removeClass('fh-hover');
  439. // 如果当前移除的元素是记录的悬停元素,清空记录
  440. if (currentHoverElement && currentHoverElement[0] === el[0]) {
  441. currentHoverElement = null;
  442. }
  443. // 严格阻止事件冒泡和默认行为
  444. e.stopPropagation();
  445. e.stopImmediatePropagation();
  446. });
  447. // 为整个jfContent区域添加鼠标离开事件,确保彻底清除悬停样式
  448. $('#jfContent').bind('mouseleave', function (e) {
  449. if (currentHoverElement) {
  450. currentHoverElement.removeClass('fh-hover');
  451. currentHoverElement = null;
  452. }
  453. });
  454. // 图片预览功能:针对所有data-is-link=1的a标签
  455. let $imgPreview = null;
  456. // 加载缓存
  457. function getImgCache() {
  458. try {
  459. return JSON.parse(sessionStorage.getItem('fehelper-img-preview-cache') || '{}');
  460. } catch (e) { return {}; }
  461. }
  462. function setImgCache(url, isImg) {
  463. let cache = getImgCache();
  464. cache[url] = isImg;
  465. sessionStorage.setItem('fehelper-img-preview-cache', JSON.stringify(cache));
  466. }
  467. $('#jfContent').on('mouseenter', 'a[data-is-link="1"]', function(e) {
  468. const url = $(this).attr('data-link-url');
  469. if (!url) return;
  470. let cache = getImgCache();
  471. if (cache.hasOwnProperty(url)) {
  472. if (cache[url]) {
  473. if (!$imgPreview || !$imgPreview.length) {
  474. $imgPreview = $('#fh-img-preview');
  475. if (!$imgPreview.length) {
  476. $imgPreview = $('<div id="fh-img-preview" style="position:fixed;z-index:999999;border:1px solid #ccc;background:#fff;padding:4px;box-shadow:0 2px 8px #0002;pointer-events:none;"><img style="max-width:300px;max-height:200px;display:block;"></div>').appendTo('body');
  477. }
  478. }
  479. $imgPreview.find('img').attr('src', url);
  480. $imgPreview.show();
  481. $(document).on('mousemove.fhimg', function(ev) {
  482. $imgPreview.css({
  483. left: ev.pageX + 20 + 'px',
  484. top: ev.pageY + 20 + 'px'
  485. });
  486. });
  487. $imgPreview.css({
  488. left: e.pageX + 20 + 'px',
  489. top: e.pageY + 20 + 'px'
  490. });
  491. }
  492. return;
  493. }
  494. // 创建图片对象尝试加载
  495. const img = new window.Image();
  496. img.src = url;
  497. img.onload = function() {
  498. // 保证页面上只存在一个浮窗节点
  499. if (!$imgPreview || !$imgPreview.length) {
  500. $imgPreview = $('#fh-img-preview');
  501. if (!$imgPreview.length) {
  502. $imgPreview = $('<div id="fh-img-preview" style="position:fixed;z-index:999999;border:1px solid #ccc;background:#fff;padding:4px;box-shadow:0 2px 8px #0002;pointer-events:none;"><img style="max-width:300px;max-height:200px;display:block;"></div>').appendTo('body');
  503. }
  504. }
  505. $imgPreview.find('img').attr('src', url);
  506. $imgPreview.show();
  507. $(document).on('mousemove.fhimg', function(ev) {
  508. $imgPreview.css({
  509. left: ev.pageX + 20 + 'px',
  510. top: ev.pageY + 20 + 'px'
  511. });
  512. });
  513. $imgPreview.css({
  514. left: e.pageX + 20 + 'px',
  515. top: e.pageY + 20 + 'px'
  516. });
  517. };
  518. img.onerror = function() {
  519. setImgCache(url, false);
  520. };
  521. }).on('mouseleave', 'a[data-is-link="1"]', function(e) {
  522. if ($imgPreview) $imgPreview.hide();
  523. $(document).off('mousemove.fhimg');
  524. });
  525. // 新增:全局监听,防止浮窗残留
  526. $(document).on('mousemove.fhimgcheck', function(ev) {
  527. let $target = $(ev.target).closest('a[data-is-link="1"]');
  528. if ($target.length === 0) {
  529. if ($imgPreview) $imgPreview.hide();
  530. $(document).off('mousemove.fhimg');
  531. }
  532. });
  533. };
  534. /**
  535. * 初始化或获取Worker实例(异步,兼容Chrome/Edge/Firefox)
  536. * @returns {Promise<Worker|null>}
  537. */
  538. let _getWorkerInstance = async function() {
  539. if (workerInstance) {
  540. return workerInstance;
  541. }
  542. let workerUrl = chrome.runtime.getURL('json-format/json-worker.js');
  543. // 判断是否为Firefox
  544. const isFirefox = typeof InstallTrigger !== 'undefined' || navigator.userAgent.includes('Firefox');
  545. try {
  546. if (isFirefox) {
  547. workerInstance = new Worker(workerUrl);
  548. return workerInstance;
  549. } else {
  550. // Chrome/Edge用fetch+Blob方式
  551. const resp = await fetch(workerUrl);
  552. const workerScript = await resp.text();
  553. const blob = new Blob([workerScript], { type: 'application/javascript' });
  554. const blobUrl = URL.createObjectURL(blob);
  555. workerInstance = new Worker(blobUrl);
  556. return workerInstance;
  557. }
  558. } catch (e) {
  559. console.error('创建Worker失败:', e);
  560. workerInstance = null;
  561. return null;
  562. }
  563. };
  564. /**
  565. * 执行代码格式化
  566. * 支持异步worker
  567. */
  568. let format = async function (jsonStr, skin) {
  569. cachedJsonString = JSON.stringify(JSON.parse(jsonStr), null, 4);
  570. _initElements();
  571. jfPre.html(htmlspecialchars(cachedJsonString));
  572. try {
  573. // 获取Worker实例(异步)
  574. let worker = await _getWorkerInstance();
  575. if (worker) {
  576. // 设置消息处理程序
  577. worker.onmessage = function (evt) {
  578. let msg = evt.data;
  579. switch (msg[0]) {
  580. case 'FORMATTING':
  581. formattingMsg.show();
  582. break;
  583. case 'FORMATTED':
  584. formattingMsg.hide();
  585. jfContent.html(msg[1]);
  586. _buildOptionBar();
  587. // 事件绑定
  588. _addEvents();
  589. // 支持文件下载
  590. _downloadSupport(cachedJsonString);
  591. break;
  592. }
  593. };
  594. // 发送格式化请求
  595. worker.postMessage({
  596. jsonString: jsonStr,
  597. skin: skin
  598. });
  599. } else {
  600. // Worker创建失败,回退到同步方式
  601. formatSync(jsonStr, skin);
  602. }
  603. } catch (e) {
  604. console.error('Worker处理失败:', e);
  605. // 出现任何错误,回退到同步方式
  606. formatSync(jsonStr, skin);
  607. }
  608. };
  609. // 同步的方式格式化
  610. let formatSync = function (jsonStr, skin) {
  611. cachedJsonString = JSON.stringify(JSON.parse(jsonStr), null, 4);
  612. _initElements();
  613. jfPre.html(htmlspecialchars(cachedJsonString));
  614. // 显示格式化进度
  615. formattingMsg.show();
  616. try {
  617. // 回退方案:使用简单模式直接显示格式化的JSON
  618. let formattedJson = JSON.stringify(JSON.parse(jsonStr), null, 4);
  619. jfContent.html(`<div id="formattedJson"><pre class="rootItem">${htmlspecialchars(formattedJson)}</pre></div>`);
  620. // 隐藏进度提示
  621. formattingMsg.hide();
  622. // 构建操作栏
  623. _buildOptionBar();
  624. // 事件绑定
  625. _addEvents();
  626. // 支持文件下载
  627. _downloadSupport(cachedJsonString);
  628. return;
  629. } catch (e) {
  630. jfContent.html(`<div class="error">JSON格式化失败: ${e.message}</div>`);
  631. // 隐藏进度提示
  632. formattingMsg.hide();
  633. }
  634. };
  635. return {
  636. format: format,
  637. formatSync: formatSync
  638. }
  639. })();