Compare commits
12 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| aafdb4af72 | |||
| 67a24ddf26 | |||
| 2a928886b9 | |||
| 4d192654df | |||
| a634c2e237 | |||
| 9b0b448cb1 | |||
| ba4aa912af | |||
| ca4f994b55 | |||
| 74844492eb | |||
| c42cedbf94 | |||
| 749725f091 | |||
| f3a8ad057a |
BIN
.playwright-mcp/after-scroll.png
Normal file
BIN
.playwright-mcp/after-scroll.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 125 KiB |
BIN
.playwright-mcp/scroll-containment-check.png
Normal file
BIN
.playwright-mcp/scroll-containment-check.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 125 KiB |
47
changelog.md
47
changelog.md
@@ -1,5 +1,52 @@
|
||||
# Changelog
|
||||
|
||||
## 2026-01-04 - 3.31.0 - feat(dees-input-list)
|
||||
enhance drag-and-drop reordering for dees-input-list and migrate tests to chromium runner
|
||||
|
||||
- Add rich drag state to dees-input-list: dragStartY, dragCurrentY, targetIndex, itemHeight and originalItemRects for accurate hit detection.
|
||||
- Introduce bound global drag handlers and centralized global drag end/cleanup logic (handleGlobalDragOver / handleGlobalDragEnd).
|
||||
- Improve drag visuals and animations: 'dragging', 'move-up', 'move-down' transforms, box-shadow, and smoother transitions; prevent hover styling while dragging.
|
||||
- Move reorder logic away from per-item drop to global drag end to avoid race/positioning issues and ensure consistent reflow and cleanup.
|
||||
- Migrate many browser test files to chromium-specific variants (added *.chromium.ts) and remove duplicate browser test counterparts.
|
||||
|
||||
## 2026-01-04 - 3.30.1 - fix(dees-statsgrid)
|
||||
refine spacing, sizing, and colors in dees-statsgrid for a tighter, more compact appearance
|
||||
|
||||
- Reduce global spacing and sizing variables (grid-gap 16→12, tile-padding 24→16, header-spacing 16→12, content-min-height 48→40, description-spacing 12→8, border-radius 8→6).
|
||||
- Adjust typographic scale (value-font-size 30→26, unit-font-size 16→14, label-font-size 13→12, title-font-size 14→13).
|
||||
- Switch color tokens to neutral hex values and tighten hover/box-shadow (tile border and backgrounds updated from HSL to hex, hover bg to #fafafa/#0d0d0d, border-color and shadow reduced).
|
||||
- Downsize graphical elements: gauge and SVG dimensions (width 140→120, height 80→70), stroke-widths 8→6, radius 48→40.
|
||||
- Slim down percentage bar and trend visuals (percentage bar height 8→6, border-radius 4→3, trend stroke-width 2→1.5, trend fill moved to RGBA).
|
||||
- No functional or API changes — purely visual/CSS and SVG adjustments.
|
||||
|
||||
## 2026-01-03 - 3.30.0 - feat(appui)
|
||||
add dees-appui-bottombar component with config, programmatic API, demo and docs
|
||||
|
||||
- Adds a new dees-appui-bottombar web component (ts_web/elements/00group-appui/dees-appui-bottombar/) implementing widget and action management (add/update/remove/get/clear).
|
||||
- Introduces bottom bar types and API in ts_web/elements/interfaces/appconfig.ts (IBottomBarWidget, IBottomBarAction, IBottomBarConfig, IBottomBarAPI) and extends the app config/type to include bottomBar and bottomBar APIs.
|
||||
- Integrates the bottom bar into dees-appui: imports and registers component, renders conditionally, exposes bottomBar proxy API, visibility controls (set/getBottomBarVisible), and wires initial config to populate widgets/actions.
|
||||
- Updates layout/styles (reduces main grid height to account for 24px fixed bottom bar and adds bottombar-hidden attribute handling) and exports component from the appui index.
|
||||
- Adds interactive demos (dees-appui-bottombar.demo.ts and integration demo) and documents usage and API in readme.hints.md.
|
||||
|
||||
## 2026-01-03 - 3.29.3 - fix(elements/appui)
|
||||
prevent scroll chaining on app UI components by adding overscroll-behavior: contain
|
||||
|
||||
- Added CSS overscroll-behavior: contain to activity log, main menu, secondary menu, profile dropdown, and tabs components to prevent scroll chaining and unintended body scrolling on touch/trackpad.
|
||||
- Styling-only change; no public API or behavioral changes beyond scroll handling.
|
||||
- Bump patch version from 3.29.2 to 3.29.3.
|
||||
|
||||
## 2026-01-03 - 3.29.2 - fix(dees-appui)
|
||||
set min-height: 0 on .maingrid > dees-appui-maincontent to prevent layout overflow in flex container
|
||||
|
||||
- Added min-height: 0 to .maingrid > dees-appui-maincontent in ts_web/elements/00group-appui/dees-appui/dees-appui.ts to prevent unwanted growth/overflow when used inside a flex container.
|
||||
- Pure CSS/layout fix — no API or behavior changes to components.
|
||||
|
||||
## 2026-01-03 - 3.29.1 - fix(dees-appui)
|
||||
prevent main grid overflow by adding overflow:hidden; and add Playwright scroll containment screenshots
|
||||
|
||||
- Add overflow: hidden to .maingrid in ts_web/elements/00group-appui/dees-appui/dees-appui.ts to prevent content from escaping during grid-template-columns transitions.
|
||||
- Add Playwright artifacts: .playwright-mcp/after-scroll.png and .playwright-mcp/scroll-containment-check.png (screenshots for scroll containment testing).
|
||||
|
||||
## 2026-01-03 - 3.29.0 - feat(docs)
|
||||
add documentation for new input components, activity log features, theming, and expand DeesAppui docs
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@design.estate/dees-catalog",
|
||||
"version": "3.29.0",
|
||||
"version": "3.31.0",
|
||||
"private": false,
|
||||
"description": "A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.",
|
||||
"main": "dist_ts_web/index.js",
|
||||
|
||||
112
readme.hints.md
112
readme.hints.md
@@ -800,4 +800,114 @@ html`
|
||||
- **External Router Support**: Integrate with Angular Router or other frameworks
|
||||
- **State Persistence**: Save/restore collapsed menus, selections, and current view
|
||||
- **View-specific Menus**: Each view can define its own secondary menu and tabs
|
||||
- **Full Backward Compatibility**: Existing code continues to work
|
||||
- **Full Backward Compatibility**: Existing code continues to work
|
||||
|
||||
## AppUI Bottom Bar (2026-01-03)
|
||||
|
||||
Added a new `dees-appui-bottombar` component similar to `dees-workspace-bottombar`, providing a 24px fixed-height status bar at the bottom of the app shell.
|
||||
|
||||
### Features:
|
||||
- **Generic status widgets**: Configurable widgets with icon, label, status colors, loading spinner
|
||||
- **App-specific actions**: Quick action buttons with icons and tooltips
|
||||
- **Always visible**: Fixed 24px height at the bottom of the app
|
||||
- **Status colors**: idle, active (blue), success (green), warning (yellow), error (red)
|
||||
- **Context menus**: Widgets can have right-click context menus
|
||||
|
||||
### New Interfaces (in `interfaces/appconfig.ts`):
|
||||
|
||||
```typescript
|
||||
interface IBottomBarWidget {
|
||||
id: string;
|
||||
iconName?: string;
|
||||
label?: string;
|
||||
status?: 'idle' | 'active' | 'success' | 'warning' | 'error';
|
||||
tooltip?: string;
|
||||
loading?: boolean;
|
||||
onClick?: () => void;
|
||||
contextMenuItems?: IBottomBarContextMenuItem[];
|
||||
position?: 'left' | 'right';
|
||||
order?: number;
|
||||
}
|
||||
|
||||
interface IBottomBarAction {
|
||||
id: string;
|
||||
iconName: string;
|
||||
tooltip?: string;
|
||||
onClick: () => void | Promise<void>;
|
||||
disabled?: boolean;
|
||||
position?: 'left' | 'right';
|
||||
}
|
||||
|
||||
interface IBottomBarConfig {
|
||||
visible?: boolean;
|
||||
widgets?: IBottomBarWidget[];
|
||||
actions?: IBottomBarAction[];
|
||||
}
|
||||
```
|
||||
|
||||
### Usage via configure():
|
||||
|
||||
```typescript
|
||||
const config: IAppConfig = {
|
||||
// ... other config
|
||||
bottomBar: {
|
||||
visible: true,
|
||||
widgets: [
|
||||
{
|
||||
id: 'status',
|
||||
iconName: 'lucide:activity',
|
||||
label: 'System Online',
|
||||
status: 'success',
|
||||
tooltip: 'All systems operational',
|
||||
onClick: () => console.log('Status clicked'),
|
||||
},
|
||||
{
|
||||
id: 'notifications',
|
||||
iconName: 'lucide:bell',
|
||||
label: '3 notifications',
|
||||
status: 'warning',
|
||||
position: 'left',
|
||||
},
|
||||
{
|
||||
id: 'version',
|
||||
iconName: 'lucide:gitBranch',
|
||||
label: 'v1.2.3',
|
||||
position: 'right',
|
||||
},
|
||||
],
|
||||
actions: [
|
||||
{
|
||||
id: 'terminal',
|
||||
iconName: 'lucide:terminal',
|
||||
tooltip: 'Open Terminal',
|
||||
position: 'right',
|
||||
onClick: () => console.log('Terminal clicked'),
|
||||
},
|
||||
],
|
||||
},
|
||||
};
|
||||
```
|
||||
|
||||
### Programmatic API:
|
||||
|
||||
```typescript
|
||||
// Add/update/remove widgets
|
||||
appui.bottomBar.addWidget({ id: 'status', ... });
|
||||
appui.bottomBar.updateWidget('status', { status: 'error', label: 'Error!' });
|
||||
appui.bottomBar.removeWidget('status');
|
||||
appui.bottomBar.clearWidgets();
|
||||
|
||||
// Add/remove actions
|
||||
appui.bottomBar.addAction({ id: 'refresh', iconName: 'lucide:refreshCw', ... });
|
||||
appui.bottomBar.removeAction('refresh');
|
||||
appui.bottomBar.clearActions();
|
||||
|
||||
// Visibility control
|
||||
appui.setBottomBarVisible(false);
|
||||
appui.getBottomBarVisible();
|
||||
```
|
||||
|
||||
### Files:
|
||||
- `ts_web/elements/00group-appui/dees-appui-bottombar/dees-appui-bottombar.ts` - Main component
|
||||
- `ts_web/elements/00group-appui/dees-appui-bottombar/dees-appui-bottombar.demo.ts` - Demo
|
||||
- `ts_web/elements/interfaces/appconfig.ts` - New interfaces added
|
||||
@@ -3,6 +3,6 @@
|
||||
*/
|
||||
export const commitinfo = {
|
||||
name: '@design.estate/dees-catalog',
|
||||
version: '3.29.0',
|
||||
version: '3.31.0',
|
||||
description: 'A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.'
|
||||
}
|
||||
|
||||
@@ -122,6 +122,7 @@ export class DeesAppuiActivitylog extends DeesElement implements IActivityLogAPI
|
||||
width: 100%;
|
||||
padding: 8px 0;
|
||||
overflow-y: auto;
|
||||
overscroll-behavior: contain;
|
||||
scrollbar-width: thin;
|
||||
scrollbar-color: ${cssManager.bdTheme('#d4d4d4', '#333333')} transparent;
|
||||
}
|
||||
|
||||
@@ -0,0 +1,210 @@
|
||||
import { html } from '@design.estate/dees-element';
|
||||
import type { DeesAppuiBottombar } from './dees-appui-bottombar.js';
|
||||
import '@design.estate/dees-wcctools/demotools';
|
||||
|
||||
export const demoFunc = () => {
|
||||
return html`
|
||||
<dees-demowrapper>
|
||||
<style>
|
||||
.demo-container {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 24px;
|
||||
padding: 24px;
|
||||
background: #1a1a1a;
|
||||
}
|
||||
|
||||
.demo-section {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.demo-label {
|
||||
font-size: 12px;
|
||||
color: #737373;
|
||||
font-family: 'Geist Sans', sans-serif;
|
||||
}
|
||||
|
||||
.demo-bottombar-wrapper {
|
||||
border: 1px solid hsl(0 0% 20%);
|
||||
border-radius: 4px;
|
||||
overflow: hidden;
|
||||
}
|
||||
</style>
|
||||
<div class="demo-container">
|
||||
<div class="demo-section">
|
||||
<div class="demo-label">Bottom bar with status widgets and actions</div>
|
||||
<div class="demo-bottombar-wrapper">
|
||||
<dees-appui-bottombar
|
||||
id="demo-bottombar"
|
||||
></dees-appui-bottombar>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="demo-section">
|
||||
<div class="demo-label">Controls</div>
|
||||
<div style="display: flex; gap: 8px; flex-wrap: wrap;">
|
||||
<button onclick="addSuccessWidget()">Add Success Widget</button>
|
||||
<button onclick="addWarningWidget()">Add Warning Widget</button>
|
||||
<button onclick="addErrorWidget()">Add Error Widget</button>
|
||||
<button onclick="addLoadingWidget()">Add Loading Widget</button>
|
||||
<button onclick="addRightWidget()">Add Right Widget</button>
|
||||
<button onclick="addAction()">Add Action</button>
|
||||
<button onclick="clearAll()">Clear All</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<script type="module">
|
||||
const bottombar = document.getElementById('demo-bottombar');
|
||||
|
||||
// Wait for component to initialize
|
||||
await bottombar.updateComplete;
|
||||
|
||||
// Add initial widgets
|
||||
bottombar.addWidget({
|
||||
id: 'status',
|
||||
iconName: 'lucide:activity',
|
||||
label: 'System Online',
|
||||
status: 'success',
|
||||
tooltip: 'All systems operational',
|
||||
onClick: () => console.log('Status clicked'),
|
||||
contextMenuItems: [
|
||||
{ name: 'View Details', iconName: 'lucide:info', action: () => alert('System details') },
|
||||
{ divider: true },
|
||||
{ name: 'Refresh Status', iconName: 'lucide:refreshCw', action: () => alert('Refreshing...') },
|
||||
],
|
||||
});
|
||||
|
||||
bottombar.addWidget({
|
||||
id: 'notifications',
|
||||
iconName: 'lucide:bell',
|
||||
label: '3 notifications',
|
||||
status: 'warning',
|
||||
tooltip: 'You have unread notifications',
|
||||
onClick: () => console.log('Notifications clicked'),
|
||||
});
|
||||
|
||||
bottombar.addWidget({
|
||||
id: 'version',
|
||||
iconName: 'lucide:gitBranch',
|
||||
label: 'v1.2.3',
|
||||
tooltip: 'Current version',
|
||||
position: 'right',
|
||||
onClick: () => console.log('Version clicked'),
|
||||
});
|
||||
|
||||
// Add initial actions
|
||||
bottombar.addAction({
|
||||
id: 'settings',
|
||||
iconName: 'lucide:settings',
|
||||
tooltip: 'Settings',
|
||||
position: 'right',
|
||||
onClick: () => alert('Settings clicked'),
|
||||
});
|
||||
|
||||
bottombar.addAction({
|
||||
id: 'help',
|
||||
iconName: 'lucide:helpCircle',
|
||||
tooltip: 'Help',
|
||||
position: 'right',
|
||||
onClick: () => alert('Help clicked'),
|
||||
});
|
||||
|
||||
// Demo control functions
|
||||
let widgetCounter = 0;
|
||||
let actionCounter = 0;
|
||||
|
||||
window.addSuccessWidget = () => {
|
||||
widgetCounter++;
|
||||
bottombar.addWidget({
|
||||
id: 'success-' + widgetCounter,
|
||||
iconName: 'lucide:checkCircle',
|
||||
label: 'Success ' + widgetCounter,
|
||||
status: 'success',
|
||||
tooltip: 'Success widget',
|
||||
onClick: () => bottombar.removeWidget('success-' + widgetCounter),
|
||||
});
|
||||
};
|
||||
|
||||
window.addWarningWidget = () => {
|
||||
widgetCounter++;
|
||||
bottombar.addWidget({
|
||||
id: 'warning-' + widgetCounter,
|
||||
iconName: 'lucide:alertTriangle',
|
||||
label: 'Warning ' + widgetCounter,
|
||||
status: 'warning',
|
||||
tooltip: 'Warning widget',
|
||||
onClick: () => bottombar.removeWidget('warning-' + widgetCounter),
|
||||
});
|
||||
};
|
||||
|
||||
window.addErrorWidget = () => {
|
||||
widgetCounter++;
|
||||
bottombar.addWidget({
|
||||
id: 'error-' + widgetCounter,
|
||||
iconName: 'lucide:xCircle',
|
||||
label: 'Error ' + widgetCounter,
|
||||
status: 'error',
|
||||
tooltip: 'Error widget',
|
||||
onClick: () => bottombar.removeWidget('error-' + widgetCounter),
|
||||
});
|
||||
};
|
||||
|
||||
window.addLoadingWidget = () => {
|
||||
widgetCounter++;
|
||||
const id = 'loading-' + widgetCounter;
|
||||
bottombar.addWidget({
|
||||
id: id,
|
||||
iconName: 'lucide:loader2',
|
||||
label: 'Loading...',
|
||||
status: 'active',
|
||||
loading: true,
|
||||
tooltip: 'Loading in progress',
|
||||
});
|
||||
|
||||
// Simulate completion after 3 seconds
|
||||
setTimeout(() => {
|
||||
bottombar.updateWidget(id, {
|
||||
iconName: 'lucide:check',
|
||||
label: 'Done!',
|
||||
status: 'success',
|
||||
loading: false,
|
||||
});
|
||||
}, 3000);
|
||||
};
|
||||
|
||||
window.addRightWidget = () => {
|
||||
widgetCounter++;
|
||||
bottombar.addWidget({
|
||||
id: 'right-' + widgetCounter,
|
||||
iconName: 'lucide:info',
|
||||
label: 'Right ' + widgetCounter,
|
||||
position: 'right',
|
||||
onClick: () => bottombar.removeWidget('right-' + widgetCounter),
|
||||
});
|
||||
};
|
||||
|
||||
window.addAction = () => {
|
||||
actionCounter++;
|
||||
bottombar.addAction({
|
||||
id: 'action-' + actionCounter,
|
||||
iconName: 'lucide:zap',
|
||||
tooltip: 'Action ' + actionCounter,
|
||||
onClick: () => {
|
||||
alert('Action ' + actionCounter + ' clicked');
|
||||
bottombar.removeAction('action-' + actionCounter);
|
||||
},
|
||||
});
|
||||
};
|
||||
|
||||
window.clearAll = () => {
|
||||
bottombar.clearWidgets();
|
||||
bottombar.clearActions();
|
||||
widgetCounter = 0;
|
||||
actionCounter = 0;
|
||||
};
|
||||
</script>
|
||||
</dees-demowrapper>
|
||||
`;
|
||||
};
|
||||
@@ -0,0 +1,314 @@
|
||||
import {
|
||||
DeesElement,
|
||||
type TemplateResult,
|
||||
customElement,
|
||||
html,
|
||||
css,
|
||||
cssManager,
|
||||
state,
|
||||
} from '@design.estate/dees-element';
|
||||
import { themeDefaultStyles } from '../../00theme.js';
|
||||
import '../../dees-icon/dees-icon.js';
|
||||
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||
import type {
|
||||
IBottomBarWidget,
|
||||
IBottomBarAction,
|
||||
IBottomBarAPI,
|
||||
} from '../../interfaces/appconfig.js';
|
||||
import { demoFunc } from './dees-appui-bottombar.demo.js';
|
||||
|
||||
declare global {
|
||||
interface HTMLElementTagNameMap {
|
||||
'dees-appui-bottombar': DeesAppuiBottombar;
|
||||
}
|
||||
}
|
||||
|
||||
@customElement('dees-appui-bottombar')
|
||||
export class DeesAppuiBottombar extends DeesElement implements IBottomBarAPI {
|
||||
public static demo = demoFunc;
|
||||
|
||||
// INSTANCE PROPERTIES
|
||||
@state()
|
||||
accessor widgets: IBottomBarWidget[] = [];
|
||||
|
||||
@state()
|
||||
accessor actions: IBottomBarAction[] = [];
|
||||
|
||||
public static styles = [
|
||||
themeDefaultStyles,
|
||||
cssManager.defaultStyles,
|
||||
css`
|
||||
:host {
|
||||
display: block;
|
||||
height: 24px;
|
||||
flex-shrink: 0;
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.bottom-bar {
|
||||
height: 24px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: 0 8px;
|
||||
gap: 4px;
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 94%)', 'hsl(0 0% 6%)')};
|
||||
border-top: 1px solid ${cssManager.bdTheme('hsl(0 0% 85%)', 'hsl(0 0% 15%)')};
|
||||
font-size: 11px;
|
||||
color: ${cssManager.bdTheme('hsl(0 0% 40%)', 'hsl(0 0% 60%)')};
|
||||
}
|
||||
|
||||
.widget {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 4px;
|
||||
padding: 2px 6px;
|
||||
border-radius: 3px;
|
||||
cursor: pointer;
|
||||
transition: background 0.15s ease, color 0.15s ease;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.widget:hover {
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 88%)', 'hsl(0 0% 12%)')};
|
||||
color: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 80%)')};
|
||||
}
|
||||
|
||||
.widget dees-icon {
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.widget-separator {
|
||||
width: 1px;
|
||||
height: 14px;
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 80%)', 'hsl(0 0% 20%)')};
|
||||
margin: 0 4px;
|
||||
}
|
||||
|
||||
/* Status colors matching dees-workspace-bottombar */
|
||||
.widget.active {
|
||||
color: ${cssManager.bdTheme('hsl(210 100% 45%)', 'hsl(210 100% 60%)')};
|
||||
}
|
||||
|
||||
.widget.success {
|
||||
color: ${cssManager.bdTheme('hsl(142 70% 35%)', 'hsl(142 70% 50%)')};
|
||||
}
|
||||
|
||||
.widget.warning {
|
||||
color: ${cssManager.bdTheme('hsl(38 92% 45%)', 'hsl(38 92% 55%)')};
|
||||
}
|
||||
|
||||
.widget.error {
|
||||
color: ${cssManager.bdTheme('hsl(0 70% 50%)', 'hsl(0 70% 60%)')};
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
from { transform: rotate(0deg); }
|
||||
to { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
.spinning {
|
||||
animation: spin 1s linear infinite;
|
||||
}
|
||||
|
||||
.spacer {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.action-button {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
border-radius: 3px;
|
||||
cursor: pointer;
|
||||
transition: background 0.15s ease;
|
||||
color: ${cssManager.bdTheme('hsl(0 0% 40%)', 'hsl(0 0% 60%)')};
|
||||
}
|
||||
|
||||
.action-button:hover {
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 88%)', 'hsl(0 0% 12%)')};
|
||||
color: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 80%)')};
|
||||
}
|
||||
|
||||
.action-button.disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.action-button.disabled:hover {
|
||||
background: transparent;
|
||||
color: ${cssManager.bdTheme('hsl(0 0% 40%)', 'hsl(0 0% 60%)')};
|
||||
}
|
||||
`,
|
||||
];
|
||||
|
||||
public render(): TemplateResult {
|
||||
const leftWidgets = this.widgets
|
||||
.filter(w => w.position !== 'right')
|
||||
.sort((a, b) => (a.order || 0) - (b.order || 0));
|
||||
|
||||
const rightWidgets = this.widgets
|
||||
.filter(w => w.position === 'right')
|
||||
.sort((a, b) => (a.order || 0) - (b.order || 0));
|
||||
|
||||
const leftActions = this.actions.filter(a => a.position === 'left');
|
||||
const rightActions = this.actions.filter(a => a.position !== 'left');
|
||||
|
||||
return html`
|
||||
<div class="bottom-bar">
|
||||
<!-- Left actions -->
|
||||
${leftActions.map(action => this.renderAction(action))}
|
||||
|
||||
<!-- Left widgets -->
|
||||
${leftWidgets.map((widget, index) => html`
|
||||
${index > 0 || leftActions.length > 0 ? html`<div class="widget-separator"></div>` : ''}
|
||||
${this.renderWidget(widget)}
|
||||
`)}
|
||||
|
||||
<div class="spacer"></div>
|
||||
|
||||
<!-- Right widgets -->
|
||||
${rightWidgets.map((widget, index) => html`
|
||||
${this.renderWidget(widget)}
|
||||
${index < rightWidgets.length - 1 || rightActions.length > 0 ? html`<div class="widget-separator"></div>` : ''}
|
||||
`)}
|
||||
|
||||
<!-- Right actions -->
|
||||
${rightActions.map(action => this.renderAction(action))}
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
private renderWidget(widget: IBottomBarWidget): TemplateResult {
|
||||
const statusClass = widget.status && widget.status !== 'idle' ? widget.status : '';
|
||||
const iconName = widget.iconName
|
||||
? (widget.iconName.startsWith('lucide:') ? widget.iconName : `lucide:${widget.iconName}`)
|
||||
: '';
|
||||
|
||||
return html`
|
||||
<div
|
||||
class="widget ${statusClass}"
|
||||
title="${widget.tooltip || ''}"
|
||||
@click=${() => widget.onClick?.()}
|
||||
@contextmenu=${(e: MouseEvent) => this.handleWidgetContextMenu(e, widget)}
|
||||
>
|
||||
${iconName ? html`
|
||||
<dees-icon
|
||||
.icon=${iconName}
|
||||
iconSize="12"
|
||||
class="${widget.loading ? 'spinning' : ''}"
|
||||
></dees-icon>
|
||||
` : ''}
|
||||
${widget.label ? html`<span>${widget.label}</span>` : ''}
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
private renderAction(action: IBottomBarAction): TemplateResult {
|
||||
const iconName = action.iconName.startsWith('lucide:')
|
||||
? action.iconName
|
||||
: `lucide:${action.iconName}`;
|
||||
|
||||
return html`
|
||||
<div
|
||||
class="action-button ${action.disabled ? 'disabled' : ''}"
|
||||
title="${action.tooltip || ''}"
|
||||
@click=${() => !action.disabled && action.onClick?.()}
|
||||
>
|
||||
<dees-icon
|
||||
.icon=${iconName}
|
||||
iconSize="12"
|
||||
></dees-icon>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
private async handleWidgetContextMenu(e: MouseEvent, widget: IBottomBarWidget): Promise<void> {
|
||||
if (!widget.contextMenuItems || widget.contextMenuItems.length === 0) return;
|
||||
|
||||
e.preventDefault();
|
||||
|
||||
const menuItems: Parameters<typeof DeesContextmenu.openContextMenuWithOptions>[1] = [];
|
||||
|
||||
for (const item of widget.contextMenuItems) {
|
||||
if (item.divider) {
|
||||
menuItems.push({ divider: true });
|
||||
} else {
|
||||
menuItems.push({
|
||||
name: item.name,
|
||||
iconName: item.iconName,
|
||||
action: async () => { await item.action(); },
|
||||
disabled: item.disabled,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
await DeesContextmenu.openContextMenuWithOptions(e, menuItems);
|
||||
}
|
||||
|
||||
// ==========================================
|
||||
// API METHODS (implements IBottomBarAPI)
|
||||
// ==========================================
|
||||
|
||||
/**
|
||||
* Add a widget to the bottom bar
|
||||
*/
|
||||
public addWidget(widget: IBottomBarWidget): void {
|
||||
// Remove existing widget with same ID if present
|
||||
this.widgets = this.widgets.filter(w => w.id !== widget.id);
|
||||
this.widgets = [...this.widgets, widget];
|
||||
}
|
||||
|
||||
/**
|
||||
* Update an existing widget by ID
|
||||
*/
|
||||
public updateWidget(id: string, update: Partial<IBottomBarWidget>): void {
|
||||
this.widgets = this.widgets.map(w =>
|
||||
w.id === id ? { ...w, ...update } : w
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove a widget by ID
|
||||
*/
|
||||
public removeWidget(id: string): void {
|
||||
this.widgets = this.widgets.filter(w => w.id !== id);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a widget by ID
|
||||
*/
|
||||
public getWidget(id: string): IBottomBarWidget | undefined {
|
||||
return this.widgets.find(w => w.id === id);
|
||||
}
|
||||
|
||||
/**
|
||||
* Clear all widgets
|
||||
*/
|
||||
public clearWidgets(): void {
|
||||
this.widgets = [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Add an action button
|
||||
*/
|
||||
public addAction(action: IBottomBarAction): void {
|
||||
this.actions = this.actions.filter(a => a.id !== action.id);
|
||||
this.actions = [...this.actions, action];
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove an action by ID
|
||||
*/
|
||||
public removeAction(id: string): void {
|
||||
this.actions = this.actions.filter(a => a.id !== id);
|
||||
}
|
||||
|
||||
/**
|
||||
* Clear all actions
|
||||
*/
|
||||
public clearActions(): void {
|
||||
this.actions = [];
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1 @@
|
||||
export * from './dees-appui-bottombar.js';
|
||||
@@ -164,6 +164,7 @@ export class DeesAppuiMainmenu extends DeesElement {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
overflow-x: hidden;
|
||||
overscroll-behavior: contain;
|
||||
padding: 8px 0;
|
||||
}
|
||||
|
||||
|
||||
@@ -285,6 +285,7 @@ export class DeesAppuiProfileDropdown extends DeesElement {
|
||||
max-width: calc(100vw - 32px);
|
||||
max-height: calc(100vh - 32px);
|
||||
overflow-y: auto;
|
||||
overscroll-behavior: contain;
|
||||
}
|
||||
|
||||
:host([isopen]) .dropdown {
|
||||
|
||||
@@ -201,6 +201,7 @@ export class DeesAppuiSecondarymenu extends DeesElement {
|
||||
flex: 1;
|
||||
overflow-y: auto;
|
||||
overflow-x: hidden;
|
||||
overscroll-behavior: contain;
|
||||
padding: 8px 0;
|
||||
}
|
||||
|
||||
|
||||
@@ -116,6 +116,7 @@ export class DeesAppuiTabs extends DeesElement {
|
||||
font-size: 14px;
|
||||
overflow-x: auto;
|
||||
overflow-y: hidden;
|
||||
overscroll-behavior: contain;
|
||||
scrollbar-width: thin;
|
||||
scrollbar-color: transparent transparent;
|
||||
height: 100%;
|
||||
|
||||
@@ -663,6 +663,44 @@ export const demoFunc = () => {
|
||||
|
||||
defaultView: 'dashboard',
|
||||
|
||||
bottomBar: {
|
||||
visible: true,
|
||||
widgets: [
|
||||
{
|
||||
id: 'status',
|
||||
iconName: 'lucide:activity',
|
||||
label: 'System Online',
|
||||
status: 'success',
|
||||
tooltip: 'All systems operational',
|
||||
onClick: () => console.log('Status clicked'),
|
||||
},
|
||||
{
|
||||
id: 'notifications',
|
||||
iconName: 'lucide:bell',
|
||||
label: '3 notifications',
|
||||
status: 'warning',
|
||||
tooltip: 'You have unread notifications',
|
||||
onClick: () => console.log('Notifications clicked'),
|
||||
},
|
||||
{
|
||||
id: 'version',
|
||||
iconName: 'lucide:gitBranch',
|
||||
label: 'v1.2.3',
|
||||
position: 'right',
|
||||
tooltip: 'Current version',
|
||||
},
|
||||
],
|
||||
actions: [
|
||||
{
|
||||
id: 'terminal',
|
||||
iconName: 'lucide:terminal',
|
||||
tooltip: 'Open Terminal',
|
||||
position: 'right',
|
||||
onClick: () => console.log('Terminal clicked'),
|
||||
},
|
||||
],
|
||||
},
|
||||
|
||||
onViewChange: (viewId, view) => {
|
||||
console.log(`View changed to: ${viewId} (${view.name})`);
|
||||
},
|
||||
|
||||
@@ -15,6 +15,7 @@ import type { DeesAppuiMainmenu } from '../dees-appui-mainmenu/dees-appui-mainme
|
||||
import type { DeesAppuiSecondarymenu } from '../dees-appui-secondarymenu/dees-appui-secondarymenu.js';
|
||||
import type { DeesAppuiMaincontent } from '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||
import type { DeesAppuiActivitylog } from '../dees-appui-activitylog/dees-appui-activitylog.js';
|
||||
import type { DeesAppuiBottombar } from '../dees-appui-bottombar/dees-appui-bottombar.js';
|
||||
import { demoFunc } from './dees-appui.demo.js';
|
||||
import { themeDefaultStyles } from '../../00theme.js';
|
||||
|
||||
@@ -23,6 +24,7 @@ import { ViewRegistry } from './view.registry.js';
|
||||
|
||||
// Import child components
|
||||
import '../dees-appui-appbar/index.js';
|
||||
import '../dees-appui-bottombar/dees-appui-bottombar.js';
|
||||
import '../dees-appui-mainmenu/dees-appui-mainmenu.js';
|
||||
import '../dees-appui-secondarymenu/dees-appui-secondarymenu.js';
|
||||
import '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||
@@ -156,6 +158,12 @@ export class DeesAppui extends DeesElement {
|
||||
@state()
|
||||
accessor activitylogElement: DeesAppuiActivitylog | undefined = undefined;
|
||||
|
||||
@state()
|
||||
accessor bottombarElement: DeesAppuiBottombar | undefined = undefined;
|
||||
|
||||
@state()
|
||||
accessor bottombarVisible: boolean = true;
|
||||
|
||||
// Current view state
|
||||
@state()
|
||||
accessor currentView: interfaces.IViewDefinition | undefined = undefined;
|
||||
@@ -179,12 +187,25 @@ export class DeesAppui extends DeesElement {
|
||||
.maingrid {
|
||||
position: absolute;
|
||||
top: 40px;
|
||||
height: calc(100% - 40px);
|
||||
height: calc(100% - 40px - 24px);
|
||||
width: 100%;
|
||||
display: grid;
|
||||
/* grid-template-columns set dynamically in template */
|
||||
grid-template-rows: 1fr;
|
||||
transition: grid-template-columns 0.3s ease;
|
||||
transition: grid-template-columns 0.3s ease, height 0.3s ease;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
:host([bottombar-hidden]) .maingrid {
|
||||
height: calc(100% - 40px);
|
||||
}
|
||||
|
||||
dees-appui-bottombar {
|
||||
position: absolute;
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
z-index: 4;
|
||||
}
|
||||
|
||||
/* Z-index layering for proper stacking */
|
||||
@@ -201,6 +222,7 @@ export class DeesAppui extends DeesElement {
|
||||
.maingrid > dees-appui-maincontent {
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
.maingrid > dees-appui-activitylog {
|
||||
@@ -293,6 +315,9 @@ export class DeesAppui extends DeesElement {
|
||||
class="${this.activityLogVisible ? 'visible' : 'hidden'}"
|
||||
></dees-appui-activitylog>
|
||||
</div>
|
||||
${this.bottombarVisible ? html`
|
||||
<dees-appui-bottombar></dees-appui-bottombar>
|
||||
` : ''}
|
||||
`;
|
||||
}
|
||||
|
||||
@@ -303,6 +328,7 @@ export class DeesAppui extends DeesElement {
|
||||
this.secondarymenu = this.shadowRoot!.querySelector('dees-appui-secondarymenu') as DeesAppuiSecondarymenu;
|
||||
this.maincontent = this.shadowRoot!.querySelector('dees-appui-maincontent') as DeesAppuiMaincontent;
|
||||
this.activitylogElement = this.shadowRoot!.querySelector('dees-appui-activitylog') as DeesAppuiActivitylog;
|
||||
this.bottombarElement = this.shadowRoot!.querySelector('dees-appui-bottombar') as DeesAppuiBottombar;
|
||||
|
||||
// Subscribe to activity log entry changes for badge count
|
||||
if (this.activitylogElement) {
|
||||
@@ -728,6 +754,72 @@ export class DeesAppui extends DeesElement {
|
||||
return this.activityLogVisible;
|
||||
}
|
||||
|
||||
// ==========================================
|
||||
// PROGRAMMATIC API: BOTTOM BAR
|
||||
// ==========================================
|
||||
|
||||
/**
|
||||
* Get the bottom bar API for widget/action management
|
||||
*/
|
||||
public get bottomBar(): interfaces.IBottomBarAPI {
|
||||
if (!this.bottombarElement) {
|
||||
// Return a deferred API that will work after firstUpdated
|
||||
return {
|
||||
addWidget: (widget) => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.addWidget(widget));
|
||||
},
|
||||
updateWidget: (id, update) => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.updateWidget(id, update));
|
||||
},
|
||||
removeWidget: (id) => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.removeWidget(id));
|
||||
},
|
||||
getWidget: (id) => this.bottombarElement?.getWidget(id),
|
||||
clearWidgets: () => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.clearWidgets());
|
||||
},
|
||||
addAction: (action) => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.addAction(action));
|
||||
},
|
||||
removeAction: (id) => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.removeAction(id));
|
||||
},
|
||||
clearActions: () => {
|
||||
this.updateComplete.then(() => this.bottombarElement?.clearActions());
|
||||
},
|
||||
};
|
||||
}
|
||||
return {
|
||||
addWidget: (widget) => this.bottombarElement!.addWidget(widget),
|
||||
updateWidget: (id, update) => this.bottombarElement!.updateWidget(id, update),
|
||||
removeWidget: (id) => this.bottombarElement!.removeWidget(id),
|
||||
getWidget: (id) => this.bottombarElement!.getWidget(id),
|
||||
clearWidgets: () => this.bottombarElement!.clearWidgets(),
|
||||
addAction: (action) => this.bottombarElement!.addAction(action),
|
||||
removeAction: (id) => this.bottombarElement!.removeAction(id),
|
||||
clearActions: () => this.bottombarElement!.clearActions(),
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Set bottom bar visibility
|
||||
*/
|
||||
public setBottomBarVisible(visible: boolean): void {
|
||||
this.bottombarVisible = visible;
|
||||
if (!visible) {
|
||||
this.setAttribute('bottombar-hidden', '');
|
||||
} else {
|
||||
this.removeAttribute('bottombar-hidden');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get bottom bar visibility state
|
||||
*/
|
||||
public getBottomBarVisible(): boolean {
|
||||
return this.bottombarVisible;
|
||||
}
|
||||
|
||||
// ==========================================
|
||||
// PROGRAMMATIC API: NAVIGATION
|
||||
// ==========================================
|
||||
@@ -840,6 +932,23 @@ export class DeesAppui extends DeesElement {
|
||||
}
|
||||
}
|
||||
|
||||
// Apply bottom bar config
|
||||
if (config.bottomBar) {
|
||||
this.setBottomBarVisible(config.bottomBar.visible ?? true);
|
||||
|
||||
if (config.bottomBar.widgets) {
|
||||
config.bottomBar.widgets.forEach(widget => {
|
||||
this.bottomBar.addWidget(widget);
|
||||
});
|
||||
}
|
||||
|
||||
if (config.bottomBar.actions) {
|
||||
config.bottomBar.actions.forEach(action => {
|
||||
this.bottomBar.addAction(action);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// Setup domtools.router integration
|
||||
this.setupRouterIntegration(config);
|
||||
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
// App UI Components
|
||||
export * from './dees-appui-activitylog/index.js';
|
||||
export * from './dees-appui-appbar/index.js';
|
||||
export * from './dees-appui-bottombar/index.js';
|
||||
export * from './dees-appui/index.js';
|
||||
export * from './dees-appui-maincontent/index.js';
|
||||
export * from './dees-appui-mainmenu/index.js';
|
||||
|
||||
@@ -64,6 +64,26 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
@state()
|
||||
accessor dragOverIndex: number = -1;
|
||||
|
||||
// Enhanced drag state for interactive reordering
|
||||
@state()
|
||||
accessor dragStartY: number = 0;
|
||||
|
||||
@state()
|
||||
accessor dragCurrentY: number = 0;
|
||||
|
||||
@state()
|
||||
accessor targetIndex: number = -1;
|
||||
|
||||
@state()
|
||||
accessor itemHeight: number = 0;
|
||||
|
||||
// Bound event handlers for cleanup
|
||||
private boundHandleGlobalDragOver: ((e: DragEvent) => void) | null = null;
|
||||
private boundHandleGlobalDragEnd: (() => void) | null = null;
|
||||
|
||||
// Store original item positions for accurate hit detection (before transforms)
|
||||
private originalItemRects: DOMRect[] = [];
|
||||
|
||||
public static styles = [
|
||||
themeDefaultStyles,
|
||||
...DeesInputBase.baseStyles,
|
||||
@@ -113,7 +133,7 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid ${cssManager.bdTheme('hsl(0 0% 89.8%)', 'hsl(0 0% 14.9%)')};
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(0 0% 3.9%)')};
|
||||
transition: all 0.15s ease;
|
||||
transition: transform 0.2s ease, background 0.15s ease, box-shadow 0.15s ease;
|
||||
position: relative;
|
||||
overflow: hidden; /* Prevent animation from affecting scroll bounds */
|
||||
}
|
||||
@@ -122,20 +142,31 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.list-item:hover:not(.disabled) {
|
||||
.list-items:not(.is-dragging) .list-item:hover:not(.disabled) {
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 97.5%)', 'hsl(0 0% 6.9%)')};
|
||||
}
|
||||
|
||||
/* Dragging item - follows cursor */
|
||||
.list-item.dragging {
|
||||
opacity: 0.4;
|
||||
background: ${cssManager.bdTheme('hsl(210 40% 96.1%)', 'hsl(215 20.2% 10.8%)')};
|
||||
position: relative;
|
||||
z-index: 100;
|
||||
background: ${cssManager.bdTheme('hsl(210 40% 98%)', 'hsl(215 20.2% 12%)')};
|
||||
box-shadow: 0 4px 12px ${cssManager.bdTheme('rgba(0, 0, 0, 0.15)', 'rgba(0, 0, 0, 0.4)')};
|
||||
border-radius: 6px;
|
||||
transition: box-shadow 0.15s ease, background 0.15s ease;
|
||||
}
|
||||
|
||||
.list-item.drag-over {
|
||||
background: ${cssManager.bdTheme('hsl(210 40% 93.1%)', 'hsl(215 20.2% 13.8%)')};
|
||||
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 51.2%)', 'hsl(217.2 91.2% 59.8%)')};
|
||||
/* Items that need to move up to make space */
|
||||
.list-item.move-up {
|
||||
transform: translateY(calc(-1 * var(--item-height, 48px)));
|
||||
}
|
||||
|
||||
/* Items that need to move down to make space */
|
||||
.list-item.move-down {
|
||||
transform: translateY(var(--item-height, 48px));
|
||||
}
|
||||
|
||||
|
||||
.drag-handle {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
@@ -313,27 +344,9 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
background: ${cssManager.bdTheme('hsl(0 0% 79.8%)', 'hsl(0 0% 34.9%)')};
|
||||
}
|
||||
|
||||
/* Animation for adding/removing items */
|
||||
@keyframes slideIn {
|
||||
from {
|
||||
opacity: 0;
|
||||
transform: translateY(-10px);
|
||||
}
|
||||
to {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
}
|
||||
}
|
||||
|
||||
.list-item {
|
||||
animation: slideIn 0.2s ease;
|
||||
}
|
||||
|
||||
/* Override any inherited contain/content-visibility that might cause scrolling issues */
|
||||
.list-items, .list-item {
|
||||
content-visibility: visible !important;
|
||||
contain: none !important;
|
||||
contain-intrinsic-size: auto !important;
|
||||
/* Disable transitions during drop to prevent flash */
|
||||
.list-items.dropping .list-item {
|
||||
transition: none !important;
|
||||
}
|
||||
`,
|
||||
];
|
||||
@@ -347,12 +360,11 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
<div class="list-items">
|
||||
${this.value.length > 0 ? this.value.map((item, index) => html`
|
||||
<div
|
||||
class="list-item ${this.draggedIndex === index ? 'dragging' : ''} ${this.dragOverIndex === index ? 'drag-over' : ''}"
|
||||
class="list-item ${this.draggedIndex === index ? 'dragging' : ''}"
|
||||
draggable="${this.sortable && !this.disabled}"
|
||||
@dragstart=${(e: DragEvent) => this.handleDragStart(e, index)}
|
||||
@dragend=${this.handleDragEnd}
|
||||
@dragover=${(e: DragEvent) => this.handleDragOver(e, index)}
|
||||
@dragleave=${this.handleDragLeave}
|
||||
@drop=${(e: DragEvent) => this.handleDrop(e, index)}
|
||||
>
|
||||
${this.sortable && !this.disabled ? html`
|
||||
@@ -547,48 +559,313 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
|
||||
return confirm(message);
|
||||
}
|
||||
|
||||
// Drag and drop handlers
|
||||
// Drag and drop handlers - Interactive implementation
|
||||
private handleDragStart(e: DragEvent, index: number) {
|
||||
if (!this.sortable || this.disabled) return;
|
||||
|
||||
|
||||
this.draggedIndex = index;
|
||||
this.targetIndex = index;
|
||||
e.dataTransfer!.effectAllowed = 'move';
|
||||
e.dataTransfer!.setData('text/plain', index.toString());
|
||||
|
||||
// Hide the default drag image
|
||||
const emptyImg = new Image();
|
||||
emptyImg.src = 'data:image/gif;base64,R0lGODlhAQABAIAAAAUEBAAAACwAAAAAAQABAAACAkQBADs=';
|
||||
e.dataTransfer!.setDragImage(emptyImg, 0, 0);
|
||||
|
||||
// Store initial mouse position
|
||||
this.dragStartY = e.clientY;
|
||||
this.dragCurrentY = e.clientY;
|
||||
|
||||
// Measure item height and store all original positions before any transforms
|
||||
const listItems = this.shadowRoot?.querySelector('.list-items');
|
||||
const allItems = Array.from(listItems?.querySelectorAll('.list-item') || []) as HTMLElement[];
|
||||
|
||||
if (allItems[index]) {
|
||||
this.itemHeight = allItems[index].offsetHeight;
|
||||
}
|
||||
|
||||
// Store original positions for accurate hit detection (before any transforms are applied)
|
||||
this.originalItemRects = allItems.map(item => item.getBoundingClientRect());
|
||||
|
||||
// Add class to container
|
||||
listItems?.classList.add('is-dragging');
|
||||
|
||||
// Set up global event listeners
|
||||
this.boundHandleGlobalDragOver = this.handleGlobalDragOver.bind(this);
|
||||
this.boundHandleGlobalDragEnd = this.handleGlobalDragEnd.bind(this);
|
||||
document.addEventListener('dragover', this.boundHandleGlobalDragOver);
|
||||
document.addEventListener('dragend', this.boundHandleGlobalDragEnd);
|
||||
}
|
||||
|
||||
private handleGlobalDragOver(e: DragEvent) {
|
||||
e.preventDefault();
|
||||
if (this.draggedIndex === -1) return;
|
||||
|
||||
this.dragCurrentY = e.clientY;
|
||||
|
||||
// Calculate which position the item should move to
|
||||
const listItems = this.shadowRoot?.querySelector('.list-items');
|
||||
if (!listItems) return;
|
||||
|
||||
const items = Array.from(listItems.querySelectorAll('.list-item')) as HTMLElement[];
|
||||
const draggedElement = items[this.draggedIndex];
|
||||
if (!draggedElement) return;
|
||||
|
||||
// Apply transform to dragged item
|
||||
const deltaY = this.dragCurrentY - this.dragStartY;
|
||||
draggedElement.style.transform = `translateY(${deltaY}px)`;
|
||||
|
||||
// Calculate the dragged item's current center position
|
||||
const draggedRect = this.originalItemRects[this.draggedIndex];
|
||||
if (!draggedRect) return;
|
||||
const draggedCenter = draggedRect.top + draggedRect.height / 2 + deltaY;
|
||||
|
||||
// Determine target index: swap when dragged item's center crosses another item's center
|
||||
// Account for items that have already shifted (their visual position changed)
|
||||
let newTargetIndex = this.draggedIndex;
|
||||
|
||||
for (let i = 0; i < items.length; i++) {
|
||||
if (i === this.draggedIndex) continue;
|
||||
|
||||
const rect = this.originalItemRects[i];
|
||||
if (!rect) continue;
|
||||
|
||||
// Adjust item center based on whether it has shifted
|
||||
let itemCenter = rect.top + rect.height / 2;
|
||||
|
||||
// If item has moved, use its shifted position
|
||||
if (items[i].classList.contains('move-up')) {
|
||||
itemCenter -= this.itemHeight;
|
||||
} else if (items[i].classList.contains('move-down')) {
|
||||
itemCenter += this.itemHeight;
|
||||
}
|
||||
|
||||
if (draggedCenter < itemCenter && i < this.draggedIndex) {
|
||||
newTargetIndex = i;
|
||||
break;
|
||||
} else if (draggedCenter > itemCenter && i > this.draggedIndex) {
|
||||
newTargetIndex = i;
|
||||
}
|
||||
}
|
||||
|
||||
// Update target index and apply move classes
|
||||
if (newTargetIndex !== this.targetIndex) {
|
||||
this.targetIndex = newTargetIndex;
|
||||
this.updateItemPositions(items);
|
||||
}
|
||||
}
|
||||
|
||||
private updateItemPositions(items: HTMLElement[]) {
|
||||
const draggedIdx = this.draggedIndex;
|
||||
const targetIdx = this.targetIndex;
|
||||
|
||||
// Set CSS variable for item height
|
||||
const listItems = this.shadowRoot?.querySelector('.list-items') as HTMLElement;
|
||||
if (listItems) {
|
||||
listItems.style.setProperty('--item-height', `${this.itemHeight}px`);
|
||||
}
|
||||
|
||||
items.forEach((item, i) => {
|
||||
if (i === draggedIdx) return; // Skip dragged item
|
||||
|
||||
item.classList.remove('move-up', 'move-down');
|
||||
item.style.setProperty('--item-height', `${this.itemHeight}px`);
|
||||
|
||||
if (draggedIdx < targetIdx) {
|
||||
// Dragging down: items between draggedIdx and targetIdx move up
|
||||
if (i > draggedIdx && i <= targetIdx) {
|
||||
item.classList.add('move-up');
|
||||
}
|
||||
} else if (draggedIdx > targetIdx) {
|
||||
// Dragging up: items between targetIdx and draggedIdx move down
|
||||
if (i >= targetIdx && i < draggedIdx) {
|
||||
item.classList.add('move-down');
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private handleGlobalDragEnd() {
|
||||
// Clean up event listeners
|
||||
if (this.boundHandleGlobalDragOver) {
|
||||
document.removeEventListener('dragover', this.boundHandleGlobalDragOver);
|
||||
this.boundHandleGlobalDragOver = null;
|
||||
}
|
||||
if (this.boundHandleGlobalDragEnd) {
|
||||
document.removeEventListener('dragend', this.boundHandleGlobalDragEnd);
|
||||
this.boundHandleGlobalDragEnd = null;
|
||||
}
|
||||
|
||||
const listItems = this.shadowRoot?.querySelector('.list-items');
|
||||
const items = listItems?.querySelectorAll('.list-item') as NodeListOf<HTMLElement>;
|
||||
const draggedElement = items?.[this.draggedIndex];
|
||||
|
||||
// If no reorder needed, animate back and clean up
|
||||
if (this.draggedIndex === -1 || this.targetIndex === -1 || this.draggedIndex === this.targetIndex) {
|
||||
// Animate dragged item back to original position
|
||||
if (draggedElement && this.draggedIndex !== -1) {
|
||||
draggedElement.style.transition = 'transform 0.15s ease';
|
||||
draggedElement.style.transform = 'translateY(0)';
|
||||
|
||||
let handled = false;
|
||||
const onReturn = () => {
|
||||
if (handled) return;
|
||||
handled = true;
|
||||
draggedElement.removeEventListener('transitionend', onReturn);
|
||||
this.cleanupDragState(listItems, items);
|
||||
};
|
||||
|
||||
draggedElement.addEventListener('transitionend', onReturn, { once: true });
|
||||
setTimeout(onReturn, 200);
|
||||
} else {
|
||||
this.cleanupDragState(listItems, items);
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// Calculate final position for dragged item
|
||||
const draggedRect = this.originalItemRects[this.draggedIndex];
|
||||
const targetRect = this.originalItemRects[this.targetIndex];
|
||||
|
||||
if (!draggedRect || !targetRect || !draggedElement) {
|
||||
this.cleanupDragState(listItems, items);
|
||||
return;
|
||||
}
|
||||
|
||||
// Calculate where dragged item needs to go
|
||||
let finalY: number;
|
||||
if (this.targetIndex > this.draggedIndex) {
|
||||
// Moving down: go to bottom of target
|
||||
finalY = targetRect.bottom - draggedRect.bottom;
|
||||
} else {
|
||||
// Moving up: go to top of target
|
||||
finalY = targetRect.top - draggedRect.top;
|
||||
}
|
||||
|
||||
// Animate dragged item to final position
|
||||
draggedElement.style.transition = 'transform 0.15s ease';
|
||||
draggedElement.style.transform = `translateY(${finalY}px)`;
|
||||
|
||||
// After animation completes, update data
|
||||
let handled = false;
|
||||
const onTransitionEnd = () => {
|
||||
if (handled) return;
|
||||
handled = true;
|
||||
draggedElement.removeEventListener('transitionend', onTransitionEnd);
|
||||
|
||||
// Disable all transitions
|
||||
listItems?.classList.add('dropping');
|
||||
|
||||
// Force reflow so dropping class takes effect immediately
|
||||
void (listItems as HTMLElement)?.offsetHeight;
|
||||
|
||||
// Clean up all element state
|
||||
items?.forEach(item => {
|
||||
item.classList.remove('move-up', 'move-down', 'dragging');
|
||||
item.style.removeProperty('transform');
|
||||
item.style.removeProperty('transition');
|
||||
});
|
||||
|
||||
// Update data
|
||||
const newValue = [...this.value];
|
||||
const [draggedItem] = newValue.splice(this.draggedIndex, 1);
|
||||
newValue.splice(this.targetIndex, 0, draggedItem);
|
||||
this.value = newValue;
|
||||
this.emitChange();
|
||||
|
||||
// Reset state
|
||||
this.draggedIndex = -1;
|
||||
this.dragOverIndex = -1;
|
||||
this.targetIndex = -1;
|
||||
this.dragStartY = 0;
|
||||
this.dragCurrentY = 0;
|
||||
this.originalItemRects = [];
|
||||
|
||||
// After render, ensure no animation then re-enable transitions
|
||||
this.updateComplete.then(() => {
|
||||
// Set inline transition:none on fresh elements
|
||||
const freshItems = this.shadowRoot?.querySelectorAll('.list-item') as NodeListOf<HTMLElement>;
|
||||
freshItems?.forEach(item => {
|
||||
item.style.transition = 'none';
|
||||
});
|
||||
|
||||
// Force reflow
|
||||
void (this.shadowRoot?.querySelector('.list-items') as HTMLElement)?.offsetHeight;
|
||||
|
||||
// Now re-enable transitions
|
||||
requestAnimationFrame(() => {
|
||||
freshItems?.forEach(item => {
|
||||
item.style.removeProperty('transition');
|
||||
});
|
||||
listItems?.classList.remove('dropping', 'is-dragging');
|
||||
});
|
||||
});
|
||||
};
|
||||
|
||||
draggedElement.addEventListener('transitionend', onTransitionEnd, { once: true });
|
||||
|
||||
// Fallback timeout in case transitionend doesn't fire
|
||||
setTimeout(onTransitionEnd, 200);
|
||||
}
|
||||
|
||||
private cleanupDragState(listItems: Element | null | undefined, items: NodeListOf<HTMLElement> | undefined) {
|
||||
listItems?.classList.add('dropping');
|
||||
|
||||
// Force reflow so dropping class takes effect immediately
|
||||
void (listItems as HTMLElement)?.offsetHeight;
|
||||
|
||||
items?.forEach(item => {
|
||||
item.classList.remove('move-up', 'move-down', 'dragging');
|
||||
item.style.removeProperty('transform');
|
||||
item.style.removeProperty('transition');
|
||||
});
|
||||
|
||||
this.draggedIndex = -1;
|
||||
this.dragOverIndex = -1;
|
||||
this.targetIndex = -1;
|
||||
this.dragStartY = 0;
|
||||
this.dragCurrentY = 0;
|
||||
this.originalItemRects = [];
|
||||
|
||||
this.updateComplete.then(() => {
|
||||
const freshItems = this.shadowRoot?.querySelectorAll('.list-item') as NodeListOf<HTMLElement>;
|
||||
freshItems?.forEach(item => {
|
||||
item.style.transition = 'none';
|
||||
});
|
||||
|
||||
void (this.shadowRoot?.querySelector('.list-items') as HTMLElement)?.offsetHeight;
|
||||
|
||||
requestAnimationFrame(() => {
|
||||
freshItems?.forEach(item => {
|
||||
item.style.removeProperty('transition');
|
||||
});
|
||||
listItems?.classList.remove('dropping', 'is-dragging');
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
private handleDragEnd() {
|
||||
this.draggedIndex = -1;
|
||||
this.dragOverIndex = -1;
|
||||
// This is called by the native dragend on the element
|
||||
// The actual cleanup is done in handleGlobalDragEnd
|
||||
this.handleGlobalDragEnd();
|
||||
}
|
||||
|
||||
private handleDragOver(e: DragEvent, index: number) {
|
||||
if (!this.sortable || this.disabled) return;
|
||||
|
||||
e.preventDefault();
|
||||
e.dataTransfer!.dropEffect = 'move';
|
||||
this.dragOverIndex = index;
|
||||
// We handle positioning in handleGlobalDragOver now
|
||||
}
|
||||
|
||||
private handleDragLeave() {
|
||||
this.dragOverIndex = -1;
|
||||
// No longer needed for visual feedback - handled by transform
|
||||
}
|
||||
|
||||
private handleDrop(e: DragEvent, dropIndex: number) {
|
||||
if (!this.sortable || this.disabled) return;
|
||||
|
||||
e.preventDefault();
|
||||
const draggedIndex = parseInt(e.dataTransfer!.getData('text/plain'));
|
||||
|
||||
if (draggedIndex !== dropIndex) {
|
||||
const newValue = [...this.value];
|
||||
const [draggedItem] = newValue.splice(draggedIndex, 1);
|
||||
newValue.splice(dropIndex, 0, draggedItem);
|
||||
this.value = newValue;
|
||||
this.emitChange();
|
||||
}
|
||||
|
||||
this.draggedIndex = -1;
|
||||
this.dragOverIndex = -1;
|
||||
// The actual reorder happens in handleGlobalDragEnd
|
||||
}
|
||||
|
||||
private emitChange() {
|
||||
|
||||
@@ -257,12 +257,13 @@ export class DeesIcon extends DeesElement {
|
||||
* @returns Object with type and name properties
|
||||
*/
|
||||
private parseIconString(iconStr: string): { type: 'fa' | 'lucide', name: string } {
|
||||
if (iconStr.startsWith('fa:')) {
|
||||
const lowerStr = iconStr.toLowerCase();
|
||||
if (lowerStr.startsWith('fa:')) {
|
||||
return {
|
||||
type: 'fa',
|
||||
name: iconStr.substring(3) // Remove 'fa:' prefix
|
||||
};
|
||||
} else if (iconStr.startsWith('lucide:')) {
|
||||
} else if (lowerStr.startsWith('lucide:')) {
|
||||
return {
|
||||
type: 'lucide',
|
||||
name: iconStr.substring(7) // Remove 'lucide:' prefix
|
||||
|
||||
@@ -88,16 +88,16 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
/* CSS Variables for consistent spacing and sizing */
|
||||
:host {
|
||||
--grid-gap: 16px;
|
||||
--tile-padding: 24px;
|
||||
--header-spacing: 16px;
|
||||
--content-min-height: 48px;
|
||||
--value-font-size: 30px;
|
||||
--unit-font-size: 16px;
|
||||
--label-font-size: 13px;
|
||||
--title-font-size: 14px;
|
||||
--description-spacing: 12px;
|
||||
--border-radius: 8px;
|
||||
--grid-gap: 12px;
|
||||
--tile-padding: 16px;
|
||||
--header-spacing: 12px;
|
||||
--content-min-height: 40px;
|
||||
--value-font-size: 26px;
|
||||
--unit-font-size: 14px;
|
||||
--label-font-size: 12px;
|
||||
--title-font-size: 13px;
|
||||
--description-spacing: 8px;
|
||||
--border-radius: 6px;
|
||||
--transition-duration: 0.15s;
|
||||
}
|
||||
|
||||
@@ -136,7 +136,7 @@ export class DeesStatsGrid extends DeesElement {
|
||||
/* Tile Base Styles */
|
||||
.stats-tile {
|
||||
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
|
||||
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(215 20.2% 11.8%)')};
|
||||
border: 1px solid ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
||||
border-radius: var(--border-radius);
|
||||
padding: var(--tile-padding);
|
||||
transition: all var(--transition-duration) ease;
|
||||
@@ -148,8 +148,8 @@ export class DeesStatsGrid extends DeesElement {
|
||||
}
|
||||
|
||||
.stats-tile:hover {
|
||||
background: ${cssManager.bdTheme('hsl(210 40% 98%)', 'hsl(215 20.2% 10.2%)')};
|
||||
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 85%)', 'hsl(215 20.2% 16.8%)')};
|
||||
background: ${cssManager.bdTheme('#fafafa', '#0d0d0d')};
|
||||
border-color: ${cssManager.bdTheme('#d0d0d0', '#2a2a2a')};
|
||||
}
|
||||
|
||||
.stats-tile.clickable {
|
||||
@@ -158,7 +158,7 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
.stats-tile.clickable:hover {
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 2px 8px ${cssManager.bdTheme('rgba(0,0,0,0.04)', 'rgba(0,0,0,0.2)')};
|
||||
box-shadow: 0 2px 6px ${cssManager.bdTheme('rgba(0,0,0,0.03)', 'rgba(0,0,0,0.15)')};
|
||||
}
|
||||
|
||||
/* Tile Header */
|
||||
@@ -230,10 +230,10 @@ export class DeesStatsGrid extends DeesElement {
|
||||
}
|
||||
|
||||
.gauge-container {
|
||||
width: 140px;
|
||||
height: 80px;
|
||||
width: 120px;
|
||||
height: 70px;
|
||||
position: relative;
|
||||
margin-top: -10px;
|
||||
margin-top: -8px;
|
||||
}
|
||||
|
||||
.gauge-svg {
|
||||
@@ -243,13 +243,13 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
.gauge-background {
|
||||
fill: none;
|
||||
stroke: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(215 20.2% 21.8%)')};
|
||||
stroke-width: 8;
|
||||
stroke: ${cssManager.bdTheme('#e8e8e8', '#1a1a1a')};
|
||||
stroke-width: 6;
|
||||
}
|
||||
|
||||
.gauge-fill {
|
||||
fill: none;
|
||||
stroke-width: 8;
|
||||
stroke-width: 6;
|
||||
stroke-linecap: round;
|
||||
transition: stroke-dashoffset 0.6s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
@@ -287,17 +287,17 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
.percentage-bar {
|
||||
width: 100%;
|
||||
height: 8px;
|
||||
background: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(215 20.2% 21.8%)')};
|
||||
border-radius: 4px;
|
||||
height: 6px;
|
||||
background: ${cssManager.bdTheme('#e8e8e8', '#1a1a1a')};
|
||||
border-radius: 3px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.percentage-fill {
|
||||
height: 100%;
|
||||
background: ${cssManager.bdTheme('hsl(215.3 25% 8.8%)', 'hsl(210 40% 98%)')};
|
||||
background: ${cssManager.bdTheme('#333333', '#e0e0e0')};
|
||||
transition: width 0.6s cubic-bezier(0.4, 0, 0.2, 1);
|
||||
border-radius: 4px;
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
/* Trend Styles */
|
||||
@@ -339,7 +339,7 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
.trend-graph {
|
||||
width: 100%;
|
||||
height: 32px;
|
||||
height: 28px;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
@@ -351,14 +351,14 @@ export class DeesStatsGrid extends DeesElement {
|
||||
|
||||
.trend-line {
|
||||
fill: none;
|
||||
stroke: ${cssManager.bdTheme('hsl(215.4 16.3% 66.9%)', 'hsl(215 20.2% 55.1%)')};
|
||||
stroke-width: 2;
|
||||
stroke: ${cssManager.bdTheme('#999999', '#666666')};
|
||||
stroke-width: 1.5;
|
||||
stroke-linejoin: round;
|
||||
stroke-linecap: round;
|
||||
}
|
||||
|
||||
.trend-area {
|
||||
fill: ${cssManager.bdTheme('hsl(215.4 16.3% 66.9% / 0.1)', 'hsl(215 20.2% 55.1% / 0.08)')};
|
||||
fill: ${cssManager.bdTheme('rgba(150, 150, 150, 0.08)', 'rgba(100, 100, 100, 0.08)')};
|
||||
}
|
||||
|
||||
/* Text Value Styles */
|
||||
@@ -480,13 +480,13 @@ export class DeesStatsGrid extends DeesElement {
|
||||
const value = typeof tile.value === 'number' ? tile.value : parseFloat(tile.value);
|
||||
const options = tile.gaugeOptions || { min: 0, max: 100 };
|
||||
const percentage = ((value - options.min) / (options.max - options.min)) * 100;
|
||||
|
||||
|
||||
// SVG dimensions and calculations
|
||||
const width = 140;
|
||||
const height = 80;
|
||||
const strokeWidth = 8;
|
||||
const width = 120;
|
||||
const height = 70;
|
||||
const strokeWidth = 6;
|
||||
const padding = strokeWidth / 2 + 2;
|
||||
const radius = 48;
|
||||
const radius = 40;
|
||||
const centerX = width / 2;
|
||||
const centerY = height - padding;
|
||||
|
||||
|
||||
@@ -4,6 +4,104 @@ import type { IMenuItem } from './tab.js';
|
||||
import type { IMenuGroup } from './menugroup.js';
|
||||
import type { ISecondaryMenuGroup, ISecondaryMenuItem } from './secondarymenu.js';
|
||||
|
||||
// ==========================================
|
||||
// BOTTOM BAR INTERFACES
|
||||
// ==========================================
|
||||
|
||||
/**
|
||||
* Bottom bar widget status for styling
|
||||
*/
|
||||
export type TBottomBarWidgetStatus = 'idle' | 'active' | 'success' | 'warning' | 'error';
|
||||
|
||||
/**
|
||||
* Generic status widget for the bottom bar
|
||||
*/
|
||||
export interface IBottomBarWidget {
|
||||
/** Unique identifier for the widget */
|
||||
id: string;
|
||||
/** Icon to display (lucide icon name) */
|
||||
iconName?: string;
|
||||
/** Text label to display */
|
||||
label?: string;
|
||||
/** Status affects styling (colors) */
|
||||
status?: TBottomBarWidgetStatus;
|
||||
/** Tooltip text */
|
||||
tooltip?: string;
|
||||
/** Whether the widget shows a loading spinner */
|
||||
loading?: boolean;
|
||||
/** Click handler for the widget */
|
||||
onClick?: () => void;
|
||||
/** Optional context menu items on right-click */
|
||||
contextMenuItems?: IBottomBarContextMenuItem[];
|
||||
/** Position: 'left' (default) or 'right' */
|
||||
position?: 'left' | 'right';
|
||||
/** Order within position group (lower = earlier) */
|
||||
order?: number;
|
||||
}
|
||||
|
||||
/**
|
||||
* Context menu item for bottom bar widgets
|
||||
*/
|
||||
export interface IBottomBarContextMenuItem {
|
||||
name: string;
|
||||
iconName?: string;
|
||||
action: () => void | Promise<void>;
|
||||
disabled?: boolean;
|
||||
divider?: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
* Bottom bar action (quick action button)
|
||||
*/
|
||||
export interface IBottomBarAction {
|
||||
/** Unique identifier */
|
||||
id: string;
|
||||
/** Icon to display */
|
||||
iconName: string;
|
||||
/** Tooltip */
|
||||
tooltip?: string;
|
||||
/** Click handler */
|
||||
onClick: () => void | Promise<void>;
|
||||
/** Whether action is disabled */
|
||||
disabled?: boolean;
|
||||
/** Position: 'left' or 'right' (default) */
|
||||
position?: 'left' | 'right';
|
||||
}
|
||||
|
||||
/**
|
||||
* Bottom bar configuration
|
||||
*/
|
||||
export interface IBottomBarConfig {
|
||||
/** Whether bottom bar is visible */
|
||||
visible?: boolean;
|
||||
/** Initial widgets */
|
||||
widgets?: IBottomBarWidget[];
|
||||
/** Initial actions */
|
||||
actions?: IBottomBarAction[];
|
||||
}
|
||||
|
||||
/**
|
||||
* Bottom bar programmatic API
|
||||
*/
|
||||
export interface IBottomBarAPI {
|
||||
/** Add a widget */
|
||||
addWidget: (widget: IBottomBarWidget) => void;
|
||||
/** Update an existing widget by ID */
|
||||
updateWidget: (id: string, update: Partial<IBottomBarWidget>) => void;
|
||||
/** Remove a widget by ID */
|
||||
removeWidget: (id: string) => void;
|
||||
/** Get a widget by ID */
|
||||
getWidget: (id: string) => IBottomBarWidget | undefined;
|
||||
/** Clear all widgets */
|
||||
clearWidgets: () => void;
|
||||
/** Add an action button */
|
||||
addAction: (action: IBottomBarAction) => void;
|
||||
/** Remove an action by ID */
|
||||
removeAction: (id: string) => void;
|
||||
/** Clear all actions */
|
||||
clearActions: () => void;
|
||||
}
|
||||
|
||||
// Forward declaration for circular reference
|
||||
export type TDeesAppui = HTMLElement & {
|
||||
setAppBarMenus: (menus: IAppBarMenuItem[]) => void;
|
||||
@@ -42,6 +140,10 @@ export type TDeesAppui = HTMLElement & {
|
||||
getActivityLogVisible: () => boolean;
|
||||
navigateToView: (viewId: string, params?: Record<string, string>) => Promise<boolean>;
|
||||
getCurrentView: () => IViewDefinition | undefined;
|
||||
// Bottom bar
|
||||
bottomBar: IBottomBarAPI;
|
||||
setBottomBarVisible: (visible: boolean) => void;
|
||||
getBottomBarVisible: () => boolean;
|
||||
};
|
||||
|
||||
/**
|
||||
@@ -233,6 +335,9 @@ export interface IAppConfig {
|
||||
/** Activity log configuration */
|
||||
activityLog?: IActivityLogConfig;
|
||||
|
||||
/** Bottom bar configuration */
|
||||
bottomBar?: IBottomBarConfig;
|
||||
|
||||
/** Event callbacks */
|
||||
onViewChange?: (viewId: string, view: IViewDefinition) => void;
|
||||
onSearch?: (query: string) => void;
|
||||
|
||||
Reference in New Issue
Block a user