Compare commits
99 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ac1ef4e497 | |||
| 9c61c0542b | |||
| 5c099c8057 | |||
| 82b4afa95a | |||
| 888430d55a | |||
| 85424d07cd | |||
| 24d3afe85d | |||
| 9735af05c8 | |||
| 9471c419fa | |||
| 778f457ed5 | |||
| a91098527c | |||
| 8f8aedc6b0 | |||
| f67be189eb | |||
| 4b8b1fa446 | |||
| 0f9bc67a8e | |||
| b33d51cebf | |||
| 021e0fda3d | |||
| 2ed0d8e0f2 | |||
| 5e4514c913 | |||
| d1bc562b5c | |||
| 7adad49cb1 | |||
| d07fec834f | |||
| 6f54bd228c | |||
| ca7aa12218 | |||
| c2ee19308d | |||
| 5e27449e50 | |||
| d69f777b25 | |||
| caa954a539 | |||
| 997520f3ba | |||
| 92f69e2aa6 | |||
| 70c29c778c | |||
| 0fc302699e | |||
| dcb7ca2df3 | |||
| ccbb0415e4 | |||
| 496f54cedd | |||
| 83b5ecebeb | |||
| 53b5cbed07 | |||
| 352fe79791 | |||
| a95d5a96a0 | |||
| ece7bb9a94 | |||
| d42859b7b2 | |||
| f5655ad20b | |||
| d3463f009b | |||
| bb883ce341 | |||
| d9703d3ce3 | |||
| 7b5ba74d8b | |||
| a61f57db13 | |||
| c33ad2e405 | |||
| 4190324cb4 | |||
| 1b108fcc8c | |||
| 0b2675c7e5 | |||
| 12b0aa0aad | |||
| 987ae70e7a | |||
| 3ba673282a | |||
| 20a52d1b3e | |||
| dafcf3834c | |||
| 639672358a | |||
| 671fb7dc66 | |||
| b92966ef28 | |||
| c1102634f3 | |||
| ee470775b2 | |||
| ba0f1602a1 | |||
| 682955212e | |||
| 0410f6c196 | |||
| 24aa7588c5 | |||
| b46fe8fe93 | |||
| b47c2053b5 | |||
| 16bf8001ae | |||
| 792e77f824 | |||
| 9b39196195 | |||
| ad59e3d334 | |||
| 0de4283fae | |||
| 6f9c92a866 | |||
| 0ec2f2aebb | |||
| cd22106597 | |||
| a212536cfa | |||
| 18297d54c4 | |||
| f790ca38d0 | |||
| ce2b42ecd5 | |||
| 09e299bc2e | |||
| bbc7dfe29a | |||
| 49b9e833e8 | |||
| f739bb608e | |||
| 286a6f9088 | |||
| e32b9589a5 | |||
| 6427510c98 | |||
| cf92a423cf | |||
| 3f3677ebaa | |||
| edc15a727c | |||
| 960085145d | |||
| 7fdb4f19a8 | |||
| e21fb79731 | |||
| 05f669a7bd | |||
| 8137d79e18 | |||
| 3b474b7dcc | |||
| e449b413d1 | |||
| 8918dc94bd | |||
| 2c595bf803 | |||
| 75f31a6cec |
174
CLAUDE.md
174
CLAUDE.md
@@ -1,174 +0,0 @@
|
|||||||
# CLAUDE.md
|
|
||||||
|
|
||||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
|
||||||
|
|
||||||
## Project Overview
|
|
||||||
|
|
||||||
@design.estate/dees-catalog is a comprehensive web components library built with TypeScript and LitElement. It provides a large collection of UI components for building modern web applications with consistent design and behavior.
|
|
||||||
|
|
||||||
## Build and Development Commands
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Install dependencies
|
|
||||||
pnpm install
|
|
||||||
|
|
||||||
# Build the project
|
|
||||||
pnpm run build
|
|
||||||
# This runs: tsbuild tsfolders --allowimplicitany && tsbundle element --production --bundler esbuild
|
|
||||||
|
|
||||||
# Run development watch mode
|
|
||||||
pnpm run watch
|
|
||||||
# This runs: tswatch element
|
|
||||||
|
|
||||||
# Run tests (browser tests)
|
|
||||||
pnpm test
|
|
||||||
# This runs: tstest test/ --web --verbose --timeout 30 --logfile
|
|
||||||
|
|
||||||
# Run a specific test file
|
|
||||||
tsx test/test.wysiwyg-basic.browser.ts --verbose
|
|
||||||
|
|
||||||
# Build documentation
|
|
||||||
pnpm run buildDocs
|
|
||||||
```
|
|
||||||
|
|
||||||
### Testing Notes
|
|
||||||
- Test files follow the pattern: `test.*.browser.ts`, `test.*.node.ts`, or `test.*.both.ts`
|
|
||||||
- Browser tests run in a headless browser environment
|
|
||||||
- Use `--logfile` option to store logs in `.nogit/testlogs/`
|
|
||||||
- For debugging, create files in `.nogit/debug/` and run with `tsx`
|
|
||||||
|
|
||||||
## Architecture Overview
|
|
||||||
|
|
||||||
### Component Structure
|
|
||||||
The library is organized into several categories:
|
|
||||||
|
|
||||||
1. **Core UI Components** (`dees-button`, `dees-badge`, `dees-icon`, etc.)
|
|
||||||
- Basic building blocks with consistent theming
|
|
||||||
- All support light/dark themes via `cssManager.bdTheme()`
|
|
||||||
|
|
||||||
2. **Form Components** (`dees-form`, `dees-input-*`)
|
|
||||||
- Complete form system with validation
|
|
||||||
- Base class `DeesInputBase` provides common functionality
|
|
||||||
- Form data collection via `DeesForm` container
|
|
||||||
|
|
||||||
3. **Layout Components** (`dees-appui-*`)
|
|
||||||
- Application shell components
|
|
||||||
- `DeesAppuiBase` orchestrates the entire layout
|
|
||||||
- Grid-based responsive design
|
|
||||||
|
|
||||||
4. **Data Display** (`dees-table`, `dees-dataview-*`, `dees-statsgrid`)
|
|
||||||
- Complex data visualization components
|
|
||||||
- Interactive tables with sorting/filtering
|
|
||||||
- Chart components using ApexCharts
|
|
||||||
|
|
||||||
5. **Overlays** (`dees-modal`, `dees-contextmenu`, `dees-toast`)
|
|
||||||
- Managed by central z-index registry
|
|
||||||
- Window layer system for proper stacking
|
|
||||||
|
|
||||||
### Key Architectural Patterns
|
|
||||||
|
|
||||||
#### Z-Index Management
|
|
||||||
All overlay components use a centralized z-index registry system:
|
|
||||||
- Definition in `ts_web/elements/00zindex.ts`
|
|
||||||
- Dynamic z-index assignment via `ZIndexRegistry` class
|
|
||||||
- Components get z-index from registry when showing
|
|
||||||
- Ensures proper stacking order (dropdowns above modals, etc.)
|
|
||||||
|
|
||||||
#### Theme System
|
|
||||||
- All components support light/dark themes
|
|
||||||
- Use `cssManager.bdTheme(lightValue, darkValue)` for theme-aware colors
|
|
||||||
- Consistent color palette defined in `00colors.ts`
|
|
||||||
|
|
||||||
#### Component Demo System
|
|
||||||
- Each component has a static `demo` property
|
|
||||||
- Demo functions in separate `.demo.ts` files
|
|
||||||
- Showcase pages aggregate demos (e.g., `input-showcase.ts`)
|
|
||||||
|
|
||||||
#### WYSIWYG Editor Architecture
|
|
||||||
The WYSIWYG editor uses a sophisticated architecture with separated concerns:
|
|
||||||
- **Main Component**: `dees-input-wysiwyg.ts` - Orchestrates the editor
|
|
||||||
- **Handler Classes**:
|
|
||||||
- `WysiwygInputHandler` - Handles text input and block transformations
|
|
||||||
- `WysiwygKeyboardHandler` - Manages keyboard shortcuts and navigation
|
|
||||||
- `WysiwygDragDropHandler` - Manages block reordering
|
|
||||||
- `WysiwygModalManager` - Shows configuration modals
|
|
||||||
- `WysiwygBlockOperations` - Core block manipulation logic
|
|
||||||
- **Global Menus**:
|
|
||||||
- `DeesSlashMenu` and `DeesFormattingMenu` render globally to avoid focus issues
|
|
||||||
- Singleton pattern ensures single instance
|
|
||||||
- **Programmatic Rendering**: Uses manual DOM manipulation to prevent focus loss
|
|
||||||
|
|
||||||
### Component Communication
|
|
||||||
- Custom events for parent-child communication
|
|
||||||
- Form components emit standardized events (`change`, `blur`, etc.)
|
|
||||||
- Complex components like `DeesAppuiBase` re-emit child events
|
|
||||||
|
|
||||||
### Build System
|
|
||||||
- TypeScript compilation with decorators support
|
|
||||||
- Web component bundling with esbuild
|
|
||||||
- Element exports in `ts_web/elements/index.ts`
|
|
||||||
- Distribution builds in `dist_ts_web/`
|
|
||||||
|
|
||||||
## Important Implementation Details
|
|
||||||
|
|
||||||
### When Creating New Components
|
|
||||||
1. Extend `DeesElement` from `@design.estate/dees-element`
|
|
||||||
2. Use `@customElement('dees-componentname')` decorator
|
|
||||||
3. Implement theme support with `cssManager.bdTheme()`
|
|
||||||
4. Create a demo function in a separate `.demo.ts` file
|
|
||||||
5. Export from `elements/index.ts`
|
|
||||||
|
|
||||||
### Form Input Components
|
|
||||||
1. Extend `DeesInputBase` for form inputs
|
|
||||||
2. Implement `getValue()` and `setValue()` methods
|
|
||||||
3. Use `changeSubject.next(this)` to emit changes
|
|
||||||
4. Support `disabled` and `required` properties
|
|
||||||
|
|
||||||
### Overlay Components
|
|
||||||
1. Import z-index from `00zindex.ts`
|
|
||||||
2. Get z-index from registry when showing: `zIndexRegistry.getNextZIndex()`
|
|
||||||
3. Register/unregister with the registry
|
|
||||||
4. Use `DeesWindowLayer` for backdrop if needed
|
|
||||||
|
|
||||||
### Testing Components
|
|
||||||
1. Create test files in `test/` directory
|
|
||||||
2. Use `@git.zone/tstest` with tap-bundle
|
|
||||||
3. Test in browser environment for web components
|
|
||||||
4. Use proper async/await for component lifecycle
|
|
||||||
|
|
||||||
## Common Patterns and Pitfalls
|
|
||||||
|
|
||||||
### Focus Management
|
|
||||||
- WYSIWYG editor uses programmatic rendering to prevent focus loss
|
|
||||||
- Use `requestAnimationFrame` for timing-sensitive focus operations
|
|
||||||
- Avoid reactive re-renders during user input
|
|
||||||
|
|
||||||
### Event Handling
|
|
||||||
- Prevent event bubbling in nested interactive components
|
|
||||||
- Use `pointer-events: none/auto` for click-through behavior
|
|
||||||
- Handle both mouse and keyboard events for accessibility
|
|
||||||
|
|
||||||
### Performance Considerations
|
|
||||||
- Large components (editor, terminal) use lazy loading
|
|
||||||
- Charts use debounced resize observers
|
|
||||||
- Tables implement virtual scrolling for large datasets
|
|
||||||
|
|
||||||
## File Organization
|
|
||||||
```
|
|
||||||
ts_web/
|
|
||||||
├── elements/ # All component files
|
|
||||||
│ ├── 00*.ts # Shared utilities (colors, z-index, plugins)
|
|
||||||
│ ├── dees-*.ts # Component implementations
|
|
||||||
│ ├── dees-*.demo.ts # Component demos
|
|
||||||
│ ├── interfaces/ # Shared TypeScript interfaces
|
|
||||||
│ ├── helperclasses/ # Utility classes (FormController)
|
|
||||||
│ └── wysiwyg/ # WYSIWYG editor subsystem
|
|
||||||
├── pages/ # Demo showcase pages
|
|
||||||
└── index.ts # Main export file
|
|
||||||
```
|
|
||||||
|
|
||||||
## Recent Major Changes
|
|
||||||
- Z-Index Registry System (2025-12-24): Dynamic stacking order management
|
|
||||||
- WYSIWYG Refactoring (2025-06-24): Complete architecture overhaul with separated concerns
|
|
||||||
- Form System Enhancement: Unified validation and data collection
|
|
||||||
- Theme System: Consistent light/dark theme support across all components
|
|
||||||
232
changelog.md
232
changelog.md
@@ -1,5 +1,235 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## 2025-12-08 - 3.1.2 - fix(DeesAppuiMainmenu, DeesAppuiSecondarymenu)
|
||||||
|
Add position: relative to main and secondary app UI menus to fix positioning of overlays and tooltips
|
||||||
|
|
||||||
|
- ts_web/elements/00group-appui/dees-appui-mainmenu/dees-appui-mainmenu.ts: add `position: relative` to host styles
|
||||||
|
- ts_web/elements/00group-appui/dees-appui-secondarymenu/dees-appui-secondarymenu.ts: add `position: relative` to host styles
|
||||||
|
- Fixes incorrect positioning for absolutely positioned children (tooltips, overlays, badges) inside the main and secondary menus
|
||||||
|
|
||||||
|
## 2025-12-08 - 3.1.1 - fix(dees-appui)
|
||||||
|
Extract demos for main and secondary app menus, adjust collapsed styles and toggle placement, bump devDependency
|
||||||
|
|
||||||
|
- Extracted inline demo markup into separate demo files: ts_web/elements/00group-appui/dees-appui-mainmenu/dees-appui-mainmenu.demo.ts and ts_web/elements/00group-appui/dees-appui-secondarymenu/dees-appui-secondarymenu.demo.ts and wired them up via imported demoFunc to reduce component size.
|
||||||
|
- Moved collapse toggle button markup in both dees-appui-mainmenu and dees-appui-secondarymenu templates to after the main container to improve layout/stacking and focus behavior.
|
||||||
|
- Adjusted collapsed logo/heading styles: removed extra padding/gap and hide logo text using display:none for a cleaner collapsed state.
|
||||||
|
- Bumped devDependency @git.zone/tswatch from ^2.3.1 to ^2.3.2 in package.json.
|
||||||
|
|
||||||
|
## 2025-12-08 - 3.1.0 - feat(dees-appui)
|
||||||
|
Add collapsible/compact mode to AppUI sidebars (mainmenu & secondarymenu) with toggles, tooltips and improved z-index stacking
|
||||||
|
|
||||||
|
- Add collapsed property to dees-appui-mainmenu and dees-appui-secondarymenu (reflect: true) to enable compact horizontal mode.
|
||||||
|
- Add floating collapse toggle buttons and public toggleCollapse() methods on mainmenu and secondarymenu; these dispatch 'collapse-change' events (bubbles & composed).
|
||||||
|
- Expose and track collapse state in dees-appui-base via mainmenuCollapsed and secondarymenuCollapsed properties; bind states to child components and re-emit collapse-change events as mainmenu-collapse-change and secondarymenu-collapse-change.
|
||||||
|
- Implement collapsed styles and animations: reduced sidebar widths, hide/compact labels and headers, center icons, hide badges, and add smooth width/opacity transitions.
|
||||||
|
- Add tooltips that appear for tabs/items when sidebars are collapsed to preserve discoverability.
|
||||||
|
- Adjust layout grid in DeesAppuiBase (use auto columns) and add explicit z-index layering to ensure proper stacking order of mainmenu, secondarymenu, maincontent and activitylog.
|
||||||
|
|
||||||
|
## 2025-12-08 - 3.0.1 - fix(dees-appui)
|
||||||
|
Normalize header heights and box-sizing for App UI components
|
||||||
|
|
||||||
|
- Set topbar/header heights to 48px (was 40px) and adjusted dependent offsets (activity container top, topShadow position) in dees-appui-activitylog.
|
||||||
|
- Make logo and secondary menu headers fixed 48px tall and replace vertical padding with horizontal padding for consistent vertical alignment (dees-appui-mainmenu, dees-appui-secondarymenu).
|
||||||
|
- Ensure tabs wrapper uses explicit 48px height and tabsContainer fills height (height:100%) to keep tab items vertically centered (dees-appui-tabs).
|
||||||
|
- Add box-sizing: border-box to affected header/logo containers to prevent overflow and ensure correct sizing.
|
||||||
|
- Minor CSS alignment and overflow fixes to improve consistent layout and scrolling behavior across the app UI components.
|
||||||
|
|
||||||
|
## 2025-12-08 - 3.0.0 - BREAKING CHANGE(dees-appui-secondarymenu)
|
||||||
|
Add SecondaryMenu component and replace Mainselector with new SecondaryMenu in AppUI
|
||||||
|
|
||||||
|
- Add dees-appui-secondarymenu component: collapsible groups, badges, dynamic heading, context menu and legacy flat-options support
|
||||||
|
- Introduce interfaces ISecondaryMenuItem and ISecondaryMenuGroup under elements/interfaces
|
||||||
|
- Replace dees-appui-mainselector usage with dees-appui-secondarymenu in DeesAppuiBase (props/events updated: secondarymenuGroups, secondarymenuHeading, secondarymenuOptions, item-select / secondarymenu-item-select)
|
||||||
|
- Remove dees-appui-mainselector implementation and its index export; update group exports and imports to expose secondarymenu
|
||||||
|
- Update demos and pages to showcase the new SecondaryMenu and adjust import paths for grouped components
|
||||||
|
- Bump devDependency @git.zone/tswatch to ^2.3.1
|
||||||
|
|
||||||
|
## 2025-12-08 - 2.0.7 - fix(structure)
|
||||||
|
Add many new UI components, input controls, charts, editors, and demos
|
||||||
|
|
||||||
|
- Introduce App UI components: dees-appui-appbar, dees-appui-mainmenu, dees-appui-mainselector, dees-appui-maincontent, dees-appui-activitylog, dees-appui-profiledropdown, dees-appui-tabs, dees-appui-base, dees-appui-view (templates, styles and demos included).
|
||||||
|
- Add a comprehensive set of input components: dees-input-text, dees-input-checkbox, dees-input-dropdown, dees-input-fileupload, dees-input-datepicker, dees-input-phone, dees-input-iban, dees-input-quantityselector, dees-input-list, dees-input-typelist, dees-input-tags, dees-input-multitoggle, dees-input-radiogroup, dees-input-richtext and supporting demos/styles/templates.
|
||||||
|
- Add form primitives and integration: dees-form and dees-form-submit with validation, collection and demo pages showcasing usage.
|
||||||
|
- Add button family and utilities: dees-button (with updated variants, sizes, status handling and demo), dees-button-group and dees-button-exit.
|
||||||
|
- Add charting components: dees-chart-area (ApexCharts integration) and dees-chart-log (log viewer) plus rich demo scenarios and realtime features.
|
||||||
|
- Add data display components: dees-dataview-codebox (highlight.js integration) and dees-dataview-statusobject with copy/context behaviours and demos.
|
||||||
|
- Add editor tooling: dees-editor (Monaco loader/version management), dees-editor-markdown and dees-editor-markdownoutlet; also TipTap-based richtext input with toolbar and link handling.
|
||||||
|
- Add global utilities and infra: dees-toast (programmatic toast API and containers), z-index registry and theme/font helpers (fonts, color tokens), plus many styles and accessibility/keyboard improvements across components.
|
||||||
|
- Export and index updates: new group exports added to ts_web/elements index and many index.ts files to expose the new components and demos.
|
||||||
|
- Extensive demos and showcase pages added (input-showcase, component demos) to illustrate integration, keyboard navigation, theming and form flows.
|
||||||
|
|
||||||
|
## 2025-12-06 - 2.0.6 - fix(dees-input-richtext)
|
||||||
|
Initialize editor and link input element references in firstUpdated to ensure they exist before editor initialization.
|
||||||
|
|
||||||
|
- Assign editorElement from shadowRoot.querySelector('.editor-content') in firstUpdated.
|
||||||
|
- Assign linkInputElement from shadowRoot.querySelector('.link-input input') in firstUpdated.
|
||||||
|
- Call initializeEditor() after DOM references are set to avoid undefined-element runtime errors.
|
||||||
|
|
||||||
|
## 2025-12-06 - 2.0.5 - fix(build)
|
||||||
|
Bump devDependencies: update @git.zone/tsbundle and @git.zone/tswatch to patched versions
|
||||||
|
|
||||||
|
- Update @git.zone/tsbundle from ^2.6.2 to ^2.6.3
|
||||||
|
- Update @git.zone/tswatch from ^2.2.2 to ^2.2.3
|
||||||
|
|
||||||
|
## 2025-12-06 - 2.0.4 - fix(imports)
|
||||||
|
Normalize and fix relative import paths for web components to ensure correct module resolution
|
||||||
|
|
||||||
|
- Replaced numerous './<component>.js' imports with explicit '../<component>/<component>.js' paths across many elements and demos to fix module resolution.
|
||||||
|
- Updated imports for core shared components such as dees-icon, dees-panel, dees-contextmenu, dees-windowlayer, dees-windowcontrols and several app-ui components (appbar, maincontent, mainselector, activitylog, mobilenavigation, modal, pdf, profilepicture, statsgrid, etc.).
|
||||||
|
- No runtime behavior changes — this is a refactor to import paths to address build/bundling and resolution issues.
|
||||||
|
|
||||||
|
## 2025-12-03 - 2.0.3 - fix(dependencies)
|
||||||
|
Bump dependencies and developer tooling versions
|
||||||
|
|
||||||
|
- Upgrade lucide from ^0.553.0 to ^0.555.0
|
||||||
|
- Bump @git.zone/tsbuild from ^3.1.0 to ^3.1.2
|
||||||
|
- Bump @git.zone/tsbundle from ^2.5.2 to ^2.6.2
|
||||||
|
- Bump @git.zone/tstest from ^2.8.1 to ^3.1.3
|
||||||
|
- Bump @git.zone/tswatch from ^2.2.1 to ^2.2.2
|
||||||
|
- Upgrade @types/node from ^22.0.0 to ^24.10.1
|
||||||
|
- Patch release: increment package version to 2.0.3
|
||||||
|
|
||||||
|
## 2025-11-30 - 2.0.2 - fix(dees-stepper)
|
||||||
|
Make step validation abortable and cancel active step listeners when navigating
|
||||||
|
|
||||||
|
- Extend IStep.validationFunc signature to accept an optional AbortSignal so validation handlers can be cancelled.
|
||||||
|
- Store an AbortController on the selected step and pass its signal into validationFunc when invoked.
|
||||||
|
- Abort the step's AbortController when navigating to the previous or next step to cancel any active listeners or async operations.
|
||||||
|
|
||||||
|
## 2025-11-30 - 2.0.1 - fix(dees-stepper)
|
||||||
|
Improve dees-stepper visual style and transitions
|
||||||
|
|
||||||
|
- Smooth animation: extend .step transition duration and use a cubic-bezier curve for smoother motion.
|
||||||
|
- Add .step.entrance class with a shorter easing for entrance animations to keep entrance timing distinct.
|
||||||
|
- Visual tweaks: reduce border-radius from 18px to 12px and increase inner content padding to 32px.
|
||||||
|
- Color and border updates: adjust background and border colors for light/dark themes to more consistent values.
|
||||||
|
- Shadow simplification: replace theme-dependent heavy shadows with a single subtle shadow (0 8px 32px rgba(0,0,0,0.4)).
|
||||||
|
- Remove selected-state border/box-shadow overrides (selection visuals simplified).
|
||||||
|
- Remove background-clip: padding-box to simplify rendering.
|
||||||
|
|
||||||
|
## 2025-11-17 - 2.0.0 - BREAKING CHANGE(decorators)
|
||||||
|
Migrate to TC39 standard decorators (accessor) across components, update tsconfig and bump dependencies
|
||||||
|
|
||||||
|
- Replaced experimental decorator-backed class fields with the TC39-compatible "accessor" form across ~69 web component files (properties and state fields) to follow Lit 3.x recommendations.
|
||||||
|
- Updated tsconfig.json to remove experimentalDecorators and useDefineForClassFields, aligning compiler settings with the standard decorators migration.
|
||||||
|
- Fixed optional/nullable fields to explicit `Type | undefined = undefined` where necessary to preserve runtime behavior and typing.
|
||||||
|
- Adjusted/remove usages of some non-reactive decorators/@query patterns to be compatible with the new decorator model (notable changes in a few components).
|
||||||
|
- Bumped several dependencies and devDependencies (examples: @design.estate/dees-domtools, @design.estate/dees-element, @design.estate/dees-wcctools, @git.zone/tsbuild, @git.zone/tstest, apexcharts, lucide).
|
||||||
|
- Added migration notes and testing summary to readme.hints.md documenting the TC39 decorators migration and verification steps.
|
||||||
|
|
||||||
|
## 2025-10-23 - 1.12.6 - fix(dependencies)
|
||||||
|
Bump FontAwesome to ^7.1.0 and add local claude settings
|
||||||
|
|
||||||
|
- Updated @fortawesome packages (@fortawesome/fontawesome-svg-core, @fortawesome/free-brands-svg-icons, @fortawesome/free-regular-svg-icons, @fortawesome/free-solid-svg-icons) to ^7.1.0 in package.json
|
||||||
|
- Added .claude/settings.local.json to configure local Claude/tooling permissions for repository operations
|
||||||
|
|
||||||
|
## 2025-09-23 - 1.12.5 - fix(ci)
|
||||||
|
Add local permissions settings for development
|
||||||
|
|
||||||
|
- Adds a new local settings file: .claude/settings.local.json
|
||||||
|
- Provides explicit permission entries for development tasks (allow running pnpm scripts, reading files, searching/replacing patterns, activating project, and helper tooling)
|
||||||
|
- Intended for local dev environment to enable tool automation without changing repository code
|
||||||
|
|
||||||
|
## 2025-09-20 - 1.12.4 - fix(ci)
|
||||||
|
Add local assistant settings to enable permitted dev tooling commands
|
||||||
|
|
||||||
|
- Add a local assistant settings file to configure allowed development tooling commands.
|
||||||
|
- Allows running pnpm scripts, file read/search/replace operations and other local project helper actions.
|
||||||
|
- Local configuration only — does not change library code or public API.
|
||||||
|
|
||||||
|
## 2025-09-19 - 1.12.3 - fix(dees-input-fileupload)
|
||||||
|
Show selected files inside dropzone and improve file upload UX
|
||||||
|
|
||||||
|
- Render the selected file list inside the dropzone container so files are displayed inline with the drop area
|
||||||
|
- Add dropzone--has-files class and styles to visually indicate when files are present
|
||||||
|
- Avoid opening the file selector when clicking on the browse button or inside the file list (prevents accidental re-opening)
|
||||||
|
- Refine file list and file-row styles (sizes, paddings, border radius, hover/background behavior and thumbnail/icon sizes) for a more compact and consistent appearance
|
||||||
|
- Simplify empty-state handling by returning an empty template when no files are present (file list is only rendered when files exist)
|
||||||
|
|
||||||
|
## 2025-09-18 - 1.12.2 - fix(dees-input-wysiwyg)
|
||||||
|
Integrate output format preview into WYSIWYG demo; update plan and add local dev settings
|
||||||
|
|
||||||
|
- Wire output format preview into the WYSIWYG demo (ts_web/elements/dees-input-wysiwyg.demo.ts) by calling setupOutputFormatDemo(editors.meeting, editors.recipe) so HTML/Markdown preview controls are initialized.
|
||||||
|
- Update readme.plan.md: mark the Output Formats review tasks as completed and document that preview controls were added.
|
||||||
|
- Add a local settings file to allow running local tooling tasks (grants permission for pnpm run scripts and related local commands).
|
||||||
|
- No library API or runtime component behavior changed — this is a demo/documentation and local-settings update.
|
||||||
|
|
||||||
|
## 2025-09-18 - 1.12.1 - fix(ci)
|
||||||
|
Add local settings to allow running pnpm scripts and enable dev chat permission
|
||||||
|
|
||||||
|
- Add a repository-local settings file granting permission to run pnpm scripts (Bash(pnpm run:*)) for development tooling.
|
||||||
|
- Enable the mcp__zen__chat permission for local dev workflows.
|
||||||
|
|
||||||
|
## 2025-09-18 - 1.12.0 - feat(dees-stepper)
|
||||||
|
Revamp dees-stepper: modern styling, new steps and improved navigation/validation
|
||||||
|
|
||||||
|
- Visual refresh for dees-stepper: updated card shapes, shadows, refined borders and stronger selected-state visuals for a modern shadcn-inspired look
|
||||||
|
- Improved transitions and animations (transform, box-shadow, filter) for smoother step selection and show/hide behavior
|
||||||
|
- Expanded default/demo steps: replaced small sample with a richer multi-step flow (Account Setup, Profile Details, Contact Information, Team Size, Goals, Brand Preferences, Integrations, Review & Launch)
|
||||||
|
- Enhanced step interactions: safer goNext/goBack handling with boundary checks and reset of validation flags to avoid stale validation state
|
||||||
|
- Better toolbar/controls placement for stepper demo (spacing, counters, accessible back control) and improved keyboard/UX affordances
|
||||||
|
- Minor documentation and meta updates: readme.plan.md extended with dees-stepper plan items and added .claude/settings.local.json
|
||||||
|
|
||||||
|
## 2025-09-18 - 1.11.8 - fix(ci)
|
||||||
|
Add local tool permissions config to allow running pnpm scripts and enable mcp__zen__chat
|
||||||
|
|
||||||
|
- Add local settings file to grant permission to run pnpm scripts (Bash(pnpm run:*))
|
||||||
|
- Enable mcp__zen__chat permission in local tool settings
|
||||||
|
|
||||||
|
## 2025-09-16 - 1.11.7 - fix(readme)
|
||||||
|
Expand README with comprehensive component documentation, examples and developer guide; add local Claude settings
|
||||||
|
|
||||||
|
- Expanded README substantially: installation, component overview, detailed component docs, usage examples, demos and developer guidance
|
||||||
|
- Updated many example snippets and API usage examples (icons, inputs, editor, forms, overlays, charts, etc.) to be more explicit and consistent
|
||||||
|
- Added .claude/settings.local.json to configure local Claude permissions for repository tooling
|
||||||
|
- No runtime or library code changes — documentation and demo content only
|
||||||
|
|
||||||
|
## 2025-09-16 - 1.11.6 - fix(dees-table)
|
||||||
|
Improve Lucene range comparisons, pin monaco-editor to 0.52.2, and add local dev metadata
|
||||||
|
|
||||||
|
- Fix lucene inRange behavior to correctly compare homogeneous types (strings, numbers, dates) and fall back to string comparison when needed (ts_web/elements/dees-table/lucene.ts).
|
||||||
|
- Pin monaco-editor to 0.52.2 in package.json to avoid a breaking upgrade regression observed with ^0.53.0.
|
||||||
|
- Add local development/tooling metadata and conveniences: .claude/settings.local.json (tool permissions) and .serena/ memory files (done_checklist, project_overview, style_and_conventions, suggested_commands).
|
||||||
|
- Minor housekeeping: update project dev docs / memories to capture build/test/checklist guidance.
|
||||||
|
|
||||||
|
## 2025-09-16 - 1.11.5 - fix(ci)
|
||||||
|
Add local Claude agent settings for CI tooling
|
||||||
|
|
||||||
|
- Add .claude/settings.local.json to configure local Claude agent permissions
|
||||||
|
- Allow Bash commands matching pnpm run:* and the mcp__zen__chat permission for development tooling
|
||||||
|
|
||||||
|
## 2025-09-10 - 1.11.4 - fix(readme)
|
||||||
|
Rewrite and expand README with Quick Start, feature highlights, demos and usage examples; add local Claude settings file
|
||||||
|
|
||||||
|
- Completely rewritten and reorganized README: added Quick Start, component highlights, usage examples, demos, development workflow, troubleshooting and links.
|
||||||
|
- Added .claude/settings.local.json with local Claude permission configuration.
|
||||||
|
|
||||||
|
## 2025-09-08 - 1.11.3 - fix(dees-input-list)
|
||||||
|
Prevent list animations from affecting scroll bounds and fix content-visibility issues in dees-input-list; add local developer settings
|
||||||
|
|
||||||
|
- dees-input-list: add overflow:hidden to list items to prevent animations from altering scroll bounds and causing visual/scroll glitches
|
||||||
|
- dees-input-list: force content-visibility/contain to visible/none to avoid unexpected scrolling/layout issues when items animate
|
||||||
|
- Add .claude/settings.local.json with local developer permissions (allows running pnpm scripts via Claude-local tooling)
|
||||||
|
|
||||||
|
## 2025-09-07 - 1.11.2 - fix(DeesFormSubmit)
|
||||||
|
Make form submit robust by locating nearest dees-form via closest(); add local CLAUDE settings
|
||||||
|
|
||||||
|
- Fix: DeesFormSubmit.submit now walks up the DOM with closest('dees-form') to find and call gatherAndDispatch on the parent form. This fixes cases where the submit button is slotted or not a direct child of the form.
|
||||||
|
- Chore: Add .claude/settings.local.json to permit running pnpm scripts in the local CLAUDE environment (allows Bash(pnpm run:*)).
|
||||||
|
|
||||||
|
## 2025-09-06 - 1.11.1 - fix(dees-input-text)
|
||||||
|
Normalize Lucide icon names for password toggle
|
||||||
|
|
||||||
|
- Updated password visibility toggle icons in dees-input-text from 'lucide:eye'/'lucide:eye-off' to 'lucide:Eye'/'lucide:EyeOff' to match Lucide exports and avoid missing icon rendering.
|
||||||
|
|
||||||
|
## 2025-09-05 - 1.11.0 - feat(dees-icon)
|
||||||
|
Add full icon list and improve dees-icon demo with copy-all functionality and UI tweaks
|
||||||
|
|
||||||
|
- Added readme.icons.md containing 1900+ icon identifiers (FontAwesome + Lucide) for easy reference and tooling
|
||||||
|
- Enhanced ts_web/elements/dees-icon.demo.ts: added a 'Copy All Icon Names' button that copies prefixed icon names (fa:..., lucide:...) to the clipboard and shows temporary feedback
|
||||||
|
- Updated demo presentation: prefixed displayed icon names (fa: / lucide:), improved search-container spacing and added button styling for better UX
|
||||||
|
- Changes are documentation/demo only — no production runtime component logic changed
|
||||||
|
|
||||||
## 2025-09-05 - 1.10.12 - fix(dees-simple-appdash)
|
## 2025-09-05 - 1.10.12 - fix(dees-simple-appdash)
|
||||||
Fix icon rendering in dees-simple-appdash to respect provided icon strings
|
Fix icon rendering in dees-simple-appdash to respect provided icon strings
|
||||||
|
|
||||||
@@ -161,7 +391,7 @@ Add dees-searchbar component with live search and filter demo
|
|||||||
## 2025-04-22 - 1.6.0 - feat(documentation/dees-heading)
|
## 2025-04-22 - 1.6.0 - feat(documentation/dees-heading)
|
||||||
Add codex documentation overview and dees-heading component demo
|
Add codex documentation overview and dees-heading component demo
|
||||||
|
|
||||||
- Introduce 'codex.md' to provide a high-level overview of project layout, component patterns, and build workflow
|
- Introduce contributor overview doc (`codex.md`, now consolidated into `readme.info.md`) to provide a high-level overview of project layout, component patterns, and build workflow
|
||||||
- Add and update dees-heading component with demo to support multiple heading levels and horizontal rule styles
|
- Add and update dees-heading component with demo to support multiple heading levels and horizontal rule styles
|
||||||
- Update component export index to include dees-heading
|
- Update component export index to include dees-heading
|
||||||
|
|
||||||
|
|||||||
43
codex.md
43
codex.md
@@ -1,43 +0,0 @@
|
|||||||
# Codex: Project Overview and Codebase Structure
|
|
||||||
|
|
||||||
## Project Overview
|
|
||||||
- Package: `@design.estate/dees-catalog`
|
|
||||||
- Focus: Web Components library providing UI elements and layouts for modern web apps.
|
|
||||||
|
|
||||||
## Directory Layout
|
|
||||||
- ts_web/: TypeScript source files
|
|
||||||
- elements/: Individual Web Component definitions
|
|
||||||
- pages/: Page-level templates for composite layouts
|
|
||||||
- html/: Demo/app entry point loading the bundled scripts
|
|
||||||
- dist_bundle/: Bundled browser JS and source maps
|
|
||||||
- dist_ts_web/: ES module outputs for TypeScript/web consumers
|
|
||||||
- dist_watch/: Watch-mode development bundle with live reload
|
|
||||||
- test/: Browser-based tests using `@push.rocks/tapbundle`
|
|
||||||
|
|
||||||
## Component Patterns
|
|
||||||
- Each component in ts_web/elements/:
|
|
||||||
- Decorated with `@customElement('tag-name')`
|
|
||||||
- Extends `DeesElement` from `@design.estate/dees-element`
|
|
||||||
- Uses `@property` for reactive, reflected attributes
|
|
||||||
- Defines `static styles = [cssManager.defaultStyles, css`...`]`
|
|
||||||
- Implements `render()` returning a Lit `html` template with slots or markup
|
|
||||||
- Exposes a demo via `public static demo` linking to `.demo.ts` files
|
|
||||||
|
|
||||||
## Build & Development Workflow
|
|
||||||
- Install dependencies: `npm install` or `pnpm install`
|
|
||||||
- Build production bundle: `npm run build`
|
|
||||||
- Start dev watch mode: `npm run watch`
|
|
||||||
- Run tests: `npm test` (launches browser fixtures)
|
|
||||||
|
|
||||||
## Theming & Utilities
|
|
||||||
- Default global styles via `cssManager.defaultStyles`
|
|
||||||
- Theme-aware values with `cssManager.bdTheme(light, dark)`
|
|
||||||
- DOM utilities set up in `html/index.ts` using `@design.estate/dees-domtools`
|
|
||||||
|
|
||||||
## Documentation
|
|
||||||
- `readme.md` provides an overview of all components and basic usage
|
|
||||||
- Live examples in `.demo.ts` files
|
|
||||||
accessible via component `demo` static property
|
|
||||||
|
|
||||||
## Updates to this file
|
|
||||||
If you have pattern insisights or general changes to the codebase, please update this file.
|
|
||||||
40
package.json
40
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@design.estate/dees-catalog",
|
"name": "@design.estate/dees-catalog",
|
||||||
"version": "1.10.12",
|
"version": "3.1.2",
|
||||||
"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",
|
||||||
@@ -10,46 +10,48 @@
|
|||||||
"test": "tstest test/ --web --verbose --timeout 30 --logfile",
|
"test": "tstest test/ --web --verbose --timeout 30 --logfile",
|
||||||
"build": "tsbuild tsfolders --allowimplicitany && tsbundle element --production --bundler esbuild",
|
"build": "tsbuild tsfolders --allowimplicitany && tsbundle element --production --bundler esbuild",
|
||||||
"watch": "tswatch element",
|
"watch": "tswatch element",
|
||||||
"buildDocs": "tsdoc"
|
"buildDocs": "tsdoc",
|
||||||
|
"postinstall": "node scripts/update-monaco-version.cjs"
|
||||||
},
|
},
|
||||||
"author": "Lossless GmbH",
|
"author": "Lossless GmbH",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@design.estate/dees-domtools": "^2.3.3",
|
"@design.estate/dees-domtools": "^2.3.6",
|
||||||
"@design.estate/dees-element": "^2.1.2",
|
"@design.estate/dees-element": "^2.1.3",
|
||||||
"@design.estate/dees-wcctools": "^1.1.1",
|
"@design.estate/dees-wcctools": "^1.2.1",
|
||||||
"@fortawesome/fontawesome-svg-core": "^7.0.1",
|
"@fortawesome/fontawesome-svg-core": "^7.1.0",
|
||||||
"@fortawesome/free-brands-svg-icons": "^7.0.1",
|
"@fortawesome/free-brands-svg-icons": "^7.1.0",
|
||||||
"@fortawesome/free-regular-svg-icons": "^7.0.1",
|
"@fortawesome/free-regular-svg-icons": "^7.1.0",
|
||||||
"@fortawesome/free-solid-svg-icons": "^7.0.1",
|
"@fortawesome/free-solid-svg-icons": "^7.1.0",
|
||||||
"@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.0.15",
|
"@push.rocks/smartstring": "^4.1.0",
|
||||||
"@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",
|
||||||
"@tiptap/extension-typography": "^2.23.0",
|
"@tiptap/extension-typography": "^2.23.0",
|
||||||
"@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.2.0",
|
"@tsclass/tsclass": "^9.3.0",
|
||||||
"@webcontainer/api": "1.2.0",
|
"@webcontainer/api": "1.2.0",
|
||||||
"apexcharts": "^5.3.4",
|
"apexcharts": "^5.3.6",
|
||||||
"highlight.js": "11.11.1",
|
"highlight.js": "11.11.1",
|
||||||
"ibantools": "^4.5.1",
|
"ibantools": "^4.5.1",
|
||||||
"lucide": "^0.542.0",
|
"lit": "^3.3.1",
|
||||||
"monaco-editor": "^0.52.2",
|
"lucide": "^0.555.0",
|
||||||
|
"monaco-editor": "0.52.2",
|
||||||
"pdfjs-dist": "^4.10.38",
|
"pdfjs-dist": "^4.10.38",
|
||||||
"xterm": "^5.3.0",
|
"xterm": "^5.3.0",
|
||||||
"xterm-addon-fit": "^0.8.0"
|
"xterm-addon-fit": "^0.8.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@git.zone/tsbuild": "^2.6.8",
|
"@git.zone/tsbuild": "^3.1.2",
|
||||||
"@git.zone/tsbundle": "^2.5.1",
|
"@git.zone/tsbundle": "^2.6.3",
|
||||||
"@git.zone/tstest": "^2.3.6",
|
"@git.zone/tstest": "^3.1.3",
|
||||||
"@git.zone/tswatch": "^2.2.1",
|
"@git.zone/tswatch": "^2.3.2",
|
||||||
"@push.rocks/projectinfo": "^5.0.2",
|
"@push.rocks/projectinfo": "^5.0.2",
|
||||||
"@push.rocks/tapbundle": "^6.0.3",
|
"@push.rocks/tapbundle": "^6.0.3",
|
||||||
"@types/node": "^22.0.0"
|
"@types/node": "^24.10.1"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
"ts/**/*",
|
"ts/**/*",
|
||||||
|
|||||||
7379
pnpm-lock.yaml
generated
7379
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -605,3 +605,79 @@ z-index: ${zIndexLayers.overlay.modal};
|
|||||||
```
|
```
|
||||||
|
|
||||||
This system ensures proper stacking order for all overlay components and prevents z-index conflicts.
|
This system ensures proper stacking order for all overlay components and prevents z-index conflicts.
|
||||||
|
|
||||||
|
## TC39 Standard Decorators Migration (2025-01-17)
|
||||||
|
|
||||||
|
Successfully migrated from experimental TypeScript decorators to standard TC39 decorators as recommended by Lit 3.x documentation.
|
||||||
|
|
||||||
|
### Migration Overview:
|
||||||
|
|
||||||
|
#### 1. Changes Made:
|
||||||
|
- **Added `accessor` keyword** to all `@property` and `@state` decorated fields across 69 component files
|
||||||
|
- **Updated tsconfig.json**: Removed `experimentalDecorators: true` and `useDefineForClassFields: false`
|
||||||
|
- **Fixed optional properties**: Changed `accessor prop?: Type` to `accessor prop: Type | undefined = undefined`
|
||||||
|
- **Removed incompatible decorators**: Removed `@query` and non-reactive `@state` decorators from regular fields
|
||||||
|
|
||||||
|
#### 2. Key Pattern Changes:
|
||||||
|
|
||||||
|
**Before (Experimental Decorators):**
|
||||||
|
```typescript
|
||||||
|
@property({ type: String })
|
||||||
|
public value: string = '';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
public disabled?: boolean;
|
||||||
|
```
|
||||||
|
|
||||||
|
**After (Standard TC39 Decorators):**
|
||||||
|
```typescript
|
||||||
|
@property({ type: String })
|
||||||
|
accessor value: string = '';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor disabled: boolean | undefined = undefined;
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 3. Important Rules:
|
||||||
|
- **@property and @state**: MUST use `accessor` keyword for reactive properties
|
||||||
|
- **@query decorators**: Should NOT use `accessor` (they work with regular fields)
|
||||||
|
- **Optional properties**: Cannot use `?` syntax with accessor, must use `| undefined = undefined`
|
||||||
|
- **Private fields**: Non-reactive private fields should not use decorators
|
||||||
|
|
||||||
|
#### 4. TypeScript Configuration:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "ES2022",
|
||||||
|
"module": "NodeNext",
|
||||||
|
"moduleResolution": "NodeNext"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
Note: `experimentalDecorators` defaults to false, and `useDefineForClassFields` defaults to true with ES2022 target.
|
||||||
|
|
||||||
|
#### 5. Build Results:
|
||||||
|
- ✅ Build successful with standard decorators
|
||||||
|
- ✅ Tests passing (7/8 - same as before migration)
|
||||||
|
- ✅ No bundle size changes reported
|
||||||
|
- ✅ All components working correctly
|
||||||
|
|
||||||
|
#### 6. Files Modified:
|
||||||
|
- 69 component files with decorator updates
|
||||||
|
- 16 files with optional property fixes
|
||||||
|
- 3 files with @query decorator removals
|
||||||
|
- tsconfig.json configuration update
|
||||||
|
|
||||||
|
### Why This Migration:
|
||||||
|
|
||||||
|
According to Lit's documentation (https://lit.dev/docs/components/decorators/#decorator-versions):
|
||||||
|
- TC39 standard decorators are the future-proof approach
|
||||||
|
- Provides better TypeScript integration
|
||||||
|
- Aligns with JavaScript specification
|
||||||
|
- While bundle sizes are slightly larger, the standardization benefits outweigh this
|
||||||
|
|
||||||
|
### Testing:
|
||||||
|
- All unit tests passing
|
||||||
|
- Manual testing of key components verified
|
||||||
|
- No regressions detected
|
||||||
|
- Focus management and interactions working correctly
|
||||||
1906
readme.icons.md
Normal file
1906
readme.icons.md
Normal file
File diff suppressed because it is too large
Load Diff
80
readme.info.md
Normal file
80
readme.info.md
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
# Contributor Information
|
||||||
|
|
||||||
|
This reference consolidates the helper notes previously split across `codex.md` and `CLAUDE.md`. Use it to get oriented quickly when working on `@design.estate/dees-catalog`, a TypeScript/Lit web-components library that ships themed UI building blocks for modern web applications.
|
||||||
|
|
||||||
|
## Project Snapshot
|
||||||
|
- Package: `@design.estate/dees-catalog`
|
||||||
|
- Description: Comprehensive catalog of reusable web components with cohesive design, advanced form inputs, data displays, and layout scaffolding.
|
||||||
|
- Entry points: builds ship to `dist_ts_web/` (ES modules) and `dist_bundle/` (browser bundle); demos live in `html/`.
|
||||||
|
- Type system: strict TypeScript targeting modern browsers (see `tsconfig.json`).
|
||||||
|
|
||||||
|
## Repository Layout
|
||||||
|
- `ts_web/` – TypeScript source
|
||||||
|
- `elements/` – component implementations (`00*.ts` shared utilities, `dees-*.ts` components, `*.demo.ts` demos)
|
||||||
|
- `pages/` – showcase pages aggregating demos
|
||||||
|
- `index.ts` – main export surface
|
||||||
|
- `html/` – demo entry point bootstrapping bundles
|
||||||
|
- `dist_bundle/`, `dist_ts_web/`, `dist_watch/` – build outputs (production, module, and watch bundles)
|
||||||
|
- `test/` – browser/node tests powered by `@push.rocks/tapbundle`
|
||||||
|
- `scripts/` – maintenance utilities (e.g., Monaco version sync postinstall)
|
||||||
|
|
||||||
|
## Build & Development Commands
|
||||||
|
All workflows use pnpm (see `package.json`).
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm install # install dependencies
|
||||||
|
pnpm run build # tsbuild tsfolders --allowimplicitany && tsbundle element --production --bundler esbuild
|
||||||
|
pnpm run watch # tswatch element (development watch/dev server)
|
||||||
|
pnpm test # tstest test/ --web --verbose --timeout 30 --logfile
|
||||||
|
pnpm run buildDocs # tsdoc (generates docs)
|
||||||
|
tsx test/test.file.ts # run a specific test file (file must be named test.*)
|
||||||
|
```
|
||||||
|
|
||||||
|
`postinstall` runs `node scripts/update-monaco-version.cjs` to sync the Monaco editor version, so keep the script intact when updating dependencies.
|
||||||
|
|
||||||
|
## Testing Guidelines
|
||||||
|
- Framework: `@push.rocks/tapbundle` with smartexpect assertions. Always review https://code.foss.global/push.rocks/smartexpect/raw/branch/master/readme.md when adding tests.
|
||||||
|
- Import pattern:
|
||||||
|
```typescript
|
||||||
|
import { tap, expect } from '@push.rocks/tapbundle';
|
||||||
|
```
|
||||||
|
- Test naming: `test.*.both.ts` for dual runtime, `.node.ts` for Node-only, `.browser.ts` for browser-only suites.
|
||||||
|
- Prefer `pnpm test` for full runs; use `tsx` for focused debugging. Type-check failing tests with `tsc --noEmit`.
|
||||||
|
- Logs live under `.nogit/testlogs/`; put ad-hoc debug artefacts in `.nogit/debug/`.
|
||||||
|
|
||||||
|
## Component Architecture
|
||||||
|
- **Base pattern**: Components extend `DeesElement` from `@design.estate/dees-element`, use Lit decorators (`@customElement`, `@property`), and combine `cssManager.defaultStyles` with component styles. Rendering happens via Lit `html` templates; demos sit on a static `demo` property referencing a `.demo.ts` module.
|
||||||
|
- **Theming**: `cssManager.bdTheme(light, dark)` selects theme-aware values. Shared palettes live in `ts_web/elements/00colors.ts`.
|
||||||
|
- **Z-index management**: Overlays consult the registry in `ts_web/elements/00zindex.ts` (`ZIndexRegistry`) to coordinate stacking.
|
||||||
|
- **Component families**:
|
||||||
|
- Core UI (`dees-button`, `dees-badge`, `dees-icon`, …) focus on consistent theming and interactions.
|
||||||
|
- Form inputs (`dees-form`, `dees-input-*`) build on `DeesInputBase` and communicate through subjects/events for validation.
|
||||||
|
- Layout shells (`dees-appui-*`) orchestrate responsive app frames with centralized event rebroadcasts.
|
||||||
|
- Data views (`dees-table`, `dees-dataview-*`, `dees-statsgrid`) handle large datasets with virtualisation and chart integrations.
|
||||||
|
- Overlays (`dees-modal`, `dees-contextmenu`, `dees-toast`) respect the z-index registry and use shared window-layer utilities.
|
||||||
|
- **WYSIWYG editor**: `dees-input-wysiwyg` coordinates specialized handler classes (`WysiwygInputHandler`, `WysiwygKeyboardHandler`, drag/drop & modal managers) and global menus (`DeesSlashMenu`, `DeesFormattingMenu`). Rendering is imperative to preserve caret focus.
|
||||||
|
|
||||||
|
## Implementation Guidelines
|
||||||
|
- Import external modules through `ts_web/elements/00plugins.ts`: `import * as plugins from './plugins.ts';` then reference `plugins.moduleName`.
|
||||||
|
- When creating new components:
|
||||||
|
1. Extend `DeesElement` and decorate with `@customElement('dees-component')`.
|
||||||
|
2. Support theming, slots, and accessibility; provide meaningful default styles.
|
||||||
|
3. Expose a `.demo.ts` for the component and re-export via `elements/index.ts`.
|
||||||
|
- Form components must implement `getValue()` / `setValue()` and emit through `changeSubject` while honoring `disabled` and `required` states.
|
||||||
|
- Overlay components retrieve z-indices from the registry, register/unregister on show/hide, and use `DeesWindowLayer` for backdrops when appropriate.
|
||||||
|
- Avoid simplifying away functionality; prefer small, targeted changes and keep compatibility with existing APIs.
|
||||||
|
|
||||||
|
## Common Patterns & Pitfalls
|
||||||
|
- Focus management: schedule DOM updates with `requestAnimationFrame` inside interactive editors to avoid focus loss.
|
||||||
|
- Event handling: stop propagation where nested interactive elements coexist; mix pointer and keyboard handling for accessibility.
|
||||||
|
- Performance: heavy blocks/components may load lazily; charts use debounced observers, tables rely on virtual scrolling. Watch bundle size when adding dependencies.
|
||||||
|
|
||||||
|
## Documentation & Demos
|
||||||
|
- `readme.md` surfaces component overviews; demos in `.demo.ts` illustrate real usage.
|
||||||
|
- Update this `readme.info.md` when architectural patterns or workflows change so contributors stay in sync.
|
||||||
|
|
||||||
|
## Recent Highlights
|
||||||
|
- Z-index registry overhaul enables dynamic stacking control across overlays.
|
||||||
|
- WYSIWYG refactor separated block handlers for maintainability.
|
||||||
|
- Dashboard grid enhancements added live drag-and-drop previews and overlap fixes.
|
||||||
|
- Monaco editor integration now reads the installed version at build time.
|
||||||
829
readme.md
829
readme.md
@@ -92,15 +92,6 @@ Display icons from FontAwesome and Lucide icon libraries with library prefixes.
|
|||||||
strokeWidth="2" // Optional: stroke width for Lucide icons
|
strokeWidth="2" // Optional: stroke width for Lucide icons
|
||||||
></dees-icon>
|
></dees-icon>
|
||||||
|
|
||||||
// Available FontAwesome icons include:
|
|
||||||
// fa:check, fa:bell, fa:gear, fa:trash, fa:copy, fa:paste, fa:eye, fa:eyeSlash,
|
|
||||||
// fa:plus, fa:minus, fa:circleInfo, fa:circleCheck, fa:circleXmark, fa:message,
|
|
||||||
// fa:arrowRight, fa:facebook, fa:twitter, fa:linkedin, fa:instagram, etc.
|
|
||||||
|
|
||||||
// Available Lucide icons include:
|
|
||||||
// lucide:menu, lucide:settings, lucide:home, lucide:file, lucide:folder,
|
|
||||||
// lucide:search, lucide:user, lucide:heart, lucide:star, lucide:download, etc.
|
|
||||||
|
|
||||||
// Legacy API (deprecated but still supported)
|
// Legacy API (deprecated but still supported)
|
||||||
<dees-icon
|
<dees-icon
|
||||||
iconFA="check" // Without prefix - assumes FontAwesome
|
iconFA="check" // Without prefix - assumes FontAwesome
|
||||||
@@ -599,10 +590,10 @@ Base container component for application layout structure with integrated appbar
|
|||||||
.appbarMenuItems=${[
|
.appbarMenuItems=${[
|
||||||
{
|
{
|
||||||
name: 'File',
|
name: 'File',
|
||||||
action: async () => {},
|
action: async () => {}, // No-op for parent menu items
|
||||||
submenu: [
|
submenu: [
|
||||||
{ name: 'New', shortcut: 'Cmd+N', iconName: 'file-plus', action: async () => {} },
|
{ name: 'New File', shortcut: 'Cmd+N', iconName: 'file-plus', action: async () => {} },
|
||||||
{ name: 'Open', shortcut: 'Cmd+O', iconName: 'folder-open', action: async () => {} },
|
{ name: 'Open...', shortcut: 'Cmd+O', iconName: 'folder-open', action: async () => {} },
|
||||||
{ divider: true },
|
{ divider: true },
|
||||||
{ name: 'Save', shortcut: 'Cmd+S', iconName: 'save', action: async () => {} }
|
{ name: 'Save', shortcut: 'Cmd+S', iconName: 'save', action: async () => {} }
|
||||||
]
|
]
|
||||||
@@ -618,10 +609,7 @@ Base container component for application layout structure with integrated appbar
|
|||||||
]}
|
]}
|
||||||
.appbarBreadcrumbs=${'Dashboard > Overview'}
|
.appbarBreadcrumbs=${'Dashboard > Overview'}
|
||||||
.appbarTheme=${'dark'}
|
.appbarTheme=${'dark'}
|
||||||
.appbarUser=${{
|
.appbarUser=${{ name: 'John Doe', status: 'online' }}
|
||||||
name: 'John Doe',
|
|
||||||
status: 'online'
|
|
||||||
}}
|
|
||||||
.appbarShowSearch=${true}
|
.appbarShowSearch=${true}
|
||||||
.appbarShowWindowControls=${true}
|
.appbarShowWindowControls=${true}
|
||||||
|
|
||||||
@@ -667,43 +655,6 @@ Key Features:
|
|||||||
- **Responsive Grid**: Uses CSS Grid for flexible, responsive layout
|
- **Responsive Grid**: Uses CSS Grid for flexible, responsive layout
|
||||||
- **Slot Support**: Main content area supports custom content via slots
|
- **Slot Support**: Main content area supports custom content via slots
|
||||||
|
|
||||||
Layout Structure:
|
|
||||||
```
|
|
||||||
┌─────────────────────────────────────────────────┐
|
|
||||||
│ AppBar │
|
|
||||||
├────┬──────────────┬─────────────────┬──────────┤
|
|
||||||
│ │ │ │ │
|
|
||||||
│ M │ Selector │ Main Content │ Activity │
|
|
||||||
│ e │ │ │ Log │
|
|
||||||
│ n │ │ │ │
|
|
||||||
│ u │ │ │ │
|
|
||||||
│ │ │ │ │
|
|
||||||
└────┴──────────────┴─────────────────┴──────────┘
|
|
||||||
```
|
|
||||||
|
|
||||||
Grid Configuration:
|
|
||||||
- Main Menu: 60px width
|
|
||||||
- Selector: 240px width
|
|
||||||
- Main Content: Flexible (1fr)
|
|
||||||
- Activity Log: 240px width
|
|
||||||
|
|
||||||
Child Component Access:
|
|
||||||
```typescript
|
|
||||||
// Access child components after firstUpdated
|
|
||||||
const base = document.querySelector('dees-appui-base');
|
|
||||||
base.appbar; // DeesAppuiAppbar instance
|
|
||||||
base.mainmenu; // DeesAppuiMainmenu instance
|
|
||||||
base.mainselector; // DeesAppuiMainselector instance
|
|
||||||
base.maincontent; // DeesAppuiMaincontent instance
|
|
||||||
base.activitylog; // DeesAppuiActivitylog instance
|
|
||||||
```
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
1. **Configuration**: Set all properties on the base component for consistency
|
|
||||||
2. **Event Handling**: Listen to events on the base component rather than child components
|
|
||||||
3. **Content**: Use the `maincontent` slot for your application's primary interface
|
|
||||||
4. **State Management**: Manage selected tabs and options at the base component level
|
|
||||||
|
|
||||||
#### `DeesAppuiMainmenu`
|
#### `DeesAppuiMainmenu`
|
||||||
Main navigation menu component for application-wide navigation.
|
Main navigation menu component for application-wide navigation.
|
||||||
|
|
||||||
@@ -860,257 +811,6 @@ Key Features:
|
|||||||
- Focus management
|
- Focus management
|
||||||
- Screen reader compatible
|
- Screen reader compatible
|
||||||
|
|
||||||
Menu Item Interface:
|
|
||||||
```typescript
|
|
||||||
// Regular menu item
|
|
||||||
interface IAppBarMenuItemRegular {
|
|
||||||
name: string; // Display text
|
|
||||||
action: () => Promise<any>; // Click handler
|
|
||||||
iconName?: string; // Optional icon (for dropdown items)
|
|
||||||
shortcut?: string; // Keyboard shortcut display
|
|
||||||
submenu?: IAppBarMenuItem[]; // Nested menu items
|
|
||||||
disabled?: boolean; // Disabled state
|
|
||||||
checked?: boolean; // For checkbox menu items
|
|
||||||
radioGroup?: string; // For radio button menu items
|
|
||||||
}
|
|
||||||
|
|
||||||
// Divider item
|
|
||||||
interface IAppBarMenuDivider {
|
|
||||||
divider: true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Combined type
|
|
||||||
type IAppBarMenuItem = IAppBarMenuItemRegular | IAppBarMenuDivider;
|
|
||||||
```
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
1. **Menu Structure**
|
|
||||||
- Keep top-level menus text-only (no icons)
|
|
||||||
- Use icons in dropdown items for visual clarity
|
|
||||||
- Group related actions with dividers
|
|
||||||
- Provide keyboard shortcuts for common actions
|
|
||||||
|
|
||||||
2. **Navigation**
|
|
||||||
- Use breadcrumbs for deep navigation hierarchies
|
|
||||||
- Keep breadcrumb labels concise
|
|
||||||
- Provide meaningful navigation events
|
|
||||||
|
|
||||||
3. **User Experience**
|
|
||||||
- Show user status when relevant
|
|
||||||
- Provide clear visual feedback
|
|
||||||
- Ensure smooth transitions
|
|
||||||
- Handle edge cases (long menus, small screens)
|
|
||||||
|
|
||||||
4. **Accessibility**
|
|
||||||
- Always provide text labels
|
|
||||||
- Ensure keyboard navigation works
|
|
||||||
- Test with screen readers
|
|
||||||
- Maintain focus management
|
|
||||||
|
|
||||||
#### `DeesAppuiActivitylog`
|
|
||||||
Activity log component for displaying system events and user actions.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-appui-activitylog
|
|
||||||
.entries=${[
|
|
||||||
{
|
|
||||||
timestamp: new Date(),
|
|
||||||
type: 'info',
|
|
||||||
message: 'User logged in',
|
|
||||||
details: { userId: '123' }
|
|
||||||
},
|
|
||||||
{
|
|
||||||
timestamp: new Date(),
|
|
||||||
type: 'error',
|
|
||||||
message: 'Failed to save document',
|
|
||||||
details: { error: 'Network error' }
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
maxEntries={100} // Maximum entries to display
|
|
||||||
@entry-click=${handleEntryClick}
|
|
||||||
></dees-appui-activitylog>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesAppuiProfiledropdown`
|
|
||||||
User profile dropdown component with status and menu options.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-appui-profiledropdown
|
|
||||||
.user=${{
|
|
||||||
name: 'John Doe',
|
|
||||||
email: 'john@example.com',
|
|
||||||
avatar: '/path/to/avatar.jpg',
|
|
||||||
status: 'online' // Options: online, offline, busy, away
|
|
||||||
}}
|
|
||||||
.menuItems=${[
|
|
||||||
{ name: 'Profile', iconName: 'user', action: async () => {} },
|
|
||||||
{ name: 'Settings', iconName: 'settings', action: async () => {} },
|
|
||||||
{ divider: true },
|
|
||||||
{ name: 'Logout', iconName: 'logOut', action: async () => {} }
|
|
||||||
]}
|
|
||||||
@status-change=${handleStatusChange}
|
|
||||||
></dees-appui-profiledropdown>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesAppuiTabs`
|
|
||||||
Tab navigation component for organizing content sections.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-appui-tabs
|
|
||||||
.tabs=${[
|
|
||||||
{
|
|
||||||
key: 'overview',
|
|
||||||
label: 'Overview',
|
|
||||||
icon: 'home',
|
|
||||||
content: html`<div>Overview content</div>`
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 'details',
|
|
||||||
label: 'Details',
|
|
||||||
icon: 'info',
|
|
||||||
content: html`<div>Details content</div>`
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
selectedTab="overview"
|
|
||||||
@tab-change=${handleTabChange}
|
|
||||||
></dees-appui-tabs>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesAppuiView`
|
|
||||||
View container component for consistent page layouts.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-appui-view
|
|
||||||
viewTitle="Dashboard"
|
|
||||||
viewSubtitle="System Overview"
|
|
||||||
.headerActions=${[
|
|
||||||
{ icon: 'refresh', action: handleRefresh },
|
|
||||||
{ icon: 'settings', action: handleSettings }
|
|
||||||
]}
|
|
||||||
>
|
|
||||||
<!-- View content -->
|
|
||||||
</dees-appui-view>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesMobileNavigation`
|
|
||||||
Responsive navigation component for mobile devices.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-mobile-navigation
|
|
||||||
.menuItems=${[
|
|
||||||
{
|
|
||||||
key: 'home',
|
|
||||||
label: 'Home',
|
|
||||||
icon: 'home',
|
|
||||||
action: () => navigate('home')
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: 'profile',
|
|
||||||
label: 'Profile',
|
|
||||||
icon: 'user',
|
|
||||||
action: () => navigate('profile')
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
activeKey="home" // Currently active item
|
|
||||||
position="bottom" // Options: bottom, top
|
|
||||||
@item-click=${handleNavigationClick}
|
|
||||||
></dees-mobile-navigation>
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesDashboardGrid`
|
|
||||||
Drag-and-drop grid layout system for creating customizable dashboards.
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
<dees-dashboardgrid
|
|
||||||
.widgets=${[
|
|
||||||
{
|
|
||||||
id: 'widget1',
|
|
||||||
x: 0, // Grid column position
|
|
||||||
y: 0, // Grid row position
|
|
||||||
w: 4, // Width in grid units
|
|
||||||
h: 3, // Height in grid units
|
|
||||||
minW: 2, // Minimum width
|
|
||||||
minH: 2, // Minimum height
|
|
||||||
maxW: 6, // Maximum width
|
|
||||||
title: 'Sales Overview',
|
|
||||||
icon: 'fa:chart-line',
|
|
||||||
content: html`<div>Widget content here</div>`,
|
|
||||||
noMove: false, // Allow moving
|
|
||||||
noResize: false // Allow resizing
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'widget2',
|
|
||||||
x: 4,
|
|
||||||
y: 0,
|
|
||||||
w: 4,
|
|
||||||
h: 3,
|
|
||||||
title: 'Recent Activity',
|
|
||||||
content: html`<dees-table .data=${activityData}></dees-table>`,
|
|
||||||
autoPosition: true // Auto-find position
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
columns={12} // Number of grid columns
|
|
||||||
cellHeight={80} // Height of each grid cell in pixels
|
|
||||||
cellHeightUnit="px" // Options: px, em, rem, auto
|
|
||||||
margin={10} // Gap between widgets
|
|
||||||
editable={true} // Enable drag and resize
|
|
||||||
showGridLines={false} // Show grid guidelines
|
|
||||||
enableAnimation={true} // Smooth transitions
|
|
||||||
rtl={false} // Right-to-left support
|
|
||||||
@widget-move=${handleWidgetMove}
|
|
||||||
@widget-resize=${handleWidgetResize}
|
|
||||||
></dees-dashboardgrid>
|
|
||||||
|
|
||||||
// Programmatic methods
|
|
||||||
const grid = document.querySelector('dees-dashboardgrid');
|
|
||||||
|
|
||||||
// Add a new widget
|
|
||||||
grid.addWidget({
|
|
||||||
id: 'newWidget',
|
|
||||||
x: 0,
|
|
||||||
y: 0,
|
|
||||||
w: 3,
|
|
||||||
h: 2,
|
|
||||||
content: html`<div>New widget</div>`
|
|
||||||
}, true); // true = auto-position
|
|
||||||
|
|
||||||
// Remove widget
|
|
||||||
grid.removeWidget('widget1');
|
|
||||||
|
|
||||||
// Update widget
|
|
||||||
grid.updateWidget('widget2', {
|
|
||||||
title: 'Updated Title',
|
|
||||||
w: 6
|
|
||||||
});
|
|
||||||
|
|
||||||
// Get/set layout
|
|
||||||
const layout = grid.getLayout(); // Returns position data
|
|
||||||
grid.setLayout(savedLayout); // Restore positions
|
|
||||||
|
|
||||||
// Compact widgets
|
|
||||||
grid.compact('vertical'); // Or 'horizontal'
|
|
||||||
|
|
||||||
// Lock/unlock editing
|
|
||||||
grid.lockGrid();
|
|
||||||
grid.unlockGrid();
|
|
||||||
```
|
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Drag-and-drop widget repositioning
|
|
||||||
- Resize handles on edges and corners
|
|
||||||
- Grid-based layout system
|
|
||||||
- Collision detection
|
|
||||||
- Auto-positioning for new widgets
|
|
||||||
- Configurable constraints (min/max dimensions)
|
|
||||||
- Lock individual widgets or entire grid
|
|
||||||
- Compact layout algorithm
|
|
||||||
- Save/restore layout positions
|
|
||||||
- RTL layout support
|
|
||||||
- Optional grid lines for alignment
|
|
||||||
- Smooth animations
|
|
||||||
- Responsive sizing
|
|
||||||
- Empty state display
|
|
||||||
|
|
||||||
### Data Display Components
|
### Data Display Components
|
||||||
|
|
||||||
#### `DeesTable`
|
#### `DeesTable`
|
||||||
@@ -1145,6 +845,19 @@ Advanced table component with sorting, filtering, and action support.
|
|||||||
></dees-table>
|
></dees-table>
|
||||||
```
|
```
|
||||||
|
|
||||||
|
##### DeesTable (Updated)
|
||||||
|
|
||||||
|
Newer features available in `dees-table`:
|
||||||
|
- Schema-first columns or `displayFunction` rendering
|
||||||
|
- Sorting via header clicks with `aria-sort` + `sortChange`
|
||||||
|
- Global search with Lucene-like syntax; modes: `table`, `data`, `server`
|
||||||
|
- Per-column quick filters row; `showColumnFilters` and `column.filterable=false`
|
||||||
|
- Selection: `none` | `single` | `multi`, with select-all and `selectionChange`
|
||||||
|
- Sticky header + internal scroll (`stickyHeader`, `--table-max-height`)
|
||||||
|
- Rich actions: header/in-row/contextmenu/footer/doubleClick; pinned Actions column
|
||||||
|
- Editable cells via `editableFields`
|
||||||
|
- Drag & drop files onto rows
|
||||||
|
|
||||||
#### `DeesDataviewCodebox`
|
#### `DeesDataviewCodebox`
|
||||||
Code display component with syntax highlighting and line numbers.
|
Code display component with syntax highlighting and line numbers.
|
||||||
|
|
||||||
@@ -1155,7 +868,7 @@ Code display component with syntax highlighting and line numbers.
|
|||||||
import { html } from '@design.estate/dees-element';
|
import { html } from '@design.estate/dees-element';
|
||||||
|
|
||||||
export const myComponent = () => {
|
export const myComponent = () => {
|
||||||
return html\`<div>Hello World</div>\`;
|
return html`<div>Hello World</div>`;
|
||||||
};
|
};
|
||||||
`}
|
`}
|
||||||
></dees-dataview-codebox>
|
></dees-dataview-codebox>
|
||||||
@@ -1208,37 +921,6 @@ PDF viewer component with navigation and zoom controls.
|
|||||||
></dees-pdf>
|
></dees-pdf>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- `DeesTable`:
|
|
||||||
- Sortable columns
|
|
||||||
- Searchable content
|
|
||||||
- Customizable row actions
|
|
||||||
- Selection support
|
|
||||||
- Form compatibility
|
|
||||||
- Custom display formatting
|
|
||||||
|
|
||||||
- `DeesDataviewCodebox`:
|
|
||||||
- Syntax highlighting for multiple languages
|
|
||||||
- Line numbering
|
|
||||||
- Copy to clipboard functionality
|
|
||||||
- Custom theme support
|
|
||||||
- Window-like appearance with controls
|
|
||||||
|
|
||||||
- `DeesDataviewStatusobject`:
|
|
||||||
- Hierarchical status display
|
|
||||||
- Color-coded status indicators
|
|
||||||
- Expandable details
|
|
||||||
- JSON export capability
|
|
||||||
- Customizable styling
|
|
||||||
|
|
||||||
- `DeesPdf`:
|
|
||||||
- Page navigation
|
|
||||||
- Zoom controls
|
|
||||||
- Download support
|
|
||||||
- Print functionality
|
|
||||||
- Responsive layout
|
|
||||||
- Loading states
|
|
||||||
|
|
||||||
#### `DeesStatsGrid`
|
#### `DeesStatsGrid`
|
||||||
A responsive grid component for displaying statistical data with various visualization types including numbers, gauges, percentages, and trends.
|
A responsive grid component for displaying statistical data with various visualization types including numbers, gauges, percentages, and trends.
|
||||||
|
|
||||||
@@ -1336,116 +1018,6 @@ A responsive grid component for displaying statistical data with various visuali
|
|||||||
></dees-statsgrid>
|
></dees-statsgrid>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Auto-responsive grid layout with configurable minimum tile width
|
|
||||||
- Multiple tile types for different data visualizations
|
|
||||||
- Full theme support (light/dark mode)
|
|
||||||
- Interactive tiles with action support
|
|
||||||
- Grid-level and tile-level actions
|
|
||||||
- Smooth animations and transitions
|
|
||||||
- Icon support for visual hierarchy
|
|
||||||
|
|
||||||
Tile Types:
|
|
||||||
1. **`number`** - Display numeric values with optional units
|
|
||||||
- Large, prominent value display
|
|
||||||
- Optional unit display
|
|
||||||
- Custom color support
|
|
||||||
- Description text
|
|
||||||
|
|
||||||
2. **`gauge`** - Circular gauge visualization
|
|
||||||
- Min/max value configuration
|
|
||||||
- Color thresholds for visual alerts
|
|
||||||
- Animated value transitions
|
|
||||||
- Compact circular design
|
|
||||||
|
|
||||||
3. **`percentage`** - Progress bar visualization
|
|
||||||
- Horizontal progress bar
|
|
||||||
- Percentage display overlay
|
|
||||||
- Custom color support
|
|
||||||
- Ideal for capacity metrics
|
|
||||||
|
|
||||||
4. **`trend`** - Mini sparkline chart
|
|
||||||
- Array of numeric values for trend data
|
|
||||||
- Area chart visualization
|
|
||||||
- Current value display
|
|
||||||
- Responsive SVG rendering
|
|
||||||
|
|
||||||
5. **`text`** - Simple text display
|
|
||||||
- Flexible text content
|
|
||||||
- Custom color support
|
|
||||||
- Ideal for status messages
|
|
||||||
|
|
||||||
Action System:
|
|
||||||
- **Grid Actions**: Displayed as buttons in the grid header
|
|
||||||
- Apply to the entire stats grid
|
|
||||||
- Use standard `dees-button` components
|
|
||||||
- Support icons and text
|
|
||||||
|
|
||||||
- **Tile Actions**: Context-specific actions per tile
|
|
||||||
- Single action: Direct click on tile
|
|
||||||
- Multiple actions: Right-click context menu
|
|
||||||
- Actions access tile data through closures
|
|
||||||
- Consistent with other library components
|
|
||||||
|
|
||||||
Configuration Options:
|
|
||||||
- `tiles`: Array of `IStatsTile` objects defining the grid content
|
|
||||||
- `gridActions`: Array of actions for the entire grid
|
|
||||||
- `minTileWidth`: Minimum width for tiles (default: 250px)
|
|
||||||
- `gap`: Space between tiles (default: 16px)
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
1. **Data Organization**
|
|
||||||
- Group related metrics together
|
|
||||||
- Use consistent units and scales
|
|
||||||
- Provide meaningful descriptions
|
|
||||||
- Choose appropriate tile types for data
|
|
||||||
|
|
||||||
2. **Visual Hierarchy**
|
|
||||||
- Use colors strategically for alerts
|
|
||||||
- Include relevant icons
|
|
||||||
- Keep titles concise
|
|
||||||
- Balance tile types for visual interest
|
|
||||||
|
|
||||||
3. **Interactivity**
|
|
||||||
- Provide relevant actions for detailed views
|
|
||||||
- Use tile actions for item-specific operations
|
|
||||||
- Use grid actions for global operations
|
|
||||||
- Keep action names clear and concise
|
|
||||||
|
|
||||||
4. **Performance**
|
|
||||||
- Update only changed tiles
|
|
||||||
- Use reasonable update intervals
|
|
||||||
- Batch updates when possible
|
|
||||||
- Consider data volume for trends
|
|
||||||
|
|
||||||
Common Use Cases:
|
|
||||||
- System monitoring dashboards
|
|
||||||
- Business intelligence displays
|
|
||||||
- Performance metrics
|
|
||||||
- Resource utilization
|
|
||||||
- Real-time statistics
|
|
||||||
- KPI tracking
|
|
||||||
|
|
||||||
Integration Example:
|
|
||||||
```typescript
|
|
||||||
// Real-time updates
|
|
||||||
setInterval(() => {
|
|
||||||
const grid = document.querySelector('dees-statsgrid');
|
|
||||||
const updatedTiles = [...grid.tiles];
|
|
||||||
|
|
||||||
// Update specific tile
|
|
||||||
const cpuTile = updatedTiles.find(t => t.id === 'cpu');
|
|
||||||
cpuTile.value = Math.round(Math.random() * 100);
|
|
||||||
|
|
||||||
// Update trend data
|
|
||||||
const trendTile = updatedTiles.find(t => t.id === 'requests');
|
|
||||||
trendTile.trendData = [...trendTile.trendData.slice(1),
|
|
||||||
Math.round(Math.random() * 100)];
|
|
||||||
|
|
||||||
grid.tiles = updatedTiles;
|
|
||||||
}, 3000);
|
|
||||||
```
|
|
||||||
|
|
||||||
#### `DeesPagination`
|
#### `DeesPagination`
|
||||||
Pagination component for navigating through large datasets.
|
Pagination component for navigating through large datasets.
|
||||||
|
|
||||||
@@ -1459,14 +1031,6 @@ Pagination component for navigating through large datasets.
|
|||||||
></dees-pagination>
|
></dees-pagination>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Page number navigation
|
|
||||||
- Previous/next buttons
|
|
||||||
- Jump to first/last page
|
|
||||||
- Configurable items per page
|
|
||||||
- Responsive design
|
|
||||||
- Keyboard navigation support
|
|
||||||
|
|
||||||
### Visualization Components
|
### Visualization Components
|
||||||
|
|
||||||
#### `DeesChartArea`
|
#### `DeesChartArea`
|
||||||
@@ -1496,22 +1060,6 @@ Area chart component built on ApexCharts for visualizing time-series data.
|
|||||||
></dees-chart-area>
|
></dees-chart-area>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Responsive design with automatic resizing
|
|
||||||
- Gradient fill support
|
|
||||||
- Interactive tooltips
|
|
||||||
- Grid customization
|
|
||||||
- Multiple series support
|
|
||||||
- Time-based x-axis
|
|
||||||
- Customizable styling
|
|
||||||
|
|
||||||
Configuration Options:
|
|
||||||
- Series data format: `{ x: timestamp, y: value }`
|
|
||||||
- Tooltip customization with datetime formatting
|
|
||||||
- Grid line styling and colors
|
|
||||||
- Gradient fill properties
|
|
||||||
- Chart dimensions and responsiveness
|
|
||||||
|
|
||||||
#### `DeesChartLog`
|
#### `DeesChartLog`
|
||||||
Specialized chart component for visualizing log data and events.
|
Specialized chart component for visualizing log data and events.
|
||||||
|
|
||||||
@@ -1535,44 +1083,6 @@ Specialized chart component for visualizing log data and events.
|
|||||||
></dees-chart-log>
|
></dees-chart-log>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Event timeline visualization
|
|
||||||
- Color-coded event types
|
|
||||||
- Interactive event details
|
|
||||||
- Filtering capabilities
|
|
||||||
- Zoom and pan support
|
|
||||||
- Time-based navigation
|
|
||||||
- Event clustering
|
|
||||||
|
|
||||||
Common Use Cases:
|
|
||||||
- System monitoring
|
|
||||||
- Performance tracking
|
|
||||||
- Resource utilization
|
|
||||||
- Event logging
|
|
||||||
- Time-series analysis
|
|
||||||
- Trend visualization
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
1. Data Formatting
|
|
||||||
- Use consistent timestamp formats
|
|
||||||
- Provide meaningful series names
|
|
||||||
- Include appropriate data points
|
|
||||||
|
|
||||||
2. Responsiveness
|
|
||||||
- Charts automatically adjust to container size
|
|
||||||
- Consider mobile viewports
|
|
||||||
- Set appropriate min/max dimensions
|
|
||||||
|
|
||||||
3. Interaction
|
|
||||||
- Enable relevant tooltips
|
|
||||||
- Configure meaningful click handlers
|
|
||||||
- Implement appropriate zoom levels
|
|
||||||
|
|
||||||
4. Styling
|
|
||||||
- Use consistent color schemes
|
|
||||||
- Configure appropriate grid lines
|
|
||||||
- Set readable font sizes
|
|
||||||
|
|
||||||
### Dialogs & Overlays Components
|
### Dialogs & Overlays Components
|
||||||
|
|
||||||
#### `DeesModal`
|
#### `DeesModal`
|
||||||
@@ -1616,14 +1126,6 @@ DeesModal.createAndShow({
|
|||||||
></dees-modal>
|
></dees-modal>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Backdrop blur effect
|
|
||||||
- Customizable content using HTML templates
|
|
||||||
- Flexible action buttons
|
|
||||||
- Outside click handling
|
|
||||||
- Animated transitions
|
|
||||||
- Automatic window layer management
|
|
||||||
|
|
||||||
#### `DeesContextmenu`
|
#### `DeesContextmenu`
|
||||||
Context menu component for right-click actions.
|
Context menu component for right-click actions.
|
||||||
|
|
||||||
@@ -1663,13 +1165,6 @@ const bubble = await DeesSpeechbubble.createAndShow(
|
|||||||
></dees-speechbubble>
|
></dees-speechbubble>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Automatic positioning
|
|
||||||
- Non-intrusive overlay
|
|
||||||
- Animated appearance
|
|
||||||
- Reference element tracking
|
|
||||||
- Custom styling options
|
|
||||||
|
|
||||||
#### `DeesWindowlayer`
|
#### `DeesWindowlayer`
|
||||||
Base overlay component used by modal dialogs and other overlay components.
|
Base overlay component used by modal dialogs and other overlay components.
|
||||||
|
|
||||||
@@ -1692,43 +1187,6 @@ const layer = await DeesWindowLayer.createAndShow({
|
|||||||
</dees-windowlayer>
|
</dees-windowlayer>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Backdrop blur support
|
|
||||||
- Click event handling
|
|
||||||
- Z-index management
|
|
||||||
- Animated transitions
|
|
||||||
- Flexible content container
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
|
|
||||||
1. Modal Dialogs
|
|
||||||
- Use for important user interactions
|
|
||||||
- Provide clear action buttons
|
|
||||||
- Include close/cancel options
|
|
||||||
- Handle outside clicks appropriately
|
|
||||||
- Use meaningful headings
|
|
||||||
|
|
||||||
2. Context Menus
|
|
||||||
- Group related actions
|
|
||||||
- Use consistent icons
|
|
||||||
- Provide keyboard shortcuts
|
|
||||||
- Consider position constraints
|
|
||||||
- Handle menu item states
|
|
||||||
|
|
||||||
3. Speech Bubbles
|
|
||||||
- Keep content concise
|
|
||||||
- Position strategically
|
|
||||||
- Avoid overlapping
|
|
||||||
- Consider mobile viewports
|
|
||||||
- Use appropriate timing
|
|
||||||
|
|
||||||
4. Window Layers
|
|
||||||
- Manage z-index carefully
|
|
||||||
- Handle backdrop interactions
|
|
||||||
- Consider performance impact
|
|
||||||
- Implement proper cleanup
|
|
||||||
- Manage multiple layers
|
|
||||||
|
|
||||||
### Navigation Components
|
### Navigation Components
|
||||||
|
|
||||||
#### `DeesStepper`
|
#### `DeesStepper`
|
||||||
@@ -1761,14 +1219,6 @@ Multi-step navigation component for guided user flows.
|
|||||||
></dees-stepper>
|
></dees-stepper>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Linear or non-linear progression
|
|
||||||
- Step validation
|
|
||||||
- Progress tracking
|
|
||||||
- Customizable step content
|
|
||||||
- Navigation controls
|
|
||||||
- Step completion indicators
|
|
||||||
|
|
||||||
#### `DeesProgressbar`
|
#### `DeesProgressbar`
|
||||||
Progress indicator component for tracking completion status.
|
Progress indicator component for tracking completion status.
|
||||||
|
|
||||||
@@ -1783,53 +1233,6 @@ Progress indicator component for tracking completion status.
|
|||||||
></dees-progressbar>
|
></dees-progressbar>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Determinate and indeterminate states
|
|
||||||
- Percentage display
|
|
||||||
- Custom styling options
|
|
||||||
- Status indicators
|
|
||||||
- Animation support
|
|
||||||
- Accessibility features
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
|
|
||||||
1. Stepper Implementation
|
|
||||||
- Clear step labels
|
|
||||||
- Validation feedback
|
|
||||||
- Progress indication
|
|
||||||
- Error handling
|
|
||||||
- Consistent navigation
|
|
||||||
|
|
||||||
2. Progress Tracking
|
|
||||||
- Accurate progress calculation
|
|
||||||
- Clear visual feedback
|
|
||||||
- Status communication
|
|
||||||
- Performance monitoring
|
|
||||||
- Error state handling
|
|
||||||
|
|
||||||
Common Use Cases:
|
|
||||||
|
|
||||||
1. Stepper
|
|
||||||
- Multi-step forms
|
|
||||||
- User onboarding
|
|
||||||
- Checkout processes
|
|
||||||
- Setup wizards
|
|
||||||
- Tutorial flows
|
|
||||||
|
|
||||||
2. Progress Bar
|
|
||||||
- File uploads
|
|
||||||
- Process tracking
|
|
||||||
- Loading indicators
|
|
||||||
- Task completion
|
|
||||||
- Step progression
|
|
||||||
|
|
||||||
Accessibility Considerations:
|
|
||||||
- Keyboard navigation support
|
|
||||||
- ARIA labels and roles
|
|
||||||
- Focus management
|
|
||||||
- Screen reader compatibility
|
|
||||||
- Color contrast compliance
|
|
||||||
|
|
||||||
### Development Components
|
### Development Components
|
||||||
|
|
||||||
#### `DeesEditor`
|
#### `DeesEditor`
|
||||||
@@ -1849,17 +1252,6 @@ Code editor component with syntax highlighting and code completion, powered by M
|
|||||||
></dees-editor>
|
></dees-editor>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Syntax highlighting for multiple languages
|
|
||||||
- Code completion
|
|
||||||
- Line numbers
|
|
||||||
- Minimap navigation
|
|
||||||
- Customizable options
|
|
||||||
- Theme support
|
|
||||||
- Search and replace
|
|
||||||
- Multiple cursors
|
|
||||||
- Code folding
|
|
||||||
|
|
||||||
#### `DeesEditorMarkdown`
|
#### `DeesEditorMarkdown`
|
||||||
Markdown editor component with live preview.
|
Markdown editor component with live preview.
|
||||||
|
|
||||||
@@ -1876,16 +1268,6 @@ Markdown editor component with live preview.
|
|||||||
></dees-editor-markdown>
|
></dees-editor-markdown>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Live preview
|
|
||||||
- Toolbar for common formatting
|
|
||||||
- Markdown syntax highlighting
|
|
||||||
- Image upload support
|
|
||||||
- Table editor
|
|
||||||
- Customizable preview styles
|
|
||||||
- Spellcheck integration
|
|
||||||
- Auto-save functionality
|
|
||||||
|
|
||||||
#### `DeesEditorMarkdownoutlet`
|
#### `DeesEditorMarkdownoutlet`
|
||||||
Markdown preview component for rendering markdown content.
|
Markdown preview component for rendering markdown content.
|
||||||
|
|
||||||
@@ -1898,14 +1280,6 @@ Markdown preview component for rendering markdown content.
|
|||||||
></dees-editor-markdownoutlet>
|
></dees-editor-markdownoutlet>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Safe markdown rendering
|
|
||||||
- Multiple themes
|
|
||||||
- Plugin support (mermaid diagrams, syntax highlighting)
|
|
||||||
- XSS protection
|
|
||||||
- Custom CSS injection
|
|
||||||
- Responsive images
|
|
||||||
|
|
||||||
#### `DeesTerminal`
|
#### `DeesTerminal`
|
||||||
Terminal emulator component for command-line interface.
|
Terminal emulator component for command-line interface.
|
||||||
|
|
||||||
@@ -1922,16 +1296,6 @@ Terminal emulator component for command-line interface.
|
|||||||
></dees-terminal>
|
></dees-terminal>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Command history
|
|
||||||
- Custom commands
|
|
||||||
- Auto-completion
|
|
||||||
- Copy/paste support
|
|
||||||
- ANSI color support
|
|
||||||
- Scrollback buffer
|
|
||||||
- Keyboard shortcuts
|
|
||||||
- Command aliases
|
|
||||||
|
|
||||||
#### `DeesUpdater`
|
#### `DeesUpdater`
|
||||||
Component for managing application updates and version control.
|
Component for managing application updates and version control.
|
||||||
|
|
||||||
@@ -1945,112 +1309,6 @@ Component for managing application updates and version control.
|
|||||||
></dees-updater>
|
></dees-updater>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Version checking
|
|
||||||
- Update notifications
|
|
||||||
- Progress tracking
|
|
||||||
- Automatic updates
|
|
||||||
- Rollback support
|
|
||||||
- Update scheduling
|
|
||||||
- Dependency management
|
|
||||||
|
|
||||||
Best Practices:
|
|
||||||
|
|
||||||
1. Code Editor Usage
|
|
||||||
- Configure appropriate language
|
|
||||||
- Set reasonable defaults
|
|
||||||
- Handle content changes
|
|
||||||
- Manage undo/redo stack
|
|
||||||
- Consider performance
|
|
||||||
|
|
||||||
2. Markdown Editing
|
|
||||||
- Provide clear toolbar
|
|
||||||
- Show live preview
|
|
||||||
- Handle image uploads
|
|
||||||
- Support shortcuts
|
|
||||||
- Maintain consistent styling
|
|
||||||
|
|
||||||
3. Terminal Implementation
|
|
||||||
- Clear command documentation
|
|
||||||
- Handle errors gracefully
|
|
||||||
- Provide command history
|
|
||||||
- Support common shortcuts
|
|
||||||
- Implement auto-completion
|
|
||||||
|
|
||||||
4. Update Management
|
|
||||||
- Regular version checks
|
|
||||||
- Clear update messaging
|
|
||||||
- Progress indication
|
|
||||||
- Error recovery
|
|
||||||
- User confirmation
|
|
||||||
|
|
||||||
Common Use Cases:
|
|
||||||
|
|
||||||
1. Code Editor
|
|
||||||
- Configuration editing
|
|
||||||
- Script development
|
|
||||||
- Code snippets
|
|
||||||
- Documentation
|
|
||||||
- Teaching tools
|
|
||||||
|
|
||||||
2. Markdown Editor
|
|
||||||
- Documentation
|
|
||||||
- Content creation
|
|
||||||
- README files
|
|
||||||
- Blog posts
|
|
||||||
- Release notes
|
|
||||||
|
|
||||||
3. Terminal
|
|
||||||
- Command execution
|
|
||||||
- System monitoring
|
|
||||||
- Development tools
|
|
||||||
- Debugging
|
|
||||||
- Training environments
|
|
||||||
|
|
||||||
4. Updater
|
|
||||||
- Application updates
|
|
||||||
- Plugin management
|
|
||||||
- Feature deployment
|
|
||||||
- Security patches
|
|
||||||
- Configuration updates
|
|
||||||
|
|
||||||
Integration Examples:
|
|
||||||
|
|
||||||
```typescript
|
|
||||||
// Combining components for a development environment
|
|
||||||
<dees-form>
|
|
||||||
<dees-editor
|
|
||||||
.language=${'javascript'}
|
|
||||||
.value=${code}
|
|
||||||
@change=${updatePreview}
|
|
||||||
></dees-editor>
|
|
||||||
|
|
||||||
<dees-terminal
|
|
||||||
.commands=${devCommands}
|
|
||||||
@command=${executeCommand}
|
|
||||||
></dees-terminal>
|
|
||||||
|
|
||||||
<dees-updater
|
|
||||||
.currentVersion=${appVersion}
|
|
||||||
@update-available=${notifyUser}
|
|
||||||
></dees-updater>
|
|
||||||
</dees-form>
|
|
||||||
```
|
|
||||||
|
|
||||||
Performance Considerations:
|
|
||||||
- Lazy loading of heavy components
|
|
||||||
- Memory management
|
|
||||||
- Resource cleanup
|
|
||||||
- Event handling optimization
|
|
||||||
- Efficient updates
|
|
||||||
|
|
||||||
Accessibility Features:
|
|
||||||
- Keyboard navigation
|
|
||||||
- Screen reader support
|
|
||||||
- High contrast themes
|
|
||||||
- Focus management
|
|
||||||
- ARIA attributes
|
|
||||||
|
|
||||||
### Auth & Utilities Components
|
### Auth & Utilities Components
|
||||||
|
|
||||||
#### `DeesSimpleAppdash`
|
#### `DeesSimpleAppdash`
|
||||||
@@ -2073,13 +1331,6 @@ Simple application dashboard component for quick prototyping.
|
|||||||
</dees-simple-appdash>
|
</dees-simple-appdash>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Quick setup dashboard layout
|
|
||||||
- Built-in navigation
|
|
||||||
- User profile section
|
|
||||||
- Responsive design
|
|
||||||
- Minimal configuration
|
|
||||||
|
|
||||||
#### `DeesSimpleLogin`
|
#### `DeesSimpleLogin`
|
||||||
Simple login form component with validation and customization.
|
Simple login form component with validation and customization.
|
||||||
|
|
||||||
@@ -2096,15 +1347,6 @@ Simple login form component with validation and customization.
|
|||||||
></dees-simple-login>
|
></dees-simple-login>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Customizable fields
|
|
||||||
- Built-in validation
|
|
||||||
- Remember me option
|
|
||||||
- Forgot password link
|
|
||||||
- Custom branding
|
|
||||||
- Responsive layout
|
|
||||||
- Loading states
|
|
||||||
|
|
||||||
### Shopping Components
|
### Shopping Components
|
||||||
|
|
||||||
#### `DeesShoppingProductcard`
|
#### `DeesShoppingProductcard`
|
||||||
@@ -2133,41 +1375,6 @@ Product card component for e-commerce applications.
|
|||||||
></dees-shopping-productcard>
|
></dees-shopping-productcard>
|
||||||
```
|
```
|
||||||
|
|
||||||
Key Features:
|
|
||||||
- Product image with fallback icon
|
|
||||||
- Category label
|
|
||||||
- Product name and description
|
|
||||||
- Price display with original price strikethrough
|
|
||||||
- Stock status indicator
|
|
||||||
- Built-in quantity selector
|
|
||||||
- Selection mode for bulk operations
|
|
||||||
- Hover effects
|
|
||||||
- Responsive design
|
|
||||||
- Theme-aware styling
|
|
||||||
|
|
||||||
Product Data Interface:
|
|
||||||
```typescript
|
|
||||||
interface IProductData {
|
|
||||||
name: string;
|
|
||||||
category?: string;
|
|
||||||
description?: string;
|
|
||||||
price: number;
|
|
||||||
originalPrice?: number;
|
|
||||||
currency?: string;
|
|
||||||
inStock?: boolean;
|
|
||||||
stockText?: string;
|
|
||||||
imageUrl?: string;
|
|
||||||
iconName?: string;
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Common Use Cases:
|
|
||||||
- Product listings
|
|
||||||
- Shopping carts
|
|
||||||
- Order summaries
|
|
||||||
- Product comparisons
|
|
||||||
- Wishlist displays
|
|
||||||
|
|
||||||
## License and Legal Information
|
## License and Legal Information
|
||||||
|
|
||||||
This repository contains open-source code that is licensed under the MIT License. A copy of the MIT License can be found in the license file within this repository.
|
This repository contains open-source code that is licensed under the MIT License. A copy of the MIT License can be found in the license file within this repository.
|
||||||
|
|||||||
BIN
readme.plan.md
BIN
readme.plan.md
Binary file not shown.
44
scripts/update-monaco-version.cjs
Executable file
44
scripts/update-monaco-version.cjs
Executable file
@@ -0,0 +1,44 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const projectRoot = path.resolve(__dirname, '..');
|
||||||
|
|
||||||
|
function resolveMonacoPackageJson() {
|
||||||
|
try {
|
||||||
|
const resolvedPath = require.resolve('monaco-editor/package.json', {
|
||||||
|
paths: [projectRoot],
|
||||||
|
});
|
||||||
|
return resolvedPath;
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[dees-editor] Unable to resolve monaco-editor/package.json');
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getMonacoVersion() {
|
||||||
|
const monacoPackagePath = resolveMonacoPackageJson();
|
||||||
|
const monacoPackage = require(monacoPackagePath);
|
||||||
|
if (!monacoPackage.version) {
|
||||||
|
throw new Error('[dees-editor] monaco-editor/package.json does not expose a version field');
|
||||||
|
}
|
||||||
|
return monacoPackage.version;
|
||||||
|
}
|
||||||
|
|
||||||
|
function writeVersionModule(version) {
|
||||||
|
const targetDir = path.join(projectRoot, 'ts_web', 'elements', 'dees-editor');
|
||||||
|
fs.mkdirSync(targetDir, { recursive: true });
|
||||||
|
const targetFile = path.join(targetDir, 'version.ts');
|
||||||
|
const fileContent = `// Auto-generated by scripts/update-monaco-version.cjs\nexport const MONACO_VERSION = '${version}';\n`;
|
||||||
|
fs.writeFileSync(targetFile, fileContent, 'utf8');
|
||||||
|
console.log(`[dees-editor] Wrote ${path.relative(projectRoot, targetFile)} with monaco-editor@${version}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const version = getMonacoVersion();
|
||||||
|
writeVersionModule(version);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[dees-editor] Failed to update Monaco version module.');
|
||||||
|
console.error(error instanceof Error ? error.message : error);
|
||||||
|
process.exitCode = 1;
|
||||||
|
}
|
||||||
28
test/test.dashboardgrid-layout.node.ts
Normal file
28
test/test.dashboardgrid-layout.node.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import { tap, expect } from '@push.rocks/tapbundle';
|
||||||
|
|
||||||
|
import {
|
||||||
|
resolveWidgetPlacement,
|
||||||
|
collectCollisions,
|
||||||
|
} from '../ts_web/elements/dees-dashboardgrid/layout.ts';
|
||||||
|
import type { DashboardWidget } from '../ts_web/elements/dees-dashboardgrid/types.ts';
|
||||||
|
|
||||||
|
tap.test('dashboardgrid does not overlap widgets after swap attempt', async () => {
|
||||||
|
const widgets: DashboardWidget[] = [
|
||||||
|
{ id: 'w0', x: 6, y: 5, w: 1, h: 3 },
|
||||||
|
{ id: 'w1', x: 6, y: 1, w: 1, h: 3 },
|
||||||
|
{ id: 'w2', x: 3, y: 0, w: 2, h: 2 },
|
||||||
|
{ id: 'w3', x: 9, y: 0, w: 1, h: 2 },
|
||||||
|
{ id: 'w4', x: 4, y: 3, w: 1, h: 2 },
|
||||||
|
];
|
||||||
|
|
||||||
|
const placement = resolveWidgetPlacement(widgets, 'w0', { x: 6, y: 3 }, 12);
|
||||||
|
expect(placement).toBeTruthy();
|
||||||
|
|
||||||
|
const layout = placement!.widgets;
|
||||||
|
for (const widget of layout) {
|
||||||
|
const collisions = collectCollisions(layout, widget, widget.x, widget.y, widget.w, widget.h);
|
||||||
|
expect(collisions).toBeEmptyArray();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export default tap.start();
|
||||||
@@ -3,6 +3,6 @@
|
|||||||
*/
|
*/
|
||||||
export const commitinfo = {
|
export const commitinfo = {
|
||||||
name: '@design.estate/dees-catalog',
|
name: '@design.estate/dees-catalog',
|
||||||
version: '1.10.12',
|
version: '3.1.2',
|
||||||
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.'
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../../00plugins.js';
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
type TemplateResult,
|
type TemplateResult,
|
||||||
@@ -10,8 +10,8 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||||
import './dees-icon.js';
|
import '../../dees-icon/dees-icon.js';
|
||||||
|
|
||||||
@customElement('dees-appui-activitylog')
|
@customElement('dees-appui-activitylog')
|
||||||
export class DeesAppuiActivitylog extends DeesElement {
|
export class DeesAppuiActivitylog extends DeesElement {
|
||||||
@@ -63,13 +63,14 @@ export class DeesAppuiActivitylog extends DeesElement {
|
|||||||
.topbar {
|
.topbar {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0px;
|
top: 0px;
|
||||||
height: 40px;
|
height: 48px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 0px 16px;
|
padding: 0px 16px;
|
||||||
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
|
background: ${cssManager.bdTheme('#ffffff', '#09090b')};
|
||||||
border-bottom: 1px solid ${cssManager.bdTheme('#e5e7eb', '#27272a')};
|
border-bottom: 1px solid ${cssManager.bdTheme('#e5e7eb', '#27272a')};
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
.topbar .heading {
|
.topbar .heading {
|
||||||
@@ -81,7 +82,7 @@ export class DeesAppuiActivitylog extends DeesElement {
|
|||||||
|
|
||||||
.activityContainer {
|
.activityContainer {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 40px;
|
top: 48px;
|
||||||
bottom: 48px;
|
bottom: 48px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 12px 0px;
|
padding: 12px 0px;
|
||||||
@@ -315,7 +316,7 @@ export class DeesAppuiActivitylog extends DeesElement {
|
|||||||
position: absolute;
|
position: absolute;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
top: 40px;
|
top: 48px;
|
||||||
background: ${cssManager.bdTheme(
|
background: ${cssManager.bdTheme(
|
||||||
'linear-gradient(0deg, transparent 0%, #fafafa 100%)',
|
'linear-gradient(0deg, transparent 0%, #fafafa 100%)',
|
||||||
'linear-gradient(0deg, transparent 0%, #0a0a0a 100%)'
|
'linear-gradient(0deg, transparent 0%, #0a0a0a 100%)'
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-activitylog.js';
|
||||||
@@ -5,19 +5,19 @@ import {
|
|||||||
property,
|
property,
|
||||||
state,
|
state,
|
||||||
html,
|
html,
|
||||||
css,
|
|
||||||
cssManager,
|
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../../00plugins.js';
|
||||||
import { demoFunc } from './dees-appui-appbar.demo.js';
|
import { demoFunc } from './demo.js';
|
||||||
|
import { appuiAppbarStyles } from './styles.js';
|
||||||
|
import { renderAppuiAppbar } from './template.js';
|
||||||
|
|
||||||
// Import required components
|
// Import required components
|
||||||
import './dees-icon.js';
|
import '../../dees-icon/dees-icon.js';
|
||||||
import './dees-windowcontrols.js';
|
import '../../dees-windowcontrols/dees-windowcontrols.js';
|
||||||
import './dees-appui-profiledropdown.js';
|
import '../dees-appui-profiledropdown/dees-appui-profiledropdown.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -31,301 +31,58 @@ export class DeesAppuiBar extends DeesElement {
|
|||||||
|
|
||||||
// INSTANCE PROPERTIES
|
// INSTANCE PROPERTIES
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public menuItems: interfaces.IAppBarMenuItem[] = [];
|
accessor menuItems: interfaces.IAppBarMenuItem[] = [];
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public breadcrumbs: string = '';
|
accessor breadcrumbs: string = '';
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public breadcrumbSeparator: string = ' > ';
|
accessor breadcrumbSeparator: string = ' > ';
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public showWindowControls: boolean = true;
|
accessor showWindowControls: boolean = true;
|
||||||
|
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public user?: {
|
accessor user: {
|
||||||
name: string;
|
name: string;
|
||||||
email?: string;
|
email?: string;
|
||||||
avatar?: string;
|
avatar?: string;
|
||||||
status?: 'online' | 'offline' | 'busy' | 'away';
|
status?: 'online' | 'offline' | 'busy' | 'away';
|
||||||
};
|
} | undefined = undefined;
|
||||||
|
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public profileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
accessor profileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public showSearch: boolean = false;
|
accessor showSearch: boolean = false;
|
||||||
|
|
||||||
// STATE
|
// STATE
|
||||||
@state()
|
@state()
|
||||||
private activeMenu: string | null = null;
|
accessor activeMenu: string | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private openDropdowns: Set<string> = new Set();
|
accessor openDropdowns: Set<string> = new Set();
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private focusedItem: string | null = null;
|
accessor focusedItem: string | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private focusedDropdownItem: number = -1;
|
accessor focusedDropdownItem: number = -1;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private isProfileDropdownOpen: boolean = false;
|
accessor isProfileDropdownOpen: boolean = false;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = appuiAppbarStyles;
|
||||||
cssManager.defaultStyles,
|
|
||||||
css`
|
|
||||||
:host {
|
|
||||||
/* CSS Variables for theming */
|
|
||||||
--appbar-height: 40px;
|
|
||||||
--appbar-font-size: 12px;
|
|
||||||
|
|
||||||
display: block;
|
|
||||||
position: relative;
|
|
||||||
width: 100%;
|
|
||||||
height: var(--appbar-height);
|
|
||||||
border-bottom: 1px solid ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
|
||||||
background: ${cssManager.bdTheme('#ffffff', '#000000')};
|
|
||||||
color: ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
|
||||||
font-size: var(--appbar-font-size);
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: ${cssManager.cssGridColumns(3, 20)};
|
|
||||||
-webkit-app-region: drag;
|
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.menus {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
padding: 0 8px;
|
|
||||||
cursor: default;
|
|
||||||
}
|
|
||||||
|
|
||||||
.menuItem {
|
|
||||||
position: relative;
|
|
||||||
line-height: 24px;
|
|
||||||
padding: 0px 12px;
|
|
||||||
margin: 8px 0px;
|
|
||||||
border-radius: 4px;
|
|
||||||
-webkit-app-region: no-drag;
|
|
||||||
transition: all 0.2s ease;
|
|
||||||
cursor: default;
|
|
||||||
outline: none;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Optional: Style for menu items with icons (not typically used for top-level items) */
|
|
||||||
.menuItem dees-icon {
|
|
||||||
font-size: 14px;
|
|
||||||
opacity: 0.8;
|
|
||||||
}
|
|
||||||
|
|
||||||
.menuItem:hover {
|
|
||||||
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
|
||||||
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.menuItem.active {
|
|
||||||
background: ${cssManager.bdTheme('#00000020', '#ffffff30')};
|
|
||||||
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.menuItem[disabled] {
|
|
||||||
opacity: 0.5;
|
|
||||||
cursor: not-allowed;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.menuItem:focus-visible {
|
|
||||||
box-shadow: 0 0 0 2px ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/* Dropdown styles */
|
|
||||||
.dropdown {
|
|
||||||
position: absolute;
|
|
||||||
top: 100%;
|
|
||||||
left: 0;
|
|
||||||
min-width: 200px;
|
|
||||||
background: ${cssManager.bdTheme('#ffffff', '#000000')};
|
|
||||||
border: 1px solid ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
|
||||||
border-radius: 4px;
|
|
||||||
box-shadow: ${cssManager.bdTheme('0 4px 12px rgba(0, 0, 0, 0.15)', '0 4px 12px rgba(0, 0, 0, 0.3)')};
|
|
||||||
margin-top: 4px;
|
|
||||||
z-index: 1000;
|
|
||||||
opacity: 0;
|
|
||||||
transform: translateY(-10px);
|
|
||||||
transition: opacity 0.2s, transform 0.2s;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown.open {
|
|
||||||
opacity: 1;
|
|
||||||
transform: translateY(0);
|
|
||||||
pointer-events: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown-item {
|
|
||||||
padding: 8px 16px;
|
|
||||||
cursor: default;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
transition: background 0.1s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown-item:hover,
|
|
||||||
.dropdown-item.focused {
|
|
||||||
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown-divider {
|
|
||||||
height: 1px;
|
|
||||||
background: ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
|
||||||
margin: 4px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown-item[disabled] {
|
|
||||||
opacity: 0.5;
|
|
||||||
cursor: not-allowed;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dropdown-item .shortcut {
|
|
||||||
margin-left: auto;
|
|
||||||
opacity: 0.6;
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Breadcrumbs */
|
|
||||||
.breadcrumbs {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
height: 100%;
|
|
||||||
padding: 0 16px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.breadcrumb-item {
|
|
||||||
color: ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
|
||||||
cursor: default;
|
|
||||||
transition: color 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.breadcrumb-item:hover {
|
|
||||||
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.breadcrumb-separator {
|
|
||||||
margin: 0 8px;
|
|
||||||
opacity: 0.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Account section */
|
|
||||||
.account {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: flex-end;
|
|
||||||
padding: 0 16px;
|
|
||||||
gap: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-icon {
|
|
||||||
cursor: default;
|
|
||||||
opacity: 0.7;
|
|
||||||
transition: opacity 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.search-icon:hover {
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-info {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
cursor: default;
|
|
||||||
padding: 4px 8px;
|
|
||||||
border-radius: 4px;
|
|
||||||
transition: background 0.2s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-info:hover {
|
|
||||||
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-avatar {
|
|
||||||
position: relative;
|
|
||||||
width: 24px;
|
|
||||||
height: 24px;
|
|
||||||
border-radius: 50%;
|
|
||||||
background: ${cssManager.bdTheme('#00000020', '#ffffff30')};
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
font-size: 10px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-avatar img {
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
border-radius: 50%;
|
|
||||||
object-fit: cover;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-status {
|
|
||||||
position: absolute;
|
|
||||||
bottom: -2px;
|
|
||||||
right: -2px;
|
|
||||||
width: 8px;
|
|
||||||
height: 8px;
|
|
||||||
border-radius: 50%;
|
|
||||||
border: 2px solid ${cssManager.bdTheme('#ffffff', '#000000')};
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-status.online {
|
|
||||||
background: #4caf50;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-status.offline {
|
|
||||||
background: #757575;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-status.busy {
|
|
||||||
background: #f44336;
|
|
||||||
}
|
|
||||||
|
|
||||||
.user-status.away {
|
|
||||||
background: #ff9800;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
];
|
|
||||||
|
|
||||||
// INSTANCE
|
// INSTANCE
|
||||||
public render(): TemplateResult {
|
public render(): TemplateResult {
|
||||||
return html`
|
return renderAppuiAppbar(this);
|
||||||
<div class="menus">
|
|
||||||
${this.showWindowControls ? html`<dees-windowcontrols></dees-windowcontrols>` : ''}
|
|
||||||
${this.renderMenuItems()}
|
|
||||||
</div>
|
|
||||||
<div class="breadcrumbs">
|
|
||||||
${this.renderBreadcrumbs()}
|
|
||||||
</div>
|
|
||||||
<div class="account">
|
|
||||||
${this.renderAccountSection()}
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private renderMenuItems(): TemplateResult {
|
|
||||||
|
|
||||||
|
public renderMenuItems(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
${this.menuItems.map((item, index) => this.renderMenuItem(item, `menu-${index}`))}
|
${this.menuItems.map((item, index) => this.renderMenuItem(item, `menu-${index}`))}
|
||||||
`;
|
`;
|
||||||
@@ -398,7 +155,7 @@ export class DeesAppuiBar extends DeesElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private renderBreadcrumbs(): TemplateResult {
|
public renderBreadcrumbs(): TemplateResult {
|
||||||
if (!this.breadcrumbs) {
|
if (!this.breadcrumbs) {
|
||||||
return html``;
|
return html``;
|
||||||
}
|
}
|
||||||
@@ -417,7 +174,7 @@ export class DeesAppuiBar extends DeesElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private renderAccountSection(): TemplateResult {
|
public renderAccountSection(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
${this.showSearch ? html`
|
${this.showSearch ? html`
|
||||||
<dees-icon
|
<dees-icon
|
||||||
@@ -1,7 +1,8 @@
|
|||||||
import { html, css } from '@design.estate/dees-element';
|
import { html, css } from '@design.estate/dees-element';
|
||||||
import type { DeesAppuiBar } from './dees-appui-appbar.js';
|
import type { DeesAppuiBar } from './component.js';
|
||||||
import type { IAppBarMenuItem } from './interfaces/appbarmenuitem.js';
|
import type { IAppBarMenuItem } from '../../interfaces/appbarmenuitem.js';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
import './component.js';
|
||||||
|
|
||||||
export const demoFunc = () => {
|
export const demoFunc = () => {
|
||||||
// Sample menu items with various configurations
|
// Sample menu items with various configurations
|
||||||
1
ts_web/elements/00group-appui/dees-appui-appbar/index.ts
Normal file
1
ts_web/elements/00group-appui/dees-appui-appbar/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './component.js';
|
||||||
238
ts_web/elements/00group-appui/dees-appui-appbar/styles.ts
Normal file
238
ts_web/elements/00group-appui/dees-appui-appbar/styles.ts
Normal file
@@ -0,0 +1,238 @@
|
|||||||
|
import { css, cssManager } from '@design.estate/dees-element';
|
||||||
|
|
||||||
|
export const appuiAppbarStyles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
/* CSS Variables for theming */
|
||||||
|
--appbar-height: 40px;
|
||||||
|
--appbar-font-size: 12px;
|
||||||
|
|
||||||
|
display: block;
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: var(--appbar-height);
|
||||||
|
border-bottom: 1px solid ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
||||||
|
background: ${cssManager.bdTheme('#ffffff', '#000000')};
|
||||||
|
color: ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
||||||
|
font-size: var(--appbar-font-size);
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: ${cssManager.cssGridColumns(3, 20)};
|
||||||
|
-webkit-app-region: drag;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menus {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 0 8px;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem {
|
||||||
|
position: relative;
|
||||||
|
line-height: 24px;
|
||||||
|
padding: 0px 12px;
|
||||||
|
margin: 8px 0px;
|
||||||
|
border-radius: 4px;
|
||||||
|
-webkit-app-region: no-drag;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
cursor: default;
|
||||||
|
outline: none;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Optional: Style for menu items with icons (not typically used for top-level items) */
|
||||||
|
.menuItem dees-icon {
|
||||||
|
font-size: 14px;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem:hover {
|
||||||
|
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
||||||
|
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem.active {
|
||||||
|
background: ${cssManager.bdTheme('#00000020', '#ffffff30')};
|
||||||
|
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem[disabled] {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem:focus-visible {
|
||||||
|
box-shadow: 0 0 0 2px ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Dropdown styles */
|
||||||
|
.dropdown {
|
||||||
|
position: absolute;
|
||||||
|
top: 100%;
|
||||||
|
left: 0;
|
||||||
|
min-width: 200px;
|
||||||
|
background: ${cssManager.bdTheme('#ffffff', '#000000')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: ${cssManager.bdTheme('0 4px 12px rgba(0, 0, 0, 0.15)', '0 4px 12px rgba(0, 0, 0, 0.3)')};
|
||||||
|
margin-top: 4px;
|
||||||
|
z-index: 1000;
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(-10px);
|
||||||
|
transition: opacity 0.2s, transform 0.2s;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown.open {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0);
|
||||||
|
pointer-events: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item {
|
||||||
|
padding: 8px 16px;
|
||||||
|
cursor: default;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
transition: background 0.1s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item:hover,
|
||||||
|
.dropdown-item.focused {
|
||||||
|
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-divider {
|
||||||
|
height: 1px;
|
||||||
|
background: ${cssManager.bdTheme('#e0e0e0', '#202020')};
|
||||||
|
margin: 4px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item[disabled] {
|
||||||
|
opacity: 0.5;
|
||||||
|
cursor: not-allowed;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-item .shortcut {
|
||||||
|
margin-left: auto;
|
||||||
|
opacity: 0.6;
|
||||||
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Breadcrumbs */
|
||||||
|
.breadcrumbs {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
height: 100%;
|
||||||
|
padding: 0 16px;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.breadcrumb-item {
|
||||||
|
color: ${cssManager.bdTheme('#00000080', '#ffffff80')};
|
||||||
|
cursor: default;
|
||||||
|
transition: color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.breadcrumb-item:hover {
|
||||||
|
color: ${cssManager.bdTheme('#000000', '#ffffff')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.breadcrumb-separator {
|
||||||
|
margin: 0 8px;
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Account section */
|
||||||
|
.account {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
padding: 0 16px;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-icon {
|
||||||
|
cursor: default;
|
||||||
|
opacity: 0.7;
|
||||||
|
transition: opacity 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-icon:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-info {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
cursor: default;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-info:hover {
|
||||||
|
background: ${cssManager.bdTheme('#00000010', '#ffffff20')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-avatar {
|
||||||
|
position: relative;
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: ${cssManager.bdTheme('#00000020', '#ffffff30')};
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-avatar img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border-radius: 50%;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-status {
|
||||||
|
position: absolute;
|
||||||
|
bottom: -2px;
|
||||||
|
right: -2px;
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 2px solid ${cssManager.bdTheme('#ffffff', '#000000')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-status.online {
|
||||||
|
background: #4caf50;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-status.offline {
|
||||||
|
background: #757575;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-status.busy {
|
||||||
|
background: #f44336;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-status.away {
|
||||||
|
background: #ff9800;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
|
||||||
18
ts_web/elements/00group-appui/dees-appui-appbar/template.ts
Normal file
18
ts_web/elements/00group-appui/dees-appui-appbar/template.ts
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import { html, type TemplateResult } from '@design.estate/dees-element';
|
||||||
|
import type { DeesAppuiBar } from './component.js';
|
||||||
|
|
||||||
|
export const renderAppuiAppbar = (component: DeesAppuiBar): TemplateResult => {
|
||||||
|
return html`
|
||||||
|
<div class="menus">
|
||||||
|
${component.showWindowControls ? html`<dees-windowcontrols></dees-windowcontrols>` : ''}
|
||||||
|
${component.renderMenuItems()}
|
||||||
|
</div>
|
||||||
|
<div class="breadcrumbs">
|
||||||
|
${component.renderBreadcrumbs()}
|
||||||
|
</div>
|
||||||
|
<div class="account">
|
||||||
|
${component.renderAccountSection()}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
};
|
||||||
@@ -0,0 +1,238 @@
|
|||||||
|
import { html, css } from '@design.estate/dees-element';
|
||||||
|
import type { DeesAppuiBase } from '../dees-appui-base/dees-appui-base.js';
|
||||||
|
import type { IAppBarMenuItem } from '../../interfaces/appbarmenuitem.js';
|
||||||
|
import type { ITab } from '../../interfaces/tab.js';
|
||||||
|
import type { ISelectionOption } from '../../interfaces/selectionoption.js';
|
||||||
|
import type { IMenuGroup } from '../../interfaces/menugroup.js';
|
||||||
|
import type { ISecondaryMenuGroup } from '../../interfaces/secondarymenu.js';
|
||||||
|
import * as plugins from '../../00plugins.js';
|
||||||
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
|
||||||
|
export const demoFunc = () => {
|
||||||
|
// Menu items for the appbar
|
||||||
|
const menuItems: IAppBarMenuItem[] = [
|
||||||
|
{
|
||||||
|
name: 'File',
|
||||||
|
action: async () => {},
|
||||||
|
submenu: [
|
||||||
|
{ name: 'New Project', shortcut: 'Cmd+N', iconName: 'filePlus', action: async () => console.log('New project') },
|
||||||
|
{ name: 'Open Project...', shortcut: 'Cmd+O', iconName: 'folderOpen', action: async () => console.log('Open project') },
|
||||||
|
{ name: 'Recent Projects', action: async () => {}, submenu: [
|
||||||
|
{ name: 'my-app', action: async () => console.log('Open my-app') },
|
||||||
|
{ name: 'component-lib', action: async () => console.log('Open component-lib') },
|
||||||
|
{ name: 'api-server', action: async () => console.log('Open api-server') },
|
||||||
|
]},
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Save All', shortcut: 'Cmd+Shift+S', iconName: 'save', action: async () => console.log('Save all') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Close Project', action: async () => console.log('Close project') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Edit',
|
||||||
|
action: async () => {},
|
||||||
|
submenu: [
|
||||||
|
{ name: 'Undo', shortcut: 'Cmd+Z', iconName: 'undo', action: async () => console.log('Undo') },
|
||||||
|
{ name: 'Redo', shortcut: 'Cmd+Shift+Z', iconName: 'redo', action: async () => console.log('Redo') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Cut', shortcut: 'Cmd+X', iconName: 'scissors', action: async () => console.log('Cut') },
|
||||||
|
{ name: 'Copy', shortcut: 'Cmd+C', iconName: 'copy', action: async () => console.log('Copy') },
|
||||||
|
{ name: 'Paste', shortcut: 'Cmd+V', iconName: 'clipboard', action: async () => console.log('Paste') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'View',
|
||||||
|
action: async () => {},
|
||||||
|
submenu: [
|
||||||
|
{ name: 'Toggle Sidebar', shortcut: 'Cmd+B', action: async () => console.log('Toggle sidebar') },
|
||||||
|
{ name: 'Toggle Terminal', shortcut: 'Cmd+J', iconName: 'terminal', action: async () => console.log('Toggle terminal') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Zoom In', shortcut: 'Cmd++', iconName: 'zoomIn', action: async () => console.log('Zoom in') },
|
||||||
|
{ name: 'Zoom Out', shortcut: 'Cmd+-', iconName: 'zoomOut', action: async () => console.log('Zoom out') },
|
||||||
|
{ name: 'Reset Zoom', shortcut: 'Cmd+0', action: async () => console.log('Reset zoom') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Help',
|
||||||
|
action: async () => {},
|
||||||
|
submenu: [
|
||||||
|
{ name: 'Documentation', iconName: 'book', action: async () => console.log('Documentation') },
|
||||||
|
{ name: 'Release Notes', iconName: 'fileText', action: async () => console.log('Release notes') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Report Issue', iconName: 'bug', action: async () => console.log('Report issue') },
|
||||||
|
{ name: 'About', iconName: 'info', action: async () => console.log('About') },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
// Main menu groups (left sidebar)
|
||||||
|
const mainMenuGroups: IMenuGroup[] = [
|
||||||
|
{
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Dashboard', iconName: 'lucide:home', action: () => console.log('Dashboard selected') },
|
||||||
|
{ key: 'Inbox', iconName: 'lucide:inbox', action: () => console.log('Inbox selected') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Workspace',
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Projects', iconName: 'lucide:folder', action: () => console.log('Projects selected') },
|
||||||
|
{ key: 'Tasks', iconName: 'lucide:checkSquare', action: () => console.log('Tasks selected') },
|
||||||
|
{ key: 'Documents', iconName: 'lucide:fileText', action: () => console.log('Documents selected') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Analytics',
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Reports', iconName: 'lucide:barChart3', action: () => console.log('Reports selected') },
|
||||||
|
{ key: 'Insights', iconName: 'lucide:lightbulb', action: () => console.log('Insights selected') },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
// Main menu bottom tabs (pinned to bottom)
|
||||||
|
const mainMenuBottomTabs: ITab[] = [
|
||||||
|
{ key: 'Settings', iconName: 'lucide:settings', action: () => console.log('Settings selected') },
|
||||||
|
{ key: 'Help', iconName: 'lucide:helpCircle', action: () => console.log('Help selected') },
|
||||||
|
];
|
||||||
|
|
||||||
|
// Secondary menu groups (second sidebar with collapsible groups)
|
||||||
|
// These showcase the new shadcn-style design with badges and collapsible sections
|
||||||
|
const secondaryMenuGroups: ISecondaryMenuGroup[] = [
|
||||||
|
{
|
||||||
|
name: 'Quick Access',
|
||||||
|
iconName: 'lucide:zap',
|
||||||
|
items: [
|
||||||
|
{ key: 'Overview', iconName: 'layoutDashboard', action: () => console.log('Overview selected') },
|
||||||
|
{ key: 'Recent Activity', iconName: 'clock', action: () => console.log('Recent Activity selected'), badge: 5 },
|
||||||
|
{ key: 'Favorites', iconName: 'star', action: () => console.log('Favorites selected') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Resources',
|
||||||
|
iconName: 'lucide:layers',
|
||||||
|
items: [
|
||||||
|
{ key: 'Components', iconName: 'package', action: () => console.log('Components selected'), badge: 24 },
|
||||||
|
{ key: 'Services', iconName: 'server', action: () => console.log('Services selected'), badge: 'new', badgeVariant: 'success' },
|
||||||
|
{ key: 'APIs', iconName: 'globe', action: () => console.log('APIs selected'), badge: 3, badgeVariant: 'warning' },
|
||||||
|
{ key: 'Webhooks', iconName: 'webhook', action: () => console.log('Webhooks selected') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Data Management',
|
||||||
|
iconName: 'lucide:database',
|
||||||
|
items: [
|
||||||
|
{ key: 'Database', iconName: 'database', action: () => console.log('Database selected') },
|
||||||
|
{ key: 'Storage', iconName: 'hardDrive', action: () => console.log('Storage selected'), badge: '85%', badgeVariant: 'warning' },
|
||||||
|
{ key: 'Backups', iconName: 'archive', action: () => console.log('Backups selected'), badge: 'OK', badgeVariant: 'success' },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'System',
|
||||||
|
iconName: 'lucide:settings',
|
||||||
|
collapsed: true,
|
||||||
|
items: [
|
||||||
|
{ key: 'Configuration', iconName: 'sliders', action: () => console.log('Configuration selected') },
|
||||||
|
{ key: 'Integrations', iconName: 'plug', action: () => console.log('Integrations selected'), badge: 2, badgeVariant: 'error' },
|
||||||
|
{ key: 'Permissions', iconName: 'shield', action: () => console.log('Permissions selected') },
|
||||||
|
{ key: 'Logs', iconName: 'fileText', action: () => console.log('Logs selected') },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
// Main content tabs
|
||||||
|
const mainContentTabs: ITab[] = [
|
||||||
|
{ key: 'Details', iconName: 'lucide:file', action: () => console.log('Details tab') },
|
||||||
|
{ key: 'Logs', iconName: 'lucide:list', action: () => console.log('Logs tab') },
|
||||||
|
{ key: 'Metrics', iconName: 'lucide:lineChart', action: () => console.log('Metrics tab') },
|
||||||
|
];
|
||||||
|
|
||||||
|
// Profile menu items
|
||||||
|
const profileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [
|
||||||
|
{ name: 'Profile Settings', iconName: 'user', action: async () => console.log('Profile settings') },
|
||||||
|
{ name: 'Account', iconName: 'settings', action: async () => console.log('Account settings') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Help & Support', iconName: 'helpCircle', action: async () => console.log('Help') },
|
||||||
|
{ name: 'Keyboard Shortcuts', iconName: 'keyboard', shortcut: 'Cmd+K', action: async () => console.log('Shortcuts') },
|
||||||
|
{ divider: true },
|
||||||
|
{ name: 'Sign Out', iconName: 'logOut', action: async () => console.log('Sign out') }
|
||||||
|
];
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<dees-demowrapper>
|
||||||
|
<style>
|
||||||
|
${css`
|
||||||
|
.demo-container {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
`}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div class="demo-container">
|
||||||
|
<dees-appui-base
|
||||||
|
.appbarMenuItems=${menuItems}
|
||||||
|
.appbarBreadcrumbs=${'Dashboard'}
|
||||||
|
.appbarUser=${{
|
||||||
|
name: 'Jane Smith',
|
||||||
|
email: 'jane.smith@example.com',
|
||||||
|
status: 'online' as 'online' | 'offline' | 'busy' | 'away'
|
||||||
|
}}
|
||||||
|
.appbarProfileMenuItems=${profileMenuItems}
|
||||||
|
.appbarShowWindowControls=${true}
|
||||||
|
.appbarShowSearch=${true}
|
||||||
|
.mainmenuLogoIcon=${'lucide:box'}
|
||||||
|
.mainmenuLogoText=${'Acme App'}
|
||||||
|
.mainmenuGroups=${mainMenuGroups}
|
||||||
|
.mainmenuBottomTabs=${mainMenuBottomTabs}
|
||||||
|
.secondarymenuHeading=${'Dashboard'}
|
||||||
|
.secondarymenuGroups=${secondaryMenuGroups}
|
||||||
|
.maincontentTabs=${mainContentTabs}
|
||||||
|
@appbar-menu-select=${(e: CustomEvent) => console.log('Menu selected:', e.detail)}
|
||||||
|
@appbar-breadcrumb-navigate=${(e: CustomEvent) => console.log('Breadcrumb:', e.detail)}
|
||||||
|
@appbar-search-click=${() => console.log('Search clicked')}
|
||||||
|
@appbar-user-menu-open=${() => console.log('User menu opened')}
|
||||||
|
@appbar-profile-menu-select=${(e: CustomEvent) => console.log('Profile menu selected:', e.detail)}
|
||||||
|
@mainmenu-tab-select=${(e: CustomEvent) => console.log('Tab selected:', e.detail)}
|
||||||
|
@secondarymenu-item-select=${(e: CustomEvent) => console.log('Item selected:', e.detail)}
|
||||||
|
>
|
||||||
|
<div slot="maincontent" style="padding: 40px; color: #a3a3a3; font-family: 'Geist Sans', 'Inter', -apple-system, sans-serif;">
|
||||||
|
<h1 style="color: #fafafa; font-weight: 600; font-size: 24px; margin-bottom: 8px;">Welcome to Acme App</h1>
|
||||||
|
<p style="color: #737373; margin-bottom: 32px;">This demo showcases the AppUI component system with the new SecondaryMenu.</p>
|
||||||
|
|
||||||
|
<div style="display: grid; grid-template-columns: repeat(2, 1fr); gap: 16px; margin-bottom: 32px;">
|
||||||
|
<div style="background: rgba(255,255,255,0.03); border: 1px solid rgba(255,255,255,0.08); border-radius: 8px; padding: 20px;">
|
||||||
|
<h3 style="color: #fafafa; font-size: 14px; font-weight: 600; margin-bottom: 8px;">SecondaryMenu Features</h3>
|
||||||
|
<ul style="margin: 0; padding-left: 20px; font-size: 13px; line-height: 1.8;">
|
||||||
|
<li>Collapsible groups with smooth animations</li>
|
||||||
|
<li>Badge support (counts, status, variants)</li>
|
||||||
|
<li>Dynamic heading from MainMenu selection</li>
|
||||||
|
<li>shadcn-inspired modern design</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div style="background: rgba(255,255,255,0.03); border: 1px solid rgba(255,255,255,0.08); border-radius: 8px; padding: 20px;">
|
||||||
|
<h3 style="color: #fafafa; font-size: 14px; font-weight: 600; margin-bottom: 8px;">Badge Variants</h3>
|
||||||
|
<div style="display: flex; flex-wrap: wrap; gap: 8px; font-size: 12px;">
|
||||||
|
<span style="background: #27272a; color: #a1a1aa; padding: 2px 8px; border-radius: 9px;">default</span>
|
||||||
|
<span style="background: #14532d; color: #4ade80; padding: 2px 8px; border-radius: 9px;">success</span>
|
||||||
|
<span style="background: #451a03; color: #fbbf24; padding: 2px 8px; border-radius: 9px;">warning</span>
|
||||||
|
<span style="background: #450a0a; color: #f87171; padding: 2px 8px; border-radius: 9px;">error</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p style="font-size: 13px; color: #525252;">
|
||||||
|
Try clicking items in the MainMenu (left) - the SecondaryMenu heading updates automatically.
|
||||||
|
Click group headers in the SecondaryMenu to collapse/expand sections.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</dees-appui-base>
|
||||||
|
</div>
|
||||||
|
</dees-demowrapper>
|
||||||
|
`;
|
||||||
|
};
|
||||||
298
ts_web/elements/00group-appui/dees-appui-base/dees-appui-base.ts
Normal file
298
ts_web/elements/00group-appui/dees-appui-base/dees-appui-base.ts
Normal file
@@ -0,0 +1,298 @@
|
|||||||
|
import {
|
||||||
|
DeesElement,
|
||||||
|
type TemplateResult,
|
||||||
|
property,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
css,
|
||||||
|
cssManager,
|
||||||
|
state,
|
||||||
|
} from '@design.estate/dees-element';
|
||||||
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
import * as plugins from '../../00plugins.js';
|
||||||
|
import type { DeesAppuiBar } from '../dees-appui-appbar/index.js';
|
||||||
|
import type { DeesAppuiMainmenu } from '../dees-appui-mainmenu/dees-appui-mainmenu.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 { DeesAppuiActivitylog } from '../dees-appui-activitylog/dees-appui-activitylog.js';
|
||||||
|
import { demoFunc } from './dees-appui-base.demo.js';
|
||||||
|
|
||||||
|
// Import child components
|
||||||
|
import '../dees-appui-appbar/index.js';
|
||||||
|
import '../dees-appui-mainmenu/dees-appui-mainmenu.js';
|
||||||
|
import '../dees-appui-secondarymenu/dees-appui-secondarymenu.js';
|
||||||
|
import '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||||
|
import '../dees-appui-activitylog/dees-appui-activitylog.js';
|
||||||
|
|
||||||
|
@customElement('dees-appui-base')
|
||||||
|
export class DeesAppuiBase extends DeesElement {
|
||||||
|
public static demo = demoFunc;
|
||||||
|
|
||||||
|
// Properties for appbar
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor appbarMenuItems: interfaces.IAppBarMenuItem[] = [];
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor appbarBreadcrumbs: string = '';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor appbarBreadcrumbSeparator: string = ' > ';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor appbarShowWindowControls: boolean = true;
|
||||||
|
|
||||||
|
|
||||||
|
@property({ type: Object })
|
||||||
|
accessor appbarUser: {
|
||||||
|
name: string;
|
||||||
|
email?: string;
|
||||||
|
avatar?: string;
|
||||||
|
status?: 'online' | 'offline' | 'busy' | 'away';
|
||||||
|
} | undefined = undefined;
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor appbarProfileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor appbarShowSearch: boolean = false;
|
||||||
|
|
||||||
|
// Properties for mainmenu
|
||||||
|
@property({ type: String })
|
||||||
|
accessor mainmenuLogoIcon: string = '';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor mainmenuLogoText: string = '';
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor mainmenuGroups: interfaces.IMenuGroup[] = [];
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor mainmenuBottomTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor mainmenuTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
|
@property({ type: Object })
|
||||||
|
accessor mainmenuSelectedTab: interfaces.ITab | undefined = undefined;
|
||||||
|
|
||||||
|
// Properties for secondarymenu
|
||||||
|
@property({ type: String })
|
||||||
|
accessor secondarymenuHeading: string = 'Menu';
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor secondarymenuGroups: interfaces.ISecondaryMenuGroup[] = [];
|
||||||
|
|
||||||
|
@property({ type: Object })
|
||||||
|
accessor secondarymenuSelectedItem: interfaces.ISecondaryMenuItem | undefined = undefined;
|
||||||
|
|
||||||
|
/** Legacy support for flat options (backward compatibility) */
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor secondarymenuOptions: (interfaces.ISelectionOption | { divider: true })[] = [];
|
||||||
|
|
||||||
|
// Collapse states
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor mainmenuCollapsed: boolean = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor secondarymenuCollapsed: boolean = false;
|
||||||
|
|
||||||
|
// Properties for maincontent
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor maincontentTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
|
// References to child components
|
||||||
|
@state()
|
||||||
|
accessor appbar: DeesAppuiBar | undefined = undefined;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor mainmenu: DeesAppuiMainmenu | undefined = undefined;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor secondarymenu: DeesAppuiSecondarymenu | undefined = undefined;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor maincontent: DeesAppuiMaincontent | undefined = undefined;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor activitylog: DeesAppuiActivitylog | undefined = undefined;
|
||||||
|
|
||||||
|
public static styles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
position: absolute;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
background: ${cssManager.bdTheme('#f0f0f0', '#1a1a1a')};
|
||||||
|
}
|
||||||
|
.maingrid {
|
||||||
|
position: absolute;
|
||||||
|
top: 40px;
|
||||||
|
height: calc(100% - 40px);
|
||||||
|
width: 100%;
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto auto 1fr 240px;
|
||||||
|
grid-template-rows: 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Z-index layering for proper stacking (position: relative required for z-index to work) */
|
||||||
|
.maingrid > dees-appui-mainmenu {
|
||||||
|
position: relative;
|
||||||
|
z-index: 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.maingrid > dees-appui-secondarymenu {
|
||||||
|
position: relative;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.maingrid > dees-appui-maincontent {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.maingrid > dees-appui-activitylog {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
|
||||||
|
// INSTANCE
|
||||||
|
public render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<style></style>
|
||||||
|
<dees-appui-appbar
|
||||||
|
.menuItems=${this.appbarMenuItems}
|
||||||
|
.breadcrumbs=${this.appbarBreadcrumbs}
|
||||||
|
.breadcrumbSeparator=${this.appbarBreadcrumbSeparator}
|
||||||
|
.showWindowControls=${this.appbarShowWindowControls}
|
||||||
|
.user=${this.appbarUser}
|
||||||
|
.profileMenuItems=${this.appbarProfileMenuItems}
|
||||||
|
.showSearch=${this.appbarShowSearch}
|
||||||
|
@menu-select=${(e: CustomEvent) => this.handleAppbarMenuSelect(e)}
|
||||||
|
@breadcrumb-navigate=${(e: CustomEvent) => this.handleAppbarBreadcrumbNavigate(e)}
|
||||||
|
@search-click=${() => this.handleAppbarSearchClick()}
|
||||||
|
@user-menu-open=${() => this.handleAppbarUserMenuOpen()}
|
||||||
|
@profile-menu-select=${(e: CustomEvent) => this.handleAppbarProfileMenuSelect(e)}
|
||||||
|
></dees-appui-appbar>
|
||||||
|
<div class="maingrid">
|
||||||
|
<dees-appui-mainmenu
|
||||||
|
.logoIcon=${this.mainmenuLogoIcon}
|
||||||
|
.logoText=${this.mainmenuLogoText}
|
||||||
|
.menuGroups=${this.mainmenuGroups}
|
||||||
|
.bottomTabs=${this.mainmenuBottomTabs}
|
||||||
|
.tabs=${this.mainmenuTabs}
|
||||||
|
.selectedTab=${this.mainmenuSelectedTab}
|
||||||
|
.collapsed=${this.mainmenuCollapsed}
|
||||||
|
@tab-select=${(e: CustomEvent) => this.handleMainmenuTabSelect(e)}
|
||||||
|
@collapse-change=${(e: CustomEvent) => this.handleMainmenuCollapseChange(e)}
|
||||||
|
></dees-appui-mainmenu>
|
||||||
|
<dees-appui-secondarymenu
|
||||||
|
.heading=${this.secondarymenuHeading}
|
||||||
|
.groups=${this.secondarymenuGroups}
|
||||||
|
.selectionOptions=${this.secondarymenuOptions}
|
||||||
|
.selectedItem=${this.secondarymenuSelectedItem}
|
||||||
|
.collapsed=${this.secondarymenuCollapsed}
|
||||||
|
@item-select=${(e: CustomEvent) => this.handleSecondarymenuItemSelect(e)}
|
||||||
|
@collapse-change=${(e: CustomEvent) => this.handleSecondarymenuCollapseChange(e)}
|
||||||
|
></dees-appui-secondarymenu>
|
||||||
|
<dees-appui-maincontent
|
||||||
|
.tabs=${this.maincontentTabs}
|
||||||
|
>
|
||||||
|
<slot name="maincontent"></slot>
|
||||||
|
</dees-appui-maincontent>
|
||||||
|
<dees-appui-activitylog></dees-appui-activitylog>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async firstUpdated() {
|
||||||
|
// Get references to child components
|
||||||
|
this.appbar = this.shadowRoot.querySelector('dees-appui-appbar');
|
||||||
|
this.mainmenu = this.shadowRoot.querySelector('dees-appui-mainmenu');
|
||||||
|
this.secondarymenu = this.shadowRoot.querySelector('dees-appui-secondarymenu');
|
||||||
|
this.maincontent = this.shadowRoot.querySelector('dees-appui-maincontent');
|
||||||
|
this.activitylog = this.shadowRoot.querySelector('dees-appui-activitylog');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event handlers for appbar
|
||||||
|
private handleAppbarMenuSelect(e: CustomEvent) {
|
||||||
|
this.dispatchEvent(new CustomEvent('appbar-menu-select', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleAppbarBreadcrumbNavigate(e: CustomEvent) {
|
||||||
|
this.dispatchEvent(new CustomEvent('appbar-breadcrumb-navigate', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleAppbarSearchClick() {
|
||||||
|
this.dispatchEvent(new CustomEvent('appbar-search-click', {
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleAppbarUserMenuOpen() {
|
||||||
|
this.dispatchEvent(new CustomEvent('appbar-user-menu-open', {
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleAppbarProfileMenuSelect(e: CustomEvent) {
|
||||||
|
this.dispatchEvent(new CustomEvent('appbar-profile-menu-select', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event handlers for mainmenu
|
||||||
|
private handleMainmenuTabSelect(e: CustomEvent) {
|
||||||
|
this.mainmenuSelectedTab = e.detail.tab;
|
||||||
|
// Update secondary menu heading based on main menu selection
|
||||||
|
this.secondarymenuHeading = e.detail.tab.key;
|
||||||
|
this.dispatchEvent(new CustomEvent('mainmenu-tab-select', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event handlers for secondarymenu
|
||||||
|
private handleSecondarymenuItemSelect(e: CustomEvent) {
|
||||||
|
this.secondarymenuSelectedItem = e.detail.item;
|
||||||
|
this.dispatchEvent(new CustomEvent('secondarymenu-item-select', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event handlers for collapse state changes
|
||||||
|
private handleMainmenuCollapseChange(e: CustomEvent) {
|
||||||
|
this.mainmenuCollapsed = e.detail.collapsed;
|
||||||
|
this.dispatchEvent(new CustomEvent('mainmenu-collapse-change', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleSecondarymenuCollapseChange(e: CustomEvent) {
|
||||||
|
this.secondarymenuCollapsed = e.detail.collapsed;
|
||||||
|
this.dispatchEvent(new CustomEvent('secondarymenu-collapse-change', {
|
||||||
|
detail: e.detail,
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
1
ts_web/elements/00group-appui/dees-appui-base/index.ts
Normal file
1
ts_web/elements/00group-appui/dees-appui-base/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-base.js';
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -11,8 +11,8 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import './dees-appui-tabs.js';
|
import '../dees-appui-tabs/dees-appui-tabs.js';
|
||||||
import type { DeesAppuiTabs } from './dees-appui-tabs.js';
|
import type { DeesAppuiTabs } from '../dees-appui-tabs/dees-appui-tabs.js';
|
||||||
|
|
||||||
@customElement('dees-appui-maincontent')
|
@customElement('dees-appui-maincontent')
|
||||||
export class DeesAppuiMaincontent extends DeesElement {
|
export class DeesAppuiMaincontent extends DeesElement {
|
||||||
@@ -35,12 +35,12 @@ export class DeesAppuiMaincontent extends DeesElement {
|
|||||||
@property({
|
@property({
|
||||||
type: Array,
|
type: Array,
|
||||||
})
|
})
|
||||||
public tabs: interfaces.ITab[] = [
|
accessor tabs: interfaces.ITab[] = [
|
||||||
{ key: '⚠️ Please set tabs', action: () => console.warn('No tabs configured for maincontent') },
|
{ key: '⚠️ Please set tabs', action: () => console.warn('No tabs configured for maincontent') },
|
||||||
];
|
];
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public selectedTab: interfaces.ITab | null = null;
|
accessor selectedTab: interfaces.ITab | null = null;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-maincontent.js';
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
import { html } from '@design.estate/dees-element';
|
||||||
|
|
||||||
|
export const demoFunc = () => html`
|
||||||
|
<style>
|
||||||
|
.demo-mainmenu-container {
|
||||||
|
display: flex;
|
||||||
|
height: 100%;
|
||||||
|
background: #1a1a1a;
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
.demo-mainmenu-container .spacer {
|
||||||
|
flex: 1;
|
||||||
|
background: #0f0f0f;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<div class="demo-mainmenu-container">
|
||||||
|
<dees-appui-mainmenu
|
||||||
|
.logoIcon=${'lucide:box'}
|
||||||
|
.logoText=${'Acme App'}
|
||||||
|
.menuGroups=${[
|
||||||
|
{
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Dashboard', iconName: 'lucide:home', action: () => console.log('Dashboard') },
|
||||||
|
{ key: 'Inbox', iconName: 'lucide:inbox', action: () => console.log('Inbox') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Workspace',
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Projects', iconName: 'lucide:folder', action: () => console.log('Projects') },
|
||||||
|
{ key: 'Tasks', iconName: 'lucide:checkSquare', action: () => console.log('Tasks') },
|
||||||
|
{ key: 'Documents', iconName: 'lucide:fileText', action: () => console.log('Documents') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Analytics',
|
||||||
|
tabs: [
|
||||||
|
{ key: 'Reports', iconName: 'lucide:barChart3', action: () => console.log('Reports') },
|
||||||
|
{ key: 'Insights', iconName: 'lucide:lightbulb', action: () => console.log('Insights') },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]}
|
||||||
|
.bottomTabs=${[
|
||||||
|
{ key: 'Settings', iconName: 'lucide:settings', action: () => console.log('Settings') },
|
||||||
|
{ key: 'Help', iconName: 'lucide:helpCircle', action: () => console.log('Help') },
|
||||||
|
]}
|
||||||
|
></dees-appui-mainmenu>
|
||||||
|
<div class="spacer"></div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
@@ -0,0 +1,460 @@
|
|||||||
|
import * as plugins from '../../00plugins.js';
|
||||||
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
import { zIndexLayers } from '../../00zindex.js';
|
||||||
|
|
||||||
|
import {
|
||||||
|
DeesElement,
|
||||||
|
type TemplateResult,
|
||||||
|
property,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
css,
|
||||||
|
cssManager,
|
||||||
|
} from '@design.estate/dees-element';
|
||||||
|
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
import { demoFunc } from './dees-appui-mainmenu.demo.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* the most left menu
|
||||||
|
* usually used as organization selector
|
||||||
|
*/
|
||||||
|
@customElement('dees-appui-mainmenu')
|
||||||
|
export class DeesAppuiMainmenu extends DeesElement {
|
||||||
|
public static demo = demoFunc;
|
||||||
|
|
||||||
|
// INSTANCE
|
||||||
|
|
||||||
|
// Logo properties
|
||||||
|
@property({ type: String })
|
||||||
|
accessor logoIcon: string = '';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor logoText: string = '';
|
||||||
|
|
||||||
|
// Menu groups (new way)
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor menuGroups: interfaces.IMenuGroup[] = [];
|
||||||
|
|
||||||
|
// Bottom tabs (pinned to bottom)
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor bottomTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
|
// Legacy tabs property (for backward compatibility)
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor tabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
|
@property()
|
||||||
|
accessor selectedTab: interfaces.ITab;
|
||||||
|
|
||||||
|
@property({ type: Boolean, reflect: true })
|
||||||
|
accessor collapsed: boolean = false;
|
||||||
|
|
||||||
|
public static styles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
--menu-width-expanded: 200px;
|
||||||
|
--menu-width-collapsed: 56px;
|
||||||
|
--tooltip-bg: ${cssManager.bdTheme('#18181b', '#fafafa')};
|
||||||
|
--tooltip-fg: ${cssManager.bdTheme('#fafafa', '#18181b')};
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mainContainer {
|
||||||
|
color: ${cssManager.bdTheme('#666', '#ccc')};
|
||||||
|
z-index: ${zIndexLayers.fixed.appBar};
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
position: relative;
|
||||||
|
width: var(--menu-width-expanded);
|
||||||
|
height: 100%;
|
||||||
|
background: ${cssManager.bdTheme('#fafafa', '#0a0a0a')};
|
||||||
|
user-select: none;
|
||||||
|
border-right: 1px solid ${cssManager.bdTheme('#e5e5e5', '#1a1a1a')};
|
||||||
|
font-family: 'Geist Sans', 'Inter', -apple-system, BlinkMacSystemFont, sans-serif;
|
||||||
|
transition: width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .mainContainer {
|
||||||
|
width: var(--menu-width-collapsed);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Floating collapse toggle button */
|
||||||
|
.collapse-toggle {
|
||||||
|
position: absolute;
|
||||||
|
right: -12px;
|
||||||
|
top: 24px;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: ${cssManager.bdTheme('#ffffff', '#27272a')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('#e5e5e5', '#3f3f46')};
|
||||||
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 10;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: ${cssManager.bdTheme('#737373', '#a1a1aa')};
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.2s ease, background 0.15s ease;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-toggle:hover {
|
||||||
|
background: ${cssManager.bdTheme('#f4f4f5', '#3f3f46')};
|
||||||
|
color: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
}
|
||||||
|
|
||||||
|
:host(:hover) .collapse-toggle {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-toggle dees-icon {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Logo Section */
|
||||||
|
.logoSection {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
height: 48px;
|
||||||
|
padding: 0 14px;
|
||||||
|
border-bottom: 1px solid ${cssManager.bdTheme('#e5e5e5', '#1a1a1a')};
|
||||||
|
flex-shrink: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logoSection .logoIcon {
|
||||||
|
font-size: 22px;
|
||||||
|
color: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.logoSection .logoText {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
transition: opacity 0.2s ease, width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .logoSection {
|
||||||
|
justify-content: center;
|
||||||
|
padding: 0;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .logoSection .logoText {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Middle Section (scrollable) */
|
||||||
|
.menuSection {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;
|
||||||
|
padding: 8px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-track {
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-thumb {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.15)', 'rgba(255, 255, 255, 0.15)')};
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.25)', 'rgba(255, 255, 255, 0.25)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Menu Group */
|
||||||
|
.menuGroup {
|
||||||
|
padding: 0 8px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuGroup:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader {
|
||||||
|
padding: 8px 12px 6px;
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: ${cssManager.bdTheme('#737373', '#737373')};
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: opacity 0.2s ease, max-height 0.25s ease;
|
||||||
|
max-height: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .groupHeader {
|
||||||
|
opacity: 0;
|
||||||
|
max-height: 0;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupTabs {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .menuGroup {
|
||||||
|
padding: 0 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tab Item */
|
||||||
|
.tab {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
padding: 10px 12px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 500;
|
||||||
|
border-radius: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.15s ease;
|
||||||
|
color: ${cssManager.bdTheme('#525252', '#a3a3a3')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab:hover {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.04)', 'rgba(255, 255, 255, 0.06)')};
|
||||||
|
color: ${cssManager.bdTheme('#262626', '#e5e5e5')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab:active {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.06)', 'rgba(255, 255, 255, 0.08)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.selectedTab {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.06)', 'rgba(255, 255, 255, 0.08)')};
|
||||||
|
color: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.selectedTab::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
width: 3px;
|
||||||
|
height: 16px;
|
||||||
|
background: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
border-radius: 0 2px 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab dees-icon {
|
||||||
|
font-size: 18px;
|
||||||
|
opacity: 0.85;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.selectedTab dees-icon {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab .tabLabel {
|
||||||
|
flex: 1;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
transition: opacity 0.2s ease, width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Collapsed tab styles */
|
||||||
|
:host([collapsed]) .tab {
|
||||||
|
justify-content: center;
|
||||||
|
padding: 10px;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .tab .tabLabel {
|
||||||
|
opacity: 0;
|
||||||
|
width: 0;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .tab.selectedTab::before {
|
||||||
|
left: -4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip for collapsed state */
|
||||||
|
.tab-tooltip {
|
||||||
|
position: absolute;
|
||||||
|
left: 100%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
margin-left: 12px;
|
||||||
|
padding: 6px 12px;
|
||||||
|
background: var(--tooltip-bg);
|
||||||
|
color: var(--tooltip-fg);
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 500;
|
||||||
|
white-space: nowrap;
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
transition: opacity 0.15s ease;
|
||||||
|
z-index: 1000;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-tooltip::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
left: -4px;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
border: 4px solid transparent;
|
||||||
|
border-right-color: var(--tooltip-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .tab:hover .tab-tooltip {
|
||||||
|
opacity: 1;
|
||||||
|
transition-delay: 1s;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bottom Section */
|
||||||
|
.bottomSection {
|
||||||
|
flex-shrink: 0;
|
||||||
|
padding: 8px;
|
||||||
|
border-top: 1px solid ${cssManager.bdTheme('#e5e5e5', '#1a1a1a')};
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .bottomSection {
|
||||||
|
padding: 8px 4px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
|
||||||
|
public render(): TemplateResult {
|
||||||
|
// Get all tabs for selection (from groups or legacy tabs)
|
||||||
|
const allTabs = this.getAllTabs();
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="mainContainer" @contextmenu=${(eventArg: MouseEvent) => {
|
||||||
|
DeesContextmenu.openContextMenuWithOptions(eventArg, [{
|
||||||
|
name: 'app settings',
|
||||||
|
action: async () => {},
|
||||||
|
iconName: 'gear',
|
||||||
|
}])
|
||||||
|
}}>
|
||||||
|
${this.logoIcon || this.logoText ? html`
|
||||||
|
<div class="logoSection">
|
||||||
|
${this.logoIcon ? html`<dees-icon class="logoIcon" .icon="${this.logoIcon}"></dees-icon>` : ''}
|
||||||
|
${this.logoText ? html`<span class="logoText">${this.logoText}</span>` : ''}
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
|
||||||
|
<div class="menuSection">
|
||||||
|
${this.menuGroups.length > 0 ? this.renderMenuGroups() : this.renderLegacyTabs()}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
${this.bottomTabs.length > 0 ? html`
|
||||||
|
<div class="bottomSection">
|
||||||
|
${this.bottomTabs.map((tabArg) => this.renderTab(tabArg))}
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
<button class="collapse-toggle" @click="${() => this.toggleCollapse()}">
|
||||||
|
<dees-icon .icon="${this.collapsed ? 'lucide:chevronRight' : 'lucide:chevronLeft'}"></dees-icon>
|
||||||
|
</button>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderMenuGroups(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
${this.menuGroups.map((group) => html`
|
||||||
|
<div class="menuGroup">
|
||||||
|
${group.name ? html`<div class="groupHeader">${group.name}</div>` : ''}
|
||||||
|
<div class="groupTabs">
|
||||||
|
${group.tabs.map((tabArg) => this.renderTab(tabArg))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`)}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderLegacyTabs(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<div class="menuGroup">
|
||||||
|
<div class="groupTabs">
|
||||||
|
${this.tabs.map((tabArg) => this.renderTab(tabArg))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderTab(tabArg: interfaces.ITab): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<div
|
||||||
|
class="tab ${tabArg === this.selectedTab ? 'selectedTab' : ''}"
|
||||||
|
@click="${() => {
|
||||||
|
this.updateTab(tabArg);
|
||||||
|
}}"
|
||||||
|
>
|
||||||
|
<dees-icon .icon="${tabArg.iconName || ''}"></dees-icon>
|
||||||
|
<span class="tabLabel">${tabArg.key}</span>
|
||||||
|
<span class="tab-tooltip">${tabArg.key}</span>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getAllTabs(): interfaces.ITab[] {
|
||||||
|
if (this.menuGroups.length > 0) {
|
||||||
|
const groupTabs = this.menuGroups.flatMap(group => group.tabs);
|
||||||
|
return [...groupTabs, ...this.bottomTabs];
|
||||||
|
}
|
||||||
|
return [...this.tabs, ...this.bottomTabs];
|
||||||
|
}
|
||||||
|
|
||||||
|
updateTab(tabArg: interfaces.ITab) {
|
||||||
|
this.selectedTab = tabArg;
|
||||||
|
this.selectedTab.action();
|
||||||
|
|
||||||
|
// Emit tab-select event
|
||||||
|
this.dispatchEvent(new CustomEvent('tab-select', {
|
||||||
|
detail: { tab: tabArg },
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
firstUpdated() {
|
||||||
|
const allTabs = this.getAllTabs();
|
||||||
|
if (allTabs.length > 0) {
|
||||||
|
this.updateTab(allTabs[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public toggleCollapse(): void {
|
||||||
|
this.collapsed = !this.collapsed;
|
||||||
|
this.dispatchEvent(new CustomEvent('collapse-change', {
|
||||||
|
detail: { collapsed: this.collapsed },
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-mainmenu.js';
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../../00plugins.js';
|
||||||
import { zIndexLayers } from './00zindex.js';
|
import { zIndexLayers } from '../../00zindex.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -36,21 +36,21 @@ export class DeesAppuiProfileDropdown extends DeesElement {
|
|||||||
`;
|
`;
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public user?: {
|
accessor user: {
|
||||||
name: string;
|
name: string;
|
||||||
email?: string;
|
email?: string;
|
||||||
avatar?: string;
|
avatar?: string;
|
||||||
status?: 'online' | 'offline' | 'busy' | 'away';
|
status?: 'online' | 'offline' | 'busy' | 'away';
|
||||||
};
|
} | undefined = undefined;
|
||||||
|
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public menuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
accessor menuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true })
|
@property({ type: Boolean, reflect: true })
|
||||||
public isOpen: boolean = false;
|
accessor isOpen: boolean = false;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public position: 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left' = 'top-right';
|
accessor position: 'top-right' | 'top-left' | 'bottom-right' | 'bottom-left' = 'top-right';
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-profiledropdown.js';
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { html } from '@design.estate/dees-element';
|
||||||
|
import type * as interfaces from '../../interfaces/index.js';
|
||||||
|
|
||||||
|
export const demoFunc = () => html`
|
||||||
|
<style>
|
||||||
|
.demo-secondarymenu-container {
|
||||||
|
display: flex;
|
||||||
|
height: 100%;
|
||||||
|
background: #1a1a1a;
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
.demo-secondarymenu-container .spacer {
|
||||||
|
flex: 1;
|
||||||
|
background: #0f0f0f;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<div class="demo-secondarymenu-container">
|
||||||
|
<dees-appui-secondarymenu
|
||||||
|
.heading=${'Projects'}
|
||||||
|
.groups=${[
|
||||||
|
{
|
||||||
|
name: 'Active',
|
||||||
|
iconName: 'lucide:folder',
|
||||||
|
items: [
|
||||||
|
{ key: 'Frontend App', iconName: 'code', action: () => console.log('Frontend'), badge: 3, badgeVariant: 'warning' },
|
||||||
|
{ key: 'API Server', iconName: 'server', action: () => console.log('API'), badge: 'new', badgeVariant: 'success' },
|
||||||
|
{ key: 'Database', iconName: 'database', action: () => console.log('Database') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Archived',
|
||||||
|
iconName: 'lucide:archive',
|
||||||
|
collapsed: true,
|
||||||
|
items: [
|
||||||
|
{ key: 'Legacy System', iconName: 'box', action: () => console.log('Legacy') },
|
||||||
|
{ key: 'Old API', iconName: 'server', action: () => console.log('Old API') },
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Settings',
|
||||||
|
iconName: 'lucide:settings',
|
||||||
|
items: [
|
||||||
|
{ key: 'Configuration', iconName: 'sliders', action: () => console.log('Config') },
|
||||||
|
{ key: 'Integrations', iconName: 'plug', action: () => console.log('Integrations'), badge: 5, badgeVariant: 'error' },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
] as interfaces.ISecondaryMenuGroup[]}
|
||||||
|
@item-select=${(e: CustomEvent) => console.log('Selected:', e.detail)}
|
||||||
|
></dees-appui-secondarymenu>
|
||||||
|
<div class="spacer"></div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
@@ -0,0 +1,601 @@
|
|||||||
|
import * as plugins from '../../00plugins.js';
|
||||||
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
|
||||||
|
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
import '../../dees-icon/dees-icon.js';
|
||||||
|
|
||||||
|
import {
|
||||||
|
DeesElement,
|
||||||
|
type TemplateResult,
|
||||||
|
property,
|
||||||
|
state,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
css,
|
||||||
|
cssManager,
|
||||||
|
} from '@design.estate/dees-element';
|
||||||
|
import { demoFunc } from './dees-appui-secondarymenu.demo.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Secondary navigation menu for sub-navigation within MainMenu views
|
||||||
|
* Supports collapsible groups, badges, and dynamic headings
|
||||||
|
*/
|
||||||
|
@customElement('dees-appui-secondarymenu')
|
||||||
|
export class DeesAppuiSecondarymenu extends DeesElement {
|
||||||
|
public static demo = demoFunc;
|
||||||
|
|
||||||
|
// INSTANCE
|
||||||
|
|
||||||
|
/** Dynamic heading - typically shows the selected MainMenu item */
|
||||||
|
@property({ type: String })
|
||||||
|
accessor heading: string = 'Menu';
|
||||||
|
|
||||||
|
/** Grouped items with collapse support */
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor groups: interfaces.ISecondaryMenuGroup[] = [];
|
||||||
|
|
||||||
|
/** Legacy flat list support for backward compatibility */
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor selectionOptions: (interfaces.ISelectionOption | { divider: true })[] = [];
|
||||||
|
|
||||||
|
/** Currently selected item */
|
||||||
|
@property({ type: Object })
|
||||||
|
accessor selectedItem: interfaces.ISecondaryMenuItem | null = null;
|
||||||
|
|
||||||
|
/** Internal state for collapsed groups */
|
||||||
|
@state()
|
||||||
|
accessor collapsedGroups: Set<string> = new Set();
|
||||||
|
|
||||||
|
/** Horizontal collapse state */
|
||||||
|
@property({ type: Boolean, reflect: true })
|
||||||
|
accessor collapsed: boolean = false;
|
||||||
|
|
||||||
|
public static styles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
--sidebar-width-expanded: 240px;
|
||||||
|
--sidebar-width-collapsed: 56px;
|
||||||
|
--sidebar-bg: ${cssManager.bdTheme('#fafafa', '#0a0a0a')};
|
||||||
|
--sidebar-fg: ${cssManager.bdTheme('#525252', '#a3a3a3')};
|
||||||
|
--sidebar-fg-muted: ${cssManager.bdTheme('#737373', '#737373')};
|
||||||
|
--sidebar-fg-active: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
--sidebar-border: ${cssManager.bdTheme('#e5e5e5', '#1a1a1a')};
|
||||||
|
--sidebar-hover: ${cssManager.bdTheme('rgba(0, 0, 0, 0.04)', 'rgba(255, 255, 255, 0.06)')};
|
||||||
|
--sidebar-active: ${cssManager.bdTheme('rgba(0, 0, 0, 0.06)', 'rgba(255, 255, 255, 0.08)')};
|
||||||
|
--sidebar-accent: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
--tooltip-bg: ${cssManager.bdTheme('#18181b', '#fafafa')};
|
||||||
|
--tooltip-fg: ${cssManager.bdTheme('#fafafa', '#18181b')};
|
||||||
|
|
||||||
|
/* Badge colors */
|
||||||
|
--badge-default-bg: ${cssManager.bdTheme('#f4f4f5', '#27272a')};
|
||||||
|
--badge-default-fg: ${cssManager.bdTheme('#3f3f46', '#a1a1aa')};
|
||||||
|
--badge-success-bg: ${cssManager.bdTheme('#dcfce7', '#14532d')};
|
||||||
|
--badge-success-fg: ${cssManager.bdTheme('#166534', '#4ade80')};
|
||||||
|
--badge-warning-bg: ${cssManager.bdTheme('#fef3c7', '#451a03')};
|
||||||
|
--badge-warning-fg: ${cssManager.bdTheme('#92400e', '#fbbf24')};
|
||||||
|
--badge-error-bg: ${cssManager.bdTheme('#fee2e2', '#450a0a')};
|
||||||
|
--badge-error-fg: ${cssManager.bdTheme('#991b1b', '#f87171')};
|
||||||
|
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
height: 100%;
|
||||||
|
width: var(--sidebar-width-expanded);
|
||||||
|
background: var(--sidebar-bg);
|
||||||
|
border-right: 1px solid var(--sidebar-border);
|
||||||
|
font-family: 'Geist Sans', 'Inter', -apple-system, BlinkMacSystemFont, sans-serif;
|
||||||
|
user-select: none;
|
||||||
|
transition: width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) {
|
||||||
|
width: var(--sidebar-width-collapsed);
|
||||||
|
}
|
||||||
|
|
||||||
|
.maincontainer {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Floating collapse toggle button */
|
||||||
|
.collapse-toggle {
|
||||||
|
position: absolute;
|
||||||
|
right: -12px;
|
||||||
|
top: 24px;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: ${cssManager.bdTheme('#ffffff', '#27272a')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('#e5e5e5', '#3f3f46')};
|
||||||
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 10;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: ${cssManager.bdTheme('#737373', '#a1a1aa')};
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.2s ease, background 0.15s ease;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-toggle:hover {
|
||||||
|
background: ${cssManager.bdTheme('#f4f4f5', '#3f3f46')};
|
||||||
|
color: ${cssManager.bdTheme('#0a0a0a', '#fafafa')};
|
||||||
|
}
|
||||||
|
|
||||||
|
:host(:hover) .collapse-toggle {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.collapse-toggle dees-icon {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Header Section */
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
height: 48px;
|
||||||
|
padding: 0 16px;
|
||||||
|
border-bottom: 1px solid var(--sidebar-border);
|
||||||
|
flex-shrink: 0;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header .heading {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--sidebar-fg-active);
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
transition: opacity 0.2s ease, width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .header {
|
||||||
|
justify-content: center;
|
||||||
|
padding: 0 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .header .heading {
|
||||||
|
opacity: 0;
|
||||||
|
width: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Scrollable Menu Section */
|
||||||
|
.menuSection {
|
||||||
|
flex: 1;
|
||||||
|
overflow-y: auto;
|
||||||
|
overflow-x: hidden;
|
||||||
|
padding: 8px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar {
|
||||||
|
width: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-track {
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-thumb {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.15)', 'rgba(255, 255, 255, 0.15)')};
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuSection::-webkit-scrollbar-thumb:hover {
|
||||||
|
background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.25)', 'rgba(255, 255, 255, 0.25)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Menu Group */
|
||||||
|
.menuGroup {
|
||||||
|
padding: 0 8px;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .menuGroup {
|
||||||
|
padding: 0 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 8px 8px;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 6px;
|
||||||
|
transition: background 0.15s ease, opacity 0.2s ease, max-height 0.25s ease;
|
||||||
|
max-height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader:hover {
|
||||||
|
background: var(--sidebar-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader .groupTitle {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--sidebar-fg-muted);
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader .groupTitle dees-icon {
|
||||||
|
font-size: 14px;
|
||||||
|
opacity: 0.7;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader .chevron {
|
||||||
|
font-size: 12px;
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
color: var(--sidebar-fg-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupHeader.collapsed .chevron {
|
||||||
|
transform: rotate(-90deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide group headers when horizontally collapsed */
|
||||||
|
:host([collapsed]) .groupHeader {
|
||||||
|
opacity: 0;
|
||||||
|
max-height: 0;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Group Items Container */
|
||||||
|
.groupItems {
|
||||||
|
overflow: hidden;
|
||||||
|
transition: max-height 0.25s ease, opacity 0.2s ease;
|
||||||
|
max-height: 500px;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.groupItems.collapsed {
|
||||||
|
max-height: 0;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Always show items when horizontally collapsed (regardless of group collapse state) */
|
||||||
|
:host([collapsed]) .groupItems {
|
||||||
|
max-height: none;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Menu Item */
|
||||||
|
.menuItem {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
margin: 2px 0;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 450;
|
||||||
|
border-radius: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.15s ease;
|
||||||
|
color: var(--sidebar-fg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem:hover {
|
||||||
|
background: var(--sidebar-hover);
|
||||||
|
color: var(--sidebar-fg-active);
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem:active {
|
||||||
|
background: var(--sidebar-active);
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem.selected {
|
||||||
|
background: var(--sidebar-active);
|
||||||
|
color: var(--sidebar-fg-active);
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem.selected::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
width: 3px;
|
||||||
|
height: 16px;
|
||||||
|
background: var(--sidebar-accent);
|
||||||
|
border-radius: 0 2px 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem dees-icon {
|
||||||
|
font-size: 16px;
|
||||||
|
opacity: 0.7;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem.selected dees-icon {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.menuItem .itemLabel {
|
||||||
|
flex: 1;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
transition: opacity 0.2s ease, width 0.25s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Collapsed menu item styles */
|
||||||
|
:host([collapsed]) .menuItem {
|
||||||
|
justify-content: center;
|
||||||
|
padding: 8px;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .menuItem .itemLabel {
|
||||||
|
opacity: 0;
|
||||||
|
width: 0;
|
||||||
|
position: absolute;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .menuItem.selected::before {
|
||||||
|
left: -4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip for collapsed state */
|
||||||
|
.item-tooltip {
|
||||||
|
position: absolute;
|
||||||
|
left: 100%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
margin-left: 12px;
|
||||||
|
padding: 6px 12px;
|
||||||
|
background: var(--tooltip-bg);
|
||||||
|
color: var(--tooltip-fg);
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 500;
|
||||||
|
white-space: nowrap;
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
transition: opacity 0.15s ease;
|
||||||
|
z-index: 1000;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-tooltip::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
left: -4px;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
border: 4px solid transparent;
|
||||||
|
border-right-color: var(--tooltip-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .menuItem:hover .item-tooltip {
|
||||||
|
opacity: 1;
|
||||||
|
transition-delay: 1s;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Badge Styles */
|
||||||
|
.badge {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
min-width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
padding: 0 6px;
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 600;
|
||||||
|
border-radius: 9px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge.default {
|
||||||
|
background: var(--badge-default-bg);
|
||||||
|
color: var(--badge-default-fg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge.success {
|
||||||
|
background: var(--badge-success-bg);
|
||||||
|
color: var(--badge-success-fg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge.warning {
|
||||||
|
background: var(--badge-warning-bg);
|
||||||
|
color: var(--badge-warning-fg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.badge.error {
|
||||||
|
background: var(--badge-error-bg);
|
||||||
|
color: var(--badge-error-fg);
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([collapsed]) .badge {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Divider */
|
||||||
|
.divider {
|
||||||
|
height: 1px;
|
||||||
|
background: var(--sidebar-border);
|
||||||
|
margin: 8px 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Legacy options container */
|
||||||
|
.legacyOptions {
|
||||||
|
padding: 0 8px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
|
||||||
|
public render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<div class="maincontainer">
|
||||||
|
<div class="header">
|
||||||
|
<span class="heading">${this.heading}</span>
|
||||||
|
</div>
|
||||||
|
<div class="menuSection">
|
||||||
|
${this.groups.length > 0
|
||||||
|
? this.renderGroups()
|
||||||
|
: this.renderLegacyOptions()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button class="collapse-toggle" @click="${() => this.toggleCollapse()}">
|
||||||
|
<dees-icon .icon="${this.collapsed ? 'lucide:chevronRight' : 'lucide:chevronLeft'}"></dees-icon>
|
||||||
|
</button>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderGroups(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
${this.groups.map((group) => html`
|
||||||
|
<div class="menuGroup">
|
||||||
|
<div
|
||||||
|
class="groupHeader ${this.collapsedGroups.has(group.name) ? 'collapsed' : ''}"
|
||||||
|
@click="${() => this.toggleGroup(group.name)}"
|
||||||
|
>
|
||||||
|
<span class="groupTitle">
|
||||||
|
${group.iconName ? html`<dees-icon .icon="${group.iconName.startsWith('lucide:') ? group.iconName : `lucide:${group.iconName}`}"></dees-icon>` : ''}
|
||||||
|
${group.name}
|
||||||
|
</span>
|
||||||
|
<dees-icon class="chevron" .icon="${'lucide:chevronDown'}"></dees-icon>
|
||||||
|
</div>
|
||||||
|
<div class="groupItems ${this.collapsedGroups.has(group.name) ? 'collapsed' : ''}">
|
||||||
|
${group.items.map((item) => this.renderMenuItem(item, group))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`)}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderMenuItem(item: interfaces.ISecondaryMenuItem, group?: interfaces.ISecondaryMenuGroup): TemplateResult {
|
||||||
|
const isSelected = this.selectedItem?.key === item.key;
|
||||||
|
return html`
|
||||||
|
<div
|
||||||
|
class="menuItem ${isSelected ? 'selected' : ''}"
|
||||||
|
@click="${() => this.selectItem(item, group)}"
|
||||||
|
@contextmenu="${(e: MouseEvent) => this.handleContextMenu(e, item)}"
|
||||||
|
>
|
||||||
|
${item.iconName ? html`<dees-icon .icon="${item.iconName.startsWith('lucide:') ? item.iconName : `lucide:${item.iconName}`}"></dees-icon>` : ''}
|
||||||
|
<span class="itemLabel">${item.key}</span>
|
||||||
|
${item.badge !== undefined ? html`
|
||||||
|
<span class="badge ${item.badgeVariant || 'default'}">${item.badge}</span>
|
||||||
|
` : ''}
|
||||||
|
<span class="item-tooltip">${item.key}</span>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderLegacyOptions(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<div class="legacyOptions">
|
||||||
|
${this.selectionOptions.map((option) => {
|
||||||
|
if ('divider' in option && option.divider) {
|
||||||
|
return html`<div class="divider"></div>`;
|
||||||
|
}
|
||||||
|
const item = option as interfaces.ISelectionOption;
|
||||||
|
return this.renderMenuItem({
|
||||||
|
key: item.key,
|
||||||
|
iconName: item.iconName,
|
||||||
|
action: item.action,
|
||||||
|
});
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private toggleGroup(groupName: string): void {
|
||||||
|
const newCollapsed = new Set(this.collapsedGroups);
|
||||||
|
if (newCollapsed.has(groupName)) {
|
||||||
|
newCollapsed.delete(groupName);
|
||||||
|
} else {
|
||||||
|
newCollapsed.add(groupName);
|
||||||
|
}
|
||||||
|
this.collapsedGroups = newCollapsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
public toggleCollapse(): void {
|
||||||
|
this.collapsed = !this.collapsed;
|
||||||
|
this.dispatchEvent(new CustomEvent('collapse-change', {
|
||||||
|
detail: { collapsed: this.collapsed },
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private selectItem(item: interfaces.ISecondaryMenuItem, group?: interfaces.ISecondaryMenuGroup): void {
|
||||||
|
this.selectedItem = item;
|
||||||
|
item.action();
|
||||||
|
|
||||||
|
this.dispatchEvent(new CustomEvent('item-select', {
|
||||||
|
detail: { item, group },
|
||||||
|
bubbles: true,
|
||||||
|
composed: true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleContextMenu(event: MouseEvent, item: interfaces.ISecondaryMenuItem): void {
|
||||||
|
DeesContextmenu.openContextMenuWithOptions(event, [
|
||||||
|
{
|
||||||
|
name: 'View details',
|
||||||
|
action: async () => {},
|
||||||
|
iconName: 'lucide:eye',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Edit',
|
||||||
|
action: async () => {},
|
||||||
|
iconName: 'lucide:pencil',
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
async firstUpdated(_changedProperties: Map<string | number | symbol, unknown>) {
|
||||||
|
await super.firstUpdated(_changedProperties);
|
||||||
|
|
||||||
|
// Initialize collapsed state from group defaults
|
||||||
|
if (this.groups.length > 0) {
|
||||||
|
const initialCollapsed = new Set<string>();
|
||||||
|
this.groups.forEach(group => {
|
||||||
|
if (group.collapsed) {
|
||||||
|
initialCollapsed.add(group.name);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.collapsedGroups = initialCollapsed;
|
||||||
|
|
||||||
|
// Auto-select first item if none selected
|
||||||
|
if (!this.selectedItem && this.groups[0]?.items.length > 0) {
|
||||||
|
this.selectItem(this.groups[0].items[0], this.groups[0]);
|
||||||
|
}
|
||||||
|
} else if (this.selectionOptions.length > 0) {
|
||||||
|
// Legacy mode: select first non-divider option
|
||||||
|
const firstOption = this.selectionOptions.find(opt => !('divider' in opt)) as interfaces.ISelectionOption;
|
||||||
|
if (firstOption && !this.selectedItem) {
|
||||||
|
this.selectItem({
|
||||||
|
key: firstOption.key,
|
||||||
|
iconName: firstOption.iconName,
|
||||||
|
action: firstOption.action,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
'dees-appui-secondarymenu': DeesAppuiSecondarymenu;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-secondarymenu.js';
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -107,16 +107,16 @@ export class DeesAppuiTabs extends DeesElement {
|
|||||||
@property({
|
@property({
|
||||||
type: Array,
|
type: Array,
|
||||||
})
|
})
|
||||||
public tabs: interfaces.ITab[] = [];
|
accessor tabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public selectedTab: interfaces.ITab | null = null;
|
accessor selectedTab: interfaces.ITab | null = null;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public showTabIndicator: boolean = true;
|
accessor showTabIndicator: boolean = true;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public tabStyle: 'horizontal' | 'vertical' = 'horizontal';
|
accessor tabStyle: 'horizontal' | 'vertical' = 'horizontal';
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
@@ -132,7 +132,9 @@ export class DeesAppuiTabs extends DeesElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.tabs-wrapper.horizontal-wrapper {
|
.tabs-wrapper.horizontal-wrapper {
|
||||||
|
height: 48px;
|
||||||
border-bottom: 1px solid ${cssManager.bdTheme('#e5e7eb', '#27272a')};
|
border-bottom: 1px solid ${cssManager.bdTheme('#e5e7eb', '#27272a')};
|
||||||
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tabsContainer {
|
.tabsContainer {
|
||||||
@@ -146,7 +148,7 @@ export class DeesAppuiTabs extends DeesElement {
|
|||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
scrollbar-width: none;
|
scrollbar-width: none;
|
||||||
height: 48px;
|
height: 100%;
|
||||||
padding: 0 16px;
|
padding: 0 16px;
|
||||||
gap: 4px;
|
gap: 4px;
|
||||||
}
|
}
|
||||||
1
ts_web/elements/00group-appui/dees-appui-tabs/index.ts
Normal file
1
ts_web/elements/00group-appui/dees-appui-tabs/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-tabs.js';
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../../interfaces/index.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -11,8 +11,8 @@ import {
|
|||||||
state,
|
state,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import './dees-appui-tabs.js';
|
import '../dees-appui-tabs/dees-appui-tabs.js';
|
||||||
import type { DeesAppuiTabs } from './dees-appui-tabs.js';
|
import type { DeesAppuiTabs } from '../dees-appui-tabs/dees-appui-tabs.js';
|
||||||
|
|
||||||
export interface IAppViewTab extends interfaces.ITab {
|
export interface IAppViewTab extends interfaces.ITab {
|
||||||
content?: TemplateResult | (() => TemplateResult);
|
content?: TemplateResult | (() => TemplateResult);
|
||||||
@@ -60,13 +60,13 @@ export class DeesAppuiView extends DeesElement {
|
|||||||
|
|
||||||
// INSTANCE
|
// INSTANCE
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public viewConfig: IAppView;
|
accessor viewConfig: IAppView;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private selectedTab: IAppViewTab | null = null;
|
accessor selectedTab: IAppViewTab | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private tabs: DeesAppuiTabs;
|
accessor tabs: DeesAppuiTabs;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
1
ts_web/elements/00group-appui/dees-appui-view/index.ts
Normal file
1
ts_web/elements/00group-appui/dees-appui-view/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-appui-view.js';
|
||||||
10
ts_web/elements/00group-appui/index.ts
Normal file
10
ts_web/elements/00group-appui/index.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
// App UI Components
|
||||||
|
export * from './dees-appui-activitylog/index.js';
|
||||||
|
export * from './dees-appui-appbar/index.js';
|
||||||
|
export * from './dees-appui-base/index.js';
|
||||||
|
export * from './dees-appui-maincontent/index.js';
|
||||||
|
export * from './dees-appui-mainmenu/index.js';
|
||||||
|
export * from './dees-appui-secondarymenu/index.js';
|
||||||
|
export * from './dees-appui-profiledropdown/index.js';
|
||||||
|
export * from './dees-appui-tabs/index.js';
|
||||||
|
export * from './dees-appui-view/index.js';
|
||||||
@@ -21,7 +21,7 @@ export class DeesButtonExit extends DeesElement {
|
|||||||
@property({
|
@property({
|
||||||
type: Number
|
type: Number
|
||||||
})
|
})
|
||||||
public size: number = 24;
|
accessor size: number = 24;
|
||||||
|
|
||||||
public styles = [
|
public styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
1
ts_web/elements/00group-button/dees-button-exit/index.ts
Normal file
1
ts_web/elements/00group-button/dees-button-exit/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-button-exit.js';
|
||||||
@@ -22,10 +22,10 @@ export class DeesButtonGroup extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public label: string = '';
|
accessor label: string = '';
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public direction: 'horizontal' | 'vertical' = 'horizontal';
|
accessor direction: 'horizontal' | 'vertical' = 'horizontal';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-button-group.js';
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
import { html, css, cssManager, domtools } from '@design.estate/dees-element';
|
import { html, css, cssManager, domtools } from '@design.estate/dees-element';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
import './dees-panel.js';
|
import '../../dees-panel/dees-panel.js';
|
||||||
import './dees-form.js';
|
import '../../00group-form/dees-form/dees-form.js';
|
||||||
import './dees-form-submit.js';
|
import '../../00group-form/dees-form-submit/dees-form-submit.js';
|
||||||
import './dees-input-text.js';
|
import '../../00group-input/dees-input-text/dees-input-text.js';
|
||||||
import './dees-icon.js';
|
import '../../dees-icon/dees-icon.js';
|
||||||
import type { DeesButton } from './dees-button.js';
|
import type { DeesButton } from '../dees-button/dees-button.js';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
<style>
|
<style>
|
||||||
@@ -29,42 +29,42 @@ export class DeesButton extends DeesElement {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
public text: string;
|
accessor text: string;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public eventDetailData: string;
|
accessor eventDetailData: string;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
reflect: true,
|
reflect: true,
|
||||||
})
|
})
|
||||||
public disabled = false;
|
accessor disabled = false;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Boolean
|
type: Boolean
|
||||||
})
|
})
|
||||||
public isHidden = false;
|
accessor isHidden = false;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String
|
type: String
|
||||||
})
|
})
|
||||||
public type: 'default' | 'destructive' | 'outline' | 'secondary' | 'ghost' | 'link' | 'normal' | 'highlighted' | 'discreet' | 'big' = 'default';
|
accessor type: 'default' | 'destructive' | 'outline' | 'secondary' | 'ghost' | 'link' | 'normal' | 'highlighted' | 'discreet' | 'big' = 'default';
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String
|
type: String
|
||||||
})
|
})
|
||||||
public size: 'default' | 'sm' | 'lg' | 'icon' = 'default';
|
accessor size: 'default' | 'sm' | 'lg' | 'icon' = 'default';
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String
|
type: String
|
||||||
})
|
})
|
||||||
public status: 'normal' | 'pending' | 'success' | 'error' = 'normal';
|
accessor status: 'normal' | 'pending' | 'success' | 'error' = 'normal';
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
reflect: true
|
reflect: true
|
||||||
})
|
})
|
||||||
public insideForm: boolean = false;
|
accessor insideForm: boolean = false;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
1
ts_web/elements/00group-button/dees-button/index.ts
Normal file
1
ts_web/elements/00group-button/dees-button/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-button.js';
|
||||||
4
ts_web/elements/00group-button/index.ts
Normal file
4
ts_web/elements/00group-button/index.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
// Button Components
|
||||||
|
export * from './dees-button/index.js';
|
||||||
|
export * from './dees-button-exit/index.js';
|
||||||
|
export * from './dees-button-group/index.js';
|
||||||
@@ -1,16 +1,15 @@
|
|||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
css,
|
|
||||||
cssManager,
|
|
||||||
customElement,
|
customElement,
|
||||||
html,
|
|
||||||
property,
|
property,
|
||||||
state,
|
state,
|
||||||
type TemplateResult,
|
type TemplateResult,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import { demoFunc } from './dees-chart-area.demo.js';
|
import { demoFunc } from './demo.js';
|
||||||
|
import { chartAreaStyles } from './styles.js';
|
||||||
|
import { renderChartArea } from './template.js';
|
||||||
|
|
||||||
import ApexCharts from 'apexcharts';
|
import ApexCharts from 'apexcharts';
|
||||||
|
|
||||||
@@ -26,13 +25,13 @@ export class DeesChartArea extends DeesElement {
|
|||||||
|
|
||||||
// instance
|
// instance
|
||||||
@state()
|
@state()
|
||||||
public chart: ApexCharts;
|
accessor chart: ApexCharts;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public label: string = 'Untitled Chart';
|
accessor label: string = 'Untitled Chart';
|
||||||
|
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public series: ApexAxisChartSeries = [];
|
accessor series: ApexAxisChartSeries = [];
|
||||||
|
|
||||||
// Override getter to return internal chart data
|
// Override getter to return internal chart data
|
||||||
get chartSeries(): ApexAxisChartSeries {
|
get chartSeries(): ApexAxisChartSeries {
|
||||||
@@ -40,22 +39,22 @@ export class DeesChartArea extends DeesElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@property({ attribute: false })
|
@property({ attribute: false })
|
||||||
public yAxisFormatter: (value: number) => string = (val) => `${val} Mbps`;
|
accessor yAxisFormatter: (value: number) => string = (val) => `${val} Mbps`;
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public rollingWindow: number = 0; // 0 means no rolling window
|
accessor rollingWindow: number = 0; // 0 means no rolling window
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public realtimeMode: boolean = false;
|
accessor realtimeMode: boolean = false;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public yAxisScaling: 'fixed' | 'dynamic' | 'percentage' = 'dynamic';
|
accessor yAxisScaling: 'fixed' | 'dynamic' | 'percentage' = 'dynamic';
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public yAxisMax: number = 100; // Used when yAxisScaling is 'fixed' or 'percentage'
|
accessor yAxisMax: number = 100; // Used when yAxisScaling is 'fixed' or 'percentage'
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public autoScrollInterval: number = 1000; // Auto-scroll interval in milliseconds (0 to disable)
|
accessor autoScrollInterval: number = 1000; // Auto-scroll interval in milliseconds (0 to disable)
|
||||||
|
|
||||||
private resizeObserver: ResizeObserver;
|
private resizeObserver: ResizeObserver;
|
||||||
private resizeTimeout: number;
|
private resizeTimeout: number;
|
||||||
@@ -141,73 +140,14 @@ export class DeesChartArea extends DeesElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static styles = [
|
public static styles = chartAreaStyles;
|
||||||
cssManager.defaultStyles,
|
|
||||||
css`
|
|
||||||
:host {
|
|
||||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', sans-serif;
|
|
||||||
color: ${cssManager.bdTheme('hsl(0 0% 3.9%)', 'hsl(0 0% 98%)')};
|
|
||||||
font-weight: 400;
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
.mainbox {
|
|
||||||
position: relative;
|
|
||||||
width: 100%;
|
|
||||||
height: 400px;
|
|
||||||
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(0 0% 3.9%)')};
|
|
||||||
border: 1px solid ${cssManager.bdTheme('hsl(0 0% 89.8%)', 'hsl(0 0% 14.9%)')};
|
|
||||||
border-radius: 8px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.chartTitle {
|
|
||||||
position: absolute;
|
|
||||||
top: 0;
|
|
||||||
left: 0;
|
|
||||||
width: 100%;
|
|
||||||
text-align: left;
|
|
||||||
padding: 16px 24px;
|
|
||||||
z-index: 10;
|
|
||||||
font-size: 14px;
|
|
||||||
font-weight: 500;
|
|
||||||
letter-spacing: -0.01em;
|
|
||||||
color: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 63.9%)')};
|
|
||||||
}
|
|
||||||
.chartContainer {
|
|
||||||
position: absolute;
|
|
||||||
top: 0px;
|
|
||||||
left: 0px;
|
|
||||||
bottom: 0px;
|
|
||||||
right: 0px;
|
|
||||||
padding: 44px 16px 16px 0px;
|
|
||||||
overflow: hidden;
|
|
||||||
background: transparent; /* Ensure container doesn't override chart background */
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ApexCharts theme overrides */
|
|
||||||
.apexcharts-canvas {
|
|
||||||
background: transparent !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.apexcharts-inner {
|
|
||||||
background: transparent !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.apexcharts-graphical {
|
|
||||||
background: transparent !important;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
];
|
|
||||||
|
|
||||||
public render(): TemplateResult {
|
public render(): TemplateResult {
|
||||||
return html`
|
return renderChartArea(this);
|
||||||
<div class="mainbox">
|
|
||||||
<div class="chartTitle">${this.label}</div>
|
|
||||||
<div class="chartContainer"></div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public async firstUpdated() {
|
public async firstUpdated() {
|
||||||
await this.domtoolsPromise;
|
await this.domtoolsPromise;
|
||||||
|
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { html, css, cssManager } from '@design.estate/dees-element';
|
import { html, css, cssManager } from '@design.estate/dees-element';
|
||||||
import type { DeesChartArea } from './dees-chart-area.js';
|
import type { DeesChartArea } from './component.js';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
import './component.js';
|
||||||
|
|
||||||
export const demoFunc = () => {
|
export const demoFunc = () => {
|
||||||
// Initial dataset values
|
// Initial dataset values
|
||||||
1
ts_web/elements/00group-chart/dees-chart-area/index.ts
Normal file
1
ts_web/elements/00group-chart/dees-chart-area/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './component.js';
|
||||||
60
ts_web/elements/00group-chart/dees-chart-area/styles.ts
Normal file
60
ts_web/elements/00group-chart/dees-chart-area/styles.ts
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import { css, cssManager } from '@design.estate/dees-element';
|
||||||
|
|
||||||
|
export const chartAreaStyles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', sans-serif;
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 0% 3.9%)', 'hsl(0 0% 98%)')};
|
||||||
|
font-weight: 400;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
.mainbox {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 400px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(0 0% 3.9%)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(0 0% 89.8%)', 'hsl(0 0% 14.9%)')};
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chartTitle {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
text-align: left;
|
||||||
|
padding: 16px 24px;
|
||||||
|
z-index: 10;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
letter-spacing: -0.01em;
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 63.9%)')};
|
||||||
|
}
|
||||||
|
.chartContainer {
|
||||||
|
position: absolute;
|
||||||
|
top: 0px;
|
||||||
|
left: 0px;
|
||||||
|
bottom: 0px;
|
||||||
|
right: 0px;
|
||||||
|
padding: 44px 16px 16px 0px;
|
||||||
|
overflow: hidden;
|
||||||
|
background: transparent; /* Ensure container doesn't override chart background */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ApexCharts theme overrides */
|
||||||
|
.apexcharts-canvas {
|
||||||
|
background: transparent !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.apexcharts-inner {
|
||||||
|
background: transparent !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.apexcharts-graphical {
|
||||||
|
background: transparent !important;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
|
||||||
12
ts_web/elements/00group-chart/dees-chart-area/template.ts
Normal file
12
ts_web/elements/00group-chart/dees-chart-area/template.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { html, type TemplateResult } from '@design.estate/dees-element';
|
||||||
|
import type { DeesChartArea } from './component.js';
|
||||||
|
|
||||||
|
export const renderChartArea = (component: DeesChartArea): TemplateResult => {
|
||||||
|
return html`
|
||||||
|
<div class="mainbox">
|
||||||
|
<div class="chartTitle">${component.label}</div>
|
||||||
|
<div class="chartContainer"></div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
};
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { html } from '@design.estate/dees-element';
|
import { html } from '@design.estate/dees-element';
|
||||||
import type { DeesChartLog } from './dees-chart-log.js';
|
import type { DeesChartLog } from '../dees-chart-log/dees-chart-log.js';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
|
||||||
export const demoFunc = () => {
|
export const demoFunc = () => {
|
||||||
@@ -30,16 +30,16 @@ export class DeesChartLog extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public label: string = 'Server Logs';
|
accessor label: string = 'Server Logs';
|
||||||
|
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public logEntries: ILogEntry[] = [];
|
accessor logEntries: ILogEntry[] = [];
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public autoScroll: boolean = true;
|
accessor autoScroll: boolean = true;
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public maxEntries: number = 1000;
|
accessor maxEntries: number = 1000;
|
||||||
|
|
||||||
private logContainer: HTMLDivElement;
|
private logContainer: HTMLDivElement;
|
||||||
|
|
||||||
1
ts_web/elements/00group-chart/dees-chart-log/index.ts
Normal file
1
ts_web/elements/00group-chart/dees-chart-log/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-chart-log.js';
|
||||||
3
ts_web/elements/00group-chart/index.ts
Normal file
3
ts_web/elements/00group-chart/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
// Chart Components
|
||||||
|
export * from './dees-chart-area/index.js';
|
||||||
|
export * from './dees-chart-log/index.js';
|
||||||
@@ -8,14 +8,14 @@ import {
|
|||||||
state,
|
state,
|
||||||
cssManager,
|
cssManager,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import { cssGeistFontFamily, cssMonoFontFamily } from './00fonts.js';
|
import { cssGeistFontFamily, cssMonoFontFamily } from '../../00fonts.js';
|
||||||
|
|
||||||
import hlight from 'highlight.js';
|
import hlight from 'highlight.js';
|
||||||
|
|
||||||
import * as smartstring from '@push.rocks/smartstring';
|
import * as smartstring from '@push.rocks/smartstring';
|
||||||
|
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -28,13 +28,13 @@ export class DeesDataviewCodebox extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public progLang: string = 'typescript';
|
accessor progLang: string = 'typescript';
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String,
|
type: String,
|
||||||
reflect: true,
|
reflect: true,
|
||||||
})
|
})
|
||||||
public codeToDisplay: string = '';
|
accessor codeToDisplay: string = '';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@@ -228,7 +228,6 @@ export class DeesDataviewCodebox extends DeesElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@state()
|
|
||||||
private codeToDisplayStore = '';
|
private codeToDisplayStore = '';
|
||||||
|
|
||||||
public async updated(_changedProperties) {
|
public async updated(_changedProperties) {
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-dataview-codebox.js';
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import * as colors from './00colors.js';
|
import * as colors from '../../00colors.js';
|
||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../../00plugins.js';
|
||||||
|
|
||||||
import { demoFunc } from './dees-dataview-statusobject.demo.js';
|
import { demoFunc } from './dees-dataview-statusobject.demo.js';
|
||||||
import {
|
import {
|
||||||
@@ -15,7 +15,7 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import * as tsclass from '@tsclass/tsclass';
|
import * as tsclass from '@tsclass/tsclass';
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -27,7 +27,7 @@ declare global {
|
|||||||
export class DeesDataviewStatusobject extends DeesElement {
|
export class DeesDataviewStatusobject extends DeesElement {
|
||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property({ type: Object }) statusObject: tsclass.code.IStatusObject;
|
@property({ type: Object }) accessor statusObject: tsclass.code.IStatusObject;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-dataview-statusobject.js';
|
||||||
3
ts_web/elements/00group-dataview/index.ts
Normal file
3
ts_web/elements/00group-dataview/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
// Data View Components
|
||||||
|
export * from './dees-dataview-codebox/index.js';
|
||||||
|
export * from './dees-dataview-statusobject/index.js';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-editor-markdown.js';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-editor-markdownoutlet.js';
|
||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
cssManager,
|
cssManager,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
|
import { MONACO_VERSION } from './version.js';
|
||||||
|
|
||||||
import type * as monaco from 'monaco-editor';
|
import type * as monaco from 'monaco-editor';
|
||||||
|
|
||||||
@@ -32,17 +33,17 @@ export class DeesEditor extends DeesElement {
|
|||||||
@property({
|
@property({
|
||||||
type: String
|
type: String
|
||||||
})
|
})
|
||||||
public content = "function hello() {\n\talert('Hello world!');\n}";
|
accessor content = "function hello() {\n\talert('Hello world!');\n}";
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Object
|
type: Object
|
||||||
})
|
})
|
||||||
public contentSubject = new domtools.plugins.smartrx.rxjs.Subject<string>();
|
accessor contentSubject = new domtools.plugins.smartrx.rxjs.Subject<string>();
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Boolean
|
type: Boolean
|
||||||
})
|
})
|
||||||
public wordWrap: monaco.editor.IStandaloneEditorConstructionOptions['wordWrap'] = 'off';
|
accessor wordWrap: monaco.editor.IStandaloneEditorConstructionOptions['wordWrap'] = 'off';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@@ -80,10 +81,11 @@ export class DeesEditor extends DeesElement {
|
|||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
super.firstUpdated(_changedProperties);
|
super.firstUpdated(_changedProperties);
|
||||||
const container = this.shadowRoot.getElementById('container');
|
const container = this.shadowRoot.getElementById('container');
|
||||||
|
const monacoCdnBase = `https://cdn.jsdelivr.net/npm/monaco-editor@${MONACO_VERSION}`;
|
||||||
|
|
||||||
if (!DeesEditor.monacoDeferred) {
|
if (!DeesEditor.monacoDeferred) {
|
||||||
DeesEditor.monacoDeferred = domtools.plugins.smartpromise.defer();
|
DeesEditor.monacoDeferred = domtools.plugins.smartpromise.defer();
|
||||||
const scriptUrl = `https://cdn.jsdelivr.net/npm/monaco-editor/min/vs/loader.js`;
|
const scriptUrl = `${monacoCdnBase}/min/vs/loader.js`;
|
||||||
const script = document.createElement('script');
|
const script = document.createElement('script');
|
||||||
script.src = scriptUrl;
|
script.src = scriptUrl;
|
||||||
script.onload = () => {
|
script.onload = () => {
|
||||||
@@ -94,7 +96,7 @@ export class DeesEditor extends DeesElement {
|
|||||||
await DeesEditor.monacoDeferred.promise;
|
await DeesEditor.monacoDeferred.promise;
|
||||||
|
|
||||||
(window as any).require.config({
|
(window as any).require.config({
|
||||||
paths: { vs: 'https://cdn.jsdelivr.net/npm/monaco-editor/min/vs' },
|
paths: { vs: `${monacoCdnBase}/min/vs` },
|
||||||
});
|
});
|
||||||
(window as any).require(['vs/editor/editor.main'], async () => {
|
(window as any).require(['vs/editor/editor.main'], async () => {
|
||||||
const editor = ((window as any).monaco.editor as typeof monaco.editor).create(container, {
|
const editor = ((window as any).monaco.editor as typeof monaco.editor).create(container, {
|
||||||
@@ -109,7 +111,7 @@ export class DeesEditor extends DeesElement {
|
|||||||
this.editorDeferred.resolve(editor);
|
this.editorDeferred.resolve(editor);
|
||||||
});
|
});
|
||||||
const css = await (
|
const css = await (
|
||||||
await fetch('https://cdn.jsdelivr.net/npm/monaco-editor/min/vs/editor/editor.main.css')
|
await fetch(`${monacoCdnBase}/min/vs/editor/editor.main.css`)
|
||||||
).text();
|
).text();
|
||||||
const styleElement = document.createElement('style');
|
const styleElement = document.createElement('style');
|
||||||
styleElement.textContent = css;
|
styleElement.textContent = css;
|
||||||
1
ts_web/elements/00group-editor/dees-editor/index.ts
Normal file
1
ts_web/elements/00group-editor/dees-editor/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-editor.js';
|
||||||
2
ts_web/elements/00group-editor/dees-editor/version.ts
Normal file
2
ts_web/elements/00group-editor/dees-editor/version.ts
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
// Auto-generated by scripts/update-monaco-version.cjs
|
||||||
|
export const MONACO_VERSION = '0.52.2';
|
||||||
4
ts_web/elements/00group-editor/index.ts
Normal file
4
ts_web/elements/00group-editor/index.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
// Editor Components
|
||||||
|
export * from './dees-editor/index.js';
|
||||||
|
export * from './dees-editor-markdown/index.js';
|
||||||
|
export * from './dees-editor-markdownoutlet/index.js';
|
||||||
@@ -7,7 +7,7 @@ import {
|
|||||||
cssManager,
|
cssManager,
|
||||||
property,
|
property,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import type { DeesForm } from './dees-form.js';
|
import type { DeesForm } from '../dees-form/dees-form.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -23,17 +23,17 @@ export class DeesFormSubmit extends DeesElement {
|
|||||||
type: Boolean,
|
type: Boolean,
|
||||||
reflect: true,
|
reflect: true,
|
||||||
})
|
})
|
||||||
public disabled = false;
|
accessor disabled = false;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String,
|
type: String,
|
||||||
})
|
})
|
||||||
public text: string;
|
accessor text: string;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String,
|
type: String,
|
||||||
})
|
})
|
||||||
public status: 'normal' | 'pending' | 'success' | 'error' = 'normal';
|
accessor status: 'normal' | 'pending' | 'success' | 'error' = 'normal';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
@@ -57,9 +57,10 @@ export class DeesFormSubmit extends DeesElement {
|
|||||||
if (this.disabled) {
|
if (this.disabled) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const parentElement: DeesForm = this.parentElement as DeesForm;
|
// Walk up the DOM tree to find the nearest dees-form element
|
||||||
if (parentElement && parentElement.gatherAndDispatch) {
|
const parentFormElement = this.closest('dees-form') as DeesForm;
|
||||||
parentElement.gatherAndDispatch();
|
if (parentFormElement && parentFormElement.gatherAndDispatch) {
|
||||||
|
parentFormElement.gatherAndDispatch();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
1
ts_web/elements/00group-form/dees-form-submit/index.ts
Normal file
1
ts_web/elements/00group-form/dees-form-submit/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-form-submit.js';
|
||||||
@@ -8,19 +8,19 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
|
|
||||||
import { DeesInputCheckbox } from './dees-input-checkbox.js';
|
import { DeesInputCheckbox } from '../../00group-input/dees-input-checkbox/dees-input-checkbox.js';
|
||||||
import { DeesInputDatepicker } from './dees-input-datepicker.js';
|
import { DeesInputDatepicker } from '../../00group-input/dees-input-datepicker/index.js';
|
||||||
import { DeesInputText } from './dees-input-text.js';
|
import { DeesInputText } from '../../00group-input/dees-input-text/dees-input-text.js';
|
||||||
import { DeesInputQuantitySelector } from './dees-input-quantityselector.js';
|
import { DeesInputQuantitySelector } from '../../00group-input/dees-input-quantityselector/dees-input-quantityselector.js';
|
||||||
import { DeesInputRadiogroup } from './dees-input-radiogroup.js';
|
import { DeesInputRadiogroup } from '../../00group-input/dees-input-radiogroup/dees-input-radiogroup.js';
|
||||||
import { DeesInputDropdown } from './dees-input-dropdown.js';
|
import { DeesInputDropdown } from '../../00group-input/dees-input-dropdown/dees-input-dropdown.js';
|
||||||
import { DeesInputFileupload } from './dees-input-fileupload.js';
|
import { DeesInputFileupload } from '../../00group-input/dees-input-fileupload/index.js';
|
||||||
import { DeesInputIban } from './dees-input-iban.js';
|
import { DeesInputIban } from '../../00group-input/dees-input-iban/dees-input-iban.js';
|
||||||
import { DeesInputMultitoggle } from './dees-input-multitoggle.js';
|
import { DeesInputMultitoggle } from '../../00group-input/dees-input-multitoggle/dees-input-multitoggle.js';
|
||||||
import { DeesInputPhone } from './dees-input-phone.js';
|
import { DeesInputPhone } from '../../00group-input/dees-input-phone/dees-input-phone.js';
|
||||||
import { DeesInputTypelist } from './dees-input-typelist.js';
|
import { DeesInputTypelist } from '../../00group-input/dees-input-typelist/dees-input-typelist.js';
|
||||||
import { DeesFormSubmit } from './dees-form-submit.js';
|
import { DeesFormSubmit } from '../dees-form-submit/dees-form-submit.js';
|
||||||
import { DeesTable } from './dees-table.js';
|
import { DeesTable } from '../../dees-table/index.js';
|
||||||
import { demoFunc } from './dees-form.demo.js';
|
import { demoFunc } from './dees-form.demo.js';
|
||||||
|
|
||||||
// Unified set for form input types
|
// Unified set for form input types
|
||||||
@@ -72,7 +72,7 @@ export class DeesForm extends DeesElement {
|
|||||||
* When true, sets all child inputs to horizontal layout
|
* When true, sets all child inputs to horizontal layout
|
||||||
*/
|
*/
|
||||||
@property({ type: Boolean, reflect: true, attribute: 'horizontal-layout' })
|
@property({ type: Boolean, reflect: true, attribute: 'horizontal-layout' })
|
||||||
public horizontalLayout: boolean = false;
|
accessor horizontalLayout: boolean = false;
|
||||||
|
|
||||||
public render(): TemplateResult {
|
public render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
1
ts_web/elements/00group-form/dees-form/index.ts
Normal file
1
ts_web/elements/00group-form/dees-form/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-form.js';
|
||||||
3
ts_web/elements/00group-form/index.ts
Normal file
3
ts_web/elements/00group-form/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
// Form Components
|
||||||
|
export * from './dees-form/index.js';
|
||||||
|
export * from './dees-form-submit/index.js';
|
||||||
@@ -19,31 +19,31 @@ export abstract class DeesInputBase<T = any> extends DeesElement {
|
|||||||
* - auto: Detect from parent context
|
* - auto: Detect from parent context
|
||||||
*/
|
*/
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public layoutMode: 'vertical' | 'horizontal' | 'auto' = 'auto';
|
accessor layoutMode: 'vertical' | 'horizontal' | 'auto' = 'auto';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Position of the label relative to the input
|
* Position of the label relative to the input
|
||||||
*/
|
*/
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public labelPosition: 'top' | 'left' | 'right' | 'none' = 'top';
|
accessor labelPosition: 'top' | 'left' | 'right' | 'none' = 'top';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Common properties for all inputs
|
* Common properties for all inputs
|
||||||
*/
|
*/
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public key: string;
|
accessor key: string;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public label: string;
|
accessor label: string;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public required: boolean = false;
|
accessor required: boolean = false;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public disabled: boolean = false;
|
accessor disabled: boolean = false;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public description: string;
|
accessor description: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Common styles for all input components
|
* Common styles for all input components
|
||||||
1
ts_web/elements/00group-input/dees-input-base/index.ts
Normal file
1
ts_web/elements/00group-input/dees-input-base/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-input-base.js';
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import { html, css, cssManager } from '@design.estate/dees-element';
|
import { html, css, cssManager } from '@design.estate/dees-element';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
import './dees-panel.js';
|
import '../../dees-panel/dees-panel.js';
|
||||||
import type { DeesInputCheckbox } from './dees-input-checkbox.js';
|
import type { DeesInputCheckbox } from '../dees-input-checkbox/dees-input-checkbox.js';
|
||||||
import './dees-button.js';
|
import '../../00group-button/dees-button/dees-button.js';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
<dees-demowrapper .runAfterRender=${async (elementArg: HTMLElement) => {
|
<dees-demowrapper .runAfterRender=${async (elementArg: HTMLElement) => {
|
||||||
@@ -6,9 +6,9 @@ import {
|
|||||||
css,
|
css,
|
||||||
cssManager,
|
cssManager,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import { DeesInputBase } from './dees-input-base.js';
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
import { demoFunc } from './dees-input-checkbox.demo.js';
|
import { demoFunc } from './dees-input-checkbox.demo.js';
|
||||||
import { cssGeistFontFamily } from './00fonts.js';
|
import { cssGeistFontFamily } from '../../00fonts.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -26,7 +26,10 @@ export class DeesInputCheckbox extends DeesInputBase<DeesInputCheckbox> {
|
|||||||
@property({
|
@property({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
})
|
})
|
||||||
public value: boolean = false;
|
accessor value: boolean = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor indeterminate: boolean = false;
|
||||||
|
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
@@ -166,6 +169,14 @@ export class DeesInputCheckbox extends DeesInputBase<DeesInputCheckbox> {
|
|||||||
</svg>
|
</svg>
|
||||||
</span>
|
</span>
|
||||||
`
|
`
|
||||||
|
: this.indeterminate
|
||||||
|
? html`
|
||||||
|
<span class="checkmark">
|
||||||
|
<svg viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M5 12H19" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
</span>
|
||||||
|
`
|
||||||
: html``}
|
: html``}
|
||||||
</div>
|
</div>
|
||||||
<div class="label-container">
|
<div class="label-container">
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-input-checkbox.js';
|
||||||
624
ts_web/elements/00group-input/dees-input-datepicker/component.ts
Normal file
624
ts_web/elements/00group-input/dees-input-datepicker/component.ts
Normal file
@@ -0,0 +1,624 @@
|
|||||||
|
import {
|
||||||
|
customElement,
|
||||||
|
type TemplateResult,
|
||||||
|
property,
|
||||||
|
state,
|
||||||
|
} from '@design.estate/dees-element';
|
||||||
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
|
import { demoFunc } from './demo.js';
|
||||||
|
import { datepickerStyles } from './styles.js';
|
||||||
|
import { renderDatepicker } from './template.js';
|
||||||
|
import type { IDateEvent } from './types.js';
|
||||||
|
import '../../dees-icon/dees-icon.js';
|
||||||
|
import '../../dees-label/dees-label.js';
|
||||||
|
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
'dees-input-datepicker': DeesInputDatepicker;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@customElement('dees-input-datepicker')
|
||||||
|
export class DeesInputDatepicker extends DeesInputBase<DeesInputDatepicker> {
|
||||||
|
public static demo = demoFunc;
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor value: string = '';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor enableTime: boolean = false;
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor timeFormat: '24h' | '12h' = '24h';
|
||||||
|
|
||||||
|
@property({ type: Number })
|
||||||
|
accessor minuteIncrement: number = 1;
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor dateFormat: string = 'YYYY-MM-DD';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor minDate: string = '';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor maxDate: string = '';
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor disabledDates: string[] = [];
|
||||||
|
|
||||||
|
@property({ type: Number })
|
||||||
|
accessor weekStartsOn: 0 | 1 = 1; // Default to Monday
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor placeholder: string = 'YYYY-MM-DD';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor enableTimezone: boolean = false;
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor timezone: string = Intl.DateTimeFormat().resolvedOptions().timeZone;
|
||||||
|
|
||||||
|
@property({ type: Array })
|
||||||
|
accessor events: IDateEvent[] = [];
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor isOpened: boolean = false;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor opensToTop: boolean = false;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor selectedDate: Date | null = null;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor viewDate: Date = new Date();
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor selectedHour: number = 0;
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor selectedMinute: number = 0;
|
||||||
|
|
||||||
|
public static styles = datepickerStyles;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public getTimezones(): { value: string; label: string }[] {
|
||||||
|
// Common timezones with their display names
|
||||||
|
return [
|
||||||
|
{ value: 'UTC', label: 'UTC (Coordinated Universal Time)' },
|
||||||
|
{ value: 'America/New_York', label: 'Eastern Time (US & Canada)' },
|
||||||
|
{ value: 'America/Chicago', label: 'Central Time (US & Canada)' },
|
||||||
|
{ value: 'America/Denver', label: 'Mountain Time (US & Canada)' },
|
||||||
|
{ value: 'America/Los_Angeles', label: 'Pacific Time (US & Canada)' },
|
||||||
|
{ value: 'America/Phoenix', label: 'Arizona' },
|
||||||
|
{ value: 'America/Anchorage', label: 'Alaska' },
|
||||||
|
{ value: 'Pacific/Honolulu', label: 'Hawaii' },
|
||||||
|
{ value: 'Europe/London', label: 'London' },
|
||||||
|
{ value: 'Europe/Paris', label: 'Paris' },
|
||||||
|
{ value: 'Europe/Berlin', label: 'Berlin' },
|
||||||
|
{ value: 'Europe/Moscow', label: 'Moscow' },
|
||||||
|
{ value: 'Asia/Dubai', label: 'Dubai' },
|
||||||
|
{ value: 'Asia/Kolkata', label: 'India Standard Time' },
|
||||||
|
{ value: 'Asia/Shanghai', label: 'China Standard Time' },
|
||||||
|
{ value: 'Asia/Tokyo', label: 'Tokyo' },
|
||||||
|
{ value: 'Australia/Sydney', label: 'Sydney' },
|
||||||
|
{ value: 'Pacific/Auckland', label: 'Auckland' },
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public render(): TemplateResult {
|
||||||
|
return renderDatepicker(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
async connectedCallback() {
|
||||||
|
super.connectedCallback();
|
||||||
|
this.handleClickOutside = this.handleClickOutside.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
async disconnectedCallback() {
|
||||||
|
await super.disconnectedCallback();
|
||||||
|
document.removeEventListener('click', this.handleClickOutside);
|
||||||
|
}
|
||||||
|
|
||||||
|
async firstUpdated() {
|
||||||
|
// Initialize with empty value if not set
|
||||||
|
if (!this.value) {
|
||||||
|
this.value = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize view date and selected time
|
||||||
|
if (this.value) {
|
||||||
|
try {
|
||||||
|
const date = new Date(this.value);
|
||||||
|
if (!isNaN(date.getTime())) {
|
||||||
|
this.selectedDate = date;
|
||||||
|
this.viewDate = new Date(date);
|
||||||
|
this.selectedHour = date.getHours();
|
||||||
|
this.selectedMinute = date.getMinutes();
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Invalid date
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const now = new Date();
|
||||||
|
this.viewDate = new Date(now);
|
||||||
|
this.selectedHour = now.getHours();
|
||||||
|
this.selectedMinute = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public formatDate(isoString: string): string {
|
||||||
|
if (!isoString) return '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const date = new Date(isoString);
|
||||||
|
if (isNaN(date.getTime())) return '';
|
||||||
|
|
||||||
|
let formatted = this.dateFormat;
|
||||||
|
|
||||||
|
// Basic date formatting
|
||||||
|
const day = date.getDate().toString().padStart(2, '0');
|
||||||
|
const month = (date.getMonth() + 1).toString().padStart(2, '0');
|
||||||
|
const year = date.getFullYear().toString();
|
||||||
|
|
||||||
|
// Replace in correct order to avoid conflicts
|
||||||
|
formatted = formatted.replace('YYYY', year);
|
||||||
|
formatted = formatted.replace('YY', year.slice(-2));
|
||||||
|
formatted = formatted.replace('MM', month);
|
||||||
|
formatted = formatted.replace('DD', day);
|
||||||
|
|
||||||
|
// Time formatting if enabled
|
||||||
|
if (this.enableTime) {
|
||||||
|
const hours24 = date.getHours();
|
||||||
|
const hours12 = hours24 === 0 ? 12 : hours24 > 12 ? hours24 - 12 : hours24;
|
||||||
|
const minutes = date.getMinutes().toString().padStart(2, '0');
|
||||||
|
const ampm = hours24 >= 12 ? 'PM' : 'AM';
|
||||||
|
|
||||||
|
if (this.timeFormat === '12h') {
|
||||||
|
formatted += ` ${hours12}:${minutes} ${ampm}`;
|
||||||
|
} else {
|
||||||
|
formatted += ` ${hours24.toString().padStart(2, '0')}:${minutes}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Timezone formatting if enabled
|
||||||
|
if (this.enableTimezone) {
|
||||||
|
const formatter = new Intl.DateTimeFormat('en-US', {
|
||||||
|
timeZoneName: 'short',
|
||||||
|
timeZone: this.timezone
|
||||||
|
});
|
||||||
|
const parts = formatter.formatToParts(date);
|
||||||
|
const tzPart = parts.find(part => part.type === 'timeZoneName');
|
||||||
|
if (tzPart) {
|
||||||
|
formatted += ` ${tzPart.value}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatted;
|
||||||
|
} catch {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleClickOutside = (event: MouseEvent) => {
|
||||||
|
const path = event.composedPath();
|
||||||
|
if (!path.includes(this)) {
|
||||||
|
this.isOpened = false;
|
||||||
|
document.removeEventListener('click', this.handleClickOutside);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public async toggleCalendar(): Promise<void> {
|
||||||
|
if (this.disabled) return;
|
||||||
|
|
||||||
|
this.isOpened = !this.isOpened;
|
||||||
|
|
||||||
|
if (this.isOpened) {
|
||||||
|
// Check available space and set position
|
||||||
|
const inputContainer = this.shadowRoot!.querySelector('.input-container') as HTMLElement;
|
||||||
|
const rect = inputContainer.getBoundingClientRect();
|
||||||
|
const spaceBelow = window.innerHeight - rect.bottom;
|
||||||
|
const spaceAbove = rect.top;
|
||||||
|
|
||||||
|
// Determine if we should open upwards (approximate height of 400px)
|
||||||
|
this.opensToTop = spaceBelow < 400 && spaceAbove > spaceBelow;
|
||||||
|
|
||||||
|
// Add click outside listener
|
||||||
|
setTimeout(() => {
|
||||||
|
document.addEventListener('click', this.handleClickOutside);
|
||||||
|
}, 0);
|
||||||
|
} else {
|
||||||
|
document.removeEventListener('click', this.handleClickOutside);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public getDaysInMonth(): Date[] {
|
||||||
|
const year = this.viewDate.getFullYear();
|
||||||
|
const month = this.viewDate.getMonth();
|
||||||
|
const firstDay = new Date(year, month, 1);
|
||||||
|
const lastDay = new Date(year, month + 1, 0);
|
||||||
|
const days: Date[] = [];
|
||||||
|
|
||||||
|
// Adjust for week start
|
||||||
|
const startOffset = this.weekStartsOn === 1
|
||||||
|
? (firstDay.getDay() === 0 ? 6 : firstDay.getDay() - 1)
|
||||||
|
: firstDay.getDay();
|
||||||
|
|
||||||
|
// Add days from previous month
|
||||||
|
for (let i = startOffset; i > 0; i--) {
|
||||||
|
days.push(new Date(year, month, 1 - i));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add days of current month
|
||||||
|
for (let i = 1; i <= lastDay.getDate(); i++) {
|
||||||
|
days.push(new Date(year, month, i));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add days from next month to complete the grid (6 rows)
|
||||||
|
const remainingDays = 42 - days.length;
|
||||||
|
for (let i = 1; i <= remainingDays; i++) {
|
||||||
|
days.push(new Date(year, month + 1, i));
|
||||||
|
}
|
||||||
|
|
||||||
|
return days;
|
||||||
|
}
|
||||||
|
|
||||||
|
public isToday(date: Date): boolean {
|
||||||
|
const today = new Date();
|
||||||
|
return date.getDate() === today.getDate() &&
|
||||||
|
date.getMonth() === today.getMonth() &&
|
||||||
|
date.getFullYear() === today.getFullYear();
|
||||||
|
}
|
||||||
|
|
||||||
|
public isSelected(date: Date): boolean {
|
||||||
|
if (!this.selectedDate) return false;
|
||||||
|
return date.getDate() === this.selectedDate.getDate() &&
|
||||||
|
date.getMonth() === this.selectedDate.getMonth() &&
|
||||||
|
date.getFullYear() === this.selectedDate.getFullYear();
|
||||||
|
}
|
||||||
|
|
||||||
|
public isDisabled(date: Date): boolean {
|
||||||
|
// Check min date
|
||||||
|
if (this.minDate) {
|
||||||
|
const min = new Date(this.minDate);
|
||||||
|
if (date < min) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check max date
|
||||||
|
if (this.maxDate) {
|
||||||
|
const max = new Date(this.maxDate);
|
||||||
|
if (date > max) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check disabled dates
|
||||||
|
if (this.disabledDates && this.disabledDates.length > 0) {
|
||||||
|
return this.disabledDates.some(disabledStr => {
|
||||||
|
try {
|
||||||
|
const disabled = new Date(disabledStr);
|
||||||
|
return date.getDate() === disabled.getDate() &&
|
||||||
|
date.getMonth() === disabled.getMonth() &&
|
||||||
|
date.getFullYear() === disabled.getFullYear();
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getEventsForDate(date: Date): IDateEvent[] {
|
||||||
|
if (!this.events || this.events.length === 0) return [];
|
||||||
|
|
||||||
|
const dateStr = `${date.getFullYear()}-${(date.getMonth() + 1).toString().padStart(2, '0')}-${date.getDate().toString().padStart(2, '0')}`;
|
||||||
|
return this.events.filter(event => event.date === dateStr);
|
||||||
|
}
|
||||||
|
|
||||||
|
public selectDate(date: Date): void {
|
||||||
|
this.selectedDate = new Date(
|
||||||
|
date.getFullYear(),
|
||||||
|
date.getMonth(),
|
||||||
|
date.getDate(),
|
||||||
|
this.selectedHour,
|
||||||
|
this.selectedMinute
|
||||||
|
);
|
||||||
|
|
||||||
|
this.value = this.formatValueWithTimezone(this.selectedDate);
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
|
||||||
|
if (!this.enableTime) {
|
||||||
|
this.isOpened = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public selectToday(): void {
|
||||||
|
const today = new Date();
|
||||||
|
this.selectedDate = today;
|
||||||
|
this.viewDate = new Date(today);
|
||||||
|
this.selectedHour = today.getHours();
|
||||||
|
this.selectedMinute = today.getMinutes();
|
||||||
|
|
||||||
|
this.value = this.formatValueWithTimezone(this.selectedDate);
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
|
||||||
|
if (!this.enableTime) {
|
||||||
|
this.isOpened = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public clear(): void {
|
||||||
|
this.value = '';
|
||||||
|
this.selectedDate = null;
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
this.isOpened = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public previousMonth(): void {
|
||||||
|
this.viewDate = new Date(this.viewDate.getFullYear(), this.viewDate.getMonth() - 1, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public nextMonth(): void {
|
||||||
|
this.viewDate = new Date(this.viewDate.getFullYear(), this.viewDate.getMonth() + 1, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleHourInput(e: InputEvent): void {
|
||||||
|
const input = e.target as HTMLInputElement;
|
||||||
|
let value = parseInt(input.value) || 0;
|
||||||
|
|
||||||
|
if (this.timeFormat === '12h') {
|
||||||
|
value = Math.max(1, Math.min(12, value));
|
||||||
|
// Convert to 24h format
|
||||||
|
if (this.selectedHour >= 12 && value !== 12) {
|
||||||
|
this.selectedHour = value + 12;
|
||||||
|
} else if (this.selectedHour < 12 && value === 12) {
|
||||||
|
this.selectedHour = 0;
|
||||||
|
} else {
|
||||||
|
this.selectedHour = value;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.selectedHour = Math.max(0, Math.min(23, value));
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updateSelectedDateTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleMinuteInput(e: InputEvent): void {
|
||||||
|
const input = e.target as HTMLInputElement;
|
||||||
|
let value = parseInt(input.value) || 0;
|
||||||
|
value = Math.max(0, Math.min(59, value));
|
||||||
|
|
||||||
|
if (this.minuteIncrement && this.minuteIncrement > 1) {
|
||||||
|
value = Math.round(value / this.minuteIncrement) * this.minuteIncrement;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.selectedMinute = value;
|
||||||
|
this.updateSelectedDateTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
public setAMPM(period: 'am' | 'pm'): void {
|
||||||
|
if (period === 'am' && this.selectedHour >= 12) {
|
||||||
|
this.selectedHour -= 12;
|
||||||
|
} else if (period === 'pm' && this.selectedHour < 12) {
|
||||||
|
this.selectedHour += 12;
|
||||||
|
}
|
||||||
|
this.updateSelectedDateTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
private updateSelectedDateTime(): void {
|
||||||
|
if (this.selectedDate) {
|
||||||
|
this.selectedDate = new Date(
|
||||||
|
this.selectedDate.getFullYear(),
|
||||||
|
this.selectedDate.getMonth(),
|
||||||
|
this.selectedDate.getDate(),
|
||||||
|
this.selectedHour,
|
||||||
|
this.selectedMinute
|
||||||
|
);
|
||||||
|
this.value = this.formatValueWithTimezone(this.selectedDate);
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleTimezoneChange(e: Event): void {
|
||||||
|
const select = e.target as HTMLSelectElement;
|
||||||
|
this.timezone = select.value;
|
||||||
|
this.updateSelectedDateTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
private formatValueWithTimezone(date: Date): string {
|
||||||
|
if (!this.enableTimezone) {
|
||||||
|
return date.toISOString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the date with timezone offset
|
||||||
|
const formatter = new Intl.DateTimeFormat('en-US', {
|
||||||
|
year: 'numeric',
|
||||||
|
month: '2-digit',
|
||||||
|
day: '2-digit',
|
||||||
|
hour: '2-digit',
|
||||||
|
minute: '2-digit',
|
||||||
|
second: '2-digit',
|
||||||
|
hour12: false,
|
||||||
|
timeZone: this.timezone,
|
||||||
|
timeZoneName: 'short'
|
||||||
|
});
|
||||||
|
|
||||||
|
const parts = formatter.formatToParts(date);
|
||||||
|
const dateParts: any = {};
|
||||||
|
parts.forEach(part => {
|
||||||
|
dateParts[part.type] = part.value;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create ISO-like format with timezone
|
||||||
|
const isoString = `${dateParts.year}-${dateParts.month}-${dateParts.day}T${dateParts.hour}:${dateParts.minute}:${dateParts.second}`;
|
||||||
|
|
||||||
|
// Get timezone offset
|
||||||
|
const tzOffset = this.getTimezoneOffset(date, this.timezone);
|
||||||
|
return `${isoString}${tzOffset}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getTimezoneOffset(date: Date, timezone: string): string {
|
||||||
|
// Create a date in the target timezone
|
||||||
|
const tzDate = new Date(date.toLocaleString('en-US', { timeZone: timezone }));
|
||||||
|
const utcDate = new Date(date.toLocaleString('en-US', { timeZone: 'UTC' }));
|
||||||
|
|
||||||
|
const offsetMinutes = (tzDate.getTime() - utcDate.getTime()) / (1000 * 60);
|
||||||
|
const hours = Math.floor(Math.abs(offsetMinutes) / 60);
|
||||||
|
const minutes = Math.abs(offsetMinutes) % 60;
|
||||||
|
const sign = offsetMinutes >= 0 ? '+' : '-';
|
||||||
|
|
||||||
|
return `${sign}${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleKeydown(e: KeyboardEvent): void {
|
||||||
|
if (e.key === 'Enter' || e.key === ' ') {
|
||||||
|
e.preventDefault();
|
||||||
|
this.toggleCalendar();
|
||||||
|
} else if (e.key === 'Escape' && this.isOpened) {
|
||||||
|
e.preventDefault();
|
||||||
|
this.isOpened = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public clearValue(e: Event): void {
|
||||||
|
e.stopPropagation();
|
||||||
|
this.value = '';
|
||||||
|
this.selectedDate = null;
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleManualInput(e: InputEvent): void {
|
||||||
|
const input = e.target as HTMLInputElement;
|
||||||
|
const inputValue = input.value.trim();
|
||||||
|
|
||||||
|
if (!inputValue) {
|
||||||
|
// Clear the value if input is empty
|
||||||
|
this.value = '';
|
||||||
|
this.selectedDate = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const parsedDate = this.parseManualDate(inputValue);
|
||||||
|
if (parsedDate && !isNaN(parsedDate.getTime())) {
|
||||||
|
// Update internal state without triggering re-render of input
|
||||||
|
this.value = parsedDate.toISOString();
|
||||||
|
this.selectedDate = parsedDate;
|
||||||
|
this.viewDate = new Date(parsedDate);
|
||||||
|
this.selectedHour = parsedDate.getHours();
|
||||||
|
this.selectedMinute = parsedDate.getMinutes();
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public handleInputBlur(e: FocusEvent): void {
|
||||||
|
const input = e.target as HTMLInputElement;
|
||||||
|
const inputValue = input.value.trim();
|
||||||
|
|
||||||
|
if (!inputValue) {
|
||||||
|
this.value = '';
|
||||||
|
this.selectedDate = null;
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const parsedDate = this.parseManualDate(inputValue);
|
||||||
|
if (parsedDate && !isNaN(parsedDate.getTime())) {
|
||||||
|
this.value = parsedDate.toISOString();
|
||||||
|
this.selectedDate = parsedDate;
|
||||||
|
this.viewDate = new Date(parsedDate);
|
||||||
|
this.selectedHour = parsedDate.getHours();
|
||||||
|
this.selectedMinute = parsedDate.getMinutes();
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
// Update the input with formatted date
|
||||||
|
input.value = this.formatDate(this.value);
|
||||||
|
} else {
|
||||||
|
// Revert to previous valid value on blur if parsing failed
|
||||||
|
input.value = this.formatDate(this.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private parseManualDate(input: string): Date | null {
|
||||||
|
if (!input) return null;
|
||||||
|
|
||||||
|
// Split date and time parts if present
|
||||||
|
const parts = input.split(' ');
|
||||||
|
let datePart = parts[0];
|
||||||
|
let timePart = parts[1] || '';
|
||||||
|
|
||||||
|
let parsedDate: Date | null = null;
|
||||||
|
|
||||||
|
// Try different date formats
|
||||||
|
// Format 1: YYYY-MM-DD (ISO-like)
|
||||||
|
const isoMatch = datePart.match(/^(\d{4})-(\d{1,2})-(\d{1,2})$/);
|
||||||
|
if (isoMatch) {
|
||||||
|
const [_, year, month, day] = isoMatch;
|
||||||
|
parsedDate = new Date(parseInt(year), parseInt(month) - 1, parseInt(day));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format 2: DD.MM.YYYY (European)
|
||||||
|
if (!parsedDate) {
|
||||||
|
const euMatch = datePart.match(/^(\d{1,2})\.(\d{1,2})\.(\d{4})$/);
|
||||||
|
if (euMatch) {
|
||||||
|
const [_, day, month, year] = euMatch;
|
||||||
|
parsedDate = new Date(parseInt(year), parseInt(month) - 1, parseInt(day));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format 3: MM/DD/YYYY (US)
|
||||||
|
if (!parsedDate) {
|
||||||
|
const usMatch = datePart.match(/^(\d{1,2})\/(\d{1,2})\/(\d{4})$/);
|
||||||
|
if (usMatch) {
|
||||||
|
const [_, month, day, year] = usMatch;
|
||||||
|
parsedDate = new Date(parseInt(year), parseInt(month) - 1, parseInt(day));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no date was parsed, return null
|
||||||
|
if (!parsedDate || isNaN(parsedDate.getTime())) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse time if present (HH:MM format)
|
||||||
|
if (timePart) {
|
||||||
|
const timeMatch = timePart.match(/^(\d{1,2}):(\d{2})$/);
|
||||||
|
if (timeMatch) {
|
||||||
|
const [_, hours, minutes] = timeMatch;
|
||||||
|
parsedDate.setHours(parseInt(hours));
|
||||||
|
parsedDate.setMinutes(parseInt(minutes));
|
||||||
|
}
|
||||||
|
} else if (!this.enableTime) {
|
||||||
|
// If time is not enabled and not provided, use current time
|
||||||
|
const now = new Date();
|
||||||
|
parsedDate.setHours(now.getHours());
|
||||||
|
parsedDate.setMinutes(now.getMinutes());
|
||||||
|
parsedDate.setSeconds(0);
|
||||||
|
parsedDate.setMilliseconds(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsedDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getValue(): string {
|
||||||
|
return this.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public setValue(value: string): void {
|
||||||
|
this.value = value;
|
||||||
|
if (value) {
|
||||||
|
try {
|
||||||
|
const date = new Date(value);
|
||||||
|
if (!isNaN(date.getTime())) {
|
||||||
|
this.selectedDate = date;
|
||||||
|
this.viewDate = new Date(date);
|
||||||
|
this.selectedHour = date.getHours();
|
||||||
|
this.selectedMinute = date.getMinutes();
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Invalid date
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import { html, css } from '@design.estate/dees-element';
|
import { html, css } from '@design.estate/dees-element';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
import './dees-panel.js';
|
import '../../dees-panel/dees-panel.js';
|
||||||
import './dees-input-datepicker.js';
|
import './component.js';
|
||||||
import type { DeesInputDatepicker } from './dees-input-datepicker.js';
|
import type { DeesInputDatepicker } from './component.js';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
<style>
|
<style>
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './component.js';
|
||||||
514
ts_web/elements/00group-input/dees-input-datepicker/styles.ts
Normal file
514
ts_web/elements/00group-input/dees-input-datepicker/styles.ts
Normal file
@@ -0,0 +1,514 @@
|
|||||||
|
import { css, cssManager } from '@design.estate/dees-element';
|
||||||
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
|
|
||||||
|
export const datepickerStyles = [
|
||||||
|
...DeesInputBase.baseStyles,
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
display: block;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.input-container {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-input {
|
||||||
|
width: 100%;
|
||||||
|
height: 40px;
|
||||||
|
padding: 0 12px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
outline: none;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-input::placeholder {
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-input:hover:not(:disabled) {
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-input:focus,
|
||||||
|
.date-input.open {
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
outline: 2px solid transparent;
|
||||||
|
outline-offset: 2px;
|
||||||
|
box-shadow: 0 0 0 2px ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')},
|
||||||
|
0 0 0 4px ${cssManager.bdTheme('hsl(222.2 47.4% 11.2% / 0.1)', 'hsl(210 20% 98% / 0.1)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.date-input:disabled {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Icon container using flexbox for better positioning */
|
||||||
|
.icon-container {
|
||||||
|
position: absolute;
|
||||||
|
right: 0;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 4px;
|
||||||
|
padding: 0 12px;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-container > * {
|
||||||
|
pointer-events: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.calendar-icon {
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
pointer-events: none;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button {
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
border: none;
|
||||||
|
background: transparent;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: 4px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
transition: opacity 0.2s ease, background-color 0.2s ease;
|
||||||
|
padding: 0;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button:disabled {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Calendar Popup Styles */
|
||||||
|
.calendar-popup {
|
||||||
|
will-change: transform, opacity;
|
||||||
|
pointer-events: none;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(-4px);
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
box-shadow: ${cssManager.bdTheme(
|
||||||
|
'0 10px 15px -3px hsl(0 0% 0% / 0.1), 0 4px 6px -4px hsl(0 0% 0% / 0.1)',
|
||||||
|
'0 10px 15px -3px hsl(0 0% 0% / 0.2), 0 4px 6px -4px hsl(0 0% 0% / 0.2)'
|
||||||
|
)};
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 12px;
|
||||||
|
position: absolute;
|
||||||
|
user-select: none;
|
||||||
|
margin-top: 4px;
|
||||||
|
z-index: 50;
|
||||||
|
left: 0;
|
||||||
|
min-width: 280px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.calendar-popup.top {
|
||||||
|
bottom: calc(100% + 4px);
|
||||||
|
top: auto;
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
transform: translateY(4px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.calendar-popup.bottom {
|
||||||
|
top: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.calendar-popup.show {
|
||||||
|
pointer-events: all;
|
||||||
|
transform: translateY(0);
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Calendar Header */
|
||||||
|
.calendar-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.month-year-display {
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 14px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-button {
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border: none;
|
||||||
|
background: transparent;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 6px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-button:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-button:active {
|
||||||
|
background: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Weekday headers */
|
||||||
|
.weekdays {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(7, 1fr);
|
||||||
|
gap: 0;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.weekday {
|
||||||
|
text-align: center;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 400;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
padding: 0 0 8px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Days grid */
|
||||||
|
.days-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(7, 1fr);
|
||||||
|
gap: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day {
|
||||||
|
aspect-ratio: 1;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
border: none;
|
||||||
|
width: 36px;
|
||||||
|
height: 36px;
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day:hover:not(.disabled) {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.other-month {
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.today {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.selected {
|
||||||
|
background: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(222.2 47.4% 11.2%)')};
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.disabled {
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Event indicators */
|
||||||
|
.day.has-event {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-indicator {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 4px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
display: flex;
|
||||||
|
gap: 2px;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-dot {
|
||||||
|
width: 4px;
|
||||||
|
height: 4px;
|
||||||
|
border-radius: 50%;
|
||||||
|
background: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-dot.info {
|
||||||
|
background: ${cssManager.bdTheme('hsl(211 70% 52%)', 'hsl(211 70% 62%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-dot.warning {
|
||||||
|
background: ${cssManager.bdTheme('hsl(45 90% 45%)', 'hsl(45 90% 55%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-dot.success {
|
||||||
|
background: ${cssManager.bdTheme('hsl(142 69% 45%)', 'hsl(142 69% 55%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-dot.error {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 72% 51%)', 'hsl(0 72% 61%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-count {
|
||||||
|
position: absolute;
|
||||||
|
top: 2px;
|
||||||
|
right: 2px;
|
||||||
|
min-width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
padding: 0 4px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 72% 51%)', 'hsl(0 72% 61%)')};
|
||||||
|
color: white;
|
||||||
|
border-radius: 8px;
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 600;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Tooltip for event details */
|
||||||
|
.event-tooltip {
|
||||||
|
position: absolute;
|
||||||
|
bottom: calc(100% + 8px);
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 90%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(0 0% 0%)')};
|
||||||
|
padding: 8px 12px;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 12px;
|
||||||
|
white-space: nowrap;
|
||||||
|
pointer-events: none;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.2s ease;
|
||||||
|
z-index: 10;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.event-tooltip::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 100%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
border: 4px solid transparent;
|
||||||
|
border-top-color: ${cssManager.bdTheme('hsl(0 0% 20%)', 'hsl(0 0% 90%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.day.has-event:hover .event-tooltip {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Time selector */
|
||||||
|
.time-selector {
|
||||||
|
margin-top: 12px;
|
||||||
|
padding-top: 12px;
|
||||||
|
border-top: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-selector-title {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 500;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-inputs {
|
||||||
|
display: flex;
|
||||||
|
gap: 8px;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-input {
|
||||||
|
width: 65px;
|
||||||
|
height: 36px;
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 0 12px;
|
||||||
|
font-size: 14px;
|
||||||
|
text-align: center;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-input:hover {
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-input:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
box-shadow: 0 0 0 2px ${cssManager.bdTheme('hsl(222.2 47.4% 11.2% / 0.1)', 'hsl(210 20% 98% / 0.1)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-separator {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.am-pm-selector {
|
||||||
|
display: flex;
|
||||||
|
gap: 4px;
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.am-pm-button {
|
||||||
|
padding: 6px 12px;
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.am-pm-button.selected {
|
||||||
|
background: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(222.2 47.4% 11.2%)')};
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.am-pm-button:hover:not(.selected) {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Action buttons */
|
||||||
|
.calendar-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 8px;
|
||||||
|
margin-top: 12px;
|
||||||
|
padding-top: 12px;
|
||||||
|
border-top: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-button {
|
||||||
|
flex: 1;
|
||||||
|
height: 36px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.today-button {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.today-button:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.today-button:active {
|
||||||
|
background: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button {
|
||||||
|
background: transparent;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 72.2% 50.6% / 0.1)', 'hsl(0 62.8% 30.6% / 0.1)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 72.2% 50.6%)', 'hsl(0 62.8% 30.6%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button:active {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 72.2% 50.6% / 0.2)', 'hsl(0 62.8% 30.6% / 0.2)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Timezone selector */
|
||||||
|
.timezone-selector {
|
||||||
|
margin-top: 12px;
|
||||||
|
padding-top: 12px;
|
||||||
|
border-top: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.timezone-selector-title {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 500;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(220 8.9% 46.1%)', 'hsl(215 20.2% 65.1%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.timezone-select {
|
||||||
|
width: 100%;
|
||||||
|
height: 36px;
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 0 12px;
|
||||||
|
font-size: 14px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(224 71.4% 4.1%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(224 71.4% 4.1%)', 'hsl(210 20% 98%)')};
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.timezone-select:hover {
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(214.3 31.8% 91.4%)', 'hsl(217.2 32.6% 17.5%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(210 20% 98%)', 'hsl(215 27.9% 16.9%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.timezone-select:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(222.2 47.4% 11.2%)', 'hsl(210 20% 98%)')};
|
||||||
|
box-shadow: 0 0 0 2px ${cssManager.bdTheme('hsl(222.2 47.4% 11.2% / 0.1)', 'hsl(210 20% 98% / 0.1)')};
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
179
ts_web/elements/00group-input/dees-input-datepicker/template.ts
Normal file
179
ts_web/elements/00group-input/dees-input-datepicker/template.ts
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
import { html, type TemplateResult } from '@design.estate/dees-element';
|
||||||
|
import type { DeesInputDatepicker } from './component.js';
|
||||||
|
|
||||||
|
export const renderDatepicker = (component: DeesInputDatepicker): TemplateResult => {
|
||||||
|
const monthNames = [
|
||||||
|
'January', 'February', 'March', 'April', 'May', 'June',
|
||||||
|
'July', 'August', 'September', 'October', 'November', 'December'
|
||||||
|
];
|
||||||
|
|
||||||
|
const weekDays = component.weekStartsOn === 1
|
||||||
|
? ['Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa', 'Su']
|
||||||
|
: ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'];
|
||||||
|
|
||||||
|
const days = component.getDaysInMonth();
|
||||||
|
const isAM = component.selectedHour < 12;
|
||||||
|
const timezones = component.getTimezones();
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="input-wrapper">
|
||||||
|
<dees-label .label=${component.label} .description=${component.description} .required=${component.required}></dees-label>
|
||||||
|
<div class="input-container">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
class="date-input ${component.isOpened ? 'open' : ''}"
|
||||||
|
.value=${component.formatDate(component.value)}
|
||||||
|
.placeholder=${component.placeholder}
|
||||||
|
?disabled=${component.disabled}
|
||||||
|
@click=${component.toggleCalendar}
|
||||||
|
@keydown=${component.handleKeydown}
|
||||||
|
@input=${component.handleManualInput}
|
||||||
|
@blur=${component.handleInputBlur}
|
||||||
|
style="padding-right: ${component.value ? '64px' : '40px'}"
|
||||||
|
/>
|
||||||
|
<div class="icon-container">
|
||||||
|
${component.value && !component.disabled ? html`
|
||||||
|
<button class="clear-button" @click=${component.clearValue} title="Clear">
|
||||||
|
<dees-icon icon="lucide:x" iconSize="14"></dees-icon>
|
||||||
|
</button>
|
||||||
|
` : ''}
|
||||||
|
<dees-icon class="calendar-icon" icon="lucide:calendar" iconSize="16"></dees-icon>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Calendar Popup -->
|
||||||
|
<div class="calendar-popup ${component.isOpened ? 'show' : ''} ${component.opensToTop ? 'top' : 'bottom'}">
|
||||||
|
<!-- Month/Year Navigation -->
|
||||||
|
<div class="calendar-header">
|
||||||
|
<button class="nav-button" @click=${component.previousMonth}>
|
||||||
|
<dees-icon icon="lucide:chevronLeft" iconSize="16"></dees-icon>
|
||||||
|
</button>
|
||||||
|
<div class="month-year-display">
|
||||||
|
${monthNames[component.viewDate.getMonth()]} ${component.viewDate.getFullYear()}
|
||||||
|
</div>
|
||||||
|
<button class="nav-button" @click=${component.nextMonth}>
|
||||||
|
<dees-icon icon="lucide:chevronRight" iconSize="16"></dees-icon>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Weekday Headers -->
|
||||||
|
<div class="weekdays">
|
||||||
|
${weekDays.map(day => html`<div class="weekday">${day}</div>`)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Days Grid -->
|
||||||
|
<div class="days-grid">
|
||||||
|
${days.map(day => {
|
||||||
|
const isToday = component.isToday(day);
|
||||||
|
const isSelected = component.isSelected(day);
|
||||||
|
const isOtherMonth = day.getMonth() !== component.viewDate.getMonth();
|
||||||
|
const isDisabled = component.isDisabled(day);
|
||||||
|
const dayEvents = component.getEventsForDate(day);
|
||||||
|
const hasEvents = dayEvents.length > 0;
|
||||||
|
const totalEventCount = dayEvents.reduce((sum, event) => sum + (event.count || 1), 0);
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div
|
||||||
|
class="day ${isOtherMonth ? 'other-month' : ''} ${isToday ? 'today' : ''} ${isSelected ? 'selected' : ''} ${isDisabled ? 'disabled' : ''} ${hasEvents ? 'has-event' : ''}"
|
||||||
|
@click=${() => !isDisabled && component.selectDate(day)}
|
||||||
|
>
|
||||||
|
${day.getDate()}
|
||||||
|
${hasEvents ? html`
|
||||||
|
${totalEventCount > 3 ? html`
|
||||||
|
<div class="event-count">${totalEventCount}</div>
|
||||||
|
` : html`
|
||||||
|
<div class="event-indicator">
|
||||||
|
${dayEvents.slice(0, 3).map(event => html`
|
||||||
|
<div class="event-dot ${event.type || 'info'}"></div>
|
||||||
|
`)}
|
||||||
|
</div>
|
||||||
|
`}
|
||||||
|
${dayEvents[0].title ? html`
|
||||||
|
<div class="event-tooltip">
|
||||||
|
${dayEvents[0].title}
|
||||||
|
${totalEventCount > 1 ? html` (+${totalEventCount - 1} more)` : ''}
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Time Selector -->
|
||||||
|
${component.enableTime ? html`
|
||||||
|
<div class="time-selector">
|
||||||
|
<div class="time-selector-title">Time</div>
|
||||||
|
<div class="time-inputs">
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
class="time-input"
|
||||||
|
.value=${component.timeFormat === '12h'
|
||||||
|
? (component.selectedHour === 0 ? 12 : component.selectedHour > 12 ? component.selectedHour - 12 : component.selectedHour).toString().padStart(2, '0')
|
||||||
|
: component.selectedHour.toString().padStart(2, '0')}
|
||||||
|
@input=${(e: InputEvent) => component.handleHourInput(e)}
|
||||||
|
min="${component.timeFormat === '12h' ? 1 : 0}"
|
||||||
|
max="${component.timeFormat === '12h' ? 12 : 23}"
|
||||||
|
/>
|
||||||
|
<span class="time-separator">:</span>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
class="time-input"
|
||||||
|
.value=${component.selectedMinute.toString().padStart(2, '0')}
|
||||||
|
@input=${(e: InputEvent) => component.handleMinuteInput(e)}
|
||||||
|
min="0"
|
||||||
|
max="59"
|
||||||
|
step="${component.minuteIncrement || 1}"
|
||||||
|
/>
|
||||||
|
${component.timeFormat === '12h' ? html`
|
||||||
|
<div class="am-pm-selector">
|
||||||
|
<button
|
||||||
|
class="am-pm-button ${isAM ? 'selected' : ''}"
|
||||||
|
@click=${() => component.setAMPM('am')}
|
||||||
|
>
|
||||||
|
AM
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
class="am-pm-button ${!isAM ? 'selected' : ''}"
|
||||||
|
@click=${() => component.setAMPM('pm')}
|
||||||
|
>
|
||||||
|
PM
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
|
||||||
|
<!-- Timezone Selector -->
|
||||||
|
${component.enableTimezone ? html`
|
||||||
|
<div class="timezone-selector">
|
||||||
|
<div class="timezone-selector-title">Timezone</div>
|
||||||
|
<select
|
||||||
|
class="timezone-select"
|
||||||
|
.value=${component.timezone}
|
||||||
|
@change=${(e: Event) => component.handleTimezoneChange(e)}
|
||||||
|
>
|
||||||
|
${timezones.map(tz => html`
|
||||||
|
<option value="${tz.value}" ?selected=${tz.value === component.timezone}>
|
||||||
|
${tz.label}
|
||||||
|
</option>
|
||||||
|
`)}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
` : ''}
|
||||||
|
|
||||||
|
<!-- Action Buttons -->
|
||||||
|
<div class="calendar-actions">
|
||||||
|
<button class="action-button today-button" @click=${component.selectToday}>
|
||||||
|
Today
|
||||||
|
</button>
|
||||||
|
<button class="action-button clear-button" @click=${component.clear}>
|
||||||
|
Clear
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
};
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
export interface IDateEvent {
|
||||||
|
date: string; // ISO date string (YYYY-MM-DD)
|
||||||
|
title?: string;
|
||||||
|
description?: string;
|
||||||
|
type?: 'info' | 'warning' | 'success' | 'error';
|
||||||
|
count?: number; // Number of events on this day
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import { html, css } from '@design.estate/dees-element';
|
import { html, css } from '@design.estate/dees-element';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
import './dees-panel.js';
|
import '../../dees-panel/dees-panel.js';
|
||||||
import './dees-form.js';
|
import '../../00group-form/dees-form/dees-form.js';
|
||||||
import './dees-form-submit.js';
|
import '../../00group-form/dees-form-submit/dees-form-submit.js';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
<style>
|
<style>
|
||||||
@@ -9,8 +9,8 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import { demoFunc } from './dees-input-dropdown.demo.js';
|
import { demoFunc } from './dees-input-dropdown.demo.js';
|
||||||
import { DeesInputBase } from './dees-input-base.js';
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
import { cssGeistFontFamily } from './00fonts.js';
|
import { cssGeistFontFamily } from '../../00fonts.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -25,10 +25,10 @@ export class DeesInputDropdown extends DeesInputBase<DeesInputDropdown> {
|
|||||||
// INSTANCE
|
// INSTANCE
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public options: { option: string; key: string; payload?: any }[] = [];
|
accessor options: { option: string; key: string; payload?: any }[] = [];
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public selectedOption: { option: string; key: string; payload?: any } = null;
|
accessor selectedOption: { option: string; key: string; payload?: any } = null;
|
||||||
|
|
||||||
// Add value property for form compatibility
|
// Add value property for form compatibility
|
||||||
public get value() {
|
public get value() {
|
||||||
@@ -42,22 +42,22 @@ export class DeesInputDropdown extends DeesInputBase<DeesInputDropdown> {
|
|||||||
@property({
|
@property({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
})
|
})
|
||||||
public enableSearch: boolean = true;
|
accessor enableSearch: boolean = true;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public opensToTop: boolean = false;
|
accessor opensToTop: boolean = false;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private filteredOptions: { option: string; key: string; payload?: any }[] = [];
|
accessor filteredOptions: { option: string; key: string; payload?: any }[] = [];
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private highlightedIndex: number = 0;
|
accessor highlightedIndex: number = 0;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public isOpened = false;
|
accessor isOpened = false;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private searchValue: string = '';
|
accessor searchValue: string = '';
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
...DeesInputBase.baseStyles,
|
...DeesInputBase.baseStyles,
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-input-dropdown.js';
|
||||||
619
ts_web/elements/00group-input/dees-input-fileupload/component.ts
Normal file
619
ts_web/elements/00group-input/dees-input-fileupload/component.ts
Normal file
@@ -0,0 +1,619 @@
|
|||||||
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
|
import { demoFunc } from './demo.js';
|
||||||
|
import { fileuploadStyles } from './styles.js';
|
||||||
|
import '../../dees-icon/dees-icon.js';
|
||||||
|
import '../../dees-label/dees-label.js';
|
||||||
|
|
||||||
|
import {
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
property,
|
||||||
|
state,
|
||||||
|
type TemplateResult,
|
||||||
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
'dees-input-fileupload': DeesInputFileupload;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@customElement('dees-input-fileupload')
|
||||||
|
export class DeesInputFileupload extends DeesInputBase<DeesInputFileupload> {
|
||||||
|
public static demo = demoFunc;
|
||||||
|
|
||||||
|
@property({ attribute: false })
|
||||||
|
accessor value: File[] = [];
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor state: 'idle' | 'dragOver' | 'dropped' | 'uploading' | 'completed' = 'idle';
|
||||||
|
|
||||||
|
@state()
|
||||||
|
accessor isLoading: boolean = false;
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor buttonText: string = 'Select files';
|
||||||
|
|
||||||
|
@property({ type: String })
|
||||||
|
accessor accept: string = '';
|
||||||
|
|
||||||
|
@property({ type: Boolean })
|
||||||
|
accessor multiple: boolean = true;
|
||||||
|
|
||||||
|
@property({ type: Number })
|
||||||
|
accessor maxSize: number = 0; // 0 means no limit
|
||||||
|
|
||||||
|
@property({ type: Number })
|
||||||
|
accessor maxFiles: number = 0; // 0 means no limit
|
||||||
|
|
||||||
|
@property({ type: String, reflect: true })
|
||||||
|
accessor validationState: 'valid' | 'invalid' | 'warn' | 'pending' = null;
|
||||||
|
|
||||||
|
accessor validationMessage: string = '';
|
||||||
|
|
||||||
|
private previewUrlMap: WeakMap<File, string> = new WeakMap();
|
||||||
|
private dropArea: HTMLElement | null = null;
|
||||||
|
|
||||||
|
public static styles = fileuploadStyles;
|
||||||
|
|
||||||
|
public render(): TemplateResult {
|
||||||
|
const acceptedSummary = this.getAcceptedSummary();
|
||||||
|
const metaEntries: string[] = [
|
||||||
|
this.multiple ? 'Multiple files supported' : 'Single file only',
|
||||||
|
this.maxSize > 0 ? `Max ${this.formatFileSize(this.maxSize)}` : 'No size limit',
|
||||||
|
];
|
||||||
|
|
||||||
|
if (acceptedSummary) {
|
||||||
|
metaEntries.push(`Accepts ${acceptedSummary}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="input-wrapper">
|
||||||
|
<dees-label
|
||||||
|
.label=${this.label}
|
||||||
|
.description=${this.description}
|
||||||
|
.required=${this.required}
|
||||||
|
></dees-label>
|
||||||
|
<div
|
||||||
|
class="dropzone ${this.state === 'dragOver' ? 'dropzone--active' : ''} ${this.disabled ? 'dropzone--disabled' : ''} ${this.value.length > 0 ? 'dropzone--has-files' : ''}"
|
||||||
|
role="button"
|
||||||
|
tabindex=${this.disabled ? -1 : 0}
|
||||||
|
aria-disabled=${this.disabled}
|
||||||
|
aria-label=${`Select files${acceptedSummary ? ` (${acceptedSummary})` : ''}`}
|
||||||
|
@click=${this.handleDropzoneClick}
|
||||||
|
@keydown=${this.handleDropzoneKeydown}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
class="file-input"
|
||||||
|
style="position: absolute; opacity: 0; pointer-events: none; width: 1px; height: 1px; top: 0; left: 0; overflow: hidden;"
|
||||||
|
type="file"
|
||||||
|
?multiple=${this.multiple}
|
||||||
|
accept=${this.accept || ''}
|
||||||
|
?disabled=${this.disabled}
|
||||||
|
@change=${this.handleFileInputChange}
|
||||||
|
tabindex="-1"
|
||||||
|
/>
|
||||||
|
<div class="dropzone__body">
|
||||||
|
<div class="dropzone__icon">
|
||||||
|
${this.isLoading
|
||||||
|
? html`<span class="dropzone__loader" aria-hidden="true"></span>`
|
||||||
|
: html`<dees-icon icon="lucide:FolderOpen"></dees-icon>`}
|
||||||
|
</div>
|
||||||
|
<div class="dropzone__content">
|
||||||
|
<span class="dropzone__headline">${this.buttonText || 'Select files'}</span>
|
||||||
|
<span class="dropzone__subline">
|
||||||
|
Drag and drop files here or
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="dropzone__browse"
|
||||||
|
@click=${this.handleBrowseClick}
|
||||||
|
?disabled=${this.disabled}
|
||||||
|
>
|
||||||
|
browse
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dropzone__meta">
|
||||||
|
${metaEntries.map((entry) => html`<span>${entry}</span>`)}
|
||||||
|
</div>
|
||||||
|
${this.renderFileList()}
|
||||||
|
</div>
|
||||||
|
${this.validationMessage
|
||||||
|
? html`<div class="validation-message" aria-live="polite">${this.validationMessage}</div>`
|
||||||
|
: html``}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderFileList(): TemplateResult {
|
||||||
|
if (this.value.length === 0) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="file-list">
|
||||||
|
<div class="file-list__header">
|
||||||
|
<span>${this.value.length} file${this.value.length === 1 ? '' : 's'} selected</span>
|
||||||
|
${this.value.length > 0
|
||||||
|
? html`<button type="button" class="file-list__clear" @click=${this.handleClearAll}>Clear ${this.value.length > 1 ? 'all' : ''}</button>`
|
||||||
|
: html``}
|
||||||
|
</div>
|
||||||
|
<div class="file-list__items">
|
||||||
|
${this.value.map((file) => this.renderFileRow(file))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderFileRow(file: File): TemplateResult {
|
||||||
|
const fileType = this.getFileType(file);
|
||||||
|
const previewUrl = this.canShowPreview(file) ? this.getPreviewUrl(file) : null;
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="file-row ${fileType}-file">
|
||||||
|
<div class="file-thumb" aria-hidden="true">
|
||||||
|
${previewUrl
|
||||||
|
? html`<img class="thumb-image" src=${previewUrl} alt=${`Preview of ${file.name}`}>`
|
||||||
|
: html`<dees-icon icon=${this.getFileIcon(file)}></dees-icon>`}
|
||||||
|
</div>
|
||||||
|
<div class="file-meta">
|
||||||
|
<div class="file-name" title=${file.name}>${file.name}</div>
|
||||||
|
<div class="file-details">
|
||||||
|
<span class="file-size">${this.formatFileSize(file.size)}</span>
|
||||||
|
${fileType !== 'file' ? html`<span class="file-type">${fileType}</span>` : html``}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="file-actions">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="remove-button"
|
||||||
|
@click=${() => this.removeFile(file)}
|
||||||
|
aria-label=${`Remove ${file.name}`}
|
||||||
|
>
|
||||||
|
<dees-icon icon="lucide:X"></dees-icon>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleFileInputChange = async (event: Event) => {
|
||||||
|
this.isLoading = false;
|
||||||
|
const target = event.target as HTMLInputElement;
|
||||||
|
const files = Array.from(target.files ?? []);
|
||||||
|
if (files.length > 0) {
|
||||||
|
await this.addFiles(files);
|
||||||
|
}
|
||||||
|
target.value = '';
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleDropzoneClick = (event: MouseEvent) => {
|
||||||
|
if (this.disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Don't open file selector if clicking on the browse button or file list
|
||||||
|
if ((event.target as HTMLElement).closest('.dropzone__browse, .file-list')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.openFileSelector();
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleBrowseClick = (event: MouseEvent) => {
|
||||||
|
if (this.disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event.stopPropagation(); // Stop propagation to prevent double trigger
|
||||||
|
this.openFileSelector();
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleDropzoneKeydown = (event: KeyboardEvent) => {
|
||||||
|
if (this.disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (event.key === 'Enter' || event.key === ' ') {
|
||||||
|
event.preventDefault();
|
||||||
|
this.openFileSelector();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleClearAll = (event: MouseEvent) => {
|
||||||
|
event.preventDefault();
|
||||||
|
this.clearAll();
|
||||||
|
};
|
||||||
|
|
||||||
|
private handleDragEvent = async (event: DragEvent) => {
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
if (this.disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.type === 'dragenter' || event.type === 'dragover') {
|
||||||
|
if (event.dataTransfer) {
|
||||||
|
event.dataTransfer.dropEffect = 'copy';
|
||||||
|
}
|
||||||
|
this.state = 'dragOver';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.type === 'dragleave') {
|
||||||
|
if (!this.dropArea) {
|
||||||
|
this.state = 'idle';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const rect = this.dropArea.getBoundingClientRect();
|
||||||
|
const { clientX = 0, clientY = 0 } = event;
|
||||||
|
if (clientX <= rect.left || clientX >= rect.right || clientY <= rect.top || clientY >= rect.bottom) {
|
||||||
|
this.state = 'idle';
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.type === 'drop') {
|
||||||
|
this.state = 'idle';
|
||||||
|
const files = Array.from(event.dataTransfer?.files ?? []);
|
||||||
|
if (files.length > 0) {
|
||||||
|
await this.addFiles(files);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private attachDropListeners(): void {
|
||||||
|
if (!this.dropArea) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
['dragenter', 'dragover', 'dragleave', 'drop'].forEach((eventName) => {
|
||||||
|
this.dropArea!.addEventListener(eventName, this.handleDragEvent);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private detachDropListeners(): void {
|
||||||
|
if (!this.dropArea) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
['dragenter', 'dragover', 'dragleave', 'drop'].forEach((eventName) => {
|
||||||
|
this.dropArea!.removeEventListener(eventName, this.handleDragEvent);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private rebindInteractiveElements(): void {
|
||||||
|
const newDropArea = this.shadowRoot?.querySelector('.dropzone') as HTMLElement | null;
|
||||||
|
|
||||||
|
if (newDropArea !== this.dropArea) {
|
||||||
|
this.detachDropListeners();
|
||||||
|
this.dropArea = newDropArea;
|
||||||
|
this.attachDropListeners();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public formatFileSize(bytes: number): string {
|
||||||
|
const units = ['Bytes', 'KB', 'MB', 'GB'];
|
||||||
|
if (bytes === 0) return '0 Bytes';
|
||||||
|
const exponent = Math.min(Math.floor(Math.log(bytes) / Math.log(1024)), units.length - 1);
|
||||||
|
const size = bytes / Math.pow(1024, exponent);
|
||||||
|
return `${Math.round(size * 100) / 100} ${units[exponent]}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getFileType(file: File): string {
|
||||||
|
const type = file.type.toLowerCase();
|
||||||
|
if (type.startsWith('image/')) return 'image';
|
||||||
|
if (type === 'application/pdf') return 'pdf';
|
||||||
|
if (type.includes('word') || type.includes('document')) return 'doc';
|
||||||
|
if (type.includes('sheet') || type.includes('excel')) return 'spreadsheet';
|
||||||
|
if (type.includes('presentation') || type.includes('powerpoint')) return 'presentation';
|
||||||
|
if (type.startsWith('video/')) return 'video';
|
||||||
|
if (type.startsWith('audio/')) return 'audio';
|
||||||
|
if (type.includes('zip') || type.includes('compressed')) return 'archive';
|
||||||
|
return 'file';
|
||||||
|
}
|
||||||
|
|
||||||
|
public getFileIcon(file: File): string {
|
||||||
|
const fileType = this.getFileType(file);
|
||||||
|
const iconMap: Record<string, string> = {
|
||||||
|
image: 'lucide:FileImage',
|
||||||
|
pdf: 'lucide:FileText',
|
||||||
|
doc: 'lucide:FileText',
|
||||||
|
spreadsheet: 'lucide:FileSpreadsheet',
|
||||||
|
presentation: 'lucide:FileBarChart',
|
||||||
|
video: 'lucide:FileVideo',
|
||||||
|
audio: 'lucide:FileAudio',
|
||||||
|
archive: 'lucide:FileArchive',
|
||||||
|
file: 'lucide:File',
|
||||||
|
};
|
||||||
|
return iconMap[fileType] ?? 'lucide:File';
|
||||||
|
}
|
||||||
|
|
||||||
|
public canShowPreview(file: File): boolean {
|
||||||
|
return file.type.startsWith('image/') && file.size < 5 * 1024 * 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
private validateFile(file: File): boolean {
|
||||||
|
if (this.maxSize > 0 && file.size > this.maxSize) {
|
||||||
|
this.validationMessage = `File "${file.name}" exceeds the maximum size of ${this.formatFileSize(this.maxSize)}`;
|
||||||
|
this.validationState = 'invalid';
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.accept) {
|
||||||
|
const acceptedTypes = this.accept
|
||||||
|
.split(',')
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter((entry) => entry.length > 0);
|
||||||
|
|
||||||
|
if (acceptedTypes.length > 0) {
|
||||||
|
let isAccepted = false;
|
||||||
|
for (const acceptType of acceptedTypes) {
|
||||||
|
if (acceptType.startsWith('.')) {
|
||||||
|
if (file.name.toLowerCase().endsWith(acceptType.toLowerCase())) {
|
||||||
|
isAccepted = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if (acceptType.endsWith('/*')) {
|
||||||
|
const prefix = acceptType.slice(0, -2);
|
||||||
|
if (file.type.startsWith(prefix)) {
|
||||||
|
isAccepted = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else if (file.type === acceptType) {
|
||||||
|
isAccepted = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isAccepted) {
|
||||||
|
this.validationMessage = `File type not accepted. Allowed: ${acceptedTypes.join(', ')}`;
|
||||||
|
this.validationState = 'invalid';
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getPreviewUrl(file: File): string {
|
||||||
|
let url = this.previewUrlMap.get(file);
|
||||||
|
if (!url) {
|
||||||
|
url = URL.createObjectURL(file);
|
||||||
|
this.previewUrlMap.set(file, url);
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
private releasePreview(file: File): void {
|
||||||
|
const url = this.previewUrlMap.get(file);
|
||||||
|
if (url) {
|
||||||
|
URL.revokeObjectURL(url);
|
||||||
|
this.previewUrlMap.delete(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private getAcceptedSummary(): string | null {
|
||||||
|
if (!this.accept) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const formatted = Array.from(
|
||||||
|
new Set(
|
||||||
|
this.accept
|
||||||
|
.split(',')
|
||||||
|
.map((token) => token.trim())
|
||||||
|
.filter((token) => token.length > 0)
|
||||||
|
.map((token) => this.formatAcceptToken(token))
|
||||||
|
)
|
||||||
|
).filter(Boolean);
|
||||||
|
|
||||||
|
if (formatted.length === 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (formatted.length === 1) {
|
||||||
|
return formatted[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (formatted.length === 2) {
|
||||||
|
return `${formatted[0]}, ${formatted[1]}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${formatted.slice(0, 2).join(', ')}…`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private formatAcceptToken(token: string): string {
|
||||||
|
if (token === '*/*') {
|
||||||
|
return 'All files';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.endsWith('/*')) {
|
||||||
|
const family = token.split('/')[0];
|
||||||
|
if (!family) {
|
||||||
|
return 'All files';
|
||||||
|
}
|
||||||
|
return `${family.charAt(0).toUpperCase()}${family.slice(1)} files`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.startsWith('.')) {
|
||||||
|
return token.slice(1).toUpperCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (token.includes('pdf')) return 'PDF';
|
||||||
|
if (token.includes('zip')) return 'ZIP';
|
||||||
|
if (token.includes('json')) return 'JSON';
|
||||||
|
if (token.includes('msword')) return 'DOC';
|
||||||
|
if (token.includes('wordprocessingml')) return 'DOCX';
|
||||||
|
if (token.includes('excel')) return 'XLS';
|
||||||
|
if (token.includes('presentation')) return 'PPT';
|
||||||
|
|
||||||
|
const segments = token.split('/');
|
||||||
|
const lastSegment = segments.pop() ?? token;
|
||||||
|
return lastSegment.toUpperCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
private attachLifecycleListeners(): void {
|
||||||
|
this.rebindInteractiveElements();
|
||||||
|
}
|
||||||
|
|
||||||
|
public firstUpdated(changedProperties: Map<string, unknown>) {
|
||||||
|
super.firstUpdated(changedProperties);
|
||||||
|
this.attachLifecycleListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
public updated(changedProperties: Map<string, unknown>) {
|
||||||
|
super.updated(changedProperties);
|
||||||
|
if (changedProperties.has('value')) {
|
||||||
|
void this.validate();
|
||||||
|
}
|
||||||
|
this.rebindInteractiveElements();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async disconnectedCallback(): Promise<void> {
|
||||||
|
this.detachDropListeners();
|
||||||
|
this.value.forEach((file) => this.releasePreview(file));
|
||||||
|
this.previewUrlMap = new WeakMap();
|
||||||
|
await super.disconnectedCallback();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async openFileSelector() {
|
||||||
|
if (this.disabled || this.isLoading) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isLoading = true;
|
||||||
|
|
||||||
|
// Ensure we have the latest reference to the file input
|
||||||
|
const inputFile = this.shadowRoot?.querySelector('.file-input') as HTMLInputElement | null;
|
||||||
|
|
||||||
|
if (!inputFile) {
|
||||||
|
this.isLoading = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleFocus = () => {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (!inputFile.files || inputFile.files.length === 0) {
|
||||||
|
this.isLoading = false;
|
||||||
|
}
|
||||||
|
window.removeEventListener('focus', handleFocus);
|
||||||
|
}, 300);
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('focus', handleFocus);
|
||||||
|
|
||||||
|
// Click the input to open file selector
|
||||||
|
inputFile.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
public removeFile(file: File) {
|
||||||
|
const index = this.value.indexOf(file);
|
||||||
|
if (index > -1) {
|
||||||
|
this.releasePreview(file);
|
||||||
|
this.value.splice(index, 1);
|
||||||
|
this.requestUpdate('value');
|
||||||
|
void this.validate();
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public clearAll() {
|
||||||
|
const existingFiles = [...this.value];
|
||||||
|
this.value = [];
|
||||||
|
existingFiles.forEach((file) => this.releasePreview(file));
|
||||||
|
this.requestUpdate('value');
|
||||||
|
void this.validate();
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
this.buttonText = 'Select files';
|
||||||
|
}
|
||||||
|
|
||||||
|
public async updateValue(eventArg: Event) {
|
||||||
|
const target = eventArg.target as HTMLInputElement;
|
||||||
|
this.value = Array.from(target.files ?? []);
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public setValue(value: File[]): void {
|
||||||
|
this.value.forEach((file) => this.releasePreview(file));
|
||||||
|
this.value = value;
|
||||||
|
if (value.length > 0) {
|
||||||
|
this.buttonText = this.multiple ? 'Add more files' : 'Replace file';
|
||||||
|
} else {
|
||||||
|
this.buttonText = 'Select files';
|
||||||
|
}
|
||||||
|
this.requestUpdate('value');
|
||||||
|
void this.validate();
|
||||||
|
}
|
||||||
|
|
||||||
|
public getValue(): File[] {
|
||||||
|
return this.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async addFiles(files: File[]) {
|
||||||
|
const filesToAdd: File[] = [];
|
||||||
|
|
||||||
|
for (const file of files) {
|
||||||
|
if (this.validateFile(file)) {
|
||||||
|
filesToAdd.push(file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filesToAdd.length === 0) {
|
||||||
|
this.isLoading = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.maxFiles > 0) {
|
||||||
|
const totalFiles = this.value.length + filesToAdd.length;
|
||||||
|
if (totalFiles > this.maxFiles) {
|
||||||
|
const allowedCount = this.maxFiles - this.value.length;
|
||||||
|
if (allowedCount <= 0) {
|
||||||
|
this.validationMessage = `Maximum ${this.maxFiles} files allowed`;
|
||||||
|
this.validationState = 'invalid';
|
||||||
|
this.isLoading = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
filesToAdd.splice(allowedCount);
|
||||||
|
this.validationMessage = `Only ${allowedCount} more file(s) can be added`;
|
||||||
|
this.validationState = 'warn';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.multiple && filesToAdd.length > 0) {
|
||||||
|
this.value.forEach((file) => this.releasePreview(file));
|
||||||
|
this.value = [filesToAdd[0]];
|
||||||
|
} else {
|
||||||
|
this.value.push(...filesToAdd);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.validationMessage = '';
|
||||||
|
this.validationState = null;
|
||||||
|
this.requestUpdate('value');
|
||||||
|
await this.validate();
|
||||||
|
this.changeSubject.next(this);
|
||||||
|
this.isLoading = false;
|
||||||
|
|
||||||
|
if (this.value.length > 0) {
|
||||||
|
this.buttonText = this.multiple ? 'Add more files' : 'Replace file';
|
||||||
|
} else {
|
||||||
|
this.buttonText = 'Select files';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async validate(): Promise<boolean> {
|
||||||
|
this.validationMessage = '';
|
||||||
|
|
||||||
|
if (this.required && this.value.length === 0) {
|
||||||
|
this.validationState = 'invalid';
|
||||||
|
this.validationMessage = 'Please select at least one file';
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const file of this.value) {
|
||||||
|
if (!this.validateFile(file)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.validationState = this.value.length > 0 ? 'valid' : null;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
159
ts_web/elements/00group-input/dees-input-fileupload/demo.ts
Normal file
159
ts_web/elements/00group-input/dees-input-fileupload/demo.ts
Normal file
@@ -0,0 +1,159 @@
|
|||||||
|
import { css, cssManager, html } from '@design.estate/dees-element';
|
||||||
|
import './component.js';
|
||||||
|
import '../../dees-panel/dees-panel.js';
|
||||||
|
|
||||||
|
export const demoFunc = () => html`
|
||||||
|
<dees-demowrapper>
|
||||||
|
<style>
|
||||||
|
${css`
|
||||||
|
.demo-shell {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 32px;
|
||||||
|
padding: 24px;
|
||||||
|
max-width: 1160px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demo-grid {
|
||||||
|
display: grid;
|
||||||
|
gap: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 960px) {
|
||||||
|
.demo-grid--two {
|
||||||
|
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.demo-stack {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demo-note {
|
||||||
|
margin-top: 16px;
|
||||||
|
padding: 16px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(217 91% 90%)', 'hsl(215 20% 26%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(213 100% 97%)', 'hsl(215 20% 12%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 25% 32%)', 'hsl(215 20% 82%)')};
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.55;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demo-note strong {
|
||||||
|
color: ${cssManager.bdTheme('hsl(217 91% 45%)', 'hsl(213 93% 68%)')};
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div class="demo-shell">
|
||||||
|
<dees-panel
|
||||||
|
.title=${'Modern file uploader'}
|
||||||
|
.subtitle=${'Shadcn-inspired layout with drag & drop, previews and validation'}
|
||||||
|
>
|
||||||
|
<div class="demo-grid demo-grid--two">
|
||||||
|
<div class="demo-stack">
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Attachments'}
|
||||||
|
.description=${'Upload supporting documents for your request'}
|
||||||
|
.accept=${'image/*,.pdf,.zip'}
|
||||||
|
.maxSize=${10 * 1024 * 1024}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Brand assets'}
|
||||||
|
.description=${'Upload high-resolution imagery (JPG/PNG)'}
|
||||||
|
.accept=${'image/jpeg,image/png'}
|
||||||
|
.multiple=${false}
|
||||||
|
.maxSize=${5 * 1024 * 1024}
|
||||||
|
.buttonText=${'Select cover image'}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="demo-stack">
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Audio uploads'}
|
||||||
|
.description=${'Share podcast drafts (MP3/WAV, max 25MB each)'}
|
||||||
|
.accept=${'audio/*'}
|
||||||
|
.maxSize=${25 * 1024 * 1024}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Disabled example'}
|
||||||
|
.description=${'Uploader is disabled while moderation is pending'}
|
||||||
|
.disabled=${true}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel
|
||||||
|
.title=${'Form integration'}
|
||||||
|
.subtitle=${'Combine file uploads with the rest of the DEES form ecosystem'}
|
||||||
|
>
|
||||||
|
<div class="demo-grid">
|
||||||
|
<dees-form>
|
||||||
|
<div class="demo-stack">
|
||||||
|
<dees-input-text
|
||||||
|
.label=${'Project name'}
|
||||||
|
.description=${'How should we refer to this project internally?'}
|
||||||
|
.required=${true}
|
||||||
|
.key=${'projectName'}
|
||||||
|
></dees-input-text>
|
||||||
|
|
||||||
|
<dees-input-text
|
||||||
|
.label=${'Contact email'}
|
||||||
|
.inputType=${'email'}
|
||||||
|
.required=${true}
|
||||||
|
.key=${'contactEmail'}
|
||||||
|
></dees-input-text>
|
||||||
|
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Statement of work'}
|
||||||
|
.description=${'Upload a signed statement of work (PDF, max 15MB)'}
|
||||||
|
.required=${true}
|
||||||
|
.accept=${'application/pdf'}
|
||||||
|
.maxSize=${15 * 1024 * 1024}
|
||||||
|
.multiple=${false}
|
||||||
|
.key=${'sow'}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
|
||||||
|
<dees-input-fileupload
|
||||||
|
.label=${'Creative references'}
|
||||||
|
.description=${'Optional. Upload up to five visual references'}
|
||||||
|
.accept=${'image/*'}
|
||||||
|
.maxFiles=${5}
|
||||||
|
.maxSize=${8 * 1024 * 1024}
|
||||||
|
.key=${'references'}
|
||||||
|
></dees-input-fileupload>
|
||||||
|
|
||||||
|
<dees-input-text
|
||||||
|
.label=${'Notes'}
|
||||||
|
.description=${'Add optional context for reviewers'}
|
||||||
|
.inputType=${'textarea'}
|
||||||
|
.key=${'notes'}
|
||||||
|
></dees-input-text>
|
||||||
|
|
||||||
|
<dees-form-submit .text=${'Submit briefing'}></dees-form-submit>
|
||||||
|
</div>
|
||||||
|
</dees-form>
|
||||||
|
|
||||||
|
<div class="demo-note">
|
||||||
|
<strong>Good to know:</strong>
|
||||||
|
<ul>
|
||||||
|
<li>Drag & drop highlights the dropzone and supports keyboard activation.</li>
|
||||||
|
<li>Accepted file types are summarised automatically from the <code>accept</code> attribute.</li>
|
||||||
|
<li>Image uploads show live previews generated via <code>URL.createObjectURL</code>.</li>
|
||||||
|
<li>File size and file-count limits surface inline validation messages.</li>
|
||||||
|
<li>The component stays compatible with <code>dees-form</code> value accessors.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
</div>
|
||||||
|
</dees-demowrapper>
|
||||||
|
`;
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './component.js';
|
||||||
313
ts_web/elements/00group-input/dees-input-fileupload/styles.ts
Normal file
313
ts_web/elements/00group-input/dees-input-fileupload/styles.ts
Normal file
@@ -0,0 +1,313 @@
|
|||||||
|
import { css, cssManager } from '@design.estate/dees-element';
|
||||||
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
|
|
||||||
|
export const fileuploadStyles = [
|
||||||
|
cssManager.defaultStyles,
|
||||||
|
...DeesInputBase.baseStyles,
|
||||||
|
css`
|
||||||
|
:host {
|
||||||
|
position: relative;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.input-wrapper {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone {
|
||||||
|
position: relative;
|
||||||
|
padding: 20px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1.5px dashed ${cssManager.bdTheme('hsl(215 16% 80%)', 'hsl(217 20% 25%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(215 20% 12%)')};
|
||||||
|
transition: border-color 0.2s ease, box-shadow 0.2s ease, background 0.2s ease;
|
||||||
|
cursor: pointer;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone:focus-visible {
|
||||||
|
box-shadow: 0 0 0 2px ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(215 20% 12%)')},
|
||||||
|
0 0 0 4px ${cssManager.bdTheme('hsl(217 91% 60% / 0.5)', 'hsl(213 93% 68% / 0.4)')};
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone--active {
|
||||||
|
border-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
box-shadow: 0 12px 32px ${cssManager.bdTheme('rgba(15, 23, 42, 0.12)', 'rgba(0, 0, 0, 0.35)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(217 91% 60% / 0.06)', 'hsl(213 93% 68% / 0.12)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone--has-files {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 99%)', 'hsl(215 20% 11%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone--disabled {
|
||||||
|
opacity: 0.6;
|
||||||
|
pointer-events: none;
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__body {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__icon {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 16px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
background: ${cssManager.bdTheme('hsl(217 91% 60% / 0.12)', 'hsl(213 93% 68% / 0.12)')};
|
||||||
|
position: relative;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__icon dees-icon {
|
||||||
|
font-size: 22px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__loader {
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
border-radius: 999px;
|
||||||
|
border: 2px solid ${cssManager.bdTheme('rgba(15, 23, 42, 0.15)', 'rgba(255, 255, 255, 0.15)')};
|
||||||
|
border-top-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
animation: loader-spin 0.6s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 4px;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__headline {
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: ${cssManager.bdTheme('hsl(222 47% 11%)', 'hsl(210 20% 96%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__subline {
|
||||||
|
font-size: 13px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 46%)', 'hsl(215 16% 70%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__browse {
|
||||||
|
appearance: none;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
padding: 0;
|
||||||
|
margin-left: 4px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__browse:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__browse:disabled {
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__meta {
|
||||||
|
margin-top: 14px;
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 50%)', 'hsl(215 16% 72%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropzone__meta span {
|
||||||
|
padding: 4px 10px;
|
||||||
|
border-radius: 999px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(217 91% 95%)', 'hsl(213 93% 18%)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(217 91% 90%)', 'hsl(213 93% 24%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 12px;
|
||||||
|
margin-top: 20px;
|
||||||
|
padding-top: 20px;
|
||||||
|
border-top: 1px solid ${cssManager.bdTheme('hsl(217 91% 90%)', 'hsl(213 93% 24%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-list__header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 68%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-list__clear {
|
||||||
|
appearance: none;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')};
|
||||||
|
cursor: pointer;
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 13px;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-list__clear:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-list__items {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
padding: 10px 12px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100% / 0.5)', 'hsl(215 20% 16% / 0.5)')};
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(213 27% 92%)', 'hsl(217 25% 26%)')};
|
||||||
|
border-radius: 8px;
|
||||||
|
transition: background 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-row:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 0% 100% / 0.8)', 'hsl(215 20% 16% / 0.8)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-thumb {
|
||||||
|
width: 36px;
|
||||||
|
height: 36px;
|
||||||
|
border-radius: 8px;
|
||||||
|
background: ${cssManager.bdTheme('hsl(214 31% 92%)', 'hsl(217 32% 18%)')};
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
overflow: hidden;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-thumb dees-icon {
|
||||||
|
font-size: 18px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 70%)')};
|
||||||
|
display: block;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
line-height: 1;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.thumb-image {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-meta {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 4px;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-name {
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: 14px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(222 47% 11%)', 'hsl(210 20% 96%)')};
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-details {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
font-size: 12px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 46%)', 'hsl(215 16% 70%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-size {
|
||||||
|
font-variant-numeric: tabular-nums;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-type {
|
||||||
|
padding: 2px 8px;
|
||||||
|
border-radius: 999px;
|
||||||
|
border: 1px solid ${cssManager.bdTheme('hsl(214 31% 86%)', 'hsl(217 32% 28%)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 46%)', 'hsl(215 16% 70%)')};
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.file-actions {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
margin-left: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.remove-button {
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border-radius: 6px;
|
||||||
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: background 0.15s ease, transform 0.15s ease, color 0.15s ease;
|
||||||
|
color: ${cssManager.bdTheme('hsl(215 16% 52%)', 'hsl(215 16% 68%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.remove-button:hover {
|
||||||
|
background: ${cssManager.bdTheme('hsl(0 72% 50% / 0.08)', 'hsl(0 62% 32% / 0.15)')};
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 72% 46%)', 'hsl(0 70% 70%)')};
|
||||||
|
}
|
||||||
|
|
||||||
|
.remove-button:active {
|
||||||
|
transform: scale(0.96);
|
||||||
|
}
|
||||||
|
|
||||||
|
.remove-button dees-icon {
|
||||||
|
display: block;
|
||||||
|
width: 14px;
|
||||||
|
height: 14px;
|
||||||
|
font-size: 14px;
|
||||||
|
line-height: 1;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.validation-message {
|
||||||
|
font-size: 13px;
|
||||||
|
color: ${cssManager.bdTheme('hsl(0 72% 40%)', 'hsl(0 70% 68%)')};
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes loader-spin {
|
||||||
|
to {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
cssManager,
|
cssManager,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import * as domtools from '@design.estate/dees-domtools';
|
import * as domtools from '@design.estate/dees-domtools';
|
||||||
import { DeesInputBase } from './dees-input-base.js';
|
import { DeesInputBase } from '../dees-input-base/dees-input-base.js';
|
||||||
import * as ibantools from 'ibantools';
|
import * as ibantools from 'ibantools';
|
||||||
import { demoFunc } from './dees-input-iban.demo.js';
|
import { demoFunc } from './dees-input-iban.demo.js';
|
||||||
|
|
||||||
@@ -19,15 +19,15 @@ export class DeesInputIban extends DeesInputBase<DeesInputIban> {
|
|||||||
|
|
||||||
// INSTANCE
|
// INSTANCE
|
||||||
@state()
|
@state()
|
||||||
enteredString: string = '';
|
accessor enteredString: string = '';
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
enteredIbanIsValid: boolean = false;
|
accessor enteredIbanIsValid: boolean = false;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String,
|
type: String,
|
||||||
})
|
})
|
||||||
public value = '';
|
accessor value = '';
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
...DeesInputBase.baseStyles,
|
...DeesInputBase.baseStyles,
|
||||||
1
ts_web/elements/00group-input/dees-input-iban/index.ts
Normal file
1
ts_web/elements/00group-input/dees-input-iban/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './dees-input-iban.js';
|
||||||
@@ -0,0 +1,275 @@
|
|||||||
|
import { html, css } from '@design.estate/dees-element';
|
||||||
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
import '../../dees-panel/dees-panel.js';
|
||||||
|
import '../../00group-form/dees-form/dees-form.js';
|
||||||
|
import '../dees-input-text/dees-input-text.js';
|
||||||
|
import '../../00group-form/dees-form-submit/dees-form-submit.js';
|
||||||
|
|
||||||
|
export const demoFunc = () => html`
|
||||||
|
<dees-demowrapper>
|
||||||
|
<style>
|
||||||
|
${css`
|
||||||
|
.demo-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 24px;
|
||||||
|
padding: 24px;
|
||||||
|
max-width: 1200px;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
dees-panel {
|
||||||
|
margin-bottom: 24px;
|
||||||
|
}
|
||||||
|
|
||||||
|
dees-panel:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid-layout {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.grid-layout {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.output-preview {
|
||||||
|
margin-top: 16px;
|
||||||
|
padding: 16px;
|
||||||
|
background: #f3f4f6;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: #374151;
|
||||||
|
word-break: break-all;
|
||||||
|
max-height: 200px;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
.output-preview {
|
||||||
|
background: #2c2c2c;
|
||||||
|
color: #e4e4e7;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.feature-note {
|
||||||
|
margin-top: 12px;
|
||||||
|
padding: 12px;
|
||||||
|
background: #eff6ff;
|
||||||
|
border-left: 3px solid #3b82f6;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #1e40af;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
.feature-note {
|
||||||
|
background: #1e3a5f;
|
||||||
|
color: #93c5fd;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div class="demo-container">
|
||||||
|
<dees-panel .title=${'1. Basic List Input'} .subtitle=${'Simple list management with add, edit, and delete'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Shopping List'}
|
||||||
|
.placeholder=${'Add item to your list...'}
|
||||||
|
.value=${['Milk', 'Bread', 'Eggs', 'Cheese']}
|
||||||
|
.description=${'Double-click to edit items, or use the edit button'}
|
||||||
|
></dees-input-list>
|
||||||
|
<div class="feature-note">
|
||||||
|
💡 Double-click any item to quickly edit it inline
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'2. Sortable List'} .subtitle=${'Drag and drop to reorder items'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Task Priority'}
|
||||||
|
.placeholder=${'Add a task...'}
|
||||||
|
.sortable=${true}
|
||||||
|
.value=${[
|
||||||
|
'Review pull requests',
|
||||||
|
'Fix critical bug',
|
||||||
|
'Update documentation',
|
||||||
|
'Deploy to production',
|
||||||
|
'Team standup meeting'
|
||||||
|
]}
|
||||||
|
.description=${'Drag items using the handle to reorder them'}
|
||||||
|
></dees-input-list>
|
||||||
|
<div class="feature-note">
|
||||||
|
🔄 Drag the grip handle to reorder tasks by priority
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'3. Validation & Constraints'} .subtitle=${'Lists with minimum/maximum items and duplicate prevention'}>
|
||||||
|
<div class="grid-layout">
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Team Members (Min 2, Max 5)'}
|
||||||
|
.placeholder=${'Add team member...'}
|
||||||
|
.minItems=${2}
|
||||||
|
.maxItems=${5}
|
||||||
|
.value=${['Alice', 'Bob']}
|
||||||
|
.required=${true}
|
||||||
|
.description=${'Add 2-5 team members'}
|
||||||
|
></dees-input-list>
|
||||||
|
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Unique Tags (No Duplicates)'}
|
||||||
|
.placeholder=${'Add unique tag...'}
|
||||||
|
.allowDuplicates=${false}
|
||||||
|
.value=${['frontend', 'backend', 'database']}
|
||||||
|
.description=${'Duplicate items are not allowed'}
|
||||||
|
></dees-input-list>
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'4. Delete Confirmation'} .subtitle=${'Require confirmation before deleting items'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Important Documents'}
|
||||||
|
.placeholder=${'Add document name...'}
|
||||||
|
.confirmDelete=${true}
|
||||||
|
.value=${[
|
||||||
|
'Contract_2024.pdf',
|
||||||
|
'Financial_Report_Q3.xlsx',
|
||||||
|
'Project_Proposal.docx',
|
||||||
|
'Meeting_Notes.txt'
|
||||||
|
]}
|
||||||
|
.description=${'Deletion requires confirmation for safety'}
|
||||||
|
></dees-input-list>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'5. Disabled State'} .subtitle=${'Read-only list display'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'System Defaults'}
|
||||||
|
.value=${['Default Setting 1', 'Default Setting 2', 'Default Setting 3']}
|
||||||
|
.disabled=${true}
|
||||||
|
.description=${'These items cannot be modified'}
|
||||||
|
></dees-input-list>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'6. Form Integration'} .subtitle=${'List input working within a form context'}>
|
||||||
|
<dees-form>
|
||||||
|
<dees-input-text
|
||||||
|
.label=${'Recipe Name'}
|
||||||
|
.placeholder=${'My Amazing Recipe'}
|
||||||
|
.required=${true}
|
||||||
|
.key=${'name'}
|
||||||
|
></dees-input-text>
|
||||||
|
|
||||||
|
<div class="grid-layout">
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Ingredients'}
|
||||||
|
.placeholder=${'Add ingredient...'}
|
||||||
|
.required=${true}
|
||||||
|
.minItems=${3}
|
||||||
|
.key=${'ingredients'}
|
||||||
|
.sortable=${true}
|
||||||
|
.value=${[
|
||||||
|
'2 cups flour',
|
||||||
|
'1 cup sugar',
|
||||||
|
'3 eggs'
|
||||||
|
]}
|
||||||
|
.description=${'Add at least 3 ingredients'}
|
||||||
|
></dees-input-list>
|
||||||
|
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Instructions'}
|
||||||
|
.placeholder=${'Add instruction step...'}
|
||||||
|
.required=${true}
|
||||||
|
.minItems=${2}
|
||||||
|
.key=${'instructions'}
|
||||||
|
.sortable=${true}
|
||||||
|
.value=${[
|
||||||
|
'Preheat oven to 350°F',
|
||||||
|
'Mix dry ingredients'
|
||||||
|
]}
|
||||||
|
.description=${'Add cooking instructions in order'}
|
||||||
|
></dees-input-list>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<dees-input-text
|
||||||
|
.label=${'Notes'}
|
||||||
|
.inputType=${'textarea'}
|
||||||
|
.placeholder=${'Any special notes or tips...'}
|
||||||
|
.key=${'notes'}
|
||||||
|
></dees-input-text>
|
||||||
|
|
||||||
|
<dees-form-submit .text=${'Save Recipe'}></dees-form-submit>
|
||||||
|
</dees-form>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'7. Interactive Demo'} .subtitle=${'Build your own feature list and see the data'}>
|
||||||
|
<dees-input-list
|
||||||
|
id="interactive-list"
|
||||||
|
.label=${'Product Features'}
|
||||||
|
.placeholder=${'Add a feature...'}
|
||||||
|
.sortable=${true}
|
||||||
|
.confirmDelete=${false}
|
||||||
|
.allowDuplicates=${false}
|
||||||
|
.maxItems=${10}
|
||||||
|
@change=${(e: CustomEvent) => {
|
||||||
|
const preview = document.querySelector('#list-json');
|
||||||
|
if (preview) {
|
||||||
|
const data = {
|
||||||
|
items: e.detail.value,
|
||||||
|
count: e.detail.value.length,
|
||||||
|
timestamp: new Date().toISOString()
|
||||||
|
};
|
||||||
|
preview.textContent = JSON.stringify(data, null, 2);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
></dees-input-list>
|
||||||
|
|
||||||
|
<div class="output-preview" id="list-json">
|
||||||
|
{
|
||||||
|
"items": [],
|
||||||
|
"count": 0,
|
||||||
|
"timestamp": "${new Date().toISOString()}"
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="feature-note">
|
||||||
|
✨ Add, edit, remove, and reorder items to see the JSON output update in real-time
|
||||||
|
</div>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'8. Advanced Configuration'} .subtitle=${'Combine all features for complex use cases'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Project Milestones'}
|
||||||
|
.placeholder=${'Add milestone...'}
|
||||||
|
.value=${[
|
||||||
|
'Project Kickoff - Week 1',
|
||||||
|
'Requirements Gathering - Week 2-3',
|
||||||
|
'Design Phase - Week 4-6',
|
||||||
|
'Development Sprint 1 - Week 7-9',
|
||||||
|
'Testing & QA - Week 10-11',
|
||||||
|
'Deployment - Week 12'
|
||||||
|
]}
|
||||||
|
.sortable=${true}
|
||||||
|
.confirmDelete=${true}
|
||||||
|
.allowDuplicates=${false}
|
||||||
|
.minItems=${3}
|
||||||
|
.maxItems=${12}
|
||||||
|
.required=${true}
|
||||||
|
.description=${'Manage project milestones (3-12 items, sortable, no duplicates)'}
|
||||||
|
></dees-input-list>
|
||||||
|
</dees-panel>
|
||||||
|
|
||||||
|
<dees-panel .title=${'9. Empty State'} .subtitle=${'How the component looks with no items'}>
|
||||||
|
<dees-input-list
|
||||||
|
.label=${'Your Ideas'}
|
||||||
|
.placeholder=${'Share your ideas...'}
|
||||||
|
.value=${[]}
|
||||||
|
.description=${'Start adding items to build your list'}
|
||||||
|
></dees-input-list>
|
||||||
|
</dees-panel>
|
||||||
|
</div>
|
||||||
|
</dees-demowrapper>
|
||||||
|
`;
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user