Compare commits
40 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 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 |
1
.serena/.gitignore
vendored
1
.serena/.gitignore
vendored
@@ -1 +0,0 @@
|
|||||||
/cache
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
Before finishing a task:
|
|
||||||
- Run `pnpm run build` to ensure TypeScript compile + bundling succeed.
|
|
||||||
- Verify `dist_ts_web/` and `dist_bundle/bundle.js` updated.
|
|
||||||
- Optionally run `pnpm run test` and inspect failures.
|
|
||||||
- Avoid changing public APIs unless required; keep changes scoped.
|
|
||||||
- Update readme or inline docs only if user-facing behavior changes.
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
Project: @design.estate/dees-catalog
|
|
||||||
Purpose: A component library of dynamic Web Components (TypeScript) for building modern web apps.
|
|
||||||
Tech stack: TypeScript (ES2022, NodeNext), decorators, custom elements via @design.estate/dees-element (Lit-style), bundling with esbuild via @git.zone/tsbundle, TypeScript building via @git.zone/tsbuild (tsfolders), tests with @git.zone/tstest, various UI libs (tiptap, apexcharts, monaco-editor runtime via CDN), DOM helpers via @design.estate/dees-domtools.
|
|
||||||
Structure:
|
|
||||||
- ts_web/: source of web components and pages
|
|
||||||
- dist_ts_web/: transpiled TS output
|
|
||||||
- dist_bundle/: production bundle (bundle.js + map)
|
|
||||||
- test/: tests
|
|
||||||
- html/: static demo assets
|
|
||||||
Key configs: tsconfig.json sets ES2022, NodeNext module/resolution, decorators enabled, skipLibCheck enabled to avoid third-party d.ts issues.
|
|
||||||
Entrypoints: ts_web/index.ts for bundling; custom elements annotated with @customElement.
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
Language: TypeScript, ES2022 target, NodeNext module + resolution.
|
|
||||||
Patterns: Web Components with @customElement decorators; class-based components extending DeesElement; styles via css/cssManager; template render via html tagged literal.
|
|
||||||
Typing: Prefer explicit types where practical; tolerate `any` for external browser-injected libs (e.g., monaco) to keep build healthy.
|
|
||||||
Config: skipLibCheck enabled to avoid third-party d.ts breakages; exclude built declaration outputs.
|
|
||||||
Formatting/Linting: Not explicitly configured; follow existing style (2-space indents, single quotes often, semicolons present).
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
Build: pnpm run build
|
|
||||||
Watch: pnpm run watch
|
|
||||||
Test: pnpm run test
|
|
||||||
Docs: pnpm run buildDocs
|
|
||||||
Inspect bundle size: ls -lh dist_bundle/bundle.js
|
|
||||||
Open demo (if applicable): serve static `html/` with any web server
|
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
# language of the project (csharp, python, rust, java, typescript, go, cpp, or ruby)
|
|
||||||
# * For C, use cpp
|
|
||||||
# * For JavaScript, use typescript
|
|
||||||
# Special requirements:
|
|
||||||
# * csharp: Requires the presence of a .sln file in the project folder.
|
|
||||||
language: typescript
|
|
||||||
|
|
||||||
# whether to use the project's gitignore file to ignore files
|
|
||||||
# Added on 2025-04-07
|
|
||||||
ignore_all_files_in_gitignore: true
|
|
||||||
# list of additional paths to ignore
|
|
||||||
# same syntax as gitignore, so you can use * and **
|
|
||||||
# Was previously called `ignored_dirs`, please update your config if you are using that.
|
|
||||||
# Added (renamed) on 2025-04-07
|
|
||||||
ignored_paths: []
|
|
||||||
|
|
||||||
# whether the project is in read-only mode
|
|
||||||
# If set to true, all editing tools will be disabled and attempts to use them will result in an error
|
|
||||||
# Added on 2025-04-18
|
|
||||||
read_only: false
|
|
||||||
|
|
||||||
# list of tool names to exclude. We recommend not excluding any tools, see the readme for more details.
|
|
||||||
# Below is the complete list of tools for convenience.
|
|
||||||
# To make sure you have the latest list of tools, and to view their descriptions,
|
|
||||||
# execute `uv run scripts/print_tool_overview.py`.
|
|
||||||
#
|
|
||||||
# * `activate_project`: Activates a project by name.
|
|
||||||
# * `check_onboarding_performed`: Checks whether project onboarding was already performed.
|
|
||||||
# * `create_text_file`: Creates/overwrites a file in the project directory.
|
|
||||||
# * `delete_lines`: Deletes a range of lines within a file.
|
|
||||||
# * `delete_memory`: Deletes a memory from Serena's project-specific memory store.
|
|
||||||
# * `execute_shell_command`: Executes a shell command.
|
|
||||||
# * `find_referencing_code_snippets`: Finds code snippets in which the symbol at the given location is referenced.
|
|
||||||
# * `find_referencing_symbols`: Finds symbols that reference the symbol at the given location (optionally filtered by type).
|
|
||||||
# * `find_symbol`: Performs a global (or local) search for symbols with/containing a given name/substring (optionally filtered by type).
|
|
||||||
# * `get_current_config`: Prints the current configuration of the agent, including the active and available projects, tools, contexts, and modes.
|
|
||||||
# * `get_symbols_overview`: Gets an overview of the top-level symbols defined in a given file.
|
|
||||||
# * `initial_instructions`: Gets the initial instructions for the current project.
|
|
||||||
# Should only be used in settings where the system prompt cannot be set,
|
|
||||||
# e.g. in clients you have no control over, like Claude Desktop.
|
|
||||||
# * `insert_after_symbol`: Inserts content after the end of the definition of a given symbol.
|
|
||||||
# * `insert_at_line`: Inserts content at a given line in a file.
|
|
||||||
# * `insert_before_symbol`: Inserts content before the beginning of the definition of a given symbol.
|
|
||||||
# * `list_dir`: Lists files and directories in the given directory (optionally with recursion).
|
|
||||||
# * `list_memories`: Lists memories in Serena's project-specific memory store.
|
|
||||||
# * `onboarding`: Performs onboarding (identifying the project structure and essential tasks, e.g. for testing or building).
|
|
||||||
# * `prepare_for_new_conversation`: Provides instructions for preparing for a new conversation (in order to continue with the necessary context).
|
|
||||||
# * `read_file`: Reads a file within the project directory.
|
|
||||||
# * `read_memory`: Reads the memory with the given name from Serena's project-specific memory store.
|
|
||||||
# * `remove_project`: Removes a project from the Serena configuration.
|
|
||||||
# * `replace_lines`: Replaces a range of lines within a file with new content.
|
|
||||||
# * `replace_symbol_body`: Replaces the full definition of a symbol.
|
|
||||||
# * `restart_language_server`: Restarts the language server, may be necessary when edits not through Serena happen.
|
|
||||||
# * `search_for_pattern`: Performs a search for a pattern in the project.
|
|
||||||
# * `summarize_changes`: Provides instructions for summarizing the changes made to the codebase.
|
|
||||||
# * `switch_modes`: Activates modes by providing a list of their names
|
|
||||||
# * `think_about_collected_information`: Thinking tool for pondering the completeness of collected information.
|
|
||||||
# * `think_about_task_adherence`: Thinking tool for determining whether the agent is still on track with the current task.
|
|
||||||
# * `think_about_whether_you_are_done`: Thinking tool for determining whether the task is truly completed.
|
|
||||||
# * `write_memory`: Writes a named memory (for future reference) to Serena's project-specific memory store.
|
|
||||||
excluded_tools: []
|
|
||||||
|
|
||||||
# initial prompt for the project. It will always be given to the LLM upon activating the project
|
|
||||||
# (contrary to the memories, which are loaded on demand).
|
|
||||||
initial_prompt: ""
|
|
||||||
|
|
||||||
project_name: "dees-catalog"
|
|
||||||
83
changelog.md
83
changelog.md
@@ -1,5 +1,88 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## 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)
|
## 2025-09-18 - 1.12.1 - fix(ci)
|
||||||
Add local settings to allow running pnpm scripts and enable dev chat permission
|
Add local settings to allow running pnpm scripts and enable dev chat permission
|
||||||
|
|
||||||
|
|||||||
33
package.json
33
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@design.estate/dees-catalog",
|
"name": "@design.estate/dees-catalog",
|
||||||
"version": "1.12.1",
|
"version": "2.0.4",
|
||||||
"private": false,
|
"private": false,
|
||||||
"description": "A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.",
|
"description": "A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.",
|
||||||
"main": "dist_ts_web/index.js",
|
"main": "dist_ts_web/index.js",
|
||||||
@@ -16,13 +16,13 @@
|
|||||||
"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.1.0",
|
"@push.rocks/smartstring": "^4.1.0",
|
||||||
@@ -32,25 +32,26 @@
|
|||||||
"@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.5",
|
"apexcharts": "^5.3.6",
|
||||||
"highlight.js": "11.11.1",
|
"highlight.js": "11.11.1",
|
||||||
"ibantools": "^4.5.1",
|
"ibantools": "^4.5.1",
|
||||||
"lucide": "^0.544.0",
|
"lit": "^3.3.1",
|
||||||
|
"lucide": "^0.555.0",
|
||||||
"monaco-editor": "0.52.2",
|
"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.2",
|
||||||
"@git.zone/tstest": "^2.3.8",
|
"@git.zone/tstest": "^3.1.3",
|
||||||
"@git.zone/tswatch": "^2.2.1",
|
"@git.zone/tswatch": "^2.2.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/**/*",
|
||||||
|
|||||||
3927
pnpm-lock.yaml
generated
3927
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -1,2 +0,0 @@
|
|||||||
onlyBuiltDependencies:
|
|
||||||
- puppeteer
|
|
||||||
@@ -604,4 +604,80 @@ import { zIndexLayers } from './00zindex.js';
|
|||||||
z-index: ${zIndexLayers.overlay.modal};
|
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
|
||||||
BIN
readme.plan.md
BIN
readme.plan.md
Binary file not shown.
@@ -3,6 +3,6 @@
|
|||||||
*/
|
*/
|
||||||
export const commitinfo = {
|
export const commitinfo = {
|
||||||
name: '@design.estate/dees-catalog',
|
name: '@design.estate/dees-catalog',
|
||||||
version: '1.12.1',
|
version: '2.0.4',
|
||||||
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 {
|
||||||
1
ts_web/elements/dees-appui-activitylog/index.ts
Normal file
1
ts_web/elements/dees-appui-activitylog/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-activitylog/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
|
||||||
3
ts_web/elements/dees-appui-appbar/index.ts
Normal file
3
ts_web/elements/dees-appui-appbar/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from './component.js';
|
||||||
|
export { appuiAppbarStyles } from './styles.js';
|
||||||
|
export { renderAppuiAppbar } from './template.js';
|
||||||
238
ts_web/elements/dees-appui-appbar/styles.ts
Normal file
238
ts_web/elements/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/dees-appui-appbar/template.ts
Normal file
18
ts_web/elements/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>
|
||||||
|
`;
|
||||||
|
|
||||||
|
};
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
import { html, css } from '@design.estate/dees-element';
|
import { html, css } from '@design.estate/dees-element';
|
||||||
import type { DeesAppuiBase } from './dees-appui-base.js';
|
import type { DeesAppuiBase } from '../dees-appui-base/dees-appui-base.js';
|
||||||
import type { IAppBarMenuItem } from './interfaces/appbarmenuitem.js';
|
import type { IAppBarMenuItem } from '../interfaces/appbarmenuitem.js';
|
||||||
import type { ITab } from './interfaces/tab.js';
|
import type { ITab } from '../interfaces/tab.js';
|
||||||
import type { ISelectionOption } from './interfaces/selectionoption.js';
|
import type { ISelectionOption } from '../interfaces/selectionoption.js';
|
||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
|
||||||
export const demoFunc = () => {
|
export const demoFunc = () => {
|
||||||
@@ -8,21 +8,21 @@ import {
|
|||||||
cssManager,
|
cssManager,
|
||||||
state,
|
state,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
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 type { DeesAppuiBar } from './dees-appui-appbar.js';
|
import type { DeesAppuiBar } from '../dees-appui-appbar/index.js';
|
||||||
import type { DeesAppuiMainmenu } from './dees-appui-mainmenu.js';
|
import type { DeesAppuiMainmenu } from '../dees-appui-mainmenu/dees-appui-mainmenu.js';
|
||||||
import type { DeesAppuiMainselector } from './dees-appui-mainselector.js';
|
import type { DeesAppuiMainselector } from '../dees-appui-mainselector/dees-appui-mainselector.js';
|
||||||
import type { DeesAppuiMaincontent } from './dees-appui-maincontent.js';
|
import type { DeesAppuiMaincontent } from '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||||
import type { DeesAppuiActivitylog } from './dees-appui-activitylog.js';
|
import type { DeesAppuiActivitylog } from '../dees-appui-activitylog/dees-appui-activitylog.js';
|
||||||
import { demoFunc } from './dees-appui-base.demo.js';
|
import { demoFunc } from './dees-appui-base.demo.js';
|
||||||
|
|
||||||
// Import child components
|
// Import child components
|
||||||
import './dees-appui-appbar.js';
|
import '../dees-appui-appbar/index.js';
|
||||||
import './dees-appui-mainmenu.js';
|
import '../dees-appui-mainmenu/dees-appui-mainmenu.js';
|
||||||
import './dees-appui-mainselector.js';
|
import '../dees-appui-mainselector/dees-appui-mainselector.js';
|
||||||
import './dees-appui-maincontent.js';
|
import '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||||
import './dees-appui-activitylog.js';
|
import '../dees-appui-activitylog/dees-appui-activitylog.js';
|
||||||
|
|
||||||
@customElement('dees-appui-base')
|
@customElement('dees-appui-base')
|
||||||
export class DeesAppuiBase extends DeesElement {
|
export class DeesAppuiBase extends DeesElement {
|
||||||
@@ -30,65 +30,65 @@ export class DeesAppuiBase extends DeesElement {
|
|||||||
|
|
||||||
// Properties for appbar
|
// Properties for appbar
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public appbarMenuItems: interfaces.IAppBarMenuItem[] = [];
|
accessor appbarMenuItems: interfaces.IAppBarMenuItem[] = [];
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public appbarBreadcrumbs: string = '';
|
accessor appbarBreadcrumbs: string = '';
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public appbarBreadcrumbSeparator: string = ' > ';
|
accessor appbarBreadcrumbSeparator: string = ' > ';
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public appbarShowWindowControls: boolean = true;
|
accessor appbarShowWindowControls: boolean = true;
|
||||||
|
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public appbarUser?: {
|
accessor appbarUser: {
|
||||||
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 appbarProfileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
accessor appbarProfileMenuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string } | { divider: true })[] = [];
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public appbarShowSearch: boolean = false;
|
accessor appbarShowSearch: boolean = false;
|
||||||
|
|
||||||
// Properties for mainmenu
|
// Properties for mainmenu
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public mainmenuTabs: interfaces.ITab[] = [];
|
accessor mainmenuTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public mainmenuSelectedTab?: interfaces.ITab;
|
accessor mainmenuSelectedTab: interfaces.ITab | undefined = undefined;
|
||||||
|
|
||||||
// Properties for mainselector
|
// Properties for mainselector
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public mainselectorOptions: (interfaces.ISelectionOption | { divider: true })[] = [];
|
accessor mainselectorOptions: (interfaces.ISelectionOption | { divider: true })[] = [];
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public mainselectorSelectedOption?: interfaces.ISelectionOption;
|
accessor mainselectorSelectedOption: interfaces.ISelectionOption | undefined = undefined;
|
||||||
|
|
||||||
// Properties for maincontent
|
// Properties for maincontent
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public maincontentTabs: interfaces.ITab[] = [];
|
accessor maincontentTabs: interfaces.ITab[] = [];
|
||||||
|
|
||||||
// References to child components
|
// References to child components
|
||||||
@state()
|
@state()
|
||||||
public appbar?: DeesAppuiBar;
|
accessor appbar: DeesAppuiBar | undefined = undefined;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public mainmenu?: DeesAppuiMainmenu;
|
accessor mainmenu: DeesAppuiMainmenu | undefined = undefined;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public mainselector?: DeesAppuiMainselector;
|
accessor mainselector: DeesAppuiMainselector | undefined = undefined;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public maincontent?: DeesAppuiMaincontent;
|
accessor maincontent: DeesAppuiMaincontent | undefined = undefined;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
public activitylog?: DeesAppuiActivitylog;
|
accessor activitylog: DeesAppuiActivitylog | undefined = undefined;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
1
ts_web/elements/dees-appui-base/index.ts
Normal file
1
ts_web/elements/dees-appui-base/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-base/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,
|
||||||
1
ts_web/elements/dees-appui-maincontent/index.ts
Normal file
1
ts_web/elements/dees-appui-maincontent/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-maincontent/dees-appui-maincontent.js';
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../interfaces/index.js';
|
||||||
import { zIndexLayers } from './00zindex.js';
|
import { zIndexLayers } from '../00zindex.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -11,7 +11,7 @@ import {
|
|||||||
css,
|
css,
|
||||||
cssManager,
|
cssManager,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* the most left menu
|
* the most left menu
|
||||||
@@ -34,12 +34,12 @@ export class DeesAppuiMainmenu extends DeesElement {
|
|||||||
|
|
||||||
// INSTANCE
|
// INSTANCE
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public tabs: interfaces.ITab[] = [
|
accessor tabs: interfaces.ITab[] = [
|
||||||
{ key: '⚠️ Please set tabs', iconName: 'lucide:alertTriangle', action: () => console.warn('No tabs configured for mainmenu') },
|
{ key: '⚠️ Please set tabs', iconName: 'lucide:alertTriangle', action: () => console.warn('No tabs configured for mainmenu') },
|
||||||
];
|
];
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public selectedTab: interfaces.ITab;
|
accessor selectedTab: interfaces.ITab;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
1
ts_web/elements/dees-appui-mainmenu/index.ts
Normal file
1
ts_web/elements/dees-appui-mainmenu/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-mainmenu/dees-appui-mainmenu.js';
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
import * as interfaces from './interfaces/index.js';
|
import * as interfaces from '../interfaces/index.js';
|
||||||
|
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../dees-contextmenu/dees-contextmenu.js';
|
||||||
import './dees-icon.js';
|
import '../dees-icon/dees-icon.js';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DeesElement,
|
DeesElement,
|
||||||
@@ -34,12 +34,12 @@ export class DeesAppuiMainselector extends DeesElement {
|
|||||||
|
|
||||||
// INSTANCE
|
// INSTANCE
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public selectionOptions: (interfaces.ISelectionOption | { divider: true })[] = [
|
accessor selectionOptions: (interfaces.ISelectionOption | { divider: true })[] = [
|
||||||
{ key: '⚠️ Please set selection options', action: () => console.warn('No selection options configured for mainselector') },
|
{ key: '⚠️ Please set selection options', action: () => console.warn('No selection options configured for mainselector') },
|
||||||
];
|
];
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public selectedOption: interfaces.ISelectionOption = null;
|
accessor selectedOption: interfaces.ISelectionOption = null;
|
||||||
|
|
||||||
public static styles = [
|
public static styles = [
|
||||||
cssManager.defaultStyles,
|
cssManager.defaultStyles,
|
||||||
1
ts_web/elements/dees-appui-mainselector/index.ts
Normal file
1
ts_web/elements/dees-appui-mainselector/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-mainselector/dees-appui-mainselector.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,
|
||||||
1
ts_web/elements/dees-appui-profiledropdown/index.ts
Normal file
1
ts_web/elements/dees-appui-profiledropdown/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-profiledropdown/dees-appui-profiledropdown.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,
|
||||||
1
ts_web/elements/dees-appui-tabs/index.ts
Normal file
1
ts_web/elements/dees-appui-tabs/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-tabs/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/dees-appui-view/index.ts
Normal file
1
ts_web/elements/dees-appui-view/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-appui-view/dees-appui-view.js';
|
||||||
@@ -23,13 +23,13 @@ export class DeesBadge extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public type: 'default' | 'primary' | 'success' | 'warning' | 'error' = 'default';
|
accessor type: 'default' | 'primary' | 'success' | 'warning' | 'error' = 'default';
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public text: string = '';
|
accessor text: string = '';
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public rounded: boolean = false;
|
accessor rounded: boolean = false;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
1
ts_web/elements/dees-badge/index.ts
Normal file
1
ts_web/elements/dees-badge/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-badge/dees-badge.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/dees-button-exit/index.ts
Normal file
1
ts_web/elements/dees-button-exit/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-button-exit/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();
|
||||||
1
ts_web/elements/dees-button-group/index.ts
Normal file
1
ts_web/elements/dees-button-group/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-button-group/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 '../dees-form/dees-form.js';
|
||||||
import './dees-form-submit.js';
|
import '../dees-form-submit/dees-form-submit.js';
|
||||||
import './dees-input-text.js';
|
import '../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({
|
|
||||||
type: String
|
|
||||||
})
|
|
||||||
public size: 'default' | 'sm' | 'lg' | 'icon' = 'default';
|
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: String
|
type: String
|
||||||
})
|
})
|
||||||
public status: 'normal' | 'pending' | 'success' | 'error' = 'normal';
|
accessor size: 'default' | 'sm' | 'lg' | 'icon' = 'default';
|
||||||
|
|
||||||
|
@property({
|
||||||
|
type: String
|
||||||
|
})
|
||||||
|
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/dees-button/index.ts
Normal file
1
ts_web/elements/dees-button/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-button/dees-button.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,36 +25,36 @@ 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 {
|
||||||
return this.internalChartData.length > 0 ? this.internalChartData : this.series;
|
return this.internalChartData.length > 0 ? this.internalChartData : this.series;
|
||||||
}
|
}
|
||||||
|
|
||||||
@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
|
||||||
3
ts_web/elements/dees-chart-area/index.ts
Normal file
3
ts_web/elements/dees-chart-area/index.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
export * from './component.js';
|
||||||
|
export { chartAreaStyles } from './styles.js';
|
||||||
|
export { renderChartArea } from './template.js';
|
||||||
60
ts_web/elements/dees-chart-area/styles.ts
Normal file
60
ts_web/elements/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/dees-chart-area/template.ts
Normal file
12
ts_web/elements/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/dees-chart-log/index.ts
Normal file
1
ts_web/elements/dees-chart-log/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-chart-log/dees-chart-log.js';
|
||||||
@@ -26,25 +26,25 @@ export class DeesChips extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public selectionMode: 'none' | 'single' | 'multiple' = 'single';
|
accessor selectionMode: 'none' | 'single' | 'multiple' = 'single';
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
})
|
})
|
||||||
public chipsAreRemovable: boolean = false;
|
accessor chipsAreRemovable: boolean = false;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Array,
|
type: Array,
|
||||||
})
|
})
|
||||||
public selectableChips: Tag[] = [];
|
accessor selectableChips: Tag[] = [];
|
||||||
|
|
||||||
@property()
|
@property()
|
||||||
public selectedChip: Tag = null;
|
accessor selectedChip: Tag = null;
|
||||||
|
|
||||||
@property({
|
@property({
|
||||||
type: Array,
|
type: Array,
|
||||||
})
|
})
|
||||||
public selectedChips: Tag[] = [];
|
accessor selectedChips: Tag[] = [];
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
1
ts_web/elements/dees-chips/index.ts
Normal file
1
ts_web/elements/dees-chips/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-chips/dees-chips.js';
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { html } from '@design.estate/dees-element';
|
import { html } from '@design.estate/dees-element';
|
||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
|
|
||||||
import { DeesContextmenu } from './dees-contextmenu.js';
|
import { DeesContextmenu } from '../dees-contextmenu/dees-contextmenu.js';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
<style>
|
<style>
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import * as plugins from './00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
import { demoFunc } from './dees-contextmenu.demo.js';
|
import { demoFunc } from './dees-contextmenu.demo.js';
|
||||||
import {
|
import {
|
||||||
customElement,
|
customElement,
|
||||||
@@ -13,9 +13,9 @@ 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 { DeesWindowLayer } from './dees-windowlayer.js';
|
import { DeesWindowLayer } from '../dees-windowlayer/dees-windowlayer.js';
|
||||||
import { zIndexLayers } from './00zindex.js';
|
import { zIndexLayers } from '../00zindex.js';
|
||||||
import './dees-icon.js';
|
import '../dees-icon/dees-icon.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -127,7 +127,7 @@ export class DeesContextmenu extends DeesElement {
|
|||||||
@property({
|
@property({
|
||||||
type: Array,
|
type: Array,
|
||||||
})
|
})
|
||||||
public menuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string; disabled?: boolean; submenu?: (plugins.tsclass.website.IMenuItem & { shortcut?: string; disabled?: boolean } | { divider: true })[]; divider?: never } | { divider: true })[] = [];
|
accessor menuItems: (plugins.tsclass.website.IMenuItem & { shortcut?: string; disabled?: boolean; submenu?: (plugins.tsclass.website.IMenuItem & { shortcut?: string; disabled?: boolean } | { divider: true })[]; divider?: never } | { divider: true })[] = [];
|
||||||
windowLayer: DeesWindowLayer;
|
windowLayer: DeesWindowLayer;
|
||||||
|
|
||||||
private submenu: DeesContextmenu | null = null;
|
private submenu: DeesContextmenu | null = null;
|
||||||
1
ts_web/elements/dees-contextmenu/index.ts
Normal file
1
ts_web/elements/dees-contextmenu/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-contextmenu/dees-contextmenu.js';
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { DashboardWidget } from './types.js';
|
import type { DashboardWidget } from './types.js';
|
||||||
import { DeesContextmenu } from '../dees-contextmenu.js';
|
import { DeesContextmenu } from '../dees-contextmenu/dees-contextmenu.js';
|
||||||
import type { DeesDashboardgrid } from './dees-dashboardgrid.js';
|
import type { DeesDashboardgrid } from './dees-dashboardgrid.js';
|
||||||
import * as plugins from '../00plugins.js';
|
import * as plugins from '../00plugins.js';
|
||||||
|
|
||||||
|
|||||||
@@ -7,8 +7,8 @@ import {
|
|||||||
type TemplateResult,
|
type TemplateResult,
|
||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import '../dees-icon.js';
|
import '../dees-icon/dees-icon.js';
|
||||||
import '../dees-contextmenu.js';
|
import '../dees-contextmenu/dees-contextmenu.js';
|
||||||
import { demoFunc } from './dees-dashboardgrid.demo.js';
|
import { demoFunc } from './dees-dashboardgrid.demo.js';
|
||||||
import { dashboardGridStyles } from './styles.js';
|
import { dashboardGridStyles } from './styles.js';
|
||||||
import {
|
import {
|
||||||
@@ -71,49 +71,49 @@ export class DeesDashboardgrid extends DeesElement {
|
|||||||
public static styles = dashboardGridStyles;
|
public static styles = dashboardGridStyles;
|
||||||
|
|
||||||
@property({ type: Array })
|
@property({ type: Array })
|
||||||
public widgets: DashboardWidget[] = [];
|
accessor widgets: DashboardWidget[] = [];
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public cellHeight: number = 80;
|
accessor cellHeight: number = 80;
|
||||||
|
|
||||||
@property({ type: Object })
|
@property({ type: Object })
|
||||||
public margin: DashboardMargin = 10;
|
accessor margin: DashboardMargin = 10;
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public columns: number = 12;
|
accessor columns: number = 12;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public editable: boolean = true;
|
accessor editable: boolean = true;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true })
|
@property({ type: Boolean, reflect: true })
|
||||||
public enableAnimation: boolean = true;
|
accessor enableAnimation: boolean = true;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public cellHeightUnit: CellHeightUnit = 'px';
|
accessor cellHeightUnit: CellHeightUnit = 'px';
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public rtl: boolean = false;
|
accessor rtl: boolean = false;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
public showGridLines: boolean = false;
|
accessor showGridLines: boolean = false;
|
||||||
|
|
||||||
@property({ attribute: false })
|
@property({ attribute: false })
|
||||||
public layouts?: Record<string, DashboardLayoutItem[]>;
|
accessor layouts: Record<string, DashboardLayoutItem[]> | undefined = undefined;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public activeBreakpoint: string = 'base';
|
accessor activeBreakpoint: string = 'base';
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private placeholderPosition: DashboardLayoutItem | null = null;
|
accessor placeholderPosition: DashboardLayoutItem | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private metrics: GridCellMetrics | null = null;
|
accessor metrics: GridCellMetrics | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private resolvedMargins: DashboardResolvedMargins | null = null;
|
accessor resolvedMargins: DashboardResolvedMargins | null = null;
|
||||||
|
|
||||||
@state()
|
@state()
|
||||||
private previewWidgets: DashboardWidget[] | null = null;
|
accessor previewWidgets: DashboardWidget[] | null = null;
|
||||||
|
|
||||||
private containerBounds: DOMRect | null = null;
|
private containerBounds: DOMRect | null = null;
|
||||||
private dragState: DragState | null = null;
|
private dragState: DragState | null = null;
|
||||||
|
|||||||
@@ -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) {
|
||||||
1
ts_web/elements/dees-dataview-codebox/index.ts
Normal file
1
ts_web/elements/dees-dataview-codebox/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-dataview-codebox/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,
|
||||||
1
ts_web/elements/dees-dataview-statusobject/index.ts
Normal file
1
ts_web/elements/dees-dataview-statusobject/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-dataview-statusobject/dees-dataview-statusobject.js';
|
||||||
1
ts_web/elements/dees-editor-markdown/index.ts
Normal file
1
ts_web/elements/dees-editor-markdown/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-editor-markdown/dees-editor-markdown.js';
|
||||||
1
ts_web/elements/dees-editor-markdownoutlet/index.ts
Normal file
1
ts_web/elements/dees-editor-markdownoutlet/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-editor-markdownoutlet/dees-editor-markdownoutlet.js';
|
||||||
@@ -33,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();
|
||||||
|
|||||||
@@ -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();
|
||||||
1
ts_web/elements/dees-form-submit/index.ts
Normal file
1
ts_web/elements/dees-form-submit/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-form-submit/dees-form-submit.js';
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { html, css, domtools, cssManager } from '@design.estate/dees-element';
|
import { html, css, domtools, cssManager } from '@design.estate/dees-element';
|
||||||
import type { DeesForm } from './dees-form.js';
|
import type { DeesForm } from '../dees-form/dees-form.js';
|
||||||
import '@design.estate/dees-wcctools/demotools';
|
import '@design.estate/dees-wcctools/demotools';
|
||||||
|
|
||||||
export const demoFunc = () => html`
|
export const demoFunc = () => html`
|
||||||
@@ -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 '../dees-input-checkbox/dees-input-checkbox.js';
|
||||||
import { DeesInputDatepicker } from './dees-input-datepicker.js';
|
import { DeesInputDatepicker } from '../dees-input-datepicker/index.js';
|
||||||
import { DeesInputText } from './dees-input-text.js';
|
import { DeesInputText } from '../dees-input-text/dees-input-text.js';
|
||||||
import { DeesInputQuantitySelector } from './dees-input-quantityselector.js';
|
import { DeesInputQuantitySelector } from '../dees-input-quantityselector/dees-input-quantityselector.js';
|
||||||
import { DeesInputRadiogroup } from './dees-input-radiogroup.js';
|
import { DeesInputRadiogroup } from '../dees-input-radiogroup/dees-input-radiogroup.js';
|
||||||
import { DeesInputDropdown } from './dees-input-dropdown.js';
|
import { DeesInputDropdown } from '../dees-input-dropdown/dees-input-dropdown.js';
|
||||||
import { DeesInputFileupload } from './dees-input-fileupload.js';
|
import { DeesInputFileupload } from '../dees-input-fileupload/index.js';
|
||||||
import { DeesInputIban } from './dees-input-iban.js';
|
import { DeesInputIban } from '../dees-input-iban/dees-input-iban.js';
|
||||||
import { DeesInputMultitoggle } from './dees-input-multitoggle.js';
|
import { DeesInputMultitoggle } from '../dees-input-multitoggle/dees-input-multitoggle.js';
|
||||||
import { DeesInputPhone } from './dees-input-phone.js';
|
import { DeesInputPhone } from '../dees-input-phone/dees-input-phone.js';
|
||||||
import { DeesInputTypelist } from './dees-input-typelist.js';
|
import { DeesInputTypelist } from '../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/index.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/dees-form/index.ts
Normal file
1
ts_web/elements/dees-form/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-form/dees-form.js';
|
||||||
@@ -10,7 +10,7 @@ import {
|
|||||||
} from '@design.estate/dees-element';
|
} from '@design.estate/dees-element';
|
||||||
|
|
||||||
import { demoFunc } from './dees-heading.demo.js';
|
import { demoFunc } from './dees-heading.demo.js';
|
||||||
import { cssCalSansFontFamily } from './00fonts.js';
|
import { cssCalSansFontFamily } from '../00fonts.js';
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
@@ -28,7 +28,7 @@ export class DeesHeading extends DeesElement {
|
|||||||
* Heading level: 1-6 for h1-h6, or 'hr' for horizontal rule style
|
* Heading level: 1-6 for h1-h6, or 'hr' for horizontal rule style
|
||||||
*/
|
*/
|
||||||
@property({ type: String, reflect: true })
|
@property({ type: String, reflect: true })
|
||||||
public level: '1' | '2' | '3' | '4' | '5' | '6' | 'hr' | 'hr-small' = '1';
|
accessor level: '1' | '2' | '3' | '4' | '5' | '6' | 'hr' | 'hr-small' = '1';
|
||||||
|
|
||||||
// STATIC STYLES
|
// STATIC STYLES
|
||||||
public static styles: CSSResult[] = [
|
public static styles: CSSResult[] = [
|
||||||
1
ts_web/elements/dees-heading/index.ts
Normal file
1
ts_web/elements/dees-heading/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-heading/dees-heading.js';
|
||||||
@@ -23,7 +23,7 @@ export class DeesHint extends DeesElement {
|
|||||||
public static demo = demoFunc;
|
public static demo = demoFunc;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public type: 'info' | 'warn' | 'error' | 'critical' = 'info';
|
accessor type: 'info' | 'warn' | 'error' | 'critical' = 'info';
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
1
ts_web/elements/dees-hint/index.ts
Normal file
1
ts_web/elements/dees-hint/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-hint/dees-hint.js';
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { html } from '@design.estate/dees-element';
|
import { html } from '@design.estate/dees-element';
|
||||||
import { icons, type IconWithPrefix } from './dees-icon.js';
|
import { icons, type IconWithPrefix } from '../dees-icon/dees-icon.js';
|
||||||
import * as lucideIcons from 'lucide';
|
import * as lucideIcons from 'lucide';
|
||||||
|
|
||||||
export const demoFunc = () => {
|
export const demoFunc = () => {
|
||||||
@@ -190,7 +190,7 @@ export class DeesIcon extends DeesElement {
|
|||||||
toAttribute: (value: TIconKey): string => value
|
toAttribute: (value: TIconKey): string => value
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
public iconFA?: TIconKey;
|
accessor iconFA: TIconKey | undefined = undefined;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The preferred icon property. Use format "fa:iconName" or "lucide:iconName"
|
* The preferred icon property. Use format "fa:iconName" or "lucide:iconName"
|
||||||
@@ -203,16 +203,16 @@ export class DeesIcon extends DeesElement {
|
|||||||
toAttribute: (value: IconWithPrefix): string => value
|
toAttribute: (value: IconWithPrefix): string => value
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
public icon?: IconWithPrefix;
|
accessor icon: IconWithPrefix | undefined = undefined;
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public iconSize: number;
|
accessor iconSize: number;
|
||||||
|
|
||||||
@property({ type: String })
|
@property({ type: String })
|
||||||
public color: string = 'currentColor';
|
accessor color: string = 'currentColor';
|
||||||
|
|
||||||
@property({ type: Number })
|
@property({ type: Number })
|
||||||
public strokeWidth: number = 2;
|
accessor strokeWidth: number = 2;
|
||||||
|
|
||||||
// For tracking when we need to re-render
|
// For tracking when we need to re-render
|
||||||
private lastIcon: IconWithPrefix | TIconKey | null = null;
|
private lastIcon: IconWithPrefix | TIconKey | null = null;
|
||||||
1
ts_web/elements/dees-icon/index.ts
Normal file
1
ts_web/elements/dees-icon/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-icon/dees-icon.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/dees-input-base/index.ts
Normal file
1
ts_web/elements/dees-input-base/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-input-base/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 '../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,10 +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 })
|
@property({ type: Boolean })
|
||||||
public indeterminate: boolean = false;
|
accessor indeterminate: boolean = false;
|
||||||
|
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
1
ts_web/elements/dees-input-checkbox/index.ts
Normal file
1
ts_web/elements/dees-input-checkbox/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from '../dees-input-checkbox/dees-input-checkbox.js';
|
||||||
File diff suppressed because it is too large
Load Diff
624
ts_web/elements/dees-input-datepicker/component.ts
Normal file
624
ts_web/elements/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>
|
||||||
4
ts_web/elements/dees-input-datepicker/index.ts
Normal file
4
ts_web/elements/dees-input-datepicker/index.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export * from './component.js';
|
||||||
|
export { datepickerStyles } from './styles.js';
|
||||||
|
export { renderDatepicker } from './template.js';
|
||||||
|
export type { IDateEvent } from './types.js';
|
||||||
514
ts_web/elements/dees-input-datepicker/styles.ts
Normal file
514
ts_web/elements/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/dees-input-datepicker/template.ts
Normal file
179
ts_web/elements/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>
|
||||||
|
`;
|
||||||
|
|
||||||
|
};
|
||||||
7
ts_web/elements/dees-input-datepicker/types.ts
Normal file
7
ts_web/elements/dees-input-datepicker/types.ts
Normal file
@@ -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
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user