mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
120 lines
3.6 KiB
TypeScript
120 lines
3.6 KiB
TypeScript
import "./menu-actions.scss"
|
|
|
|
import React, { isValidElement } from "react";
|
|
import { observable } from "mobx";
|
|
import { observer } from "mobx-react";
|
|
import { t, Trans } from "@lingui/macro";
|
|
import { autobind, cssNames } from "../../utils";
|
|
import { ConfirmDialog } from "../confirm-dialog";
|
|
import { Icon, IconProps } from "../icon";
|
|
import { Menu, MenuItem, MenuProps } from "../menu";
|
|
import uniqueId from "lodash/uniqueId";
|
|
import isString from "lodash/isString";
|
|
import { _i18n } from "../../i18n";
|
|
|
|
export interface MenuActionsProps extends Partial<MenuProps> {
|
|
className?: string;
|
|
toolbar?: boolean; // display menu as toolbar with icons
|
|
triggerIcon?: string | IconProps | React.ReactNode;
|
|
removeConfirmationMessage?: React.ReactNode | (() => React.ReactNode);
|
|
updateAction?(): void;
|
|
removeAction?(): void;
|
|
}
|
|
|
|
@observer
|
|
export class MenuActions extends React.Component<MenuActionsProps> {
|
|
static defaultProps: MenuActionsProps = {
|
|
get removeConfirmationMessage() {
|
|
return _i18n._(t`Remove item?`)
|
|
}
|
|
}
|
|
|
|
public id = uniqueId("menu_actions_");
|
|
|
|
@observable isOpen = !!this.props.toolbar;
|
|
|
|
toggle = () => {
|
|
if (this.props.toolbar) return;
|
|
this.isOpen = !this.isOpen;
|
|
};
|
|
|
|
@autobind()
|
|
remove() {
|
|
const { removeAction} = this.props;
|
|
let { removeConfirmationMessage } = this.props;
|
|
|
|
if (typeof removeConfirmationMessage === "function") {
|
|
removeConfirmationMessage = removeConfirmationMessage();
|
|
}
|
|
ConfirmDialog.open({
|
|
ok: removeAction,
|
|
labelOk: _i18n._(t`Remove`),
|
|
message: <div>{removeConfirmationMessage}</div>,
|
|
})
|
|
}
|
|
|
|
renderTriggerIcon() {
|
|
if (this.props.toolbar) return;
|
|
const { triggerIcon = "more_vert" } = this.props;
|
|
if (isValidElement<HTMLElement>(triggerIcon)) {
|
|
const className = cssNames(triggerIcon.props.className, { active: this.isOpen });
|
|
return React.cloneElement(triggerIcon, { id: this.id, className } as any);
|
|
}
|
|
const iconProps: Partial<IconProps> = {
|
|
id: this.id,
|
|
interactive: true,
|
|
material: isString(triggerIcon) ? triggerIcon : undefined,
|
|
active: this.isOpen,
|
|
...(typeof triggerIcon === "object" ? triggerIcon : {}),
|
|
}
|
|
if (iconProps.tooltip && this.isOpen) {
|
|
delete iconProps.tooltip; // don't show tooltip for icon when menu is open
|
|
}
|
|
return (
|
|
<Icon {...iconProps}/>
|
|
)
|
|
}
|
|
|
|
render() {
|
|
const {
|
|
toolbar, children, updateAction, removeAction, triggerIcon, removeConfirmationMessage,
|
|
...menuProps
|
|
} = this.props;
|
|
let { className } = this.props;
|
|
className = cssNames("MenuActions flex", className, {
|
|
toolbar: toolbar,
|
|
gaps: toolbar, // add spacing for .flex
|
|
});
|
|
const autoClose = !toolbar;
|
|
return (
|
|
<>
|
|
{this.renderTriggerIcon()}
|
|
<Menu
|
|
htmlFor={this.id}
|
|
isOpen={this.isOpen} open={this.toggle} close={this.toggle}
|
|
className={className}
|
|
usePortal={autoClose}
|
|
closeOnScroll={autoClose}
|
|
closeOnClickItem={autoClose}
|
|
closeOnClickOutside={autoClose}
|
|
{...menuProps}
|
|
>
|
|
{children}
|
|
{updateAction && (
|
|
<MenuItem onClick={updateAction}>
|
|
<Icon material="edit" interactive={toolbar} title={_i18n._(t`Edit`)}/>
|
|
<span className="title"><Trans>Edit</Trans></span>
|
|
</MenuItem>
|
|
)}
|
|
{removeAction && (
|
|
<MenuItem onClick={this.remove}>
|
|
<Icon material="delete" interactive={toolbar} title={_i18n._(t`Delete`)}/>
|
|
<span className="title"><Trans>Remove</Trans></span>
|
|
</MenuItem>
|
|
)}
|
|
</Menu>
|
|
</>
|
|
)
|
|
}
|
|
}
|