devtools.ts 3.4 KB

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