@elastic/eui
Version:
Elastic UI Component Library
401 lines (396 loc) • 20.1 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.EuiContextMenuClass = exports.EuiContextMenu = void 0;
var _react = _interopRequireWildcard(require("react"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _classnames = _interopRequireDefault(require("classnames"));
var _services = require("../../services");
var _horizontal_rule = require("../horizontal_rule");
var _context_menu_panel = require("./context_menu_panel");
var _context_menu_item = require("./context_menu_item");
var _context_menu = require("./context_menu.styles");
var _react2 = require("@emotion/react");
var _excluded = ["isSeparator", "key"],
_excluded2 = ["panel", "name", "key", "icon", "onClick"],
_excluded3 = ["id", "title", "items", "content", "width", "initialFocusedItemIndex"],
_excluded4 = ["stylesMemoizer", "panels", "onPanelChange", "className", "initialPanelId", "height"];
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
function _objectWithoutProperties(e, t) { if (null == e) return {}; var o, r, i = _objectWithoutPropertiesLoose(e, t); if (Object.getOwnPropertySymbols) { var n = Object.getOwnPropertySymbols(e); for (r = 0; r < n.length; r++) o = n[r], t.indexOf(o) >= 0 || {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]); } return i; }
function _objectWithoutPropertiesLoose(r, e) { if (null == r) return {}; var t = {}; for (var n in r) if ({}.hasOwnProperty.call(r, n)) { if (e.indexOf(n) >= 0) continue; t[n] = r[n]; } return t; }
function _classCallCheck(a, n) { if (!(a instanceof n)) throw new TypeError("Cannot call a class as a function"); }
function _defineProperties(e, r) { for (var t = 0; t < r.length; t++) { var o = r[t]; o.enumerable = o.enumerable || !1, o.configurable = !0, "value" in o && (o.writable = !0), Object.defineProperty(e, _toPropertyKey(o.key), o); } }
function _createClass(e, r, t) { return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, "prototype", { writable: !1 }), e; }
function _callSuper(t, o, e) { return o = _getPrototypeOf(o), _possibleConstructorReturn(t, _isNativeReflectConstruct() ? Reflect.construct(o, e || [], _getPrototypeOf(t).constructor) : o.apply(t, e)); }
function _possibleConstructorReturn(t, e) { if (e && ("object" == _typeof(e) || "function" == typeof e)) return e; if (void 0 !== e) throw new TypeError("Derived constructors may only return object or undefined"); return _assertThisInitialized(t); }
function _assertThisInitialized(e) { if (void 0 === e) throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); return e; }
function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); }
function _getPrototypeOf(t) { return _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf.bind() : function (t) { return t.__proto__ || Object.getPrototypeOf(t); }, _getPrototypeOf(t); }
function _inherits(t, e) { if ("function" != typeof e && null !== e) throw new TypeError("Super expression must either be null or a function"); t.prototype = Object.create(e && e.prototype, { constructor: { value: t, writable: !0, configurable: !0 } }), Object.defineProperty(t, "prototype", { writable: !1 }), e && _setPrototypeOf(t, e); }
function _setPrototypeOf(t, e) { return _setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function (t, e) { return t.__proto__ = e, t; }, _setPrototypeOf(t, e); }
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); } /*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/
var isItemSeparator = function isItemSeparator(item) {
return item.isSeparator === true;
};
function mapIdsToPanels(panels) {
var map = {};
panels.forEach(function (panel) {
map[panel.id] = panel;
});
return map;
}
function mapIdsToPreviousPanels(panels) {
var idToPreviousPanelIdMap = {};
panels.forEach(function (panel) {
if (Array.isArray(panel.items)) {
panel.items.forEach(function (item) {
if (isItemSeparator(item)) return;
var isCloseable = item.panel !== undefined;
if (isCloseable) {
idToPreviousPanelIdMap[item.panel] = panel.id;
}
});
}
});
return idToPreviousPanelIdMap;
}
function mapPanelItemsToPanels(panels) {
var idAndItemIndexToPanelIdMap = {};
panels.forEach(function (panel) {
idAndItemIndexToPanelIdMap[panel.id] = {};
if (panel.items) {
panel.items.forEach(function (item, index) {
if (isItemSeparator(item)) return;
if (item.panel) {
idAndItemIndexToPanelIdMap[panel.id][index] = item.panel;
}
});
}
});
return idAndItemIndexToPanelIdMap;
}
var EuiContextMenuClass = exports.EuiContextMenuClass = /*#__PURE__*/function (_Component) {
function EuiContextMenuClass(props) {
var _this;
_classCallCheck(this, EuiContextMenuClass);
_this = _callSuper(this, EuiContextMenuClass, [props]);
_defineProperty(_this, "hasPreviousPanel", function (panelId) {
var previousPanelId = _this.state.idToPreviousPanelIdMap[panelId];
return typeof previousPanelId !== 'undefined';
});
_defineProperty(_this, "showNextPanel", function (itemIndex) {
if (itemIndex == null) {
return;
}
var nextPanelId = _this.state.idAndItemIndexToPanelIdMap[_this.state.incomingPanelId][itemIndex];
if (nextPanelId) {
if (_this.state.isUsingKeyboardToNavigate) {
_this.setState(function (_ref) {
var idToPanelMap = _ref.idToPanelMap;
return {
focusedItemIndex: idToPanelMap[nextPanelId].initialFocusedItemIndex
};
});
}
_this.showPanel(nextPanelId, 'next');
}
});
_defineProperty(_this, "showPreviousPanel", function () {
// If there's a previous panel, then we can close the current panel to go back to it.
if (_this.hasPreviousPanel(_this.state.incomingPanelId)) {
var previousPanelId = _this.state.idToPreviousPanelIdMap[_this.state.incomingPanelId];
// Set focus on the item which shows the panel we're leaving.
var previousPanel = _this.state.idToPanelMap[previousPanelId];
var focusedItemIndex = previousPanel.items.filter(function (item) {
return !isItemSeparator(item);
}).findIndex(function (item) {
return item.panel === _this.state.incomingPanelId;
});
if (focusedItemIndex !== -1) {
_this.setState({
focusedItemIndex: focusedItemIndex
});
}
_this.showPanel(previousPanelId, 'previous');
}
});
_defineProperty(_this, "onIncomingPanelHeightChange", function (height) {
_this.setState(function (_ref2) {
var prevHeight = _ref2.height;
if (height === prevHeight) {
return null;
}
return {
height: height
};
});
});
_defineProperty(_this, "onOutGoingPanelTransitionComplete", function () {
_this.setState({
isOutgoingPanelVisible: false
});
});
_defineProperty(_this, "onUseKeyboardToNavigate", function () {
if (!_this.state.isUsingKeyboardToNavigate) {
_this.setState({
isUsingKeyboardToNavigate: true
});
}
});
_defineProperty(_this, "mapIdsToRenderedItems", function () {
var panels = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
var idToRenderedItemsMap = {};
var idToPanelMap = mapIdsToPanels(panels);
// Pre-rendering the items lets us check reference equality inside of EuiContextMenuPanel.
panels.forEach(function (panel) {
idToRenderedItemsMap[panel.id] = _this.renderItems(panel.items, idToPanelMap);
});
return idToRenderedItemsMap;
});
_this.state = {
prevProps: {},
idToPanelMap: {},
idToPreviousPanelIdMap: {},
idAndItemIndexToPanelIdMap: {},
idToRenderedItemsMap: _this.mapIdsToRenderedItems(_this.props.panels),
height: undefined,
outgoingPanelId: undefined,
incomingPanelId: props.initialPanelId,
transitionDirection: undefined,
isOutgoingPanelVisible: false,
focusedItemIndex: undefined,
isUsingKeyboardToNavigate: false
};
return _this;
}
_inherits(EuiContextMenuClass, _Component);
return _createClass(EuiContextMenuClass, [{
key: "componentDidUpdate",
value: function componentDidUpdate(prevProps) {
if (prevProps.panels !== this.props.panels) {
this.setState({
idToRenderedItemsMap: this.mapIdsToRenderedItems(this.props.panels)
});
}
}
}, {
key: "showPanel",
value: function showPanel(panelId, direction) {
var _this$props$onPanelCh, _this$props;
this.setState({
outgoingPanelId: this.state.incomingPanelId,
incomingPanelId: panelId,
transitionDirection: direction,
isOutgoingPanelVisible: true
});
(_this$props$onPanelCh = (_this$props = this.props).onPanelChange) === null || _this$props$onPanelCh === void 0 || _this$props$onPanelCh.call(_this$props, {
panelId: panelId,
direction: direction
});
}
}, {
key: "renderItems",
value: function renderItems() {
var _this2 = this;
var items = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : [];
var idToPanelMap = arguments.length > 1 ? arguments[1] : undefined;
return items.map(function (item, index) {
if (item.renderItem) {
var _item$key;
return (0, _react2.jsx)(_react.Fragment, {
key: (_item$key = item.key) !== null && _item$key !== void 0 ? _item$key : index
}, item.renderItem());
}
if (isItemSeparator(item)) {
var omit = item.isSeparator,
_item$key2 = item.key,
_key = _item$key2 === void 0 ? index : _item$key2,
_rest = _objectWithoutProperties(item, _excluded);
return (0, _react2.jsx)(_horizontal_rule.EuiHorizontalRule, _extends({
key: _key,
margin: "xs",
role: "separator"
}, _rest));
}
var panel = item.panel,
name = item.name,
key = item.key,
icon = item.icon,
onClick = item.onClick,
rest = _objectWithoutProperties(item, _excluded2);
var targetPanel = panel != null ? idToPanelMap[String(panel)] : undefined;
var ariaHasPopup = panel == null || targetPanel == null ? undefined : targetPanel && Array.isArray(targetPanel.items) && targetPanel.items.length > 0 ? 'menu' : 'true'; // for custom content (no items) we shouldn't indicate it as "menu"
var onClickHandler = panel ? function (event) {
if (onClick && event) {
event.persist();
}
// This component is commonly wrapped in a EuiOutsideClickDetector, which means we'll
// need to wait for that logic to complete before re-rendering the DOM via showPanel.
window.requestAnimationFrame(function () {
if (onClick) {
onClick(event);
}
_this2.showNextPanel(index);
});
} : onClick;
return (0, _react2.jsx)(_context_menu_item.EuiContextMenuItem, _extends({
key: key || (typeof name === 'string' ? name : undefined) || index,
icon: icon,
onClick: onClickHandler,
hasPanel: Boolean(panel),
"aria-haspopup": ariaHasPopup
}, rest), name);
});
}
}, {
key: "renderPanel",
value: function renderPanel(panelId, transitionType) {
var _this3 = this;
var panel = this.state.idToPanelMap[panelId];
if (!panel) {
return;
}
var _id = panel.id,
title = panel.title,
_items = panel.items,
content = panel.content,
_width = panel.width,
initialFocusedItemIndex = panel.initialFocusedItemIndex,
rest = _objectWithoutProperties(panel, _excluded3);
// As above, we need to wait for EuiOutsideClickDetector to complete its logic before
// re-rendering via showPanel.
var onClose;
if (this.hasPreviousPanel(panelId)) {
onClose = function onClose() {
return window.requestAnimationFrame(_this3.showPreviousPanel);
};
}
var cssStyles = {
position: 'absolute',
label: 'euiContextMenu__panel'
};
return (0, _react2.jsx)(_context_menu_panel.EuiContextMenuPanel, _extends({
key: panelId,
height: this.props.height,
css: cssStyles,
onHeightChange: transitionType === 'in' ? this.onIncomingPanelHeightChange : undefined,
onTransitionComplete: transitionType === 'out' ? this.onOutGoingPanelTransitionComplete : undefined,
title: title,
onClose: onClose,
transitionType: this.state.isOutgoingPanelVisible ? transitionType : undefined,
transitionDirection: this.state.isOutgoingPanelVisible ? this.state.transitionDirection : undefined,
items: this.state.idToRenderedItemsMap[panelId],
initialFocusedItemIndex: this.state.isUsingKeyboardToNavigate ? this.state.focusedItemIndex : initialFocusedItemIndex,
onUseKeyboardToNavigate: this.onUseKeyboardToNavigate,
showNextPanel: this.showNextPanel,
showPreviousPanel: this.showPreviousPanel
}, rest), content);
}
}, {
key: "render",
value: function render() {
var _this$props2 = this.props,
stylesMemoizer = _this$props2.stylesMemoizer,
panels = _this$props2.panels,
onPanelChange = _this$props2.onPanelChange,
className = _this$props2.className,
initialPanelId = _this$props2.initialPanelId,
height = _this$props2.height,
rest = _objectWithoutProperties(_this$props2, _excluded4);
var incomingPanel = this.renderPanel(this.state.incomingPanelId, 'in');
var outgoingPanel;
if (this.state.isOutgoingPanelVisible) {
outgoingPanel = this.renderPanel(this.state.outgoingPanelId, 'out');
}
var width = this.state.idToPanelMap[this.state.incomingPanelId] && this.state.idToPanelMap[this.state.incomingPanelId].width ? this.state.idToPanelMap[this.state.incomingPanelId].width : undefined;
var classes = (0, _classnames.default)('euiContextMenu', className);
var styles = stylesMemoizer(_context_menu.euiContextMenuStyles);
var cssStyles = [styles.euiContextMenu, height && styles.fixedHeight];
return (0, _react2.jsx)("div", _extends({
css: cssStyles,
className: classes,
style: {
height: height !== null && height !== void 0 ? height : this.state.height,
width: width
}
}, rest), outgoingPanel, incomingPanel);
}
}], [{
key: "getDerivedStateFromProps",
value: function getDerivedStateFromProps(nextProps, prevState) {
var panels = nextProps.panels;
if (panels && prevState.prevProps.panels !== panels) {
return {
prevProps: {
panels: panels
},
idToPanelMap: mapIdsToPanels(panels),
idToPreviousPanelIdMap: mapIdsToPreviousPanels(panels),
idAndItemIndexToPanelIdMap: mapPanelItemsToPanels(panels)
};
}
return null;
}
}]);
}(_react.Component);
_defineProperty(EuiContextMenuClass, "defaultProps", {
panels: []
});
EuiContextMenuClass.propTypes = {
className: _propTypes.default.string,
"aria-label": _propTypes.default.string,
"data-test-subj": _propTypes.default.string,
css: _propTypes.default.any,
panels: _propTypes.default.arrayOf(_propTypes.default.shape({
id: _propTypes.default.oneOfType([_propTypes.default.string.isRequired, _propTypes.default.number.isRequired]).isRequired,
title: _propTypes.default.node,
items: _propTypes.default.arrayOf(_propTypes.default.shape({
name: _propTypes.default.node,
key: _propTypes.default.string,
panel: _propTypes.default.oneOfType([_propTypes.default.string.isRequired, _propTypes.default.number.isRequired]),
isSeparator: _propTypes.default.oneOf([true]),
/**
* Defines the width of the HR.
*/
size: _propTypes.default.any,
margin: _propTypes.default.any,
className: _propTypes.default.string,
"aria-label": _propTypes.default.string,
"data-test-subj": _propTypes.default.string,
css: _propTypes.default.any,
/**
* Allows rendering any custom content alongside your array of context menu items.
* Accepts either a component or an inline function component that returns any JSX.
*/
renderItem: _propTypes.default.oneOfType([_propTypes.default.func.isRequired, _propTypes.default.func.isRequired])
}).isRequired),
content: _propTypes.default.node,
width: _propTypes.default.any,
initialFocusedItemIndex: _propTypes.default.number,
className: _propTypes.default.string,
"aria-label": _propTypes.default.string,
"data-test-subj": _propTypes.default.string,
css: _propTypes.default.any
}).isRequired),
/**
* Optional callback that fires on every panel change. Passes back
* the new panel ID and whether its direction was `next` or `previous`.
*/
onPanelChange: _propTypes.default.func,
initialPanelId: _propTypes.default.oneOfType([_propTypes.default.string.isRequired, _propTypes.default.number.isRequired]),
height: _propTypes.default.any
};
var EuiContextMenu = exports.EuiContextMenu = (0, _services.withEuiStylesMemoizer)(EuiContextMenuClass);