devtools.ts 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. import { dialog, Menu } from 'electron/main';
  2. import * as fs from 'fs';
  3. import * as url from 'url';
  4. import { ipcMainInternal } from '@electron/internal/browser/ipc-main-internal';
  5. import * as ipcMainUtils from '@electron/internal/browser/ipc-main-internal-utils';
  6. import { IPC_MESSAGES } from '@electron/internal//common/ipc-messages';
  7. const convertToMenuTemplate = function (items: ContextMenuItem[], handler: (id: number) => void) {
  8. return items.map(function (item) {
  9. const transformed: Electron.MenuItemConstructorOptions = item.type === 'subMenu' ? {
  10. type: 'submenu',
  11. label: item.label,
  12. enabled: item.enabled,
  13. submenu: convertToMenuTemplate(item.subItems, handler)
  14. } : item.type === 'separator' ? {
  15. type: 'separator'
  16. } : item.type === 'checkbox' ? {
  17. type: 'checkbox',
  18. label: item.label,
  19. enabled: item.enabled,
  20. checked: item.checked
  21. } : {
  22. type: 'normal',
  23. label: item.label,
  24. enabled: item.enabled
  25. };
  26. if (item.id != null) {
  27. transformed.click = () => handler(item.id);
  28. }
  29. return transformed;
  30. });
  31. };
  32. const getEditMenuItems = function (): Electron.MenuItemConstructorOptions[] {
  33. return [
  34. { role: 'undo' },
  35. { role: 'redo' },
  36. { type: 'separator' },
  37. { role: 'cut' },
  38. { role: 'copy' },
  39. { role: 'paste' },
  40. { role: 'pasteAndMatchStyle' },
  41. { role: 'delete' },
  42. { role: 'selectAll' }
  43. ];
  44. };
  45. const isChromeDevTools = function (pageURL: string) {
  46. const { protocol } = url.parse(pageURL);
  47. return protocol === 'devtools:';
  48. };
  49. const assertChromeDevTools = function (contents: Electron.WebContents, api: string) {
  50. const pageURL = contents.getURL();
  51. if (!isChromeDevTools(pageURL)) {
  52. console.error(`Blocked ${pageURL} from calling ${api}`);
  53. throw new Error(`Blocked ${api}`);
  54. }
  55. };
  56. ipcMainInternal.handle(IPC_MESSAGES.INSPECTOR_CONTEXT_MENU, function (event, items: ContextMenuItem[], isEditMenu: boolean) {
  57. return new Promise<number | void>(resolve => {
  58. assertChromeDevTools(event.sender, 'window.InspectorFrontendHost.showContextMenuAtPoint()');
  59. const template = isEditMenu ? getEditMenuItems() : convertToMenuTemplate(items, resolve);
  60. const menu = Menu.buildFromTemplate(template);
  61. const window = event.sender.getOwnerBrowserWindow()!;
  62. menu.popup({ window, callback: () => resolve() });
  63. });
  64. });
  65. ipcMainInternal.handle(IPC_MESSAGES.INSPECTOR_SELECT_FILE, async function (event) {
  66. assertChromeDevTools(event.sender, 'window.UI.createFileSelectorElement()');
  67. const result = await dialog.showOpenDialog({});
  68. if (result.canceled) return [];
  69. const path = result.filePaths[0];
  70. const data = await fs.promises.readFile(path);
  71. return [path, data];
  72. });
  73. ipcMainUtils.handleSync(IPC_MESSAGES.INSPECTOR_CONFIRM, async function (event, message: string = '', title: string = '') {
  74. assertChromeDevTools(event.sender, 'window.confirm()');
  75. const options = {
  76. message: String(message),
  77. title: String(title),
  78. buttons: ['OK', 'Cancel'],
  79. cancelId: 1
  80. };
  81. const window = event.sender.getOwnerBrowserWindow()!;
  82. const { response } = await dialog.showMessageBox(window, options);
  83. return response === 0;
  84. });