Move app/src (mostly) to flipper-ui-core/src
Summary: This diff moves all UI code from app/src to app/flipper-ui-core. That is now slightly too much (e.g. node deps are not removed yet), but from here it should be easier to move things out again, as I don't want this diff to be open for too long to avoid too much merge conflicts. * But at least flipper-ui-core is Electron free :) * Killed all cross module imports as well, as they where now even more in the way * Some unit test needed some changes, most not too big (but emotion hashes got renumbered in the snapshots, feel free to ignore that) * Found some files that were actually meaningless (tsconfig in plugins, WatchTools files, that start generating compile errors, removed those Follow up work: * make flipper-ui-core configurable, and wire up flipper-server-core in Electron instead of here * remove node deps (aigoncharov) * figure out correct place to load GKs, plugins, make intern requests etc., and move to the correct module * clean up deps Reviewed By: aigoncharov Differential Revision: D32427722 fbshipit-source-id: 14fe92e1ceb15b9dcf7bece367c8ab92df927a70
This commit is contained in:
committed by
Facebook GitHub Bot
parent
54b7ce9308
commit
7e50c0466a
164
desktop/flipper-ui-core/src/deeplink.tsx
Normal file
164
desktop/flipper-ui-core/src/deeplink.tsx
Normal file
@@ -0,0 +1,164 @@
|
||||
/**
|
||||
* 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
|
||||
*/
|
||||
|
||||
import {Group, SUPPORTED_GROUPS} from './reducers/supportForm';
|
||||
import {getLogger, Logger} from 'flipper-common';
|
||||
import {Store} from './reducers/index';
|
||||
import {importDataToStore} from './utils/exportData';
|
||||
import {selectPlugin, getAllClients} from './reducers/connections';
|
||||
import {Dialog} from 'flipper-plugin';
|
||||
import {handleOpenPluginDeeplink} from './dispatcher/handleOpenPluginDeeplink';
|
||||
import {message} from 'antd';
|
||||
import {showLoginDialog} from './chrome/fb-stubs/SignInSheet';
|
||||
import {track} from './deeplinkTracking';
|
||||
|
||||
const UNKNOWN = 'Unknown deeplink';
|
||||
/**
|
||||
* Handle a flipper:// deeplink. Will throw if the URL pattern couldn't be recognised
|
||||
*/
|
||||
export async function handleDeeplink(
|
||||
store: Store,
|
||||
logger: Logger,
|
||||
query: string,
|
||||
): Promise<void> {
|
||||
const trackInteraction = track.bind(null, logger, query);
|
||||
const unknownError = () => {
|
||||
trackInteraction({
|
||||
state: 'ERROR',
|
||||
errorMessage: UNKNOWN,
|
||||
});
|
||||
throw new Error(UNKNOWN);
|
||||
};
|
||||
const uri = new URL(query);
|
||||
|
||||
trackInteraction({
|
||||
state: 'INIT',
|
||||
});
|
||||
if (uri.protocol !== 'flipper:') {
|
||||
throw unknownError();
|
||||
}
|
||||
if (uri.href === 'flipper://' || uri.pathname === '//welcome') {
|
||||
// We support an empty protocol for just opening Flipper from anywhere
|
||||
// or alternatively flipper://welcome to open the welcome screen.
|
||||
return;
|
||||
}
|
||||
if (uri.href.startsWith('flipper://open-plugin')) {
|
||||
return handleOpenPluginDeeplink(store, query, trackInteraction);
|
||||
}
|
||||
if (uri.pathname.match(/^\/*import\/*$/)) {
|
||||
const url = uri.searchParams.get('url');
|
||||
if (url) {
|
||||
const handle = Dialog.loading({
|
||||
message: 'Importing Flipper trace...',
|
||||
});
|
||||
return fetch(url)
|
||||
.then((res) => res.text())
|
||||
.then((data) => importDataToStore(url, data, store))
|
||||
.catch((e: Error) => {
|
||||
console.warn('Failed to download Flipper trace', e);
|
||||
message.error({
|
||||
duration: 0,
|
||||
content: 'Failed to download Flipper trace: ' + e,
|
||||
});
|
||||
})
|
||||
.finally(() => {
|
||||
handle.close();
|
||||
});
|
||||
}
|
||||
throw unknownError();
|
||||
} else if (uri.pathname.match(/^\/*support-form\/*$/)) {
|
||||
const formParam = uri.searchParams.get('form');
|
||||
const grp = deeplinkFormParamToGroups(formParam);
|
||||
if (grp) {
|
||||
grp.handleSupportFormDeeplinks(store);
|
||||
return;
|
||||
}
|
||||
throw unknownError();
|
||||
} else if (uri.pathname.match(/^\/*login\/*$/)) {
|
||||
const token = uri.searchParams.get('token');
|
||||
showLoginDialog(token ?? '');
|
||||
return;
|
||||
}
|
||||
const match = uriComponents(query);
|
||||
if (match.length > 1) {
|
||||
// deprecated, use the open-plugin format instead, which is more flexible
|
||||
// and will guide the user through any necessary set up steps
|
||||
// flipper://<client>/<pluginId>/<payload>
|
||||
console.warn(
|
||||
`Deprecated deeplink format: '${query}', use 'flipper://open-plugin?plugin-id=${
|
||||
match[1]
|
||||
}&client=${match[0]}&payload=${encodeURIComponent(match[2])}' instead.`,
|
||||
);
|
||||
const deepLinkPayload = match[2];
|
||||
const deepLinkParams = new URLSearchParams(deepLinkPayload);
|
||||
const deviceParam = deepLinkParams.get('device');
|
||||
|
||||
// if there is a device Param, find a matching device
|
||||
const selectedDevice = deviceParam
|
||||
? store
|
||||
.getState()
|
||||
.connections.devices.find((v) => v.title === deviceParam)
|
||||
: undefined;
|
||||
|
||||
// if a client is specified, find it, withing the device if applicable
|
||||
const selectedClient = getAllClients(store.getState().connections).find(
|
||||
(c) =>
|
||||
c.query.app === match[0] &&
|
||||
(selectedDevice == null || c.device === selectedDevice),
|
||||
);
|
||||
|
||||
store.dispatch(
|
||||
selectPlugin({
|
||||
selectedAppId: selectedClient?.id,
|
||||
selectedDevice: selectedClient ? selectedClient.device : selectedDevice,
|
||||
selectedPlugin: match[1],
|
||||
deepLinkPayload,
|
||||
}),
|
||||
);
|
||||
return;
|
||||
} else {
|
||||
throw unknownError();
|
||||
}
|
||||
}
|
||||
|
||||
function deeplinkFormParamToGroups(
|
||||
formParam: string | null,
|
||||
): Group | undefined {
|
||||
if (!formParam) {
|
||||
return undefined;
|
||||
}
|
||||
return SUPPORTED_GROUPS.find((grp) => {
|
||||
return grp.deeplinkSuffix.toLowerCase() === formParam.toLowerCase();
|
||||
});
|
||||
}
|
||||
|
||||
export const uriComponents = (url: string): Array<string> => {
|
||||
if (!url) {
|
||||
return [];
|
||||
}
|
||||
const match: Array<string> | undefined | null = url.match(
|
||||
/^flipper:\/\/([^\/]*)\/([^\/\?]*)\/?(.*)$/,
|
||||
);
|
||||
if (match) {
|
||||
return match.map(decodeURIComponent).slice(1).filter(Boolean);
|
||||
}
|
||||
return [];
|
||||
};
|
||||
|
||||
export function openDeeplinkDialog(store: Store) {
|
||||
Dialog.prompt({
|
||||
title: 'Open deeplink',
|
||||
message: 'Enter a deeplink:',
|
||||
defaultValue: 'flipper://',
|
||||
onConfirm: async (deeplink) => {
|
||||
await handleDeeplink(store, getLogger(), deeplink);
|
||||
return deeplink;
|
||||
},
|
||||
});
|
||||
}
|
||||
Reference in New Issue
Block a user