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

Replacing <Avatar/> component by native one (#4452)

This commit is contained in:
Alex Andreev 2021-12-01 16:28:56 +03:00 committed by GitHub
parent 80eeffc229
commit 9ce91884c2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
19 changed files with 337 additions and 261 deletions

View File

@ -33,9 +33,8 @@
margin-right: calc(var(--margin) * 3); margin-right: calc(var(--margin) * 3);
.avatar { .avatar {
:global(.MuiAvatar-root) { font-size: 3ch;
font-size: 3ch; cursor: pointer;
}
} }
.hint { .hint {

View File

@ -27,10 +27,10 @@ import type { CatalogCategory, CatalogEntity } from "../../../common/catalog";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { CatalogEntityDrawerMenu } from "./catalog-entity-drawer-menu"; import { CatalogEntityDrawerMenu } from "./catalog-entity-drawer-menu";
import { CatalogEntityDetailRegistry } from "../../../extensions/registries"; import { CatalogEntityDetailRegistry } from "../../../extensions/registries";
import { HotbarIcon } from "../hotbar/hotbar-icon";
import type { CatalogEntityItem } from "./catalog-entity-item"; import type { CatalogEntityItem } from "./catalog-entity-item";
import { isDevelopment } from "../../../common/vars"; import { isDevelopment } from "../../../common/vars";
import { cssNames } from "../../utils"; import { cssNames } from "../../utils";
import { Avatar } from "../avatar";
interface Props<T extends CatalogEntity> { interface Props<T extends CatalogEntity> {
item: CatalogEntityItem<T> | null | undefined; item: CatalogEntityItem<T> | null | undefined;
@ -60,19 +60,18 @@ export class CatalogEntityDetails<T extends CatalogEntity> extends Component<Pro
{showDetails && ( {showDetails && (
<div className="flex"> <div className="flex">
<div className={styles.entityIcon}> <div className={styles.entityIcon}>
<HotbarIcon <Avatar
uid={item.id}
title={item.name} title={item.name}
source={item.source} colorHash={`${item.name}-${item.source}`}
src={item.entity.spec.icon?.src}
material={item.entity.spec.icon?.material}
background={item.entity.spec.icon?.background}
disabled={!item?.enabled}
onClick={() => item.onRun()}
size={128} size={128}
src={item.entity.spec.icon?.src}
data-testid="detail-panel-hot-bar-icon" data-testid="detail-panel-hot-bar-icon"
background={item.entity.spec.icon?.background}
onClick={() => item.onRun()}
className={styles.avatar} className={styles.avatar}
/> >
{item.entity.spec.icon?.material && <Icon material={item.entity.spec.icon?.material}/>}
</Avatar>
{item?.enabled && ( {item?.enabled && (
<div className={styles.hint}> <div className={styles.hint}>
Click to open Click to open

View File

@ -140,3 +140,7 @@
background-color: var(--blue); background-color: var(--blue);
--color-active: white; --color-active: white;
} }
.catalogAvatar {
font-size: 1.2ch;
}

View File

@ -41,10 +41,10 @@ import { createStorage, prevDefault } from "../../utils";
import { CatalogEntityDetails } from "./catalog-entity-details"; import { CatalogEntityDetails } from "./catalog-entity-details";
import { browseCatalogTab, catalogURL, CatalogViewRouteParam } from "../../../common/routes"; import { browseCatalogTab, catalogURL, CatalogViewRouteParam } from "../../../common/routes";
import { CatalogMenu } from "./catalog-menu"; import { CatalogMenu } from "./catalog-menu";
import { HotbarIcon } from "../hotbar/hotbar-icon";
import { RenderDelay } from "../render-delay/render-delay"; import { RenderDelay } from "../render-delay/render-delay";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { HotbarToggleMenuItem } from "./hotbar-toggle-menu-item"; import { HotbarToggleMenuItem } from "./hotbar-toggle-menu-item";
import { Avatar } from "../avatar";
export const previousActiveTab = createStorage("catalog-previous-active-tab", browseCatalogTab); export const previousActiveTab = createStorage("catalog-previous-active-tab", browseCatalogTab);
@ -213,15 +213,16 @@ export class Catalog extends React.Component<Props> {
return ( return (
<> <>
<HotbarIcon <Avatar
uid={`catalog-icon-${item.getId()}`}
title={item.getName()} title={item.getName()}
source={item.source} colorHash={`${item.getName()}-${item.source}`}
src={item.entity.spec.icon?.src} src={item.entity.spec.icon?.src}
material={item.entity.spec.icon?.material}
background={item.entity.spec.icon?.background} background={item.entity.spec.icon?.background}
className={styles.catalogAvatar}
size={24} size={24}
/> >
{item.entity.spec.icon?.material && <Icon material={item.entity.spec.icon?.material} small/>}
</Avatar>
<span>{item.name}</span> <span>{item.name}</span>
<Icon <Icon
small small

View File

@ -27,4 +27,8 @@
/* Simulate text-overflow:ellipsis styles but for multiple text lines */ /* Simulate text-overflow:ellipsis styles but for multiple text lines */
-webkit-line-clamp: 3; -webkit-line-clamp: 3;
-webkit-box-orient: vertical; -webkit-box-orient: vertical;
}
.settingsAvatar {
margin: 0 10px;
} }

View File

@ -33,8 +33,8 @@ import { EntitySettingRegistry } from "../../../extensions/registries";
import type { EntitySettingsRouteParams } from "../../../common/routes"; import type { EntitySettingsRouteParams } from "../../../common/routes";
import { groupBy } from "lodash"; import { groupBy } from "lodash";
import { SettingLayout } from "../layout/setting-layout"; import { SettingLayout } from "../layout/setting-layout";
import { HotbarIcon } from "../hotbar/hotbar-icon";
import logger from "../../../common/logger"; import logger from "../../../common/logger";
import { Avatar } from "../avatar";
interface Props extends RouteComponentProps<EntitySettingsRouteParams> { interface Props extends RouteComponentProps<EntitySettingsRouteParams> {
} }
@ -96,11 +96,12 @@ export class EntitySettings extends React.Component<Props> {
return ( return (
<> <>
<div className="flex items-center pb-8"> <div className="flex items-center pb-8">
<HotbarIcon <Avatar
uid={this.entity.metadata.uid}
title={this.entity.metadata.name} title={this.entity.metadata.name}
source={this.entity.metadata.source} colorHash={`${this.entity.metadata.name}-${this.entity.metadata.source}`}
src={this.entity.spec.icon?.src} src={this.entity.spec.icon?.src}
className={styles.settingsAvatar}
size={40}
/> />
<div className={styles.entityName}> <div className={styles.entityName}>
{this.entity.metadata.name} {this.entity.metadata.name}

View File

@ -0,0 +1,52 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
import React from "react";
import "@testing-library/jest-dom/extend-expect";
import { render } from "@testing-library/react";
import { Avatar } from "../avatar";
import { Icon } from "../../icon";
describe("<Avatar/>", () => {
test("renders w/o errors", () => {
const { container } = render(<Avatar title="John Ferguson"/>);
expect(container).toBeInstanceOf(HTMLElement);
});
test("shows capital letters from title", () => {
const { getByText } = render(<Avatar title="John Ferguson"/>);
expect(getByText("JF")).toBeInTheDocument();
});
test("shows custom icon passed as children", () => {
const { getByTestId } = render(<Avatar title="John Ferguson"><Icon material="alarm" data-testid="alarm-icon"/></Avatar>);
expect(getByTestId("alarm-icon")).toBeInTheDocument();
});
test("shows <img/> element if src prop passed", () => {
const { getByAltText } = render(<Avatar title="John Ferguson" src="someurl"/>);
expect(getByAltText("John Ferguson")).toBeInTheDocument();
});
});

View File

@ -0,0 +1,51 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
.Avatar {
background-color: hsl(0deg, 0%, 45%);
text-transform: uppercase;
color: white;
font-size: 1.6ch;
overflow: hidden;
display: flex;
align-items: center;
justify-content: center;
user-select: none;
img {
width: 100%;
height: 100%;
object-fit: cover;
}
}
.circle {
border-radius: 50%;
}
.rounded {
border-radius: 4px;
}
.disabled {
opacity: 0.5;
filter: grayscale(0.7);
}

View File

@ -19,20 +19,22 @@
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/ */
import React, { DOMAttributes } from "react"; import styles from "./avatar.module.css";
import React, { HTMLAttributes, ImgHTMLAttributes } from "react";
import randomColor from "randomcolor"; import randomColor from "randomcolor";
import GraphemeSplitter from "grapheme-splitter"; import GraphemeSplitter from "grapheme-splitter";
import { Avatar as MaterialAvatar, AvatarTypeMap } from "@material-ui/core"; import { cssNames, iter } from "../../utils";
import { iter } from "../../utils";
interface Props extends DOMAttributes<HTMLElement>, Partial<AvatarTypeMap> { export interface AvatarProps extends HTMLAttributes<HTMLElement> {
title: string; title: string;
colorHash?: string; colorHash?: string;
width?: number; size?: number;
height?: number;
src?: string; src?: string;
className?: string;
background?: string; background?: string;
variant?: "circle" | "rounded" | "square";
imgProps?: ImgHTMLAttributes<HTMLImageElement>;
disabled?: boolean;
} }
function getNameParts(name: string): string[] { function getNameParts(name: string): string[] {
@ -51,7 +53,7 @@ function getNameParts(name: string): string[] {
return name.split(/@+/); return name.split(/@+/);
} }
function getIconString(title: string) { function getLabelFromTitle(title: string) {
if (!title) { if (!title) {
return "??"; return "??";
} }
@ -69,37 +71,32 @@ function getIconString(title: string) {
].filter(Boolean).join(""); ].filter(Boolean).join("");
} }
export function Avatar(props: Props) { export function Avatar(props: AvatarProps) {
const { title, width = 32, height = 32, colorHash, children, background, ...settings } = props; const { title, variant = "rounded", size = 32, colorHash, children, background, imgProps, src, className, disabled, ...rest } = props;
const getBackgroundColor = () => { const getBackgroundColor = () => {
if (background) { return background || randomColor({ seed: colorHash, luminosity: "dark" });
return background;
}
if (settings.src) {
return "transparent";
}
return randomColor({ seed: colorHash, luminosity: "dark" });
}; };
const generateAvatarStyle = (): React.CSSProperties => { const renderContents = () => {
return { if (src) {
backgroundColor: getBackgroundColor(), return <img src={src} {...imgProps} alt={title}/>;
width, }
height,
textTransform: "uppercase", return children || getLabelFromTitle(title);
};
}; };
return ( return (
<MaterialAvatar <div
variant="rounded" className={cssNames(styles.Avatar, {
style={generateAvatarStyle()} [styles.circle]: variant == "circle",
{...settings} [styles.rounded]: variant == "rounded",
[styles.disabled]: disabled,
}, className)}
style={{ width: `${size}px`, height: `${size}px`, backgroundColor: getBackgroundColor() }}
{...rest}
> >
{children || getIconString(title)} {renderContents()}
</MaterialAvatar> </div>
); );
} }

View File

@ -0,0 +1,22 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
export * from "./avatar";

View File

@ -24,10 +24,10 @@ import type { Cluster } from "../../../../main/cluster";
import { boundMethod } from "../../../utils"; import { boundMethod } from "../../../utils";
import { observable } from "mobx"; import { observable } from "mobx";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { HotbarIcon } from "../../hotbar/hotbar-icon";
import type { KubernetesCluster } from "../../../../common/catalog-entities"; import type { KubernetesCluster } from "../../../../common/catalog-entities";
import { FilePicker, OverSizeLimitStyle } from "../../file-picker"; import { FilePicker, OverSizeLimitStyle } from "../../file-picker";
import { MenuActions, MenuItem } from "../../menu"; import { MenuActions, MenuItem } from "../../menu";
import { Avatar } from "../../avatar";
enum GeneralInputStatus { enum GeneralInputStatus {
CLEAN = "clean", CLEAN = "clean",
@ -86,10 +86,9 @@ export class ClusterIconSetting extends React.Component<Props> {
<FilePicker <FilePicker
accept="image/*" accept="image/*"
label={ label={
<HotbarIcon <Avatar
uid={entity.metadata.uid} colorHash={`${entity.metadata.name}-${entity.metadata.source}`}
title={entity.metadata.name} title={entity.metadata.name}
source={entity.metadata.source}
src={entity.spec.icon?.src} src={entity.spec.icon?.src}
size={53} size={53}
/> />

View File

@ -0,0 +1,56 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
.led {
position: absolute;
left: 3px;
top: 3px;
background-color: var(--layoutBackground);
border: 1px solid var(--clusterMenuBackground);
border-radius: 50%;
padding: 0px;
width: 8px;
height: 8px;
&.online {
background-color: var(--primary);
box-shadow: 0 0 5px var(--clusterMenuBackground), 0 0 5px var(--primary);
}
}
.badge {
position: absolute;
right: -2px;
bottom: -3px;
margin: -8px;
font-size: var(--font-size-small);
background: var(--clusterMenuBackground);
color: var(--textColorAccent);
padding: 0px;
border-radius: 50%;
border: 2px solid var(--clusterMenuBackground);
width: 15px;
height: 15px;
svg {
width: 13px;
}
}

View File

@ -19,7 +19,9 @@
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/ */
import React, { DOMAttributes } from "react"; import styles from "./hotbar-entity-icon.module.css";
import React, { HTMLAttributes } from "react";
import { makeObservable, observable } from "mobx"; import { makeObservable, observable } from "mobx";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
@ -32,10 +34,9 @@ import { Icon } from "../icon";
import { HotbarIcon } from "./hotbar-icon"; import { HotbarIcon } from "./hotbar-icon";
import { LensKubernetesClusterStatus } from "../../../common/catalog-entities/kubernetes-cluster"; import { LensKubernetesClusterStatus } from "../../../common/catalog-entities/kubernetes-cluster";
interface Props extends DOMAttributes<HTMLElement> { interface Props extends HTMLAttributes<HTMLElement> {
entity: CatalogEntity; entity: CatalogEntity;
index: number; index: number;
className?: IClassName;
errorClass?: IClassName; errorClass?: IClassName;
add: (item: CatalogEntity, index: number) => void; add: (item: CatalogEntity, index: number) => void;
remove: (uid: string) => void; remove: (uid: string) => void;
@ -55,7 +56,7 @@ export class HotbarEntityIcon extends React.Component<Props> {
} }
get kindIcon() { get kindIcon() {
const className = "badge"; const className = styles.badge;
const category = catalogCategoryRegistry.getCategoryForEntity(this.props.entity); const category = catalogCategoryRegistry.getCategoryForEntity(this.props.entity);
if (!category) { if (!category) {
@ -74,7 +75,7 @@ export class HotbarEntityIcon extends React.Component<Props> {
return null; return null;
} }
const className = cssNames("led", { online: this.props.entity.status.phase === LensKubernetesClusterStatus.CONNECTED }); // TODO: make it more generic const className = cssNames(styles.led, { [styles.online]: this.props.entity.status.phase === LensKubernetesClusterStatus.CONNECTED }); // TODO: make it more generic
return <div className={className} />; return <div className={className} />;
} }
@ -83,34 +84,25 @@ export class HotbarEntityIcon extends React.Component<Props> {
return catalogEntityRegistry.activeEntity?.metadata?.uid == item.getId(); return catalogEntityRegistry.activeEntity?.metadata?.uid == item.getId();
} }
async onMenuOpen() {
const menuItems: CatalogEntityContextMenu[] = [];
menuItems.unshift({
title: "Remove from Hotbar",
onClick: () => this.props.remove(this.props.entity.metadata.uid),
});
this.contextMenu.menuItems = menuItems;
await this.props.entity.onContextMenuOpen(this.contextMenu);
}
render() { render() {
if (!this.contextMenu) { if (!this.contextMenu) {
return null; return null;
} }
const { const { entity, errorClass, add, remove, index, children, ...elemProps } = this.props;
entity, errorClass, add, remove,
index, children, ...elemProps
} = this.props;
const className = cssNames("HotbarEntityIcon", this.props.className, {
interactive: true,
active: this.isActive(entity),
disabled: !entity,
});
const onOpen = async () => {
const menuItems: CatalogEntityContextMenu[] = [];
menuItems.unshift({
title: "Remove from Hotbar",
onClick: () => remove(entity.metadata.uid),
});
this.contextMenu.menuItems = menuItems;
await entity.onContextMenuOpen(this.contextMenu);
};
const isActive = this.isActive(entity);
return ( return (
<HotbarIcon <HotbarIcon
@ -120,9 +112,10 @@ export class HotbarEntityIcon extends React.Component<Props> {
src={entity.spec.icon?.src} src={entity.spec.icon?.src}
material={entity.spec.icon?.material} material={entity.spec.icon?.material}
background={entity.spec.icon?.background} background={entity.spec.icon?.background}
className={className} className={this.props.className}
active={isActive} active={this.isActive(entity)}
onMenuOpen={onOpen} onMenuOpen={() => this.onMenuOpen()}
disabled={!entity}
menuItems={this.contextMenu.menuItems} menuItems={this.contextMenu.menuItems}
tooltip={`${entity.metadata.name} (${entity.metadata.source})`} tooltip={`${entity.metadata.name} (${entity.metadata.source})`}
{...elemProps} {...elemProps}

View File

@ -0,0 +1,47 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
.HotbarIcon {
--iconActiveShadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px var(--textColorAccent);
--iconHoverShadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px #ffffff50;
display: flex;
cursor: pointer;
position: relative;
border-radius: 6px;
transition: box-shadow 0.1s ease-in-out;
&:not(.active):hover {
box-shadow: var(--iconHoverShadow);
}
}
.contextMenuAvailable {
cursor: context-menu;
}
.avatar {
border-radius: 6px;
}
.active {
box-shadow: var(--iconActiveShadow);
}

View File

@ -1,131 +0,0 @@
/**
* Copyright (c) 2021 OpenLens Authors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
.HotbarMenu {
.HotbarIconMenu {
left: 30px;
min-width: 250px;
li.MenuItem {
font-size: 0.9em;
}
}
}
.HotbarIcon {
--iconActiveShadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px var(--textColorAccent);
--iconHoverShadow: 0 0 0px 3px var(--clusterMenuBackground), 0 0 0px 6px #ffffff50;
border-radius: 6px;
user-select: none;
cursor: pointer;
transition: none;
text-shadow: 0 0 4px #0000008f;
position: relative;
z-index: 0; // allows to catch state of :active pseudo-selector
&:active .MuiAvatar-root {
box-shadow: var(--iconActiveShadow) !important;
}
div.MuiAvatar-colorDefault {
font-weight:500;
text-transform: uppercase;
border-radius: 6px;
}
&.disabled {
opacity: 0.4;
cursor: default;
filter: grayscale(0.7);
&.contextMenuAvailable {
cursor: context-menu;
}
&:hover {
&:not(.active) {
box-shadow: none;
}
}
}
&.isDragging {
box-shadow: none;
}
.MuiAvatar-root {
width: var(--size);
height: var(--size);
border-radius: 6px;
&.active {
box-shadow: var(--iconActiveShadow);
}
&.interactive:not(.active):hover {
box-shadow: var(--iconHoverShadow);
}
}
.led {
position: absolute;
left: 3px;
top: 3px;
background-color: var(--layoutBackground);
border: 1px solid var(--clusterMenuBackground);
border-radius: 50%;
padding: 0px;
width: 8px;
height: 8px;
&.online {
background-color: var(--primary);
box-shadow: 0 0 5px var(--clusterMenuBackground), 0 0 5px var(--primary);
}
}
.badge {
position: absolute;
right: -2px;
bottom: -3px;
margin: -8px;
font-size: var(--font-size-small);
background: var(--clusterMenuBackground);
color: var(--textColorAccent);
padding: 0px;
border-radius: 50%;
border: 2px solid var(--clusterMenuBackground);
width: 15px;
height: 15px;
svg {
width: 13px;
}
}
.materialIcon {
margin-left: 1px;
margin-top: 1px;
text-shadow: none;
font-size: 180%;
}
}

View File

@ -19,32 +19,27 @@
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/ */
import "./hotbar-icon.scss"; import styles from "./hotbar-icon.module.css";
import React, { DOMAttributes, useState } from "react"; import React, { useState } from "react";
import type { CatalogEntityContextMenu } from "../../../common/catalog"; import type { CatalogEntityContextMenu } from "../../../common/catalog";
import { cssNames, IClassName } from "../../utils"; import { cssNames } from "../../utils";
import { ConfirmDialog } from "../confirm-dialog"; import { ConfirmDialog } from "../confirm-dialog";
import { Menu, MenuItem } from "../menu"; import { Menu, MenuItem } from "../menu";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Avatar } from "../avatar/avatar"; import { Avatar, AvatarProps } from "../avatar";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { Tooltip } from "../tooltip"; import { Tooltip } from "../tooltip";
export interface HotbarIconProps extends DOMAttributes<HTMLElement> { export interface Props extends AvatarProps {
uid: string; uid: string;
title: string;
source: string; source: string;
src?: string;
material?: string; material?: string;
onMenuOpen?: () => void; onMenuOpen?: () => void;
className?: IClassName;
active?: boolean; active?: boolean;
menuItems?: CatalogEntityContextMenu[]; menuItems?: CatalogEntityContextMenu[];
disabled?: boolean; disabled?: boolean;
size?: number;
background?: string;
tooltip?: string; tooltip?: string;
} }
@ -65,7 +60,7 @@ function onMenuItemClick(menuItem: CatalogEntityContextMenu) {
} }
} }
export const HotbarIcon = observer(({ menuItems = [], size = 40, tooltip, ...props }: HotbarIconProps) => { export const HotbarIcon = observer(({ menuItems = [], size = 40, tooltip, ...props }: Props) => {
const { uid, title, src, material, active, className, source, disabled, onMenuOpen, onClick, children, ...rest } = props; const { uid, title, src, material, active, className, source, disabled, onMenuOpen, onClick, children, ...rest } = props;
const id = `hotbarIcon-${uid}`; const id = `hotbarIcon-${uid}`;
const [menuOpen, setMenuOpen] = useState(false); const [menuOpen, setMenuOpen] = useState(false);
@ -75,33 +70,25 @@ export const HotbarIcon = observer(({ menuItems = [], size = 40, tooltip, ...pro
}; };
return ( return (
<div className={cssNames("HotbarIcon flex", className, { disabled, contextMenuAvailable: menuItems.length > 0 })}> <div className={cssNames(styles.HotbarIcon, className, { [styles.contextMenuAvailable]: menuItems.length > 0 })}>
{tooltip && <Tooltip targetId={id}>{tooltip}</Tooltip>} {tooltip && <Tooltip targetId={id}>{tooltip}</Tooltip>}
<div <Avatar
{...rest}
id={id} id={id}
onClick={(event) => { title={title}
if (!disabled) { colorHash={`${title}-${source}`}
onClick?.(event); className={cssNames(styles.avatar, { [styles.active]: active })}
} disabled={disabled}
}} size={size}
src={src}
onClick={(event) => !disabled && onClick?.(event)}
> >
<Avatar {material && <Icon material={material} />}
{...rest} </Avatar>
title={title} {children}
colorHash={`${title}-${source}`}
className={cssNames(active ? "active" : "default", { interactive: !!onClick })}
width={size}
height={size}
src={src}
>
{material && <Icon className="materialIcon" material={material} />}
</Avatar>
{children}
</div>
<Menu <Menu
usePortal usePortal
htmlFor={id} htmlFor={id}
className="HotbarIconMenu"
isOpen={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

View File

@ -105,7 +105,7 @@
} }
&:not(:empty) { &:not(:empty) {
.HotbarIcon { > div {
animation: click .1s; animation: click .1s;
} }
} }

View File

@ -120,10 +120,6 @@
} }
} }
} }
.HotbarIcon {
margin: 0 11px;
}
} }
} }

View File

@ -26,7 +26,7 @@ import { HotbarStore } from "../../../common/hotbar-store";
import { broadcastMessage } from "../../../common/ipc"; import { broadcastMessage } from "../../../common/ipc";
import type { CatalogEntity, CatalogEntityContextMenu, CatalogEntityContextMenuContext } from "../../api/catalog-entity"; import type { CatalogEntity, CatalogEntityContextMenu, CatalogEntityContextMenuContext } from "../../api/catalog-entity";
import { IpcRendererNavigationEvents } from "../../navigation/events"; import { IpcRendererNavigationEvents } from "../../navigation/events";
import { Avatar } from "../avatar/avatar"; import { Avatar } from "../avatar";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { navigate } from "../../navigation"; import { navigate } from "../../navigation";
import { Menu, MenuItem } from "../menu"; import { Menu, MenuItem } from "../menu";
@ -108,8 +108,7 @@ export function SidebarCluster({ clusterEntity }: { clusterEntity: CatalogEntity
<Avatar <Avatar
title={metadata.name} title={metadata.name}
colorHash={`${metadata.name}-${metadata.source}`} colorHash={`${metadata.name}-${metadata.source}`}
width={40} size={40}
height={40}
src={spec.icon?.src} src={spec.icon?.src}
className={styles.avatar} className={styles.avatar}
/> />