background.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. /**
  2. * FeJson后台运行程序
  3. * @author zhaoxianlie
  4. */
  5. import MSG_TYPE from '../static/js/common.js';
  6. import Settings from '../options/settings.js';
  7. import Menu from './menu.js';
  8. import Awesome from './awesome.js';
  9. import InjectTools from './inject-tools.js';
  10. import Monkey from './monkey.js';
  11. let BgPageInstance = (function () {
  12. let FeJson = {
  13. notifyTimeoutId: -1
  14. };
  15. // 黑名单页面
  16. let blacklist = [
  17. /^https:\/\/chrome\.google\.com/
  18. ];
  19. /**
  20. * 文本格式,可以设置一个图标和标题
  21. * @param {Object} options
  22. * @config {string} type notification的类型,可选值:html、text
  23. * @config {string} icon 图标
  24. * @config {string} title 标题
  25. * @config {string} message 内容
  26. */
  27. let notifyText = function (options) {
  28. let notifyId = 'FeJson-notify-id';
  29. clearTimeout(FeJson.notifyTimeoutId);
  30. if (options.closeImmediately) {
  31. return chrome.notifications.clear(notifyId);
  32. }
  33. if (!options.icon) {
  34. options.icon = "static/img/fe-48.png";
  35. }
  36. if (!options.title) {
  37. options.title = "温馨提示";
  38. }
  39. chrome.notifications.create(notifyId, {
  40. type: 'basic',
  41. title: options.title,
  42. iconUrl: chrome.runtime.getURL(options.icon),
  43. message: options.message
  44. });
  45. FeJson.notifyTimeoutId = setTimeout(() => {
  46. chrome.notifications.clear(notifyId);
  47. }, parseInt(options.autoClose || 3000, 10));
  48. };
  49. // 像页面注入css脚本
  50. let _injectContentCss = function(tabId,toolName,isDevTool){
  51. if(isDevTool){
  52. Awesome.getContentScript(toolName, true)
  53. .then(css => {
  54. InjectTools.inject(tabId, { css })
  55. });
  56. }else{
  57. InjectTools.inject(tabId, {files: [`${toolName}/content-script.css`]});
  58. }
  59. };
  60. // 往当前页面直接注入脚本,不再使用content-script的配置了
  61. let _injectContentScripts = function (tabId) {
  62. // FH工具脚本注入
  63. Awesome.getInstalledTools().then(tools => {
  64. // 注入js
  65. let jsTools = Object.keys(tools)
  66. .filter(tool => !tools[tool]._devTool
  67. && (tools[tool].contentScriptJs || tools[tool].contentScript));
  68. let jsCodes = [];
  69. jsTools.forEach((t, i) => {
  70. let func = `window['${t.replace(/-/g, '')}ContentScript']`;
  71. jsCodes.push(`(()=>{let func=${func};func&&func();})()`);
  72. });
  73. let jsFiles = jsTools.map(tool => `${tool}/content-script.js`);
  74. InjectTools.inject(tabId, {files: jsFiles,js: jsCodes.join(';')});
  75. });
  76. // 其他开发者自定义工具脚本注入======For FH DevTools
  77. Awesome.getInstalledTools().then(tools => {
  78. let list = Object.keys(tools).filter(tool => tools[tool]._devTool);
  79. // 注入js脚本
  80. list.filter(tool => (tools[tool].contentScriptJs || tools[tool].contentScript))
  81. .map(tool => Awesome.getContentScript(tool).then(js => {
  82. InjectTools.inject(tabId, { js });
  83. }));
  84. });
  85. };
  86. /**
  87. * 动态运行工具
  88. * @param configs
  89. * @config tool 工具名称
  90. * @config withContent 默认携带的内容
  91. * @config query 请求参数
  92. * @config noPage 无页面模式
  93. * @constructor
  94. */
  95. chrome.DynamicToolRunner = async function (configs) {
  96. let tool = configs.tool || configs.page;
  97. let withContent = configs.withContent;
  98. let activeTab = null;
  99. let query = configs.query;
  100. // 如果是noPage模式,则表名只完成content-script的工作,直接发送命令即可
  101. if (configs.noPage) {
  102. let toolFunc = tool.replace(/-/g, '');
  103. chrome.tabs.query({active: true, currentWindow: true}, tabs => {
  104. let found = tabs.some(tab => {
  105. if (/^(http(s)?|file):\/\//.test(tab.url) && blacklist.every(reg => !reg.test(tab.url))) {
  106. let codes = `window['${toolFunc}NoPage'] && window['${toolFunc}NoPage'](${JSON.stringify(tab)});`;
  107. InjectTools.inject(tab.id, {js: codes});
  108. return true;
  109. }
  110. return false;
  111. });
  112. if (!found) {
  113. notifyText({
  114. message: '抱歉,此工具无法在当前页面使用!'
  115. });
  116. }
  117. });
  118. return;
  119. }
  120. chrome.tabs.query({currentWindow: true}, function (tabs) {
  121. activeTab = tabs.filter(tab => tab.active)[0];
  122. Settings.getOptions((opts) => {
  123. let isOpened = false;
  124. let tabId;
  125. // 允许在新窗口打开
  126. if (String(opts['FORBID_OPEN_IN_NEW_TAB']) === 'true') {
  127. let reg = new RegExp("^chrome.*\\/" + tool + "\\/index.html" + (query ? "\\?" + query : '') + "$", "i");
  128. for (let i = 0, len = tabs.length; i < len; i++) {
  129. if (reg.test(tabs[i].url)) {
  130. isOpened = true;
  131. tabId = tabs[i].id;
  132. break;
  133. }
  134. }
  135. }
  136. if (!isOpened) {
  137. let url = `/${tool}/index.html` + (query ? "?" + query : '');
  138. chrome.tabs.create({
  139. url,
  140. active: true
  141. }).then(tab => { FeJson[tab.id] = { content: withContent }; });
  142. } else {
  143. chrome.tabs.update(tabId, {highlighted: true}).then(tab => {
  144. FeJson[tab.id] = { content: withContent };
  145. chrome.tabs.reload(tabId);
  146. });
  147. }
  148. });
  149. });
  150. };
  151. /**
  152. * 动态在icon处显示提示
  153. * @param tips
  154. * @private
  155. */
  156. let _animateTips = (tips) => {
  157. setTimeout(() => {
  158. chrome.action.setBadgeText({text: tips});
  159. setTimeout(() => {
  160. chrome.action.setBadgeText({text: ''});
  161. }, 2000);
  162. }, 3000);
  163. };
  164. /**
  165. * 插件图标点击后的默认动作
  166. * @param request
  167. * @param sender
  168. * @param callback
  169. */
  170. let browserActionClickedHandler = function (request, sender, callback) {
  171. chrome.DynamicToolRunner({
  172. tool: MSG_TYPE.JSON_FORMAT
  173. });
  174. };
  175. /**
  176. * 更新browser action的点击动作
  177. * @param action install / upgrade / offload
  178. * @param showTips 是否notify
  179. * @param menuOnly 只管理Menu
  180. * @private
  181. */
  182. let _updateBrowserAction = function (action, showTips, menuOnly) {
  183. if (!menuOnly) {
  184. // 如果有安装过工具,则显示Popup模式
  185. Awesome.getInstalledTools().then(tools => {
  186. if (Object.keys(tools).length > 1) {
  187. chrome.action.setPopup({ popup: '/popup/index.html' });
  188. } else {
  189. // 删除popup page
  190. chrome.action.setPopup({ popup: '' });
  191. // 否则点击图标,直接打开页面
  192. if (!chrome.action.onClicked.hasListener(browserActionClickedHandler)) {
  193. chrome.action.onClicked.addListener(browserActionClickedHandler);
  194. }
  195. }
  196. });
  197. if (action === 'offload') {
  198. _animateTips('-1');
  199. } else if(!!action) {
  200. _animateTips('+1');
  201. }
  202. } else {
  203. // 重绘菜单
  204. Menu.rebuild();
  205. }
  206. if (showTips) {
  207. let actionTxt = '';
  208. switch (action) {
  209. case 'install':
  210. actionTxt = '工具已「安装」成功,并已添加到弹出下拉列表,点击FeHelper图标可正常使用!';
  211. break;
  212. case 'offload':
  213. actionTxt = '工具已「卸载」成功,并已从弹出下拉列表中移除!';
  214. break;
  215. case 'menu-install':
  216. actionTxt = '已将此工具快捷方式加入到「右键菜单」中!';
  217. break;
  218. case 'menu-offload':
  219. actionTxt = '已将此工具快捷方式从「右键菜单」中移除!';
  220. break;
  221. default:
  222. actionTxt = '恭喜,操作成功!';
  223. }
  224. notifyText({
  225. message: actionTxt,
  226. autoClose: 2500
  227. });
  228. }
  229. };
  230. // 捕获当前页面可视区域
  231. let _captureVisibleTab = function (callback) {
  232. chrome.tabs.captureVisibleTab(null, {format: 'png', quality: 100}, uri => {
  233. callback && callback(uri);
  234. });
  235. };
  236. let _addScreenShotByPages = function(params,callback){
  237. chrome.tabs.captureVisibleTab(null, {format: 'png', quality: 100}, uri => {
  238. callback({ params, uri });
  239. });
  240. };
  241. let _showScreenShotResult = function(data){
  242. // 确保截图数据完整有效
  243. if (!data || !data.screenshots || !data.screenshots.length) {
  244. return;
  245. }
  246. chrome.DynamicToolRunner({
  247. tool: 'screenshot',
  248. withContent: data
  249. });
  250. };
  251. let _colorPickerCapture = function(params) {
  252. chrome.tabs.query({active: true, currentWindow: true}, function (tabs) {
  253. chrome.tabs.captureVisibleTab(null, {format: 'png'}, function (dataUrl) {
  254. let js = `window.colorpickerNoPage(${JSON.stringify({
  255. setPickerImage: true,
  256. pickerImage: dataUrl
  257. })})`;
  258. InjectTools.inject(tabs[0].id, { js });
  259. });
  260. });
  261. };
  262. let _codeBeautify = function(params){
  263. if (['javascript', 'css'].includes(params.fileType)) {
  264. Awesome.StorageMgr.get('JS_CSS_PAGE_BEAUTIFY').then(val => {
  265. if(val !== '0') {
  266. let js = `window._codebutifydetect_('${params.fileType}')`;
  267. InjectTools.inject(params.tabId, { js });
  268. }
  269. });
  270. }
  271. };
  272. /**
  273. * 接收来自content_scripts发来的消息
  274. */
  275. let _addExtensionListener = function () {
  276. _updateBrowserAction();
  277. chrome.runtime.onMessage.addListener(function (request, sender, callback) {
  278. // 如果发生了错误,就啥都别干了
  279. if (chrome.runtime.lastError) {
  280. return true;
  281. }
  282. // 动态安装工具或者卸载工具,需要更新browserAction
  283. if (request.type === MSG_TYPE.DYNAMIC_TOOL_INSTALL_OR_OFFLOAD) {
  284. _updateBrowserAction(request.action, request.showTips, request.menuOnly);
  285. callback && callback();
  286. }
  287. // 截屏
  288. else if (request.type === MSG_TYPE.CAPTURE_VISIBLE_PAGE) {
  289. _captureVisibleTab(callback);
  290. }
  291. // 直接处理content-script.js中的截图请求
  292. else if (request.type === 'fh-screenshot-capture-visible') {
  293. _captureVisibleTab(callback);
  294. }
  295. // 打开动态工具页面
  296. else if (request.type === MSG_TYPE.OPEN_DYNAMIC_TOOL) {
  297. chrome.DynamicToolRunner(request);
  298. callback && callback();
  299. }
  300. // 打开其他页面
  301. else if (request.type === MSG_TYPE.OPEN_PAGE) {
  302. chrome.DynamicToolRunner({
  303. tool: request.page
  304. });
  305. callback && callback();
  306. }
  307. // 任何事件,都可以通过这个钩子来完成
  308. else if (request.type === MSG_TYPE.DYNAMIC_ANY_THING) {
  309. switch(request.thing){
  310. case 'save-options':
  311. notifyText({
  312. message: '配置修改已生效,请继续使用!',
  313. autoClose: 2000
  314. });
  315. break;
  316. case 'trigger-screenshot':
  317. // 处理从popup触发的截图请求
  318. if (request.tabId) {
  319. _triggerScreenshotTool(request.tabId);
  320. } else {
  321. chrome.DynamicToolRunner({
  322. tool: 'screenshot',
  323. noPage: true
  324. });
  325. }
  326. break;
  327. case 'request-jsonformat-options':
  328. Awesome.StorageMgr.get(request.params).then(result => {
  329. Object.keys(result).forEach(key => {
  330. if (['MAX_JSON_KEYS_NUMBER', 'JSON_FORMAT_THEME'].includes(key)) {
  331. result[key] = parseInt(result[key]);
  332. } else {
  333. result[key] = (result[key] !== 'false');
  334. }
  335. });
  336. callback && callback(result);
  337. });
  338. return true; // 这个返回true是非常重要的!!!要不然callback会拿不到结果
  339. case 'save-jsonformat-options':
  340. Awesome.StorageMgr.set(request.params).then(() => {
  341. callback && callback();
  342. });
  343. return true;
  344. case 'toggle-jsonformat-options':
  345. Awesome.StorageMgr.get('JSON_TOOL_BAR_ALWAYS_SHOW').then(result => {
  346. let show = result !== false;
  347. Awesome.StorageMgr.set('JSON_TOOL_BAR_ALWAYS_SHOW',!show).then(() => {
  348. callback && callback(!show);
  349. });
  350. });
  351. return true; // 这个返回true是非常重要的!!!要不然callback会拿不到结果
  352. case 'code-beautify':
  353. _codeBeautify(request.params);
  354. break;
  355. case 'close-beautify':
  356. Awesome.StorageMgr.set('JS_CSS_PAGE_BEAUTIFY',0);
  357. break;
  358. case 'qr-decode':
  359. chrome.DynamicToolRunner({
  360. withContent: request.params.uri,
  361. tool: 'qr-code',
  362. query: `mode=decode`
  363. });
  364. break;
  365. case 'request-page-content':
  366. request.params = FeJson[request.tabId];
  367. delete FeJson[request.tabId];
  368. break;
  369. case 'set-page-timing-data':
  370. chrome.DynamicToolRunner({
  371. tool: 'page-timing',
  372. withContent: request.wpoInfo
  373. });
  374. break;
  375. case 'color-picker-capture':
  376. _colorPickerCapture(request.params);
  377. break;
  378. case 'add-screen-shot-by-pages':
  379. _addScreenShotByPages(request.params,callback);
  380. return true;
  381. case 'page-screenshot-done':
  382. _showScreenShotResult(request.params);
  383. break;
  384. case 'request-monkey-start':
  385. Monkey.start(request.params);
  386. break;
  387. case 'inject-content-css':
  388. _injectContentCss(sender.tab.id,request.tool,!!request.devTool);
  389. break;
  390. }
  391. callback && callback(request.params);
  392. } else {
  393. callback && callback();
  394. }
  395. return true;
  396. });
  397. // 每开一个窗口,都向内容脚本注入一个js,绑定tabId
  398. chrome.tabs.onUpdated.addListener(function (tabId, changeInfo, tab) {
  399. if (String(changeInfo.status).toLowerCase() === "complete") {
  400. if(/^(http(s)?|file):\/\//.test(tab.url) && blacklist.every(reg => !reg.test(tab.url))){
  401. InjectTools.inject(tabId, { js: `window.__FH_TAB_ID__=${tabId};` });
  402. _injectContentScripts(tabId);
  403. }
  404. }
  405. });
  406. // 安装与更新
  407. chrome.runtime.onInstalled.addListener(({reason, previousVersion}) => {
  408. switch (reason) {
  409. case 'install':
  410. chrome.runtime.openOptionsPage();
  411. break;
  412. case 'update':
  413. _animateTips('+++1');
  414. if (previousVersion === '2019.12.2415') {
  415. notifyText({
  416. message: '历尽千辛万苦,FeHelper已升级到最新版本,可以到插件设置页去安装旧版功能了!',
  417. autoClose: 5000
  418. });
  419. }
  420. // 从V2020.02.1413版本开始,本地的数据存储大部分迁移至chrome.storage.local
  421. // 这里需要对老版本升级过来的情况进行强制数据迁移
  422. let getAbsNum = num => parseInt(num.split(/\./).map(n => n.padStart(4, '0')).join(''), 10);
  423. // let preVN = getAbsNum(previousVersion);
  424. // let minVN = getAbsNum('2020.02.1413');
  425. // if (preVN < minVN) {
  426. // Awesome.makeStorageUnlimited();
  427. // setTimeout(() => chrome.runtime.reload(), 1000 * 5);
  428. // }
  429. break;
  430. }
  431. });
  432. // 卸载
  433. chrome.runtime.setUninstallURL(chrome.runtime.getManifest().homepage_url);
  434. };
  435. /**
  436. * 检查插件更新
  437. * @private
  438. */
  439. let _checkUpdate = function () {
  440. setTimeout(() => {
  441. chrome.runtime.requestUpdateCheck((status) => {
  442. if (status === "update_available") {
  443. chrome.runtime.reload();
  444. }
  445. });
  446. }, 1000 * 30);
  447. };
  448. /**
  449. * 初始化
  450. */
  451. let _init = function () {
  452. _checkUpdate();
  453. _addExtensionListener();
  454. // 添加截图工具直接命令 - 通过右键菜单触发
  455. chrome.contextMenus.onClicked.addListener((info, tab) => {
  456. if (info.menuItemId === 'fehelper-screenshot-page') {
  457. _triggerScreenshotTool(tab.id);
  458. }
  459. });
  460. // 创建截图工具右键菜单
  461. chrome.contextMenus.create({
  462. id: 'fehelper-screenshot-page',
  463. title: '网页截图',
  464. contexts: ['page']
  465. });
  466. Menu.rebuild();
  467. // 定期清理冗余的垃圾
  468. setTimeout(() => {
  469. Awesome.gcLocalFiles();
  470. }, 1000 * 10);
  471. };
  472. /**
  473. * 触发截图工具的执行
  474. * @param {number} tabId - 标签页ID
  475. */
  476. function _triggerScreenshotTool(tabId) {
  477. // 先尝试直接发送消息给content script
  478. chrome.tabs.sendMessage(tabId, {
  479. type: 'fh-screenshot-start'
  480. }).then(() => {
  481. // 成功触发
  482. }).catch(() => {
  483. // 如果发送消息失败,使用noPage模式
  484. chrome.DynamicToolRunner({
  485. tool: 'screenshot',
  486. noPage: true
  487. });
  488. });
  489. }
  490. return {
  491. pageCapture: _captureVisibleTab,
  492. init: _init
  493. };
  494. })();
  495. BgPageInstance.init();