mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
- Add distinction between `getInstance` and `getInstanceOrCreate` since it is not always possible to create an instance (since you might not know the correct arguments) - Remove all the `export const *Store = *Store.getInstance<*Store>();` calls as it defeats the purpose of `Singleton`. Plus with the typing changes the appropriate `*Store.getInstance()` is "short enough". - Special case the two extension export facades to not need to use `getInstanceOrCreate`. Plus since they are just facades it is always possible to create them. - Move some other types to be also `Singleton`'s: ExtensionLoader, ExtensionDiscovery, ThemeStore, LocalizationStore, ... - Fixed dev-run always using the same port with electron inspect - Update Store documentation with new recommendations about creating instances of singletons - Fix all unit tests to create their dependent singletons Signed-off-by: Sebastian Malton <sebastian@malton.name>
134 lines
4.2 KiB
TypeScript
134 lines
4.2 KiB
TypeScript
import "./hotbar-icon.scss";
|
|
|
|
import React, { DOMAttributes } from "react";
|
|
import { observer } from "mobx-react";
|
|
import { cssNames, IClassName } from "../../utils";
|
|
import { Tooltip } from "../tooltip";
|
|
import { Avatar } from "@material-ui/core";
|
|
import { CatalogEntity, CatalogEntityContextMenu, CatalogEntityContextMenuContext } from "../../../common/catalog-entity";
|
|
import { Menu, MenuItem } from "../menu";
|
|
import { Icon } from "../icon";
|
|
import { observable } from "mobx";
|
|
import { navigate } from "../../navigation";
|
|
import { HotbarStore } from "../../../common/hotbar-store";
|
|
import { ConfirmDialog } from "../confirm-dialog";
|
|
|
|
interface Props extends DOMAttributes<HTMLElement> {
|
|
entity: CatalogEntity;
|
|
index: number;
|
|
className?: IClassName;
|
|
errorClass?: IClassName;
|
|
isActive?: boolean;
|
|
}
|
|
|
|
@observer
|
|
export class HotbarIcon extends React.Component<Props> {
|
|
@observable.deep private contextMenu: CatalogEntityContextMenuContext;
|
|
@observable menuOpen = false;
|
|
|
|
componentDidMount() {
|
|
this.contextMenu = {
|
|
menuItems: [],
|
|
navigate: (url: string) => navigate(url)
|
|
};
|
|
}
|
|
|
|
get iconString() {
|
|
let splittedName = this.props.entity.metadata.name.split(" ");
|
|
|
|
if (splittedName.length === 1) {
|
|
splittedName = splittedName[0].split("-");
|
|
}
|
|
|
|
if (splittedName.length === 1) {
|
|
splittedName = splittedName[0].split("@");
|
|
}
|
|
|
|
splittedName = splittedName.map((part) => part.replace(/\W/g, ""));
|
|
|
|
if (splittedName.length === 1) {
|
|
return splittedName[0].substring(0, 2);
|
|
} else if (splittedName.length === 2) {
|
|
return splittedName[0].substring(0, 1) + splittedName[1].substring(0, 1);
|
|
} else {
|
|
return splittedName[0].substring(0, 1) + splittedName[1].substring(0, 1) + splittedName[2].substring(0, 1);
|
|
}
|
|
}
|
|
|
|
toggleMenu() {
|
|
this.menuOpen = !this.menuOpen;
|
|
}
|
|
|
|
removeFromHotbar(item: CatalogEntity) {
|
|
const hotbar = HotbarStore.getInstance().getByName("default"); // FIXME
|
|
|
|
if (!hotbar) {
|
|
return;
|
|
}
|
|
|
|
hotbar.items = hotbar.items.filter((i) => i.entity.uid !== item.metadata.uid);
|
|
}
|
|
|
|
onMenuItemClick(menuItem: CatalogEntityContextMenu) {
|
|
if (menuItem.confirm) {
|
|
ConfirmDialog.open({
|
|
okButtonProps: {
|
|
primary: false,
|
|
accent: true,
|
|
},
|
|
ok: () => {
|
|
menuItem.onClick();
|
|
},
|
|
message: menuItem.confirm.message
|
|
});
|
|
} else {
|
|
menuItem.onClick();
|
|
}
|
|
}
|
|
|
|
render() {
|
|
const {
|
|
entity, errorClass, isActive,
|
|
children, ...elemProps
|
|
} = this.props;
|
|
const entityIconId = `hotbar-icon-${this.props.index}`;
|
|
const className = cssNames("HotbarIcon flex inline", this.props.className, {
|
|
interactive: true,
|
|
active: isActive,
|
|
});
|
|
const onOpen = async () => {
|
|
await entity.onContextMenuOpen(this.contextMenu);
|
|
this.toggleMenu();
|
|
};
|
|
const menuItems = this.contextMenu?.menuItems.filter((menuItem) => !menuItem.onlyVisibleForSource || menuItem.onlyVisibleForSource === entity.metadata.source);
|
|
|
|
return (
|
|
<div className={className}>
|
|
<Tooltip targetId={entityIconId}>{entity.metadata.name}</Tooltip>
|
|
<Avatar {...elemProps} id={entityIconId} variant="square" className={isActive ? "active" : "default"}>{this.iconString}</Avatar>
|
|
<Menu
|
|
usePortal={false}
|
|
htmlFor={entityIconId}
|
|
className="HotbarIconMenu"
|
|
isOpen={this.menuOpen}
|
|
toggleEvent="contextmenu"
|
|
position={{right: true, bottom: true }} // FIXME: position does not work
|
|
open={() => onOpen()}
|
|
close={() => this.toggleMenu()}>
|
|
<MenuItem key="remove-from-hotbar" onClick={() => this.removeFromHotbar(entity) }>
|
|
<Icon material="clear" small interactive={true} title="Remove from hotbar"/> Remove from Hotbar
|
|
</MenuItem>
|
|
{ this.contextMenu && menuItems.map((menuItem) => {
|
|
return (
|
|
<MenuItem key={menuItem.title} onClick={() => this.onMenuItemClick(menuItem) }>
|
|
<Icon material={menuItem.icon} small interactive={true} title={menuItem.title}/> {menuItem.title}
|
|
</MenuItem>
|
|
);
|
|
})}
|
|
</Menu>
|
|
{children}
|
|
</div>
|
|
);
|
|
}
|
|
}
|