mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Hotbar inner drag-n-drop (#2691)
* Configure ts to use react-jsx rule Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Moving HotbarSelector to separate component Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Initial drag-n-drop implementation Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Revert tsconfig and linter changes Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Reverting back active cell effect Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Adding drag-n-drop behavior Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Fix drag-n-drop logic Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * White border on dragging over Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Adding test coverage Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Fixing cell hover effect Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Increase PageLayout z-index Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com> * Styling hotbar selector tooltip Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>
This commit is contained in:
parent
83e63bf959
commit
1af12fe59e
@ -1,7 +1,71 @@
|
|||||||
import mockFs from "mock-fs";
|
import mockFs from "mock-fs";
|
||||||
|
import { CatalogEntityItem } from "../../renderer/components/+catalog/catalog-entity.store";
|
||||||
import { ClusterStore } from "../cluster-store";
|
import { ClusterStore } from "../cluster-store";
|
||||||
import { HotbarStore } from "../hotbar-store";
|
import { HotbarStore } from "../hotbar-store";
|
||||||
|
|
||||||
|
const testCluster = {
|
||||||
|
uid: "test",
|
||||||
|
name: "test",
|
||||||
|
apiVersion: "v1",
|
||||||
|
kind: "Cluster",
|
||||||
|
status: {
|
||||||
|
phase: "Running"
|
||||||
|
},
|
||||||
|
spec: {},
|
||||||
|
getName: jest.fn(),
|
||||||
|
getId: jest.fn(),
|
||||||
|
onDetailsOpen: jest.fn(),
|
||||||
|
onContextMenuOpen: jest.fn(),
|
||||||
|
onSettingsOpen: jest.fn(),
|
||||||
|
metadata: {
|
||||||
|
uid: "test",
|
||||||
|
name: "test",
|
||||||
|
labels: {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const minikubeCluster = {
|
||||||
|
uid: "minikube",
|
||||||
|
name: "minikube",
|
||||||
|
apiVersion: "v1",
|
||||||
|
kind: "Cluster",
|
||||||
|
status: {
|
||||||
|
phase: "Running"
|
||||||
|
},
|
||||||
|
spec: {},
|
||||||
|
getName: jest.fn(),
|
||||||
|
getId: jest.fn(),
|
||||||
|
onDetailsOpen: jest.fn(),
|
||||||
|
onContextMenuOpen: jest.fn(),
|
||||||
|
onSettingsOpen: jest.fn(),
|
||||||
|
metadata: {
|
||||||
|
uid: "minikube",
|
||||||
|
name: "minikube",
|
||||||
|
labels: {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const awsCluster = {
|
||||||
|
uid: "aws",
|
||||||
|
name: "aws",
|
||||||
|
apiVersion: "v1",
|
||||||
|
kind: "Cluster",
|
||||||
|
status: {
|
||||||
|
phase: "Running"
|
||||||
|
},
|
||||||
|
spec: {},
|
||||||
|
getName: jest.fn(),
|
||||||
|
getId: jest.fn(),
|
||||||
|
onDetailsOpen: jest.fn(),
|
||||||
|
onContextMenuOpen: jest.fn(),
|
||||||
|
onSettingsOpen: jest.fn(),
|
||||||
|
metadata: {
|
||||||
|
uid: "aws",
|
||||||
|
name: "aws",
|
||||||
|
labels: {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
describe("HotbarStore", () => {
|
describe("HotbarStore", () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
ClusterStore.resetInstance();
|
ClusterStore.resetInstance();
|
||||||
@ -31,4 +95,137 @@ describe("HotbarStore", () => {
|
|||||||
expect(hotbarStore.hotbars.length).toEqual(2);
|
expect(hotbarStore.hotbars.length).toEqual(2);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("hotbar items", () => {
|
||||||
|
it("initially creates 12 empty cells", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
expect(hotbarStore.getActive().items.length).toEqual(12);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("adds items", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const entity = new CatalogEntityItem(testCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(entity);
|
||||||
|
const items = hotbarStore.getActive().items.filter(Boolean);
|
||||||
|
|
||||||
|
expect(items.length).toEqual(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("removes items", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const entity = new CatalogEntityItem(testCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(entity);
|
||||||
|
hotbarStore.removeFromHotbar("test");
|
||||||
|
const items = hotbarStore.getActive().items.filter(Boolean);
|
||||||
|
|
||||||
|
expect(items.length).toEqual(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does nothing if removing with invalid uid", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const entity = new CatalogEntityItem(testCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(entity);
|
||||||
|
hotbarStore.removeFromHotbar("invalid uid");
|
||||||
|
const items = hotbarStore.getActive().items.filter(Boolean);
|
||||||
|
|
||||||
|
expect(items.length).toEqual(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("moves item to empty cell", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const test = new CatalogEntityItem(testCluster);
|
||||||
|
const minikube = new CatalogEntityItem(minikubeCluster);
|
||||||
|
const aws = new CatalogEntityItem(awsCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(test);
|
||||||
|
hotbarStore.addToHotbar(minikube);
|
||||||
|
hotbarStore.addToHotbar(aws);
|
||||||
|
|
||||||
|
expect(hotbarStore.getActive().items[5]).toBeNull();
|
||||||
|
|
||||||
|
hotbarStore.restackItems(1, 5);
|
||||||
|
|
||||||
|
expect(hotbarStore.getActive().items[5]).toBeTruthy();
|
||||||
|
expect(hotbarStore.getActive().items[5].entity.uid).toEqual("minikube");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("moves items down", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const test = new CatalogEntityItem(testCluster);
|
||||||
|
const minikube = new CatalogEntityItem(minikubeCluster);
|
||||||
|
const aws = new CatalogEntityItem(awsCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(test);
|
||||||
|
hotbarStore.addToHotbar(minikube);
|
||||||
|
hotbarStore.addToHotbar(aws);
|
||||||
|
|
||||||
|
// aws -> test
|
||||||
|
hotbarStore.restackItems(2, 0);
|
||||||
|
|
||||||
|
const items = hotbarStore.getActive().items.map(item => item?.entity.uid || null);
|
||||||
|
|
||||||
|
expect(items.slice(0, 4)).toEqual(["aws", "test", "minikube", null]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("moves items up", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const test = new CatalogEntityItem(testCluster);
|
||||||
|
const minikube = new CatalogEntityItem(minikubeCluster);
|
||||||
|
const aws = new CatalogEntityItem(awsCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(test);
|
||||||
|
hotbarStore.addToHotbar(minikube);
|
||||||
|
hotbarStore.addToHotbar(aws);
|
||||||
|
|
||||||
|
// test -> aws
|
||||||
|
hotbarStore.restackItems(0, 2);
|
||||||
|
|
||||||
|
const items = hotbarStore.getActive().items.map(item => item?.entity.uid || null);
|
||||||
|
|
||||||
|
expect(items.slice(0, 4)).toEqual(["minikube", "aws", "test", null]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("does nothing when item moved to same cell", () => {
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const test = new CatalogEntityItem(testCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(test);
|
||||||
|
hotbarStore.restackItems(0, 0);
|
||||||
|
|
||||||
|
expect(hotbarStore.getActive().items[0].entity.uid).toEqual("test");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("throws if invalid arguments provided", () => {
|
||||||
|
// Prevent writing to stderr during this render.
|
||||||
|
const err = console.error;
|
||||||
|
|
||||||
|
console.error = jest.fn();
|
||||||
|
|
||||||
|
const hotbarStore = HotbarStore.createInstance();
|
||||||
|
const test = new CatalogEntityItem(testCluster);
|
||||||
|
|
||||||
|
hotbarStore.load();
|
||||||
|
hotbarStore.addToHotbar(test);
|
||||||
|
|
||||||
|
expect(() => hotbarStore.restackItems(-5, 0)).toThrow();
|
||||||
|
expect(() => hotbarStore.restackItems(2, -1)).toThrow();
|
||||||
|
expect(() => hotbarStore.restackItems(14, 1)).toThrow();
|
||||||
|
expect(() => hotbarStore.restackItems(11, 112)).toThrow();
|
||||||
|
|
||||||
|
// Restore writing to stderr.
|
||||||
|
console.error = err;
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@ -4,7 +4,6 @@ import migrations from "../migrations/hotbar-store";
|
|||||||
import * as uuid from "uuid";
|
import * as uuid from "uuid";
|
||||||
import { CatalogEntityItem } from "../renderer/components/+catalog/catalog-entity.store";
|
import { CatalogEntityItem } from "../renderer/components/+catalog/catalog-entity.store";
|
||||||
import isNull from "lodash/isNull";
|
import isNull from "lodash/isNull";
|
||||||
import { CatalogEntity } from "./catalog/catalog-entity";
|
|
||||||
|
|
||||||
export interface HotbarItem {
|
export interface HotbarItem {
|
||||||
entity: {
|
entity: {
|
||||||
@ -147,9 +146,9 @@ export class HotbarStore extends BaseStore<HotbarStoreModel> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
removeFromHotbar(item: CatalogEntity) {
|
removeFromHotbar(uid: string) {
|
||||||
const hotbar = this.getActive();
|
const hotbar = this.getActive();
|
||||||
const index = hotbar.items.findIndex((i) => i?.entity.uid === item.getId());
|
const index = hotbar.items.findIndex((i) => i?.entity.uid === uid);
|
||||||
|
|
||||||
if (index == -1) {
|
if (index == -1) {
|
||||||
return;
|
return;
|
||||||
@ -158,6 +157,40 @@ export class HotbarStore extends BaseStore<HotbarStoreModel> {
|
|||||||
hotbar.items[index] = null;
|
hotbar.items[index] = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
findClosestEmptyIndex(from: number, direction = 1) {
|
||||||
|
let index = from;
|
||||||
|
|
||||||
|
while(this.getActive().items[index] != null) {
|
||||||
|
index += direction;
|
||||||
|
}
|
||||||
|
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
restackItems(from: number, to: number): void {
|
||||||
|
const { items } = this.getActive();
|
||||||
|
const source = items[from];
|
||||||
|
const moveDown = from < to;
|
||||||
|
|
||||||
|
if (from < 0 || to < 0 || from >= items.length || to >= items.length || isNaN(from) || isNaN(to)) {
|
||||||
|
throw new Error("Invalid 'from' or 'to' arguments");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (from == to) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
items.splice(from, 1, null);
|
||||||
|
|
||||||
|
if (items[to] == null) {
|
||||||
|
items.splice(to, 1, source);
|
||||||
|
} else {
|
||||||
|
// Move cells up or down to closes empty cell
|
||||||
|
items.splice(this.findClosestEmptyIndex(to, moveDown ? -1 : 1), 1);
|
||||||
|
items.splice(to, 0, source);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
switchToPrevious() {
|
switchToPrevious() {
|
||||||
const hotbarStore = HotbarStore.getInstance();
|
const hotbarStore = HotbarStore.getInstance();
|
||||||
let index = hotbarStore.activeHotbarIndex - 1;
|
let index = hotbarStore.activeHotbarIndex - 1;
|
||||||
|
|||||||
@ -14,7 +14,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.active {
|
&.active {
|
||||||
box-shadow: 0 0 0px 3px #ffffff;
|
box-shadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px var(--textColorAccent);
|
||||||
transition: all 0s 0.8s;
|
transition: all 0s 0.8s;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -24,6 +24,14 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
box-shadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px #ffffff30;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.isDragging {
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
> .led {
|
> .led {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 3px;
|
left: 3px;
|
||||||
|
|||||||
@ -94,7 +94,7 @@ export class HotbarIcon extends React.Component<Props> {
|
|||||||
remove(item: CatalogEntity) {
|
remove(item: CatalogEntity) {
|
||||||
const hotbar = HotbarStore.getInstance();
|
const hotbar = HotbarStore.getInstance();
|
||||||
|
|
||||||
hotbar.removeFromHotbar(item);
|
hotbar.removeFromHotbar(item.getId());
|
||||||
}
|
}
|
||||||
|
|
||||||
onMenuItemClick(menuItem: CatalogEntityContextMenu) {
|
onMenuItemClick(menuItem: CatalogEntityContextMenu) {
|
||||||
|
|||||||
@ -44,15 +44,14 @@
|
|||||||
background: var(--layoutBackground);
|
background: var(--layoutBackground);
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
position: relative;
|
position: relative;
|
||||||
transform: translateZ(0); // Remove flickering artifacts
|
|
||||||
|
|
||||||
&:hover {
|
&.isDraggingOver {
|
||||||
&:not(:empty) {
|
box-shadow: 0 0 0px 3px $clusterMenuBackground, 0 0 0px 6px #fff;
|
||||||
box-shadow: 0 0 0px 3px #ffffff1a;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
&.animating {
|
&.animating {
|
||||||
|
transform: translateZ(0); // Remove flickering artifacts
|
||||||
|
|
||||||
&:empty {
|
&:empty {
|
||||||
animation: shake .6s cubic-bezier(.36,.07,.19,.97) both;
|
animation: shake .6s cubic-bezier(.36,.07,.19,.97) both;
|
||||||
transform: translate3d(0, 0, 0);
|
transform: translate3d(0, 0, 0);
|
||||||
@ -61,48 +60,11 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
&:not(:empty) {
|
&:not(:empty) {
|
||||||
animation: outline 0.8s cubic-bezier(0.19, 1, 0.22, 1);
|
animation: outline 1s cubic-bezier(0.19, 1, 0.22, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.HotbarSelector {
|
|
||||||
height: 26px;
|
|
||||||
background-color: var(--layoutBackground);
|
|
||||||
position: relative;
|
|
||||||
|
|
||||||
&:before {
|
|
||||||
content: " ";
|
|
||||||
position: absolute;
|
|
||||||
width: 100%;
|
|
||||||
height: 20px;
|
|
||||||
background: linear-gradient(0deg, var(--clusterMenuBackground), transparent);
|
|
||||||
top: -20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.Badge {
|
|
||||||
cursor: pointer;
|
|
||||||
background: var(--secondaryBackground);
|
|
||||||
width: 100%;
|
|
||||||
color: var(--settingsColor);
|
|
||||||
padding-top: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.Icon {
|
|
||||||
--size: 16px;
|
|
||||||
padding: 0 4px;
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
box-shadow: none;
|
|
||||||
background-color: transparent;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.previous {
|
|
||||||
transform: rotateY(180deg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@keyframes shake {
|
@keyframes shake {
|
||||||
@ -130,6 +92,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
100% {
|
100% {
|
||||||
box-shadow: 0 0 0px 0px $clusterMenuBackground, 0 0 0px 3px #ffffff;
|
box-shadow: 0 0 0px 3px $clusterMenuBackground, 0 0 0px 6px #ffffff;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,18 +1,15 @@
|
|||||||
import "./hotbar-menu.scss";
|
import "./hotbar-menu.scss";
|
||||||
import "./hotbar.commands";
|
import "./hotbar.commands";
|
||||||
|
|
||||||
import React, { ReactNode, useState } from "react";
|
import React, { HTMLAttributes, ReactNode, useState } from "react";
|
||||||
import { observer } from "mobx-react";
|
import { observer } from "mobx-react";
|
||||||
import { HotbarIcon } from "./hotbar-icon";
|
import { HotbarIcon } from "./hotbar-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 { HotbarItem, HotbarStore } from "../../../common/hotbar-store";
|
import { defaultHotbarCells, HotbarItem, HotbarStore } from "../../../common/hotbar-store";
|
||||||
import { CatalogEntity, catalogEntityRunContext } from "../../api/catalog-entity";
|
import { CatalogEntity, catalogEntityRunContext } from "../../api/catalog-entity";
|
||||||
import { Icon } from "../icon";
|
import { DragDropContext, Draggable, Droppable, DropResult } from "react-beautiful-dnd";
|
||||||
import { Badge } from "../badge";
|
import { HotbarSelector } from "./hotbar-selector";
|
||||||
import { CommandOverlay } from "../command-palette";
|
|
||||||
import { HotbarSwitchCommand } from "./hotbar-switch-command";
|
|
||||||
import { Tooltip, TooltipPosition } from "../tooltip";
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
className?: IClassName;
|
className?: IClassName;
|
||||||
@ -38,36 +35,67 @@ export class HotbarMenu extends React.Component<Props> {
|
|||||||
return item ? catalogEntityRegistry.items.find((entity) => entity.metadata.uid === item.entity.uid) : null;
|
return item ? catalogEntityRegistry.items.find((entity) => entity.metadata.uid === item.entity.uid) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
previous() {
|
onDragEnd(result: DropResult) {
|
||||||
HotbarStore.getInstance().switchToPrevious();
|
const { source, destination } = result;
|
||||||
}
|
|
||||||
|
|
||||||
next() {
|
if (!destination) { // Dropped outside of the list
|
||||||
HotbarStore.getInstance().switchToNext();
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
openSelector() {
|
const from = parseInt(source.droppableId);
|
||||||
CommandOverlay.open(<HotbarSwitchCommand />);
|
const to = parseInt(destination.droppableId);
|
||||||
|
|
||||||
|
HotbarStore.getInstance().restackItems(from, to);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderGrid() {
|
renderGrid() {
|
||||||
if (!this.hotbar.items.length) return;
|
|
||||||
|
|
||||||
return this.hotbar.items.map((item, index) => {
|
return this.hotbar.items.map((item, index) => {
|
||||||
const entity = this.getEntity(item);
|
const entity = this.getEntity(item);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HotbarCell key={index} index={index}>
|
<Droppable droppableId={`${index}`} key={index}>
|
||||||
{entity && (
|
{(provided, snapshot) => (
|
||||||
<HotbarIcon
|
<HotbarCell
|
||||||
key={index}
|
|
||||||
index={index}
|
index={index}
|
||||||
entity={entity}
|
key={entity ? entity.getId() : `cell${index}`}
|
||||||
isActive={this.isActive(entity)}
|
innerRef={provided.innerRef}
|
||||||
onClick={() => entity.onRun(catalogEntityRunContext)}
|
className={cssNames({ isDraggingOver: snapshot.isDraggingOver })}
|
||||||
/>
|
{...provided.droppableProps}
|
||||||
|
>
|
||||||
|
{entity && (
|
||||||
|
<Draggable draggableId={item.entity.uid} key={item.entity.uid} index={0}>
|
||||||
|
{(provided, snapshot) => {
|
||||||
|
const style = {
|
||||||
|
zIndex: defaultHotbarCells - index,
|
||||||
|
position: "absolute",
|
||||||
|
...provided.draggableProps.style,
|
||||||
|
} as React.CSSProperties;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={item.entity.uid}
|
||||||
|
ref={provided.innerRef}
|
||||||
|
{...provided.draggableProps}
|
||||||
|
{...provided.dragHandleProps}
|
||||||
|
style={style}
|
||||||
|
>
|
||||||
|
<HotbarIcon
|
||||||
|
key={index}
|
||||||
|
index={index}
|
||||||
|
entity={entity}
|
||||||
|
isActive={this.isActive(entity)}
|
||||||
|
onClick={() => entity.onRun(catalogEntityRunContext)}
|
||||||
|
className={cssNames({ isDragging: snapshot.isDragging })}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</Draggable>
|
||||||
|
)}
|
||||||
|
{provided.placeholder}
|
||||||
|
</HotbarCell>
|
||||||
)}
|
)}
|
||||||
</HotbarCell>
|
</Droppable>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -76,48 +104,46 @@ export class HotbarMenu extends React.Component<Props> {
|
|||||||
const { className } = this.props;
|
const { className } = this.props;
|
||||||
const hotbarStore = HotbarStore.getInstance();
|
const hotbarStore = HotbarStore.getInstance();
|
||||||
const hotbar = hotbarStore.getActive();
|
const hotbar = hotbarStore.getActive();
|
||||||
const activeIndexDisplay = hotbarStore.activeHotbarIndex + 1;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={cssNames("HotbarMenu flex column", className)}>
|
<div className={cssNames("HotbarMenu flex column", className)}>
|
||||||
<div className="HotbarItems flex column gaps">
|
<div className="HotbarItems flex column gaps">
|
||||||
{this.renderGrid()}
|
<DragDropContext onDragEnd={this.onDragEnd}>
|
||||||
</div>
|
{this.renderGrid()}
|
||||||
<div className="HotbarSelector flex align-center">
|
</DragDropContext>
|
||||||
<Icon material="play_arrow" className="previous box" onClick={() => this.previous()} />
|
|
||||||
<div className="box grow flex align-center">
|
|
||||||
<Badge id="hotbarIndex" small label={activeIndexDisplay} onClick={() => this.openSelector()} />
|
|
||||||
<Tooltip
|
|
||||||
targetId="hotbarIndex"
|
|
||||||
preferredPositions={TooltipPosition.TOP}
|
|
||||||
>
|
|
||||||
{hotbar.name}
|
|
||||||
</Tooltip>
|
|
||||||
</div>
|
|
||||||
<Icon material="play_arrow" className="next box" onClick={() => this.next()} />
|
|
||||||
</div>
|
</div>
|
||||||
|
<HotbarSelector hotbar={hotbar}/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface HotbarCellProps {
|
interface HotbarCellProps extends HTMLAttributes<HTMLDivElement> {
|
||||||
children?: ReactNode;
|
children?: ReactNode;
|
||||||
index: number;
|
index: number;
|
||||||
|
innerRef?: React.LegacyRef<HTMLDivElement>;
|
||||||
}
|
}
|
||||||
|
|
||||||
function HotbarCell(props: HotbarCellProps) {
|
function HotbarCell({ innerRef, children, className, ...rest }: HotbarCellProps) {
|
||||||
const [animating, setAnimating] = useState(false);
|
const [animating, setAnimating] = useState(false);
|
||||||
const onAnimationEnd = () => { setAnimating(false); };
|
const onAnimationEnd = () => { setAnimating(false); };
|
||||||
const onClick = () => { setAnimating(true); };
|
const onClick = () => {
|
||||||
|
if (className.includes("isDraggingOver")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setAnimating(true);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={cssNames("HotbarCell", { animating })}
|
className={cssNames("HotbarCell", { animating }, className)}
|
||||||
onAnimationEnd={onAnimationEnd}
|
onAnimationEnd={onAnimationEnd}
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
|
ref={innerRef}
|
||||||
|
{...rest}
|
||||||
>
|
>
|
||||||
{props.children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
36
src/renderer/components/hotbar/hotbar-selector.scss
Normal file
36
src/renderer/components/hotbar/hotbar-selector.scss
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
.HotbarSelector {
|
||||||
|
height: 26px;
|
||||||
|
background-color: var(--layoutBackground);
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
&:before {
|
||||||
|
content: " ";
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 20px;
|
||||||
|
background: linear-gradient(0deg, var(--clusterMenuBackground), transparent);
|
||||||
|
top: -20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Badge {
|
||||||
|
cursor: pointer;
|
||||||
|
background: var(--secondaryBackground);
|
||||||
|
width: 100%;
|
||||||
|
color: var(--settingsColor);
|
||||||
|
padding-top: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Icon {
|
||||||
|
--size: 16px;
|
||||||
|
padding: 0 4px;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
box-shadow: none;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.previous {
|
||||||
|
transform: rotateY(180deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
46
src/renderer/components/hotbar/hotbar-selector.tsx
Normal file
46
src/renderer/components/hotbar/hotbar-selector.tsx
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
import "./hotbar-selector.scss";
|
||||||
|
import React from "react";
|
||||||
|
import { Icon } from "../icon";
|
||||||
|
import { Badge } from "../badge";
|
||||||
|
import { makeStyles, Tooltip } from "@material-ui/core";
|
||||||
|
import { Hotbar, HotbarStore } from "../../../common/hotbar-store";
|
||||||
|
import { CommandOverlay } from "../command-palette";
|
||||||
|
import { HotbarSwitchCommand } from "./hotbar-switch-command";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
hotbar: Hotbar;
|
||||||
|
}
|
||||||
|
|
||||||
|
const useStyles = makeStyles(() => ({
|
||||||
|
arrow: {
|
||||||
|
color: "#222",
|
||||||
|
},
|
||||||
|
tooltip: {
|
||||||
|
fontSize: 12,
|
||||||
|
backgroundColor: "#222",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
|
||||||
|
export function HotbarSelector({ hotbar }: Props) {
|
||||||
|
const store = HotbarStore.getInstance();
|
||||||
|
const activeIndexDisplay = store.activeHotbarIndex + 1;
|
||||||
|
const classes = useStyles();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="HotbarSelector flex align-center">
|
||||||
|
<Icon material="play_arrow" className="previous box" onClick={() => store.switchToPrevious()} />
|
||||||
|
<div className="box grow flex align-center">
|
||||||
|
<Tooltip arrow title={hotbar.name} classes={classes}>
|
||||||
|
<Badge
|
||||||
|
id="hotbarIndex"
|
||||||
|
small
|
||||||
|
label={activeIndexDisplay}
|
||||||
|
onClick={() => CommandOverlay.open(<HotbarSwitchCommand />)}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
<Icon material="play_arrow" className="next box" onClick={() => store.switchToNext()} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@ -24,7 +24,7 @@
|
|||||||
// covers whole app view area
|
// covers whole app view area
|
||||||
&.showOnTop {
|
&.showOnTop {
|
||||||
position: fixed !important; // allow to cover ClustersMenu
|
position: fixed !important; // allow to cover ClustersMenu
|
||||||
z-index: 3;
|
z-index: 13;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user