Feat: Add workspace context actions
This commit is contained in:
@@ -30,6 +30,12 @@
|
|||||||
border-bottom: 1px solid color-mix(in srgb, var(--color-border) 84%, transparent);
|
border-bottom: 1px solid color-mix(in srgb, var(--color-border) 84%, transparent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.headerActions {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: var(--space-2);
|
||||||
|
}
|
||||||
|
|
||||||
.brandBlock {
|
.brandBlock {
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -74,6 +80,22 @@
|
|||||||
color: var(--color-text-subtle);
|
color: var(--color-text-subtle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.createButton {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: var(--space-2);
|
||||||
|
min-height: var(--control-size-md);
|
||||||
|
padding: 0 var(--space-3);
|
||||||
|
border: 1px solid color-mix(in srgb, var(--color-border-strong, var(--color-border)) 82%, transparent);
|
||||||
|
border-radius: 999px;
|
||||||
|
background: color-mix(in srgb, var(--color-surface) 94%, var(--color-canvas) 6%);
|
||||||
|
color: var(--color-text);
|
||||||
|
box-shadow: var(--shadow-soft);
|
||||||
|
font: inherit;
|
||||||
|
font-weight: var(--font-weight-semibold);
|
||||||
|
}
|
||||||
|
|
||||||
.sheetBody {
|
.sheetBody {
|
||||||
min-height: 0;
|
min-height: 0;
|
||||||
display: grid;
|
display: grid;
|
||||||
|
|||||||
@@ -1,6 +1,21 @@
|
|||||||
import { For, Show, type JSX } from "solid-js";
|
import { For, Show, createSignal, type JSX } from "solid-js";
|
||||||
import { ChevronRight, X } from "../../../lib/icons";
|
import { ChevronRight, Plus, X } from "../../../lib/icons";
|
||||||
import { activeProject, activeServer, workspaceStaticItems, workspaceTree, type SidebarItem, type WorkspaceTreeNode } from "../data/shell.data";
|
import { createLongPressGesture } from "../createLongPressGesture";
|
||||||
|
import {
|
||||||
|
activeProject,
|
||||||
|
activeServer,
|
||||||
|
createWorkspaceStaticTarget,
|
||||||
|
createWorkspaceSurfaceTarget,
|
||||||
|
createWorkspaceTreeTarget,
|
||||||
|
workspaceStaticItems,
|
||||||
|
workspaceTree,
|
||||||
|
type SidebarItem,
|
||||||
|
type WorkspaceContextMenuAction,
|
||||||
|
type WorkspaceContextMenuTarget,
|
||||||
|
type WorkspaceStaticItem,
|
||||||
|
type WorkspaceTreeNode,
|
||||||
|
} from "../data/shell.data";
|
||||||
|
import { WorkspaceMobileActionSheet } from "../WorkspaceMobileActionSheet/WorkspaceMobileActionSheet";
|
||||||
import styles from "./MobileWorkspaceBrowser.module.scss";
|
import styles from "./MobileWorkspaceBrowser.module.scss";
|
||||||
|
|
||||||
type MobileWorkspaceBrowserProps = {
|
type MobileWorkspaceBrowserProps = {
|
||||||
@@ -14,29 +29,29 @@ const TreeRow = (props: { node: WorkspaceTreeNode; depth?: number }): JSX.Elemen
|
|||||||
const hasChildren = (props.node.children?.length ?? 0) > 0;
|
const hasChildren = (props.node.children?.length ?? 0) > 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li class={styles.treeListItem}>
|
<button
|
||||||
<button classList={{ [styles.treeRow]: true, [styles.treeRowActive]: props.node.active ?? false, [styles.treeRowBranch]: hasChildren }} type="button" style={{ "--tree-depth": `${depth}` }}>
|
classList={{
|
||||||
<span class={styles.treeRowLead}>
|
[styles.treeRow]: true,
|
||||||
<Icon size={16} strokeWidth={2} />
|
[styles.treeRowActive]: props.node.active ?? false,
|
||||||
<span class={styles.treeLabel}>{props.node.label}</span>
|
[styles.treeRowBranch]: hasChildren,
|
||||||
</span>
|
}}
|
||||||
|
type="button"
|
||||||
|
style={{ "--tree-depth": `${depth}` }}
|
||||||
|
>
|
||||||
|
<span class={styles.treeRowLead}>
|
||||||
|
<Icon size={16} strokeWidth={2} />
|
||||||
|
<span class={styles.treeLabel}>{props.node.label}</span>
|
||||||
|
</span>
|
||||||
|
|
||||||
<span class={styles.treeRowTrail}>
|
<span class={styles.treeRowTrail}>
|
||||||
<Show when={props.node.meta}>
|
<Show when={props.node.meta}>
|
||||||
<span class={styles.treeMeta}>{props.node.meta}</span>
|
<span class={styles.treeMeta}>{props.node.meta}</span>
|
||||||
</Show>
|
</Show>
|
||||||
<Show when={hasChildren}>
|
<Show when={hasChildren}>
|
||||||
<ChevronRight size={14} strokeWidth={2} class={styles.treeChevron} />
|
<ChevronRight size={14} strokeWidth={2} class={styles.treeChevron} />
|
||||||
</Show>
|
</Show>
|
||||||
</span>
|
</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Show when={hasChildren}>
|
|
||||||
<ul class={styles.treeListNested}>
|
|
||||||
<For each={props.node.children}>{(child): JSX.Element => <TreeRow node={child} depth={depth + 1} />}</For>
|
|
||||||
</ul>
|
|
||||||
</Show>
|
|
||||||
</li>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -44,55 +59,158 @@ const StaticRow = (props: { item: SidebarItem }): JSX.Element => {
|
|||||||
const Icon = props.item.icon;
|
const Icon = props.item.icon;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li class={styles.treeListItem}>
|
<button classList={{ [styles.treeRow]: true, [styles.treeRowActive]: props.item.active ?? false }} type="button" style={{ "--tree-depth": "0" }}>
|
||||||
<button classList={{ [styles.treeRow]: true, [styles.treeRowActive]: props.item.active ?? false }} type="button" style={{ "--tree-depth": "0" }}>
|
<span class={styles.treeRowLead}>
|
||||||
<span class={styles.treeRowLead}>
|
<Icon size={16} strokeWidth={2} />
|
||||||
<Icon size={16} strokeWidth={2} />
|
<span class={styles.treeLabel}>{props.item.label}</span>
|
||||||
<span class={styles.treeLabel}>{props.item.label}</span>
|
</span>
|
||||||
</span>
|
<span class={styles.treeRowTrail}>
|
||||||
<span class={styles.treeRowTrail}>
|
<Show when={props.item.meta}>
|
||||||
<Show when={props.item.meta}>
|
<span class={styles.treeMeta}>{props.item.meta}</span>
|
||||||
<span class={styles.treeMeta}>{props.item.meta}</span>
|
</Show>
|
||||||
</Show>
|
<ChevronRight size={14} strokeWidth={2} class={styles.treeChevron} />
|
||||||
<ChevronRight size={14} strokeWidth={2} class={styles.treeChevron} />
|
</span>
|
||||||
</span>
|
</button>
|
||||||
</button>
|
);
|
||||||
|
};
|
||||||
|
const WorkspaceStaticRow = (props: {
|
||||||
|
item: WorkspaceStaticItem;
|
||||||
|
onOpenActionSheet: (target: WorkspaceContextMenuTarget) => void;
|
||||||
|
}): JSX.Element => {
|
||||||
|
const target = createWorkspaceStaticTarget(props.item);
|
||||||
|
const longPress = createLongPressGesture({
|
||||||
|
onLongPress: () => {
|
||||||
|
props.onOpenActionSheet(target);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<li
|
||||||
|
class={styles.treeListItem}
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
event.preventDefault();
|
||||||
|
props.onOpenActionSheet(target);
|
||||||
|
}}
|
||||||
|
{...longPress}
|
||||||
|
>
|
||||||
|
<StaticRow item={props.item} />
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const WorkspaceTreeBranch = (props: {
|
||||||
|
nodes: readonly WorkspaceTreeNode[];
|
||||||
|
depth?: number;
|
||||||
|
onOpenActionSheet: (target: WorkspaceContextMenuTarget) => void;
|
||||||
|
}): JSX.Element => {
|
||||||
|
const depth = props.depth ?? 0;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<For each={props.nodes}>
|
||||||
|
{(node): JSX.Element => {
|
||||||
|
const target = createWorkspaceTreeTarget(node);
|
||||||
|
const longPress = createLongPressGesture({
|
||||||
|
onLongPress: () => {
|
||||||
|
props.onOpenActionSheet(target);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<li
|
||||||
|
class={styles.treeListItem}
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
event.preventDefault();
|
||||||
|
props.onOpenActionSheet(target);
|
||||||
|
}}
|
||||||
|
{...longPress}
|
||||||
|
>
|
||||||
|
<TreeRow node={node} depth={depth} />
|
||||||
|
|
||||||
|
<Show when={node.children?.length}>
|
||||||
|
<ul class={styles.treeListNested}>
|
||||||
|
<WorkspaceTreeBranch nodes={node.children ?? []} depth={depth + 1} onOpenActionSheet={props.onOpenActionSheet} />
|
||||||
|
</ul>
|
||||||
|
</Show>
|
||||||
|
</li>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</For>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
export const MobileWorkspaceBrowser = (props: MobileWorkspaceBrowserProps): JSX.Element => {
|
export const MobileWorkspaceBrowser = (props: MobileWorkspaceBrowserProps): JSX.Element => {
|
||||||
|
const [actionSheetTarget, setActionSheetTarget] = createSignal<WorkspaceContextMenuTarget | null>(null);
|
||||||
const sectionNodes = workspaceTree.filter((node) => (node.children?.length ?? 0) > 0);
|
const sectionNodes = workspaceTree.filter((node) => (node.children?.length ?? 0) > 0);
|
||||||
const looseNodes = workspaceTree.filter((node) => (node.children?.length ?? 0) === 0);
|
const looseNodes = workspaceTree.filter((node) => (node.children?.length ?? 0) === 0);
|
||||||
|
const workspaceTarget = createWorkspaceSurfaceTarget(activeProject);
|
||||||
|
const openActionSheet = (target: WorkspaceContextMenuTarget): void => {
|
||||||
|
setActionSheetTarget(target);
|
||||||
|
};
|
||||||
|
const closeActionSheet = (): void => {
|
||||||
|
setActionSheetTarget(null);
|
||||||
|
};
|
||||||
|
const openWorkspaceActionSheet = (): void => {
|
||||||
|
openActionSheet(workspaceTarget);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleActionSelect = (_action: WorkspaceContextMenuAction, _target: WorkspaceContextMenuTarget): void => {
|
||||||
|
// Mobile first pass only establishes the action-sheet IA and long-press behavior.
|
||||||
|
};
|
||||||
|
|
||||||
|
const workspaceLongPress = createLongPressGesture({
|
||||||
|
onLongPress: openWorkspaceActionSheet,
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Show when={props.open}>
|
<Show when={props.open}>
|
||||||
<div class={styles.browserLayer}>
|
<div class={styles.browserLayer}>
|
||||||
<section class={styles.sheet} aria-label="Mobile workspace browser">
|
<section class={styles.sheet} aria-label="Mobile workspace browser">
|
||||||
<header class={styles.sheetHeader}>
|
<header class={styles.sheetHeader}>
|
||||||
<div class={styles.brandBlock}>
|
<div
|
||||||
|
class={styles.brandBlock}
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
event.preventDefault();
|
||||||
|
openWorkspaceActionSheet();
|
||||||
|
}}
|
||||||
|
{...workspaceLongPress}
|
||||||
|
>
|
||||||
|
{/* Long-pressing the browser header exposes workspace-level actions on mobile. */}
|
||||||
<span class={styles.brandEyebrow}>Moku Work</span>
|
<span class={styles.brandEyebrow}>Moku Work</span>
|
||||||
<strong class={styles.brandTitle}>{activeProject.name}</strong>
|
<strong class={styles.brandTitle}>{activeProject.name}</strong>
|
||||||
<span class={styles.brandContext}>{activeServer.name}</span>
|
<span class={styles.brandContext}>{activeServer.name}</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button class={styles.closeButton} type="button" aria-label="Close workspace browser" onClick={props.onClose}>
|
<div class={styles.headerActions}>
|
||||||
<X size={18} strokeWidth={2} />
|
<button
|
||||||
</button>
|
class={styles.createButton}
|
||||||
|
type="button"
|
||||||
|
aria-label="Create"
|
||||||
|
onClick={openWorkspaceActionSheet}
|
||||||
|
>
|
||||||
|
<Plus size={16} strokeWidth={2.25} />
|
||||||
|
<span>Create</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button class={styles.closeButton} type="button" aria-label="Close workspace browser" onClick={props.onClose}>
|
||||||
|
<X size={18} strokeWidth={2} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<div class={styles.sheetBody}>
|
<div class={styles.sheetBody}>
|
||||||
<section class={styles.sectionBlock}>
|
<section class={styles.sectionBlock}>
|
||||||
<span class={styles.sectionLabel}>Workspace</span>
|
<span class={styles.sectionLabel}>Workspace</span>
|
||||||
<ul class={styles.treeList}>
|
<ul class={styles.treeList}>
|
||||||
<For each={workspaceStaticItems}>{(item): JSX.Element => <StaticRow item={item} />}</For>
|
<For each={workspaceStaticItems}>
|
||||||
|
{(item): JSX.Element => <WorkspaceStaticRow item={item} onOpenActionSheet={openActionSheet} />}
|
||||||
|
</For>
|
||||||
</ul>
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class={styles.sectionBlock}>
|
<section class={styles.sectionBlock}>
|
||||||
<span class={styles.sectionLabel}>Items</span>
|
<span class={styles.sectionLabel}>Items</span>
|
||||||
<ul class={styles.treeList}>
|
<ul class={styles.treeList}>
|
||||||
<For each={sectionNodes}>{(node): JSX.Element => <TreeRow node={node} />}</For>
|
<WorkspaceTreeBranch nodes={sectionNodes} onOpenActionSheet={openActionSheet} />
|
||||||
</ul>
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
@@ -100,12 +218,18 @@ export const MobileWorkspaceBrowser = (props: MobileWorkspaceBrowserProps): JSX.
|
|||||||
<section class={styles.sectionBlock}>
|
<section class={styles.sectionBlock}>
|
||||||
<span class={styles.sectionLabel}>More</span>
|
<span class={styles.sectionLabel}>More</span>
|
||||||
<ul class={styles.treeList}>
|
<ul class={styles.treeList}>
|
||||||
<For each={looseNodes}>{(node): JSX.Element => <TreeRow node={node} />}</For>
|
<WorkspaceTreeBranch nodes={looseNodes} onOpenActionSheet={openActionSheet} />
|
||||||
</ul>
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
</Show>
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
<WorkspaceMobileActionSheet
|
||||||
|
target={actionSheetTarget()}
|
||||||
|
onClose={closeActionSheet}
|
||||||
|
onSelect={handleActionSelect}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</Show>
|
</Show>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,194 @@
|
|||||||
|
.menu {
|
||||||
|
--context-menu-width: 13.5rem;
|
||||||
|
position: fixed;
|
||||||
|
width: min(var(--context-menu-width), calc(100vw - (var(--space-4) * 2)));
|
||||||
|
display: grid;
|
||||||
|
gap: 0;
|
||||||
|
padding: var(--space-1);
|
||||||
|
border: 1px solid color-mix(in srgb, var(--color-border-strong) 56%, transparent);
|
||||||
|
border-radius: calc(var(--radius-md) + (var(--space-1) / 2));
|
||||||
|
background: var(--color-surface);
|
||||||
|
box-shadow: var(--shadow-soft);
|
||||||
|
z-index: 2147483647;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: grid;
|
||||||
|
gap: calc(var(--space-1) / 2);
|
||||||
|
padding: var(--space-2) calc(var(--space-2) + (var(--space-1) / 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.eyebrow {
|
||||||
|
@include text-caption;
|
||||||
|
color: var(--color-text-muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.06em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
@include text-label;
|
||||||
|
color: var(--color-text);
|
||||||
|
font-weight: var(--font-weight-title);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionList {
|
||||||
|
display: grid;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionListCompact {
|
||||||
|
gap: calc(var(--space-1) / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
display: grid;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section:first-child {
|
||||||
|
padding-top: calc(var(--space-1) / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.section + .section {
|
||||||
|
margin-top: calc(var(--space-1) / 2);
|
||||||
|
padding-top: var(--space-2);
|
||||||
|
border-top: 1px solid color-mix(in srgb, var(--color-border) 78%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionLabel {
|
||||||
|
@include text-caption;
|
||||||
|
color: var(--color-text-subtle);
|
||||||
|
padding: 0 var(--space-2);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.04em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionList {
|
||||||
|
display: grid;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionItem {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action {
|
||||||
|
width: 100%;
|
||||||
|
min-height: calc(var(--control-size-md) - var(--space-2));
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
gap: var(--space-2);
|
||||||
|
padding: var(--space-2) calc(var(--space-2) + (var(--space-1) / 2));
|
||||||
|
border: 1px solid transparent;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
background: transparent;
|
||||||
|
color: var(--color-text);
|
||||||
|
text-align: left;
|
||||||
|
font-size: var(--font-size-label);
|
||||||
|
line-height: var(--line-height-label);
|
||||||
|
font-weight: var(--font-weight-label);
|
||||||
|
transition:
|
||||||
|
background var(--motion-duration-fast) var(--motion-ease-standard),
|
||||||
|
border-color var(--motion-duration-fast) var(--motion-ease-standard),
|
||||||
|
color var(--motion-duration-fast) var(--motion-ease-standard);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionCreate {
|
||||||
|
border-color: transparent;
|
||||||
|
background: transparent;
|
||||||
|
font-weight: var(--font-weight-title);
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionCreate:hover,
|
||||||
|
.actionCreate:focus-visible,
|
||||||
|
.actionCreate.actionSubmenuOpen {
|
||||||
|
background: color-mix(in srgb, var(--color-surface-hover) 82%, var(--color-surface));
|
||||||
|
border-color: color-mix(in srgb, var(--color-border) 72%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionCreateIcon {
|
||||||
|
width: calc(var(--control-size-md) - var(--space-3));
|
||||||
|
height: calc(var(--control-size-md) - var(--space-3));
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
background: color-mix(in srgb, var(--color-surface-hover) 72%, var(--color-surface));
|
||||||
|
color: var(--color-text);
|
||||||
|
box-shadow: inset 0 0 0 1px color-mix(in srgb, var(--color-border) 74%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionLabel {
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionMeta {
|
||||||
|
margin-left: auto;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: var(--space-2);
|
||||||
|
padding-left: var(--space-3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionShortcut {
|
||||||
|
color: var(--color-text-muted);
|
||||||
|
font-size: var(--font-size-caption);
|
||||||
|
line-height: var(--line-height-caption);
|
||||||
|
font-weight: var(--font-weight-caption);
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionChevron {
|
||||||
|
color: var(--color-text-subtle);
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionSubmenuOpen {
|
||||||
|
background: color-mix(in srgb, var(--color-surface-hover) 82%, var(--color-surface));
|
||||||
|
border-color: color-mix(in srgb, var(--color-border-strong) 72%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action:hover,
|
||||||
|
.action:focus-visible {
|
||||||
|
background: color-mix(in srgb, var(--color-surface-hover) 78%, var(--color-surface));
|
||||||
|
border-color: color-mix(in srgb, var(--color-border) 72%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionDanger {
|
||||||
|
color: var(--color-danger-text, var(--color-text));
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionDanger:hover,
|
||||||
|
.actionDanger:focus-visible {
|
||||||
|
background: color-mix(in srgb, var(--color-danger-soft, var(--color-surface-hover)) 72%, transparent);
|
||||||
|
border-color: color-mix(in srgb, var(--color-danger-border, var(--color-border)) 72%, transparent);
|
||||||
|
color: var(--color-danger-text, var(--color-text));
|
||||||
|
}
|
||||||
|
|
||||||
|
.submenu {
|
||||||
|
position: absolute;
|
||||||
|
top: calc(var(--space-1) * -1);
|
||||||
|
left: calc(100% + var(--space-2));
|
||||||
|
width: min(var(--context-menu-width), calc(100vw - (var(--space-4) * 2)));
|
||||||
|
padding: var(--space-1);
|
||||||
|
border: 1px solid color-mix(in srgb, var(--color-border-strong) 56%, transparent);
|
||||||
|
border-radius: calc(var(--radius-md) + (var(--space-1) / 2));
|
||||||
|
background: var(--color-surface);
|
||||||
|
box-shadow: var(--shadow-soft);
|
||||||
|
z-index: 2147483647;
|
||||||
|
}
|
||||||
|
|
||||||
|
.submenuList {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--space-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@include respond-down(mobile) {
|
||||||
|
.menu {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,231 @@
|
|||||||
|
import { For, Show, createEffect, createMemo, createSignal, onMount, type JSX } from "solid-js";
|
||||||
|
import { Portal } from "solid-js/web";
|
||||||
|
import { ChevronRight, Plus } from "../../../lib/icons";
|
||||||
|
import {
|
||||||
|
getWorkspaceContextMenuEyebrow,
|
||||||
|
getWorkspaceContextMenuSections,
|
||||||
|
type WorkspaceContextMenuAction,
|
||||||
|
type WorkspaceContextMenuShortcut,
|
||||||
|
type WorkspaceContextMenuTarget,
|
||||||
|
} from "../data/shell.data";
|
||||||
|
import styles from "./WorkspaceContextMenu.module.scss";
|
||||||
|
|
||||||
|
type ShortcutPlatform = "mac" | "windows";
|
||||||
|
|
||||||
|
type NavigatorWithUserAgentData = Navigator & {
|
||||||
|
userAgentData?: {
|
||||||
|
platform?: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type WorkspaceContextMenuPosition = {
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
type WorkspaceContextMenuProps = {
|
||||||
|
target: WorkspaceContextMenuTarget | null;
|
||||||
|
position: WorkspaceContextMenuPosition | null;
|
||||||
|
onClose: VoidFunction;
|
||||||
|
onSelect: (action: WorkspaceContextMenuAction, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
menuRef: (element: HTMLDivElement) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getShortcutPlatform = (): ShortcutPlatform => {
|
||||||
|
if (typeof navigator === "undefined") {
|
||||||
|
return "mac";
|
||||||
|
}
|
||||||
|
|
||||||
|
const navigatorWithUserAgentData = navigator as NavigatorWithUserAgentData;
|
||||||
|
|
||||||
|
const platform =
|
||||||
|
typeof navigatorWithUserAgentData.userAgentData?.platform === "string"
|
||||||
|
? navigatorWithUserAgentData.userAgentData.platform
|
||||||
|
: navigator.platform;
|
||||||
|
|
||||||
|
return /mac|iphone|ipad|ipod/i.test(platform) ? "mac" : "windows";
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatShortcut = (shortcut: WorkspaceContextMenuShortcut, platform: ShortcutPlatform): string => {
|
||||||
|
const keyLabel = (() => {
|
||||||
|
switch (shortcut.key) {
|
||||||
|
case "enter":
|
||||||
|
return platform === "mac" ? "↩" : "Enter";
|
||||||
|
case "delete":
|
||||||
|
return platform === "mac" ? "⌫" : "Del";
|
||||||
|
default:
|
||||||
|
return shortcut.key.toUpperCase();
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
|
||||||
|
const modifierLabels =
|
||||||
|
shortcut.modifiers?.map((modifier) => {
|
||||||
|
switch (modifier) {
|
||||||
|
case "meta":
|
||||||
|
return platform === "mac" ? "⌘" : "Ctrl";
|
||||||
|
case "alt":
|
||||||
|
return platform === "mac" ? "⌥" : "Alt";
|
||||||
|
case "shift":
|
||||||
|
return platform === "mac" ? "⇧" : "Shift";
|
||||||
|
}
|
||||||
|
}) ?? [];
|
||||||
|
|
||||||
|
if (platform === "mac") {
|
||||||
|
return `${modifierLabels.join("")}${keyLabel}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...modifierLabels, keyLabel].join("+");
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WorkspaceContextMenu = (props: WorkspaceContextMenuProps): JSX.Element => {
|
||||||
|
const [activeSubmenuActionId, setActiveSubmenuActionId] = createSignal<string | null>(null);
|
||||||
|
const [shortcutPlatform, setShortcutPlatform] = createSignal<ShortcutPlatform>("mac");
|
||||||
|
const sections = createMemo(() => (props.target ? getWorkspaceContextMenuSections(props.target) : []));
|
||||||
|
const handleActionSelect = (action: WorkspaceContextMenuAction, target: WorkspaceContextMenuTarget): void => {
|
||||||
|
props.onSelect(action, target);
|
||||||
|
props.onClose();
|
||||||
|
};
|
||||||
|
const menuState = createMemo<{
|
||||||
|
target: WorkspaceContextMenuTarget;
|
||||||
|
position: WorkspaceContextMenuPosition;
|
||||||
|
} | null>(() =>
|
||||||
|
props.target && props.position
|
||||||
|
? {
|
||||||
|
target: props.target,
|
||||||
|
position: props.position,
|
||||||
|
}
|
||||||
|
: null,
|
||||||
|
);
|
||||||
|
const showHeader = (): boolean => props.target?.kind !== "workspace";
|
||||||
|
const sectionHasLabel = createMemo(() => sections().some((section) => Boolean(section.label)));
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
setShortcutPlatform(getShortcutPlatform());
|
||||||
|
});
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
void props.target;
|
||||||
|
setActiveSubmenuActionId(null);
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Show when={menuState()}>
|
||||||
|
{(resolvedMenuState): JSX.Element => {
|
||||||
|
const target = resolvedMenuState().target;
|
||||||
|
const position = resolvedMenuState().position;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Portal>
|
||||||
|
<div
|
||||||
|
ref={props.menuRef}
|
||||||
|
class={styles.menu}
|
||||||
|
role="menu"
|
||||||
|
aria-label={`${target.label} context menu`}
|
||||||
|
style={{
|
||||||
|
left: `${position.x}px`,
|
||||||
|
top: `${position.y}px`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Show when={target.kind !== "workspace"}>
|
||||||
|
<header class={styles.header}>
|
||||||
|
<span class={styles.eyebrow}>{getWorkspaceContextMenuEyebrow(target)}</span>
|
||||||
|
<strong class={styles.title}>{target.label}</strong>
|
||||||
|
</header>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<div classList={{ [styles.sectionList]: true, [styles.sectionListCompact]: !sectionHasLabel() }}>
|
||||||
|
<For each={sections()}>
|
||||||
|
{(section): JSX.Element => (
|
||||||
|
<section class={styles.section}>
|
||||||
|
<Show when={section.label}>
|
||||||
|
<span class={styles.sectionLabel}>{section.label}</span>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<div class={styles.actionList}>
|
||||||
|
<For each={section.items}>
|
||||||
|
{(action): JSX.Element => {
|
||||||
|
const isSubmenuOpen = () => activeSubmenuActionId() === action.id;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
class={styles.actionItem}
|
||||||
|
onMouseEnter={() => {
|
||||||
|
setActiveSubmenuActionId(action.children ? action.id : null);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
role="menuitem"
|
||||||
|
classList={{
|
||||||
|
[styles.action]: true,
|
||||||
|
[styles.actionCreate]: action.id === "create",
|
||||||
|
[styles.actionDanger]: action.tone === "danger",
|
||||||
|
[styles.actionSubmenuOpen]: isSubmenuOpen(),
|
||||||
|
}}
|
||||||
|
onClick={() => {
|
||||||
|
if (action.children) {
|
||||||
|
setActiveSubmenuActionId(isSubmenuOpen() ? null : action.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleActionSelect(action, target);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Show when={action.id === "create"}>
|
||||||
|
<span class={styles.actionCreateIcon} aria-hidden="true">
|
||||||
|
<Plus size={14} strokeWidth={2.25} />
|
||||||
|
</span>
|
||||||
|
</Show>
|
||||||
|
<span class={styles.actionLabel}>{action.label}</span>
|
||||||
|
<div class={styles.actionMeta}>
|
||||||
|
<Show when={action.shortcut}>
|
||||||
|
<span class={styles.actionShortcut}>{formatShortcut(action.shortcut!, shortcutPlatform())}</span>
|
||||||
|
</Show>
|
||||||
|
<Show when={action.children}>
|
||||||
|
<ChevronRight class={styles.actionChevron} size={16} strokeWidth={2} />
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<Show when={action.children && isSubmenuOpen()}>
|
||||||
|
<div class={styles.submenu} role="menu" aria-label={`${action.label} submenu`}>
|
||||||
|
<div class={styles.submenuList}>
|
||||||
|
<For each={action.children ?? []}>
|
||||||
|
{(childAction): JSX.Element => (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
role="menuitem"
|
||||||
|
classList={{
|
||||||
|
[styles.action]: true,
|
||||||
|
[styles.actionDanger]: childAction.tone === "danger",
|
||||||
|
}}
|
||||||
|
onClick={() => handleActionSelect(childAction, target)}
|
||||||
|
>
|
||||||
|
<span class={styles.actionLabel}>{childAction.label}</span>
|
||||||
|
<div class={styles.actionMeta}>
|
||||||
|
<Show when={childAction.shortcut}>
|
||||||
|
<span class={styles.actionShortcut}>{formatShortcut(childAction.shortcut!, shortcutPlatform())}</span>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
)}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Portal>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</Show>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,134 @@
|
|||||||
|
import { createEffect, createSignal, onCleanup } from "solid-js";
|
||||||
|
import type { WorkspaceContextMenuTarget } from "../data/shell.data";
|
||||||
|
|
||||||
|
type WorkspaceContextMenuState = {
|
||||||
|
target: WorkspaceContextMenuTarget;
|
||||||
|
x: number;
|
||||||
|
y: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
const readRootPixelToken = (name: string, fallback: number): number => {
|
||||||
|
if (typeof window === "undefined") {
|
||||||
|
return fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
const value = window.getComputedStyle(document.documentElement).getPropertyValue(name).trim();
|
||||||
|
const parsed = Number.parseFloat(value);
|
||||||
|
|
||||||
|
if (!Number.isFinite(parsed)) {
|
||||||
|
return fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value.endsWith("px")) {
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsed * 16;
|
||||||
|
};
|
||||||
|
|
||||||
|
const clampMenuPosition = (value: number, min: number, max: number): number => {
|
||||||
|
if (max <= min) {
|
||||||
|
return min;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Math.min(Math.max(value, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createWorkspaceContextMenuController = () => {
|
||||||
|
const [menuState, setMenuState] = createSignal<WorkspaceContextMenuState | null>(null);
|
||||||
|
let menuRef: HTMLDivElement | undefined;
|
||||||
|
|
||||||
|
const closeMenu = (): void => {
|
||||||
|
setMenuState(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const repositionMenu = (): void => {
|
||||||
|
if (typeof window === "undefined" || !menuRef) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const current = menuState();
|
||||||
|
|
||||||
|
if (!current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const viewportPadding = readRootPixelToken("--space-4", 16);
|
||||||
|
const rect = menuRef.getBoundingClientRect();
|
||||||
|
const nextX = clampMenuPosition(current.x, viewportPadding, window.innerWidth - rect.width - viewportPadding);
|
||||||
|
const nextY = clampMenuPosition(current.y, viewportPadding, window.innerHeight - rect.height - viewportPadding);
|
||||||
|
|
||||||
|
if (nextX === current.x && nextY === current.y) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setMenuState({ ...current, x: nextX, y: nextY });
|
||||||
|
};
|
||||||
|
|
||||||
|
const openMenu = (event: MouseEvent, target: WorkspaceContextMenuTarget): void => {
|
||||||
|
event.preventDefault();
|
||||||
|
setMenuState({ target, x: event.clientX, y: event.clientY });
|
||||||
|
};
|
||||||
|
|
||||||
|
const openMenuFromElement = (element: HTMLElement, target: WorkspaceContextMenuTarget): void => {
|
||||||
|
const rect = element.getBoundingClientRect();
|
||||||
|
setMenuState({
|
||||||
|
target,
|
||||||
|
x: rect.left + rect.width / 2,
|
||||||
|
y: rect.top + rect.height / 2,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
if (!menuState()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof window === "undefined") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const frame = window.requestAnimationFrame(() => {
|
||||||
|
repositionMenu();
|
||||||
|
});
|
||||||
|
|
||||||
|
const handlePointerDown = (event: PointerEvent): void => {
|
||||||
|
if (!menuRef?.contains(event.target as Node)) {
|
||||||
|
closeMenu();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleKeyDown = (event: KeyboardEvent): void => {
|
||||||
|
if (event.key === "Escape") {
|
||||||
|
closeMenu();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleViewportChange = (): void => {
|
||||||
|
closeMenu();
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener("pointerdown", handlePointerDown);
|
||||||
|
window.addEventListener("resize", handleViewportChange);
|
||||||
|
window.addEventListener("scroll", handleViewportChange, true);
|
||||||
|
window.addEventListener("keydown", handleKeyDown);
|
||||||
|
|
||||||
|
onCleanup(() => {
|
||||||
|
window.cancelAnimationFrame(frame);
|
||||||
|
document.removeEventListener("pointerdown", handlePointerDown);
|
||||||
|
window.removeEventListener("resize", handleViewportChange);
|
||||||
|
window.removeEventListener("scroll", handleViewportChange, true);
|
||||||
|
window.removeEventListener("keydown", handleKeyDown);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
menuState,
|
||||||
|
openMenu,
|
||||||
|
openMenuFromElement,
|
||||||
|
closeMenu,
|
||||||
|
setMenuRef: (element: HTMLDivElement): void => {
|
||||||
|
menuRef = element;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
||||||
@@ -0,0 +1,146 @@
|
|||||||
|
.layer {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include respond-down(mobile) {
|
||||||
|
.layer {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
display: block;
|
||||||
|
z-index: var(--z-modal);
|
||||||
|
}
|
||||||
|
|
||||||
|
.backdrop {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
border: 0;
|
||||||
|
background: color-mix(in srgb, black 52%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sheet {
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
max-height: calc(100dvh - (var(--space-12) * 2));
|
||||||
|
display: grid;
|
||||||
|
gap: var(--space-3);
|
||||||
|
padding: var(--space-3) var(--space-3) calc(var(--space-4) + env(safe-area-inset-bottom, 0px));
|
||||||
|
border-top: 1px solid color-mix(in srgb, var(--color-border) 88%, transparent);
|
||||||
|
border-radius: var(--radius-xl) var(--radius-xl) 0 0;
|
||||||
|
background: var(--color-surface);
|
||||||
|
box-shadow: var(--shadow-strong);
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.handle {
|
||||||
|
width: var(--space-10);
|
||||||
|
height: var(--space-1);
|
||||||
|
margin: 0 auto;
|
||||||
|
border-radius: var(--radius-pill);
|
||||||
|
background: color-mix(in srgb, var(--color-text-muted) 24%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
align-items: start;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: var(--space-3);
|
||||||
|
padding-bottom: var(--space-3);
|
||||||
|
border-bottom: 1px solid color-mix(in srgb, var(--color-border) 88%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.headerCopy {
|
||||||
|
min-width: 0;
|
||||||
|
display: grid;
|
||||||
|
gap: calc(var(--space-1) / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eyebrow {
|
||||||
|
@include text-caption;
|
||||||
|
color: var(--color-text-muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.06em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
@include text-title;
|
||||||
|
color: var(--color-text);
|
||||||
|
font-weight: var(--font-weight-semibold);
|
||||||
|
}
|
||||||
|
|
||||||
|
.closeButton {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: var(--control-size-md);
|
||||||
|
height: var(--control-size-md);
|
||||||
|
padding: 0;
|
||||||
|
border: 1px solid color-mix(in srgb, var(--color-border) 88%, transparent);
|
||||||
|
border-radius: var(--radius-pill);
|
||||||
|
background: color-mix(in srgb, var(--color-surface) 84%, var(--color-surface-elevated, var(--color-surface)) 16%);
|
||||||
|
color: var(--color-text-subtle);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionList {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--space-3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.section {
|
||||||
|
display: grid;
|
||||||
|
gap: var(--space-2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.section + .section {
|
||||||
|
padding-top: var(--space-3);
|
||||||
|
border-top: 1px solid color-mix(in srgb, var(--color-border) 92%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.sectionLabel {
|
||||||
|
@include text-caption;
|
||||||
|
padding-inline: var(--space-1);
|
||||||
|
color: var(--color-text-muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.06em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionList {
|
||||||
|
display: grid;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid color-mix(in srgb, var(--color-border) 88%, transparent);
|
||||||
|
border-radius: var(--radius-lg);
|
||||||
|
background: color-mix(in srgb, var(--color-surface) 88%, var(--color-surface-elevated, var(--color-surface)) 12%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action {
|
||||||
|
min-width: 0;
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
min-height: calc(var(--control-size-lg) + var(--space-2));
|
||||||
|
padding: 0 var(--space-3);
|
||||||
|
border: 0;
|
||||||
|
background: transparent;
|
||||||
|
color: var(--color-text);
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action:active {
|
||||||
|
background: color-mix(in srgb, var(--color-text) 6%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.action + .action {
|
||||||
|
border-top: 1px solid color-mix(in srgb, var(--color-border) 92%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionLabel {
|
||||||
|
@include text-label;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actionDanger {
|
||||||
|
color: var(--color-danger-text, var(--color-text));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,131 @@
|
|||||||
|
import { For, Show, createMemo, type JSX } from "solid-js";
|
||||||
|
import { Portal } from "solid-js/web";
|
||||||
|
import { X } from "../../../lib/icons";
|
||||||
|
import {
|
||||||
|
getWorkspaceContextMenuEyebrow,
|
||||||
|
getWorkspaceContextMenuSections,
|
||||||
|
type WorkspaceContextMenuAction,
|
||||||
|
type WorkspaceContextMenuSection,
|
||||||
|
type WorkspaceContextMenuTarget,
|
||||||
|
} from "../data/shell.data";
|
||||||
|
import styles from "./WorkspaceMobileActionSheet.module.scss";
|
||||||
|
|
||||||
|
type WorkspaceMobileActionSheetProps = {
|
||||||
|
target: WorkspaceContextMenuTarget | null;
|
||||||
|
onClose: VoidFunction;
|
||||||
|
onSelect: (action: WorkspaceContextMenuAction, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
type FlattenedActionSection = {
|
||||||
|
id: string;
|
||||||
|
label?: string;
|
||||||
|
items: readonly WorkspaceContextMenuAction[];
|
||||||
|
};
|
||||||
|
|
||||||
|
const flattenMobileSections = (
|
||||||
|
sections: readonly WorkspaceContextMenuSection[],
|
||||||
|
): readonly FlattenedActionSection[] => {
|
||||||
|
// Mobile uses a flat action-sheet model, so desktop flyout groups become
|
||||||
|
// standalone labeled sections instead of nested menus.
|
||||||
|
return sections.flatMap((section) => {
|
||||||
|
const directActions = section.items.filter((action) => !action.children?.length);
|
||||||
|
const nestedSections = section.items
|
||||||
|
.filter((action) => action.children?.length)
|
||||||
|
.map((action) => ({
|
||||||
|
id: `${section.id}-${action.id}`,
|
||||||
|
label: action.label,
|
||||||
|
items: action.children ?? [],
|
||||||
|
}));
|
||||||
|
|
||||||
|
const flattenedSections: FlattenedActionSection[] = [];
|
||||||
|
|
||||||
|
if (directActions.length > 0) {
|
||||||
|
flattenedSections.push({
|
||||||
|
id: section.id,
|
||||||
|
label: section.label,
|
||||||
|
items: directActions,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...flattenedSections, ...nestedSections];
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WorkspaceMobileActionSheet = (props: WorkspaceMobileActionSheetProps): JSX.Element => {
|
||||||
|
const sheetState = createMemo(() => {
|
||||||
|
if (!props.target) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
target: props.target,
|
||||||
|
sections: flattenMobileSections(getWorkspaceContextMenuSections(props.target)),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const handleActionSelect = (action: WorkspaceContextMenuAction, target: WorkspaceContextMenuTarget): void => {
|
||||||
|
props.onSelect(action, target);
|
||||||
|
props.onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Show when={sheetState()}>
|
||||||
|
{(sheetState): JSX.Element => {
|
||||||
|
const target = sheetState().target;
|
||||||
|
const sections = sheetState().sections;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Portal>
|
||||||
|
<div class={styles.layer}>
|
||||||
|
<button class={styles.backdrop} type="button" aria-label="Close action sheet" onClick={props.onClose} />
|
||||||
|
|
||||||
|
<section class={styles.sheet} aria-label={`${target.label} actions`}>
|
||||||
|
<div class={styles.handle} aria-hidden="true" />
|
||||||
|
|
||||||
|
<header class={styles.header}>
|
||||||
|
<div class={styles.headerCopy}>
|
||||||
|
<span class={styles.eyebrow}>{getWorkspaceContextMenuEyebrow(target)}</span>
|
||||||
|
<strong class={styles.title}>{target.label}</strong>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button class={styles.closeButton} type="button" aria-label="Close action sheet" onClick={props.onClose}>
|
||||||
|
<X size={18} strokeWidth={2} />
|
||||||
|
</button>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class={styles.sectionList}>
|
||||||
|
<For each={sections}>
|
||||||
|
{(section): JSX.Element => (
|
||||||
|
<section class={styles.section}>
|
||||||
|
<Show when={section.label}>
|
||||||
|
<span class={styles.sectionLabel}>{section.label}</span>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<div class={styles.actionList}>
|
||||||
|
<For each={section.items}>
|
||||||
|
{(action): JSX.Element => (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
classList={{
|
||||||
|
[styles.action]: true,
|
||||||
|
[styles.actionDanger]: action.tone === "danger",
|
||||||
|
}}
|
||||||
|
onClick={() => handleActionSelect(action, target)}
|
||||||
|
>
|
||||||
|
<span class={styles.actionLabel}>{action.label}</span>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
)}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</Portal>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
</Show>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,9 +1,23 @@
|
|||||||
// Path: Frontend/src/components/shell/WorkspaceSidebar/WorkspaceSidebar.tsx
|
// Path: Frontend/src/components/shell/WorkspaceSidebar/WorkspaceSidebar.tsx
|
||||||
|
|
||||||
import { For, Show, createSignal, type JSX } from "solid-js";
|
import { For, Show, createMemo, createSignal, type JSX } from "solid-js";
|
||||||
import { ChevronLeft, ChevronRight } from "../../../lib/icons";
|
import { ChevronLeft, ChevronRight } from "../../../lib/icons";
|
||||||
import { ProjectSelector } from "../ProjectSelector/ProjectSelector";
|
import { ProjectSelector } from "../ProjectSelector/ProjectSelector";
|
||||||
import { workspaceSidebarHeaderActions, workspaceStaticItems, workspaceTree, type SidebarItem, type WorkspaceTreeNode } from "../data/shell.data";
|
import {
|
||||||
|
activeProject,
|
||||||
|
createWorkspaceStaticTarget,
|
||||||
|
createWorkspaceSurfaceTarget,
|
||||||
|
createWorkspaceTreeTarget,
|
||||||
|
workspaceSidebarHeaderActions,
|
||||||
|
workspaceStaticItems,
|
||||||
|
workspaceTree,
|
||||||
|
type WorkspaceContextMenuAction,
|
||||||
|
type WorkspaceContextMenuTarget,
|
||||||
|
type WorkspaceStaticItem,
|
||||||
|
type WorkspaceTreeNode,
|
||||||
|
} from "../data/shell.data";
|
||||||
|
import { WorkspaceContextMenu } from "../WorkspaceContextMenu/WorkspaceContextMenu";
|
||||||
|
import { createWorkspaceContextMenuController } from "../WorkspaceContextMenu/createWorkspaceContextMenuController";
|
||||||
import styles from "./WorkspaceSidebar.module.scss";
|
import styles from "./WorkspaceSidebar.module.scss";
|
||||||
|
|
||||||
type WorkspaceSidebarProps = {
|
type WorkspaceSidebarProps = {
|
||||||
@@ -12,8 +26,15 @@ type WorkspaceSidebarProps = {
|
|||||||
onToggleRailCollapse: () => void;
|
onToggleRailCollapse: () => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
const WorkspaceHomeEntry = (props: { item: SidebarItem }): JSX.Element => {
|
const isContextMenuKeyboardTrigger = (event: KeyboardEvent): boolean => event.key === "ContextMenu" || (event.shiftKey && event.key === "F10");
|
||||||
|
|
||||||
|
const WorkspaceHomeEntry = (props: {
|
||||||
|
item: WorkspaceStaticItem;
|
||||||
|
onOpenContextMenu: (event: MouseEvent, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
onOpenContextMenuFromKeyboard: (element: HTMLElement, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
}): JSX.Element => {
|
||||||
const Icon = props.item.icon;
|
const Icon = props.item.icon;
|
||||||
|
const target = createWorkspaceStaticTarget(props.item);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li>
|
<li>
|
||||||
@@ -26,6 +47,18 @@ const WorkspaceHomeEntry = (props: { item: SidebarItem }): JSX.Element => {
|
|||||||
aria-current={props.item.active ? "page" : undefined}
|
aria-current={props.item.active ? "page" : undefined}
|
||||||
aria-label={props.item.label}
|
aria-label={props.item.label}
|
||||||
title={props.item.label}
|
title={props.item.label}
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
event.stopPropagation();
|
||||||
|
props.onOpenContextMenu(event, target);
|
||||||
|
}}
|
||||||
|
onKeyDown={(event): void => {
|
||||||
|
if (!isContextMenuKeyboardTrigger(event)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
props.onOpenContextMenuFromKeyboard(event.currentTarget, target);
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<Icon class={styles.icon} size={18} strokeWidth={2} />
|
<Icon class={styles.icon} size={18} strokeWidth={2} />
|
||||||
<span class={styles.label}>{props.item.label}</span>
|
<span class={styles.label}>{props.item.label}</span>
|
||||||
@@ -37,7 +70,12 @@ const WorkspaceHomeEntry = (props: { item: SidebarItem }): JSX.Element => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const WorkspaceTreeBranch = (props: { nodes: readonly WorkspaceTreeNode[]; depth?: number }): JSX.Element => {
|
const WorkspaceTreeBranch = (props: {
|
||||||
|
nodes: readonly WorkspaceTreeNode[];
|
||||||
|
depth?: number;
|
||||||
|
onOpenContextMenu: (event: MouseEvent, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
onOpenContextMenuFromKeyboard: (element: HTMLElement, target: WorkspaceContextMenuTarget) => void;
|
||||||
|
}): JSX.Element => {
|
||||||
const depth = () => props.depth ?? 0;
|
const depth = () => props.depth ?? 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -45,6 +83,7 @@ const WorkspaceTreeBranch = (props: { nodes: readonly WorkspaceTreeNode[]; depth
|
|||||||
<For each={props.nodes}>
|
<For each={props.nodes}>
|
||||||
{(node): JSX.Element => {
|
{(node): JSX.Element => {
|
||||||
const Icon = node.icon;
|
const Icon = node.icon;
|
||||||
|
const target = createWorkspaceTreeTarget(node);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li>
|
<li>
|
||||||
@@ -59,6 +98,18 @@ const WorkspaceTreeBranch = (props: { nodes: readonly WorkspaceTreeNode[]; depth
|
|||||||
aria-current={node.active ? "page" : undefined}
|
aria-current={node.active ? "page" : undefined}
|
||||||
aria-label={node.label}
|
aria-label={node.label}
|
||||||
title={node.label}
|
title={node.label}
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
event.stopPropagation();
|
||||||
|
props.onOpenContextMenu(event, target);
|
||||||
|
}}
|
||||||
|
onKeyDown={(event): void => {
|
||||||
|
if (!isContextMenuKeyboardTrigger(event)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
props.onOpenContextMenuFromKeyboard(event.currentTarget, target);
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<Icon class={styles.icon} size={18} strokeWidth={2} />
|
<Icon class={styles.icon} size={18} strokeWidth={2} />
|
||||||
<span class={styles.label}>{node.label}</span>
|
<span class={styles.label}>{node.label}</span>
|
||||||
@@ -68,7 +119,12 @@ const WorkspaceTreeBranch = (props: { nodes: readonly WorkspaceTreeNode[]; depth
|
|||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Show when={node.children?.length}>
|
<Show when={node.children?.length}>
|
||||||
<WorkspaceTreeBranch nodes={node.children ?? []} depth={depth() + 1} />
|
<WorkspaceTreeBranch
|
||||||
|
nodes={node.children ?? []}
|
||||||
|
depth={depth() + 1}
|
||||||
|
onOpenContextMenu={props.onOpenContextMenu}
|
||||||
|
onOpenContextMenuFromKeyboard={props.onOpenContextMenuFromKeyboard}
|
||||||
|
/>
|
||||||
</Show>
|
</Show>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
@@ -78,86 +134,128 @@ const WorkspaceTreeBranch = (props: { nodes: readonly WorkspaceTreeNode[]; depth
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const WorkspaceSidebar = (props: WorkspaceSidebarProps): JSX.Element => {
|
export const WorkspaceSidebar = (props: WorkspaceSidebarProps): JSX.Element => {
|
||||||
const [isProjectDrawerOpen, setIsProjectDrawerOpen] = createSignal(false);
|
const [isProjectDrawerOpen, setIsProjectDrawerOpen] = createSignal(false);
|
||||||
|
const contextMenu = createWorkspaceContextMenuController();
|
||||||
const railToggleLabel = (): string => (props.railCollapsed ? "Expand server rail" : "Collapse server rail");
|
const railToggleLabel = (): string => (props.railCollapsed ? "Expand server rail" : "Collapse server rail");
|
||||||
|
const sidebarContextMenuTarget = createWorkspaceSurfaceTarget(activeProject);
|
||||||
|
const contextMenuTarget = createMemo(() => contextMenu.menuState()?.target ?? null);
|
||||||
|
const contextMenuPosition = createMemo(() => {
|
||||||
|
const state = contextMenu.menuState();
|
||||||
|
|
||||||
|
return state
|
||||||
|
? {
|
||||||
|
x: state.x,
|
||||||
|
y: state.y,
|
||||||
|
}
|
||||||
|
: null;
|
||||||
|
});
|
||||||
|
|
||||||
|
const handleContextActionSelect = (_action: WorkspaceContextMenuAction, _target: WorkspaceContextMenuTarget): void => {
|
||||||
|
// Initial implementation only establishes the menu IA and placement.
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<aside
|
<>
|
||||||
classList={{
|
<aside
|
||||||
[styles.sidebar]: true,
|
|
||||||
[styles.sidebarCollapsed]: props.collapsed,
|
|
||||||
}}
|
|
||||||
aria-label="Left workspace sidebar"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
classList={{
|
classList={{
|
||||||
[styles.header]: true,
|
[styles.sidebar]: true,
|
||||||
[styles.headerDrawerOpen]: isProjectDrawerOpen(),
|
[styles.sidebarCollapsed]: props.collapsed,
|
||||||
|
}}
|
||||||
|
aria-label="Left workspace sidebar"
|
||||||
|
onContextMenu={(event): void => {
|
||||||
|
contextMenu.openMenu(event, sidebarContextMenuTarget);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div class={styles.headerActions}>
|
<div
|
||||||
<button
|
classList={{
|
||||||
type="button"
|
[styles.header]: true,
|
||||||
classList={{
|
[styles.headerDrawerOpen]: isProjectDrawerOpen(),
|
||||||
[styles.headerActionButton]: true,
|
}}
|
||||||
[styles.headerCollapseButton]: true,
|
>
|
||||||
}}
|
<div class={styles.headerActions}>
|
||||||
aria-label={railToggleLabel()}
|
<button
|
||||||
title={railToggleLabel()}
|
type="button"
|
||||||
onClick={props.onToggleRailCollapse}
|
classList={{
|
||||||
>
|
[styles.headerActionButton]: true,
|
||||||
{props.railCollapsed ? <ChevronRight size={16} strokeWidth={2} /> : <ChevronLeft size={16} strokeWidth={2} />}
|
[styles.headerCollapseButton]: true,
|
||||||
</button>
|
}}
|
||||||
|
aria-label={railToggleLabel()}
|
||||||
|
title={railToggleLabel()}
|
||||||
|
onClick={props.onToggleRailCollapse}
|
||||||
|
>
|
||||||
|
{props.railCollapsed ? <ChevronRight size={16} strokeWidth={2} /> : <ChevronLeft size={16} strokeWidth={2} />}
|
||||||
|
</button>
|
||||||
|
|
||||||
<For each={workspaceSidebarHeaderActions}>
|
<For each={workspaceSidebarHeaderActions}>
|
||||||
{(action): JSX.Element => {
|
{(action): JSX.Element => {
|
||||||
const Icon = action.icon;
|
const Icon = action.icon;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<button type="button" class={styles.headerActionButton} aria-label={action.label} title={action.label}>
|
<button type="button" class={styles.headerActionButton} aria-label={action.label} title={action.label}>
|
||||||
<Icon size={16} strokeWidth={2} />
|
<Icon size={16} strokeWidth={2} />
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
</For>
|
</For>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class={styles.headerControls}>
|
||||||
|
<ProjectSelector
|
||||||
|
compact={props.collapsed}
|
||||||
|
isOpen={isProjectDrawerOpen()}
|
||||||
|
onToggle={(): void => {
|
||||||
|
setIsProjectDrawerOpen(true);
|
||||||
|
}}
|
||||||
|
onClose={(): void => {
|
||||||
|
setIsProjectDrawerOpen(false);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class={styles.headerControls}>
|
<div
|
||||||
<ProjectSelector
|
classList={{
|
||||||
compact={props.collapsed}
|
[styles.section]: true,
|
||||||
isOpen={isProjectDrawerOpen()}
|
[styles.sectionHidden]: isProjectDrawerOpen(),
|
||||||
onToggle={(): void => {
|
}}
|
||||||
setIsProjectDrawerOpen(true);
|
>
|
||||||
}}
|
|
||||||
onClose={(): void => {
|
|
||||||
setIsProjectDrawerOpen(false);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
classList={{
|
|
||||||
[styles.section]: true,
|
|
||||||
[styles.sectionHidden]: isProjectDrawerOpen(),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Show when={!props.collapsed}>
|
|
||||||
<span class={styles.sectionLabel}>Workspace</span>
|
|
||||||
</Show>
|
|
||||||
<div class={styles.navScroller}>
|
|
||||||
<ul class={styles.navList} role="list">
|
|
||||||
<For each={workspaceStaticItems}>{(item): JSX.Element => <WorkspaceHomeEntry item={item} />}</For>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<Show when={!props.collapsed}>
|
<Show when={!props.collapsed}>
|
||||||
<div class={styles.treeSectionLabel}>Items</div>
|
<span class={styles.sectionLabel}>Workspace</span>
|
||||||
</Show>
|
</Show>
|
||||||
|
<div class={styles.navScroller}>
|
||||||
|
<ul class={styles.navList} role="list">
|
||||||
|
<For each={workspaceStaticItems}>
|
||||||
|
{(item): JSX.Element => (
|
||||||
|
<WorkspaceHomeEntry
|
||||||
|
item={item}
|
||||||
|
onOpenContextMenu={contextMenu.openMenu}
|
||||||
|
onOpenContextMenuFromKeyboard={contextMenu.openMenuFromElement}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</For>
|
||||||
|
</ul>
|
||||||
|
|
||||||
<WorkspaceTreeBranch nodes={workspaceTree} />
|
<Show when={!props.collapsed}>
|
||||||
|
<div class={styles.treeSectionLabel}>Items</div>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<WorkspaceTreeBranch
|
||||||
|
nodes={workspaceTree}
|
||||||
|
onOpenContextMenu={contextMenu.openMenu}
|
||||||
|
onOpenContextMenuFromKeyboard={contextMenu.openMenuFromElement}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</aside>
|
||||||
</aside>
|
|
||||||
|
<WorkspaceContextMenu
|
||||||
|
target={contextMenuTarget()}
|
||||||
|
position={contextMenuPosition()}
|
||||||
|
menuRef={contextMenu.setMenuRef}
|
||||||
|
onClose={contextMenu.closeMenu}
|
||||||
|
onSelect={handleContextActionSelect}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
73
Frontend/src/components/shell/createLongPressGesture.ts
Normal file
73
Frontend/src/components/shell/createLongPressGesture.ts
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
import type { JSX } from "solid-js";
|
||||||
|
|
||||||
|
type PointerHandler = NonNullable<JSX.DOMAttributes<Element>["onPointerDown"]>;
|
||||||
|
|
||||||
|
type LongPressGestureOptions = {
|
||||||
|
onLongPress: () => void;
|
||||||
|
delay?: number;
|
||||||
|
movementThreshold?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
type LongPressGestureHandlers = {
|
||||||
|
onPointerDown: PointerHandler;
|
||||||
|
onPointerMove: PointerHandler;
|
||||||
|
onPointerUp: PointerHandler;
|
||||||
|
onPointerCancel: PointerHandler;
|
||||||
|
onPointerLeave: PointerHandler;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createLongPressGesture = (options: LongPressGestureOptions): LongPressGestureHandlers => {
|
||||||
|
let timeoutId: number | undefined;
|
||||||
|
let originX = 0;
|
||||||
|
let originY = 0;
|
||||||
|
|
||||||
|
const delay = options.delay ?? 420;
|
||||||
|
const movementThreshold = options.movementThreshold ?? 10;
|
||||||
|
|
||||||
|
const clearPendingLongPress = (): void => {
|
||||||
|
if (typeof timeoutId === "number") {
|
||||||
|
window.clearTimeout(timeoutId);
|
||||||
|
timeoutId = undefined;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPointerDown: PointerHandler = (event): void => {
|
||||||
|
// Mobile long-press should only respond to the primary touch/pen pointer.
|
||||||
|
if (event.pointerType === "mouse" || !event.isPrimary) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
originX = event.clientX;
|
||||||
|
originY = event.clientY;
|
||||||
|
clearPendingLongPress();
|
||||||
|
timeoutId = window.setTimeout(() => {
|
||||||
|
timeoutId = undefined;
|
||||||
|
options.onLongPress();
|
||||||
|
}, delay);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPointerMove: PointerHandler = (event): void => {
|
||||||
|
if (typeof timeoutId !== "number") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const deltaX = Math.abs(event.clientX - originX);
|
||||||
|
const deltaY = Math.abs(event.clientY - originY);
|
||||||
|
|
||||||
|
if (deltaX > movementThreshold || deltaY > movementThreshold) {
|
||||||
|
clearPendingLongPress();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onPointerUp: PointerHandler = (): void => {
|
||||||
|
clearPendingLongPress();
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
onPointerDown,
|
||||||
|
onPointerMove,
|
||||||
|
onPointerUp,
|
||||||
|
onPointerCancel: onPointerUp,
|
||||||
|
onPointerLeave: onPointerUp,
|
||||||
|
};
|
||||||
|
};
|
||||||
@@ -82,6 +82,12 @@ export type SidebarItem = {
|
|||||||
meta?: string;
|
meta?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type WorkspaceStaticKind = "workspace" | "home" | "settings";
|
||||||
|
|
||||||
|
export type WorkspaceStaticItem = SidebarItem & {
|
||||||
|
contextKind: WorkspaceStaticKind;
|
||||||
|
};
|
||||||
|
|
||||||
export type WorkspaceTreeNode = {
|
export type WorkspaceTreeNode = {
|
||||||
id: string;
|
id: string;
|
||||||
label: string;
|
label: string;
|
||||||
@@ -111,6 +117,69 @@ export type MobileBottomNavItem = {
|
|||||||
active?: boolean;
|
active?: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuTarget = {
|
||||||
|
id: string;
|
||||||
|
label: string;
|
||||||
|
kind: WorkspaceStaticKind | WorkspaceTreeNode["kind"];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuAction = {
|
||||||
|
id: string;
|
||||||
|
label: string;
|
||||||
|
tone?: "default" | "danger";
|
||||||
|
shortcut?: WorkspaceContextMenuShortcut;
|
||||||
|
children?: readonly WorkspaceContextMenuAction[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuShortcutModifier = "meta" | "alt" | "shift";
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuShortcutKey = "b" | "c" | "d" | "delete" | "enter" | "f" | "m" | "r";
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuShortcut = {
|
||||||
|
modifiers?: readonly WorkspaceContextMenuShortcutModifier[];
|
||||||
|
key: WorkspaceContextMenuShortcutKey;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WorkspaceContextMenuSection = {
|
||||||
|
id: string;
|
||||||
|
label?: string;
|
||||||
|
items: readonly WorkspaceContextMenuAction[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getWorkspaceContextMenuEyebrow = (target: WorkspaceContextMenuTarget): string => {
|
||||||
|
switch (target.kind) {
|
||||||
|
case "workspace":
|
||||||
|
case "home":
|
||||||
|
return "Workspace";
|
||||||
|
case "settings":
|
||||||
|
return "Configuration";
|
||||||
|
case "folder":
|
||||||
|
return "Folder";
|
||||||
|
case "board":
|
||||||
|
return "Board";
|
||||||
|
case "doc":
|
||||||
|
return "Doc";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createWorkspaceSurfaceTarget = (workspace: ActiveProject): WorkspaceContextMenuTarget => ({
|
||||||
|
id: `workspace-${workspace.id}`,
|
||||||
|
label: workspace.name,
|
||||||
|
kind: "workspace",
|
||||||
|
});
|
||||||
|
|
||||||
|
export const createWorkspaceStaticTarget = (item: WorkspaceStaticItem): WorkspaceContextMenuTarget => ({
|
||||||
|
id: item.id,
|
||||||
|
label: item.label,
|
||||||
|
kind: item.contextKind,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const createWorkspaceTreeTarget = (node: WorkspaceTreeNode): WorkspaceContextMenuTarget => ({
|
||||||
|
id: node.id,
|
||||||
|
label: node.label,
|
||||||
|
kind: node.kind,
|
||||||
|
});
|
||||||
|
|
||||||
export type NotificationItem = {
|
export type NotificationItem = {
|
||||||
id: string;
|
id: string;
|
||||||
title: string;
|
title: string;
|
||||||
@@ -190,9 +259,9 @@ export const departmentItems: readonly DepartmentItem[] = [
|
|||||||
|
|
||||||
// Sidebar and topbar scaffold data
|
// Sidebar and topbar scaffold data
|
||||||
// These static entries stay pinned in both desktop and mobile workspace navigation.
|
// These static entries stay pinned in both desktop and mobile workspace navigation.
|
||||||
export const workspaceStaticItems: readonly SidebarItem[] = [
|
export const workspaceStaticItems: readonly WorkspaceStaticItem[] = [
|
||||||
{ id: "home", label: "Home", icon: Home, active: true },
|
{ id: "home", label: "Home", icon: Home, active: true, contextKind: "home" },
|
||||||
{ id: "workspace-settings", label: "Settings", icon: Settings },
|
{ id: "workspace-settings", label: "Settings", icon: Settings, contextKind: "settings" },
|
||||||
] as const;
|
] as const;
|
||||||
|
|
||||||
// Freeform workspace tree scaffold: folders, boards, and docs are first-class siblings.
|
// Freeform workspace tree scaffold: folders, boards, and docs are first-class siblings.
|
||||||
@@ -240,6 +309,129 @@ export const mobileBottomNavItems: readonly MobileBottomNavItem[] = [
|
|||||||
{ id: "browse", label: "Browse", icon: Folder },
|
{ id: "browse", label: "Browse", icon: Folder },
|
||||||
] as const;
|
] as const;
|
||||||
|
|
||||||
|
// Initial context-menu IA scaffold. Behavior wiring can evolve later, but the
|
||||||
|
// target kinds and action grouping should stay shared across workspace surfaces.
|
||||||
|
export const getWorkspaceContextMenuSections = (
|
||||||
|
target: WorkspaceContextMenuTarget,
|
||||||
|
): readonly WorkspaceContextMenuSection[] => {
|
||||||
|
const createActions = [
|
||||||
|
{ id: "new-folder", label: "New folder", shortcut: { modifiers: ["alt"], key: "f" } },
|
||||||
|
{ id: "new-board", label: "New board", shortcut: { modifiers: ["alt"], key: "b" } },
|
||||||
|
{ id: "new-doc", label: "New doc", shortcut: { modifiers: ["alt"], key: "d" } },
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
const createSubmenuAction = {
|
||||||
|
id: "create",
|
||||||
|
label: "Create",
|
||||||
|
children: createActions,
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
switch (target.kind) {
|
||||||
|
case "workspace":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "create",
|
||||||
|
label: undefined,
|
||||||
|
items: [createSubmenuAction],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "workspace",
|
||||||
|
label: undefined,
|
||||||
|
items: [
|
||||||
|
{ id: "rename-workspace", label: "Rename workspace", shortcut: { key: "enter" } },
|
||||||
|
{ id: "copy-workspace-link", label: "Copy link", shortcut: { modifiers: ["meta"], key: "c" } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
case "home":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "create",
|
||||||
|
label: undefined,
|
||||||
|
items: [createSubmenuAction],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "workspace",
|
||||||
|
label: undefined,
|
||||||
|
items: [{ id: "open-home", label: "Open home", shortcut: { key: "enter" } }],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
case "settings":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "settings",
|
||||||
|
label: undefined,
|
||||||
|
items: [
|
||||||
|
{ id: "open-settings", label: "Open settings", shortcut: { key: "enter" } },
|
||||||
|
{ id: "copy-settings-link", label: "Copy link", shortcut: { modifiers: ["meta"], key: "c" } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
case "folder":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "open",
|
||||||
|
items: [
|
||||||
|
{ id: "open-folder", label: "Open folder", shortcut: { key: "enter" } },
|
||||||
|
{ id: "rename-folder", label: "Rename", shortcut: { modifiers: ["meta"], key: "r" } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "create",
|
||||||
|
label: undefined,
|
||||||
|
items: [createSubmenuAction],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "organize",
|
||||||
|
label: undefined,
|
||||||
|
items: [
|
||||||
|
{ id: "duplicate-folder", label: "Duplicate", shortcut: { modifiers: ["meta"], key: "d" } },
|
||||||
|
{ id: "move-folder", label: "Move", shortcut: { modifiers: ["meta"], key: "m" } },
|
||||||
|
{ id: "delete-folder", label: "Delete", shortcut: { modifiers: ["meta"], key: "delete" }, tone: "danger" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
case "board":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "board",
|
||||||
|
items: [
|
||||||
|
{ id: "open-board", label: "Open board", shortcut: { key: "enter" } },
|
||||||
|
{ id: "rename-board", label: "Rename", shortcut: { modifiers: ["meta"], key: "r" } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "organize",
|
||||||
|
label: undefined,
|
||||||
|
items: [
|
||||||
|
{ id: "duplicate-board", label: "Duplicate", shortcut: { modifiers: ["meta"], key: "d" } },
|
||||||
|
{ id: "move-board", label: "Move", shortcut: { modifiers: ["meta"], key: "m" } },
|
||||||
|
{ id: "delete-board", label: "Delete", shortcut: { modifiers: ["meta"], key: "delete" }, tone: "danger" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
case "doc":
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "doc",
|
||||||
|
items: [
|
||||||
|
{ id: "open-doc", label: "Open doc", shortcut: { key: "enter" } },
|
||||||
|
{ id: "rename-doc", label: "Rename", shortcut: { modifiers: ["meta"], key: "r" } },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "organize",
|
||||||
|
label: undefined,
|
||||||
|
items: [
|
||||||
|
{ id: "duplicate-doc", label: "Duplicate", shortcut: { modifiers: ["meta"], key: "d" } },
|
||||||
|
{ id: "move-doc", label: "Move", shortcut: { modifiers: ["meta"], key: "m" } },
|
||||||
|
{ id: "delete-doc", label: "Delete", shortcut: { modifiers: ["meta"], key: "delete" }, tone: "danger" },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as const;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export const topBarActions: readonly TopBarAction[] = [
|
export const topBarActions: readonly TopBarAction[] = [
|
||||||
{ id: "search", label: "Search", icon: Search },
|
{ id: "search", label: "Search", icon: Search },
|
||||||
] as const;
|
] as const;
|
||||||
|
|||||||
Reference in New Issue
Block a user