webamp/js/components/ContextMenuWrapper.js
2018-08-17 21:43:26 -07:00

80 lines
2 KiB
JavaScript

import React from "react";
import PropTypes from "prop-types";
import ContextMenu from "./ContextMenu";
export default class ContextMenuWraper extends React.Component {
constructor(props) {
super(props);
this.state = {
selected: false,
offsetTop: null,
offsetLeft: null
};
}
componentWillUnmount() {
this._closeMenu();
}
_closeMenu() {
this.setState({ selected: false, offsetTop: null, offsetLeft: null });
document.removeEventListener("click", this._handleGlobalClick);
document.body.removeEventListener(
"contextmenu",
this._handleGlobalRightClick
);
}
_handleGlobalRightClick = () => {
this._closeMenu();
};
_handleGlobalClick = e => {
if (e.button === 2) {
return;
}
this._closeMenu();
};
_handleRightClick = e => {
const { pageX, pageY } = e;
this.setState({
selected: true,
// TODO: We could do an initial render to see if the menu fits here
// and do a second render if it does not.
offsetTop: pageY,
offsetLeft: pageX
});
// Even if you right click multiple times before closeing,
// we should only end up with one global listener.
document.addEventListener("click", this._handleGlobalClick);
document.body.addEventListener("contextmenu", this._handleGlobalRightClick);
e.preventDefault();
e.stopPropagation();
};
render() {
const { children, renderContents, ...passThroughProps } = this.props;
return (
<div
onContextMenu={this._handleRightClick}
style={{ width: "100%", height: "100%" }}
{...passThroughProps}
>
<ContextMenu
selected={this.state.selected}
offsetTop={this.state.offsetTop}
offsetLeft={this.state.offsetLeft}
>
{renderContents()}
</ContextMenu>
{children}
</div>
);
}
}
ContextMenuWraper.propTypes = {
children: PropTypes.any.isRequired,
renderContents: PropTypes.func.isRequired
};