showMenu refactor - context is now available inside menu

This commit is contained in:
Jan Prochazka
2020-12-03 09:33:29 +01:00
parent e4b605162e
commit 327f2140cf
11 changed files with 144 additions and 65 deletions

View File

@@ -17,6 +17,7 @@ import UploadsProvider from './utility/UploadsProvider';
import ThemeHelmet from './themes/ThemeHelmet'; import ThemeHelmet from './themes/ThemeHelmet';
import PluginsProvider from './plugins/PluginsProvider'; import PluginsProvider from './plugins/PluginsProvider';
import { ExtensionsProvider } from './utility/useExtensions'; import { ExtensionsProvider } from './utility/useExtensions';
import { MenuLayerProvider } from './modals/showMenu';
function App() { function App() {
return ( return (
@@ -33,8 +34,10 @@ function App() {
<CurrentThemeProvider> <CurrentThemeProvider>
<UploadsProvider> <UploadsProvider>
<ModalLayerProvider> <ModalLayerProvider>
<MenuLayerProvider>
<ThemeHelmet /> <ThemeHelmet />
<Screen /> <Screen />
</MenuLayerProvider>
</ModalLayerProvider> </ModalLayerProvider>
</UploadsProvider> </UploadsProvider>
</CurrentThemeProvider> </CurrentThemeProvider>

View File

@@ -15,6 +15,7 @@ import { ModalLayer } from './modals/showModal';
import DragAndDropFileTarget from './DragAndDropFileTarget'; import DragAndDropFileTarget from './DragAndDropFileTarget';
import { useUploadsZone } from './utility/UploadsProvider'; import { useUploadsZone } from './utility/UploadsProvider';
import useTheme from './theme/useTheme'; import useTheme from './theme/useTheme';
import { MenuLayer } from './modals/showMenu';
const BodyDiv = styled.div` const BodyDiv = styled.div`
position: fixed; position: fixed;
@@ -132,6 +133,7 @@ export default function Screen() {
<StatusBar /> <StatusBar />
</StausBarContainer> </StausBarContainer>
<ModalLayer /> <ModalLayer />
<MenuLayer />
<DragAndDropFileTarget inputProps={getInputProps()} isDragActive={isDragActive} /> <DragAndDropFileTarget inputProps={getInputProps()} isDragActive={isDragActive} />
</div> </div>

View File

@@ -4,11 +4,11 @@ import styled from 'styled-components';
import { DropDownMenuItem, DropDownMenuDivider } from './modals/DropDownMenu'; import { DropDownMenuItem, DropDownMenuDivider } from './modals/DropDownMenu';
import { useOpenedTabs, useSetOpenedTabs, useCurrentDatabase, useSetCurrentDatabase } from './utility/globalState'; import { useOpenedTabs, useSetOpenedTabs, useCurrentDatabase, useSetCurrentDatabase } from './utility/globalState';
import { showMenu } from './modals/DropDownMenu';
import { getConnectionInfo } from './utility/metadataLoaders'; import { getConnectionInfo } from './utility/metadataLoaders';
import { FontIcon } from './icons'; import { FontIcon } from './icons';
import useTheme from './theme/useTheme'; import useTheme from './theme/useTheme';
import usePropsCompare from './utility/usePropsCompare'; import usePropsCompare from './utility/usePropsCompare';
import { useShowMenu } from './modals/showMenu';
// const files = [ // const files = [
// { name: 'app.js' }, // { name: 'app.js' },
@@ -126,6 +126,7 @@ function getDbIcon(key) {
export default function TabsPanel() { export default function TabsPanel() {
// const formatDbKey = (conid, database) => `${database}-${conid}`; // const formatDbKey = (conid, database) => `${database}-${conid}`;
const theme = useTheme(); const theme = useTheme();
const showMenu = useShowMenu();
const tabs = useOpenedTabs(); const tabs = useOpenedTabs();
const setOpenedTabs = useSetOpenedTabs(); const setOpenedTabs = useSetOpenedTabs();

View File

@@ -4,7 +4,7 @@ import _ from 'lodash';
import React from 'react'; import React from 'react';
import styled from 'styled-components'; import styled from 'styled-components';
import { FontIcon } from '../icons'; import { FontIcon } from '../icons';
import { showMenu } from '../modals/DropDownMenu'; // import { showMenu } from '../modals/DropDownMenu';
import useTheme from '../theme/useTheme'; import useTheme from '../theme/useTheme';
import { useSetOpenedTabs, useAppObjectParams } from '../utility/globalState'; import { useSetOpenedTabs, useAppObjectParams } from '../utility/globalState';
@@ -60,7 +60,7 @@ export function AppObjectCore({
if (!Menu) return; if (!Menu) return;
event.preventDefault(); event.preventDefault();
showMenu(event.pageX, event.pageY, <Menu data={data} makeAppObj={makeAppObj} {...appObjectParams} />); // showMenu(event.pageX, event.pageY, <Menu data={data} makeAppObj={makeAppObj} {...appObjectParams} />);
}; };
const Component = component == 'div' ? AppObjectDiv : AppObjectSpan; const Component = component == 'div' ? AppObjectDiv : AppObjectSpan;

View File

@@ -4,8 +4,10 @@ import { DropDownMenuItem } from '../modals/DropDownMenu';
import { openNewTab } from '../utility/common'; import { openNewTab } from '../utility/common';
import ImportExportModal from '../modals/ImportExportModal'; import ImportExportModal from '../modals/ImportExportModal';
import { getDefaultFileFormat } from '../utility/fileformats'; import { getDefaultFileFormat } from '../utility/fileformats';
import { useSetOpenedTabs } from '../utility/globalState';
function Menu({ data, setOpenedTabs, showModal, extensions }) { function Menu({ data, showModal, extensions }) {
const setOpenedTabs = useSetOpenedTabs();
const { connection, name } = data; const { connection, name } = data;
const tooltip = `${connection.displayName || connection.server}\n${name}`; const tooltip = `${connection.displayName || connection.server}\n${name}`;

View File

@@ -1,6 +1,6 @@
// @ts-nocheck // @ts-nocheck
import React from 'react'; import React from 'react';
import { DropDownMenuItem, DropDownMenuDivider, showMenu } from '../modals/DropDownMenu'; import { DropDownMenuItem, DropDownMenuDivider } from '../modals/DropDownMenu';
import styled from 'styled-components'; import styled from 'styled-components';
import keycodes from '../utility/keycodes'; import keycodes from '../utility/keycodes';
import { parseFilter, createMultiLineFilter } from 'dbgate-filterparser'; import { parseFilter, createMultiLineFilter } from 'dbgate-filterparser';
@@ -10,6 +10,7 @@ import FilterMultipleValuesModal from '../modals/FilterMultipleValuesModal';
import SetFilterModal from '../modals/SetFilterModal'; import SetFilterModal from '../modals/SetFilterModal';
import { FontIcon } from '../icons'; import { FontIcon } from '../icons';
import useTheme from '../theme/useTheme'; import useTheme from '../theme/useTheme';
import { useShowMenu } from '../modals/showMenu';
// import { $ } from '../../Utility/jquery'; // import { $ } from '../../Utility/jquery';
// import autobind from 'autobind-decorator'; // import autobind from 'autobind-decorator';
// import * as React from 'react'; // import * as React from 'react';
@@ -182,6 +183,7 @@ export default function DataFilterControl({
onFocusGrid, onFocusGrid,
}) { }) {
const showModal = useShowModal(); const showModal = useShowModal();
const showMenu = useShowMenu();
const theme = useTheme(); const theme = useTheme();
const [filterState, setFilterState] = React.useState('empty'); const [filterState, setFilterState] = React.useState('empty');
const setFilterText = (filter) => { const setFilterText = (filter) => {

View File

@@ -22,7 +22,6 @@ import DataGridToolbar from './DataGridToolbar';
// import usePropsCompare from '../utility/usePropsCompare'; // import usePropsCompare from '../utility/usePropsCompare';
import ColumnHeaderControl from './ColumnHeaderControl'; import ColumnHeaderControl from './ColumnHeaderControl';
import InlineButton from '../widgets/InlineButton'; import InlineButton from '../widgets/InlineButton';
import { showMenu } from '../modals/DropDownMenu';
import DataGridContextMenu from './DataGridContextMenu'; import DataGridContextMenu from './DataGridContextMenu';
import LoadingInfo from '../widgets/LoadingInfo'; import LoadingInfo from '../widgets/LoadingInfo';
import ErrorInfo from '../widgets/ErrorInfo'; import ErrorInfo from '../widgets/ErrorInfo';
@@ -30,6 +29,7 @@ import { openNewTab } from '../utility/common';
import { useSetOpenedTabs } from '../utility/globalState'; import { useSetOpenedTabs } from '../utility/globalState';
import { FontIcon } from '../icons'; import { FontIcon } from '../icons';
import useTheme from '../theme/useTheme'; import useTheme from '../theme/useTheme';
import { useShowMenu } from '../modals/showMenu';
const GridContainer = styled.div` const GridContainer = styled.div`
position: absolute; position: absolute;
@@ -138,6 +138,7 @@ export default function DataGridCore(props) {
const [autofillDragStartCell, setAutofillDragStartCell] = React.useState(nullCell); const [autofillDragStartCell, setAutofillDragStartCell] = React.useState(nullCell);
const [autofillSelectedCells, setAutofillSelectedCells] = React.useState(emptyCellArray); const [autofillSelectedCells, setAutofillSelectedCells] = React.useState(emptyCellArray);
const [focusFilterInputs, setFocusFilterInputs] = React.useState({}); const [focusFilterInputs, setFocusFilterInputs] = React.useState({});
const showMenu = useShowMenu();
const autofillMarkerCell = React.useMemo( const autofillMarkerCell = React.useMemo(
() => () =>

View File

@@ -2,6 +2,8 @@ import React from 'react';
import ReactDOM from 'react-dom'; import ReactDOM from 'react-dom';
import styled from 'styled-components'; import styled from 'styled-components';
import { LoadingToken, sleep } from '../utility/common'; import { LoadingToken, sleep } from '../utility/common';
import useDocumentClick from '../utility/useDocumentClick';
import { useHideMenu } from './showMenu';
const ContextMenuStyled = styled.ul` const ContextMenuStyled = styled.ul`
position: absolute; position: absolute;
@@ -167,6 +169,11 @@ export function DropDownMenuItem({ children, keyText = undefined, onClick }) {
// } // }
export function ContextMenu({ left, top, children }) { export function ContextMenu({ left, top, children }) {
const hideMenu = useHideMenu();
useDocumentClick(async () => {
await sleep(0);
hideMenu();
});
return <ContextMenuStyled style={{ left: `${left}px`, top: `${top}px` }}>{children}</ContextMenuStyled>; return <ContextMenuStyled style={{ left: `${left}px`, top: `${top}px` }}>{children}</ContextMenuStyled>;
} }
@@ -204,71 +211,71 @@ export function ContextMenu({ left, top, children }) {
// parentMenu: PropTypes.any // parentMenu: PropTypes.any
// }; // };
let menuHandle = null; // let menuHandle = null;
let hideToken = null; // let hideToken = null;
function showMenuCore(left, top, contentHolder, closeCallback = null) { // function showMenuCore(left, top, contentHolder, closeCallback = null) {
let container = document.createElement('div'); // let container = document.createElement('div');
let handle = { // let handle = {
container, // container,
closeCallback, // closeCallback,
close() { // close() {
this.container.remove(); // this.container.remove();
}, // },
}; // };
document.body.appendChild(container); // document.body.appendChild(container);
ReactDOM.render( // ReactDOM.render(
<ContextMenu left={left} top={top}> // <ContextMenu left={left} top={top}>
{contentHolder} // {contentHolder}
</ContextMenu>, // </ContextMenu>,
container // container
); // );
return handle; // return handle;
} // }
export function showMenu(left, top, contentHolder, closeCallback = null) { // export function showMenu(left, top, contentHolder, closeCallback = null) {
hideMenu(); // hideMenu();
if (hideToken) hideToken.cancel(); // if (hideToken) hideToken.cancel();
menuHandle = showMenuCore(left, top, contentHolder, closeCallback); // menuHandle = showMenuCore(left, top, contentHolder, closeCallback);
captureMouseDownEvents(); // captureMouseDownEvents();
} // }
function captureMouseDownEvents() { // function captureMouseDownEvents() {
document.addEventListener('mousedown', mouseDownListener, true); // document.addEventListener('mousedown', mouseDownListener, true);
} // }
function releaseMouseDownEvents() { // function releaseMouseDownEvents() {
document.removeEventListener('mousedown', mouseDownListener, true); // document.removeEventListener('mousedown', mouseDownListener, true);
} // }
function captureMouseUpEvents() { // function captureMouseUpEvents() {
document.addEventListener('mouseup', mouseUpListener, true); // document.addEventListener('mouseup', mouseUpListener, true);
} // }
function releaseMouseUpEvents() { // function releaseMouseUpEvents() {
document.removeEventListener('mouseup', mouseUpListener, true); // document.removeEventListener('mouseup', mouseUpListener, true);
} // }
async function mouseDownListener(e) { // async function mouseDownListener(e) {
captureMouseUpEvents(); // captureMouseUpEvents();
} // }
async function mouseUpListener(e) { // async function mouseUpListener(e) {
let token = new LoadingToken(); // let token = new LoadingToken();
hideToken = token; // hideToken = token;
await sleep(0); // await sleep(0);
if (token.isCanceled) return; // if (token.isCanceled) return;
hideMenu(); // hideMenu();
} // }
function hideMenu() { // function hideMenu() {
if (menuHandle == null) return; // if (menuHandle == null) return;
menuHandle.close(); // menuHandle.close();
if (menuHandle.closeCallback) menuHandle.closeCallback(); // if (menuHandle.closeCallback) menuHandle.closeCallback();
menuHandle = null; // menuHandle = null;
releaseMouseDownEvents(); // releaseMouseDownEvents();
releaseMouseUpEvents(); // releaseMouseUpEvents();
} // }
function getElementOffset(element) { function getElementOffset(element) {
var de = document.documentElement; var de = document.documentElement;

View File

@@ -0,0 +1,40 @@
import React from 'react';
import { ContextMenu } from './DropDownMenu';
import uuidv1 from 'uuid/v1';
const Context = React.createContext(null);
export function MenuLayerProvider({ children }) {
const [menu, setMenu] = React.useState(null);
return <Context.Provider value={[menu, setMenu]}>{children}</Context.Provider>;
}
export function MenuLayer() {
const [menu] = React.useContext(Context);
return (
<div>
{menu != null && (
<ContextMenu key={menu.menuid} left={menu.left} top={menu.top}>
{menu.menu}
</ContextMenu>
)}
</div>
);
}
export function useHideMenu() {
const [, setMenu] = React.useContext(Context);
return () => setMenu(null);
}
export function useShowMenu() {
const [, setMenu] = React.useContext(Context);
const showMenu = (left, top, menu) => {
const menuid = uuidv1();
setMenu({ menuid, left, top, menu });
};
return showMenu;
// const container = document.createElement('div');
// document.body.appendChild(container);
// ReactDOM.render(<ShowModalComponent renderModal={renderModal} container={container} />, container);
}

View File

@@ -0,0 +1,20 @@
import React from 'react';
export default function useDocumentClick(callback) {
const mouseUpListener = React.useCallback((e) => {
callback();
document.removeEventListener('mouseup', mouseUpListener, true);
}, []);
const mouseDownListener = React.useCallback((e) => {
document.addEventListener('mouseup', mouseUpListener, true);
document.removeEventListener('mousedown', mouseDownListener, true);
}, []);
React.useEffect(() => {
document.addEventListener('mousedown', mouseDownListener, true);
return () => {
document.removeEventListener('mouseup', mouseUpListener, true);
document.removeEventListener('mousedown', mouseDownListener, true);
};
}, []);
}

View File

@@ -1,10 +1,11 @@
import React from 'react'; import React from 'react';
import { FontIcon } from '../icons'; import { FontIcon } from '../icons';
import { showMenu } from '../modals/DropDownMenu'; import { useShowMenu } from '../modals/showMenu';
import InlineButton from './InlineButton'; import InlineButton from './InlineButton';
export default function DropDownButton({ children }) { export default function DropDownButton({ children }) {
const buttonRef = React.useRef(null); const buttonRef = React.useRef(null);
const showMenu = useShowMenu();
const handleShowMenu = () => { const handleShowMenu = () => {
const rect = buttonRef.current.getBoundingClientRect(); const rect = buttonRef.current.getBoundingClientRect();