Encapsulate electron bar setup
Summary: See D32311662 for details Reviewed By: mweststrate Differential Revision: D32357953 fbshipit-source-id: f951e82761f081876ae8e0409f00e19e87047726
This commit is contained in:
committed by
Facebook GitHub Bot
parent
15a59c3aea
commit
eb28fc411b
@@ -1,460 +0,0 @@
|
|||||||
/**
|
|
||||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
||||||
*
|
|
||||||
* This source code is licensed under the MIT license found in the
|
|
||||||
* LICENSE file in the root directory of this source tree.
|
|
||||||
*
|
|
||||||
* @format
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Deliberate use of remote in this context.
|
|
||||||
/* eslint-disable no-restricted-properties */
|
|
||||||
|
|
||||||
import {FlipperPlugin, FlipperDevicePlugin, PluginDefinition} from './plugin';
|
|
||||||
import {
|
|
||||||
showOpenDialog,
|
|
||||||
startFileExport,
|
|
||||||
startLinkExport,
|
|
||||||
} from './utils/exportData';
|
|
||||||
import {setStaticView} from './reducers/connections';
|
|
||||||
import {Store} from './reducers/';
|
|
||||||
import electron, {MenuItemConstructorOptions} from 'electron';
|
|
||||||
import constants from './fb-stubs/constants';
|
|
||||||
import {Logger} from 'flipper-common';
|
|
||||||
import {
|
|
||||||
_buildInMenuEntries,
|
|
||||||
_wrapInteractionHandler,
|
|
||||||
getFlipperLib,
|
|
||||||
Dialog,
|
|
||||||
} from 'flipper-plugin';
|
|
||||||
import {StyleGuide} from './sandy-chrome/StyleGuide';
|
|
||||||
import {showEmulatorLauncher} from './sandy-chrome/appinspect/LaunchEmulator';
|
|
||||||
import {webFrame} from 'electron';
|
|
||||||
import {openDeeplinkDialog} from './deeplink';
|
|
||||||
import React from 'react';
|
|
||||||
import ChangelogSheet from './chrome/ChangelogSheet';
|
|
||||||
import PluginManager from './chrome/plugin-manager/PluginManager';
|
|
||||||
import SettingsSheet from './chrome/SettingsSheet';
|
|
||||||
import reloadFlipper from './utils/reloadFlipper';
|
|
||||||
|
|
||||||
export type DefaultKeyboardAction = keyof typeof _buildInMenuEntries;
|
|
||||||
export type TopLevelMenu = 'Edit' | 'View' | 'Window' | 'Help';
|
|
||||||
|
|
||||||
export type KeyboardAction = {
|
|
||||||
action: string;
|
|
||||||
label: string;
|
|
||||||
accelerator?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type KeyboardActions = Array<DefaultKeyboardAction | KeyboardAction>;
|
|
||||||
|
|
||||||
const menuItems: Map<string, electron.MenuItem> = new Map();
|
|
||||||
|
|
||||||
let pluginActionHandler: ((action: string) => void) | null;
|
|
||||||
function actionHandler(action: string) {
|
|
||||||
if (pluginActionHandler) {
|
|
||||||
pluginActionHandler(action);
|
|
||||||
} else {
|
|
||||||
console.warn(`Unhandled keyboard action "${action}".`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export function setupMenuBar(
|
|
||||||
plugins: PluginDefinition[],
|
|
||||||
store: Store,
|
|
||||||
logger: Logger,
|
|
||||||
) {
|
|
||||||
const template = getTemplate(
|
|
||||||
electron.remote.app,
|
|
||||||
electron.remote.shell,
|
|
||||||
store,
|
|
||||||
logger,
|
|
||||||
);
|
|
||||||
|
|
||||||
// create actual menu instance
|
|
||||||
const applicationMenu = electron.remote.Menu.buildFromTemplate(template);
|
|
||||||
|
|
||||||
// update menubar
|
|
||||||
electron.remote.Menu.setApplicationMenu(applicationMenu);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function activateMenuItems(
|
|
||||||
activePlugin:
|
|
||||||
| FlipperPlugin<any, any, any>
|
|
||||||
| FlipperDevicePlugin<any, any, any>,
|
|
||||||
) {
|
|
||||||
// disable all keyboard actions
|
|
||||||
for (const item of menuItems) {
|
|
||||||
item[1].enabled = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// set plugin action handler
|
|
||||||
if (activePlugin.onKeyboardAction) {
|
|
||||||
pluginActionHandler = activePlugin.onKeyboardAction;
|
|
||||||
}
|
|
||||||
|
|
||||||
// enable keyboard actions for the current plugin
|
|
||||||
if (activePlugin.constructor.keyboardActions != null) {
|
|
||||||
(activePlugin.constructor.keyboardActions || []).forEach(
|
|
||||||
(keyboardAction) => {
|
|
||||||
const action =
|
|
||||||
typeof keyboardAction === 'string'
|
|
||||||
? keyboardAction
|
|
||||||
: keyboardAction.action;
|
|
||||||
const item = menuItems.get(action);
|
|
||||||
if (item != null) {
|
|
||||||
item.enabled = true;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// set the application menu again to make sure it updates
|
|
||||||
electron.remote.Menu?.setApplicationMenu(
|
|
||||||
electron.remote.Menu.getApplicationMenu(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function trackMenuItems(menu: string, items: MenuItemConstructorOptions[]) {
|
|
||||||
items.forEach((item) => {
|
|
||||||
if (item.label && item.click) {
|
|
||||||
item.click = _wrapInteractionHandler(
|
|
||||||
item.click,
|
|
||||||
'MenuItem',
|
|
||||||
'onClick',
|
|
||||||
'flipper:menu:' + menu,
|
|
||||||
item.label,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function getTemplate(
|
|
||||||
app: electron.App,
|
|
||||||
shell: electron.Shell,
|
|
||||||
store: Store,
|
|
||||||
logger: Logger,
|
|
||||||
): Array<MenuItemConstructorOptions> {
|
|
||||||
const exportSubmenu = [
|
|
||||||
{
|
|
||||||
label: 'File...',
|
|
||||||
accelerator: 'CommandOrControl+E',
|
|
||||||
click: () => startFileExport(store.dispatch),
|
|
||||||
},
|
|
||||||
];
|
|
||||||
if (constants.ENABLE_SHAREABLE_LINK) {
|
|
||||||
exportSubmenu.push({
|
|
||||||
label: 'Shareable Link',
|
|
||||||
accelerator: 'CommandOrControl+Shift+E',
|
|
||||||
click: () => startLinkExport(store.dispatch),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
trackMenuItems('export', exportSubmenu);
|
|
||||||
|
|
||||||
const fileSubmenu: MenuItemConstructorOptions[] = [
|
|
||||||
{
|
|
||||||
label: 'Launch Emulator...',
|
|
||||||
click() {
|
|
||||||
showEmulatorLauncher(store);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Preferences',
|
|
||||||
accelerator: 'Cmd+,',
|
|
||||||
click: () => {
|
|
||||||
Dialog.showModal((onHide) => (
|
|
||||||
<SettingsSheet platform={process.platform} onHide={onHide} />
|
|
||||||
));
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Import Flipper File...',
|
|
||||||
accelerator: 'CommandOrControl+O',
|
|
||||||
click: function () {
|
|
||||||
showOpenDialog(store);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Export',
|
|
||||||
submenu: exportSubmenu,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
trackMenuItems('file', fileSubmenu);
|
|
||||||
|
|
||||||
const supportRequestSubmenu = [
|
|
||||||
{
|
|
||||||
label: 'Create...',
|
|
||||||
click: function () {
|
|
||||||
// Dispatch an action to open the export screen of Support Request form
|
|
||||||
store.dispatch(
|
|
||||||
setStaticView(require('./fb-stubs/SupportRequestFormV2').default),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
trackMenuItems('support', supportRequestSubmenu);
|
|
||||||
|
|
||||||
fileSubmenu.push({
|
|
||||||
label: 'Support Requests',
|
|
||||||
submenu: supportRequestSubmenu,
|
|
||||||
});
|
|
||||||
|
|
||||||
const viewMenu: MenuItemConstructorOptions[] = [
|
|
||||||
{
|
|
||||||
label: 'Reload',
|
|
||||||
accelerator: 'CmdOrCtrl+R',
|
|
||||||
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
logger.track('usage', 'reload');
|
|
||||||
reloadFlipper();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Toggle Full Screen',
|
|
||||||
accelerator: (function () {
|
|
||||||
if (process.platform === 'darwin') {
|
|
||||||
return 'Ctrl+Command+F';
|
|
||||||
} else {
|
|
||||||
return 'F11';
|
|
||||||
}
|
|
||||||
})(),
|
|
||||||
click: function (_, focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
if (focusedWindow) {
|
|
||||||
focusedWindow.setFullScreen(!focusedWindow.isFullScreen());
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Actual Size',
|
|
||||||
accelerator: (function () {
|
|
||||||
return 'CmdOrCtrl+0';
|
|
||||||
})(),
|
|
||||||
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
webFrame.setZoomFactor(1);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Zoom In',
|
|
||||||
accelerator: (function () {
|
|
||||||
return 'CmdOrCtrl+=';
|
|
||||||
})(),
|
|
||||||
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
webFrame.setZoomFactor(webFrame.getZoomFactor() + 0.25);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Zoom Out',
|
|
||||||
accelerator: (function () {
|
|
||||||
return 'CmdOrCtrl+-';
|
|
||||||
})(),
|
|
||||||
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
webFrame.setZoomFactor(webFrame.getZoomFactor() - 0.25);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Manage Plugins...',
|
|
||||||
click: function () {
|
|
||||||
Dialog.showModal((onHide) => <PluginManager onHide={onHide} />);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Flipper style guide',
|
|
||||||
click() {
|
|
||||||
store.dispatch(setStaticView(StyleGuide));
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Toggle Developer Tools',
|
|
||||||
accelerator: (function () {
|
|
||||||
if (process.platform === 'darwin') {
|
|
||||||
return 'Alt+Command+I';
|
|
||||||
} else {
|
|
||||||
return 'Ctrl+Shift+I';
|
|
||||||
}
|
|
||||||
})(),
|
|
||||||
click: function (_, focusedWindow: electron.BrowserWindow | undefined) {
|
|
||||||
if (focusedWindow) {
|
|
||||||
// @ts-ignore: https://github.com/electron/electron/issues/7832
|
|
||||||
focusedWindow.toggleDevTools();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Trigger deeplink...',
|
|
||||||
click() {
|
|
||||||
openDeeplinkDialog(store);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
];
|
|
||||||
trackMenuItems('view', viewMenu);
|
|
||||||
|
|
||||||
const helpMenu: MenuItemConstructorOptions[] = [
|
|
||||||
{
|
|
||||||
label: 'Getting started',
|
|
||||||
click: function () {
|
|
||||||
getFlipperLib().openLink(
|
|
||||||
'https://fbflipper.com/docs/getting-started/index',
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Create plugins',
|
|
||||||
click: function () {
|
|
||||||
getFlipperLib().openLink('https://fbflipper.com/docs/tutorial/intro');
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Report problems',
|
|
||||||
click: function () {
|
|
||||||
getFlipperLib().openLink(constants.FEEDBACK_GROUP_LINK);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Changelog',
|
|
||||||
click() {
|
|
||||||
Dialog.showModal((onHide) => <ChangelogSheet onHide={onHide} />);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
trackMenuItems('help', helpMenu);
|
|
||||||
|
|
||||||
const template: MenuItemConstructorOptions[] = [
|
|
||||||
{
|
|
||||||
label: 'File',
|
|
||||||
submenu: fileSubmenu,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Edit',
|
|
||||||
submenu: [
|
|
||||||
{
|
|
||||||
label: 'Undo',
|
|
||||||
accelerator: 'CmdOrCtrl+Z',
|
|
||||||
role: 'undo',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Redo',
|
|
||||||
accelerator: 'Shift+CmdOrCtrl+Z',
|
|
||||||
role: 'redo',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Cut',
|
|
||||||
accelerator: 'CmdOrCtrl+X',
|
|
||||||
role: 'cut',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Copy',
|
|
||||||
accelerator: 'CmdOrCtrl+C',
|
|
||||||
role: 'copy',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Paste',
|
|
||||||
accelerator: 'CmdOrCtrl+V',
|
|
||||||
role: 'paste',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Select All',
|
|
||||||
accelerator: 'CmdOrCtrl+A',
|
|
||||||
role: 'selectAll',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'View',
|
|
||||||
submenu: viewMenu,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Window',
|
|
||||||
role: 'window',
|
|
||||||
submenu: [
|
|
||||||
{
|
|
||||||
label: 'Minimize',
|
|
||||||
accelerator: 'CmdOrCtrl+M',
|
|
||||||
role: 'minimize',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Close',
|
|
||||||
accelerator: 'CmdOrCtrl+W',
|
|
||||||
role: 'close',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Help',
|
|
||||||
role: 'help',
|
|
||||||
submenu: helpMenu,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
trackMenuItems('support', supportRequestSubmenu);
|
|
||||||
|
|
||||||
if (process.platform === 'darwin') {
|
|
||||||
const name = app.name;
|
|
||||||
template.unshift({
|
|
||||||
label: name,
|
|
||||||
submenu: [
|
|
||||||
{
|
|
||||||
label: 'About ' + name,
|
|
||||||
role: 'about',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Services',
|
|
||||||
role: 'services',
|
|
||||||
submenu: [],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Hide ' + name,
|
|
||||||
accelerator: 'Command+H',
|
|
||||||
role: 'hide',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Hide Others',
|
|
||||||
accelerator: 'Command+Shift+H',
|
|
||||||
role: 'hideOthers',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Show All',
|
|
||||||
role: 'unhide',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Quit',
|
|
||||||
accelerator: 'Command+Q',
|
|
||||||
click: function () {
|
|
||||||
app.quit();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
|
||||||
const windowMenu = template.find(function (m) {
|
|
||||||
return m.role === 'window';
|
|
||||||
});
|
|
||||||
if (windowMenu) {
|
|
||||||
(windowMenu.submenu as MenuItemConstructorOptions[]).push(
|
|
||||||
{
|
|
||||||
type: 'separator',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Bring All to Front',
|
|
||||||
role: 'front',
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return template;
|
|
||||||
}
|
|
||||||
@@ -29,7 +29,6 @@ import React, {PureComponent} from 'react';
|
|||||||
import {connect, ReactReduxContext, ReactReduxContextValue} from 'react-redux';
|
import {connect, ReactReduxContext, ReactReduxContextValue} from 'react-redux';
|
||||||
import {selectPlugin} from './reducers/connections';
|
import {selectPlugin} from './reducers/connections';
|
||||||
import {State as Store, MiddlewareAPI} from './reducers/index';
|
import {State as Store, MiddlewareAPI} from './reducers/index';
|
||||||
import {activateMenuItems} from './MenuBar';
|
|
||||||
import {Message} from './reducers/pluginMessageQueue';
|
import {Message} from './reducers/pluginMessageQueue';
|
||||||
import {IdlerImpl} from './utils/Idler';
|
import {IdlerImpl} from './utils/Idler';
|
||||||
import {processMessageQueue} from './utils/messageQueue';
|
import {processMessageQueue} from './utils/messageQueue';
|
||||||
@@ -130,26 +129,6 @@ class PluginContainer extends PureComponent<Props, State> {
|
|||||||
| null
|
| null
|
||||||
| undefined;
|
| undefined;
|
||||||
|
|
||||||
refChanged = (
|
|
||||||
ref:
|
|
||||||
| FlipperPlugin<any, any, any>
|
|
||||||
| FlipperDevicePlugin<any, any, any>
|
|
||||||
| null
|
|
||||||
| undefined,
|
|
||||||
) => {
|
|
||||||
// N.B. for Sandy plugins this lifecycle is managed by PluginRenderer
|
|
||||||
if (this.plugin) {
|
|
||||||
this.plugin._teardown();
|
|
||||||
this.plugin = null;
|
|
||||||
}
|
|
||||||
if (ref && this.props.target) {
|
|
||||||
activateMenuItems(ref);
|
|
||||||
ref._init();
|
|
||||||
this.props.logger.trackTimeSince(`activePlugin-${ref.constructor.id}`);
|
|
||||||
this.plugin = ref;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
idler?: IdlerImpl;
|
idler?: IdlerImpl;
|
||||||
pluginBeingProcessed: string | null = null;
|
pluginBeingProcessed: string | null = null;
|
||||||
|
|
||||||
|
|||||||
@@ -7,11 +7,12 @@
|
|||||||
* @format
|
* @format
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import Icon from '@ant-design/icons';
|
import Icon, {MacCommandOutlined} from '@ant-design/icons';
|
||||||
import {css} from '@emotion/css';
|
import {css} from '@emotion/css';
|
||||||
import {Button, Menu, MenuItemProps} from 'antd';
|
import {Button, Menu, MenuItemProps, Row, Tooltip} from 'antd';
|
||||||
import {
|
import {
|
||||||
NormalizedMenuEntry,
|
NormalizedMenuEntry,
|
||||||
|
NUX,
|
||||||
TrackingScope,
|
TrackingScope,
|
||||||
useTrackedCallback,
|
useTrackedCallback,
|
||||||
} from 'flipper-plugin';
|
} from 'flipper-plugin';
|
||||||
@@ -108,7 +109,14 @@ function PluginActionMenuItem({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Menu.Item onClick={trackedHandler} {...antdProps}>
|
<Menu.Item onClick={trackedHandler} {...antdProps}>
|
||||||
|
<Row justify="space-between" align="middle">
|
||||||
{label}
|
{label}
|
||||||
|
{accelerator ? (
|
||||||
|
<Tooltip title={accelerator} placement="right">
|
||||||
|
<MacCommandOutlined />
|
||||||
|
</Tooltip>
|
||||||
|
) : null}
|
||||||
|
</Row>
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -122,6 +130,7 @@ export function PluginActionsMenu() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<TrackingScope scope={`PluginActionsButton:${activePlugin.details.id}`}>
|
<TrackingScope scope={`PluginActionsButton:${activePlugin.details.id}`}>
|
||||||
|
<NUX title="Use custom plugin actions and shortcuts" placement="right">
|
||||||
<Menu mode="vertical" className={menu} selectable={false}>
|
<Menu mode="vertical" className={menu} selectable={false}>
|
||||||
<Menu.SubMenu
|
<Menu.SubMenu
|
||||||
popupOffset={[15, 0]}
|
popupOffset={[15, 0]}
|
||||||
@@ -139,6 +148,7 @@ export function PluginActionsMenu() {
|
|||||||
))}
|
))}
|
||||||
</Menu.SubMenu>
|
</Menu.SubMenu>
|
||||||
</Menu>
|
</Menu>
|
||||||
|
</NUX>
|
||||||
</TrackingScope>
|
</TrackingScope>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,12 +26,10 @@ import {
|
|||||||
} from '../reducers/plugins';
|
} from '../reducers/plugins';
|
||||||
import GK from '../fb-stubs/GK';
|
import GK from '../fb-stubs/GK';
|
||||||
import {FlipperBasePlugin} from '../plugin';
|
import {FlipperBasePlugin} from '../plugin';
|
||||||
import {setupMenuBar} from '../MenuBar';
|
|
||||||
import fs from 'fs-extra';
|
import fs from 'fs-extra';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import {default as config} from '../utils/processConfig';
|
import {default as config} from '../utils/processConfig';
|
||||||
import {notNull} from '../utils/typeUtils';
|
import {notNull} from '../utils/typeUtils';
|
||||||
import {sideEffect} from '../utils/sideEffect';
|
|
||||||
import {
|
import {
|
||||||
ActivatablePluginDetails,
|
ActivatablePluginDetails,
|
||||||
BundledPluginDetails,
|
BundledPluginDetails,
|
||||||
@@ -57,7 +55,7 @@ import {getStaticPath} from '../utils/pathUtils';
|
|||||||
import {createSandyPluginWrapper} from '../utils/createSandyPluginWrapper';
|
import {createSandyPluginWrapper} from '../utils/createSandyPluginWrapper';
|
||||||
let defaultPluginsIndex: any = null;
|
let defaultPluginsIndex: any = null;
|
||||||
|
|
||||||
export default async (store: Store, logger: Logger) => {
|
export default async (store: Store, _logger: Logger) => {
|
||||||
// expose Flipper and exact globally for dynamically loaded plugins
|
// expose Flipper and exact globally for dynamically loaded plugins
|
||||||
const globalObject: any = typeof window === 'undefined' ? global : window;
|
const globalObject: any = typeof window === 'undefined' ? global : window;
|
||||||
|
|
||||||
@@ -125,19 +123,6 @@ export default async (store: Store, logger: Logger) => {
|
|||||||
store.dispatch(addFailedPlugins(failedPlugins));
|
store.dispatch(addFailedPlugins(failedPlugins));
|
||||||
store.dispatch(registerPlugins(initialPlugins));
|
store.dispatch(registerPlugins(initialPlugins));
|
||||||
store.dispatch(pluginsInitialized());
|
store.dispatch(pluginsInitialized());
|
||||||
|
|
||||||
sideEffect(
|
|
||||||
store,
|
|
||||||
{name: 'setupMenuBar', throttleMs: 1000, fireImmediately: true},
|
|
||||||
(state) => state.plugins,
|
|
||||||
(plugins, store) => {
|
|
||||||
setupMenuBar(
|
|
||||||
[...plugins.devicePlugins.values(), ...plugins.clientPlugins.values()],
|
|
||||||
store,
|
|
||||||
logger,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
function reportVersion(pluginDetails: ActivatablePluginDetails) {
|
function reportVersion(pluginDetails: ActivatablePluginDetails) {
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ import {
|
|||||||
import {getRenderHostInstance, setRenderHostInstance} from '../RenderHost';
|
import {getRenderHostInstance, setRenderHostInstance} from '../RenderHost';
|
||||||
import isProduction from '../utils/isProduction';
|
import isProduction from '../utils/isProduction';
|
||||||
import fs from 'fs';
|
import fs from 'fs';
|
||||||
|
import {setupMenuBar} from './setupMenuBar';
|
||||||
|
|
||||||
export function initializeElectron() {
|
export function initializeElectron() {
|
||||||
const app = remote.app;
|
const app = remote.app;
|
||||||
@@ -100,6 +101,8 @@ export function initializeElectron() {
|
|||||||
desktopPath: app.getPath('desktop'),
|
desktopPath: app.getPath('desktop'),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setupMenuBar();
|
||||||
}
|
}
|
||||||
|
|
||||||
function getStaticDir() {
|
function getStaticDir() {
|
||||||
|
|||||||
236
desktop/app/src/electron/setupMenuBar.tsx
Normal file
236
desktop/app/src/electron/setupMenuBar.tsx
Normal file
@@ -0,0 +1,236 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
||||||
|
*
|
||||||
|
* This source code is licensed under the MIT license found in the
|
||||||
|
* LICENSE file in the root directory of this source tree.
|
||||||
|
*
|
||||||
|
* @format
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Deliberate use of remote in this context.
|
||||||
|
/* eslint-disable no-restricted-properties */
|
||||||
|
|
||||||
|
import electron, {MenuItemConstructorOptions} from 'electron';
|
||||||
|
import {getLogger} from 'flipper-common';
|
||||||
|
import {_buildInMenuEntries, _wrapInteractionHandler} from 'flipper-plugin';
|
||||||
|
import {webFrame} from 'electron';
|
||||||
|
import reloadFlipper from '../utils/reloadFlipper';
|
||||||
|
|
||||||
|
export function setupMenuBar() {
|
||||||
|
const template = getTemplate(electron.remote.app);
|
||||||
|
// create actual menu instance
|
||||||
|
const applicationMenu = electron.remote.Menu.buildFromTemplate(template);
|
||||||
|
// update menubar
|
||||||
|
electron.remote.Menu.setApplicationMenu(applicationMenu);
|
||||||
|
}
|
||||||
|
|
||||||
|
function trackMenuItems(menu: string, items: MenuItemConstructorOptions[]) {
|
||||||
|
items.forEach((item) => {
|
||||||
|
if (item.label && item.click) {
|
||||||
|
item.click = _wrapInteractionHandler(
|
||||||
|
item.click,
|
||||||
|
'MenuItem',
|
||||||
|
'onClick',
|
||||||
|
'flipper:menu:' + menu,
|
||||||
|
item.label,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTemplate(app: electron.App): Array<MenuItemConstructorOptions> {
|
||||||
|
const viewMenu: MenuItemConstructorOptions[] = [
|
||||||
|
{
|
||||||
|
label: 'Reload',
|
||||||
|
accelerator: 'CmdOrCtrl+R',
|
||||||
|
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
getLogger().track('usage', 'reload');
|
||||||
|
reloadFlipper();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Toggle Full Screen',
|
||||||
|
accelerator: (function () {
|
||||||
|
if (process.platform === 'darwin') {
|
||||||
|
return 'Ctrl+Command+F';
|
||||||
|
} else {
|
||||||
|
return 'F11';
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
click: function (_, focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
if (focusedWindow) {
|
||||||
|
focusedWindow.setFullScreen(!focusedWindow.isFullScreen());
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Actual Size',
|
||||||
|
accelerator: (function () {
|
||||||
|
return 'CmdOrCtrl+0';
|
||||||
|
})(),
|
||||||
|
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
webFrame.setZoomFactor(1);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Zoom In',
|
||||||
|
accelerator: (function () {
|
||||||
|
return 'CmdOrCtrl+=';
|
||||||
|
})(),
|
||||||
|
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
webFrame.setZoomFactor(webFrame.getZoomFactor() + 0.25);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Zoom Out',
|
||||||
|
accelerator: (function () {
|
||||||
|
return 'CmdOrCtrl+-';
|
||||||
|
})(),
|
||||||
|
click: function (_, _focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
webFrame.setZoomFactor(webFrame.getZoomFactor() - 0.25);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Toggle Developer Tools',
|
||||||
|
accelerator: (function () {
|
||||||
|
if (process.platform === 'darwin') {
|
||||||
|
return 'Alt+Command+I';
|
||||||
|
} else {
|
||||||
|
return 'Ctrl+Shift+I';
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
click: function (_, focusedWindow: electron.BrowserWindow | undefined) {
|
||||||
|
if (focusedWindow) {
|
||||||
|
// @ts-ignore: https://github.com/electron/electron/issues/7832
|
||||||
|
focusedWindow.toggleDevTools();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
trackMenuItems('view', viewMenu);
|
||||||
|
|
||||||
|
const template: MenuItemConstructorOptions[] = [
|
||||||
|
{
|
||||||
|
label: 'Edit',
|
||||||
|
submenu: [
|
||||||
|
{
|
||||||
|
label: 'Undo',
|
||||||
|
accelerator: 'CmdOrCtrl+Z',
|
||||||
|
role: 'undo',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Redo',
|
||||||
|
accelerator: 'Shift+CmdOrCtrl+Z',
|
||||||
|
role: 'redo',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'separator',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Cut',
|
||||||
|
accelerator: 'CmdOrCtrl+X',
|
||||||
|
role: 'cut',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Copy',
|
||||||
|
accelerator: 'CmdOrCtrl+C',
|
||||||
|
role: 'copy',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Paste',
|
||||||
|
accelerator: 'CmdOrCtrl+V',
|
||||||
|
role: 'paste',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Select All',
|
||||||
|
accelerator: 'CmdOrCtrl+A',
|
||||||
|
role: 'selectAll',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'View',
|
||||||
|
submenu: viewMenu,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Window',
|
||||||
|
role: 'window',
|
||||||
|
submenu: [
|
||||||
|
{
|
||||||
|
label: 'Minimize',
|
||||||
|
accelerator: 'CmdOrCtrl+M',
|
||||||
|
role: 'minimize',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Close',
|
||||||
|
accelerator: 'CmdOrCtrl+W',
|
||||||
|
role: 'close',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
if (process.platform === 'darwin') {
|
||||||
|
const name = app.name;
|
||||||
|
template.unshift({
|
||||||
|
label: name,
|
||||||
|
submenu: [
|
||||||
|
{
|
||||||
|
label: 'About ' + name,
|
||||||
|
role: 'about',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'separator',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Services',
|
||||||
|
role: 'services',
|
||||||
|
submenu: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'separator',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Hide ' + name,
|
||||||
|
accelerator: 'Command+H',
|
||||||
|
role: 'hide',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Hide Others',
|
||||||
|
accelerator: 'Command+Shift+H',
|
||||||
|
role: 'hideOthers',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Show All',
|
||||||
|
role: 'unhide',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'separator',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Quit',
|
||||||
|
accelerator: 'Command+Q',
|
||||||
|
click: function () {
|
||||||
|
app.quit();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
const windowMenu = template.find(function (m) {
|
||||||
|
return m.role === 'window';
|
||||||
|
});
|
||||||
|
if (windowMenu) {
|
||||||
|
(windowMenu.submenu as MenuItemConstructorOptions[]).push(
|
||||||
|
{
|
||||||
|
type: 'separator',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Bring All to Front',
|
||||||
|
role: 'front',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return template;
|
||||||
|
}
|
||||||
@@ -24,7 +24,7 @@ export {
|
|||||||
getUser,
|
getUser,
|
||||||
} from './fb-stubs/user';
|
} from './fb-stubs/user';
|
||||||
export {FlipperPlugin, FlipperDevicePlugin, BaseAction} from './plugin';
|
export {FlipperPlugin, FlipperDevicePlugin, BaseAction} from './plugin';
|
||||||
export {PluginClient, Props} from './plugin';
|
export {PluginClient, Props, KeyboardActions} from './plugin';
|
||||||
export {default as Client} from './Client';
|
export {default as Client} from './Client';
|
||||||
export {reportUsage} from 'flipper-common';
|
export {reportUsage} from 'flipper-common';
|
||||||
export {default as promiseTimeout} from './utils/promiseTimeout';
|
export {default as promiseTimeout} from './utils/promiseTimeout';
|
||||||
@@ -119,7 +119,6 @@ export {
|
|||||||
export {ElementFramework} from './ui/components/elements-inspector/ElementFramework';
|
export {ElementFramework} from './ui/components/elements-inspector/ElementFramework';
|
||||||
export {InspectorSidebar} from './ui/components/elements-inspector/sidebar';
|
export {InspectorSidebar} from './ui/components/elements-inspector/sidebar';
|
||||||
export {default as FileSelector} from './ui/components/FileSelector';
|
export {default as FileSelector} from './ui/components/FileSelector';
|
||||||
export {KeyboardActions} from './MenuBar';
|
|
||||||
export {getFlipperMediaCDN, appendAccessTokenToUrl} from './fb-stubs/user';
|
export {getFlipperMediaCDN, appendAccessTokenToUrl} from './fb-stubs/user';
|
||||||
export {Rect} from './utils/geometry';
|
export {Rect} from './utils/geometry';
|
||||||
export {Logger} from 'flipper-common';
|
export {Logger} from 'flipper-common';
|
||||||
|
|||||||
@@ -7,7 +7,6 @@
|
|||||||
* @format
|
* @format
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {KeyboardActions} from './MenuBar';
|
|
||||||
import {Logger} from 'flipper-common';
|
import {Logger} from 'flipper-common';
|
||||||
import Client from './Client';
|
import Client from './Client';
|
||||||
import {Component} from 'react';
|
import {Component} from 'react';
|
||||||
@@ -23,8 +22,19 @@ import {
|
|||||||
_SandyPluginDefinition,
|
_SandyPluginDefinition,
|
||||||
_makeShallowSerializable,
|
_makeShallowSerializable,
|
||||||
_deserializeShallowObject,
|
_deserializeShallowObject,
|
||||||
|
_buildInMenuEntries,
|
||||||
} from 'flipper-plugin';
|
} from 'flipper-plugin';
|
||||||
|
|
||||||
|
export type DefaultKeyboardAction = keyof typeof _buildInMenuEntries;
|
||||||
|
|
||||||
|
export type KeyboardAction = {
|
||||||
|
action: string;
|
||||||
|
label: string;
|
||||||
|
accelerator?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type KeyboardActions = Array<DefaultKeyboardAction | KeyboardAction>;
|
||||||
|
|
||||||
type Parameters = {[key: string]: any};
|
type Parameters = {[key: string]: any};
|
||||||
|
|
||||||
export type PluginDefinition = _SandyPluginDefinition;
|
export type PluginDefinition = _SandyPluginDefinition;
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ import {
|
|||||||
withTrackingScope,
|
withTrackingScope,
|
||||||
Dialog,
|
Dialog,
|
||||||
useTrackedCallback,
|
useTrackedCallback,
|
||||||
|
NUX,
|
||||||
} from 'flipper-plugin';
|
} from 'flipper-plugin';
|
||||||
import SetupDoctorScreen, {checkHasNewProblem} from './SetupDoctorScreen';
|
import SetupDoctorScreen, {checkHasNewProblem} from './SetupDoctorScreen';
|
||||||
import SettingsSheet from '../chrome/SettingsSheet';
|
import SettingsSheet from '../chrome/SettingsSheet';
|
||||||
@@ -43,7 +44,7 @@ import config from '../fb-stubs/config';
|
|||||||
import styled from '@emotion/styled';
|
import styled from '@emotion/styled';
|
||||||
import {showEmulatorLauncher} from './appinspect/LaunchEmulator';
|
import {showEmulatorLauncher} from './appinspect/LaunchEmulator';
|
||||||
import SupportRequestFormV2 from '../fb-stubs/SupportRequestFormV2';
|
import SupportRequestFormV2 from '../fb-stubs/SupportRequestFormV2';
|
||||||
import {setStaticView, StaticView} from '../reducers/connections';
|
import {setStaticView} from '../reducers/connections';
|
||||||
import {getLogger} from 'flipper-common';
|
import {getLogger} from 'flipper-common';
|
||||||
import {SandyRatingButton} from '../chrome/RatingButton';
|
import {SandyRatingButton} from '../chrome/RatingButton';
|
||||||
import {filterNotifications} from './notification/notificationUtils';
|
import {filterNotifications} from './notification/notificationUtils';
|
||||||
@@ -207,6 +208,11 @@ const submenu = css`
|
|||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
const MenuDividerPadded = styled(Menu.Divider)({
|
||||||
|
marginBottom: '8px !important',
|
||||||
|
});
|
||||||
|
|
||||||
function ExtrasMenu() {
|
function ExtrasMenu() {
|
||||||
const store = useStore();
|
const store = useStore();
|
||||||
|
|
||||||
@@ -235,6 +241,9 @@ function ExtrasMenu() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
<NUX
|
||||||
|
title="Find import, export, deeplink, feedback, settings, and help (welcome) here"
|
||||||
|
placement="right">
|
||||||
<Menu mode="vertical" className={menu} selectable={false}>
|
<Menu mode="vertical" className={menu} selectable={false}>
|
||||||
<SubMenu
|
<SubMenu
|
||||||
popupOffset={[10, 0]}
|
popupOffset={[10, 0]}
|
||||||
@@ -265,7 +274,7 @@ function ExtrasMenu() {
|
|||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
{config.isFBBuild ? (
|
{config.isFBBuild ? (
|
||||||
<>
|
<>
|
||||||
<Menu.Divider />
|
<MenuDividerPadded />
|
||||||
<Menu.Item
|
<Menu.Item
|
||||||
key="feedback"
|
key="feedback"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@@ -279,7 +288,7 @@ function ExtrasMenu() {
|
|||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
</>
|
</>
|
||||||
) : null}
|
) : null}
|
||||||
<Menu.Divider />
|
<MenuDividerPadded />
|
||||||
<Menu.Item key="settings" onClick={() => setShowSettings(true)}>
|
<Menu.Item key="settings" onClick={() => setShowSettings(true)}>
|
||||||
Settings
|
Settings
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
@@ -289,6 +298,7 @@ function ExtrasMenu() {
|
|||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
</SubMenu>
|
</SubMenu>
|
||||||
</Menu>
|
</Menu>
|
||||||
|
</NUX>
|
||||||
{showSettings && (
|
{showSettings && (
|
||||||
<SettingsSheet platform={process.platform} onHide={onSettingsClose} />
|
<SettingsSheet platform={process.platform} onHide={onSettingsClose} />
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -17,7 +17,14 @@ import {
|
|||||||
BugOutlined,
|
BugOutlined,
|
||||||
HistoryOutlined,
|
HistoryOutlined,
|
||||||
} from '@ant-design/icons';
|
} from '@ant-design/icons';
|
||||||
import {Dialog, Layout, theme, Tracked, TrackingScope} from 'flipper-plugin';
|
import {
|
||||||
|
Dialog,
|
||||||
|
Layout,
|
||||||
|
NUX,
|
||||||
|
theme,
|
||||||
|
Tracked,
|
||||||
|
TrackingScope,
|
||||||
|
} from 'flipper-plugin';
|
||||||
|
|
||||||
const {Text, Title} = Typography;
|
const {Text, Title} = Typography;
|
||||||
|
|
||||||
@@ -180,14 +187,18 @@ function WelcomeScreenContent() {
|
|||||||
{isProduction() ? `Version ${getAppVersion()}` : 'Development Mode'}
|
{isProduction() ? `Version ${getAppVersion()}` : 'Development Mode'}
|
||||||
</Text>
|
</Text>
|
||||||
<Tooltip title="Changelog" placement="bottom">
|
<Tooltip title="Changelog" placement="bottom">
|
||||||
|
<NUX title="See Flipper changelog" placement="top">
|
||||||
<Button
|
<Button
|
||||||
size="small"
|
size="small"
|
||||||
icon={<HistoryOutlined />}
|
icon={<HistoryOutlined />}
|
||||||
title="Changelog"
|
title="Changelog"
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
Dialog.showModal((onHide) => <ChangelogSheet onHide={onHide} />)
|
Dialog.showModal((onHide) => (
|
||||||
|
<ChangelogSheet onHide={onHide} />
|
||||||
|
))
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
</NUX>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
</Space>
|
</Space>
|
||||||
</Space>
|
</Space>
|
||||||
|
|||||||
Reference in New Issue
Block a user