Compare commits

..

16 Commits

Author SHA1 Message Date
43db777f2c v3.32.0
Some checks failed
Default (tags) / security (push) Failing after 12s
Default (tags) / test (push) Failing after 12s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-04 17:09:18 +00:00
9bd1734d09 feat(demo): add demoGroup metadata to components and update related dependencies 2026-01-04 17:09:18 +00:00
aafdb4af72 v3.31.0
Some checks failed
Default (tags) / security (push) Failing after 15s
Default (tags) / test (push) Failing after 14s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-04 10:16:50 +00:00
67a24ddf26 feat(dees-input-list): enhance drag-and-drop reordering for dees-input-list and migrate tests to chromium runner 2026-01-04 10:16:50 +00:00
2a928886b9 v3.30.1
Some checks failed
Default (tags) / security (push) Failing after 15s
Default (tags) / test (push) Failing after 15s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-04 09:21:02 +00:00
4d192654df fix(dees-statsgrid): refine spacing, sizing, and colors in dees-statsgrid for a tighter, more compact appearance 2026-01-04 09:21:02 +00:00
a634c2e237 v3.30.0
Some checks failed
Default (tags) / security (push) Failing after 15s
Default (tags) / test (push) Failing after 16s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-03 12:40:11 +00:00
9b0b448cb1 feat(appui): add dees-appui-bottombar component with config, programmatic API, demo and docs 2026-01-03 12:40:11 +00:00
ba4aa912af v3.29.3
Some checks failed
Default (tags) / security (push) Failing after 14s
Default (tags) / test (push) Failing after 16s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-03 02:50:37 +00:00
ca4f994b55 fix(elements/appui): prevent scroll chaining on app UI components by adding overscroll-behavior: contain 2026-01-03 02:50:37 +00:00
74844492eb v3.29.2
Some checks failed
Default (tags) / security (push) Failing after 14s
Default (tags) / test (push) Failing after 15s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-03 02:45:32 +00:00
c42cedbf94 fix(dees-appui): set min-height: 0 on .maingrid > dees-appui-maincontent to prevent layout overflow in flex container 2026-01-03 02:45:32 +00:00
749725f091 v3.29.1
Some checks failed
Default (tags) / security (push) Failing after 14s
Default (tags) / test (push) Failing after 15s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-03 02:33:01 +00:00
f3a8ad057a fix(dees-appui): prevent main grid overflow by adding overflow:hidden; and add Playwright scroll containment screenshots 2026-01-03 02:33:01 +00:00
7b8918705e v3.29.0
Some checks failed
Default (tags) / security (push) Failing after 13s
Default (tags) / test (push) Failing after 15s
Default (tags) / release (push) Has been skipped
Default (tags) / metadata (push) Has been skipped
2026-01-03 02:09:13 +00:00
8313c24c9d feat(docs): add documentation for new input components, activity log features, theming, and expand DeesAppui docs 2026-01-03 02:09:13 +00:00
88 changed files with 1912 additions and 218 deletions

Binary file not shown.

After

Width:  |  Height:  |  Size: 125 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 125 KiB

View File

@@ -1,5 +1,72 @@
# Changelog # Changelog
## 2026-01-04 - 3.32.0 - feat(demo)
add demoGroup metadata to components and update related dependencies
- Add public static demoGroup properties to many components to categorize demos (groups added: App UI, Button, Chart, Data View, Form, Input, PDF, Simple, Workspace).
- Bump package deps/devDeps: @design.estate/dees-domtools -> ^2.3.7, @design.estate/dees-element -> ^2.1.5, @design.estate/dees-wcctools -> ^3.7.1.
- Clean up package.json deps (removed duplicate entries and removed 'lit' dependency).
- Refactor dees-pdf-viewer to use consolidated directives import (directives.keyed and directives.repeat) instead of separate keyed/repeat imports.
## 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
- Updated top-level README to reflect component count increase (75+ → 80+) and added many new component docs
- Added documentation and examples for DeesInputList (sortable list input) and DeesInputProfilepicture (cropping, upload, processing)
- Introduced DeesTheme documentation with usage examples and CSS custom property overrides
- Expanded DeesAppui readme with architecture overview, activity log panel docs, activity entry types, and navigation/secondary menu guidance
- Documented activity log APIs and controls (activityLog.add, addMany, clear, getEntries, filter, search) and new control API helpers (setActivityLogVisible, toggleActivityLog, getActivityLogVisible)
- Updated Appbar examples to include activity log toggle properties (.showActivityLogToggle, .activityLogCount, .activityLogActive) and @activity-toggle event
- Added interface docs (IViewDefinition, IActivityEntry) and updated menu/secondary menu type references
- Changes are documentation-focused (README/element readmes); no source code logic changes shown in this diff
## 2026-01-03 - 3.28.1 - fix(appui) ## 2026-01-03 - 3.28.1 - fix(appui)
adjust layout and spacing in app UI components: fix activity log overflow, contain main content overscroll, and refine secondary menu padding/transition adjust layout and spacing in app UI components: fix activity log overflow, contain main content overscroll, and refine secondary menu padding/transition

View File

@@ -1,6 +1,6 @@
{ {
"name": "@design.estate/dees-catalog", "name": "@design.estate/dees-catalog",
"version": "3.28.1", "version": "3.32.0",
"private": false, "private": false,
"description": "A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.", "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", "main": "dist_ts_web/index.js",
@@ -16,8 +16,8 @@
"author": "Lossless GmbH", "author": "Lossless GmbH",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@design.estate/dees-domtools": "^2.3.6", "@design.estate/dees-domtools": "^2.3.7",
"@design.estate/dees-element": "^2.1.3", "@design.estate/dees-element": "^2.1.5",
"@fortawesome/fontawesome-svg-core": "^7.1.0", "@fortawesome/fontawesome-svg-core": "^7.1.0",
"@fortawesome/free-brands-svg-icons": "^7.1.0", "@fortawesome/free-brands-svg-icons": "^7.1.0",
"@fortawesome/free-regular-svg-icons": "^7.1.0", "@fortawesome/free-regular-svg-icons": "^7.1.0",
@@ -25,6 +25,7 @@
"@push.rocks/smarti18n": "^1.0.4", "@push.rocks/smarti18n": "^1.0.4",
"@push.rocks/smartpromise": "^4.2.0", "@push.rocks/smartpromise": "^4.2.0",
"@push.rocks/smartstring": "^4.1.0", "@push.rocks/smartstring": "^4.1.0",
"@tempfix/webcontainer__api": "1.6.1",
"@tiptap/core": "^2.23.0", "@tiptap/core": "^2.23.0",
"@tiptap/extension-link": "^2.23.0", "@tiptap/extension-link": "^2.23.0",
"@tiptap/extension-text-align": "^2.23.0", "@tiptap/extension-text-align": "^2.23.0",
@@ -32,11 +33,9 @@
"@tiptap/extension-underline": "^2.23.0", "@tiptap/extension-underline": "^2.23.0",
"@tiptap/starter-kit": "^2.23.0", "@tiptap/starter-kit": "^2.23.0",
"@tsclass/tsclass": "^9.3.0", "@tsclass/tsclass": "^9.3.0",
"@tempfix/webcontainer__api": "1.6.1",
"apexcharts": "^5.3.6", "apexcharts": "^5.3.6",
"highlight.js": "11.11.1", "highlight.js": "11.11.1",
"ibantools": "^4.5.1", "ibantools": "^4.5.1",
"lit": "^3.3.1",
"lucide": "^0.562.0", "lucide": "^0.562.0",
"monaco-editor": "0.55.1", "monaco-editor": "0.55.1",
"pdfjs-dist": "^4.10.38", "pdfjs-dist": "^4.10.38",
@@ -44,7 +43,7 @@
"xterm-addon-fit": "^0.8.0" "xterm-addon-fit": "^0.8.0"
}, },
"devDependencies": { "devDependencies": {
"@design.estate/dees-wcctools": "^3.4.0", "@design.estate/dees-wcctools": "^3.7.1",
"@git.zone/tsbuild": "^4.0.2", "@git.zone/tsbuild": "^4.0.2",
"@git.zone/tsbundle": "^2.6.3", "@git.zone/tsbundle": "^2.6.3",
"@git.zone/tstest": "^3.1.4", "@git.zone/tstest": "^3.1.4",

143
pnpm-lock.yaml generated
View File

@@ -9,11 +9,11 @@ importers:
.: .:
dependencies: dependencies:
'@design.estate/dees-domtools': '@design.estate/dees-domtools':
specifier: ^2.3.6 specifier: ^2.3.7
version: 2.3.6 version: 2.3.7
'@design.estate/dees-element': '@design.estate/dees-element':
specifier: ^2.1.3 specifier: ^2.1.5
version: 2.1.3 version: 2.1.5
'@fortawesome/fontawesome-svg-core': '@fortawesome/fontawesome-svg-core':
specifier: ^7.1.0 specifier: ^7.1.0
version: 7.1.0 version: 7.1.0
@@ -68,9 +68,6 @@ importers:
ibantools: ibantools:
specifier: ^4.5.1 specifier: ^4.5.1
version: 4.5.1 version: 4.5.1
lit:
specifier: ^3.3.1
version: 3.3.1
lucide: lucide:
specifier: ^0.562.0 specifier: ^0.562.0
version: 0.562.0 version: 0.562.0
@@ -88,8 +85,8 @@ importers:
version: 0.8.0(xterm@5.3.0) version: 0.8.0(xterm@5.3.0)
devDependencies: devDependencies:
'@design.estate/dees-wcctools': '@design.estate/dees-wcctools':
specifier: ^3.4.0 specifier: ^3.7.1
version: 3.4.0 version: 3.7.1
'@git.zone/tsbuild': '@git.zone/tsbuild':
specifier: ^4.0.2 specifier: ^4.0.2
version: 4.0.2 version: 4.0.2
@@ -328,17 +325,17 @@ packages:
'@design.estate/dees-comms@1.0.30': '@design.estate/dees-comms@1.0.30':
resolution: {integrity: sha512-KchMlklJfKAjQiJiR0xmofXtQ27VgZtBIxcMwPE9d+h3jJRv+lPZxzBQVOM0eyM0uS44S5vJMZ11IeV4uDXSHg==} resolution: {integrity: sha512-KchMlklJfKAjQiJiR0xmofXtQ27VgZtBIxcMwPE9d+h3jJRv+lPZxzBQVOM0eyM0uS44S5vJMZ11IeV4uDXSHg==}
'@design.estate/dees-domtools@2.3.6': '@design.estate/dees-domtools@2.3.7':
resolution: {integrity: sha512-cKaPNtSpp/ZuuXVx2dXO3K2FU3/HjC4ZkqtXb8Kl6yy9rNDbgtjcI4PuOk9Ux1SJzw7FgcxqVh7OSEV60htbmg==} resolution: {integrity: sha512-MXoDBrP7JTOpni8b12aFXHJKnKBoQppM8cYBuL9cesRmCVGdB7p39XMRQ7dRyMhmmyr66L3cOczhiCV6febCwg==}
'@design.estate/dees-element@2.1.3': '@design.estate/dees-element@2.1.5':
resolution: {integrity: sha512-TjXWxVcdSPaT1IOk31ckfxvAZnJLuTxhFGsNCKoh63/UE2FVf6slp8//UFvN+ADigiA9ZsY0azkY99XbJCwDDA==} resolution: {integrity: sha512-czUOFvBiUKi34I+/keDRDc71fuORZS0NfbSuD2jJ4D1ODiTPjaZ6A6SkdQ2QqCEzVsx73XF99Pu8pxPnaOLnHg==}
'@design.estate/dees-wcctools@1.3.0': '@design.estate/dees-wcctools@1.3.0':
resolution: {integrity: sha512-+yd8c1gTIKNRQYCvG0xu6Am8dHsRm7ymluX2gnoBQN4aFOpZgIBi/v9CvGyPhTD1p/VRouIBz1wsUCejnwrFCA==} resolution: {integrity: sha512-+yd8c1gTIKNRQYCvG0xu6Am8dHsRm7ymluX2gnoBQN4aFOpZgIBi/v9CvGyPhTD1p/VRouIBz1wsUCejnwrFCA==}
'@design.estate/dees-wcctools@3.4.0': '@design.estate/dees-wcctools@3.7.1':
resolution: {integrity: sha512-B263qJxK1Ob5ZmC+qj/utiuKZvdewIO6WwTfrTKF3X0Y24pcxoJVwJsDDcJID4kRd44EcNU9CP0FfWD2uYX9GQ==} resolution: {integrity: sha512-BiNWghUoC05RTQOGVCTK+wis6d18LyLY+2p8vHC0q2OBw9hrPoY8k9EplpQgY40MvP0sTXWUwaa7VPXra8ASjA==}
'@emnapi/core@1.7.1': '@emnapi/core@1.7.1':
resolution: {integrity: sha512-o1uhUASyo921r2XtHYOHy7gdkGLge8ghBEQHMWmyJFoXlpU58kIrhhN3w26lpQb6dspetweapMn2CSNwQ8I4wg==} resolution: {integrity: sha512-o1uhUASyo921r2XtHYOHy7gdkGLge8ghBEQHMWmyJFoXlpU58kIrhhN3w26lpQb6dspetweapMn2CSNwQ8I4wg==}
@@ -568,11 +565,11 @@ packages:
'@leichtgewicht/ip-codec@2.0.5': '@leichtgewicht/ip-codec@2.0.5':
resolution: {integrity: sha512-Vo+PSpZG2/fmgmiNzYK9qWRh8h/CHrwD0mo1h1DzL4yzHNSfWYujGTYsWGreD000gcgmZ7K4Ys6Tx9TxtsKdDw==} resolution: {integrity: sha512-Vo+PSpZG2/fmgmiNzYK9qWRh8h/CHrwD0mo1h1DzL4yzHNSfWYujGTYsWGreD000gcgmZ7K4Ys6Tx9TxtsKdDw==}
'@lit-labs/ssr-dom-shim@1.4.0': '@lit-labs/ssr-dom-shim@1.5.0':
resolution: {integrity: sha512-ficsEARKnmmW5njugNYKipTm4SFnbik7CXtoencDZzmzo/dQ+2Q0bgkzJuoJP20Aj0F+izzJjOqsnkd6F/o1bw==} resolution: {integrity: sha512-HLomZXMmrCFHSRKESF5vklAKsDY7/fsT/ZhqCu3V0UoW/Qbv8wxmO4W9bx4KnCCF2Zak4yuk+AGraK/bPmI4kA==}
'@lit/reactive-element@2.1.1': '@lit/reactive-element@2.1.2':
resolution: {integrity: sha512-N+dm5PAYdQ8e6UlywyyrgI2t++wFGXfHx+dSJ1oBrg6FAxUj40jId++EaRm80MKX5JnlH1sBsyZ5h0bcZKemCg==} resolution: {integrity: sha512-pbCDiVMnne1lYUIaYNN5wrwQXDtHaYtg7YEFPeW+hws6U47WeFvISGUWekPGKWOP1ygrs0ef0o1VJMk1exos5A==}
'@mixmark-io/domino@2.2.0': '@mixmark-io/domino@2.2.0':
resolution: {integrity: sha512-Y28PR25bHXUg88kCV7nivXrP2Nj2RueZ3/l/jdx6J9f8J4nsEGcgX0Qe6lt7Pa+J79+kPiJU3LguR6O/6zrLOw==} resolution: {integrity: sha512-Y28PR25bHXUg88kCV7nivXrP2Nj2RueZ3/l/jdx6J9f8J4nsEGcgX0Qe6lt7Pa+J79+kPiJU3LguR6O/6zrLOw==}
@@ -1450,6 +1447,20 @@ packages:
'@tempfix/idb@8.0.3': '@tempfix/idb@8.0.3':
resolution: {integrity: sha512-hPJQKO7+oAIY+pDNImrZ9QAINbz9KmwT+yO4iRVwdPanok2YKpaUxdJzIvCUwY0YgAawlvYdffbLvRLV5hbs2g==} resolution: {integrity: sha512-hPJQKO7+oAIY+pDNImrZ9QAINbz9KmwT+yO4iRVwdPanok2YKpaUxdJzIvCUwY0YgAawlvYdffbLvRLV5hbs2g==}
'@tempfix/lenis@1.3.17':
resolution: {integrity: sha512-IqbEB2jLGd0CZrr6TQgjPlhIJJwjDD/53e60KmEr2MEMxwRFUn6pg/H2EvxtoeS7ItmQdhWkJwPgtvVRUCctNw==}
peerDependencies:
'@nuxt/kit': '>=3.0.0'
react: '>=17.0.0'
vue: '>=3.0.0'
peerDependenciesMeta:
'@nuxt/kit':
optional: true
react:
optional: true
vue:
optional: true
'@tempfix/webcontainer__api@1.6.1': '@tempfix/webcontainer__api@1.6.1':
resolution: {integrity: sha512-Hgn3cwy0vPzjrVBqeVnY0jNZLaOCW7d+dxBe7Jv9YGHAjJ8udUMS+KbTywSv5paAfld3A/RN/iolmMzOwZxLTA==} resolution: {integrity: sha512-Hgn3cwy0vPzjrVBqeVnY0jNZLaOCW7d+dxBe7Jv9YGHAjJ8udUMS+KbTywSv5paAfld3A/RN/iolmMzOwZxLTA==}
@@ -2661,20 +2672,6 @@ packages:
resolution: {integrity: sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==} resolution: {integrity: sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==}
engines: {node: '>=0.10.0'} engines: {node: '>=0.10.0'}
lenis@1.3.15:
resolution: {integrity: sha512-zSYOFs0ydafX70uygFoipaHHQouPeE4DpZZhdOUyLJxVf2ZVvBCBBaolDDaQztTRsa6+stBlxq2GmFGJPAVryQ==}
peerDependencies:
'@nuxt/kit': '>=3.0.0'
react: '>=17.0.0'
vue: '>=3.0.0'
peerDependenciesMeta:
'@nuxt/kit':
optional: true
react:
optional: true
vue:
optional: true
lines-and-columns@1.2.4: lines-and-columns@1.2.4:
resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==}
@@ -2684,14 +2681,14 @@ packages:
linkifyjs@4.3.2: linkifyjs@4.3.2:
resolution: {integrity: sha512-NT1CJtq3hHIreOianA8aSXn6Cw0JzYOuDQbOrSPe7gqFnCpKP++MQe3ODgO3oh2GJFORkAAdqredOa60z63GbA==} resolution: {integrity: sha512-NT1CJtq3hHIreOianA8aSXn6Cw0JzYOuDQbOrSPe7gqFnCpKP++MQe3ODgO3oh2GJFORkAAdqredOa60z63GbA==}
lit-element@4.2.1: lit-element@4.2.2:
resolution: {integrity: sha512-WGAWRGzirAgyphK2urmYOV72tlvnxw7YfyLDgQ+OZnM9vQQBQnumQ7jUJe6unEzwGU3ahFOjuz1iz1jjrpCPuw==} resolution: {integrity: sha512-aFKhNToWxoyhkNDmWZwEva2SlQia+jfG0fjIWV//YeTaWrVnOxD89dPKfigCUspXFmjzOEUQpOkejH5Ly6sG0w==}
lit-html@3.3.1: lit-html@3.3.2:
resolution: {integrity: sha512-S9hbyDu/vs1qNrithiNyeyv64c9yqiW9l+DBgI18fL+MTvOtWoFR0FWiyq1TxaYef5wNlpEmzlXoBlZEO+WjoA==} resolution: {integrity: sha512-Qy9hU88zcmaxBXcc10ZpdK7cOLXvXpRoBxERdtqV9QOrfpMZZ6pSYP91LhpPtap3sFMUiL7Tw2RImbe0Al2/kw==}
lit@3.3.1: lit@3.3.2:
resolution: {integrity: sha512-Ksr/8L3PTapbdXJCk+EJVB78jDodUMaP54gD24W186zGRARvwrsPfS60wae/SSCTCNZVPd1chXqio1qHQmu4NA==} resolution: {integrity: sha512-NF9zbsP79l4ao2SNrH3NkfmFgN/hBYSQo90saIVI1o5GpjAdCPVstVzO1MrLOakHoEhYkrtRjPK6Ob521aoYWQ==}
locate-path@5.0.0: locate-path@5.0.0:
resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==} resolution: {integrity: sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==}
@@ -3938,7 +3935,7 @@ snapshots:
cors: 2.8.5 cors: 2.8.5
express: 5.2.1 express: 5.2.1
express-force-ssl: 0.3.2 express-force-ssl: 0.3.2
lit: 3.3.1 lit: 3.3.2
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
- '@push.rocks/smartserve' - '@push.rocks/smartserve'
@@ -3984,7 +3981,7 @@ snapshots:
'@push.rocks/webrequest': 4.0.1 '@push.rocks/webrequest': 4.0.1
'@push.rocks/webstore': 2.0.20 '@push.rocks/webstore': 2.0.20
'@tsclass/tsclass': 9.3.0 '@tsclass/tsclass': 9.3.0
lit: 3.3.1 lit: 3.3.2
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
- '@tiptap/pm' - '@tiptap/pm'
@@ -4528,8 +4525,8 @@ snapshots:
'@design.estate/dees-catalog@3.3.0(@tiptap/pm@2.27.1)': '@design.estate/dees-catalog@3.3.0(@tiptap/pm@2.27.1)':
dependencies: dependencies:
'@design.estate/dees-domtools': 2.3.6 '@design.estate/dees-domtools': 2.3.7
'@design.estate/dees-element': 2.1.3 '@design.estate/dees-element': 2.1.5
'@design.estate/dees-wcctools': 1.3.0 '@design.estate/dees-wcctools': 1.3.0
'@fortawesome/fontawesome-svg-core': 7.1.0 '@fortawesome/fontawesome-svg-core': 7.1.0
'@fortawesome/free-brands-svg-icons': 7.1.0 '@fortawesome/free-brands-svg-icons': 7.1.0
@@ -4549,7 +4546,7 @@ snapshots:
apexcharts: 5.3.6 apexcharts: 5.3.6
highlight.js: 11.11.1 highlight.js: 11.11.1
ibantools: 4.5.1 ibantools: 4.5.1
lit: 3.3.1 lit: 3.3.2
lucide: 0.555.0 lucide: 0.555.0
monaco-editor: 0.52.2 monaco-editor: 0.52.2
pdfjs-dist: 4.10.38 pdfjs-dist: 4.10.38
@@ -4569,7 +4566,7 @@ snapshots:
'@push.rocks/smartdelay': 3.0.5 '@push.rocks/smartdelay': 3.0.5
broadcast-channel: 7.2.0 broadcast-channel: 7.2.0
'@design.estate/dees-domtools@2.3.6': '@design.estate/dees-domtools@2.3.7':
dependencies: dependencies:
'@api.global/typedrequest': 3.2.5 '@api.global/typedrequest': 3.2.5
'@design.estate/dees-comms': 1.0.30 '@design.estate/dees-comms': 1.0.30
@@ -4586,8 +4583,8 @@ snapshots:
'@push.rocks/webrequest': 3.0.37 '@push.rocks/webrequest': 3.0.37
'@push.rocks/websetup': 3.0.19 '@push.rocks/websetup': 3.0.19
'@push.rocks/webstore': 2.0.20 '@push.rocks/webstore': 2.0.20
lenis: 1.3.15 '@tempfix/lenis': 1.3.17
lit: 3.3.1 lit: 3.3.2
sweet-scroll: 4.0.0 sweet-scroll: 4.0.0
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
@@ -4595,12 +4592,12 @@ snapshots:
- supports-color - supports-color
- vue - vue
'@design.estate/dees-element@2.1.3': '@design.estate/dees-element@2.1.5':
dependencies: dependencies:
'@design.estate/dees-domtools': 2.3.6 '@design.estate/dees-domtools': 2.3.7
'@push.rocks/isounique': 1.0.5 '@push.rocks/isounique': 1.0.5
'@push.rocks/smartrx': 3.0.10 '@push.rocks/smartrx': 3.0.10
lit: 3.3.1 lit: 3.3.2
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
- react - react
@@ -4609,22 +4606,22 @@ snapshots:
'@design.estate/dees-wcctools@1.3.0': '@design.estate/dees-wcctools@1.3.0':
dependencies: dependencies:
'@design.estate/dees-domtools': 2.3.6 '@design.estate/dees-domtools': 2.3.7
'@design.estate/dees-element': 2.1.3 '@design.estate/dees-element': 2.1.5
'@push.rocks/smartdelay': 3.0.5 '@push.rocks/smartdelay': 3.0.5
lit: 3.3.1 lit: 3.3.2
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
- react - react
- supports-color - supports-color
- vue - vue
'@design.estate/dees-wcctools@3.4.0': '@design.estate/dees-wcctools@3.7.1':
dependencies: dependencies:
'@design.estate/dees-domtools': 2.3.6 '@design.estate/dees-domtools': 2.3.7
'@design.estate/dees-element': 2.1.3 '@design.estate/dees-element': 2.1.5
'@push.rocks/smartdelay': 3.0.5 '@push.rocks/smartdelay': 3.0.5
lit: 3.3.1 lit: 3.3.2
transitivePeerDependencies: transitivePeerDependencies:
- '@nuxt/kit' - '@nuxt/kit'
- react - react
@@ -4908,11 +4905,11 @@ snapshots:
'@leichtgewicht/ip-codec@2.0.5': {} '@leichtgewicht/ip-codec@2.0.5': {}
'@lit-labs/ssr-dom-shim@1.4.0': {} '@lit-labs/ssr-dom-shim@1.5.0': {}
'@lit/reactive-element@2.1.1': '@lit/reactive-element@2.1.2':
dependencies: dependencies:
'@lit-labs/ssr-dom-shim': 1.4.0 '@lit-labs/ssr-dom-shim': 1.5.0
'@mixmark-io/domino@2.2.0': {} '@mixmark-io/domino@2.2.0': {}
@@ -5646,7 +5643,7 @@ snapshots:
'@push.rocks/smartntml@2.0.8': '@push.rocks/smartntml@2.0.8':
dependencies: dependencies:
'@design.estate/dees-element': 2.1.3 '@design.estate/dees-element': 2.1.5
'@happy-dom/global-registrator': 15.11.7 '@happy-dom/global-registrator': 15.11.7
'@push.rocks/smartpromise': 4.2.3 '@push.rocks/smartpromise': 4.2.3
fake-indexeddb: 6.2.5 fake-indexeddb: 6.2.5
@@ -5903,7 +5900,7 @@ snapshots:
'@push.rocks/taskbuffer@3.5.0': '@push.rocks/taskbuffer@3.5.0':
dependencies: dependencies:
'@design.estate/dees-element': 2.1.3 '@design.estate/dees-element': 2.1.5
'@push.rocks/lik': 6.2.2 '@push.rocks/lik': 6.2.2
'@push.rocks/smartdelay': 3.0.5 '@push.rocks/smartdelay': 3.0.5
'@push.rocks/smartlog': 3.1.10 '@push.rocks/smartlog': 3.1.10
@@ -6466,6 +6463,8 @@ snapshots:
'@tempfix/idb@8.0.3': {} '@tempfix/idb@8.0.3': {}
'@tempfix/lenis@1.3.17': {}
'@tempfix/webcontainer__api@1.6.1': {} '@tempfix/webcontainer__api@1.6.1': {}
'@tiptap/core@2.27.1(@tiptap/pm@2.27.1)': '@tiptap/core@2.27.1(@tiptap/pm@2.27.1)':
@@ -7801,8 +7800,6 @@ snapshots:
kind-of@6.0.3: {} kind-of@6.0.3: {}
lenis@1.3.15: {}
lines-and-columns@1.2.4: {} lines-and-columns@1.2.4: {}
linkify-it@5.0.0: linkify-it@5.0.0:
@@ -7811,21 +7808,21 @@ snapshots:
linkifyjs@4.3.2: {} linkifyjs@4.3.2: {}
lit-element@4.2.1: lit-element@4.2.2:
dependencies: dependencies:
'@lit-labs/ssr-dom-shim': 1.4.0 '@lit-labs/ssr-dom-shim': 1.5.0
'@lit/reactive-element': 2.1.1 '@lit/reactive-element': 2.1.2
lit-html: 3.3.1 lit-html: 3.3.2
lit-html@3.3.1: lit-html@3.3.2:
dependencies: dependencies:
'@types/trusted-types': 2.0.7 '@types/trusted-types': 2.0.7
lit@3.3.1: lit@3.3.2:
dependencies: dependencies:
'@lit/reactive-element': 2.1.1 '@lit/reactive-element': 2.1.2
lit-element: 4.2.1 lit-element: 4.2.2
lit-html: 3.3.1 lit-html: 3.3.2
locate-path@5.0.0: locate-path@5.0.0:
dependencies: dependencies:

View File

@@ -800,4 +800,114 @@ html`
- **External Router Support**: Integrate with Angular Router or other frameworks - **External Router Support**: Integrate with Angular Router or other frameworks
- **State Persistence**: Save/restore collapsed menus, selections, and current view - **State Persistence**: Save/restore collapsed menus, selections, and current view
- **View-specific Menus**: Each view can define its own secondary menu and tabs - **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

159
readme.md
View File

@@ -1,6 +1,6 @@
# @design.estate/dees-catalog # @design.estate/dees-catalog
A comprehensive web components library built with TypeScript and LitElement, providing **75+ UI components** for building modern web applications with consistent design and behavior. 🚀 A comprehensive web components library built with TypeScript and LitElement, providing **80+ production-ready UI components** for building modern web applications with consistent design and behavior. 🚀
[![TypeScript](https://img.shields.io/badge/TypeScript-5.0+-blue.svg)](https://www.typescriptlang.org/) [![TypeScript](https://img.shields.io/badge/TypeScript-5.0+-blue.svg)](https://www.typescriptlang.org/)
[![LitElement](https://img.shields.io/badge/LitElement-4.0+-orange.svg)](https://lit.dev/) [![LitElement](https://img.shields.io/badge/LitElement-4.0+-orange.svg)](https://lit.dev/)
@@ -53,13 +53,14 @@ For developers working on this library, please refer to the [UI Components Playb
| Category | Components | | Category | Components |
|----------|------------| |----------|------------|
| **Core UI** | [`DeesButton`](#deesbutton), [`DeesButtonExit`](#deesbuttonexit), [`DeesButtonGroup`](#deesbuttongroup), [`DeesBadge`](#deesbadge), [`DeesChips`](#deeschips), [`DeesHeading`](#deesheading), [`DeesHint`](#deeshint), [`DeesIcon`](#deesicon), [`DeesLabel`](#deeslabel), [`DeesPanel`](#deespanel), [`DeesSearchbar`](#deessearchbar), [`DeesSpinner`](#deesspinner), [`DeesToast`](#deestoast), [`DeesWindowcontrols`](#deeswindowcontrols) | | **Core UI** | [`DeesButton`](#deesbutton), [`DeesButtonExit`](#deesbuttonexit), [`DeesButtonGroup`](#deesbuttongroup), [`DeesBadge`](#deesbadge), [`DeesChips`](#deeschips), [`DeesHeading`](#deesheading), [`DeesHint`](#deeshint), [`DeesIcon`](#deesicon), [`DeesLabel`](#deeslabel), [`DeesPanel`](#deespanel), [`DeesSearchbar`](#deessearchbar), [`DeesSpinner`](#deesspinner), [`DeesToast`](#deestoast), [`DeesWindowcontrols`](#deeswindowcontrols) |
| **Forms** | [`DeesForm`](#deesform), [`DeesInputText`](#deesinputtext), [`DeesInputCheckbox`](#deesinputcheckbox), [`DeesInputDropdown`](#deesinputdropdown), [`DeesInputRadiogroup`](#deesinputradiogroup), [`DeesInputFileupload`](#deesinputfileupload), [`DeesInputIban`](#deesinputiban), [`DeesInputPhone`](#deesinputphone), [`DeesInputQuantitySelector`](#deesinputquantityselector), [`DeesInputMultitoggle`](#deesinputmultitoggle), [`DeesInputTags`](#deesinputtags), [`DeesInputTypelist`](#deesinputtypelist), [`DeesInputRichtext`](#deesinputrichtext), [`DeesInputWysiwyg`](#deesinputwysiwyg), [`DeesInputDatepicker`](#deesinputdatepicker), [`DeesInputSearchselect`](#deesinputsearchselect), [`DeesFormSubmit`](#deesformsubmit) | | **Forms** | [`DeesForm`](#deesform), [`DeesInputText`](#deesinputtext), [`DeesInputCheckbox`](#deesinputcheckbox), [`DeesInputDropdown`](#deesinputdropdown), [`DeesInputRadiogroup`](#deesinputradiogroup), [`DeesInputFileupload`](#deesinputfileupload), [`DeesInputIban`](#deesinputiban), [`DeesInputPhone`](#deesinputphone), [`DeesInputQuantitySelector`](#deesinputquantityselector), [`DeesInputMultitoggle`](#deesinputmultitoggle), [`DeesInputTags`](#deesinputtags), [`DeesInputTypelist`](#deesinputtypelist), [`DeesInputList`](#deesinputlist), [`DeesInputProfilepicture`](#deesinputprofilepicture), [`DeesInputRichtext`](#deesinputrichtext), [`DeesInputWysiwyg`](#deesinputwysiwyg), [`DeesInputDatepicker`](#deesinputdatepicker), [`DeesInputSearchselect`](#deesinputsearchselect), [`DeesFormSubmit`](#deesformsubmit) |
| **Layout** | [`DeesAppui`](#deesappui), [`DeesAppuiMainmenu`](#deesappuimainmenu), [`DeesAppuiSecondarymenu`](#deesappuisecondarymenu), [`DeesAppuiMaincontent`](#deesappuimaincontent), [`DeesAppuiAppbar`](#deesappuiappbar), [`DeesAppuiActivitylog`](#deesappuiactivitylog), [`DeesAppuiProfiledropdown`](#deesappuiprofiledropdown), [`DeesAppuiTabs`](#deesappuitabs), [`DeesMobileNavigation`](#deesmobilenavigation), [`DeesDashboardGrid`](#deesdashboardgrid) | | **Layout** | [`DeesAppui`](#deesappui), [`DeesAppuiMainmenu`](#deesappuimainmenu), [`DeesAppuiSecondarymenu`](#deesappuisecondarymenu), [`DeesAppuiMaincontent`](#deesappuimaincontent), [`DeesAppuiAppbar`](#deesappuiappbar), [`DeesAppuiActivitylog`](#deesappuiactivitylog), [`DeesAppuiProfiledropdown`](#deesappuiprofiledropdown), [`DeesAppuiTabs`](#deesappuitabs), [`DeesMobileNavigation`](#deesmobilenavigation), [`DeesDashboardGrid`](#deesdashboardgrid) |
| **Data Display** | [`DeesTable`](#deestable), [`DeesDataviewCodebox`](#deesdataviewcodebox), [`DeesDataviewStatusobject`](#deesdataviewstatusobject), [`DeesPdf`](#deespdf), [`DeesStatsGrid`](#deesstatsgrid), [`DeesPagination`](#deespagination) | | **Data Display** | [`DeesTable`](#deestable), [`DeesDataviewCodebox`](#deesdataviewcodebox), [`DeesDataviewStatusobject`](#deesdataviewstatusobject), [`DeesPdf`](#deespdf), [`DeesStatsGrid`](#deesstatsgrid), [`DeesPagination`](#deespagination) |
| **Visualization** | [`DeesChartArea`](#deeschartarea), [`DeesChartLog`](#deeschartlog) | | **Visualization** | [`DeesChartArea`](#deeschartarea), [`DeesChartLog`](#deeschartlog) |
| **Dialogs & Overlays** | [`DeesModal`](#deesmodal), [`DeesContextmenu`](#deescontextmenu), [`DeesSpeechbubble`](#deesspeechbubble), [`DeesWindowlayer`](#deeswindowlayer) | | **Dialogs & Overlays** | [`DeesModal`](#deesmodal), [`DeesContextmenu`](#deescontextmenu), [`DeesSpeechbubble`](#deesspeechbubble), [`DeesWindowlayer`](#deeswindowlayer) |
| **Navigation** | [`DeesStepper`](#deesstepper), [`DeesProgressbar`](#deesprogressbar) | | **Navigation** | [`DeesStepper`](#deesstepper), [`DeesProgressbar`](#deesprogressbar) |
| **Development** | [`DeesEditor`](#deeseditor), [`DeesEditorMarkdown`](#deeseditormarkdown), [`DeesEditorMarkdownoutlet`](#deeseditormarkdownoutlet), [`DeesTerminal`](#deesterminal), [`DeesUpdater`](#deesupdater) | | **Development** | [`DeesEditor`](#deeseditor), [`DeesEditorMarkdown`](#deeseditormarkdown), [`DeesEditorMarkdownoutlet`](#deeseditormarkdownoutlet), [`DeesTerminal`](#deesterminal), [`DeesUpdater`](#deesupdater) |
| **Theming** | [`DeesTheme`](#deestheme) |
| **Auth & Utilities** | [`DeesSimpleAppdash`](#deessimpleappdash), [`DeesSimpleLogin`](#deessimplelogin) | | **Auth & Utilities** | [`DeesSimpleAppdash`](#deessimpleappdash), [`DeesSimpleLogin`](#deessimplelogin) |
| **Shopping** | [`DeesShoppingProductcard`](#deesshoppingproductcard) | | **Shopping** | [`DeesShoppingProductcard`](#deesshoppingproductcard) |
@@ -482,6 +483,62 @@ Dynamic list input for managing arrays of typed values.
></dees-input-typelist> ></dees-input-typelist>
``` ```
#### `DeesInputList`
Advanced list input with drag-and-drop reordering, inline editing, and validation.
```typescript
<dees-input-list
key="items"
label="List Items"
placeholder="Add new item..."
.value=${['Item 1', 'Item 2', 'Item 3']}
maxItems={10} // Optional: maximum items
minItems={1} // Optional: minimum items
allowDuplicates={false} // Optional: allow duplicate values
sortable={true} // Optional: enable drag-and-drop reordering
confirmDelete={true} // Optional: confirm before deletion
@change=${handleListChange}
></dees-input-list>
```
**Key Features:**
- Add, edit, and remove items inline
- Drag-and-drop reordering with visual feedback
- Optional duplicate prevention
- Min/max item constraints
- Delete confirmation dialog
- Full keyboard support
- Form validation integration
#### `DeesInputProfilepicture`
Profile picture input with cropping, zoom, and image processing.
```typescript
<dees-input-profilepicture
key="avatar"
label="Profile Picture"
shape="round" // Options: round, square
size={120} // Display size in pixels
.value=${imageBase64} // Base64 encoded image or URL
allowUpload={true} // Enable upload button
allowDelete={true} // Enable delete button
maxFileSize={5242880} // Max file size in bytes (5MB)
.acceptedFormats=${['image/jpeg', 'image/png', 'image/webp']}
outputSize={800} // Output resolution in pixels
outputQuality={0.95} // JPEG quality (0-1)
@change=${handleAvatarChange}
></dees-input-profilepicture>
```
**Key Features:**
- Interactive cropping modal with zoom and pan
- Drag-and-drop file upload
- Round or square output shapes
- Configurable output size and quality
- File size and format validation
- Delete functionality
- Preview on hover
#### `DeesInputDatepicker` #### `DeesInputDatepicker`
Date and time picker component with calendar interface and manual typing support. Date and time picker component with calendar interface and manual typing support.
@@ -661,6 +718,7 @@ class MyApp extends DeesElement {
- **Hash-based Routing**: Automatic URL synchronization with view navigation - **Hash-based Routing**: Automatic URL synchronization with view navigation
- **RxJS Observables**: `viewChanged$` and `viewLifecycle$` for reactive programming - **RxJS Observables**: `viewChanged$` and `viewLifecycle$` for reactive programming
- **TypeScript-first**: Typed `IViewActivationContext` passed to views on activation - **TypeScript-first**: Typed `IViewActivationContext` passed to views on activation
- **Activity Log Toggle**: Built-in appbar button to show/hide activity panel with entry count badge
**Programmatic APIs include:** **Programmatic APIs include:**
- `navigateToView(viewId, params?)` - Navigate between views - `navigateToView(viewId, params?)` - Navigate between views
@@ -670,6 +728,7 @@ class MyApp extends DeesElement {
- `setSecondaryMenu()`, `setSecondaryMenuCollapsed()`, `setSecondaryMenuVisible()` - Control secondary menu - `setSecondaryMenu()`, `setSecondaryMenuCollapsed()`, `setSecondaryMenuVisible()` - Control secondary menu
- `setContentTabs()`, `setContentTabsVisible()` - Control view-specific UI - `setContentTabs()`, `setContentTabsVisible()` - Control view-specific UI
- `activityLog.add()`, `activityLog.addMany()`, `activityLog.clear()` - Manage activity entries - `activityLog.add()`, `activityLog.addMany()`, `activityLog.clear()` - Manage activity entries
- `setActivityLogVisible()`, `toggleActivityLog()`, `getActivityLogVisible()` - Control activity panel
**View Visibility Control:** **View Visibility Control:**
```typescript ```typescript
@@ -740,7 +799,7 @@ Main content area with tab management support.
``` ```
#### `DeesAppuiAppbar` #### `DeesAppuiAppbar`
Professional application bar component with hierarchical menus, breadcrumb navigation, and user account management. Professional application bar component with hierarchical menus, breadcrumb navigation, user account management, and activity log toggle.
```typescript ```typescript
<dees-appui-appbar <dees-appui-appbar
@@ -775,6 +834,9 @@ Professional application bar component with hierarchical menus, breadcrumb navig
.breadcrumbs=${'Project > src > components'} .breadcrumbs=${'Project > src > components'}
.showWindowControls=${true} .showWindowControls=${true}
.showSearch=${true} .showSearch=${true}
.showActivityLogToggle=${true}
.activityLogCount=${5}
.activityLogActive=${false}
.user=${{ .user=${{
name: 'John Doe', name: 'John Doe',
avatar: '/path/to/avatar.jpg', avatar: '/path/to/avatar.jpg',
@@ -782,6 +844,7 @@ Professional application bar component with hierarchical menus, breadcrumb navig
}} }}
@menu-select=${(e) => handleMenuSelect(e.detail.item)} @menu-select=${(e) => handleMenuSelect(e.detail.item)}
@breadcrumb-navigate=${(e) => handleBreadcrumbClick(e.detail)} @breadcrumb-navigate=${(e) => handleBreadcrumbClick(e.detail)}
@activity-toggle=${() => handleActivityToggle()}
></dees-appui-appbar> ></dees-appui-appbar>
``` ```
@@ -789,9 +852,40 @@ Professional application bar component with hierarchical menus, breadcrumb navig
- **Hierarchical Menu System** - Top-level menus with dropdown submenus, icons, and keyboard shortcuts - **Hierarchical Menu System** - Top-level menus with dropdown submenus, icons, and keyboard shortcuts
- **Keyboard Navigation** - Full keyboard support (Tab, Arrow keys, Enter, Escape) - **Keyboard Navigation** - Full keyboard support (Tab, Arrow keys, Enter, Escape)
- **Breadcrumb Navigation** - Customizable breadcrumb trail with click events - **Breadcrumb Navigation** - Customizable breadcrumb trail with click events
- **User Account Section** - Avatar with status indicator - **User Account Section** - Avatar with status indicator and profile dropdown
- **Activity Log Toggle** - Button with badge count to show/hide activity panel
- **Accessibility** - Full ARIA support with menubar roles - **Accessibility** - Full ARIA support with menubar roles
#### `DeesAppuiActivitylog`
Real-time activity log panel for displaying user actions and system events.
```typescript
<dees-appui-activitylog></dees-appui-activitylog>
// Programmatic API
activityLog.add({
type: 'update', // Options: login, logout, view, create, update, delete, custom
user: 'John Doe',
message: 'Updated project settings',
iconName: 'lucide:settings' // Optional: custom icon
});
activityLog.addMany(entries); // Add multiple entries
activityLog.clear(); // Clear all entries
activityLog.getEntries(); // Get all entries
activityLog.filter({ user: 'John' }); // Filter by user/type
activityLog.search('settings'); // Search by message
```
**Key Features:**
- Stacked entry layout with icon, user, timestamp, and message
- Date grouping (Today, Yesterday, etc.)
- Search and filter functionality
- Context menu for entry actions
- Live streaming indicator
- Animated slide-in/out panel
- Theme-aware styling
#### `DeesAppuiTabs` #### `DeesAppuiTabs`
Reusable tab component with horizontal/vertical layout support. Reusable tab component with horizontal/vertical layout support.
@@ -1089,6 +1183,40 @@ Progress indicator component for tracking completion status.
--- ---
### Theming Components
#### `DeesTheme`
Theme provider component that wraps children and provides CSS custom properties for consistent theming.
```typescript
// Basic usage - wrap your app
<dees-theme>
<my-app></my-app>
</dees-theme>
// With custom overrides
<dees-theme
.customColors=${{
primary: '#007bff',
success: '#28a745'
}}
.customSpacing=${{
lg: '24px',
xl: '32px'
}}
>
<my-section></my-section>
</dees-theme>
```
**Key Features:**
- Provides CSS custom properties for colors, spacing, radius, shadows, and transitions
- Can be nested for section-specific theming
- Works with dark/light mode
- Overrides cascade to all child components
---
### Development Components ### Development Components
#### `DeesEditor` #### `DeesEditor`
@@ -1241,6 +1369,29 @@ interface IMenuGroup {
collapsed?: boolean; collapsed?: boolean;
iconName?: string; iconName?: string;
} }
// View definition for app navigation
interface IViewDefinition {
id: string;
name: string;
iconName?: string;
content: string | HTMLElement | (() => TemplateResult);
secondaryMenu?: ISecondaryMenuGroup[];
contentTabs?: IMenuItem[];
route?: string;
badge?: string | number;
}
// Activity log entry
interface IActivityEntry {
id?: string;
timestamp?: Date;
type: 'login' | 'logout' | 'view' | 'create' | 'update' | 'delete' | 'custom';
user: string;
message: string;
iconName?: string;
data?: Record<string, unknown>;
}
``` ```
--- ---

View File

@@ -3,6 +3,6 @@
*/ */
export const commitinfo = { export const commitinfo = {
name: '@design.estate/dees-catalog', name: '@design.estate/dees-catalog',
version: '3.28.1', version: '3.32.0',
description: 'A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.' description: 'A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.'
} }

View File

@@ -20,6 +20,7 @@ import { themeDefaultStyles } from '../../00theme.js';
export class DeesAppuiActivitylog extends DeesElement implements IActivityLogAPI { export class DeesAppuiActivitylog extends DeesElement implements IActivityLogAPI {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// INSTANCE PROPERTIES // INSTANCE PROPERTIES
@state() @state()
@@ -122,6 +123,7 @@ export class DeesAppuiActivitylog extends DeesElement implements IActivityLogAPI
width: 100%; width: 100%;
padding: 8px 0; padding: 8px 0;
overflow-y: auto; overflow-y: auto;
overscroll-behavior: contain;
scrollbar-width: thin; scrollbar-width: thin;
scrollbar-color: ${cssManager.bdTheme('#d4d4d4', '#333333')} transparent; scrollbar-color: ${cssManager.bdTheme('#d4d4d4', '#333333')} transparent;
} }

View File

@@ -28,6 +28,7 @@ declare global {
@customElement('dees-appui-appbar') @customElement('dees-appui-appbar')
export class DeesAppuiBar extends DeesElement { export class DeesAppuiBar extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// INSTANCE PROPERTIES // INSTANCE PROPERTIES
@property({ type: Array }) @property({ type: Array })

View File

@@ -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>
`;
};

View File

@@ -0,0 +1,315 @@
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;
public static demoGroup = 'App UI';
// 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 = [];
}
}

View File

@@ -0,0 +1 @@
export * from './dees-appui-bottombar.js';

View File

@@ -31,6 +31,7 @@ export class DeesAppuiMaincontent extends DeesElement {
</div> </div>
</dees-appui-maincontent> </dees-appui-maincontent>
`; `;
public static demoGroup = 'App UI';
// INSTANCE // INSTANCE
@property({ @property({

View File

@@ -22,6 +22,7 @@ import { themeDefaultStyles } from '../../00theme.js';
@customElement('dees-appui-mainmenu') @customElement('dees-appui-mainmenu')
export class DeesAppuiMainmenu extends DeesElement { export class DeesAppuiMainmenu extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// INSTANCE // INSTANCE
@@ -164,6 +165,7 @@ export class DeesAppuiMainmenu extends DeesElement {
flex: 1; flex: 1;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; overflow-x: hidden;
overscroll-behavior: contain;
padding: 8px 0; padding: 8px 0;
} }

View File

@@ -35,6 +35,7 @@ export class DeesAppuiProfileDropdown extends DeesElement {
.isOpen=${true} .isOpen=${true}
></dees-appui-profiledropdown> ></dees-appui-profiledropdown>
`; `;
public static demoGroup = 'App UI';
@property({ type: Object }) @property({ type: Object })
accessor user: { accessor user: {
@@ -285,6 +286,7 @@ export class DeesAppuiProfileDropdown extends DeesElement {
max-width: calc(100vw - 32px); max-width: calc(100vw - 32px);
max-height: calc(100vh - 32px); max-height: calc(100vh - 32px);
overflow-y: auto; overflow-y: auto;
overscroll-behavior: contain;
} }
:host([isopen]) .dropdown { :host([isopen]) .dropdown {

View File

@@ -33,6 +33,7 @@ import { themeDefaultStyles } from '../../00theme.js';
@customElement('dees-appui-secondarymenu') @customElement('dees-appui-secondarymenu')
export class DeesAppuiSecondarymenu extends DeesElement { export class DeesAppuiSecondarymenu extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// INSTANCE // INSTANCE
@@ -201,6 +202,7 @@ export class DeesAppuiSecondarymenu extends DeesElement {
flex: 1; flex: 1;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; overflow-x: hidden;
overscroll-behavior: contain;
padding: 8px 0; padding: 8px 0;
} }

View File

@@ -18,6 +18,7 @@ import { themeDefaultStyles } from '../../00theme.js';
@customElement('dees-appui-tabs') @customElement('dees-appui-tabs')
export class DeesAppuiTabs extends DeesElement { export class DeesAppuiTabs extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// INSTANCE // INSTANCE
@property({ @property({
@@ -116,6 +117,7 @@ export class DeesAppuiTabs extends DeesElement {
font-size: 14px; font-size: 14px;
overflow-x: auto; overflow-x: auto;
overflow-y: hidden; overflow-y: hidden;
overscroll-behavior: contain;
scrollbar-width: thin; scrollbar-width: thin;
scrollbar-color: transparent transparent; scrollbar-color: transparent transparent;
height: 100%; height: 100%;

View File

@@ -663,6 +663,44 @@ export const demoFunc = () => {
defaultView: 'dashboard', 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) => { onViewChange: (viewId, view) => {
console.log(`View changed to: ${viewId} (${view.name})`); console.log(`View changed to: ${viewId} (${view.name})`);
}, },

View File

@@ -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 { DeesAppuiSecondarymenu } from '../dees-appui-secondarymenu/dees-appui-secondarymenu.js';
import type { DeesAppuiMaincontent } from '../dees-appui-maincontent/dees-appui-maincontent.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 { 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 { demoFunc } from './dees-appui.demo.js';
import { themeDefaultStyles } from '../../00theme.js'; import { themeDefaultStyles } from '../../00theme.js';
@@ -23,6 +24,7 @@ import { ViewRegistry } from './view.registry.js';
// Import child components // Import child components
import '../dees-appui-appbar/index.js'; 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-mainmenu/dees-appui-mainmenu.js';
import '../dees-appui-secondarymenu/dees-appui-secondarymenu.js'; import '../dees-appui-secondarymenu/dees-appui-secondarymenu.js';
import '../dees-appui-maincontent/dees-appui-maincontent.js'; import '../dees-appui-maincontent/dees-appui-maincontent.js';
@@ -37,6 +39,7 @@ declare global {
@customElement('dees-appui') @customElement('dees-appui')
export class DeesAppui extends DeesElement { export class DeesAppui extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'App UI';
// ========================================== // ==========================================
// REACTIVE OBSERVABLES (RxJS Subjects) // REACTIVE OBSERVABLES (RxJS Subjects)
@@ -156,6 +159,12 @@ export class DeesAppui extends DeesElement {
@state() @state()
accessor activitylogElement: DeesAppuiActivitylog | undefined = undefined; accessor activitylogElement: DeesAppuiActivitylog | undefined = undefined;
@state()
accessor bottombarElement: DeesAppuiBottombar | undefined = undefined;
@state()
accessor bottombarVisible: boolean = true;
// Current view state // Current view state
@state() @state()
accessor currentView: interfaces.IViewDefinition | undefined = undefined; accessor currentView: interfaces.IViewDefinition | undefined = undefined;
@@ -179,12 +188,25 @@ export class DeesAppui extends DeesElement {
.maingrid { .maingrid {
position: absolute; position: absolute;
top: 40px; top: 40px;
height: calc(100% - 40px); height: calc(100% - 40px - 24px);
width: 100%; width: 100%;
display: grid; display: grid;
/* grid-template-columns set dynamically in template */ /* grid-template-columns set dynamically in template */
grid-template-rows: 1fr; 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 */ /* Z-index layering for proper stacking */
@@ -201,6 +223,7 @@ export class DeesAppui extends DeesElement {
.maingrid > dees-appui-maincontent { .maingrid > dees-appui-maincontent {
position: relative; position: relative;
z-index: 1; z-index: 1;
min-height: 0;
} }
.maingrid > dees-appui-activitylog { .maingrid > dees-appui-activitylog {
@@ -293,6 +316,9 @@ export class DeesAppui extends DeesElement {
class="${this.activityLogVisible ? 'visible' : 'hidden'}" class="${this.activityLogVisible ? 'visible' : 'hidden'}"
></dees-appui-activitylog> ></dees-appui-activitylog>
</div> </div>
${this.bottombarVisible ? html`
<dees-appui-bottombar></dees-appui-bottombar>
` : ''}
`; `;
} }
@@ -303,6 +329,7 @@ export class DeesAppui extends DeesElement {
this.secondarymenu = this.shadowRoot!.querySelector('dees-appui-secondarymenu') as DeesAppuiSecondarymenu; this.secondarymenu = this.shadowRoot!.querySelector('dees-appui-secondarymenu') as DeesAppuiSecondarymenu;
this.maincontent = this.shadowRoot!.querySelector('dees-appui-maincontent') as DeesAppuiMaincontent; this.maincontent = this.shadowRoot!.querySelector('dees-appui-maincontent') as DeesAppuiMaincontent;
this.activitylogElement = this.shadowRoot!.querySelector('dees-appui-activitylog') as DeesAppuiActivitylog; 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 // Subscribe to activity log entry changes for badge count
if (this.activitylogElement) { if (this.activitylogElement) {
@@ -728,6 +755,72 @@ export class DeesAppui extends DeesElement {
return this.activityLogVisible; 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 // PROGRAMMATIC API: NAVIGATION
// ========================================== // ==========================================
@@ -840,6 +933,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 // Setup domtools.router integration
this.setupRouterIntegration(config); this.setupRouterIntegration(config);

View File

@@ -1,6 +1,6 @@
# DeesAppui # DeesAppui
A comprehensive application shell component providing a complete UI framework with navigation, menus, activity logging, and view management. A comprehensive application shell component providing a complete UI framework with navigation, menus, activity logging, and view management. 🚀
## Quick Start ## Quick Start
@@ -35,6 +35,34 @@ class MyApp extends DeesElement {
} }
``` ```
## Architecture Overview
The DeesAppui shell consists of several interconnected components:
```
┌─────────────────────────────────────────────────────────────────────┐
│ AppBar (dees-appui-appbar) │
│ ├── Menus (File, Edit, View...) │
│ ├── Breadcrumbs │
│ ├── User Profile + Dropdown │
│ └── Activity Log Toggle │
├─────────────┬───────────────────────────────────┬───────────────────┤
│ Main Menu │ Content Area │ Activity Log │
│ (collapsed/ │ ├── Content Tabs │ (slide panel) │
│ expanded) │ │ (closable, from tables/lists)│ │
│ │ └── View Container │ │
│ ┌─────────┐ │ └── Active View │ │
│ │ 🏠 Home │ ├─────────────────────────────────┐ │ │
│ │ 📁 Files│ │ Secondary Menu │ │ │
│ │ ⚙ Settings ├── Collapsible Groups │ │ │
│ │ │ │ ├── Item 1 │ │ │
│ └─────────┘ │ ├── Item 2 (with badge) │ │ │
│ │ └── Item 3 │ │ │
└─────────────┴─────────────────────────────────┴───────────────────────┘
```
---
## Configuration API ## Configuration API
### `configure(config: IAppConfig)` ### `configure(config: IAppConfig)`
@@ -155,74 +183,289 @@ appui.removeMainMenuItem('Main', 'tasks');
// Selection // Selection
appui.setMainMenuSelection('dashboard'); appui.setMainMenuSelection('dashboard');
appui.setMainMenuCollapsed(true);
// Visibility control
appui.setMainMenuCollapsed(true); // Collapse to icon-only sidebar
appui.setMainMenuVisible(false); // Hide completely
// Badges // Badges
appui.setMainMenuBadge('inbox', 12); appui.setMainMenuBadge('inbox', 12);
appui.clearMainMenuBadge('inbox'); appui.clearMainMenuBadge('inbox');
``` ```
### Secondary Menu API ---
Views can control the secondary (contextual) menu. ## Secondary Menu API 📋
The secondary menu is a contextual sidebar that appears next to the main content area. It supports **collapsible groups** with icons and badges, making it perfect for:
- **Settings pages** (grouped settings categories)
- **File browsers** (folder trees)
- **Project navigation** (grouped by category)
- **Documentation** (chapters/sections)
### Collapsible Groups
Groups can be collapsed/expanded by clicking the group header. The state is visually indicated with an icon rotation.
```typescript ```typescript
// Set menu // Set secondary menu with collapsible groups
appui.setSecondaryMenu({ appui.setSecondaryMenu({
heading: 'Settings', heading: 'Settings',
groups: [ groups: [
{ {
name: 'Account', name: 'Account',
iconName: 'lucide:user', // Group icon
collapsed: false, // Initial state (default: false)
items: [ items: [
{ key: 'profile', iconName: 'lucide:user', action: () => {} }, { key: 'profile', iconName: 'lucide:user', action: () => showProfile() },
{ key: 'security', iconName: 'lucide:shield', action: () => {} }, { key: 'security', iconName: 'lucide:shield', badge: '!', badgeVariant: 'warning', action: () => showSecurity() },
{ key: 'billing', iconName: 'lucide:credit-card', action: () => showBilling() }
]
},
{
name: 'Preferences',
iconName: 'lucide:settings',
collapsed: true, // Start collapsed
items: [
{ key: 'notifications', iconName: 'lucide:bell', action: () => {} },
{ key: 'appearance', iconName: 'lucide:palette', action: () => {} },
{ key: 'language', iconName: 'lucide:globe', action: () => {} }
] ]
} }
] ]
}); });
```
// Update group ### Secondary Menu Item Properties
appui.updateSecondaryMenuGroup('Account', { items: newItems });
// Add item ```typescript
appui.addSecondaryMenuItem('Account', { interface ISecondaryMenuItem {
key: 'notifications', key: string; // Unique identifier
iconName: 'lucide:bell', iconName?: string; // Icon (e.g., 'lucide:user')
action: () => {} action: () => void; // Click handler
badge?: string | number; // Badge text/count
badgeVariant?: 'default' | 'success' | 'warning' | 'error';
}
interface ISecondaryMenuGroup {
name: string; // Group name (shown in header)
iconName?: string; // Group icon
collapsed?: boolean; // Initial collapsed state
items: ISecondaryMenuItem[]; // Items in this group
}
```
### Updating Secondary Menu
```typescript
// Update a specific group
appui.updateSecondaryMenuGroup('Account', {
items: [...newItems]
}); });
// Selection // Add item to a group
appui.addSecondaryMenuItem('Account', {
key: 'api-keys',
iconName: 'lucide:key',
action: () => showApiKeys()
});
// Selection (highlights the item)
appui.setSecondaryMenuSelection('profile'); appui.setSecondaryMenuSelection('profile');
// Visibility control
appui.setSecondaryMenuCollapsed(true); // Collapse panel
appui.setSecondaryMenuVisible(false); // Hide completely
// Clear // Clear
appui.clearSecondaryMenu(); appui.clearSecondaryMenu();
``` ```
### Content Tabs API ### View-Specific Secondary Menus
Control tabs in the main content area. Each view can define its own secondary menu that appears when the view is activated:
```typescript ```typescript
// Set tabs // In view definition
appui.setContentTabs([ {
{ key: 'code', iconName: 'lucide:code', action: () => {} }, id: 'settings',
{ key: 'preview', iconName: 'lucide:eye', action: () => {} } name: 'Settings',
]); content: 'my-settings-view',
secondaryMenu: [
{
name: 'General',
items: [
{ key: 'account', iconName: 'lucide:user', action: () => {} },
{ key: 'security', iconName: 'lucide:shield', action: () => {} }
]
}
]
}
// Add/remove // Or set dynamically in view's onActivate hook
appui.addContentTab({ key: 'debug', iconName: 'lucide:bug', action: () => {} }); onActivate(context: IViewActivationContext) {
appui.removeContentTab('debug'); context.appui.setSecondaryMenu({
heading: 'Project Files',
// Select groups: [...]
appui.selectContentTab('preview'); });
}
// Get current
const current = appui.getSelectedContentTab();
``` ```
### Activity Log API ---
Add activity entries to the right-side activity log. ## Content Tabs API 📑
Content tabs appear above the main view content. They're designed for **opening multiple items** from tables, lists, or other data sources—similar to browser tabs or IDE editor tabs.
### Common Use Cases
- **Table row details** - Click a row to open it as a tab
- **Document editing** - Open multiple documents
- **Entity inspection** - View customer, order, product details
- **Multi-file editing** - Edit multiple configuration files
### Closable Tabs
Tabs can be closable, allowing users to open items, work with them, and close when done:
```typescript
// Set initial tabs
appui.setContentTabs([
{ key: 'overview', iconName: 'lucide:home', action: () => showOverview() },
{ key: 'activity', iconName: 'lucide:activity', action: () => showActivity() }
]);
// Add a closable tab when user clicks a table row
table.addEventListener('row-click', (e) => {
const item = e.detail.item;
appui.addContentTab({
key: `item-${item.id}`,
label: item.name, // Display label
iconName: 'lucide:file',
closable: true, // Allow closing
action: () => showItemDetails(item)
});
// Select the new tab
appui.selectContentTab(`item-${item.id}`);
});
// Handle tab close
appui.addEventListener('tab-close', (e) => {
const tabKey = e.detail.key;
// Cleanup resources if needed
console.log(`Tab ${tabKey} closed`);
});
```
### Tab Management
```typescript
// Add/remove tabs
appui.addContentTab({
key: 'debug',
iconName: 'lucide:bug',
closable: true,
action: () => {}
});
appui.removeContentTab('debug');
// Select tab
appui.selectContentTab('preview');
// Get current tab
const current = appui.getSelectedContentTab();
// Visibility control
appui.setContentTabsVisible(false); // Hide tab bar
// Auto-hide when only one tab
appui.setContentTabsAutoHide(true, 1); // Hide when ≤ 1 tab
```
### Opening Items from Tables/Lists
A common pattern is opening table rows as closable tabs:
```typescript
@customElement('my-customers-view')
class MyCustomersView extends DeesElement {
private appui: DeesAppui;
onActivate(context: IViewActivationContext) {
this.appui = context.appui;
// Set base tabs
this.appui.setContentTabs([
{ key: 'list', label: 'All Customers', iconName: 'lucide:users', action: () => this.showList() }
]);
}
render() {
return html`
<dees-table
.data=${this.customers}
@row-dblclick=${this.openCustomerTab}
></dees-table>
`;
}
openCustomerTab(e: CustomEvent) {
const customer = e.detail.item;
const tabKey = `customer-${customer.id}`;
// Check if tab already exists
const existingTab = this.appui.getSelectedContentTab();
if (existingTab?.key === tabKey) {
return; // Already viewing this customer
}
// Add new closable tab
this.appui.addContentTab({
key: tabKey,
label: customer.name,
iconName: 'lucide:user',
closable: true,
action: () => this.showCustomerDetails(customer)
});
this.appui.selectContentTab(tabKey);
}
showCustomerDetails(customer: Customer) {
// Render customer details
this.currentView = html`<customer-details .customer=${customer}></customer-details>`;
}
showList() {
this.currentView = html`<dees-table ...></dees-table>`;
}
}
```
---
## Activity Log API 📊
The activity log is a slide-out panel on the right side showing user actions and system events.
### Activity Log Toggle
The appbar includes a toggle button with a badge showing the entry count:
```typescript
// Control visibility
appui.setActivityLogVisible(true); // Show panel
appui.toggleActivityLog(); // Toggle state
const isVisible = appui.getActivityLogVisible();
// The toggle button automatically shows entry count
// Add entries and the badge updates automatically
```
### Adding Entries
```typescript ```typescript
// Add single entry // Add single entry
@@ -234,19 +477,35 @@ appui.activityLog.add({
data: { invoiceId: '123' } // Optional metadata data: { invoiceId: '123' } // Optional metadata
}); });
// Add multiple // Add multiple entries (e.g., from backend)
appui.activityLog.addMany([...entries]); appui.activityLog.addMany([...entries]);
// Clear // Clear all entries
appui.activityLog.clear(); appui.activityLog.clear();
// Query // Query entries
const entries = appui.activityLog.getEntries(); const entries = appui.activityLog.getEntries();
const filtered = appui.activityLog.filter({ user: 'John', type: 'create' }); const filtered = appui.activityLog.filter({ user: 'John', type: 'create' });
const searched = appui.activityLog.search('invoice'); const searched = appui.activityLog.search('invoice');
``` ```
### Navigation API ### Activity Entry Types
Each type has a default icon that can be overridden:
| Type | Default Icon | Use Case |
|------|--------------|----------|
| `login` | `lucide:log-in` | User sign-in |
| `logout` | `lucide:log-out` | User sign-out |
| `view` | `lucide:eye` | Page/item viewed |
| `create` | `lucide:plus` | New item created |
| `update` | `lucide:pencil` | Item modified |
| `delete` | `lucide:trash` | Item deleted |
| `custom` | `lucide:activity` | Custom events |
---
## Navigation API
Navigate between views programmatically. Navigate between views programmatically.
@@ -512,6 +771,7 @@ class CrmSettings extends DeesElement {
groups: [ groups: [
{ {
name: 'Account', name: 'Account',
iconName: 'lucide:user',
items: [ items: [
{ key: 'profile', iconName: 'lucide:user', action: () => this.showSection('profile') }, { key: 'profile', iconName: 'lucide:user', action: () => this.showSection('profile') },
{ key: 'security', iconName: 'lucide:shield', action: () => this.showSection('security') } { key: 'security', iconName: 'lucide:shield', action: () => this.showSection('security') }
@@ -519,6 +779,8 @@ class CrmSettings extends DeesElement {
}, },
{ {
name: 'Preferences', name: 'Preferences',
iconName: 'lucide:settings',
collapsed: true,
items: [ items: [
{ key: 'notifications', iconName: 'lucide:bell', action: () => this.showSection('notifications') } { key: 'notifications', iconName: 'lucide:bell', action: () => this.showSection('notifications') }
] ]
@@ -557,4 +819,5 @@ All interfaces are exported from `@design.estate/dees-catalog`:
- `IAppBarMenuItem` - App bar menu item - `IAppBarMenuItem` - App bar menu item
- `IMainMenuConfig` - Main menu configuration - `IMainMenuConfig` - Main menu configuration
- `ISecondaryMenuGroup` - Secondary menu group - `ISecondaryMenuGroup` - Secondary menu group
- `ITab` - Tab definition - `ISecondaryMenuItem` - Secondary menu item
- `IMenuItem` - Tab/menu item definition

View File

@@ -1,6 +1,7 @@
// App UI Components // App UI Components
export * from './dees-appui-activitylog/index.js'; export * from './dees-appui-activitylog/index.js';
export * from './dees-appui-appbar/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/index.js';
export * from './dees-appui-maincontent/index.js'; export * from './dees-appui-maincontent/index.js';
export * from './dees-appui-mainmenu/index.js'; export * from './dees-appui-mainmenu/index.js';

View File

@@ -16,6 +16,7 @@ export class DeesButtonExit extends DeesElement {
public static demo = () => html` public static demo = () => html`
<dees-button-exit></dees-button-exit> <dees-button-exit></dees-button-exit>
`; `;
public static demoGroup = 'Button';
// INSTANCE // INSTANCE
@property({ @property({

View File

@@ -21,6 +21,7 @@ declare global {
@customElement('dees-button-group') @customElement('dees-button-group')
export class DeesButtonGroup extends DeesElement { export class DeesButtonGroup extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Button';
@property() @property()
accessor label: string = ''; accessor label: string = '';

View File

@@ -23,6 +23,7 @@ declare global {
@customElement('dees-button') @customElement('dees-button')
export class DeesButton extends DeesElement { export class DeesButton extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Button';
@property({ @property({
reflect: true, reflect: true,

View File

@@ -23,6 +23,7 @@ declare global {
@customElement('dees-chart-area') @customElement('dees-chart-area')
export class DeesChartArea extends DeesElement { export class DeesChartArea extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Chart';
// instance // instance
@state() @state()

View File

@@ -29,6 +29,7 @@ export interface ILogEntry {
@customElement('dees-chart-log') @customElement('dees-chart-log')
export class DeesChartLog extends DeesElement { export class DeesChartLog extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Chart';
@property() @property()
accessor label: string = 'Server Logs'; accessor label: string = 'Server Logs';

View File

@@ -27,6 +27,7 @@ declare global {
@customElement('dees-dataview-codebox') @customElement('dees-dataview-codebox')
export class DeesDataviewCodebox extends DeesElement { export class DeesDataviewCodebox extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Data View';
@property() @property()
accessor progLang: string = 'typescript'; accessor progLang: string = 'typescript';

View File

@@ -27,6 +27,7 @@ declare global {
@customElement('dees-dataview-statusobject') @customElement('dees-dataview-statusobject')
export class DeesDataviewStatusobject extends DeesElement { export class DeesDataviewStatusobject extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Data View';
@property({ type: Object }) accessor statusObject: tsclass.code.IStatusObject; @property({ type: Object }) accessor statusObject: tsclass.code.IStatusObject;

View File

@@ -19,6 +19,7 @@ declare global {
@customElement('dees-form-submit') @customElement('dees-form-submit')
export class DeesFormSubmit extends DeesElement { export class DeesFormSubmit extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Form';
@property({ @property({
type: Boolean, type: Boolean,

View File

@@ -65,6 +65,7 @@ declare global {
@customElement('dees-form') @customElement('dees-form')
export class DeesForm extends DeesElement { export class DeesForm extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Form';
public name: string = 'myform'; public name: string = 'myform';
public changeSubject = new domtools.plugins.smartrx.rxjs.Subject(); public changeSubject = new domtools.plugins.smartrx.rxjs.Subject();

View File

@@ -21,6 +21,7 @@ declare global {
export class DeesInputCheckbox extends DeesInputBase<DeesInputCheckbox> { export class DeesInputCheckbox extends DeesInputBase<DeesInputCheckbox> {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE

View File

@@ -54,6 +54,7 @@ export class DeesInputCode extends DeesInputBase<string> {
.value=${'const greeting: string = "Hello World";\nconsole.log(greeting);'} .value=${'const greeting: string = "Hello World";\nconsole.log(greeting);'}
></dees-input-code> ></dees-input-code>
`; `;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@property({ type: String }) @property({ type: String })

View File

@@ -22,6 +22,7 @@ declare global {
@customElement('dees-input-datepicker') @customElement('dees-input-datepicker')
export class DeesInputDatepicker extends DeesInputBase<DeesInputDatepicker> { export class DeesInputDatepicker extends DeesInputBase<DeesInputDatepicker> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
@property({ type: String }) @property({ type: String })
accessor value: string = ''; accessor value: string = '';

View File

@@ -22,6 +22,7 @@ declare global {
@customElement('dees-input-dropdown') @customElement('dees-input-dropdown')
export class DeesInputDropdown extends DeesInputBase<DeesInputDropdown> { export class DeesInputDropdown extends DeesInputBase<DeesInputDropdown> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE

View File

@@ -21,6 +21,7 @@ declare global {
@customElement('dees-input-fileupload') @customElement('dees-input-fileupload')
export class DeesInputFileupload extends DeesInputBase<DeesInputFileupload> { export class DeesInputFileupload extends DeesInputBase<DeesInputFileupload> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
@property({ attribute: false }) @property({ attribute: false })
accessor value: File[] = []; accessor value: File[] = [];

View File

@@ -17,6 +17,7 @@ import { themeDefaultStyles } from '../../00theme.js';
export class DeesInputIban extends DeesInputBase<DeesInputIban> { export class DeesInputIban extends DeesInputBase<DeesInputIban> {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@state() @state()

View File

@@ -23,6 +23,7 @@ declare global {
export class DeesInputList extends DeesInputBase<DeesInputList> { export class DeesInputList extends DeesInputBase<DeesInputList> {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@property({ type: Array }) @property({ type: Array })
@@ -64,6 +65,26 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
@state() @state()
accessor dragOverIndex: number = -1; 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 = [ public static styles = [
themeDefaultStyles, themeDefaultStyles,
...DeesInputBase.baseStyles, ...DeesInputBase.baseStyles,
@@ -113,7 +134,7 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
padding: 12px 16px; padding: 12px 16px;
border-bottom: 1px solid ${cssManager.bdTheme('hsl(0 0% 89.8%)', 'hsl(0 0% 14.9%)')}; 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%)')}; 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; position: relative;
overflow: hidden; /* Prevent animation from affecting scroll bounds */ overflow: hidden; /* Prevent animation from affecting scroll bounds */
} }
@@ -122,20 +143,31 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
border-bottom: none; 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%)')}; background: ${cssManager.bdTheme('hsl(0 0% 97.5%)', 'hsl(0 0% 6.9%)')};
} }
/* Dragging item - follows cursor */
.list-item.dragging { .list-item.dragging {
opacity: 0.4; position: relative;
background: ${cssManager.bdTheme('hsl(210 40% 96.1%)', 'hsl(215 20.2% 10.8%)')}; 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 { /* Items that need to move up to make space */
background: ${cssManager.bdTheme('hsl(210 40% 93.1%)', 'hsl(215 20.2% 13.8%)')}; .list-item.move-up {
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 51.2%)', 'hsl(217.2 91.2% 59.8%)')}; 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 { .drag-handle {
display: flex; display: flex;
align-items: center; align-items: center;
@@ -313,27 +345,9 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
background: ${cssManager.bdTheme('hsl(0 0% 79.8%)', 'hsl(0 0% 34.9%)')}; background: ${cssManager.bdTheme('hsl(0 0% 79.8%)', 'hsl(0 0% 34.9%)')};
} }
/* Animation for adding/removing items */ /* Disable transitions during drop to prevent flash */
@keyframes slideIn { .list-items.dropping .list-item {
from { transition: none !important;
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;
} }
`, `,
]; ];
@@ -347,12 +361,11 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
<div class="list-items"> <div class="list-items">
${this.value.length > 0 ? this.value.map((item, index) => html` ${this.value.length > 0 ? this.value.map((item, index) => html`
<div <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}" draggable="${this.sortable && !this.disabled}"
@dragstart=${(e: DragEvent) => this.handleDragStart(e, index)} @dragstart=${(e: DragEvent) => this.handleDragStart(e, index)}
@dragend=${this.handleDragEnd} @dragend=${this.handleDragEnd}
@dragover=${(e: DragEvent) => this.handleDragOver(e, index)} @dragover=${(e: DragEvent) => this.handleDragOver(e, index)}
@dragleave=${this.handleDragLeave}
@drop=${(e: DragEvent) => this.handleDrop(e, index)} @drop=${(e: DragEvent) => this.handleDrop(e, index)}
> >
${this.sortable && !this.disabled ? html` ${this.sortable && !this.disabled ? html`
@@ -547,48 +560,313 @@ export class DeesInputList extends DeesInputBase<DeesInputList> {
return confirm(message); return confirm(message);
} }
// Drag and drop handlers // Drag and drop handlers - Interactive implementation
private handleDragStart(e: DragEvent, index: number) { private handleDragStart(e: DragEvent, index: number) {
if (!this.sortable || this.disabled) return; if (!this.sortable || this.disabled) return;
this.draggedIndex = index; this.draggedIndex = index;
this.targetIndex = index;
e.dataTransfer!.effectAllowed = 'move'; e.dataTransfer!.effectAllowed = 'move';
e.dataTransfer!.setData('text/plain', index.toString()); 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() { private handleDragEnd() {
this.draggedIndex = -1; // This is called by the native dragend on the element
this.dragOverIndex = -1; // The actual cleanup is done in handleGlobalDragEnd
this.handleGlobalDragEnd();
} }
private handleDragOver(e: DragEvent, index: number) { private handleDragOver(e: DragEvent, index: number) {
if (!this.sortable || this.disabled) return; if (!this.sortable || this.disabled) return;
e.preventDefault(); e.preventDefault();
e.dataTransfer!.dropEffect = 'move'; e.dataTransfer!.dropEffect = 'move';
this.dragOverIndex = index; // We handle positioning in handleGlobalDragOver now
} }
private handleDragLeave() { private handleDragLeave() {
this.dragOverIndex = -1; // No longer needed for visual feedback - handled by transform
} }
private handleDrop(e: DragEvent, dropIndex: number) { private handleDrop(e: DragEvent, dropIndex: number) {
if (!this.sortable || this.disabled) return; if (!this.sortable || this.disabled) return;
e.preventDefault(); e.preventDefault();
const draggedIndex = parseInt(e.dataTransfer!.getData('text/plain')); // The actual reorder happens in handleGlobalDragEnd
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;
} }
private emitChange() { private emitChange() {

View File

@@ -22,6 +22,7 @@ declare global {
@customElement('dees-input-multitoggle') @customElement('dees-input-multitoggle')
export class DeesInputMultitoggle extends DeesInputBase<DeesInputMultitoggle> { export class DeesInputMultitoggle extends DeesInputBase<DeesInputMultitoggle> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
@property() @property()

View File

@@ -22,6 +22,7 @@ declare global {
export class DeesInputPhone extends DeesInputBase<DeesInputPhone> { export class DeesInputPhone extends DeesInputBase<DeesInputPhone> {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@state() @state()

View File

@@ -13,6 +13,7 @@ declare global {
@customElement('dees-input-quantityselector') @customElement('dees-input-quantityselector')
export class DeesInputQuantitySelector extends DeesInputBase<DeesInputQuantitySelector> { export class DeesInputQuantitySelector extends DeesInputBase<DeesInputQuantitySelector> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE

View File

@@ -21,6 +21,7 @@ type RadioOption = string | { option: string; key: string; payload?: any };
@customElement('dees-input-radiogroup') @customElement('dees-input-radiogroup')
export class DeesInputRadiogroup extends DeesInputBase<string | object> { export class DeesInputRadiogroup extends DeesInputBase<string | object> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE

View File

@@ -28,6 +28,7 @@ declare global {
@customElement('dees-input-richtext') @customElement('dees-input-richtext')
export class DeesInputRichtext extends DeesInputBase<string> { export class DeesInputRichtext extends DeesInputBase<string> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@property({ @property({

View File

@@ -22,6 +22,7 @@ declare global {
export class DeesInputTags extends DeesInputBase<DeesInputTags> { export class DeesInputTags extends DeesInputBase<DeesInputTags> {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@property({ type: Array }) @property({ type: Array })

View File

@@ -22,6 +22,7 @@ declare global {
@customElement('dees-input-text') @customElement('dees-input-text')
export class DeesInputText extends DeesInputBase { export class DeesInputText extends DeesInputBase {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE
@property({ @property({

View File

@@ -16,6 +16,7 @@ import { themeDefaultStyles } from '../../00theme.js';
@customElement('dees-input-typelist') @customElement('dees-input-typelist')
export class DeesInputTypelist extends DeesInputBase<DeesInputTypelist> { export class DeesInputTypelist extends DeesInputBase<DeesInputTypelist> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
// INSTANCE // INSTANCE

View File

@@ -39,6 +39,7 @@ declare global {
@customElement('dees-input-wysiwyg') @customElement('dees-input-wysiwyg')
export class DeesInputWysiwyg extends DeesInputBase<string> { export class DeesInputWysiwyg extends DeesInputBase<string> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
@property({ type: String }) @property({ type: String })
accessor value: string = ''; accessor value: string = '';

View File

@@ -25,6 +25,7 @@ export type ProfileShape = 'square' | 'round';
@customElement('dees-input-profilepicture') @customElement('dees-input-profilepicture')
export class DeesInputProfilePicture extends DeesInputBase<DeesInputProfilePicture> { export class DeesInputProfilePicture extends DeesInputBase<DeesInputProfilePicture> {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Input';
@property({ type: String }) @property({ type: String })
accessor value: string = ''; // Base64 encoded image or URL accessor value: string = ''; // Base64 encoded image or URL

View File

@@ -15,6 +15,7 @@ declare global {
@customElement('dees-pdf-preview') @customElement('dees-pdf-preview')
export class DeesPdfPreview extends DeesElement { export class DeesPdfPreview extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'PDF';
public static styles = previewStyles; public static styles = previewStyles;
@property({ type: String }) @property({ type: String })

View File

@@ -1,6 +1,4 @@
import { DeesElement, property, html, customElement, type TemplateResult } from '@design.estate/dees-element'; import { DeesElement, property, html, customElement, type TemplateResult, directives } from '@design.estate/dees-element';
import { keyed } from 'lit/directives/keyed.js';
import { repeat } from 'lit/directives/repeat.js';
import { PdfManager } from '../dees-pdf-shared/PdfManager.js'; import { PdfManager } from '../dees-pdf-shared/PdfManager.js';
import { viewerStyles } from './styles.js'; import { viewerStyles } from './styles.js';
import { demo as demoFunc } from './demo.js'; import { demo as demoFunc } from './demo.js';
@@ -17,6 +15,7 @@ type RenderState = 'idle' | 'loading' | 'rendering-main' | 'rendering-thumbs' |
@customElement('dees-pdf-viewer') @customElement('dees-pdf-viewer')
export class DeesPdfViewer extends DeesElement { export class DeesPdfViewer extends DeesElement {
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'PDF';
public static styles = viewerStyles; public static styles = viewerStyles;
@property({ type: String }) @property({ type: String })
@@ -196,8 +195,8 @@ export class DeesPdfViewer extends DeesElement {
</button> </button>
</div> </div>
<div class="sidebar-content"> <div class="sidebar-content">
${keyed(this.documentId, html` ${directives.keyed(this.documentId, html`
${repeat( ${directives.repeat(
this.thumbnailData, this.thumbnailData,
(item) => item.page, (item) => item.page,
(item) => html` (item) => html`
@@ -224,7 +223,7 @@ export class DeesPdfViewer extends DeesElement {
</div> </div>
` : html` ` : html`
<div class="pages-container"> <div class="pages-container">
${repeat( ${directives.repeat(
this.pageData, this.pageData,
(item) => item.page, (item) => item.page,
(item) => html` (item) => html`

View File

@@ -21,6 +21,7 @@ declare global {
export class DeesPdf extends DeesElement { export class DeesPdf extends DeesElement {
// DEMO // DEMO
public static demo = () => html` <dees-pdf></dees-pdf> `; public static demo = () => html` <dees-pdf></dees-pdf> `;
public static demoGroup = 'PDF';
// INSTANCE // INSTANCE

View File

@@ -33,6 +33,7 @@ export interface IView {
export class DeesSimpleAppDash extends DeesElement { export class DeesSimpleAppDash extends DeesElement {
// STATIC // STATIC
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Simple';
// INSTANCE // INSTANCE
@property() @property()

View File

@@ -20,7 +20,8 @@ declare global {
@customElement('dees-simple-login') @customElement('dees-simple-login')
export class DeesSimpleLogin extends DeesElement { export class DeesSimpleLogin extends DeesElement {
// STATIC // STATIC
public static demo = demoFunc public static demo = demoFunc;
public static demoGroup = 'Simple';
// INSTANCE // INSTANCE
@property() @property()

View File

@@ -31,6 +31,7 @@ export class DeesWorkspaceDiffEditor extends DeesElement {
.filePath=${'/demo/example.ts'} .filePath=${'/demo/example.ts'}
></dees-workspace-diff-editor> ></dees-workspace-diff-editor>
`; `;
public static demoGroup = 'Workspace';
// INSTANCE // INSTANCE
public diffEditorDeferred = domtools.plugins.smartpromise.defer<monaco.editor.IStandaloneDiffEditor>(); public diffEditorDeferred = domtools.plugins.smartpromise.defer<monaco.editor.IStandaloneDiffEditor>();

View File

@@ -37,6 +37,7 @@ export class DeesWorkspaceFiletree extends DeesElement {
<dees-workspace-filetree></dees-workspace-filetree> <dees-workspace-filetree></dees-workspace-filetree>
</div> </div>
`; `;
public static demoGroup = 'Workspace';
// INSTANCE // INSTANCE
@property({ type: Object }) @property({ type: Object })

View File

@@ -23,6 +23,7 @@ declare global {
@customElement('dees-workspace-markdown') @customElement('dees-workspace-markdown')
export class DeesWorkspaceMarkdown extends DeesElement { export class DeesWorkspaceMarkdown extends DeesElement {
public static demo = () => html`<dees-workspace-markdown></dees-workspace-markdown>`; public static demo = () => html`<dees-workspace-markdown></dees-workspace-markdown>`;
public static demoGroup = 'Workspace';
public static styles = [ public static styles = [
themeDefaultStyles, themeDefaultStyles,

View File

@@ -19,6 +19,7 @@ declare global {
export class DeesWorkspaceMarkdownoutlet extends DeesElement { export class DeesWorkspaceMarkdownoutlet extends DeesElement {
// DEMO // DEMO
public static demo = demoFunc; public static demo = demoFunc;
public static demoGroup = 'Workspace';
public static styles = [ public static styles = [
themeDefaultStyles, themeDefaultStyles,

View File

@@ -23,6 +23,7 @@ declare global {
export class DeesWorkspaceMonaco extends DeesElement { export class DeesWorkspaceMonaco extends DeesElement {
// DEMO // DEMO
public static demo = () => html`<dees-workspace-monaco></dees-workspace-monaco>`; public static demo = () => html`<dees-workspace-monaco></dees-workspace-monaco>`;
public static demoGroup = 'Workspace';
// STATIC // STATIC
public static monacoDeferred: ReturnType<typeof domtools.plugins.smartpromise.defer>; public static monacoDeferred: ReturnType<typeof domtools.plugins.smartpromise.defer>;

View File

@@ -39,6 +39,7 @@ export class DeesWorkspaceTerminalPreview extends DeesElement {
]} ]}
></dees-workspace-terminal-preview> ></dees-workspace-terminal-preview>
`; `;
public static demoGroup = 'Workspace';
/** /**
* The command being displayed (shown in header) * The command being displayed (shown in header)

View File

@@ -37,6 +37,7 @@ export class DeesWorkspaceTerminal extends DeesElement {
const env = new WebContainerEnvironment(); const env = new WebContainerEnvironment();
return html`<dees-workspace-terminal .executionEnvironment=${env}></dees-workspace-terminal>`; return html`<dees-workspace-terminal .executionEnvironment=${env}></dees-workspace-terminal>`;
}; };
public static demoGroup = 'Workspace';
// INSTANCE // INSTANCE
private resizeObserver: ResizeObserver; private resizeObserver: ResizeObserver;

View File

@@ -197,6 +197,7 @@ testSmartPromise();
</dees-demowrapper> </dees-demowrapper>
`; `;
}; };
public static demoGroup = 'Workspace';
// INSTANCE // INSTANCE
@property({ type: Object }) @property({ type: Object })

View File

@@ -257,12 +257,13 @@ export class DeesIcon extends DeesElement {
* @returns Object with type and name properties * @returns Object with type and name properties
*/ */
private parseIconString(iconStr: string): { type: 'fa' | 'lucide', name: string } { private parseIconString(iconStr: string): { type: 'fa' | 'lucide', name: string } {
if (iconStr.startsWith('fa:')) { const lowerStr = iconStr.toLowerCase();
if (lowerStr.startsWith('fa:')) {
return { return {
type: 'fa', type: 'fa',
name: iconStr.substring(3) // Remove 'fa:' prefix name: iconStr.substring(3) // Remove 'fa:' prefix
}; };
} else if (iconStr.startsWith('lucide:')) { } else if (lowerStr.startsWith('lucide:')) {
return { return {
type: 'lucide', type: 'lucide',
name: iconStr.substring(7) // Remove 'lucide:' prefix name: iconStr.substring(7) // Remove 'lucide:' prefix

View File

@@ -88,16 +88,16 @@ export class DeesStatsGrid extends DeesElement {
/* CSS Variables for consistent spacing and sizing */ /* CSS Variables for consistent spacing and sizing */
:host { :host {
--grid-gap: 16px; --grid-gap: 12px;
--tile-padding: 24px; --tile-padding: 16px;
--header-spacing: 16px; --header-spacing: 12px;
--content-min-height: 48px; --content-min-height: 40px;
--value-font-size: 30px; --value-font-size: 26px;
--unit-font-size: 16px; --unit-font-size: 14px;
--label-font-size: 13px; --label-font-size: 12px;
--title-font-size: 14px; --title-font-size: 13px;
--description-spacing: 12px; --description-spacing: 8px;
--border-radius: 8px; --border-radius: 6px;
--transition-duration: 0.15s; --transition-duration: 0.15s;
} }
@@ -136,7 +136,7 @@ export class DeesStatsGrid extends DeesElement {
/* Tile Base Styles */ /* Tile Base Styles */
.stats-tile { .stats-tile {
background: ${cssManager.bdTheme('#ffffff', '#09090b')}; 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); border-radius: var(--border-radius);
padding: var(--tile-padding); padding: var(--tile-padding);
transition: all var(--transition-duration) ease; transition: all var(--transition-duration) ease;
@@ -148,8 +148,8 @@ export class DeesStatsGrid extends DeesElement {
} }
.stats-tile:hover { .stats-tile:hover {
background: ${cssManager.bdTheme('hsl(210 40% 98%)', 'hsl(215 20.2% 10.2%)')}; background: ${cssManager.bdTheme('#fafafa', '#0d0d0d')};
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 85%)', 'hsl(215 20.2% 16.8%)')}; border-color: ${cssManager.bdTheme('#d0d0d0', '#2a2a2a')};
} }
.stats-tile.clickable { .stats-tile.clickable {
@@ -158,7 +158,7 @@ export class DeesStatsGrid extends DeesElement {
.stats-tile.clickable:hover { .stats-tile.clickable:hover {
transform: translateY(-1px); 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 */ /* Tile Header */
@@ -230,10 +230,10 @@ export class DeesStatsGrid extends DeesElement {
} }
.gauge-container { .gauge-container {
width: 140px; width: 120px;
height: 80px; height: 70px;
position: relative; position: relative;
margin-top: -10px; margin-top: -8px;
} }
.gauge-svg { .gauge-svg {
@@ -243,13 +243,13 @@ export class DeesStatsGrid extends DeesElement {
.gauge-background { .gauge-background {
fill: none; fill: none;
stroke: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(215 20.2% 21.8%)')}; stroke: ${cssManager.bdTheme('#e8e8e8', '#1a1a1a')};
stroke-width: 8; stroke-width: 6;
} }
.gauge-fill { .gauge-fill {
fill: none; fill: none;
stroke-width: 8; stroke-width: 6;
stroke-linecap: round; stroke-linecap: round;
transition: stroke-dashoffset 0.6s cubic-bezier(0.4, 0, 0.2, 1); transition: stroke-dashoffset 0.6s cubic-bezier(0.4, 0, 0.2, 1);
} }
@@ -287,17 +287,17 @@ export class DeesStatsGrid extends DeesElement {
.percentage-bar { .percentage-bar {
width: 100%; width: 100%;
height: 8px; height: 6px;
background: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(215 20.2% 21.8%)')}; background: ${cssManager.bdTheme('#e8e8e8', '#1a1a1a')};
border-radius: 4px; border-radius: 3px;
overflow: hidden; overflow: hidden;
} }
.percentage-fill { .percentage-fill {
height: 100%; 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); transition: width 0.6s cubic-bezier(0.4, 0, 0.2, 1);
border-radius: 4px; border-radius: 3px;
} }
/* Trend Styles */ /* Trend Styles */
@@ -339,7 +339,7 @@ export class DeesStatsGrid extends DeesElement {
.trend-graph { .trend-graph {
width: 100%; width: 100%;
height: 32px; height: 28px;
position: relative; position: relative;
} }
@@ -351,14 +351,14 @@ export class DeesStatsGrid extends DeesElement {
.trend-line { .trend-line {
fill: none; fill: none;
stroke: ${cssManager.bdTheme('hsl(215.4 16.3% 66.9%)', 'hsl(215 20.2% 55.1%)')}; stroke: ${cssManager.bdTheme('#999999', '#666666')};
stroke-width: 2; stroke-width: 1.5;
stroke-linejoin: round; stroke-linejoin: round;
stroke-linecap: round; stroke-linecap: round;
} }
.trend-area { .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 */ /* Text Value Styles */
@@ -480,13 +480,13 @@ export class DeesStatsGrid extends DeesElement {
const value = typeof tile.value === 'number' ? tile.value : parseFloat(tile.value); const value = typeof tile.value === 'number' ? tile.value : parseFloat(tile.value);
const options = tile.gaugeOptions || { min: 0, max: 100 }; const options = tile.gaugeOptions || { min: 0, max: 100 };
const percentage = ((value - options.min) / (options.max - options.min)) * 100; const percentage = ((value - options.min) / (options.max - options.min)) * 100;
// SVG dimensions and calculations // SVG dimensions and calculations
const width = 140; const width = 120;
const height = 80; const height = 70;
const strokeWidth = 8; const strokeWidth = 6;
const padding = strokeWidth / 2 + 2; const padding = strokeWidth / 2 + 2;
const radius = 48; const radius = 40;
const centerX = width / 2; const centerX = width / 2;
const centerY = height - padding; const centerY = height - padding;

View File

@@ -4,6 +4,104 @@ import type { IMenuItem } from './tab.js';
import type { IMenuGroup } from './menugroup.js'; import type { IMenuGroup } from './menugroup.js';
import type { ISecondaryMenuGroup, ISecondaryMenuItem } from './secondarymenu.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 // Forward declaration for circular reference
export type TDeesAppui = HTMLElement & { export type TDeesAppui = HTMLElement & {
setAppBarMenus: (menus: IAppBarMenuItem[]) => void; setAppBarMenus: (menus: IAppBarMenuItem[]) => void;
@@ -42,6 +140,10 @@ export type TDeesAppui = HTMLElement & {
getActivityLogVisible: () => boolean; getActivityLogVisible: () => boolean;
navigateToView: (viewId: string, params?: Record<string, string>) => Promise<boolean>; navigateToView: (viewId: string, params?: Record<string, string>) => Promise<boolean>;
getCurrentView: () => IViewDefinition | undefined; getCurrentView: () => IViewDefinition | undefined;
// Bottom bar
bottomBar: IBottomBarAPI;
setBottomBarVisible: (visible: boolean) => void;
getBottomBarVisible: () => boolean;
}; };
/** /**
@@ -233,6 +335,9 @@ export interface IAppConfig {
/** Activity log configuration */ /** Activity log configuration */
activityLog?: IActivityLogConfig; activityLog?: IActivityLogConfig;
/** Bottom bar configuration */
bottomBar?: IBottomBarConfig;
/** Event callbacks */ /** Event callbacks */
onViewChange?: (viewId: string, view: IViewDefinition) => void; onViewChange?: (viewId: string, view: IViewDefinition) => void;
onSearch?: (query: string) => void; onSearch?: (query: string) => void;