1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00

Hotbar disabled items (#2710)

* Saving more entity data to HotbarItem

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Adding generic MaterialTooltip component

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Move HotbarCell to separate component

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Abstract out HotbarEntityIcon from HotbarIcon

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Styling disabled hotbar items

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Migration for adding extra data to hotbar items

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Testing migration

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Some cleaning up

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Bump migration version

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>

* Bump app version in package.json

Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>
This commit is contained in:
Alex Andreev 2021-05-13 10:24:58 +03:00 committed by GitHub
parent 2d0609ed24
commit a930d5f14f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 501 additions and 204 deletions

View File

@ -2,7 +2,7 @@
"name": "open-lens", "name": "open-lens",
"productName": "OpenLens", "productName": "OpenLens",
"description": "OpenLens - Open Source IDE for Kubernetes", "description": "OpenLens - Open Source IDE for Kubernetes",
"version": "5.0.0-beta.4", "version": "5.0.0-beta.5",
"main": "static/build/main.js", "main": "static/build/main.js",
"copyright": "© 2021 OpenLens Authors", "copyright": "© 2021 OpenLens Authors",
"license": "MIT", "license": "MIT",

View File

@ -24,6 +24,27 @@ import { CatalogEntityItem } from "../../renderer/components/+catalog/catalog-en
import { ClusterStore } from "../cluster-store"; import { ClusterStore } from "../cluster-store";
import { HotbarStore } from "../hotbar-store"; import { HotbarStore } from "../hotbar-store";
jest.mock("../../renderer/api/catalog-entity-registry", () => ({
catalogEntityRegistry: {
items: [
{
metadata: {
uid: "1dfa26e2ebab15780a3547e9c7fa785c",
name: "mycluster",
source: "local"
}
},
{
metadata: {
uid: "55b42c3c7ba3b04193416cda405269a5",
name: "my_shiny_cluster",
source: "remote"
}
}
]
}
}));
const testCluster = { const testCluster = {
uid: "test", uid: "test",
name: "test", name: "test",
@ -87,6 +108,17 @@ const awsCluster = {
} }
}; };
jest.mock("electron", () => {
return {
app: {
getVersion: () => "99.99.99",
getPath: () => "tmp",
getLocale: () => "en",
setLoginItemSettings: (): void => void 0,
}
};
});
describe("HotbarStore", () => { describe("HotbarStore", () => {
beforeEach(() => { beforeEach(() => {
ClusterStore.resetInstance(); ClusterStore.resetInstance();
@ -264,4 +296,111 @@ describe("HotbarStore", () => {
console.error = err; console.error = err;
}); });
}); });
describe("pre beta-5 migrations", () => {
beforeEach(() => {
HotbarStore.resetInstance();
const mockOpts = {
"tmp": {
"lens-hotbar-store.json": JSON.stringify({
__internal__: {
migrations: {
version: "5.0.0-beta.3"
}
},
"hotbars": [
{
"id": "3caac17f-aec2-4723-9694-ad204465d935",
"name": "myhotbar",
"items": [
{
"entity": {
"uid": "1dfa26e2ebab15780a3547e9c7fa785c"
}
},
{
"entity": {
"uid": "55b42c3c7ba3b04193416cda405269a5"
}
},
{
"entity": {
"uid": "176fd331968660832f62283219d7eb6e"
}
},
{
"entity": {
"uid": "61c4fb45528840ebad1badc25da41d14",
"name": "user1-context",
"source": "local"
}
},
{
"entity": {
"uid": "27d6f99fe9e7548a6e306760bfe19969",
"name": "foo2",
"source": "local"
}
},
null,
{
"entity": {
"uid": "c0b20040646849bb4dcf773e43a0bf27",
"name": "multinode-demo",
"source": "local"
}
},
null,
null,
null,
null,
null
]
}
],
})
}
};
mockFs(mockOpts);
return HotbarStore.createInstance().load();
});
afterEach(() => {
mockFs.restore();
});
it("allows to retrieve a hotbar", () => {
const hotbar = HotbarStore.getInstance().getById("3caac17f-aec2-4723-9694-ad204465d935");
expect(hotbar.id).toBe("3caac17f-aec2-4723-9694-ad204465d935");
});
it("clears cells without entity", () => {
const items = HotbarStore.getInstance().hotbars[0].items;
expect(items[2]).toBeNull();
});
it("adds extra data to cells with according entity", () => {
const items = HotbarStore.getInstance().hotbars[0].items;
expect(items[0]).toEqual({
entity: {
name: "mycluster",
source: "local",
uid: "1dfa26e2ebab15780a3547e9c7fa785c"
}
});
expect(items[1]).toEqual({
entity: {
name: "my_shiny_cluster",
source: "remote",
uid: "55b42c3c7ba3b04193416cda405269a5"
}
});
});
});
}); });

View File

@ -29,6 +29,8 @@ import isNull from "lodash/isNull";
export interface HotbarItem { export interface HotbarItem {
entity: { entity: {
uid: string; uid: string;
name?: string;
source?: string;
}; };
params?: { params?: {
[key: string]: string; [key: string]: string;
@ -144,9 +146,14 @@ export class HotbarStore extends BaseStore<HotbarStoreModel> {
} }
} }
@action
addToHotbar(item: CatalogEntityItem, cellIndex = -1) { addToHotbar(item: CatalogEntityItem, cellIndex = -1) {
const hotbar = this.getActive(); const hotbar = this.getActive();
const newItem = { entity: { uid: item.id }}; const newItem = { entity: {
uid: item.id,
name: item.name,
source: item.source
}};
if (hotbar.items.find(i => i?.entity.uid === item.id)) { if (hotbar.items.find(i => i?.entity.uid === item.id)) {
return; return;
@ -167,6 +174,7 @@ export class HotbarStore extends BaseStore<HotbarStoreModel> {
} }
} }
@action
removeFromHotbar(uid: string) { removeFromHotbar(uid: string) {
const hotbar = this.getActive(); const hotbar = this.getActive();
const index = hotbar.items.findIndex((i) => i?.entity.uid === uid); const index = hotbar.items.findIndex((i) => i?.entity.uid === uid);

View File

@ -0,0 +1,30 @@
import type { Hotbar } from "../../common/hotbar-store";
import { migration } from "../migration-wrapper";
import { catalogEntityRegistry } from "../../renderer/api/catalog-entity-registry";
export default migration({
version: "5.0.0-beta.5",
run(store) {
const hotbars: Hotbar[] = store.get("hotbars");
hotbars.forEach((hotbar, hotbarIndex) => {
hotbar.items.forEach((item, itemIndex) => {
const entity = catalogEntityRegistry.items.find((entity) => entity.metadata.uid === item?.entity.uid);
if (!entity) {
// Clear disabled item
hotbars[hotbarIndex].items[itemIndex] = null;
} else {
// Save additional data
hotbars[hotbarIndex].items[itemIndex].entity = {
...item.entity,
name: entity.metadata.name,
source: entity.metadata.source
};
}
});
});
store.set("hotbars", hotbars);
}
});

View File

@ -2,8 +2,10 @@
import version500alpha0 from "./5.0.0-alpha.0"; import version500alpha0 from "./5.0.0-alpha.0";
import version500alpha2 from "./5.0.0-alpha.2"; import version500alpha2 from "./5.0.0-alpha.2";
import version500beta5 from "./5.0.0-beta.5";
export default { export default {
...version500alpha0, ...version500alpha0,
...version500alpha2 ...version500alpha2,
...version500beta5,
}; };

View File

@ -0,0 +1,28 @@
import React from "react";
import { makeStyles, Tooltip, TooltipProps } from "@material-ui/core";
const useStyles = makeStyles(() => ({
arrow: {
color: "var(--tooltipBackground)",
},
tooltip: {
fontSize: 12,
backgroundColor: "var(--tooltipBackground)",
color: "var(--textColorAccent)",
padding: 8,
boxShadow: "0 8px 16px rgba(0,0,0,0.24)"
},
}));
export function MaterialTooltip(props: TooltipProps) {
const classes = useStyles();
return (
<Tooltip classes={classes} {...props}/>
);
}
MaterialTooltip.defaultProps = {
arrow: true
};

View File

@ -0,0 +1,32 @@
import "./hotbar-menu.scss";
import "./hotbar.commands";
import React, { HTMLAttributes, ReactNode, useState } from "react";
import { cssNames } from "../../utils";
interface Props extends HTMLAttributes<HTMLDivElement> {
children?: ReactNode;
index: number;
innerRef?: React.LegacyRef<HTMLDivElement>;
}
export function HotbarCell({ innerRef, children, className, ...rest }: Props) {
const [animating, setAnimating] = useState(false);
const onAnimationEnd = () => { setAnimating(false); };
const onClick = () => {
setAnimating(!className.includes("isDraggingOver"));
};
return (
<div
className={cssNames("HotbarCell", { animating }, className)}
onAnimationEnd={onAnimationEnd}
onClick={onClick}
ref={innerRef}
{...rest}
>
{children}
</div>
);
}

View File

@ -0,0 +1,115 @@
import "./hotbar-icon.scss";
import React, { DOMAttributes } from "react";
import { observable } from "mobx";
import { observer } from "mobx-react";
import randomColor from "randomcolor";
import { CatalogEntity, CatalogEntityContextMenu, CatalogEntityContextMenuContext } from "../../../common/catalog";
import { catalogCategoryRegistry } from "../../api/catalog-category-registry";
import { catalogEntityRegistry } from "../../api/catalog-entity-registry";
import { navigate } from "../../navigation";
import { cssNames, IClassName } from "../../utils";
import { ConfirmDialog } from "../confirm-dialog";
import { Icon } from "../icon";
import { HotbarIcon } from "./hotbar-icon";
interface Props extends DOMAttributes<HTMLElement> {
entity: CatalogEntity;
className?: IClassName;
errorClass?: IClassName;
remove: (uid: string) => void;
}
@observer
export class HotbarEntityIcon extends React.Component<Props> {
@observable.deep private contextMenu: CatalogEntityContextMenuContext;
componentDidMount() {
this.contextMenu = {
menuItems: [],
navigate: (url: string) => navigate(url)
};
}
get kindIcon() {
const className = "badge";
const category = catalogCategoryRegistry.getCategoryForEntity(this.props.entity);
if (!category) {
return <Icon material="bug_report" className={className} />;
}
if (category.metadata.icon.includes("<svg")) {
return <Icon svg={category.metadata.icon} className={className} />;
} else {
return <Icon material={category.metadata.icon} className={className} />;
}
}
get ledIcon() {
const className = cssNames("led", { online: this.props.entity.status.phase == "connected"}); // TODO: make it more generic
return <div className={className} />;
}
isActive(item: CatalogEntity) {
return catalogEntityRegistry.activeEntity?.metadata?.uid == item.getId();
}
onMenuItemClick(menuItem: CatalogEntityContextMenu) {
if (menuItem.confirm) {
ConfirmDialog.open({
okButtonProps: {
primary: false,
accent: true,
},
ok: () => {
menuItem.onClick();
},
message: menuItem.confirm.message
});
} else {
menuItem.onClick();
}
}
generateAvatarStyle(entity: CatalogEntity): React.CSSProperties {
return {
"backgroundColor": randomColor({ seed: `${entity.metadata.name}-${entity.metadata.source}`, luminosity: "dark" })
};
}
render() {
const {
entity, errorClass, remove,
children, ...elemProps
} = this.props;
const className = cssNames("HotbarEntityIcon", this.props.className, {
interactive: true,
active: this.isActive(entity),
disabled: !entity
});
const onOpen = async () => {
await entity.onContextMenuOpen(this.contextMenu);
};
const menuItems = this.contextMenu?.menuItems.filter((menuItem) => !menuItem.onlyVisibleForSource || menuItem.onlyVisibleForSource === entity.metadata.source);
return (
<HotbarIcon
uid={entity.getId()}
title={entity.getName()}
source={`${entity.metadata.source || "local"}`}
className={className}
active={this.isActive(entity)}
remove={remove}
onMenuOpen={onOpen}
menuItems={menuItems}
{...elemProps}
>
{ this.ledIcon }
{ this.kindIcon }
</HotbarIcon>
);
}
}

View File

@ -41,6 +41,18 @@
transition: all 0s 0.8s; transition: all 0s 0.8s;
} }
&.disabled {
opacity: 0.4;
cursor: default;
filter: grayscale(0.7);
&:hover {
&:not(.active) {
box-shadow: none;
}
}
}
&.active, &.interactive:hover { &.active, &.interactive:hover {
img { img {
opacity: 1; opacity: 1;

View File

@ -21,28 +21,51 @@
import "./hotbar-icon.scss"; import "./hotbar-icon.scss";
import React, { DOMAttributes } from "react"; import React, { DOMAttributes, useState } from "react";
import { observer } from "mobx-react";
import { cssNames, IClassName, iter } from "../../utils";
import { Tooltip } from "../tooltip";
import { Avatar } from "@material-ui/core"; import { Avatar } from "@material-ui/core";
import { CatalogEntity, CatalogEntityContextMenu, CatalogEntityContextMenuContext } from "../../../common/catalog";
import { Menu, MenuItem } from "../menu";
import { Icon } from "../icon";
import { computed, observable } from "mobx";
import { navigate } from "../../navigation";
import { HotbarStore } from "../../../common/hotbar-store";
import { ConfirmDialog } from "../confirm-dialog";
import randomColor from "randomcolor"; import randomColor from "randomcolor";
import { catalogCategoryRegistry } from "../../api/catalog-category-registry";
import GraphemeSplitter from "grapheme-splitter"; import GraphemeSplitter from "grapheme-splitter";
import { CatalogEntityContextMenu } from "../../../common/catalog";
import { cssNames, IClassName, iter } from "../../utils";
import { ConfirmDialog } from "../confirm-dialog";
import { Icon } from "../icon";
import { Menu, MenuItem } from "../menu";
import { MaterialTooltip } from "../+catalog/material-tooltip/material-tooltip";
interface Props extends DOMAttributes<HTMLElement> { interface Props extends DOMAttributes<HTMLElement> {
entity: CatalogEntity; uid: string;
index: number; title: string;
source: string;
remove: (uid: string) => void;
onMenuOpen?: () => void;
className?: IClassName; className?: IClassName;
errorClass?: IClassName; active?: boolean;
isActive?: boolean; menuItems?: CatalogEntityContextMenu[];
disabled?: boolean;
}
function generateAvatarStyle(seed: string): React.CSSProperties {
return {
"backgroundColor": randomColor({ seed, luminosity: "dark" })
};
}
function onMenuItemClick(menuItem: CatalogEntityContextMenu) {
if (menuItem.confirm) {
ConfirmDialog.open({
okButtonProps: {
primary: false,
accent: true,
},
ok: () => {
menuItem.onClick();
},
message: menuItem.confirm.message
});
} else {
menuItem.onClick();
}
} }
function getNameParts(name: string): string[] { function getNameParts(name: string): string[] {
@ -61,20 +84,17 @@ function getNameParts(name: string): string[] {
return name.split(/@+/); return name.split(/@+/);
} }
@observer export function HotbarIcon(props: Props) {
export class HotbarIcon extends React.Component<Props> { const { uid, title, className, source, active, remove, disabled, menuItems, onMenuOpen, children, ...rest } = props;
@observable.deep private contextMenu: CatalogEntityContextMenuContext; const id = `hotbarIcon-${uid}`;
@observable menuOpen = false; const [menuOpen, setMenuOpen] = useState(false);
componentDidMount() { const toggleMenu = () => {
this.contextMenu = { setMenuOpen(!menuOpen);
menuItems: [],
navigate: (url: string) => navigate(url)
}; };
}
@computed get iconString() { const getIconString = () => {
const [rawFirst, rawSecond, rawThird] = getNameParts(this.props.entity.metadata.name); const [rawFirst, rawSecond, rawThird] = getNameParts(title);
const splitter = new GraphemeSplitter(); const splitter = new GraphemeSplitter();
const first = splitter.iterateGraphemes(rawFirst); const first = splitter.iterateGraphemes(rawFirst);
const second = rawSecond ? splitter.iterateGraphemes(rawSecond): first; const second = rawSecond ? splitter.iterateGraphemes(rawSecond): first;
@ -85,114 +105,53 @@ export class HotbarIcon extends React.Component<Props> {
...iter.take(second, 1), ...iter.take(second, 1),
...iter.take(third, 1), ...iter.take(third, 1),
].filter(Boolean).join(""); ].filter(Boolean).join("");
}
get kindIcon() {
const className = "badge";
const category = catalogCategoryRegistry.getCategoryForEntity(this.props.entity);
if (!category) {
return <Icon material="bug_report" className={className} />;
}
if (category.metadata.icon.includes("<svg")) {
return <Icon svg={category.metadata.icon} className={className} />;
} else {
return <Icon material={category.metadata.icon} className={className} />;
}
}
get ledIcon() {
const className = cssNames("led", { online: this.props.entity.status.phase == "connected"}); // TODO: make it more generic
return <div className={className} />;
}
toggleMenu() {
this.menuOpen = !this.menuOpen;
}
remove(item: CatalogEntity) {
const hotbar = HotbarStore.getInstance();
hotbar.removeFromHotbar(item.getId());
}
onMenuItemClick(menuItem: CatalogEntityContextMenu) {
if (menuItem.confirm) {
ConfirmDialog.open({
okButtonProps: {
primary: false,
accent: true,
},
ok: () => {
menuItem.onClick();
},
message: menuItem.confirm.message
});
} else {
menuItem.onClick();
}
}
generateAvatarStyle(entity: CatalogEntity): React.CSSProperties {
return {
"backgroundColor": randomColor({ seed: `${entity.metadata.name}-${entity.metadata.source}`, luminosity: "dark" })
}; };
}
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 ( return (
<div className={className}> <div className={cssNames("HotbarIcon flex inline", className, { disabled })}>
<Tooltip targetId={entityIconId}>{entity.metadata.name} ({entity.metadata.source || "local"})</Tooltip> <MaterialTooltip title={`${title} (${source})`} placement="right">
<div id={id}>
<Avatar <Avatar
{...elemProps} {...rest}
id={entityIconId}
variant="square" variant="square"
className={isActive ? "active" : "default"} className={active ? "active" : "default"}
style={this.generateAvatarStyle(entity)} style={generateAvatarStyle(`${title}-${source}`)}
> >
{this.iconString} {getIconString()}
</Avatar> </Avatar>
{ this.ledIcon } {children}
{ this.kindIcon } </div>
</MaterialTooltip>
<Menu <Menu
usePortal usePortal
htmlFor={entityIconId} htmlFor={id}
className="HotbarIconMenu" className="HotbarIconMenu"
isOpen={this.menuOpen} isOpen={menuOpen}
toggleEvent="contextmenu" toggleEvent="contextmenu"
position={{right: true, bottom: true }} // FIXME: position does not work position={{right: true, bottom: true }} // FIXME: position does not work
open={() => onOpen()} open={() => {
close={() => this.toggleMenu()}> onMenuOpen?.();
<MenuItem key="remove-from-hotbar" onClick={() => this.remove(entity) }> toggleMenu();
}}
close={() => toggleMenu()}>
<MenuItem key="remove-from-hotbar" onClick={(evt) => {
evt.stopPropagation();
remove(uid);
}}>
<Icon material="clear" small interactive={true} title="Remove from hotbar"/> Remove from Hotbar <Icon material="clear" small interactive={true} title="Remove from hotbar"/> Remove from Hotbar
</MenuItem> </MenuItem>
{ this.contextMenu && menuItems.map((menuItem) => { { menuItems.map((menuItem) => {
return ( return (
<MenuItem key={menuItem.title} onClick={() => this.onMenuItemClick(menuItem) }> <MenuItem key={menuItem.title} onClick={() => onMenuItemClick(menuItem) }>
<Icon material={menuItem.icon} small interactive={true} title={menuItem.title}/> {menuItem.title} <Icon material={menuItem.icon} small interactive={true} title={menuItem.title}/> {menuItem.title}
</MenuItem> </MenuItem>
); );
})} })}
</Menu> </Menu>
{children}
</div> </div>
); );
}
} }
HotbarIcon.defaultProps = {
menuItems: []
};

View File

@ -22,15 +22,17 @@
import "./hotbar-menu.scss"; import "./hotbar-menu.scss";
import "./hotbar.commands"; import "./hotbar.commands";
import React, { HTMLAttributes, ReactNode, useState } from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { HotbarIcon } from "./hotbar-icon"; import { HotbarEntityIcon } from "./hotbar-entity-icon";
import { cssNames, IClassName } from "../../utils"; import { cssNames, IClassName } from "../../utils";
import { catalogEntityRegistry } from "../../api/catalog-entity-registry"; import { catalogEntityRegistry } from "../../api/catalog-entity-registry";
import { defaultHotbarCells, HotbarItem, HotbarStore } from "../../../common/hotbar-store"; import { defaultHotbarCells, HotbarItem, HotbarStore } from "../../../common/hotbar-store";
import { CatalogEntity, catalogEntityRunContext } from "../../api/catalog-entity"; import { catalogEntityRunContext } from "../../api/catalog-entity";
import { DragDropContext, Draggable, Droppable, DropResult } from "react-beautiful-dnd"; import { DragDropContext, Draggable, Droppable, DropResult } from "react-beautiful-dnd";
import { HotbarSelector } from "./hotbar-selector"; import { HotbarSelector } from "./hotbar-selector";
import { HotbarCell } from "./hotbar-cell";
import { HotbarIcon } from "./hotbar-icon";
interface Props { interface Props {
className?: IClassName; className?: IClassName;
@ -42,10 +44,6 @@ export class HotbarMenu extends React.Component<Props> {
return HotbarStore.getInstance().getActive(); return HotbarStore.getInstance().getActive();
} }
isActive(item: CatalogEntity) {
return catalogEntityRegistry.activeEntity?.metadata?.uid == item.getId();
}
getEntity(item: HotbarItem) { getEntity(item: HotbarItem) {
const hotbar = HotbarStore.getInstance().getActive(); const hotbar = HotbarStore.getInstance().getActive();
@ -69,6 +67,12 @@ export class HotbarMenu extends React.Component<Props> {
HotbarStore.getInstance().restackItems(from, to); HotbarStore.getInstance().restackItems(from, to);
} }
removeItem(uid: string) {
const hotbar = HotbarStore.getInstance();
hotbar.removeFromHotbar(uid);
}
getMoveAwayDirection(entityId: string, cellIndex: number) { getMoveAwayDirection(entityId: string, cellIndex: number) {
const draggableItemIndex = this.hotbar.items.findIndex(item => item?.entity.uid == entityId); const draggableItemIndex = this.hotbar.items.findIndex(item => item?.entity.uid == entityId);
@ -78,7 +82,6 @@ export class HotbarMenu extends React.Component<Props> {
renderGrid() { renderGrid() {
return this.hotbar.items.map((item, index) => { return this.hotbar.items.map((item, index) => {
const entity = this.getEntity(item); const entity = this.getEntity(item);
const isActive = !entity ? false : this.isActive(entity);
return ( return (
<Droppable droppableId={`${index}`} key={index}> <Droppable droppableId={`${index}`} key={index}>
@ -93,7 +96,7 @@ export class HotbarMenu extends React.Component<Props> {
}, this.getMoveAwayDirection(snapshot.draggingOverWith, index))} }, this.getMoveAwayDirection(snapshot.draggingOverWith, index))}
{...provided.droppableProps} {...provided.droppableProps}
> >
{entity && ( {item && (
<Draggable draggableId={item.entity.uid} key={item.entity.uid} index={0} > <Draggable draggableId={item.entity.uid} key={item.entity.uid} index={0} >
{(provided, snapshot) => { {(provided, snapshot) => {
const style = { const style = {
@ -110,14 +113,23 @@ export class HotbarMenu extends React.Component<Props> {
{...provided.dragHandleProps} {...provided.dragHandleProps}
style={style} style={style}
> >
<HotbarIcon {entity ? (
<HotbarEntityIcon
key={index} key={index}
index={index}
entity={entity} entity={entity}
isActive={isActive}
onClick={() => entity.onRun(catalogEntityRunContext)} onClick={() => entity.onRun(catalogEntityRunContext)}
className={cssNames({ isDragging: snapshot.isDragging })} className={cssNames({ isDragging: snapshot.isDragging })}
remove={this.removeItem}
/> />
) : (
<HotbarIcon
uid={item.entity.uid}
title={item.entity.name}
source={item.entity.source}
remove={this.removeItem}
disabled
/>
)}
</div> </div>
); );
}} }}
@ -148,33 +160,3 @@ export class HotbarMenu extends React.Component<Props> {
); );
} }
} }
interface HotbarCellProps extends HTMLAttributes<HTMLDivElement> {
children?: ReactNode;
index: number;
innerRef?: React.LegacyRef<HTMLDivElement>;
}
function HotbarCell({ innerRef, children, className, ...rest }: HotbarCellProps) {
const [animating, setAnimating] = useState(false);
const onAnimationEnd = () => { setAnimating(false); };
const onClick = () => {
if (className.includes("isDraggingOver")) {
return;
}
setAnimating(true);
};
return (
<div
className={cssNames("HotbarCell", { animating }, className)}
onAnimationEnd={onAnimationEnd}
onClick={onClick}
ref={innerRef}
{...rest}
>
{children}
</div>
);
}

View File

@ -23,43 +23,31 @@ import "./hotbar-selector.scss";
import React from "react"; import React from "react";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { Badge } from "../badge"; import { Badge } from "../badge";
import { makeStyles, Tooltip } from "@material-ui/core";
import { Hotbar, HotbarStore } from "../../../common/hotbar-store"; import { Hotbar, HotbarStore } from "../../../common/hotbar-store";
import { CommandOverlay } from "../command-palette"; import { CommandOverlay } from "../command-palette";
import { HotbarSwitchCommand } from "./hotbar-switch-command"; import { HotbarSwitchCommand } from "./hotbar-switch-command";
import { MaterialTooltip } from "../+catalog/material-tooltip/material-tooltip";
interface Props { interface Props {
hotbar: Hotbar; hotbar: Hotbar;
} }
const useStyles = makeStyles(() => ({
arrow: {
color: "#222",
},
tooltip: {
fontSize: 12,
backgroundColor: "#222",
},
}));
export function HotbarSelector({ hotbar }: Props) { export function HotbarSelector({ hotbar }: Props) {
const store = HotbarStore.getInstance(); const store = HotbarStore.getInstance();
const activeIndexDisplay = store.activeHotbarIndex + 1; const activeIndexDisplay = store.activeHotbarIndex + 1;
const classes = useStyles();
return ( return (
<div className="HotbarSelector flex align-center"> <div className="HotbarSelector flex align-center">
<Icon material="play_arrow" className="previous box" onClick={() => store.switchToPrevious()} /> <Icon material="play_arrow" className="previous box" onClick={() => store.switchToPrevious()} />
<div className="box grow flex align-center"> <div className="box grow flex align-center">
<Tooltip arrow title={hotbar.name} classes={classes}> <MaterialTooltip arrow title={hotbar.name}>
<Badge <Badge
id="hotbarIndex" id="hotbarIndex"
small small
label={activeIndexDisplay} label={activeIndexDisplay}
onClick={() => CommandOverlay.open(<HotbarSwitchCommand />)} onClick={() => CommandOverlay.open(<HotbarSwitchCommand />)}
/> />
</Tooltip> </MaterialTooltip>
</div> </div>
<Icon material="play_arrow" className="next box" onClick={() => store.switchToNext()} /> <Icon material="play_arrow" className="next box" onClick={() => store.switchToNext()} />
</div> </div>

View File

@ -128,6 +128,7 @@
"settingsColor": "#909ba6", "settingsColor": "#909ba6",
"navSelectedBackground": "#262b2e", "navSelectedBackground": "#262b2e",
"navHoverColor": "#dcddde", "navHoverColor": "#dcddde",
"hrColor": "#ffffff0f" "hrColor": "#ffffff0f",
"tooltipBackground": "#18191c"
} }
} }

View File

@ -130,6 +130,7 @@
"settingsColor": "#555555", "settingsColor": "#555555",
"navSelectedBackground": "#ffffff", "navSelectedBackground": "#ffffff",
"navHoverColor": "#2e3135", "navHoverColor": "#2e3135",
"hrColor": "#06060714" "hrColor": "#06060714",
"tooltipBackground": "#ffffff"
} }
} }