Compare commits
	
		
			5 Commits
		
	
	
		
			master
			...
			auto-jsdoc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 28e3c75ff4 | ||
|   | b2ec3c8c37 | ||
|   | fe37f8fad5 | ||
|   | c46368b141 | ||
|   | c49b0803cd | 
							
								
								
									
										6
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -11,7 +11,7 @@ body: | ||||
|  | ||||
|         **Please do not report issues for custom cards.** | ||||
|  | ||||
|         [fr]: https://github.com/orgs/home-assistant/discussions | ||||
|         [fr]: https://github.com/home-assistant/frontend/discussions | ||||
|         [releases]: https://github.com/home-assistant/home-assistant/releases | ||||
|   - type: checkboxes | ||||
|     attributes: | ||||
| @@ -108,9 +108,9 @@ body: | ||||
|       render: yaml | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: JavaScript errors shown in your browser console/inspector | ||||
|       label: Javascript errors shown in your browser console/inspector | ||||
|       description: > | ||||
|         If you come across any JavaScript or other error logs, e.g., in your | ||||
|         If you come across any Javascript or other error logs, e.g., in your | ||||
|         browser console/inspector please provide them. | ||||
|       render: txt | ||||
|   - type: textarea | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,7 +1,7 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: Request a feature for the UI / Dashboards | ||||
|     url: https://github.com/orgs/home-assistant/discussions | ||||
|     url: https://github.com/home-assistant/frontend/discussions/category_choices | ||||
|     about: Request a new feature for the Home Assistant frontend. | ||||
|   - name: Report a bug that is NOT related to the UI / Dashboards | ||||
|     url: https://github.com/home-assistant/core/issues | ||||
|   | ||||
							
								
								
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,53 +0,0 @@ | ||||
| name: Task | ||||
| description: For staff only - Create a task | ||||
| type: Task | ||||
| body: | ||||
|   - type: markdown | ||||
|     attributes: | ||||
|       value: | | ||||
|         ## ⚠️ RESTRICTED ACCESS | ||||
|  | ||||
|         **This form is restricted to Open Home Foundation staff and authorized contributors only.** | ||||
|  | ||||
|         If you are a community member wanting to contribute, please: | ||||
|         - For bug reports: Use the [bug report form](https://github.com/home-assistant/frontend/issues/new?template=bug_report.yml) | ||||
|         - For feature requests: Submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions) | ||||
|  | ||||
|         --- | ||||
|  | ||||
|         ### For authorized contributors | ||||
|  | ||||
|         Use this form to create tasks for development work, improvements, or other actionable items that need to be tracked. | ||||
|   - type: textarea | ||||
|     id: description | ||||
|     attributes: | ||||
|       label: Description | ||||
|       description: | | ||||
|         Provide a clear and detailed description of the task that needs to be accomplished. | ||||
|  | ||||
|         Be specific about what needs to be done, why it's important, and any constraints or requirements. | ||||
|       placeholder: | | ||||
|         Describe the task, including: | ||||
|         - What needs to be done | ||||
|         - Why this task is needed | ||||
|         - Expected outcome | ||||
|         - Any constraints or requirements | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: additional_context | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|       description: | | ||||
|         Any additional information, links, research, or context that would be helpful. | ||||
|  | ||||
|         Include links to related issues, research, prototypes, roadmap opportunities etc. | ||||
|       placeholder: | | ||||
|         - Roadmap opportunity: [link] | ||||
|         - Epic: [link] | ||||
|         - Feature request: [link] | ||||
|         - Technical design documents: [link] | ||||
|         - Prototype/mockup: [link] | ||||
|         - Dependencies: [links] | ||||
|     validations: | ||||
|       required: false | ||||
							
								
								
									
										596
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,596 +0,0 @@ | ||||
| # GitHub Copilot & Claude Code Instructions | ||||
|  | ||||
| You are an assistant helping with development of the Home Assistant frontend. The frontend is built using Lit-based Web Components and TypeScript, providing a responsive and performant interface for home automation control. | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| - [Quick Reference](#quick-reference) | ||||
| - [Core Architecture](#core-architecture) | ||||
| - [Development Standards](#development-standards) | ||||
| - [Component Library](#component-library) | ||||
| - [Common Patterns](#common-patterns) | ||||
| - [Text and Copy Guidelines](#text-and-copy-guidelines) | ||||
| - [Development Workflow](#development-workflow) | ||||
| - [Review Guidelines](#review-guidelines) | ||||
|  | ||||
| ## Quick Reference | ||||
|  | ||||
| ### Essential Commands | ||||
|  | ||||
| ```bash | ||||
| yarn lint          # ESLint + Prettier + TypeScript + Lit | ||||
| yarn format        # Auto-fix ESLint + Prettier | ||||
| yarn lint:types    # TypeScript compiler | ||||
| yarn test          # Vitest | ||||
| script/develop     # Development server | ||||
| ``` | ||||
|  | ||||
| ### Component Prefixes | ||||
|  | ||||
| - `ha-` - Home Assistant components | ||||
| - `hui-` - Lovelace UI components | ||||
| - `dialog-` - Dialog components | ||||
|  | ||||
| ### Import Patterns | ||||
|  | ||||
| ```typescript | ||||
| import type { HomeAssistant } from "../types"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { showAlertDialog } from "../dialogs/generic/show-alert-dialog"; | ||||
| ``` | ||||
|  | ||||
| ## Core Architecture | ||||
|  | ||||
| The Home Assistant frontend is a modern web application that: | ||||
|  | ||||
| - Uses Web Components (custom elements) built with Lit framework | ||||
| - Is written entirely in TypeScript with strict type checking | ||||
| - Communicates with the backend via WebSocket API | ||||
| - Provides comprehensive theming and internationalization | ||||
|  | ||||
| ## Development Standards | ||||
|  | ||||
| ### Code Quality Requirements | ||||
|  | ||||
| **Linting and Formatting (Enforced by Tools)** | ||||
|  | ||||
| - ESLint config extends Airbnb, TypeScript strict, Lit, Web Components, Accessibility | ||||
| - Prettier with ES5 trailing commas enforced | ||||
| - No console statements (`no-console: "error"`) - use proper logging | ||||
| - Import organization: No unused imports, consistent type imports | ||||
|  | ||||
| **Naming Conventions** | ||||
|  | ||||
| - PascalCase for types and classes | ||||
| - camelCase for variables, methods | ||||
| - Private methods require leading underscore | ||||
| - Public methods forbid leading underscore | ||||
|  | ||||
| ### TypeScript Usage | ||||
|  | ||||
| - **Always use strict TypeScript**: Enable all strict flags, avoid `any` types | ||||
| - **Proper type imports**: Use `import type` for type-only imports | ||||
| - **Define interfaces**: Create proper interfaces for data structures | ||||
| - **Type component properties**: All Lit properties must be properly typed | ||||
| - **No unused variables**: Prefix with `_` if intentionally unused | ||||
| - **Consistent imports**: Use `@typescript-eslint/consistent-type-imports` | ||||
|  | ||||
| ```typescript | ||||
| // Good | ||||
| import type { HomeAssistant } from "../types"; | ||||
|  | ||||
| interface EntityConfig { | ||||
|   entity: string; | ||||
|   name?: string; | ||||
| } | ||||
|  | ||||
| @property({ type: Object }) | ||||
| hass!: HomeAssistant; | ||||
|  | ||||
| // Bad | ||||
| @property() | ||||
| hass: any; | ||||
| ``` | ||||
|  | ||||
| ### Web Components with Lit | ||||
|  | ||||
| - **Use Lit 3.x patterns**: Follow modern Lit practices | ||||
| - **Extend appropriate base classes**: Use `LitElement`, `SubscribeMixin`, or other mixins as needed | ||||
| - **Define custom element names**: Use `ha-` prefix for components | ||||
|  | ||||
| ```typescript | ||||
| @customElement("ha-my-component") | ||||
| export class HaMyComponent extends LitElement { | ||||
|   @property({ attribute: false }) | ||||
|   hass!: HomeAssistant; | ||||
|  | ||||
|   @state() | ||||
|   private _config?: MyComponentConfig; | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: block; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   render() { | ||||
|     return html`<div>Content</div>`; | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Component Guidelines | ||||
|  | ||||
| - **Use composition**: Prefer composition over inheritance | ||||
| - **Lazy load panels**: Heavy panels should be dynamically imported | ||||
| - **Optimize renders**: Use `@state()` for internal state, `@property()` for public API | ||||
| - **Handle loading states**: Always show appropriate loading indicators | ||||
| - **Support themes**: Use CSS custom properties from theme | ||||
|  | ||||
| ### Data Management | ||||
|  | ||||
| - **Use WebSocket API**: All backend communication via home-assistant-js-websocket | ||||
| - **Cache appropriately**: Use collections and caching for frequently accessed data | ||||
| - **Handle errors gracefully**: All API calls should have error handling | ||||
| - **Update real-time**: Subscribe to state changes for live updates | ||||
|  | ||||
| ```typescript | ||||
| // Good | ||||
| try { | ||||
|   const result = await fetchEntityRegistry(this.hass.connection); | ||||
|   this._processResult(result); | ||||
| } catch (err) { | ||||
|   showAlertDialog(this, { | ||||
|     text: `Failed to load: ${err.message}`, | ||||
|   }); | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Styling Guidelines | ||||
|  | ||||
| - **Use CSS custom properties**: Leverage the theme system | ||||
| - **Mobile-first responsive**: Design for mobile, enhance for desktop | ||||
| - **Follow Material Design**: Use Material Web Components where appropriate | ||||
| - **Support RTL**: Ensure all layouts work in RTL languages | ||||
|  | ||||
| ```typescript | ||||
| static get styles() { | ||||
|   return css` | ||||
|     :host { | ||||
|       --spacing: 16px; | ||||
|       padding: var(--spacing); | ||||
|       color: var(--primary-text-color); | ||||
|       background-color: var(--card-background-color); | ||||
|     } | ||||
|  | ||||
|     @media (max-width: 600px) { | ||||
|       :host { | ||||
|         --spacing: 8px; | ||||
|       } | ||||
|     } | ||||
|   `; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Performance Best Practices | ||||
|  | ||||
| - **Code split**: Split code at the panel/dialog level | ||||
| - **Lazy load**: Use dynamic imports for heavy components | ||||
| - **Optimize bundle**: Keep initial bundle size minimal | ||||
| - **Use virtual scrolling**: For long lists, implement virtual scrolling | ||||
| - **Memoize computations**: Cache expensive calculations | ||||
|  | ||||
| ### Testing Requirements | ||||
|  | ||||
| - **Write tests**: Add tests for data processing and utilities | ||||
| - **Test with Vitest**: Use the established test framework | ||||
| - **Mock appropriately**: Mock WebSocket connections and API calls | ||||
| - **Test accessibility**: Ensure components are accessible | ||||
|  | ||||
| ## Component Library | ||||
|  | ||||
| ### Dialog Components | ||||
|  | ||||
| **Available Dialog Types:** | ||||
|  | ||||
| - `ha-md-dialog` - Preferred for new code (Material Design 3) | ||||
| - `ha-dialog` - Legacy component still widely used | ||||
|  | ||||
| **Opening Dialogs (Fire Event Pattern - Recommended):** | ||||
|  | ||||
| ```typescript | ||||
| fireEvent(this, "show-dialog", { | ||||
|   dialogTag: "dialog-example", | ||||
|   dialogImport: () => import("./dialog-example"), | ||||
|   dialogParams: { title: "Example", data: someData }, | ||||
| }); | ||||
| ``` | ||||
|  | ||||
| **Dialog Implementation Requirements:** | ||||
|  | ||||
| - Implement `HassDialog<T>` interface | ||||
| - Use `createCloseHeading()` for standard headers | ||||
| - Import `haStyleDialog` for consistent styling | ||||
| - Return `nothing` when no params (loading state) | ||||
| - Fire `dialog-closed` event when closing | ||||
| - Add `dialogInitialFocus` for accessibility | ||||
|  | ||||
| ```` | ||||
|  | ||||
| ### Form Component (ha-form) | ||||
| - Schema-driven using `HaFormSchema[]` | ||||
| - Supports entity, device, area, target, number, boolean, time, action, text, object, select, icon, media, location selectors | ||||
| - Built-in validation with error display | ||||
| - Use `dialogInitialFocus` in dialogs | ||||
| - Use `computeLabel`, `computeError`, `computeHelper` for translations | ||||
|  | ||||
| ```typescript | ||||
| <ha-form | ||||
|   .hass=${this.hass} | ||||
|   .data=${this._data} | ||||
|   .schema=${this._schema} | ||||
|   .error=${this._errors} | ||||
|   .computeLabel=${(schema) => this.hass.localize(`ui.panel.${schema.name}`)} | ||||
|   @value-changed=${this._valueChanged} | ||||
| ></ha-form> | ||||
| ```` | ||||
|  | ||||
| ### Alert Component (ha-alert) | ||||
|  | ||||
| - Types: `error`, `warning`, `info`, `success` | ||||
| - Properties: `title`, `alert-type`, `dismissable`, `icon`, `action`, `rtl` | ||||
| - Content announced by screen readers when dynamically displayed | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="error">Error message</ha-alert> | ||||
| <ha-alert alert-type="warning" title="Warning">Description</ha-alert> | ||||
| <ha-alert alert-type="success" dismissable>Success message</ha-alert> | ||||
| ``` | ||||
|  | ||||
| ## Common Patterns | ||||
|  | ||||
| ### Creating a Panel | ||||
|  | ||||
| ```typescript | ||||
| @customElement("ha-panel-myfeature") | ||||
| export class HaPanelMyFeature extends SubscribeMixin(LitElement) { | ||||
|   @property({ attribute: false }) | ||||
|   hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) | ||||
|   narrow!: boolean; | ||||
|  | ||||
|   @property() | ||||
|   route!: Route; | ||||
|  | ||||
|   hassSubscribe() { | ||||
|     return [ | ||||
|       subscribeEntityRegistry(this.hass.connection, (entities) => { | ||||
|         this._entities = entities; | ||||
|       }), | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Creating a Dialog | ||||
|  | ||||
| ```typescript | ||||
| @customElement("dialog-my-feature") | ||||
| export class DialogMyFeature | ||||
|   extends LitElement | ||||
|   implements HassDialog<MyDialogParams> | ||||
| { | ||||
|   @property({ attribute: false }) | ||||
|   hass!: HomeAssistant; | ||||
|  | ||||
|   @state() | ||||
|   private _params?: MyDialogParams; | ||||
|  | ||||
|   public async showDialog(params: MyDialogParams): Promise<void> { | ||||
|     this._params = params; | ||||
|   } | ||||
|  | ||||
|   public closeDialog(): void { | ||||
|     this._params = undefined; | ||||
|     fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|     if (!this._params) { | ||||
|       return nothing; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <ha-dialog | ||||
|         open | ||||
|         @closed=${this.closeDialog} | ||||
|         .heading=${createCloseHeading(this.hass, this._params.title)} | ||||
|       > | ||||
|         <!-- Dialog content --> | ||||
|         <ha-button | ||||
|           appearance="plain" | ||||
|           @click=${this.closeDialog} | ||||
|           slot="secondaryAction" | ||||
|         > | ||||
|           ${this.hass.localize("ui.common.cancel")} | ||||
|         </ha-button> | ||||
|         <ha-button @click=${this._submit} slot="primaryAction"> | ||||
|           ${this.hass.localize("ui.common.save")} | ||||
|         </ha-button> | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static styles = [haStyleDialog, css``]; | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ### Dialog Design Guidelines | ||||
|  | ||||
| - Max width: 560px (Alert/confirmation: 320px fixed width) | ||||
| - Close X-icon on top left (all screen sizes) | ||||
| - Submit button grouped with cancel at bottom right | ||||
| - Keep button labels short: "Save", "Delete", "Enable" | ||||
| - Destructive actions use red warning button | ||||
| - Always use a title (best practice) | ||||
| - Strive for minimalism | ||||
|  | ||||
| #### Creating a Lovelace Card | ||||
|  | ||||
| **Purpose**: Cards allow users to tell different stories about their house (based on gallery) | ||||
|  | ||||
| ```typescript | ||||
| @customElement("hui-my-card") | ||||
| export class HuiMyCard extends LitElement implements LovelaceCard { | ||||
|   @property({ attribute: false }) | ||||
|   hass!: HomeAssistant; | ||||
|  | ||||
|   @state() | ||||
|   private _config?: MyCardConfig; | ||||
|  | ||||
|   public setConfig(config: MyCardConfig): void { | ||||
|     if (!config.entity) { | ||||
|       throw new Error("Entity required"); | ||||
|     } | ||||
|     this._config = config; | ||||
|   } | ||||
|  | ||||
|   public getCardSize(): number { | ||||
|     return 3; // Height in grid units | ||||
|   } | ||||
|  | ||||
|   // Optional: Editor for card configuration | ||||
|   public static getConfigElement(): LovelaceCardEditor { | ||||
|     return document.createElement("hui-my-card-editor"); | ||||
|   } | ||||
|  | ||||
|   // Optional: Stub config for card picker | ||||
|   public static getStubConfig(): object { | ||||
|     return { entity: "" }; | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| **Card Guidelines:** | ||||
|  | ||||
| - Cards are highly customizable for different households | ||||
| - Implement `LovelaceCard` interface with `setConfig()` and `getCardSize()` | ||||
| - Use proper error handling in `setConfig()` | ||||
| - Consider all possible states (loading, error, unavailable) | ||||
| - Support different entity types and states | ||||
| - Follow responsive design principles | ||||
| - Add configuration editor when needed | ||||
|  | ||||
| ### Internationalization | ||||
|  | ||||
| - **Use localize**: Always use the localization system | ||||
| - **Add translation keys**: Add keys to src/translations/en.json | ||||
| - **Support placeholders**: Use proper placeholder syntax | ||||
|  | ||||
| ```typescript | ||||
| this.hass.localize("ui.panel.config.updates.update_available", { | ||||
|   count: 5, | ||||
| }); | ||||
| ``` | ||||
|  | ||||
| ### Accessibility | ||||
|  | ||||
| - **ARIA labels**: Add appropriate ARIA labels | ||||
| - **Keyboard navigation**: Ensure all interactions work with keyboard | ||||
| - **Screen reader support**: Test with screen readers | ||||
| - **Color contrast**: Meet WCAG AA standards | ||||
|  | ||||
| ## Development Workflow | ||||
|  | ||||
| ### Setup and Commands | ||||
|  | ||||
| 1. **Setup**: `script/setup` - Install dependencies | ||||
| 2. **Develop**: `script/develop` - Development server | ||||
| 3. **Lint**: `yarn lint` - Run all linting before committing | ||||
| 4. **Test**: `yarn test` - Add and run tests | ||||
| 5. **Build**: `script/build_frontend` - Test production build | ||||
|  | ||||
| ### Common Pitfalls to Avoid | ||||
|  | ||||
| - Don't use `querySelector` - Use refs or component properties | ||||
| - Don't manipulate DOM directly - Let Lit handle rendering | ||||
| - Don't use global styles - Scope styles to components | ||||
| - Don't block the main thread - Use web workers for heavy computation | ||||
| - Don't ignore TypeScript errors - Fix all type issues | ||||
|  | ||||
| ### Security Best Practices | ||||
|  | ||||
| - Sanitize HTML - Never use `unsafeHTML` with user content | ||||
| - Validate inputs - Always validate user inputs | ||||
| - Use HTTPS - All external resources must use HTTPS | ||||
| - CSP compliance - Ensure code works with Content Security Policy | ||||
|  | ||||
| ### Text and Copy Guidelines | ||||
|  | ||||
| #### Terminology Standards | ||||
|  | ||||
| **Delete vs Remove** (Based on gallery/src/pages/Text/remove-delete-add-create.markdown) | ||||
|  | ||||
| - **Use "Remove"** for actions that can be restored or reapplied: | ||||
|   - Removing a user's permission | ||||
|   - Removing a user from a group | ||||
|   - Removing links between items | ||||
|   - Removing a widget from dashboard | ||||
|   - Removing an item from a cart | ||||
| - **Use "Delete"** for permanent, non-recoverable actions: | ||||
|   - Deleting a field | ||||
|   - Deleting a value in a field | ||||
|   - Deleting a task | ||||
|   - Deleting a group | ||||
|   - Deleting a permission | ||||
|   - Deleting a calendar event | ||||
|  | ||||
| **Create vs Add** (Create pairs with Delete, Add pairs with Remove) | ||||
|  | ||||
| - **Use "Add"** for already-existing items: | ||||
|   - Adding a permission to a user | ||||
|   - Adding a user to a group | ||||
|   - Adding links between items | ||||
|   - Adding a widget to dashboard | ||||
|   - Adding an item to a cart | ||||
| - **Use "Create"** for something made from scratch: | ||||
|   - Creating a new field | ||||
|   - Creating a new task | ||||
|   - Creating a new group | ||||
|   - Creating a new permission | ||||
|   - Creating a new calendar event | ||||
|  | ||||
| #### Writing Style (Consistent with Home Assistant Documentation) | ||||
|  | ||||
| - **Use American English**: Standard spelling and terminology | ||||
| - **Friendly, informational tone**: Be inspiring, personal, comforting, engaging | ||||
| - **Address users directly**: Use "you" and "your" | ||||
| - **Be inclusive**: Objective, non-discriminatory language | ||||
| - **Be concise**: Use clear, direct language | ||||
| - **Be consistent**: Follow established terminology patterns | ||||
| - **Use active voice**: "Delete the automation" not "The automation should be deleted" | ||||
| - **Avoid jargon**: Use terms familiar to home automation users | ||||
|  | ||||
| #### Language Standards | ||||
|  | ||||
| - **Always use "Home Assistant"** in full, never "HA" or "HASS" | ||||
| - **Avoid abbreviations**: Spell out terms when possible | ||||
| - **Use sentence case everywhere**: Titles, headings, buttons, labels, UI elements | ||||
|   - ✅ "Create new automation" | ||||
|   - ❌ "Create New Automation" | ||||
|   - ✅ "Device settings" | ||||
|   - ❌ "Device Settings" | ||||
| - **Oxford comma**: Use in lists (item 1, item 2, and item 3) | ||||
| - **Replace Latin terms**: Use "like" instead of "e.g.", "for example" instead of "i.e." | ||||
| - **Avoid CAPS for emphasis**: Use bold or italics instead | ||||
| - **Write for all skill levels**: Both technical and non-technical users | ||||
|  | ||||
| #### Key Terminology | ||||
|  | ||||
| - **"add-on"** (hyphenated, not "addon") | ||||
| - **"integration"** (preferred over "component") | ||||
| - **Technical terms**: Use lowercase (automation, entity, device, service) | ||||
|  | ||||
| #### Translation Considerations | ||||
|  | ||||
| - **Add translation keys**: All user-facing text must be translatable | ||||
| - **Use placeholders**: Support dynamic content in translations | ||||
| - **Keep context**: Provide enough context for translators | ||||
|  | ||||
| ```typescript | ||||
| // Good | ||||
| this.hass.localize("ui.panel.config.automation.delete_confirm", { | ||||
|   name: automation.alias, | ||||
| }); | ||||
|  | ||||
| // Bad - hardcoded text | ||||
| ("Are you sure you want to delete this automation?"); | ||||
| ``` | ||||
|  | ||||
| ### Common Review Issues (From PR Analysis) | ||||
|  | ||||
| #### User Experience and Accessibility | ||||
|  | ||||
| - **Form validation**: Always provide proper field labels and validation feedback | ||||
| - **Form accessibility**: Prevent password managers from incorrectly identifying fields | ||||
| - **Loading states**: Show clear progress indicators during async operations | ||||
| - **Error handling**: Display meaningful error messages when operations fail | ||||
| - **Mobile responsiveness**: Ensure components work well on small screens | ||||
| - **Hit targets**: Make clickable areas large enough for touch interaction | ||||
| - **Visual feedback**: Provide clear indication of interactive states | ||||
|  | ||||
| #### Dialog and Modal Patterns | ||||
|  | ||||
| - **Dialog width constraints**: Respect minimum and maximum width requirements | ||||
| - **Interview progress**: Show clear progress for multi-step operations | ||||
| - **State persistence**: Handle dialog state properly during background operations | ||||
| - **Cancel behavior**: Ensure cancel/close buttons work consistently | ||||
| - **Form prefilling**: Use smart defaults but allow user override | ||||
|  | ||||
| #### Component Design Patterns | ||||
|  | ||||
| - **Terminology consistency**: Use "Join"/"Apply" instead of "Group" when appropriate | ||||
| - **Visual hierarchy**: Ensure proper font sizes and spacing ratios | ||||
| - **Grid alignment**: Components should align to the design grid system | ||||
| - **Badge placement**: Position badges and indicators consistently | ||||
| - **Color theming**: Respect theme variables and design system colors | ||||
|  | ||||
| #### Code Quality Issues | ||||
|  | ||||
| - **Null checking**: Always check if entities exist before accessing properties | ||||
| - **TypeScript safety**: Handle potentially undefined array/object access | ||||
| - **Import organization**: Remove unused imports and use proper type imports | ||||
| - **Event handling**: Properly subscribe and unsubscribe from events | ||||
| - **Memory leaks**: Clean up subscriptions and event listeners | ||||
|  | ||||
| #### Configuration and Props | ||||
|  | ||||
| - **Optional parameters**: Make configuration fields optional when sensible | ||||
| - **Smart defaults**: Provide reasonable default values | ||||
| - **Future extensibility**: Design APIs that can be extended later | ||||
| - **Validation**: Validate configuration before applying changes | ||||
|  | ||||
| ## Review Guidelines | ||||
|  | ||||
| ### Core Requirements Checklist | ||||
|  | ||||
| - [ ] TypeScript strict mode passes (`yarn lint:types`) | ||||
| - [ ] No ESLint errors or warnings (`yarn lint:eslint`) | ||||
| - [ ] Prettier formatting applied (`yarn lint:prettier`) | ||||
| - [ ] Lit analyzer passes (`yarn lint:lit`) | ||||
| - [ ] Component follows Lit best practices | ||||
| - [ ] Proper error handling implemented | ||||
| - [ ] Loading states handled | ||||
| - [ ] Mobile responsive | ||||
| - [ ] Theme variables used | ||||
| - [ ] Translations added | ||||
| - [ ] Accessible to screen readers | ||||
| - [ ] Tests added (where applicable) | ||||
| - [ ] No console statements (use proper logging) | ||||
| - [ ] Unused imports removed | ||||
| - [ ] Proper naming conventions | ||||
|  | ||||
| ### Text and Copy Checklist | ||||
|  | ||||
| - [ ] Follows terminology guidelines (Delete vs Remove, Create vs Add) | ||||
| - [ ] Localization keys added for all user-facing text | ||||
| - [ ] Uses "Home Assistant" (never "HA" or "HASS") | ||||
| - [ ] Sentence case for ALL text (titles, headings, buttons, labels) | ||||
| - [ ] American English spelling | ||||
| - [ ] Friendly, informational tone | ||||
| - [ ] Avoids abbreviations and jargon | ||||
| - [ ] Correct terminology (add-on not addon, integration not component) | ||||
|  | ||||
| ### Component-Specific Checks | ||||
|  | ||||
| - [ ] Dialogs implement HassDialog interface | ||||
| - [ ] Dialog styling uses haStyleDialog | ||||
| - [ ] Dialog accessibility includes dialogInitialFocus | ||||
| - [ ] ha-alert used correctly for messages | ||||
| - [ ] ha-form uses proper schema structure | ||||
| - [ ] Components handle all states (loading, error, unavailable) | ||||
| - [ ] Entity existence checked before property access | ||||
| - [ ] Event subscriptions properly cleaned up | ||||
							
								
								
									
										12
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -21,12 +21,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -42,7 +42,7 @@ jobs: | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --alias dev | ||||
|           npx -y netlify-cli deploy --dir=cast/dist --alias dev | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
| @@ -56,12 +56,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -77,7 +77,7 @@ jobs: | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --prod | ||||
|           npx -y netlify-cli deploy --dir=cast/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										22
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -24,9 +24,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -37,7 +37,7 @@ jobs: | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages | ||||
|       - name: Setup lint cache | ||||
|         uses: actions/cache@0400d5f644dc74513175e3cd8d07132dd4860809 # v4.2.4 | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         with: | ||||
|           path: | | ||||
|             node_modules/.cache/prettier | ||||
| @@ -58,9 +58,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -76,9 +76,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -89,7 +89,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: frontend-bundle-stats | ||||
|           path: build/stats/*.json | ||||
| @@ -100,9 +100,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -113,7 +113,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: supervisor-bundle-stats | ||||
|           path: build/stats/*.json | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -23,7 +23,7 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           # We must fetch at least the immediate parents so that if this is | ||||
|           # a pull request then we can checkout the head. | ||||
| @@ -36,14 +36,14 @@ jobs: | ||||
|  | ||||
|       # Initializes the CodeQL tools for scanning. | ||||
|       - name: Initialize CodeQL | ||||
|         uses: github/codeql-action/init@192325c86100d080feab897ff886c34abd4c83a3 # v3.30.3 | ||||
|         uses: github/codeql-action/init@v3 | ||||
|         with: | ||||
|           languages: ${{ matrix.language }} | ||||
|  | ||||
|       # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). | ||||
|       # If this step fails, then you should remove it and run the build manually (see below) | ||||
|       - name: Autobuild | ||||
|         uses: github/codeql-action/autobuild@192325c86100d080feab897ff886c34abd4c83a3 # v3.30.3 | ||||
|         uses: github/codeql-action/autobuild@v3 | ||||
|  | ||||
|       # ℹ️ Command-line programs to run using the OS shell. | ||||
|       # 📚 https://git.io/JvXDl | ||||
| @@ -57,4 +57,4 @@ jobs: | ||||
|       #   make release | ||||
|  | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@192325c86100d080feab897ff886c34abd4c83a3 # v3.30.3 | ||||
|         uses: github/codeql-action/analyze@v3 | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -22,12 +22,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -43,7 +43,7 @@ jobs: | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||
|           npx -y netlify-cli deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} | ||||
| @@ -57,12 +57,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -78,7 +78,7 @@ jobs: | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||
|           npx -y netlify-cli deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -16,10 +16,10 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -35,7 +35,7 @@ jobs: | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --prod | ||||
|           npx -y netlify-cli deploy --dir=gallery/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -21,10 +21,10 @@ jobs: | ||||
|     if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview') | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -40,7 +40,7 @@ jobs: | ||||
|       - name: Deploy preview to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \ | ||||
|           npx -y netlify-cli deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \ | ||||
|             --json > deploy_output.json | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -10,6 +10,6 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Apply labels | ||||
|         uses: actions/labeler@634933edcd8ababfe52f92936142cc22ac488b1b # v6.0.1 | ||||
|         uses: actions/labeler@v5.0.0 | ||||
|         with: | ||||
|           sync-labels: true | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -9,7 +9,7 @@ jobs: | ||||
|   lock: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1 | ||||
|       - uses: dessant/lock-threads@v5.0.1 | ||||
|         with: | ||||
|           github-token: ${{ github.token }} | ||||
|           process-only: "issues, prs" | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -20,15 +20,15 @@ jobs: | ||||
|       contents: write | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -57,14 +57,14 @@ jobs: | ||||
|         run: tar -czvf translations.tar.gz translations | ||||
|  | ||||
|       - name: Upload build artifacts | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: wheels | ||||
|           path: dist/home_assistant_frontend*.whl | ||||
|           if-no-files-found: error | ||||
|  | ||||
|       - name: Upload translations | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: translations | ||||
|           path: translations.tar.gz | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -17,7 +17,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Send bundle stats and build information to RelativeCI | ||||
|         uses: relative-ci/agent-action@1707825cbfcc7452b2913d273414705415ae64d4 # v3.0.1 | ||||
|         uses: relative-ci/agent-action@v2.2.0 | ||||
|         with: | ||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} | ||||
|           token: ${{ github.token }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -18,6 +18,6 @@ jobs: | ||||
|       pull-requests: read | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: release-drafter/release-drafter@b1476f6e6eb133afa41ed8589daba6dc69b4d3f5 # v6.1.0 | ||||
|       - uses: release-drafter/release-drafter@v6.1.0 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										23
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -23,10 +23,10 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
| @@ -34,7 +34,7 @@ jobs: | ||||
|         uses: home-assistant/actions/helpers/verify-version@master | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -55,7 +55,7 @@ jobs: | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@6cbd405e2c4e67a21c47fa9e383d020e4e28b836 # v2.3.3 | ||||
|         uses: softprops/action-gh-release@v2.2.2 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
| @@ -73,9 +73,8 @@ jobs: | ||||
|           version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' ) | ||||
|           echo "home-assistant-frontend==$version" > ./requirements.txt | ||||
|  | ||||
|       # home-assistant/wheels doesn't support SHA pinning | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2025.07.0 | ||||
|         uses: home-assistant/wheels@2025.03.0 | ||||
|         with: | ||||
|           abi: cp313 | ||||
|           tag: musllinux_1_2 | ||||
| @@ -91,9 +90,9 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -108,7 +107,7 @@ jobs: | ||||
|       - name: Tar folder | ||||
|         run: tar -czf landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz -C landing-page/dist . | ||||
|       - name: Upload release asset | ||||
|         uses: softprops/action-gh-release@6cbd405e2c4e67a21c47fa9e383d020e4e28b836 # v2.3.3 | ||||
|         uses: softprops/action-gh-release@v2.2.2 | ||||
|         with: | ||||
|           files: landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz | ||||
|  | ||||
| @@ -120,9 +119,9 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||
|         uses: actions/setup-node@v4.4.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -137,6 +136,6 @@ jobs: | ||||
|       - name: Tar folder | ||||
|         run: tar -czf hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz -C hassio/build . | ||||
|       - name: Upload release asset | ||||
|         uses: softprops/action-gh-release@6cbd405e2c4e67a21c47fa9e383d020e4e28b836 # v2.3.3 | ||||
|         uses: softprops/action-gh-release@v2.2.2 | ||||
|         with: | ||||
|           files: hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz | ||||
|   | ||||
							
								
								
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,58 +0,0 @@ | ||||
| name: Restrict task creation | ||||
|  | ||||
| # yamllint disable-line rule:truthy | ||||
| on: | ||||
|   issues: | ||||
|     types: [opened] | ||||
|  | ||||
| jobs: | ||||
|   check-authorization: | ||||
|     runs-on: ubuntu-latest | ||||
|     # Only run if this is a Task issue type (from the issue form) | ||||
|     if: github.event.issue.type.name == 'Task' | ||||
|     steps: | ||||
|       - name: Check if user is authorized | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8 | ||||
|         with: | ||||
|           script: | | ||||
|             const issueAuthor = context.payload.issue.user.login; | ||||
|  | ||||
|             // Check if user is an organization member | ||||
|             try { | ||||
|               await github.rest.orgs.checkMembershipForUser({ | ||||
|                 org: 'home-assistant', | ||||
|                 username: issueAuthor | ||||
|               }); | ||||
|               console.log(`✅ ${issueAuthor} is an organization member`); | ||||
|               return; // Authorized | ||||
|             } catch (error) { | ||||
|               console.log(`❌ ${issueAuthor} is not authorized to create Task issues`); | ||||
|             } | ||||
|  | ||||
|             // Close the issue with a comment | ||||
|             await github.rest.issues.createComment({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               body: `Hi @${issueAuthor}, thank you for your contribution!\n\n` + | ||||
|                     `Task issues are restricted to Open Home Foundation staff and authorized contributors.\n\n` + | ||||
|                     `If you would like to:\n` + | ||||
|                     `- Report a bug: Please use the [bug report form](https://github.com/home-assistant/frontend/issues/new?template=bug_report.yml)\n` + | ||||
|                     `- Request a feature: Please submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions)\n\n` + | ||||
|                     `If you believe you should have access to create Task issues, please contact the maintainers.` | ||||
|             }); | ||||
|  | ||||
|             await github.rest.issues.update({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               state: 'closed' | ||||
|             }); | ||||
|  | ||||
|             // Add a label to indicate this was auto-closed | ||||
|             await github.rest.issues.addLabels({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               labels: ['auto-closed'] | ||||
|             }); | ||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -10,7 +10,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: 90 days stale policy | ||||
|         uses: actions/stale@3a9db7e6a41a89f618792c92c0e97cc736e1b13f # v10.0.0 | ||||
|         uses: actions/stale@v9.1.0 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 90 | ||||
|   | ||||
							
								
								
									
										3
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,7 +1,6 @@ | ||||
| name: Translations | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
| @@ -14,7 +13,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Upload Translations | ||||
|         run: | | ||||
|   | ||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -53,7 +53,3 @@ src/cast/dev_const.ts | ||||
|  | ||||
| # test coverage | ||||
| test/coverage/ | ||||
|  | ||||
| # AI tooling | ||||
| .claude | ||||
|  | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| yarn run lint-staged --relative | ||||
| yarn run lint-staged --relative --shell "/bin/bash" | ||||
|   | ||||
							
								
								
									
										942
									
								
								.yarn/releases/yarn-4.10.2.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
							
								
								
									
										948
									
								
								.yarn/releases/yarn-4.9.1.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						| @@ -6,4 +6,4 @@ enableGlobalCache: false | ||||
|  | ||||
| nodeLinker: node-modules | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-4.10.2.cjs | ||||
| yarnPath: .yarn/releases/yarn-4.9.1.cjs | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| # People marked here will be automatically requested for a review | ||||
| # when the code that they own is touched. | ||||
| # https://github.com/blog/2392-introducing-code-owners | ||||
| # https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners | ||||
|  | ||||
| # Part of the frontend that mobile developper should review | ||||
| src/external_app/ @bgoncal @TimoPtr | ||||
| test/external_app/ @bgoncal @TimoPtr | ||||
| @@ -3,6 +3,9 @@ import { glob } from "glob"; | ||||
| import gulp from "gulp"; | ||||
| import yaml from "js-yaml"; | ||||
| import { marked } from "marked"; | ||||
| import ts from "typescript"; | ||||
| import { create } from "@custom-elements-manifest/analyzer"; | ||||
| import { litPlugin } from "@custom-elements-manifest/analyzer/src/features/framework-plugins/lit/lit.js"; | ||||
| import path from "path"; | ||||
| import paths from "../paths.cjs"; | ||||
| import "./clean.js"; | ||||
| @@ -13,6 +16,28 @@ import "./service-worker.js"; | ||||
| import "./translations.js"; | ||||
| import "./rspack.js"; | ||||
|  | ||||
| gulp.task("generate-component-docs", async function generateComponentDocs() { | ||||
|   const filePaths = ["src/components/ha-alert.ts"]; | ||||
|  | ||||
|   const modules = await Promise.all( | ||||
|     filePaths.map(async (file) => { | ||||
|       const filePath = path.resolve(file); | ||||
|       console.log(`Reading ${file} -> ${filePath}`); | ||||
|       const source = fs.readFileSync(filePath).toString(); | ||||
|  | ||||
|       return ts.createSourceFile(file, source, ts.ScriptTarget.ES2015, true); | ||||
|     }) | ||||
|   ); | ||||
|  | ||||
|   const manifest = create({ | ||||
|     modules, | ||||
|     plugins: litPlugin(), | ||||
|     context: { dev: true }, | ||||
|   }); | ||||
|  | ||||
|   console.log(manifest); | ||||
| }); | ||||
|  | ||||
| gulp.task("gather-gallery-pages", async function gatherPages() { | ||||
|   const pageDir = path.resolve(paths.gallery_dir, "src/pages"); | ||||
|   const files = await glob(path.resolve(pageDir, "**/*")); | ||||
|   | ||||
| @@ -14,5 +14,5 @@ | ||||
|   "name": "Home Assistant Cast", | ||||
|   "short_name": "HA Cast", | ||||
|   "start_url": "/?homescreen=1", | ||||
|   "theme_color": "#009ac7" | ||||
|   "theme_color": "#03A9F4" | ||||
| } | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
|  | ||||
| import type { ActionDetail } from "@material/mwc-list/mwc-list"; | ||||
| import { mdiCast, mdiCastConnected, mdiViewDashboard } from "@mdi/js"; | ||||
| import type { Auth, Connection } from "home-assistant-js-websocket"; | ||||
| @@ -18,7 +20,6 @@ import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | ||||
| import "../../../../src/components/ha-icon"; | ||||
| import "../../../../src/components/ha-list"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { | ||||
| @@ -62,20 +63,12 @@ class HcCast extends LitElement { | ||||
|               <p class="question action-item"> | ||||
|                 Stay logged in? | ||||
|                 <span> | ||||
|                   <ha-button | ||||
|                     appearance="plain" | ||||
|                     size="small" | ||||
|                     @click=${this._handleSaveTokens} | ||||
|                   > | ||||
|                   <mwc-button @click=${this._handleSaveTokens}> | ||||
|                     YES | ||||
|                   </ha-button> | ||||
|                   <ha-button | ||||
|                     appearance="plain" | ||||
|                     size="small" | ||||
|                     @click=${this._handleSkipSaveTokens} | ||||
|                   > | ||||
|                   </mwc-button> | ||||
|                   <mwc-button @click=${this._handleSkipSaveTokens}> | ||||
|                     NO | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </span> | ||||
|               </p> | ||||
|             ` | ||||
| @@ -85,10 +78,10 @@ class HcCast extends LitElement { | ||||
|           : !this.castManager.status | ||||
|             ? html` | ||||
|                 <p class="center-item"> | ||||
|                   <ha-button @click=${this._handleLaunch}> | ||||
|                     <ha-svg-icon slot="start" .path=${mdiCast}></ha-svg-icon> | ||||
|                   <mwc-button raised @click=${this._handleLaunch}> | ||||
|                     <ha-svg-icon .path=${mdiCast}></ha-svg-icon> | ||||
|                     Start Casting | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </p> | ||||
|               ` | ||||
|             : html` | ||||
| @@ -128,22 +121,14 @@ class HcCast extends LitElement { | ||||
|         <div class="card-actions"> | ||||
|           ${this.castManager.status | ||||
|             ? html` | ||||
|                 <ha-button appearance="plain" @click=${this._handleLaunch}> | ||||
|                   <ha-svg-icon | ||||
|                     slot="start" | ||||
|                     .path=${mdiCastConnected} | ||||
|                   ></ha-svg-icon> | ||||
|                 <mwc-button @click=${this._handleLaunch}> | ||||
|                   <ha-svg-icon .path=${mdiCastConnected}></ha-svg-icon> | ||||
|                   Manage | ||||
|                 </ha-button> | ||||
|                 </mwc-button> | ||||
|               ` | ||||
|             : ""} | ||||
|           <div class="spacer"></div> | ||||
|           <ha-button | ||||
|             variant="danger" | ||||
|             appearance="plain" | ||||
|             @click=${this._handleLogout} | ||||
|             >Log out</ha-button | ||||
|           > | ||||
|           <mwc-button @click=${this._handleLogout}>Log out</mwc-button> | ||||
|         </div> | ||||
|       </hc-layout> | ||||
|     `; | ||||
| @@ -260,6 +245,13 @@ class HcCast extends LitElement { | ||||
|       color: var(--secondary-text-color); | ||||
|     } | ||||
|  | ||||
|     mwc-button ha-svg-icon { | ||||
|       margin-right: 8px; | ||||
|       margin-inline-end: 8px; | ||||
|       margin-inline-start: initial; | ||||
|       height: 18px; | ||||
|     } | ||||
|  | ||||
|     ha-list-item ha-icon, | ||||
|     ha-list-item ha-svg-icon { | ||||
|       padding: 12px; | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { mdiCastConnected, mdiCast } from "@mdi/js"; | ||||
| import type { | ||||
|   Auth, | ||||
| @@ -27,7 +28,6 @@ import "../../../../src/layouts/hass-loading-screen"; | ||||
| import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | ||||
| import "./hc-layout"; | ||||
| import "../../../../src/components/ha-textfield"; | ||||
| import "../../../../src/components/ha-button"; | ||||
|  | ||||
| const seeFAQ = (qid) => html` | ||||
|   See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more | ||||
| @@ -83,14 +83,11 @@ export class HcConnect extends LitElement { | ||||
|             Unable to connect to ${tokens!.hassUrl}. | ||||
|           </div> | ||||
|           <div class="card-actions"> | ||||
|             <ha-button appearance="plain" href="/">Retry</ha-button> | ||||
|             <a href="/"> | ||||
|               <mwc-button> Retry </mwc-button> | ||||
|             </a> | ||||
|             <div class="spacer"></div> | ||||
|             <ha-button | ||||
|               appearance="plain" | ||||
|               variant="danger" | ||||
|               @click=${this._handleLogout} | ||||
|               >Log out</ha-button | ||||
|             > | ||||
|             <mwc-button @click=${this._handleLogout}>Log out</mwc-button> | ||||
|           </div> | ||||
|         </hc-layout> | ||||
|       `; | ||||
| @@ -131,19 +128,16 @@ export class HcConnect extends LitElement { | ||||
|             ${this.error ? html` <p class="error">${this.error}</p> ` : ""} | ||||
|           </div> | ||||
|           <div class="card-actions"> | ||||
|             <ha-button appearance="plain" @click=${this._handleDemo}> | ||||
|             <mwc-button @click=${this._handleDemo}> | ||||
|               Show Demo | ||||
|               <ha-svg-icon | ||||
|                 slot="end" | ||||
|                 .path=${this.castManager.castState === "CONNECTED" | ||||
|                   ? mdiCastConnected | ||||
|                   : mdiCast} | ||||
|               ></ha-svg-icon> | ||||
|             </ha-button> | ||||
|             </mwc-button> | ||||
|             <div class="spacer"></div> | ||||
|             <ha-button appearance="plain" @click=${this._handleConnect} | ||||
|               >Authorize</ha-button | ||||
|             > | ||||
|             <mwc-button @click=${this._handleConnect}>Authorize</mwc-button> | ||||
|           </div> | ||||
|         </hc-layout> | ||||
|       `; | ||||
| @@ -315,6 +309,10 @@ export class HcConnect extends LitElement { | ||||
|       color: darkred; | ||||
|     } | ||||
|  | ||||
|     mwc-button ha-svg-icon { | ||||
|       margin-left: 8px; | ||||
|     } | ||||
|  | ||||
|     .spacer { | ||||
|       flex: 1; | ||||
|     } | ||||
|   | ||||
| @@ -88,7 +88,7 @@ class HcLayout extends LitElement { | ||||
|       font-family: var(--ha-card-header-font-family, inherit); | ||||
|       font-size: var(--ha-card-header-font-size, var(--ha-font-size-2xl)); | ||||
|       letter-spacing: -0.012em; | ||||
|       line-height: var(--ha-line-height-condensed); | ||||
|       line-height: 32px; | ||||
|       padding: 24px 16px 16px; | ||||
|       display: block; | ||||
|       margin: 0; | ||||
|   | ||||
							
								
								
									
										186
									
								
								custom-elements.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,186 @@ | ||||
| { | ||||
|   "schemaVersion": "1.0.0", | ||||
|   "readme": "", | ||||
|   "modules": [ | ||||
|     { | ||||
|       "kind": "javascript-module", | ||||
|       "path": "src/components/ha-alert.ts", | ||||
|       "declarations": [ | ||||
|         { | ||||
|           "kind": "class", | ||||
|           "description": "A custom alert component for displaying messages with various alert types.", | ||||
|           "name": "HaAlert", | ||||
|           "cssProperties": [ | ||||
|             { | ||||
|               "description": "The color used for \"info\" alerts.", | ||||
|               "name": "--info-color" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The color used for \"warning\" alerts.", | ||||
|               "name": "--warning-color" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The color used for \"error\" alerts.", | ||||
|               "name": "--error-color" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The color used for \"success\" alerts.", | ||||
|               "name": "--success-color" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The primary text color used in the alert.", | ||||
|               "name": "--primary-text-color" | ||||
|             } | ||||
|           ], | ||||
|           "cssParts": [ | ||||
|             { | ||||
|               "description": "The container for the alert.", | ||||
|               "name": "issue-type" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The container for the alert icon.", | ||||
|               "name": "icon" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The container for the alert content.", | ||||
|               "name": "content" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The container for the alert actions.", | ||||
|               "name": "action" | ||||
|             }, | ||||
|             { | ||||
|               "description": "The container for the alert title.", | ||||
|               "name": "title" | ||||
|             } | ||||
|           ], | ||||
|           "slots": [ | ||||
|             { | ||||
|               "description": "The main content of the alert.", | ||||
|               "name": "" | ||||
|             }, | ||||
|             { | ||||
|               "description": "Slot for providing a custom icon for the alert.", | ||||
|               "name": "icon" | ||||
|             }, | ||||
|             { | ||||
|               "description": "Slot for providing custom actions or buttons for the alert.", | ||||
|               "name": "action" | ||||
|             } | ||||
|           ], | ||||
|           "members": [ | ||||
|             { | ||||
|               "kind": "field", | ||||
|               "name": "title", | ||||
|               "type": { | ||||
|                 "text": "string" | ||||
|               }, | ||||
|               "privacy": "public", | ||||
|               "default": "\"\"", | ||||
|               "description": "The title of the alert. Defaults to an empty string.", | ||||
|               "attribute": "title" | ||||
|             }, | ||||
|             { | ||||
|               "kind": "field", | ||||
|               "name": "alertType", | ||||
|               "type": { | ||||
|                 "text": "\"info\" | \"warning\" | \"error\" | \"success\"" | ||||
|               }, | ||||
|               "privacy": "public", | ||||
|               "default": "\"info\"", | ||||
|               "description": "The type of alert to display. Defaults to \"info\". Determines the styling and icon used.", | ||||
|               "attribute": "alert-type" | ||||
|             }, | ||||
|             { | ||||
|               "kind": "field", | ||||
|               "name": "dismissable", | ||||
|               "type": { | ||||
|                 "text": "boolean" | ||||
|               }, | ||||
|               "privacy": "public", | ||||
|               "default": "false", | ||||
|               "description": "Whether the alert can be dismissed. Defaults to `false`. If `true`, a dismiss button is displayed.", | ||||
|               "attribute": "dismissable" | ||||
|             }, | ||||
|             { | ||||
|               "kind": "field", | ||||
|               "name": "narrow", | ||||
|               "type": { | ||||
|                 "text": "boolean" | ||||
|               }, | ||||
|               "privacy": "public", | ||||
|               "default": "false", | ||||
|               "description": "Whether the alert should use a narrow layout. Defaults to `false`.", | ||||
|               "attribute": "narrow" | ||||
|             }, | ||||
|             { | ||||
|               "kind": "method", | ||||
|               "name": "_dismissClicked", | ||||
|               "privacy": "private" | ||||
|             } | ||||
|           ], | ||||
|           "events": [ | ||||
|             { | ||||
|               "description": "Fired when the dismiss button is clicked.", | ||||
|               "name": "alert-dismissed-clicked" | ||||
|             } | ||||
|           ], | ||||
|           "attributes": [ | ||||
|             { | ||||
|               "name": "title", | ||||
|               "type": { | ||||
|                 "text": "string" | ||||
|               }, | ||||
|               "default": "\"\"", | ||||
|               "description": "The title of the alert. Defaults to an empty string.", | ||||
|               "fieldName": "title" | ||||
|             }, | ||||
|             { | ||||
|               "name": "alert-type", | ||||
|               "type": { | ||||
|                 "text": "\"info\" | \"warning\" | \"error\" | \"success\"" | ||||
|               }, | ||||
|               "default": "\"info\"", | ||||
|               "description": "The type of alert to display. Defaults to \"info\". Determines the styling and icon used.", | ||||
|               "fieldName": "alertType" | ||||
|             }, | ||||
|             { | ||||
|               "name": "dismissable", | ||||
|               "type": { | ||||
|                 "text": "boolean" | ||||
|               }, | ||||
|               "default": "false", | ||||
|               "description": "Whether the alert can be dismissed. Defaults to `false`. If `true`, a dismiss button is displayed.", | ||||
|               "fieldName": "dismissable" | ||||
|             }, | ||||
|             { | ||||
|               "name": "narrow", | ||||
|               "type": { | ||||
|                 "text": "boolean" | ||||
|               }, | ||||
|               "default": "false", | ||||
|               "description": "Whether the alert should use a narrow layout. Defaults to `false`.", | ||||
|               "fieldName": "narrow" | ||||
|             } | ||||
|           ], | ||||
|           "superclass": { | ||||
|             "name": "LitElement", | ||||
|             "package": "lit" | ||||
|           }, | ||||
|           "tagName": "ha-alert", | ||||
|           "customElement": true | ||||
|         } | ||||
|       ], | ||||
|       "exports": [ | ||||
|         { | ||||
|           "kind": "custom-element-definition", | ||||
|           "name": "ha-alert", | ||||
|           "declaration": { | ||||
|             "name": "HaAlert", | ||||
|             "module": "src/components/ha-alert.ts" | ||||
|           } | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| @@ -75,5 +75,5 @@ | ||||
|   "name": "Home Assistant Demo", | ||||
|   "short_name": "HA Demo", | ||||
|   "start_url": "/?homescreen=1", | ||||
|   "theme_color": "#009ac7" | ||||
|   "theme_color": "#03A9F4" | ||||
| } | ||||
|   | ||||
| @@ -89,14 +89,11 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|           )} | ||||
|         </div> | ||||
|         <div class="actions small-hidden"> | ||||
|           <ha-button | ||||
|             appearance="plain" | ||||
|             size="small" | ||||
|             href="https://www.home-assistant.io" | ||||
|             target="_blank" | ||||
|           > | ||||
|             ${this.hass.localize("ui.panel.page-demo.cards.demo.learn_more")} | ||||
|           </ha-button> | ||||
|           <a href="https://www.home-assistant.io" target="_blank"> | ||||
|             <ha-button> | ||||
|               ${this.hass.localize("ui.panel.page-demo.cards.demo.learn_more")} | ||||
|             </ha-button> | ||||
|           </a> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   | ||||
| @@ -68,7 +68,7 @@ | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-top { | ||||
|         flex: 1; | ||||
|         margin-top: calc( 2 * max(var(--safe-area-inset-top, 0px), 48px) + 46px ); | ||||
|         margin-top: calc( 2 * max(env(safe-area-inset-bottom), 48px) + 46px ); | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-bottom { | ||||
| @@ -76,7 +76,7 @@ | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       .ohf-logo { | ||||
|         margin: max(var(--safe-area-inset-bottom, 0px), 48px) 0; | ||||
|         margin: max(env(safe-area-inset-bottom), 48px) 0; | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|   | ||||
| @@ -1,30 +1,7 @@ | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| let changeFunction; | ||||
|  | ||||
| export const mockFrontend = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS("frontend/get_user_data", () => ({ | ||||
|     value: null, | ||||
|   })); | ||||
|   hass.mockWS("frontend/set_user_data", ({ key, value }) => { | ||||
|     if (key === "sidebar") { | ||||
|       changeFunction?.({ | ||||
|         value: { | ||||
|           panelOrder: value.panelOrder || [], | ||||
|           hiddenPanels: value.hiddenPanels || [], | ||||
|         }, | ||||
|       }); | ||||
|     } | ||||
|   }); | ||||
|   hass.mockWS("frontend/subscribe_user_data", (_msg, _hass, onChange) => { | ||||
|     changeFunction = onChange; | ||||
|     onChange?.({ | ||||
|       value: { | ||||
|         panelOrder: [], | ||||
|         hiddenPanels: [], | ||||
|       }, | ||||
|     }); | ||||
|     // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|     return () => {}; | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -11,7 +11,6 @@ import tseslint from "typescript-eslint"; | ||||
| import eslintConfigPrettier from "eslint-config-prettier"; | ||||
| import { configs as litConfigs } from "eslint-plugin-lit"; | ||||
| import { configs as wcConfigs } from "eslint-plugin-wc"; | ||||
| import { configs as a11yConfigs } from "eslint-plugin-lit-a11y"; | ||||
|  | ||||
| const _filename = fileURLToPath(import.meta.url); | ||||
| const _dirname = path.dirname(_filename); | ||||
| @@ -22,14 +21,13 @@ const compat = new FlatCompat({ | ||||
| }); | ||||
|  | ||||
| export default tseslint.config( | ||||
|   ...compat.extends("airbnb-base"), | ||||
|   ...compat.extends("airbnb-base", "plugin:lit-a11y/recommended"), | ||||
|   eslintConfigPrettier, | ||||
|   litConfigs["flat/all"], | ||||
|   tseslint.configs.recommended, | ||||
|   tseslint.configs.strict, | ||||
|   tseslint.configs.stylistic, | ||||
|   wcConfigs["flat/recommended"], | ||||
|   a11yConfigs.recommended, | ||||
|   { | ||||
|     plugins: { | ||||
|       "unused-imports": unusedImports, | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import type { Button } from "@material/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { html, LitElement, css, nothing } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import type { HaButton } from "../../../src/components/ha-button"; | ||||
|  | ||||
| @customElement("demo-black-white-row") | ||||
| class DemoBlackWhiteRow extends LitElement { | ||||
| @@ -25,9 +25,12 @@ class DemoBlackWhiteRow extends LitElement { | ||||
|               <slot name="light"></slot> | ||||
|             </div> | ||||
|             <div class="card-actions"> | ||||
|               <ha-button .disabled=${this.disabled} @click=${this.handleSubmit}> | ||||
|               <mwc-button | ||||
|                 .disabled=${this.disabled} | ||||
|                 @click=${this.handleSubmit} | ||||
|               > | ||||
|                 Submit | ||||
|               </ha-button> | ||||
|               </mwc-button> | ||||
|             </div> | ||||
|           </ha-card> | ||||
|         </div> | ||||
| @@ -37,9 +40,12 @@ class DemoBlackWhiteRow extends LitElement { | ||||
|               <slot name="dark"></slot> | ||||
|             </div> | ||||
|             <div class="card-actions"> | ||||
|               <ha-button .disabled=${this.disabled} @click=${this.handleSubmit}> | ||||
|               <mwc-button | ||||
|                 .disabled=${this.disabled} | ||||
|                 @click=${this.handleSubmit} | ||||
|               > | ||||
|                 Submit | ||||
|               </ha-button> | ||||
|               </mwc-button> | ||||
|             </div> | ||||
|           </ha-card> | ||||
|           ${this.value | ||||
| @@ -68,7 +74,7 @@ class DemoBlackWhiteRow extends LitElement { | ||||
|   } | ||||
|  | ||||
|   handleSubmit(ev) { | ||||
|     const content = (ev.target as HaButton).closest(".content")!; | ||||
|     const content = (ev.target as Button).closest(".content")!; | ||||
|     fireEvent(this, "submitted" as any, { | ||||
|       slot: content.classList.contains("light") ? "light" : "dark", | ||||
|     }); | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| import { LitElement, css, html, nothing } from "lit"; | ||||
| import { LitElement, css, html } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/dialogs/more-info/more-info-content"; | ||||
| import "../../../src/state-summary/state-card-content"; | ||||
| import "../ha-demo-options"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import { computeShowNewMoreInfo } from "../../../src/dialogs/more-info/const"; | ||||
|  | ||||
| @customElement("demo-more-info") | ||||
| class DemoMoreInfo extends LitElement { | ||||
| @@ -22,13 +21,11 @@ class DemoMoreInfo extends LitElement { | ||||
|       <div class="root"> | ||||
|         <div id="card"> | ||||
|           <ha-card> | ||||
|             ${!computeShowNewMoreInfo(state) | ||||
|               ? html`<state-card-content | ||||
|                   .stateObj=${state} | ||||
|                   .hass=${this.hass} | ||||
|                   in-dialog | ||||
|                 ></state-card-content>` | ||||
|               : nothing} | ||||
|             <state-card-content | ||||
|               .stateObj=${state} | ||||
|               .hass=${this.hass} | ||||
|               in-dialog | ||||
|             ></state-card-content> | ||||
|  | ||||
|             <more-info-content | ||||
|               .hass=${this.hass} | ||||
|   | ||||
| @@ -38,12 +38,12 @@ class PageDescription extends HaMarkdown { | ||||
|       } | ||||
|       .title { | ||||
|         font-size: 42px; | ||||
|         line-height: var(--ha-line-height-condensed); | ||||
|         line-height: 56px; | ||||
|         padding-bottom: 8px; | ||||
|       } | ||||
|       .subtitle { | ||||
|         font-size: var(--ha-font-size-l); | ||||
|         line-height: var(--ha-line-height-normal); | ||||
|         line-height: 24px; | ||||
|       } | ||||
|       .root { | ||||
|         max-width: 800px; | ||||
|   | ||||
| @@ -1106,7 +1106,7 @@ export default { | ||||
|       friendly_name: "Philips Hue", | ||||
|       entity_picture: null, | ||||
|       description: | ||||
|         "Press the button on the bridge to register Philips Hue with Home Assistant.", | ||||
|         "Press the button on the bridge to register Philips Hue with Home Assistant.\n\n", | ||||
|       submit_caption: "I have pressed the button", | ||||
|     }, | ||||
|     last_changed: "2018-07-19T10:44:46.515160+00:00", | ||||
|   | ||||
| @@ -17,10 +17,6 @@ export const createMediaPlayerEntities = () => [ | ||||
|       new Date().getTime() - 23000 | ||||
|     ).toISOString(), | ||||
|     volume_level: 0.5, | ||||
|     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||
|     source: "AirPlay", | ||||
|     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||
|     sound_mode: "Music", | ||||
|   }), | ||||
|   getEntity("media_player", "music_playing", "playing", { | ||||
|     friendly_name: "Playing The Music", | ||||
| @@ -28,8 +24,8 @@ export const createMediaPlayerEntities = () => [ | ||||
|     media_title: "I Wanna Be A Hippy (Flamman & Abraxas Radio Mix)", | ||||
|     media_artist: "Technohead", | ||||
|     // Pause + Seek + Volume Set + Volume Mute + Previous Track + Next Track + Play Media + | ||||
|     // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media + Grouping | ||||
|     supported_features: 784959, | ||||
|     // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media | ||||
|     supported_features: 195135, | ||||
|     entity_picture: "/images/album_cover.jpg", | ||||
|     media_duration: 300, | ||||
|     media_position: 0, | ||||
| @@ -38,9 +34,6 @@ export const createMediaPlayerEntities = () => [ | ||||
|       new Date().getTime() - 23000 | ||||
|     ).toISOString(), | ||||
|     volume_level: 0.5, | ||||
|     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||
|     sound_mode: "Music", | ||||
|     group_members: ["media_player.playing", "media_player.stream_playing"], | ||||
|   }), | ||||
|   getEntity("media_player", "stream_playing", "playing", { | ||||
|     friendly_name: "Playing the Stream", | ||||
| @@ -156,18 +149,15 @@ export const createMediaPlayerEntities = () => [ | ||||
|   }), | ||||
|   getEntity("media_player", "receiver_on", "on", { | ||||
|     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||
|     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||
|     volume_level: 0.63, | ||||
|     is_volume_muted: false, | ||||
|     source: "TV", | ||||
|     sound_mode: "Movie", | ||||
|     friendly_name: "Receiver (selectable sources)", | ||||
|     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode | ||||
|     supported_features: 84364, | ||||
|   }), | ||||
|   getEntity("media_player", "receiver_off", "off", { | ||||
|     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||
|     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||
|     friendly_name: "Receiver (selectable sources)", | ||||
|     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode | ||||
|     supported_features: 84364, | ||||
|   | ||||
| @@ -252,12 +252,12 @@ class HaGallery extends LitElement { | ||||
|       .page-footer .header { | ||||
|         font-size: var(--ha-font-size-l); | ||||
|         font-weight: var(--ha-font-weight-medium); | ||||
|         line-height: var(--ha-line-height-normal); | ||||
|         line-height: 28px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer .secondary { | ||||
|         line-height: var(--ha-line-height-normal); | ||||
|         line-height: 23px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|   | ||||
| @@ -18,6 +18,7 @@ import { HaDeviceAction } from "../../../../src/panels/config/automation/action/ | ||||
| import { HaEventAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-event"; | ||||
| import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | ||||
| import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | ||||
| import { HaPlayMediaAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-play_media"; | ||||
| import { HaRepeatAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-repeat"; | ||||
| import { HaSequenceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-sequence"; | ||||
| import { HaServiceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-service"; | ||||
| @@ -31,6 +32,7 @@ const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
|   { name: "Service", actions: [HaServiceAction.defaultConfig] }, | ||||
|   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, | ||||
|   { name: "Delay", actions: [HaDelayAction.defaultConfig] }, | ||||
|   { name: "Play media", actions: [HaPlayMediaAction.defaultConfig] }, | ||||
|   { name: "Wait", actions: [HaWaitAction.defaultConfig] }, | ||||
|   { name: "WaitForTrigger", actions: [HaWaitForTriggerAction.defaultConfig] }, | ||||
|   { name: "Repeat", actions: [HaRepeatAction.defaultConfig] }, | ||||
|   | ||||
| @@ -147,13 +147,13 @@ The `title ` option should not be used without a description. | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
|   <ha-button slot="action">Undo</ha-button> | ||||
|   <mwc-button slot="action" label="Undo"></mwc-button> | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
|   <ha-button slot="action">Undo</ha-button> | ||||
|   <mwc-button slot="action" label="Undo"></mwc-button> | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
|   | ||||
| @@ -1,10 +1,10 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-logo-svg"; | ||||
|  | ||||
| const alerts: { | ||||
| @@ -78,13 +78,13 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action", | ||||
|     type: "error", | ||||
|     actionSlot: html`<ha-button size="small" slot="action">restart</ha-button>`, | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Unsaved data", | ||||
|     description: "You have unsaved data", | ||||
|     type: "warning", | ||||
|     actionSlot: html`<ha-button size="small" slot="action">save</ha-button>`, | ||||
|     actionSlot: html`<mwc-button slot="action" label="save"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted icon", | ||||
| @@ -108,7 +108,7 @@ const alerts: { | ||||
|     title: "Slotted action", | ||||
|     description: "Alert with slotted action", | ||||
|     type: "info", | ||||
|     actionSlot: html`<ha-button slot="action">action</ha-button>`, | ||||
|     actionSlot: html`<mwc-button slot="action" label="action"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     description: "Dismissable information (RTL)", | ||||
| @@ -120,7 +120,7 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action (RTL)", | ||||
|     type: "error", | ||||
|     actionSlot: html`<ha-button slot="action">restart</ha-button>`, | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|     rtl: true, | ||||
|   }, | ||||
|   { | ||||
| @@ -211,7 +211,7 @@ export class DemoHaAlert extends LitElement { | ||||
|       max-height: 24px; | ||||
|       width: 24px; | ||||
|     } | ||||
|     ha-button { | ||||
|     mwc-button { | ||||
|       --mdc-theme-primary: var(--primary-text-color); | ||||
|     } | ||||
|   `; | ||||
|   | ||||
| @@ -1,67 +0,0 @@ | ||||
| --- | ||||
| title: Button | ||||
| --- | ||||
|  | ||||
| <style> | ||||
|   .wrapper { | ||||
|     display: flex; | ||||
|     gap: 24px; | ||||
|   } | ||||
| </style> | ||||
|  | ||||
| # Button `<ha-button>` | ||||
|  | ||||
| ## Implementation | ||||
|  | ||||
| ### Example Usage | ||||
|  | ||||
| <div class="wrapper"> | ||||
|   <ha-button> | ||||
|     simple button | ||||
|   </ha-button> | ||||
|   <ha-button appearance="plain"> | ||||
|     plain button | ||||
|   </ha-button> | ||||
|   <ha-button appearance="filled"> | ||||
|     filled button | ||||
|   </ha-button> | ||||
|  | ||||
|   <ha-button size="small"> | ||||
|     small | ||||
|   </ha-button> | ||||
| </div> | ||||
|  | ||||
| ```html | ||||
| <ha-button> simple button </ha-button> | ||||
|  | ||||
| <ha-button size="small"> small </ha-button> | ||||
| ``` | ||||
|  | ||||
| ### API | ||||
|  | ||||
| This component is based on the webawesome button component. | ||||
| Check the [webawesome documentation](https://webawesome.com/docs/components/button/) for more details. | ||||
|  | ||||
| **Slots** | ||||
|  | ||||
| - default slot: Label of the button | ||||
|   ` - no default | ||||
| - `start`: The prefix container (usually for icons). | ||||
|   ` - no default | ||||
| - `end`: The suffix container (usually for icons). | ||||
|   ` - no default | ||||
|  | ||||
| **Properties/Attributes** | ||||
|  | ||||
| | Name       | Type                                           | Default  | Description                                                                       | | ||||
| | ---------- | ---------------------------------------------- | -------- | --------------------------------------------------------------------------------- | | ||||
| | appearance | "accent"/"filled"/"plain"                      | "accent" | Sets the button appearance.                                                       | | ||||
| | variants   | "brand"/"danger"/"neutral"/"warning"/"success" | "brand"  | Sets the button color variant. "brand" is default.                                | | ||||
| | size       | "small"/"medium"                               | "medium" | Sets the button size.                                                             | | ||||
| | loading    | Boolean                                        | false    | Shows a loading indicator instead of the buttons label and disable buttons click. | | ||||
| | disabled   | Boolean                                        | false    | Disables the button and prevents user interaction.                                | | ||||
|  | ||||
| **CSS Custom Properties** | ||||
|  | ||||
| - `--ha-button-height` - Height of the button. | ||||
| - `--ha-button-border-radius` - Border radius of the button. Defaults to `var(--ha-border-radius-pill)`. | ||||
| @@ -1,171 +0,0 @@ | ||||
| import { mdiHome } from "@mdi/js"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||
| import { titleCase } from "../../../../src/common/string/title-case"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg"; | ||||
|  | ||||
| const appearances = ["accent", "filled", "plain"]; | ||||
| const variants = ["brand", "danger", "neutral", "warning", "success"]; | ||||
|  | ||||
| @customElement("demo-components-ha-button") | ||||
| export class DemoHaButton extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${["light", "dark"].map( | ||||
|         (mode) => html` | ||||
|           <div class=${mode}> | ||||
|             <ha-card header="ha-button in ${mode}"> | ||||
|               <div class="card-content"> | ||||
|                 ${variants.map( | ||||
|                   (variant) => html` | ||||
|                     <div> | ||||
|                       ${appearances.map( | ||||
|                         (appearance) => html` | ||||
|                           <ha-button | ||||
|                             .appearance=${appearance} | ||||
|                             .variant=${variant} | ||||
|                           > | ||||
|                             <ha-svg-icon | ||||
|                               .path=${mdiHomeAssistant} | ||||
|                               slot="start" | ||||
|                             ></ha-svg-icon> | ||||
|                             ${titleCase(`${variant} ${appearance}`)} | ||||
|                             <ha-svg-icon | ||||
|                               .path=${mdiHome} | ||||
|                               slot="end" | ||||
|                             ></ha-svg-icon> | ||||
|                           </ha-button> | ||||
|                         ` | ||||
|                       )} | ||||
|                     </div> | ||||
|                     <div> | ||||
|                       ${appearances.map( | ||||
|                         (appearance) => html` | ||||
|                           <ha-button | ||||
|                             .appearance=${appearance} | ||||
|                             .variant=${variant} | ||||
|                             size="small" | ||||
|                           > | ||||
|                             ${titleCase(`${variant} ${appearance}`)} | ||||
|                           </ha-button> | ||||
|                         ` | ||||
|                       )} | ||||
|                     </div> | ||||
|                     <div> | ||||
|                       ${appearances.map( | ||||
|                         (appearance) => html` | ||||
|                           <ha-button | ||||
|                             .appearance=${appearance} | ||||
|                             .variant=${variant} | ||||
|                             loading | ||||
|                           > | ||||
|                             <ha-svg-icon | ||||
|                               .path=${mdiHomeAssistant} | ||||
|                               slot="start" | ||||
|                             ></ha-svg-icon> | ||||
|                             ${titleCase(`${variant} ${appearance}`)} | ||||
|                             <ha-svg-icon | ||||
|                               .path=${mdiHome} | ||||
|                               slot="end" | ||||
|                             ></ha-svg-icon> | ||||
|                           </ha-button> | ||||
|                         ` | ||||
|                       )} | ||||
|                     </div> | ||||
|                   ` | ||||
|                 )} | ||||
|                 ${variants.map( | ||||
|                   (variant) => html` | ||||
|                     <div> | ||||
|                       ${appearances.map( | ||||
|                         (appearance) => html` | ||||
|                           <ha-button | ||||
|                             .variant=${variant} | ||||
|                             .appearance=${appearance} | ||||
|                             disabled | ||||
|                           > | ||||
|                             ${titleCase(`${appearance}`)} | ||||
|                           </ha-button> | ||||
|                         ` | ||||
|                       )} | ||||
|                     </div> | ||||
|                     <div> | ||||
|                       ${appearances.map( | ||||
|                         (appearance) => html` | ||||
|                           <ha-button | ||||
|                             .variant=${variant} | ||||
|                             .appearance=${appearance} | ||||
|                             size="small" | ||||
|                             disabled | ||||
|                           > | ||||
|                             ${titleCase(`${appearance}`)} | ||||
|                           </ha-button> | ||||
|                         ` | ||||
|                       )} | ||||
|                     </div> | ||||
|                   ` | ||||
|                 )} | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           </div> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: "default", | ||||
|         themes: {}, | ||||
|         darkMode: true, | ||||
|         theme: "default", | ||||
|       }, | ||||
|       undefined, | ||||
|       undefined, | ||||
|       true | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host { | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|     } | ||||
|     .dark, | ||||
|     .light { | ||||
|       display: block; | ||||
|       background-color: var(--primary-background-color); | ||||
|       padding: 0 50px; | ||||
|     } | ||||
|     .button { | ||||
|       padding: unset; | ||||
|     } | ||||
|     ha-card { | ||||
|       margin: 24px auto; | ||||
|     } | ||||
|     .card-content { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       gap: 24px; | ||||
|     } | ||||
|     .card-content div { | ||||
|       display: flex; | ||||
|       gap: 8px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-components-ha-button": DemoHaButton; | ||||
|   } | ||||
| } | ||||
| @@ -135,7 +135,7 @@ export class DemoHaControlSelect extends LitElement { | ||||
|                 .options=${options} | ||||
|                 class=${ifDefined(config.class)} | ||||
|                 @value-changed=${this.handleValueChanged} | ||||
|                 .label=${label} | ||||
|                 aria-labelledby=${id} | ||||
|                 ?disabled=${config.disabled} | ||||
|               > | ||||
|               </ha-control-select> | ||||
| @@ -156,7 +156,7 @@ export class DemoHaControlSelect extends LitElement { | ||||
|                   vertical | ||||
|                   class=${ifDefined(config.class)} | ||||
|                   @value-changed=${this.handleValueChanged} | ||||
|                   .label=${label} | ||||
|                   aria-labelledby=${id} | ||||
|                   ?disabled=${config.disabled} | ||||
|                 > | ||||
|                 </ha-control-select> | ||||
|   | ||||
| @@ -97,7 +97,7 @@ export class DemoHaBarSlider extends LitElement { | ||||
|                 class=${ifDefined(config.class)} | ||||
|                 @value-changed=${this.handleValueChanged} | ||||
|                 @slider-moved=${this.handleSliderMoved} | ||||
|                 .label=${label} | ||||
|                 aria-labelledby=${id} | ||||
|                 .unit=${config.unit} | ||||
|               > | ||||
|               </ha-control-slider> | ||||
| @@ -119,7 +119,7 @@ export class DemoHaBarSlider extends LitElement { | ||||
|                   class=${ifDefined(config.class)} | ||||
|                   @value-changed=${this.handleValueChanged} | ||||
|                   @slider-moved=${this.handleSliderMoved} | ||||
|                   .label=${label} | ||||
|                   aria-label=${label} | ||||
|                   .unit=${config.unit} | ||||
|                 > | ||||
|                 </ha-control-slider> | ||||
|   | ||||
| @@ -63,7 +63,7 @@ export class DemoHaControlSwitch extends LitElement { | ||||
|                 @change=${this.handleValueChanged} | ||||
|                 .pathOn=${mdiLightbulb} | ||||
|                 .pathOff=${mdiLightbulbOff} | ||||
|                 .label=${label} | ||||
|                 aria-labelledby=${id} | ||||
|                 ?disabled=${config.disabled} | ||||
|                 ?reversed=${config.reversed} | ||||
|               > | ||||
| @@ -84,7 +84,7 @@ export class DemoHaControlSwitch extends LitElement { | ||||
|                   vertical | ||||
|                   class=${ifDefined(config.class)} | ||||
|                   @change=${this.handleValueChanged} | ||||
|                   .label=${label} | ||||
|                   aria-label=${label} | ||||
|                   .pathOn=${mdiGarageOpen} | ||||
|                   .pathOff=${mdiGarage} | ||||
|                   ?disabled=${config.disabled} | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import "@material/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
|   | ||||
| @@ -1,37 +0,0 @@ | ||||
| --- | ||||
| title: Marquee Text | ||||
| --- | ||||
|  | ||||
| # Marquee Text `<ha-marquee-text>` | ||||
|  | ||||
| Marquee text component scrolls text horizontally if it overflows its container. It supports pausing on hover and customizable speed and pause duration. | ||||
|  | ||||
| ## Implementation | ||||
|  | ||||
| ### Example Usage | ||||
|  | ||||
| <ha-marquee-text style="width: 200px;"> | ||||
|     This is a long text that will scroll horizontally if it overflows the container. | ||||
| </ha-marquee-text> | ||||
|  | ||||
| ```html | ||||
| <ha-marquee-text style="width: 200px;"> | ||||
|   This is a long text that will scroll horizontally if it overflows the | ||||
|   container. | ||||
| </ha-marquee-text> | ||||
| ``` | ||||
|  | ||||
| ### API | ||||
|  | ||||
| **Slots** | ||||
|  | ||||
| - default slot: The text content to be displayed and scrolled. | ||||
|   - no default | ||||
|  | ||||
| **Properties/Attributes** | ||||
|  | ||||
| | Name           | Type    | Default | Description                                                                  | | ||||
| | -------------- | ------- | ------- | ---------------------------------------------------------------------------- | | ||||
| | speed          | number  | `15`    | The speed of the scrolling animation. Higher values result in faster scroll. | | ||||
| | pause-on-hover | boolean | `true`  | Whether to pause the scrolling animation when                                | | ||||
| | pause-duration | number  | `1000`  | The delay in milliseconds before the scrolling animation starts/restarts.    | | ||||
| @@ -1,25 +0,0 @@ | ||||
| import { css, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-marquee-text"; | ||||
|  | ||||
| @customElement("demo-components-ha-marquee-text") | ||||
| export class DemoHaMarqueeText extends LitElement { | ||||
|   static styles = css` | ||||
|     ha-card { | ||||
|       max-width: 600px; | ||||
|       margin: 24px auto; | ||||
|     } | ||||
|     .card-content { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: flex-start; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-components-ha-marquee-text": DemoHaMarqueeText; | ||||
|   } | ||||
| } | ||||
| @@ -1,32 +0,0 @@ | ||||
| --- | ||||
| title: Progress Button | ||||
| --- | ||||
|  | ||||
| <style> | ||||
|   .wrapper { | ||||
|     display: flex; | ||||
|     gap: 24px; | ||||
|   } | ||||
| </style> | ||||
|  | ||||
| # Progress Button `<ha-progress-button>` | ||||
|  | ||||
| ### API | ||||
|  | ||||
| This component is a wrapper around `<ha-button>` that adds support for showing progress | ||||
|  | ||||
| **Slots** | ||||
|  | ||||
| - default slot: Label of the button | ||||
|   ` - no default | ||||
|  | ||||
| **Properties/Attributes** | ||||
|  | ||||
| | Name       | Type                                           | Default   | Description                                        | | ||||
| | ---------- | ---------------------------------------------- | --------- | -------------------------------------------------- | | ||||
| | label      | string                                         | "accent"  | Sets the button label.                             | | ||||
| | disabled   | Boolean                                        | false     | Disables the button if true.                       | | ||||
| | progress   | Boolean                                        | false     | Shows a progress indicator on the button.          | | ||||
| | appearance | "accent"/"filled"/"plain"                      | "accent"  | Sets the button appearance.                        | | ||||
| | variants   | "brand"/"danger"/"neutral"/"warning"/"success" | "brand"   | Sets the button color variant. "brand" is default. | | ||||
| | iconPath   | string                                         | undefined | Sets the icon path for the button.                 | | ||||
| @@ -1,139 +0,0 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg"; | ||||
|  | ||||
| @customElement("demo-components-ha-progress-button") | ||||
| export class DemoHaProgressButton extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${["light", "dark"].map( | ||||
|         (mode) => html` | ||||
|           <div class=${mode}> | ||||
|             <ha-card header="ha-progress-button in ${mode}"> | ||||
|               <div class="card-content"> | ||||
|                 <ha-progress-button @click=${this._clickedSuccess}> | ||||
|                   Success | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button @click=${this._clickedFail}> | ||||
|                   Fail | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button size="small" @click=${this._clickedSuccess}> | ||||
|                   small | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button | ||||
|                   appearance="filled" | ||||
|                   @click=${this._clickedSuccess} | ||||
|                 > | ||||
|                   filled | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button | ||||
|                   appearance="plain" | ||||
|                   @click=${this._clickedSuccess} | ||||
|                 > | ||||
|                   plain | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button | ||||
|                   variant="warning" | ||||
|                   @click=${this._clickedSuccess} | ||||
|                 > | ||||
|                   warning | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button | ||||
|                   variant="neutral" | ||||
|                   @click=${this._clickedSuccess} | ||||
|                   label="with icon" | ||||
|                   .iconPath=${mdiHomeAssistant} | ||||
|                 > | ||||
|                   With Icon | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button progress @click=${this._clickedSuccess}> | ||||
|                   progress | ||||
|                 </ha-progress-button> | ||||
|                 <ha-progress-button disabled @click=${this._clickedSuccess}> | ||||
|                   disabled | ||||
|                 </ha-progress-button> | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           </div> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: "default", | ||||
|         themes: {}, | ||||
|         darkMode: true, | ||||
|         theme: "default", | ||||
|       }, | ||||
|       undefined, | ||||
|       undefined, | ||||
|       true | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   private async _clickedSuccess(ev: CustomEvent): Promise<void> { | ||||
|     console.log("Clicked success"); | ||||
|     const button = ev.currentTarget as any; | ||||
|     button.progress = true; | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       button.actionSuccess(); | ||||
|       button.progress = false; | ||||
|     }, 1000); | ||||
|   } | ||||
|  | ||||
|   private async _clickedFail(ev: CustomEvent): Promise<void> { | ||||
|     const button = ev.currentTarget as any; | ||||
|     button.progress = true; | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       button.actionError(); | ||||
|       button.progress = false; | ||||
|     }, 1000); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host { | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|     } | ||||
|     .dark, | ||||
|     .light { | ||||
|       display: block; | ||||
|       background-color: var(--primary-background-color); | ||||
|       padding: 0 50px; | ||||
|     } | ||||
|     .button { | ||||
|       padding: unset; | ||||
|     } | ||||
|     ha-card { | ||||
|       margin: 24px auto; | ||||
|     } | ||||
|     .card-content { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       gap: 24px; | ||||
|     } | ||||
|     .card-content div { | ||||
|       display: flex; | ||||
|       gap: 8px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-components-ha-progress-button": DemoHaProgressButton; | ||||
|   } | ||||
| } | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| @@ -415,34 +416,6 @@ const SCHEMAS: { | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       items: { | ||||
|         name: "Items", | ||||
|         selector: { | ||||
|           object: { | ||||
|             label_field: "name", | ||||
|             description_field: "value", | ||||
|             multiple: true, | ||||
|             fields: { | ||||
|               name: { | ||||
|                 label: "Name", | ||||
|                 selector: { text: {} }, | ||||
|                 required: true, | ||||
|               }, | ||||
|               value: { | ||||
|                 label: "Value", | ||||
|                 selector: { | ||||
|                   number: { | ||||
|                     mode: "slider", | ||||
|                     min: 0, | ||||
|                     max: 100, | ||||
|                     unit_of_measurement: "%", | ||||
|                   }, | ||||
|                 }, | ||||
|               }, | ||||
|             }, | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -1,38 +0,0 @@ | ||||
| --- | ||||
| title: Slider | ||||
| subtitle: A slider component for selecting a value from a range. | ||||
| --- | ||||
|  | ||||
| <style> | ||||
|   .wrapper { | ||||
|     display: flex; | ||||
|     gap: 24px; | ||||
|   } | ||||
| </style> | ||||
|  | ||||
| # Slider `<ha-slider>` | ||||
|  | ||||
| ## Implementation | ||||
|  | ||||
| ### Example Usage | ||||
|  | ||||
| <div class="wrapper"> | ||||
|   <ha-slider size="small" with-markers min="0" max="8" value="4"></ha-slider> | ||||
|   <ha-slider size="medium"></ha-slider> | ||||
| </div> | ||||
|  | ||||
| ```html | ||||
| <ha-slider size="small" with-markers min="0" max="8" value="4"></ha-slider> | ||||
| <ha-slider size="medium"></ha-slider> | ||||
| ``` | ||||
|  | ||||
| ### API | ||||
|  | ||||
| This component is based on the webawesome slider component. | ||||
| Check the [webawesome documentation](https://webawesome.com/docs/components/slider/) for more details. | ||||
|  | ||||
| **CSS Custom Properties** | ||||
|  | ||||
| - `--ha-slider-track-size` - Height of the slider track. Defaults to `4px`. | ||||
| - `--ha-slider-thumb-color` - Color of the slider thumb. Defaults to `var(--primary-color)`. | ||||
| - `--ha-slider-indicator-color` - Color of the filled portion of the slider track. Defaults to `var(--primary-color)`. | ||||
| @@ -1,100 +0,0 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../../src/components/ha-bar"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-spinner"; | ||||
| import "../../../../src/components/ha-slider"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
|  | ||||
| @customElement("demo-components-ha-slider") | ||||
| export class DemoHaSlider extends LitElement { | ||||
|   @property({ attribute: false }) hass!: HomeAssistant; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${["light", "dark"].map( | ||||
|         (mode) => html` | ||||
|           <div class=${mode}> | ||||
|             <ha-card header="ha-slider ${mode} demo"> | ||||
|               <div class="card-content"> | ||||
|                 <span>Default (disabled)</span> | ||||
|                 <ha-slider | ||||
|                   disabled | ||||
|                   min="0" | ||||
|                   max="8" | ||||
|                   value="4" | ||||
|                   with-markers | ||||
|                 ></ha-slider> | ||||
|                 <span>Small</span> | ||||
|                 <ha-slider | ||||
|                   size="small" | ||||
|                   min="0" | ||||
|                   max="8" | ||||
|                   value="4" | ||||
|                   with-markers | ||||
|                 ></ha-slider> | ||||
|                 <span>Medium</span> | ||||
|                 <ha-slider | ||||
|                   size="medium" | ||||
|                   min="0" | ||||
|                   max="8" | ||||
|                   value="4" | ||||
|                   with-markers | ||||
|                 ></ha-slider> | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           </div> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: "default", | ||||
|         themes: {}, | ||||
|         darkMode: true, | ||||
|         theme: "default", | ||||
|       }, | ||||
|       undefined, | ||||
|       undefined, | ||||
|       true | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host { | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|     } | ||||
|     .dark, | ||||
|     .light { | ||||
|       display: block; | ||||
|       background-color: var(--primary-background-color); | ||||
|       padding: 0 50px; | ||||
|       margin: 16px; | ||||
|       border-radius: 8px; | ||||
|     } | ||||
|     ha-card { | ||||
|       margin: 24px auto; | ||||
|     } | ||||
|     .card-content { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       gap: 24px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-components-ha-slider": DemoHaSlider; | ||||
|   } | ||||
| } | ||||
| @@ -6,23 +6,21 @@ A tooltip's target is its _first child element_, so you should only wrap one ele | ||||
|  | ||||
| Tooltips use `display: contents` so they won't interfere with how elements are positioned in a flex or grid layout. | ||||
|  | ||||
| <ha-button id="hover">Hover Me</ha-button> | ||||
| <ha-tooltip for="hover"> | ||||
| This is a tooltip | ||||
| <ha-tooltip content="This is a tooltip"> | ||||
|   <ha-button>Hover Me</ha-button> | ||||
| </ha-tooltip> | ||||
|  | ||||
| ``` | ||||
| <ha-button id="hover">Hover Me</ha-button> | ||||
| <ha-tooltip for="hover"> | ||||
| This is a tooltip | ||||
| <ha-tooltip content="This is a tooltip"> | ||||
|   <ha-button>Hover Me</ha-button> | ||||
| </ha-tooltip> | ||||
| ``` | ||||
|  | ||||
| ## Documentation | ||||
|  | ||||
| This element is based on webawesome `wa-tooltip` it only sets some css tokens and has a custom show/hide animation. | ||||
| This element is based on shoelace `sl-tooltip` it only sets some css tokens and has a custom show/hide animation. | ||||
|  | ||||
| <a href="https://webawesome.com/docs/components/tooltip/" target="_blank" rel="noopener noreferrer">Webawesome documentation</a> | ||||
| <a href="https://shoelace.style/components/tooltip" target="_blank" rel="noopener noreferrer">Shoelace documentation</a> | ||||
|  | ||||
| ### HA style tokens | ||||
|  | ||||
| @@ -30,7 +28,7 @@ In your theme settings use this without the prefixed `--`. | ||||
|  | ||||
| - `--ha-tooltip-border-radius` (Default: 4px) | ||||
| - `--ha-tooltip-arrow-size` (Default: 8px) | ||||
| - `--wa-tooltip-font-family` (Default: `var(--ha-font-family-body)`) | ||||
| - `--sl-tooltip-font-family` (Default: `var(--ha-font-family-body)`) | ||||
| - `--ha-tooltip-font-size` (Default: `var(--ha-font-size-s)`) | ||||
| - `--wa-tooltip-font-weight` (Default: `var(--ha-font-weight-normal)`) | ||||
| - `--wa-tooltip-line-height` (Default: `var(--ha-line-height-condensed)`) | ||||
| - `--sl-tooltip-font-weight` (Default: `var(--ha-font-weight-normal)`) | ||||
| - `--sl-tooltip-line-height` (Default: `var(--ha-line-height-condensed)`) | ||||
|   | ||||
| @@ -11,7 +11,6 @@ import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| import { mockIcons } from "../../../../demo/src/stubs/icons"; | ||||
| import { ClimateEntityFeature } from "../../../../src/data/climate"; | ||||
| import { FanEntityFeature } from "../../../../src/data/fan"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("switch", "tv_outlet", "on", { | ||||
| @@ -101,15 +100,6 @@ const ENTITIES = [ | ||||
|       ClimateEntityFeature.FAN_MODE + | ||||
|       ClimateEntityFeature.TARGET_TEMPERATURE_RANGE, | ||||
|   }), | ||||
|   getEntity("fan", "fan_demo", "on", { | ||||
|     friendly_name: "Ceiling fan", | ||||
|     device_class: "fan", | ||||
|     direction: "reverse", | ||||
|     supported_features: | ||||
|       FanEntityFeature.DIRECTION + | ||||
|       FanEntityFeature.SET_SPEED + | ||||
|       FanEntityFeature.OSCILLATE, | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
| @@ -271,33 +261,6 @@ const CONFIGS = [ | ||||
|   - type: target-temperature | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Fan direction feature", | ||||
|     config: ` | ||||
| - type: tile | ||||
|   entity: fan.fan_demo | ||||
|   features: | ||||
|   - type: fan-direction | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Fan speed feature", | ||||
|     config: ` | ||||
| - type: tile | ||||
|   entity: fan.fan_demo | ||||
|   features: | ||||
|   - type: fan-speed | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Fan oscillate feature", | ||||
|     config: ` | ||||
| - type: tile | ||||
|   entity: fan.fan_demo | ||||
|   features: | ||||
|   - type: fan-oscillate | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-tile-card") | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import type { ActionHandlerEvent } from "../../../../src/data/lovelace/action_handler"; | ||||
| import { actionHandler } from "../../../../src/panels/lovelace/common/directives/action-handler-directive"; | ||||
| @@ -13,16 +13,12 @@ export class DemoUtilLongPress extends LitElement { | ||||
|       ${[1, 2, 3].map( | ||||
|         () => html` | ||||
|           <ha-card> | ||||
|             <ha-button | ||||
|               appearance="plain" | ||||
|             <mwc-button | ||||
|               @action=${this._handleAction} | ||||
|               .actionHandler=${actionHandler({ | ||||
|                 hasHold: true, | ||||
|                 hasDoubleClick: true, | ||||
|               })} | ||||
|               .actionHandler=${actionHandler({})} | ||||
|             > | ||||
|               (long) press me! | ||||
|             </ha-button> | ||||
|             </mwc-button> | ||||
|  | ||||
|             <textarea></textarea> | ||||
|  | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| --- | ||||
| title: Fan | ||||
| --- | ||||
| @@ -1,50 +0,0 @@ | ||||
| import type { PropertyValues, TemplateResult } from "lit"; | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/dialogs/more-info/more-info-content"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import type { MockHomeAssistant } from "../../../../src/fake_data/provide_hass"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-more-infos"; | ||||
| import { FanEntityFeature } from "../../../../src/data/fan"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("fan", "fan", "on", { | ||||
|     friendly_name: "Fan", | ||||
|     device_class: "fan", | ||||
|     supported_features: | ||||
|       FanEntityFeature.OSCILLATE + | ||||
|       FanEntityFeature.DIRECTION + | ||||
|       FanEntityFeature.SET_SPEED, | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-more-info-fan") | ||||
| class DemoMoreInfoFan extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: MockHomeAssistant; | ||||
|  | ||||
|   @query("demo-more-infos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <demo-more-infos | ||||
|         .hass=${this.hass} | ||||
|         .entities=${ENTITIES.map((ent) => ent.entityId)} | ||||
|       ></demo-more-infos> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-more-info-fan": DemoMoreInfoFan; | ||||
|   } | ||||
| } | ||||
| @@ -11,10 +11,7 @@ import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button-menu"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import type { | ||||
|   HaFormSchema, | ||||
|   HaFormDataContainer, | ||||
| } from "../../../../src/components/ha-form/types"; | ||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
| @@ -36,7 +33,6 @@ import { haStyle } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart"; | ||||
| import { hassioStyle } from "../../resources/hassio-style"; | ||||
| import type { ObjectSelector, Selector } from "../../../../src/data/selector"; | ||||
|  | ||||
| const SUPPORTED_UI_TYPES = [ | ||||
|   "string", | ||||
| @@ -82,125 +78,78 @@ class HassioAddonConfig extends LitElement { | ||||
|  | ||||
|   @query("ha-yaml-editor") private _editor?: HaYamlEditor; | ||||
|  | ||||
|   private _getTranslationEntry( | ||||
|     language: string, | ||||
|     entry: HaFormSchema, | ||||
|     options?: { path?: string[] } | ||||
|   ) { | ||||
|     let parent = this.addon.translations[language]?.configuration; | ||||
|     if (!parent) return undefined; | ||||
|     if (options?.path) { | ||||
|       for (const key of options.path) { | ||||
|         parent = parent[key]?.fields; | ||||
|         if (!parent) return undefined; | ||||
|       } | ||||
|     } | ||||
|     return parent[entry.name]; | ||||
|   } | ||||
|  | ||||
|   public computeLabel = ( | ||||
|     entry: HaFormSchema, | ||||
|     _data: HaFormDataContainer, | ||||
|     options?: { path?: string[] } | ||||
|   ): string => | ||||
|     this._getTranslationEntry(this.hass.language, entry, options)?.name || | ||||
|     this._getTranslationEntry("en", entry, options)?.name || | ||||
|   public computeLabel = (entry: HaFormSchema): string => | ||||
|     this.addon.translations[this.hass.language]?.configuration?.[entry.name] | ||||
|       ?.name || | ||||
|     this.addon.translations.en?.configuration?.[entry.name]?.name || | ||||
|     entry.name; | ||||
|  | ||||
|   public computeHelper = ( | ||||
|     entry: HaFormSchema, | ||||
|     options?: { path?: string[] } | ||||
|   ): string => | ||||
|     this._getTranslationEntry(this.hass.language, entry, options) | ||||
|   public computeHelper = (entry: HaFormSchema): string => | ||||
|     this.addon.translations[this.hass.language]?.configuration?.[entry.name] | ||||
|       ?.description || | ||||
|     this._getTranslationEntry("en", entry, options)?.description || | ||||
|     this.addon.translations.en?.configuration?.[entry.name]?.description || | ||||
|     ""; | ||||
|  | ||||
|   private _convertSchema = memoizeOne( | ||||
|     // Convert supervisor schema to selectors | ||||
|     (schema: readonly HaFormSchema[]): HaFormSchema[] => | ||||
|       this._convertSchemaElements(schema) | ||||
|     (schema: Record<string, any>): HaFormSchema[] => | ||||
|       schema.map((entry) => | ||||
|         entry.type === "select" | ||||
|           ? { | ||||
|               name: entry.name, | ||||
|               required: entry.required, | ||||
|               selector: { select: { options: entry.options } }, | ||||
|             } | ||||
|           : entry.type === "string" | ||||
|             ? entry.multiple | ||||
|               ? { | ||||
|                   name: entry.name, | ||||
|                   required: entry.required, | ||||
|                   selector: { | ||||
|                     select: { options: [], multiple: true, custom_value: true }, | ||||
|                   }, | ||||
|                 } | ||||
|               : { | ||||
|                   name: entry.name, | ||||
|                   required: entry.required, | ||||
|                   selector: { | ||||
|                     text: { | ||||
|                       type: entry.format | ||||
|                         ? entry.format | ||||
|                         : MASKED_FIELDS.includes(entry.name) | ||||
|                           ? "password" | ||||
|                           : "text", | ||||
|                     }, | ||||
|                   }, | ||||
|                 } | ||||
|             : entry.type === "boolean" | ||||
|               ? { | ||||
|                   name: entry.name, | ||||
|                   required: entry.required, | ||||
|                   selector: { boolean: {} }, | ||||
|                 } | ||||
|               : entry.type === "schema" | ||||
|                 ? { | ||||
|                     name: entry.name, | ||||
|                     required: entry.required, | ||||
|                     selector: { object: {} }, | ||||
|                   } | ||||
|                 : entry.type === "float" || entry.type === "integer" | ||||
|                   ? { | ||||
|                       name: entry.name, | ||||
|                       required: entry.required, | ||||
|                       selector: { | ||||
|                         number: { | ||||
|                           mode: "box", | ||||
|                           step: entry.type === "float" ? "any" : undefined, | ||||
|                         }, | ||||
|                       }, | ||||
|                     } | ||||
|                   : entry | ||||
|       ) | ||||
|   ); | ||||
|  | ||||
|   private _convertSchemaElements( | ||||
|     schema: readonly HaFormSchema[] | ||||
|   ): HaFormSchema[] { | ||||
|     return schema.map((entry) => this._convertSchemaElement(entry)); | ||||
|   } | ||||
|  | ||||
|   private _convertSchemaElement(entry: any): HaFormSchema { | ||||
|     if (entry.type === "schema" && !entry.multiple) { | ||||
|       return { | ||||
|         name: entry.name, | ||||
|         type: "expandable", | ||||
|         required: entry.required, | ||||
|         schema: this._convertSchemaElements(entry.schema), | ||||
|       }; | ||||
|     } | ||||
|     const selector = this._convertSchemaElementToSelector(entry, false); | ||||
|     if (selector) { | ||||
|       return { | ||||
|         name: entry.name, | ||||
|         required: entry.required, | ||||
|         selector, | ||||
|       }; | ||||
|     } | ||||
|     return entry; | ||||
|   } | ||||
|  | ||||
|   private _convertSchemaElementToSelector( | ||||
|     entry: any, | ||||
|     force: boolean | ||||
|   ): Selector | null { | ||||
|     if (entry.type === "select") { | ||||
|       return { select: { options: entry.options } }; | ||||
|     } | ||||
|     if (entry.type === "string") { | ||||
|       return entry.multiple | ||||
|         ? { select: { options: [], multiple: true, custom_value: true } } | ||||
|         : { | ||||
|             text: { | ||||
|               type: entry.format | ||||
|                 ? entry.format | ||||
|                 : MASKED_FIELDS.includes(entry.name) | ||||
|                   ? "password" | ||||
|                   : "text", | ||||
|             }, | ||||
|           }; | ||||
|     } | ||||
|     if (entry.type === "boolean") { | ||||
|       return { boolean: {} }; | ||||
|     } | ||||
|     if (entry.type === "schema") { | ||||
|       const fields: NonNullable<ObjectSelector["object"]>["fields"] = {}; | ||||
|       for (const child_entry of entry.schema) { | ||||
|         fields[child_entry.name] = { | ||||
|           required: child_entry.required, | ||||
|           selector: this._convertSchemaElementToSelector(child_entry, true)!, | ||||
|         }; | ||||
|       } | ||||
|       return { | ||||
|         object: { | ||||
|           multiple: entry.multiple, | ||||
|           fields, | ||||
|         }, | ||||
|       }; | ||||
|     } | ||||
|     if (entry.type === "float" || entry.type === "integer") { | ||||
|       return { | ||||
|         number: { | ||||
|           mode: "box", | ||||
|           step: entry.type === "float" ? "any" : undefined, | ||||
|         }, | ||||
|       }; | ||||
|     } | ||||
|     if (force) { | ||||
|       return { object: {} }; | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
|  | ||||
|   private _filteredSchema = memoizeOne( | ||||
|   private _filteredShchema = memoizeOne( | ||||
|     (options: Record<string, unknown>, schema: HaFormSchema[]) => | ||||
|       schema.filter((entry) => entry.name in options || entry.required) | ||||
|   ); | ||||
| @@ -212,7 +161,7 @@ class HassioAddonConfig extends LitElement { | ||||
|       showForm && | ||||
|       JSON.stringify(this.addon.schema) !== | ||||
|         JSON.stringify( | ||||
|           this._filteredSchema(this.addon.options, this.addon.schema!) | ||||
|           this._filteredShchema(this.addon.options, this.addon.schema!) | ||||
|         ); | ||||
|     return html` | ||||
|       <h1>${this.addon.name}</h1> | ||||
| @@ -250,7 +199,6 @@ class HassioAddonConfig extends LitElement { | ||||
|         <div class="card-content"> | ||||
|           ${showForm | ||||
|             ? html`<ha-form | ||||
|                 .hass=${this.hass} | ||||
|                 .disabled=${this.disabled} | ||||
|                 .data=${this._options!} | ||||
|                 @value-changed=${this._configChanged} | ||||
| @@ -259,7 +207,7 @@ class HassioAddonConfig extends LitElement { | ||||
|                 .schema=${this._convertSchema( | ||||
|                   this._showOptional | ||||
|                     ? this.addon.schema! | ||||
|                     : this._filteredSchema( | ||||
|                     : this._filteredShchema( | ||||
|                         this.addon.options, | ||||
|                         this.addon.schema! | ||||
|                       ) | ||||
| @@ -482,7 +430,7 @@ class HassioAddonConfig extends LitElement { | ||||
|           font-family: var(--ha-card-header-font-family, inherit); | ||||
|           font-size: var(--ha-card-header-font-size, var(--ha-font-size-2xl)); | ||||
|           letter-spacing: -0.012em; | ||||
|           line-height: var(--ha-line-height-expanded); | ||||
|           line-height: 48px; | ||||
|           padding: 12px 16px 16px; | ||||
|           display: block; | ||||
|           margin-block: 0px; | ||||
|   | ||||
| @@ -99,8 +99,7 @@ class HassioAddonNetwork extends LitElement { | ||||
|           : nothing} | ||||
|         <div class="card-actions"> | ||||
|           <ha-progress-button | ||||
|             variant="danger" | ||||
|             appearance="plain" | ||||
|             class="warning" | ||||
|             .disabled=${this.disabled} | ||||
|             @click=${this._resetTapped} | ||||
|           > | ||||
|   | ||||
| @@ -25,7 +25,6 @@ import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { LitElement, css, html, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import { ifDefined } from "lit/directives/if-defined"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| @@ -188,13 +187,12 @@ class HassioAddonInfo extends LitElement { | ||||
|                 "addon.dashboard.protection_mode.content" | ||||
|               )} | ||||
|               <ha-button | ||||
|                 variant="danger" | ||||
|                 slot="action" | ||||
|                 @click=${this._protectionToggled} | ||||
|               > | ||||
|                 ${this.supervisor.localize( | ||||
|                 .label=${this.supervisor.localize( | ||||
|                   "addon.dashboard.protection_mode.enable" | ||||
|                 )} | ||||
|                 @click=${this._protectionToggled} | ||||
|               > | ||||
|               </ha-button> | ||||
|             </ha-alert> | ||||
|           ` | ||||
| @@ -694,16 +692,14 @@ class HassioAddonInfo extends LitElement { | ||||
|               ? this._computeIsRunning | ||||
|                 ? html` | ||||
|                     <ha-progress-button | ||||
|                       variant="danger" | ||||
|                       appearance="plain" | ||||
|                       class="warning" | ||||
|                       @click=${this._stopClicked} | ||||
|                       .disabled=${systemManaged && !this.controlEnabled} | ||||
|                     > | ||||
|                       ${this.supervisor.localize("addon.dashboard.stop")} | ||||
|                     </ha-progress-button> | ||||
|                     <ha-progress-button | ||||
|                       variant="danger" | ||||
|                       appearance="plain" | ||||
|                       class="warning" | ||||
|                       @click=${this._restartClicked} | ||||
|                     > | ||||
|                       ${this.supervisor.localize("addon.dashboard.restart")} | ||||
| @@ -713,60 +709,10 @@ class HassioAddonInfo extends LitElement { | ||||
|                     <ha-progress-button | ||||
|                       @click=${this._startClicked} | ||||
|                       .progress=${this.addon.state === "startup"} | ||||
|                       appearance="plain" | ||||
|                     > | ||||
|                       ${this.supervisor.localize("addon.dashboard.start")} | ||||
|                     </ha-progress-button> | ||||
|                   ` | ||||
|               : nothing} | ||||
|           </div> | ||||
|           <div> | ||||
|             ${this.addon.version | ||||
|               ? html` | ||||
|                   <ha-progress-button | ||||
|                     variant="danger" | ||||
|                     appearance="plain" | ||||
|                     @click=${this._uninstallClicked} | ||||
|                     .disabled=${systemManaged && !this.controlEnabled} | ||||
|                   > | ||||
|                     ${this.supervisor.localize("addon.dashboard.uninstall")} | ||||
|                   </ha-progress-button> | ||||
|                   ${this.addon.build | ||||
|                     ? html` | ||||
|                         <ha-progress-button | ||||
|                           variant="danger" | ||||
|                           appearance="plain" | ||||
|                           @click=${this._rebuildClicked} | ||||
|                         > | ||||
|                           ${this.supervisor.localize("addon.dashboard.rebuild")} | ||||
|                         </ha-progress-button> | ||||
|                       ` | ||||
|                     : nothing} | ||||
|                   ${this._computeShowWebUI || this._computeShowIngressUI | ||||
|                     ? html` | ||||
|                         <ha-button | ||||
|                           href=${ifDefined( | ||||
|                             !this._computeShowIngressUI | ||||
|                               ? this._pathWebui! | ||||
|                               : nothing | ||||
|                           )} | ||||
|                           target=${ifDefined( | ||||
|                             !this._computeShowIngressUI ? "_blank" : nothing | ||||
|                           )} | ||||
|                           rel=${ifDefined( | ||||
|                             !this._computeShowIngressUI ? "noopener" : nothing | ||||
|                           )} | ||||
|                           @click=${!this._computeShowWebUI | ||||
|                             ? this._openIngress | ||||
|                             : undefined} | ||||
|                         > | ||||
|                           ${this.supervisor.localize( | ||||
|                             "addon.dashboard.open_web_ui" | ||||
|                           )} | ||||
|                         </ha-button> | ||||
|                       ` | ||||
|                     : nothing} | ||||
|                 ` | ||||
|               : html` | ||||
|                   <ha-progress-button | ||||
|                     .disabled=${!this.addon.available} | ||||
| @@ -776,12 +722,58 @@ class HassioAddonInfo extends LitElement { | ||||
|                   </ha-progress-button> | ||||
|                 `} | ||||
|           </div> | ||||
|           <div> | ||||
|             ${this.addon.version | ||||
|               ? html` ${this._computeShowWebUI | ||||
|                     ? html` | ||||
|                         <a | ||||
|                           href=${this._pathWebui!} | ||||
|                           tabindex="-1" | ||||
|                           target="_blank" | ||||
|                           rel="noopener" | ||||
|                         > | ||||
|                           <ha-button> | ||||
|                             ${this.supervisor.localize( | ||||
|                               "addon.dashboard.open_web_ui" | ||||
|                             )} | ||||
|                           </ha-button> | ||||
|                         </a> | ||||
|                       ` | ||||
|                     : nothing} | ||||
|                   ${this._computeShowIngressUI | ||||
|                     ? html` | ||||
|                         <ha-button @click=${this._openIngress}> | ||||
|                           ${this.supervisor.localize( | ||||
|                             "addon.dashboard.open_web_ui" | ||||
|                           )} | ||||
|                         </ha-button> | ||||
|                       ` | ||||
|                     : nothing} | ||||
|                   <ha-progress-button | ||||
|                     class="warning" | ||||
|                     @click=${this._uninstallClicked} | ||||
|                     .disabled=${systemManaged && !this.controlEnabled} | ||||
|                   > | ||||
|                     ${this.supervisor.localize("addon.dashboard.uninstall")} | ||||
|                   </ha-progress-button> | ||||
|                   ${this.addon.build | ||||
|                     ? html` | ||||
|                         <ha-progress-button | ||||
|                           class="warning" | ||||
|                           @click=${this._rebuildClicked} | ||||
|                         > | ||||
|                           ${this.supervisor.localize("addon.dashboard.rebuild")} | ||||
|                         </ha-progress-button> | ||||
|                       ` | ||||
|                     : nothing}` | ||||
|               : nothing} | ||||
|           </div> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|  | ||||
|       ${this.addon.long_description | ||||
|         ? html` | ||||
|             <ha-card class="long-description" outlined> | ||||
|             <ha-card outlined> | ||||
|               <div class="card-content"> | ||||
|                 <ha-markdown | ||||
|                   .content=${this.addon.long_description} | ||||
| @@ -1154,17 +1146,15 @@ class HassioAddonInfo extends LitElement { | ||||
|           ), | ||||
|           dismissText: this.supervisor.localize("common.cancel"), | ||||
|         }); | ||||
|         button.actionError(); | ||||
|         button.progress = false; | ||||
|         return; | ||||
|       } | ||||
|     } catch (err: any) { | ||||
|       button.actionError(); | ||||
|       button.progress = false; | ||||
|       showAlertDialog(this, { | ||||
|         title: "Failed to validate addon configuration", | ||||
|         text: extractApiErrorMessage(err), | ||||
|       }); | ||||
|       button.progress = false; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
| @@ -1178,15 +1168,11 @@ class HassioAddonInfo extends LitElement { | ||||
|       }; | ||||
|       fireEvent(this, "hass-api-called", eventdata); | ||||
|     } catch (err: any) { | ||||
|       button.actionError(); | ||||
|       button.progress = false; | ||||
|       showAlertDialog(this, { | ||||
|         title: this.supervisor.localize("addon.dashboard.action_error.start"), | ||||
|         text: extractApiErrorMessage(err), | ||||
|       }); | ||||
|       return; | ||||
|     } | ||||
|     button.actionSuccess(); | ||||
|     button.progress = false; | ||||
|   } | ||||
|  | ||||
| @@ -1242,7 +1228,6 @@ class HassioAddonInfo extends LitElement { | ||||
|         path: "uninstall", | ||||
|       }; | ||||
|       fireEvent(this, "hass-api-called", eventdata); | ||||
|       button.actionSuccess(); | ||||
|     } catch (err: any) { | ||||
|       showAlertDialog(this, { | ||||
|         title: this.supervisor.localize( | ||||
| @@ -1250,7 +1235,6 @@ class HassioAddonInfo extends LitElement { | ||||
|         ), | ||||
|         text: extractApiErrorMessage(err), | ||||
|       }); | ||||
|       button.actionError(); | ||||
|     } | ||||
|     button.progress = false; | ||||
|   } | ||||
| @@ -1333,9 +1317,6 @@ class HassioAddonInfo extends LitElement { | ||||
|         .description a { | ||||
|           color: var(--primary-color); | ||||
|         } | ||||
|         .long-description { | ||||
|           direction: ltr; | ||||
|         } | ||||
|         ha-assist-chip { | ||||
|           --md-sys-color-primary: var(--text-primary-color); | ||||
|           --md-sys-color-on-surface: var(--text-primary-color); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { LitElement, css, html, nothing } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { ActionDetail } from "@material/mwc-list"; | ||||
|  | ||||
| import { mdiBackupRestore, mdiDelete, mdiDotsVertical, mdiPlus } from "@mdi/js"; | ||||
| @@ -16,7 +17,6 @@ import type { | ||||
| } from "../../../src/components/data-table/ha-data-table"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-fab"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-list-item"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| @@ -241,13 +241,12 @@ export class HassioBackups extends LitElement { | ||||
|               <div class="header-btns"> | ||||
|                 ${!this.narrow | ||||
|                   ? html` | ||||
|                       <ha-button | ||||
|                         appearance="plain" | ||||
|                         variant="danger" | ||||
|                       <mwc-button | ||||
|                         @click=${this._deleteSelected} | ||||
|                         class="warning" | ||||
|                       > | ||||
|                         ${this.supervisor.localize("backup.delete_selected")} | ||||
|                       </ha-button> | ||||
|                       </mwc-button> | ||||
|                     ` | ||||
|                   : html` | ||||
|                       <ha-icon-button | ||||
| @@ -409,7 +408,7 @@ export class HassioBackups extends LitElement { | ||||
|           margin-inline-end: -12px; | ||||
|           margin-inline-start: initial; | ||||
|         } | ||||
|         .header-btns > ha-button, | ||||
|         .header-btns > mwc-button, | ||||
|         .header-btns > ha-icon-button { | ||||
|           margin: 8px; | ||||
|         } | ||||
|   | ||||
| @@ -101,7 +101,7 @@ class HassioCardContent extends LitElement { | ||||
|       overflow: hidden; | ||||
|       position: relative; | ||||
|       height: 2.4em; | ||||
|       line-height: var(--ha-line-height-condensed); | ||||
|       line-height: 1.2em; | ||||
|     } | ||||
|     .icon_image img { | ||||
|       max-height: 40px; | ||||
|   | ||||
| @@ -132,9 +132,9 @@ class HassioDashboard extends LitElement { | ||||
|         } | ||||
|         ha-fab.non-tabs { | ||||
|           position: fixed; | ||||
|           right: calc(16px + var(--safe-area-inset-right)); | ||||
|           bottom: calc(16px + var(--safe-area-inset-bottom)); | ||||
|           inset-inline-end: calc(16px + var(--safe-area-inset-right)); | ||||
|           right: calc(16px + env(safe-area-inset-right)); | ||||
|           bottom: calc(16px + env(safe-area-inset-bottom)); | ||||
|           inset-inline-end: calc(16px + env(safe-area-inset-right)); | ||||
|           inset-inline-start: initial; | ||||
|           z-index: 1; | ||||
|         } | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import type { HassioHassOSInfo } from "../../../src/data/hassio/host"; | ||||
| @@ -108,9 +109,10 @@ export class HassioUpdate extends LitElement { | ||||
|           </ha-settings-row> | ||||
|         </div> | ||||
|         <div class="card-actions"> | ||||
|           <ha-button appearance="plain" href="/hassio/update-available/${key}"> | ||||
|             ${this.supervisor.localize("common.show")} | ||||
|           </ha-button> | ||||
|           <a href="/hassio/update-available/${key}"> | ||||
|             <mwc-button .label=${this.supervisor.localize("common.show")}> | ||||
|             </mwc-button> | ||||
|           </a> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   | ||||
| @@ -1,10 +1,10 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| @@ -77,21 +77,20 @@ class HassioBackupLocationDialog extends LitElement { | ||||
|           @value-changed=${this._valueChanged} | ||||
|           dialogInitialFocus | ||||
|         ></ha-form> | ||||
|         <ha-button | ||||
|           appearance="plain" | ||||
|         <mwc-button | ||||
|           slot="secondaryAction" | ||||
|           @click=${this.closeDialog} | ||||
|           dialogInitialFocus | ||||
|         > | ||||
|           ${this._dialogParams.supervisor.localize("common.cancel")} | ||||
|         </ha-button> | ||||
|         <ha-button | ||||
|         </mwc-button> | ||||
|         <mwc-button | ||||
|           .disabled=${this._waiting || !this._data} | ||||
|           slot="primaryAction" | ||||
|           @click=${this._changeMount} | ||||
|         > | ||||
|           ${this._dialogParams.supervisor.localize("common.save")} | ||||
|         </ha-button> | ||||
|         </mwc-button> | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -8,6 +8,7 @@ import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { stopPropagation } from "../../../../src/common/dom/stop_propagation"; | ||||
| import { slugify } from "../../../../src/common/string/slugify"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-button-menu"; | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-spinner"; | ||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||
| import { | ||||
| @@ -68,20 +69,16 @@ class HassioCreateBackupDialog extends LitElement { | ||||
|         ${this._error | ||||
|           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||
|           : ""} | ||||
|         <ha-button | ||||
|           appearance="plain" | ||||
|           slot="secondaryAction" | ||||
|           @click=${this.closeDialog} | ||||
|         > | ||||
|         <mwc-button slot="secondaryAction" @click=${this.closeDialog}> | ||||
|           ${this._dialogParams.supervisor.localize("common.close")} | ||||
|         </ha-button> | ||||
|         <ha-button | ||||
|         </mwc-button> | ||||
|         <mwc-button | ||||
|           .disabled=${this._creatingBackup} | ||||
|           slot="primaryAction" | ||||
|           @click=${this._createBackup} | ||||
|         > | ||||
|           ${this._dialogParams.supervisor.localize("backup.create")} | ||||
|         </ha-button> | ||||
|         </mwc-button> | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -4,7 +4,6 @@ import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
| import "../../../../src/components/ha-select"; | ||||
| import "../../../../src/components/ha-spinner"; | ||||
| @@ -21,8 +20,8 @@ import type { HomeAssistant } from "../../../../src/types"; | ||||
| import type { HassioDatatiskDialogParams } from "./show-dialog-hassio-datadisk"; | ||||
|  | ||||
| const calculateMoveTime = memoizeOne((supervisor: Supervisor): number => { | ||||
|   // Assume a speed of 30 MB/s. | ||||
|   const moveTime = (supervisor.host.disk_used * 1000) / 60 / 30; | ||||
|   const speed = supervisor.host.disk_life_time !== "" ? 30 : 10; | ||||
|   const moveTime = (supervisor.host.disk_used * 1000) / 60 / speed; | ||||
|   const rebootTime = (supervisor.host.startup_time * 4) / 60; | ||||
|   return Math.ceil((moveTime + rebootTime) / 10) * 10; | ||||
| }); | ||||
| @@ -110,18 +109,17 @@ class HassioDatadiskDialog extends LitElement { | ||||
|                       "dialog.datadisk_move.no_devices" | ||||
|                     )} | ||||
|  | ||||
|               <ha-button | ||||
|                 appearance="plain" | ||||
|                 slot="primaryAction" | ||||
|               <mwc-button | ||||
|                 slot="secondaryAction" | ||||
|                 @click=${this.closeDialog} | ||||
|                 dialogInitialFocus | ||||
|               > | ||||
|                 ${this.dialogParams.supervisor.localize( | ||||
|                   "dialog.datadisk_move.cancel" | ||||
|                 )} | ||||
|               </ha-button> | ||||
|               </mwc-button> | ||||
|  | ||||
|               <ha-button | ||||
|               <mwc-button | ||||
|                 .disabled=${!this.selectedDevice} | ||||
|                 slot="primaryAction" | ||||
|                 @click=${this._moveDatadisk} | ||||
| @@ -129,7 +127,7 @@ class HassioDatadiskDialog extends LitElement { | ||||
|                 ${this.dialogParams.supervisor.localize( | ||||
|                   "dialog.datadisk_move.move" | ||||
|                 )} | ||||
|               </ha-button>`} | ||||
|               </mwc-button>`} | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { mdiClose } from "@mdi/js"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| @@ -5,7 +6,6 @@ import { customElement, property, state } from "lit/decorators"; | ||||
| import { cache } from "lit/directives/cache"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-expansion-panel"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| @@ -15,8 +15,7 @@ import "../../../../src/components/ha-list"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
| import "../../../../src/components/ha-password-field"; | ||||
| import "../../../../src/components/ha-radio"; | ||||
| import "../../../../src/components/ha-tab-group"; | ||||
| import "../../../../src/components/ha-tab-group-tab"; | ||||
| import "../../../../src/components/ha-spinner"; | ||||
| import "../../../../src/components/ha-textfield"; | ||||
| import type { HaTextField } from "../../../../src/components/ha-textfield"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| @@ -38,6 +37,7 @@ import type { HassDialog } from "../../../../src/dialogs/make-dialog-manager"; | ||||
| import { haStyleDialog } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import type { HassioNetworkDialogParams } from "./show-dialog-network"; | ||||
| import "../../../../src/components/sl-tab-group"; | ||||
|  | ||||
| const IP_VERSIONS = ["ipv4", "ipv6"]; | ||||
|  | ||||
| @@ -115,19 +115,19 @@ export class DialogHassioNetwork | ||||
|             ></ha-icon-button> | ||||
|           </ha-header-bar> | ||||
|           ${this._interfaces.length > 1 | ||||
|             ? html`<ha-tab-group @wa-tab-show=${this._handleTabActivated} | ||||
|             ? html`<sl-tab-group @sl-tab-show=${this._handleTabActivated} | ||||
|                 >${this._interfaces.map( | ||||
|                   (device, index) => | ||||
|                     html`<ha-tab-group-tab | ||||
|                     html`<sl-tab | ||||
|                       slot="nav" | ||||
|                       .id=${device.interface} | ||||
|                       .panel=${index.toString()} | ||||
|                       .active=${this._curTabIndex === index} | ||||
|                     > | ||||
|                       ${device.interface} | ||||
|                     </ha-tab-group-tab>` | ||||
|                     </sl-tab>` | ||||
|                 )} | ||||
|               </ha-tab-group>` | ||||
|               </sl-tab-group>` | ||||
|             : ""} | ||||
|         </div> | ||||
|         ${cache(this._renderTab())} | ||||
| @@ -154,16 +154,16 @@ export class DialogHassioNetwork | ||||
|                       )} | ||||
|                     </p>` | ||||
|                   : ""} | ||||
|                 <ha-button | ||||
|                   appearance="plain" | ||||
|                   size="small" | ||||
|                 <mwc-button | ||||
|                   class="scan" | ||||
|                   @click=${this._scanForAP} | ||||
|                   .disabled=${this._scanning} | ||||
|                   .loading=${this._scanning} | ||||
|                 > | ||||
|                   ${this.supervisor.localize("dialog.network.scan_ap")} | ||||
|                 </ha-button> | ||||
|                   ${this._scanning | ||||
|                     ? html`<ha-spinner aria-label="Scanning" size="small"> | ||||
|                       </ha-spinner>` | ||||
|                     : this.supervisor.localize("dialog.network.scan_ap")} | ||||
|                 </mwc-button> | ||||
|                 ${this._accessPoints && | ||||
|                 this._accessPoints.accesspoints && | ||||
|                 this._accessPoints.accesspoints.length !== 0 | ||||
| @@ -270,16 +270,16 @@ export class DialogHassioNetwork | ||||
|           : ""} | ||||
|       </div> | ||||
|       <div class="buttons"> | ||||
|         <ha-button @click=${this.closeDialog} appearance="plain"> | ||||
|           ${this.supervisor.localize("common.cancel")} | ||||
|         </ha-button> | ||||
|         <ha-button | ||||
|           @click=${this._updateNetwork} | ||||
|           .disabled=${!this._dirty} | ||||
|           .loading=${this._processing} | ||||
|         <mwc-button | ||||
|           .label=${this.supervisor.localize("common.cancel")} | ||||
|           @click=${this.closeDialog} | ||||
|         > | ||||
|           ${this.supervisor.localize("common.save")} | ||||
|         </ha-button> | ||||
|         </mwc-button> | ||||
|         <mwc-button @click=${this._updateNetwork} .disabled=${!this._dirty}> | ||||
|           ${this._processing | ||||
|             ? html`<ha-spinner size="small"> </ha-spinner>` | ||||
|             : this.supervisor.localize("common.save")} | ||||
|         </mwc-button> | ||||
|       </div>`; | ||||
|   } | ||||
|  | ||||
| @@ -584,7 +584,11 @@ export class DialogHassioNetwork | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         ha-button.scan { | ||||
|         mwc-button.warning { | ||||
|           --mdc-theme-primary: var(--error-color); | ||||
|         } | ||||
|  | ||||
|         mwc-button.scan { | ||||
|           margin-left: 8px; | ||||
|           margin-inline-start: 8px; | ||||
|           margin-inline-end: initial; | ||||
| @@ -605,8 +609,8 @@ export class DialogHassioNetwork | ||||
|             var(--mdc-dialog-scroll-divider-color, rgba(0, 0, 0, 0.12)); | ||||
|           display: flex; | ||||
|           justify-content: space-between; | ||||
|           padding: 16px; | ||||
|           padding-bottom: max(var(--safe-area-inset-bottom), 16px); | ||||
|           padding: 8px; | ||||
|           padding-bottom: max(env(safe-area-inset-bottom), 8px); | ||||
|           background-color: var(--mdc-theme-surface, #fff); | ||||
|         } | ||||
|         .warning { | ||||
| @@ -628,10 +632,10 @@ export class DialogHassioNetwork | ||||
|           --mdc-list-side-padding: 10px; | ||||
|         } | ||||
|  | ||||
|         ha-tab-group-tab { | ||||
|         sl-tab { | ||||
|           flex: 1; | ||||
|         } | ||||
|         ha-tab-group-tab::part(base) { | ||||
|         sl-tab::part(base) { | ||||
|           width: 100%; | ||||
|           justify-content: center; | ||||
|         } | ||||
|   | ||||
| @@ -1,14 +1,13 @@ | ||||
| import { mdiDelete, mdiPlus } from "@mdi/js"; | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { mdiDelete } from "@mdi/js"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| import { | ||||
|   addHassioDockerRegistry, | ||||
| @@ -85,19 +84,16 @@ class HassioRegistriesDialog extends LitElement { | ||||
|                 dialogInitialFocus | ||||
|               ></ha-form> | ||||
|               <div class="action"> | ||||
|                 <ha-button | ||||
|                 <mwc-button | ||||
|                   ?disabled=${Boolean( | ||||
|                     !this._input.registry || | ||||
|                       !this._input.username || | ||||
|                       !this._input.password | ||||
|                   )} | ||||
|                   @click=${this._addNewRegistry} | ||||
|                   appearance="filled" | ||||
|                   size="small" | ||||
|                 > | ||||
|                   <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||
|                   ${this.supervisor.localize("dialog.registries.add_registry")} | ||||
|                 </ha-button> | ||||
|                 </mwc-button> | ||||
|               </div> | ||||
|             ` | ||||
|           : html`${this._registries?.length | ||||
| @@ -130,17 +126,11 @@ class HassioRegistriesDialog extends LitElement { | ||||
|                     </ha-alert> | ||||
|                   `} | ||||
|               <div class="action"> | ||||
|                 <ha-button | ||||
|                   @click=${this._addRegistry} | ||||
|                   dialogInitialFocus | ||||
|                   appearance="filled" | ||||
|                   size="small" | ||||
|                 > | ||||
|                   <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||
|                 <mwc-button @click=${this._addRegistry} dialogInitialFocus> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "dialog.registries.add_new_registry" | ||||
|                   )} | ||||
|                 </ha-button> | ||||
|                 </mwc-button> | ||||
|               </div> `} | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { mdiDelete, mdiDeleteOff, mdiPlus } from "@mdi/js"; | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { mdiDelete, mdiDeleteOff } from "@mdi/js"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| @@ -6,15 +7,10 @@ import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { caseInsensitiveStringCompare } from "../../../../src/common/string/compare"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button"; | ||||
| import "../../../../src/components/ha-tooltip"; | ||||
| import "../../../../src/components/ha-spinner"; | ||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-md-list"; | ||||
| import "../../../../src/components/ha-md-list-item"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-textfield"; | ||||
| import type { HaTextField } from "../../../../src/components/ha-textfield"; | ||||
| import "../../../../src/components/ha-tooltip"; | ||||
| import type { | ||||
|   HassioAddonInfo, | ||||
|   HassioAddonRepository, | ||||
| @@ -28,6 +24,10 @@ import { | ||||
| import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import type { HassioRepositoryDialogParams } from "./show-dialog-repositories"; | ||||
| import type { HaTextField } from "../../../../src/components/ha-textfield"; | ||||
| import "../../../../src/components/ha-textfield"; | ||||
| import "../../../../src/components/ha-md-list"; | ||||
| import "../../../../src/components/ha-md-list-item"; | ||||
|  | ||||
| @customElement("dialog-hassio-repositories") | ||||
| class HassioRepositoriesDialog extends LitElement { | ||||
| @@ -119,27 +119,26 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|                         <div>${repo.url}</div> | ||||
|                       </div> | ||||
|                       <ha-tooltip | ||||
|                         .for="icon-button-${repo.slug}" | ||||
|                         class="delete" | ||||
|                         slot="end" | ||||
|                       > | ||||
|                         ${this._dialogParams!.supervisor.localize( | ||||
|                         .content=${this._dialogParams!.supervisor.localize( | ||||
|                           usedRepositories.includes(repo.slug) | ||||
|                             ? "dialog.repositories.used" | ||||
|                             : "dialog.repositories.remove" | ||||
|                         )} | ||||
|                       > | ||||
|                         <div> | ||||
|                           <ha-icon-button | ||||
|                             .disabled=${usedRepositories.includes(repo.slug)} | ||||
|                             .slug=${repo.slug} | ||||
|                             .path=${usedRepositories.includes(repo.slug) | ||||
|                               ? mdiDeleteOff | ||||
|                               : mdiDelete} | ||||
|                             @click=${this._removeRepository} | ||||
|                           > | ||||
|                           </ha-icon-button> | ||||
|                         </div> | ||||
|                       </ha-tooltip> | ||||
|                       <div .id="icon-button-${repo.slug}"> | ||||
|                         <ha-icon-button | ||||
|                           .disabled=${usedRepositories.includes(repo.slug)} | ||||
|                           .slug=${repo.slug} | ||||
|                           .path=${usedRepositories.includes(repo.slug) | ||||
|                             ? mdiDeleteOff | ||||
|                             : mdiDelete} | ||||
|                           @click=${this._removeRepository} | ||||
|                         > | ||||
|                         </ha-icon-button> | ||||
|                       </div> | ||||
|                     </ha-md-list-item> | ||||
|                   ` | ||||
|                 ) | ||||
| @@ -160,22 +159,18 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|               @keydown=${this._handleKeyAdd} | ||||
|               dialogInitialFocus | ||||
|             ></ha-textfield> | ||||
|             <ha-button | ||||
|               .loading=${this._processing} | ||||
|               @click=${this._addRepository} | ||||
|               appearance="filled" | ||||
|               size="small" | ||||
|             > | ||||
|               <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||
|               ${this._dialogParams!.supervisor.localize( | ||||
|                 "dialog.repositories.add" | ||||
|               )} | ||||
|             </ha-button> | ||||
|             <mwc-button @click=${this._addRepository}> | ||||
|               ${this._processing | ||||
|                 ? html`<ha-spinner size="small"></ha-spinner>` | ||||
|                 : this._dialogParams!.supervisor.localize( | ||||
|                     "dialog.repositories.add" | ||||
|                   )} | ||||
|             </mwc-button> | ||||
|           </div> | ||||
|         </div> | ||||
|         <ha-button slot="primaryAction" @click=${this.closeDialog}> | ||||
|         <mwc-button slot="primaryAction" @click=${this.closeDialog}> | ||||
|           ${this._dialogParams?.supervisor.localize("common.close")} | ||||
|         </ha-button> | ||||
|         </mwc-button> | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
| @@ -196,11 +191,16 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|           border-radius: 4px; | ||||
|           margin-top: 4px; | ||||
|         } | ||||
|         ha-button { | ||||
|         mwc-button { | ||||
|           margin-left: 8px; | ||||
|           margin-inline-start: 8px; | ||||
|           margin-inline-end: initial; | ||||
|         } | ||||
|         ha-spinner { | ||||
|           display: block; | ||||
|           margin: 32px; | ||||
|           text-align: center; | ||||
|         } | ||||
|         div.delete ha-icon-button { | ||||
|           color: var(--error-color); | ||||
|         } | ||||
| @@ -249,8 +249,6 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|       await addStoreRepository(this.hass, input.value); | ||||
|       await this._loadData(); | ||||
|  | ||||
|       fireEvent(this, "supervisor-collection-refresh", { collection: "store" }); | ||||
|  | ||||
|       input.value = ""; | ||||
|     } catch (err: any) { | ||||
|       this._error = extractApiErrorMessage(err); | ||||
| @@ -263,8 +261,6 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|     try { | ||||
|       await removeStoreRepository(this.hass, slug); | ||||
|       await this._loadData(); | ||||
|  | ||||
|       fireEvent(this, "supervisor-collection-refresh", { collection: "store" }); | ||||
|     } catch (err: any) { | ||||
|       this._error = extractApiErrorMessage(err); | ||||
|     } | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import type { PropertyValues, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { goBack, navigate } from "../../../src/common/navigate"; | ||||
| import { navigate } from "../../../src/common/navigate"; | ||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||
| import { nextRender } from "../../../src/common/util/render-status"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| @@ -193,7 +193,7 @@ class HassioIngressView extends LitElement { | ||||
|         title: addon.name, | ||||
|       }); | ||||
|       await nextRender(); | ||||
|       goBack(); | ||||
|       history.back(); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
| @@ -275,7 +275,7 @@ class HassioIngressView extends LitElement { | ||||
|         title: addon.name, | ||||
|       }); | ||||
|       await nextRender(); | ||||
|       goBack(); | ||||
|       history.back(); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
| @@ -354,7 +354,7 @@ class HassioIngressView extends LitElement { | ||||
|  | ||||
|     .main-title { | ||||
|       margin: var(--margin-title); | ||||
|       line-height: var(--ha-line-height-condensed); | ||||
|       line-height: 20px; | ||||
|       flex-grow: 1; | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import "@material/mwc-button"; | ||||
|  | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| @@ -69,12 +70,12 @@ class HassioCoreInfo extends LitElement { | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.core.update_available | ||||
|                 ? html` | ||||
|                     <ha-button | ||||
|                       appearance="plain" | ||||
|                       href="/hassio/update-available/core" | ||||
|                     > | ||||
|                       ${this.supervisor.localize("common.show")} | ||||
|                     </ha-button> | ||||
|                     <a href="/hassio/update-available/core"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -94,7 +95,7 @@ class HassioCoreInfo extends LitElement { | ||||
|         <div class="card-actions"> | ||||
|           <ha-progress-button | ||||
|             slot="primaryAction" | ||||
|             variant="danger" | ||||
|             class="warning" | ||||
|             @click=${this._coreRestart} | ||||
|             .title=${this.supervisor.localize("common.restart_name", { | ||||
|               name: "Core", | ||||
| @@ -187,6 +188,11 @@ class HassioCoreInfo extends LitElement { | ||||
|           white-space: normal; | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|  | ||||
|         .warning { | ||||
|           --mdc-theme-primary: var(--error-color); | ||||
|         } | ||||
|  | ||||
|         ha-button-menu { | ||||
|           color: var(--secondary-text-color); | ||||
|           --mdc-menu-min-width: 200px; | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
|  | ||||
| import { mdiDotsVertical } from "@mdi/js"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| @@ -6,11 +8,10 @@ import memoizeOne from "memoize-one"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-list-item"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
| @@ -76,28 +77,24 @@ class HassioHostInfo extends LitElement { | ||||
|                   <span slot="description"> | ||||
|                     ${this.supervisor.host.hostname} | ||||
|                   </span> | ||||
|                   <ha-button | ||||
|                   <mwc-button | ||||
|                     .label=${this.supervisor.localize("system.host.change")} | ||||
|                     @click=${this._changeHostnameClicked} | ||||
|                     appearance="plain" | ||||
|                     size="small" | ||||
|                   > | ||||
|                     ${this.supervisor.localize("system.host.change")} | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </ha-settings-row>` | ||||
|               : ""} | ||||
|             ${this.supervisor.host.features.includes("network") | ||||
|               ? html`<ha-settings-row> | ||||
|               ? html` <ha-settings-row> | ||||
|                   <span slot="heading"> | ||||
|                     ${this.supervisor.localize("system.host.ip_address")} | ||||
|                   </span> | ||||
|                   <span slot="description"> ${primaryIpAddress} </span> | ||||
|                   <ha-button | ||||
|                   <mwc-button | ||||
|                     .label=${this.supervisor.localize("system.host.change")} | ||||
|                     @click=${this._changeNetworkClicked} | ||||
|                     appearance="plain" | ||||
|                     size="small" | ||||
|                   > | ||||
|                     ${this.supervisor.localize("system.host.change")} | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </ha-settings-row>` | ||||
|               : ""} | ||||
|  | ||||
| @@ -111,13 +108,12 @@ class HassioHostInfo extends LitElement { | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.os.update_available | ||||
|                 ? html` | ||||
|                     <ha-button | ||||
|                       appearance="plain" | ||||
|                       size="small" | ||||
|                       href="/hassio/update-available/os" | ||||
|                     > | ||||
|                       ${this.supervisor.localize("common.show")} | ||||
|                     </ha-button> | ||||
|                     <a href="/hassio/update-available/os"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -143,12 +139,16 @@ class HassioHostInfo extends LitElement { | ||||
|               : ""} | ||||
|           </div> | ||||
|           <div> | ||||
|             ${this.supervisor.host.disk_life_time !== null | ||||
|             ${this.supervisor.host.disk_life_time !== "" && | ||||
|             this.supervisor.host.disk_life_time >= 10 | ||||
|               ? html` <ha-settings-row> | ||||
|                   <span slot="heading"> | ||||
|                     ${this.supervisor.localize("system.host.lifetime_used")} | ||||
|                     ${this.supervisor.localize( | ||||
|                       "system.host.emmc_lifetime_used" | ||||
|                     )} | ||||
|                   </span> | ||||
|                   <span slot="description"> | ||||
|                     ${this.supervisor.host.disk_life_time - 10} % - | ||||
|                     ${this.supervisor.host.disk_life_time} % | ||||
|                   </span> | ||||
|                 </ha-settings-row>` | ||||
| @@ -167,7 +167,7 @@ class HassioHostInfo extends LitElement { | ||||
|         <div class="card-actions"> | ||||
|           ${this.supervisor.host.features.includes("reboot") | ||||
|             ? html` | ||||
|                 <ha-progress-button variant="danger" @click=${this._hostReboot}> | ||||
|                 <ha-progress-button class="warning" @click=${this._hostReboot}> | ||||
|                   ${this.supervisor.localize("system.host.reboot_host")} | ||||
|                 </ha-progress-button> | ||||
|               ` | ||||
| @@ -175,7 +175,7 @@ class HassioHostInfo extends LitElement { | ||||
|           ${this.supervisor.host.features.includes("shutdown") | ||||
|             ? html` | ||||
|                 <ha-progress-button | ||||
|                   variant="danger" | ||||
|                   class="warning" | ||||
|                   @click=${this._hostShutdown} | ||||
|                 > | ||||
|                   ${this.supervisor.localize("system.host.shutdown_host")} | ||||
| @@ -431,6 +431,10 @@ class HassioHostInfo extends LitElement { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|  | ||||
|         .warning { | ||||
|           --mdc-theme-primary: var(--error-color); | ||||
|         } | ||||
|  | ||||
|         ha-button-menu { | ||||
|           color: var(--secondary-text-color); | ||||
|           --mdc-menu-min-width: 200px; | ||||
|   | ||||
| @@ -5,7 +5,6 @@ import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import "../../../src/components/ha-switch"; | ||||
| @@ -81,13 +80,12 @@ class HassioSupervisorInfo extends LitElement { | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.supervisor.update_available | ||||
|                 ? html` | ||||
|                     <ha-button | ||||
|                       appearance="plain" | ||||
|                       size="small" | ||||
|                       href="/hassio/update-available/supervisor" | ||||
|                     > | ||||
|                       ${this.supervisor.localize("common.show")} | ||||
|                     </ha-button> | ||||
|                     <a href="/hassio/update-available/supervisor"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -158,28 +156,24 @@ class HassioSupervisorInfo extends LitElement { | ||||
|                   ${this.supervisor.localize( | ||||
|                     "system.supervisor.unsupported_title" | ||||
|                   )} | ||||
|                   <ha-button | ||||
|                   <mwc-button | ||||
|                     slot="action" | ||||
|                     .label=${this.supervisor.localize("common.learn_more")} | ||||
|                     @click=${this._unsupportedDialog} | ||||
|                     variant="warning" | ||||
|                     size="small" | ||||
|                   > | ||||
|                     ${this.supervisor.localize("common.learn_more")} | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </ha-alert>`} | ||||
|             ${!this.supervisor.supervisor.healthy | ||||
|               ? html`<ha-alert alert-type="error"> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "system.supervisor.unhealthy_title" | ||||
|                   )} | ||||
|                   <ha-button | ||||
|                     variant="danger" | ||||
|                     size="small" | ||||
|                   <mwc-button | ||||
|                     slot="action" | ||||
|                     .label=${this.supervisor.localize("common.learn_more")} | ||||
|                     @click=${this._unhealthyDialog} | ||||
|                   > | ||||
|                     ${this.supervisor.localize("common.learn_more")} | ||||
|                   </ha-button> | ||||
|                   </mwc-button> | ||||
|                 </ha-alert>` | ||||
|               : ""} | ||||
|           </div> | ||||
| @@ -454,6 +448,9 @@ class HassioSupervisorInfo extends LitElement { | ||||
|           white-space: normal; | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         ha-alert mwc-button { | ||||
|           --mdc-theme-primary: var(--primary-text-color); | ||||
|         } | ||||
|         a { | ||||
|           text-decoration: none; | ||||
|         } | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
|  | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
|   | ||||
| @@ -208,16 +208,14 @@ class UpdateAvailableCard extends LitElement { | ||||
|               <div class="card-actions"> | ||||
|                 ${changelog | ||||
|                   ? html` | ||||
|                       <ha-button | ||||
|                         href=${changelog} | ||||
|                         target="_blank" | ||||
|                         rel="noreferrer" | ||||
|                         appearance="plain" | ||||
|                       > | ||||
|                         ${this.supervisor.localize( | ||||
|                           "update_available.open_release_notes" | ||||
|                         )} | ||||
|                       </ha-button> | ||||
|                       <a href=${changelog} target="_blank" rel="noreferrer"> | ||||
|                         <ha-button | ||||
|                           .label=${this.supervisor.localize( | ||||
|                             "update_available.open_release_notes" | ||||
|                           )} | ||||
|                         > | ||||
|                         </ha-button> | ||||
|                       </a> | ||||
|                     ` | ||||
|                   : nothing} | ||||
|                 <span></span> | ||||
|   | ||||
| @@ -2,7 +2,6 @@ import type { TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { goBack } from "../../../src/common/navigate"; | ||||
| import "../../../src/layouts/hass-subpage"; | ||||
| import type { HomeAssistant, Route } from "../../../src/types"; | ||||
| import "./update-available-card"; | ||||
| @@ -36,7 +35,7 @@ class UpdateAvailableDashboard extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private _updateComplete() { | ||||
|     goBack(); | ||||
|     history.back(); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   | ||||
| @@ -3,26 +3,26 @@ import { mdiArrowCollapseDown, mdiDownload } from "@mdi/js"; | ||||
| // eslint-disable-next-line import/extensions | ||||
| import { IntersectionController } from "@lit-labs/observers/intersection-controller.js"; | ||||
| import { LitElement, type PropertyValues, css, html, nothing } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import type { | ||||
|   LandingPageKeys, | ||||
|   LocalizeFunc, | ||||
| } from "../../../src/common/translations/localize"; | ||||
| import { waitForSeconds } from "../../../src/common/util/wait"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-ansi-to-html"; | ||||
| import type { HaAnsiToHtml } from "../../../src/components/ha-ansi-to-html"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import { fileDownload } from "../../../src/util/file_download"; | ||||
| import "../../../src/components/ha-ansi-to-html"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import type { HaAnsiToHtml } from "../../../src/components/ha-ansi-to-html"; | ||||
| import { | ||||
|   getObserverLogs, | ||||
|   downloadUrl as observerLogsDownloadUrl, | ||||
| } from "../data/observer"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { fileDownload } from "../../../src/util/file_download"; | ||||
| import { getSupervisorLogs, getSupervisorLogsFollow } from "../data/supervisor"; | ||||
| import { waitForSeconds } from "../../../src/common/util/wait"; | ||||
| import { ASSUME_CORE_START_SECONDS } from "../ha-landing-page"; | ||||
|  | ||||
| const ERROR_CHECK = /^[\d\s-:]+(ERROR|CRITICAL)(.*)/gm; | ||||
| @@ -64,7 +64,7 @@ class LandingPageLogs extends LitElement { | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <div class="actions"> | ||||
|         <ha-button appearance="plain" @click=${this._toggleLogDetails}> | ||||
|         <ha-button @click=${this._toggleLogDetails}> | ||||
|           ${this.localize(this._show ? "hide_details" : "show_details")} | ||||
|         </ha-button> | ||||
|         ${this._show | ||||
| @@ -81,11 +81,7 @@ class LandingPageLogs extends LitElement { | ||||
|               alert-type="error" | ||||
|               .title=${this.localize("logs.fetch_error")} | ||||
|             > | ||||
|               <ha-button | ||||
|                 size="small" | ||||
|                 variant="danger" | ||||
|                 @click=${this._startLogStream} | ||||
|               > | ||||
|               <ha-button @click=${this._startLogStream}> | ||||
|                 ${this.localize("logs.retry")} | ||||
|               </ha-button> | ||||
|             </ha-alert> | ||||
| @@ -108,13 +104,14 @@ class LandingPageLogs extends LitElement { | ||||
|               !this._scrolledToBottomController.value) || | ||||
|             false, | ||||
|         })}" | ||||
|         size="small" | ||||
|         appearance="filled" | ||||
|         @click=${this._scrollToBottom} | ||||
|       > | ||||
|         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="start"></ha-svg-icon> | ||||
|         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="icon"></ha-svg-icon> | ||||
|         ${this.localize("logs.scroll_down_button")} | ||||
|         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="end"></ha-svg-icon> | ||||
|         <ha-svg-icon | ||||
|           .path=${mdiArrowCollapseDown} | ||||
|           slot="trailingIcon" | ||||
|         ></ha-svg-icon> | ||||
|       </ha-button> | ||||
|     `; | ||||
|   } | ||||
| @@ -311,14 +308,21 @@ class LandingPageLogs extends LitElement { | ||||
|       } | ||||
|  | ||||
|       .new-logs-indicator { | ||||
|         --mdc-theme-primary: var(--text-primary-color); | ||||
|  | ||||
|         overflow: hidden; | ||||
|         position: absolute; | ||||
|         bottom: 4px; | ||||
|         left: 4px; | ||||
|         bottom: 0; | ||||
|         left: 0; | ||||
|         right: 0; | ||||
|         height: 0; | ||||
|         background-color: var(--primary-color); | ||||
|         border-radius: 8px; | ||||
|  | ||||
|         transition: height 0.4s ease-out; | ||||
|         display: flex; | ||||
|         justify-content: space-between; | ||||
|         align-items: center; | ||||
|       } | ||||
|  | ||||
|       .new-logs-indicator.visible { | ||||
|   | ||||
| @@ -67,7 +67,6 @@ class LandingPageNetwork extends LitElement { | ||||
|           ${ALTERNATIVE_DNS_SERVERS.map( | ||||
|             ({ translationKey }, key) => | ||||
|               html`<ha-button | ||||
|                 size="small" | ||||
|                 .index=${key} | ||||
|                 .disabled=${!dnsPrimaryInterfaceNameservers} | ||||
|                 @click=${this._setDns} | ||||
|   | ||||
| @@ -19,9 +19,8 @@ | ||||
|         height: auto; | ||||
|         padding: 32px 0; | ||||
|       } | ||||
|  | ||||
|       .content { | ||||
|         max-width: min(560px, calc(100vw - var(--safe-area-inset-right, 0px) - var(--safe-area-inset-left, 0px))); | ||||
|         max-width: 560px; | ||||
|         margin: 0 auto; | ||||
|         padding: 0 16px; | ||||
|         box-sizing: content-box; | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| export default { | ||||
|   "*.?(c|m){js,ts}": [ | ||||
|     "eslint --flag v10_config_lookup_from_file --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --fix", | ||||
|     "eslint --flag unstable_config_lookup_from_file --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --fix", | ||||
|     "prettier --cache --write", | ||||
|     "lit-analyzer --quiet", | ||||
|   ], | ||||
|   | ||||
							
								
								
									
										166
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @@ -8,8 +8,8 @@ | ||||
|   "version": "1.0.0", | ||||
|   "scripts": { | ||||
|     "build": "script/build_frontend", | ||||
|     "lint:eslint": "eslint --flag v10_config_lookup_from_file \"**/src/**/*.{js,ts,html}\" --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --ignore-pattern=.gitignore --max-warnings=0", | ||||
|     "format:eslint": "eslint --flag v10_config_lookup_from_file \"**/src/**/*.{js,ts,html}\" --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --ignore-pattern=.gitignore --fix", | ||||
|     "lint:eslint": "eslint --flag unstable_config_lookup_from_file \"**/src/**/*.{js,ts,html}\" --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --ignore-pattern=.gitignore --max-warnings=0", | ||||
|     "format:eslint": "eslint --flag unstable_config_lookup_from_file \"**/src/**/*.{js,ts,html}\" --cache --cache-strategy=content --cache-location=node_modules/.cache/eslint/.eslintcache --ignore-pattern=.gitignore --fix", | ||||
|     "lint:prettier": "prettier . --cache --check", | ||||
|     "format:prettier": "prettier . --cache --write", | ||||
|     "lint:types": "tsc", | ||||
| @@ -20,22 +20,23 @@ | ||||
|     "prepack": "pinst --disable", | ||||
|     "postpack": "pinst --enable", | ||||
|     "test": "vitest run --config test/vitest.config.ts", | ||||
|     "test:coverage": "vitest run --config test/vitest.config.ts --coverage" | ||||
|     "test:coverage": "vitest run --config test/vitest.config.ts --coverage", | ||||
|     "analyze": "cem analyze --litelement --globs \"src/components/ha-alert.ts\" --dev", | ||||
|     "doc": "gulp generate-component-docs" | ||||
|   }, | ||||
|   "author": "Paulus Schoutsen <Paulus@PaulusSchoutsen.nl> (http://paulusschoutsen.nl)", | ||||
|   "license": "Apache-2.0", | ||||
|   "type": "module", | ||||
|   "dependencies": { | ||||
|     "@babel/runtime": "7.28.4", | ||||
|     "@babel/runtime": "7.27.1", | ||||
|     "@braintree/sanitize-url": "7.1.1", | ||||
|     "@codemirror/autocomplete": "6.18.7", | ||||
|     "@codemirror/autocomplete": "6.18.6", | ||||
|     "@codemirror/commands": "6.8.1", | ||||
|     "@codemirror/language": "6.11.3", | ||||
|     "@codemirror/language": "6.11.0", | ||||
|     "@codemirror/legacy-modes": "6.5.1", | ||||
|     "@codemirror/search": "6.5.11", | ||||
|     "@codemirror/search": "6.5.10", | ||||
|     "@codemirror/state": "6.5.2", | ||||
|     "@codemirror/view": "6.38.2", | ||||
|     "@date-fns/tz": "1.4.1", | ||||
|     "@codemirror/view": "6.36.7", | ||||
|     "@egjs/hammerjs": "2.0.17", | ||||
|     "@formatjs/intl-datetimeformat": "6.18.0", | ||||
|     "@formatjs/intl-displaynames": "6.8.11", | ||||
| @@ -46,22 +47,22 @@ | ||||
|     "@formatjs/intl-numberformat": "8.15.4", | ||||
|     "@formatjs/intl-pluralrules": "5.4.4", | ||||
|     "@formatjs/intl-relativetimeformat": "11.4.11", | ||||
|     "@fullcalendar/core": "6.1.19", | ||||
|     "@fullcalendar/daygrid": "6.1.19", | ||||
|     "@fullcalendar/interaction": "6.1.19", | ||||
|     "@fullcalendar/list": "6.1.19", | ||||
|     "@fullcalendar/luxon3": "6.1.19", | ||||
|     "@fullcalendar/timegrid": "6.1.19", | ||||
|     "@home-assistant/webawesome": "3.0.0-beta.6.ha.4", | ||||
|     "@fullcalendar/core": "6.1.17", | ||||
|     "@fullcalendar/daygrid": "6.1.17", | ||||
|     "@fullcalendar/interaction": "6.1.17", | ||||
|     "@fullcalendar/list": "6.1.17", | ||||
|     "@fullcalendar/luxon3": "6.1.17", | ||||
|     "@fullcalendar/timegrid": "6.1.17", | ||||
|     "@lezer/highlight": "1.2.1", | ||||
|     "@lit-labs/motion": "1.0.9", | ||||
|     "@lit-labs/observers": "2.0.6", | ||||
|     "@lit-labs/virtualizer": "2.1.1", | ||||
|     "@lit/context": "1.1.6", | ||||
|     "@lit/reactive-element": "2.1.1", | ||||
|     "@lit-labs/motion": "1.0.8", | ||||
|     "@lit-labs/observers": "2.0.5", | ||||
|     "@lit-labs/virtualizer": "2.1.0", | ||||
|     "@lit/context": "1.1.5", | ||||
|     "@lit/reactive-element": "2.1.0", | ||||
|     "@material/chips": "=14.0.0-canary.53b3cad2f.0", | ||||
|     "@material/data-table": "=14.0.0-canary.53b3cad2f.0", | ||||
|     "@material/mwc-base": "0.27.0", | ||||
|     "@material/mwc-button": "0.27.0", | ||||
|     "@material/mwc-checkbox": "0.27.0", | ||||
|     "@material/mwc-dialog": "0.27.0", | ||||
|     "@material/mwc-drawer": "0.27.0", | ||||
| @@ -81,37 +82,38 @@ | ||||
|     "@material/mwc-top-app-bar": "0.27.0", | ||||
|     "@material/mwc-top-app-bar-fixed": "0.27.0", | ||||
|     "@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0", | ||||
|     "@material/web": "2.4.0", | ||||
|     "@material/web": "2.3.0", | ||||
|     "@mdi/js": "7.4.47", | ||||
|     "@mdi/svg": "7.4.47", | ||||
|     "@replit/codemirror-indentation-markers": "6.5.3", | ||||
|     "@shoelace-style/shoelace": "2.20.1", | ||||
|     "@swc/helpers": "0.5.17", | ||||
|     "@thomasloven/round-slider": "0.6.0", | ||||
|     "@tsparticles/engine": "3.9.1", | ||||
|     "@tsparticles/engine": "3.8.1", | ||||
|     "@tsparticles/preset-links": "3.2.0", | ||||
|     "@vaadin/combo-box": "24.9.0", | ||||
|     "@vaadin/vaadin-themable-mixin": "24.9.0", | ||||
|     "@vaadin/combo-box": "24.7.5", | ||||
|     "@vaadin/vaadin-themable-mixin": "24.7.5", | ||||
|     "@vibrant/color": "4.0.0", | ||||
|     "@vue/web-component-wrapper": "1.3.0", | ||||
|     "@webcomponents/scoped-custom-element-registry": "0.0.10", | ||||
|     "@webcomponents/webcomponentsjs": "2.8.0", | ||||
|     "app-datepicker": "5.1.1", | ||||
|     "barcode-detector": "3.0.5", | ||||
|     "color-name": "2.0.2", | ||||
|     "barcode-detector": "3.0.4", | ||||
|     "color-name": "2.0.0", | ||||
|     "comlink": "4.4.2", | ||||
|     "core-js": "3.45.1", | ||||
|     "core-js": "3.42.0", | ||||
|     "cropperjs": "1.6.2", | ||||
|     "culori": "4.0.2", | ||||
|     "date-fns": "4.1.0", | ||||
|     "date-fns-tz": "3.2.0", | ||||
|     "deep-clone-simple": "1.1.1", | ||||
|     "deep-freeze": "0.0.1", | ||||
|     "dialog-polyfill": "0.5.6", | ||||
|     "echarts": "6.0.0", | ||||
|     "echarts": "5.6.0", | ||||
|     "element-internals-polyfill": "3.0.2", | ||||
|     "fuse.js": "7.1.0", | ||||
|     "google-timezones-json": "1.2.0", | ||||
|     "gulp-zopfli-green": "6.0.2", | ||||
|     "hls.js": "1.6.12", | ||||
|     "hls.js": "1.6.2", | ||||
|     "home-assistant-js-websocket": "9.5.0", | ||||
|     "idb-keyval": "6.2.2", | ||||
|     "intl-messageformat": "10.7.16", | ||||
| @@ -119,10 +121,10 @@ | ||||
|     "leaflet": "1.9.4", | ||||
|     "leaflet-draw": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch", | ||||
|     "leaflet.markercluster": "1.5.3", | ||||
|     "lit": "3.3.1", | ||||
|     "lit-html": "3.3.1", | ||||
|     "luxon": "3.7.2", | ||||
|     "marked": "16.3.0", | ||||
|     "lit": "3.3.0", | ||||
|     "lit-html": "3.3.0", | ||||
|     "luxon": "3.6.1", | ||||
|     "marked": "15.0.11", | ||||
|     "memoize-one": "6.0.0", | ||||
|     "node-vibrant": "4.0.3", | ||||
|     "object-hash": "3.0.0", | ||||
| @@ -135,7 +137,9 @@ | ||||
|     "stacktrace-js": "2.0.2", | ||||
|     "superstruct": "2.0.2", | ||||
|     "tinykeys": "3.0.0", | ||||
|     "ua-parser-js": "2.0.5", | ||||
|     "ua-parser-js": "2.0.3", | ||||
|     "vis-data": "7.1.9", | ||||
|     "vis-network": "9.1.9", | ||||
|     "vue": "2.7.16", | ||||
|     "vue2-daterange-picker": "0.6.8", | ||||
|     "weekstart": "2.0.0", | ||||
| @@ -148,92 +152,94 @@ | ||||
|     "xss": "1.0.15" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/core": "7.28.4", | ||||
|     "@babel/helper-define-polyfill-provider": "0.6.5", | ||||
|     "@babel/plugin-transform-runtime": "7.28.3", | ||||
|     "@babel/preset-env": "7.28.3", | ||||
|     "@bundle-stats/plugin-webpack-filter": "4.21.3", | ||||
|     "@lokalise/node-api": "15.2.1", | ||||
|     "@octokit/auth-oauth-device": "8.0.1", | ||||
|     "@octokit/plugin-retry": "8.0.1", | ||||
|     "@octokit/rest": "22.0.0", | ||||
|     "@rsdoctor/rspack-plugin": "1.2.3", | ||||
|     "@rspack/core": "1.5.5", | ||||
|     "@rspack/dev-server": "1.1.4", | ||||
|     "@babel/core": "7.27.1", | ||||
|     "@babel/helper-define-polyfill-provider": "0.6.4", | ||||
|     "@babel/plugin-transform-runtime": "7.27.1", | ||||
|     "@babel/preset-env": "7.27.2", | ||||
|     "@bundle-stats/plugin-webpack-filter": "4.20.1", | ||||
|     "@custom-elements-manifest/analyzer": "0.10.4", | ||||
|     "@custom-elements-manifest/to-markdown": "0.1.0", | ||||
|     "@lokalise/node-api": "14.7.0", | ||||
|     "@octokit/auth-oauth-device": "7.1.5", | ||||
|     "@octokit/plugin-retry": "7.2.1", | ||||
|     "@octokit/rest": "21.1.1", | ||||
|     "@rsdoctor/rspack-plugin": "1.1.2", | ||||
|     "@rspack/cli": "1.3.9", | ||||
|     "@rspack/core": "1.3.9", | ||||
|     "@types/babel__plugin-transform-runtime": "7.9.5", | ||||
|     "@types/chromecast-caf-receiver": "6.0.22", | ||||
|     "@types/chromecast-caf-receiver": "6.0.21", | ||||
|     "@types/chromecast-caf-sender": "1.0.11", | ||||
|     "@types/color-name": "2.0.0", | ||||
|     "@types/culori": "4.0.1", | ||||
|     "@types/glob": "8.1.0", | ||||
|     "@types/html-minifier-terser": "7.0.2", | ||||
|     "@types/js-yaml": "4.0.9", | ||||
|     "@types/leaflet": "1.9.20", | ||||
|     "@types/leaflet-draw": "1.0.13", | ||||
|     "@types/leaflet.markercluster": "1.5.6", | ||||
|     "@types/leaflet": "1.9.17", | ||||
|     "@types/leaflet-draw": "1.0.12", | ||||
|     "@types/leaflet.markercluster": "1.5.5", | ||||
|     "@types/lodash.merge": "4.6.9", | ||||
|     "@types/luxon": "3.7.1", | ||||
|     "@types/luxon": "3.6.2", | ||||
|     "@types/mocha": "10.0.10", | ||||
|     "@types/qrcode": "1.5.5", | ||||
|     "@types/sortablejs": "1.15.8", | ||||
|     "@types/tar": "6.1.13", | ||||
|     "@types/ua-parser-js": "0.7.39", | ||||
|     "@types/webspeechapi": "0.0.29", | ||||
|     "@vitest/coverage-v8": "3.2.4", | ||||
|     "@vitest/coverage-v8": "3.1.3", | ||||
|     "babel-loader": "10.0.0", | ||||
|     "babel-plugin-template-html-minifier": "4.1.0", | ||||
|     "browserslist-useragent-regexp": "4.1.3", | ||||
|     "del": "8.0.1", | ||||
|     "eslint": "9.36.0", | ||||
|     "del": "8.0.0", | ||||
|     "eslint": "9.26.0", | ||||
|     "eslint-config-airbnb-base": "15.0.0", | ||||
|     "eslint-config-prettier": "10.1.8", | ||||
|     "eslint-config-prettier": "10.1.5", | ||||
|     "eslint-import-resolver-webpack": "0.13.10", | ||||
|     "eslint-plugin-import": "2.32.0", | ||||
|     "eslint-plugin-import": "2.31.0", | ||||
|     "eslint-plugin-lit": "2.1.1", | ||||
|     "eslint-plugin-lit-a11y": "5.1.1", | ||||
|     "eslint-plugin-unused-imports": "4.2.0", | ||||
|     "eslint-plugin-lit-a11y": "4.1.4", | ||||
|     "eslint-plugin-unused-imports": "4.1.4", | ||||
|     "eslint-plugin-wc": "3.0.1", | ||||
|     "fancy-log": "2.0.0", | ||||
|     "fs-extra": "11.3.2", | ||||
|     "glob": "11.0.3", | ||||
|     "gulp": "5.0.1", | ||||
|     "fs-extra": "11.3.0", | ||||
|     "glob": "11.0.2", | ||||
|     "gulp": "5.0.0", | ||||
|     "gulp-brotli": "3.0.0", | ||||
|     "gulp-json-transform": "0.5.0", | ||||
|     "gulp-rename": "2.1.0", | ||||
|     "gulp-rename": "2.0.0", | ||||
|     "html-minifier-terser": "7.2.0", | ||||
|     "husky": "9.1.7", | ||||
|     "jsdom": "27.0.0", | ||||
|     "jsdom": "26.1.0", | ||||
|     "jszip": "3.10.1", | ||||
|     "lint-staged": "16.1.6", | ||||
|     "lint-staged": "15.5.2", | ||||
|     "lit-analyzer": "2.0.3", | ||||
|     "lodash.merge": "4.6.2", | ||||
|     "lodash.template": "4.5.0", | ||||
|     "map-stream": "0.0.7", | ||||
|     "pinst": "3.0.0", | ||||
|     "prettier": "3.6.2", | ||||
|     "rspack-manifest-plugin": "5.1.0", | ||||
|     "serve": "14.2.5", | ||||
|     "sinon": "21.0.0", | ||||
|     "prettier": "3.5.3", | ||||
|     "rspack-manifest-plugin": "5.0.3", | ||||
|     "serve": "14.2.4", | ||||
|     "sinon": "20.0.0", | ||||
|     "tar": "7.4.3", | ||||
|     "terser-webpack-plugin": "5.3.14", | ||||
|     "ts-lit-plugin": "2.0.2", | ||||
|     "typescript": "5.9.2", | ||||
|     "typescript-eslint": "8.44.0", | ||||
|     "typescript": "5.8.3", | ||||
|     "typescript-eslint": "8.32.0", | ||||
|     "vite-tsconfig-paths": "5.1.4", | ||||
|     "vitest": "3.2.4", | ||||
|     "vitest": "3.1.3", | ||||
|     "webpack-stats-plugin": "1.1.3", | ||||
|     "webpackbar": "7.0.0", | ||||
|     "workbox-build": "patch:workbox-build@npm%3A7.1.1#~/.yarn/patches/workbox-build-npm-7.1.1-a854f3faae.patch" | ||||
|   }, | ||||
|   "resolutions": { | ||||
|     "@material/mwc-button@^0.25.3": "^0.27.0", | ||||
|     "lit": "3.3.1", | ||||
|     "lit-html": "3.3.1", | ||||
|     "lit": "3.3.0", | ||||
|     "lit-html": "3.3.0", | ||||
|     "clean-css": "5.3.3", | ||||
|     "@lit/reactive-element": "2.1.1", | ||||
|     "@fullcalendar/daygrid": "6.1.19", | ||||
|     "globals": "16.4.0", | ||||
|     "@lit/reactive-element": "2.1.0", | ||||
|     "@fullcalendar/daygrid": "6.1.17", | ||||
|     "globals": "16.1.0", | ||||
|     "tslib": "2.8.1", | ||||
|     "@material/mwc-list@^0.27.0": "patch:@material/mwc-list@npm%3A0.27.0#~/.yarn/patches/@material-mwc-list-npm-0.27.0-5344fc9de4.patch" | ||||
|   }, | ||||
|   "packageManager": "yarn@4.10.2" | ||||
|   "packageManager": "yarn@4.9.1" | ||||
| } | ||||
|   | ||||
| @@ -1,11 +1,3 @@ | ||||
| export default { | ||||
|   trailingComma: "es5", | ||||
|   overrides: [ | ||||
|     { | ||||
|       files: "*.globals.ts", | ||||
|       options: { | ||||
|         printWidth: 9999, // Effectively disables line wrapping for these files | ||||
|       }, | ||||
|     }, | ||||
|   ], | ||||
| }; | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								public/static/images/config_philips_hue.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 5.4 KiB | 
| @@ -1,32 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_3969_57097)"> | ||||
| <path d="M0 4C0 1.79086 1.79086 0 4 0H16C18.2091 0 20 1.79086 20 4V4C20 6.20914 18.2091 8 16 8H4C1.79086 8 0 6.20914 0 4V4Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M0 20C0 15.5817 3.58172 12 8 12H68C72.4183 12 76 15.5817 76 20V36C76 40.4183 72.4183 44 68 44H8C3.58172 44 0 40.4183 0 36V20Z" fill="#1C1C1C"/> | ||||
| <path d="M8 12.5H68C72.1421 12.5 75.5 15.8579 75.5 20V36C75.5 40.1421 72.1421 43.5 68 43.5H8C3.85786 43.5 0.5 40.1421 0.5 36V20C0.5 15.8579 3.85786 12.5 8 12.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M32.9844 27.0156C32.9844 26.0781 32.7031 25.2656 32.1406 24.5781C31.5781 23.8594 30.8594 23.375 29.9844 23.125V22C29.9844 21.4375 30.125 20.9375 30.4062 20.5C30.6875 20.0312 31.0469 19.6719 31.4844 19.4219C31.9531 19.1406 32.4531 19 32.9844 19H43.0156C43.5469 19 44.0312 19.1406 44.4688 19.4219C44.9375 19.6719 45.3125 20.0312 45.5938 20.5C45.875 20.9375 46.0156 21.4375 46.0156 22V23.125C45.1406 23.375 44.4219 23.8594 43.8594 24.5781C43.2969 25.2656 43.0156 26.0781 43.0156 27.0156V28.9844H32.9844V27.0156ZM47 25C47.5625 25 48.0312 25.2031 48.4062 25.6094C48.8125 25.9844 49.0156 26.4531 49.0156 27.0156V31.9844C49.0156 32.5469 48.875 33.0625 48.5938 33.5312C48.3125 33.9688 47.9375 34.3281 47.4688 34.6094C47.0312 34.8594 46.5469 34.9844 46.0156 34.9844V36.0156C46.0156 36.2656 45.9062 36.5 45.6875 36.7188C45.5 36.9062 45.2656 37 44.9844 37C44.7344 37 44.5 36.9062 44.2812 36.7188C44.0938 36.5 44 36.2656 44 36.0156V34.9844H32V36.0156C32 36.2656 31.8906 36.5 31.6719 36.7188C31.4844 36.9062 31.2656 37 31.0156 37C30.7344 37 30.4844 36.9062 30.2656 36.7188C30.0781 36.5 29.9844 36.2656 29.9844 36.0156V34.9844C29.4531 34.9844 28.9531 34.8594 28.4844 34.6094C28.0469 34.3281 27.6875 33.9688 27.4062 33.5312C27.125 33.0625 26.9844 32.5469 26.9844 31.9844V27.0156C26.9844 26.4531 27.1719 25.9844 27.5469 25.6094C27.9531 25.2031 28.4375 25 29 25C29.5625 25 30.0312 25.2031 30.4062 25.6094C30.8125 25.9844 31.0156 26.4531 31.0156 27.0156V31H44.9844V27.0156C44.9844 26.4531 45.1719 25.9844 45.5469 25.6094C45.9531 25.2031 46.4375 25 47 25Z" fill="#03A9F4"/> | ||||
| <path d="M0 56C0 51.5817 3.58172 48 8 48H68C72.4183 48 76 51.5817 76 56V72C76 76.4183 72.4183 80 68 80H8C3.58172 80 0 76.4183 0 72V56Z" fill="#1C1C1C"/> | ||||
| <path d="M8 48.5H68C72.1421 48.5 75.5 51.8579 75.5 56V72C75.5 76.1421 72.1421 79.5 68 79.5H8C3.85786 79.5 0.5 76.1421 0.5 72V56C0.5 51.8579 3.85786 48.5 8 48.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M44 61.9844H47.9844V64H46.0156V72.0156H29.9844V64H28.0156V61.9844H32C31.4375 61.9844 30.9531 61.7969 30.5469 61.4219C30.1719 61.0156 29.9844 60.5469 29.9844 60.0156V55.9844H35.9844V60.0156C35.9844 60.5469 35.7812 61.0156 35.375 61.4219C35 61.7969 34.5469 61.9844 34.0156 61.9844H41.9844V58.9844C41.9844 58.7344 41.8906 58.5156 41.7031 58.3281C41.5156 58.1094 41.2812 58 41 58C40.7188 58 40.4844 58.1094 40.2969 58.3281C40.1094 58.5156 40.0156 58.7344 40.0156 58.9844H38C38 58.4531 38.125 57.9688 38.375 57.5312C38.6562 57.0625 39.0156 56.6875 39.4531 56.4062C39.9219 56.125 40.4375 55.9844 41 55.9844C41.5625 55.9844 42.0625 56.125 42.5 56.4062C42.9688 56.6875 43.3281 57.0625 43.5781 57.5312C43.8594 57.9688 44 58.4531 44 58.9844V61.9844ZM38.9844 70V64H37.0156V70H38.9844Z" fill="#03A9F4"/> | ||||
| <path d="M0 92C0 87.5817 3.58172 84 8 84H68C72.4183 84 76 87.5817 76 92V108C76 112.418 72.4183 116 68 116H8C3.58172 116 0 112.418 0 108V92Z" fill="#1C1C1C"/> | ||||
| <path d="M8 84.5H68C72.1421 84.5 75.5 87.8579 75.5 92V108C75.5 112.142 72.1421 115.5 68 115.5H8C3.85786 115.5 0.5 112.142 0.5 108V92C0.5 87.8579 3.85786 84.5 8 84.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M44.9844 94.9844C46.0781 94.9844 47.0156 95.3906 47.7969 96.2031C48.6094 96.9844 49.0156 97.9219 49.0156 99.0156V108.016H47V105.016H29V108.016H26.9844V93.0156H29V102.016H37.0156V94.9844H44.9844ZM35.0938 100.094C34.5 100.688 33.7969 100.984 32.9844 100.984C32.1719 100.984 31.4688 100.688 30.875 100.094C30.2812 99.5 29.9844 98.7969 29.9844 97.9844C29.9844 97.1719 30.2812 96.4688 30.875 95.875C31.4688 95.2812 32.1719 94.9844 32.9844 94.9844C33.7969 94.9844 34.5 95.2812 35.0938 95.875C35.6875 96.4688 35.9844 97.1719 35.9844 97.9844C35.9844 98.7969 35.6875 99.5 35.0938 100.094Z" fill="#03A9F4"/> | ||||
| <path d="M0 128C0 123.582 3.58172 120 8 120H68C72.4183 120 76 123.582 76 128V144C76 148.418 72.4183 152 68 152H8C3.58172 152 0 148.418 0 144V128Z" fill="#1C1C1C"/> | ||||
| <path d="M8 120.5H68C72.1421 120.5 75.5 123.858 75.5 128V144C75.5 148.142 72.1421 151.5 68 151.5H8C3.85786 151.5 0.5 148.142 0.5 144V128C0.5 123.858 3.85786 120.5 8 120.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M46.0156 136.984H47.9844V142.984C47.9844 143.516 47.7812 143.984 47.375 144.391C47 144.797 46.5469 145 46.0156 145C46.0156 145.281 45.9062 145.516 45.6875 145.703C45.5 145.891 45.2656 145.984 44.9844 145.984H31.0156C30.7344 145.984 30.4844 145.891 30.2656 145.703C30.0781 145.516 29.9844 145.281 29.9844 145C29.4531 145 28.9844 144.797 28.5781 144.391C28.2031 143.984 28.0156 143.516 28.0156 142.984V136.984H31.0156V136.234C31.0156 135.641 31.2344 135.125 31.6719 134.688C32.1406 134.219 32.6719 133.984 33.2656 133.984C33.8906 133.984 34.4531 134.234 34.9531 134.734L36.3125 136.281C36.5 136.5 36.7812 136.734 37.1562 136.984H44V128.828C44 128.609 43.9219 128.422 43.7656 128.266C43.6094 128.078 43.4062 127.984 43.1562 127.984C42.9375 127.984 42.75 128.062 42.5938 128.219L41.3281 129.484C41.3906 129.734 41.4219 129.906 41.4219 130C41.4219 130.344 41.3125 130.703 41.0938 131.078L38.3281 128.312C38.7031 128.094 39.0625 127.984 39.4062 127.984C39.5625 127.984 39.7344 128.016 39.9219 128.078L41.1875 126.812C41.7188 126.281 42.375 126.016 43.1562 126.016C43.9375 126.016 44.6094 126.297 45.1719 126.859C45.7344 127.391 46.0156 128.047 46.0156 128.828V136.984ZM31.5781 132.438C31.2031 132.031 31.0156 131.547 31.0156 130.984C31.0156 130.422 31.2031 129.953 31.5781 129.578C31.9531 129.203 32.4219 129.016 32.9844 129.016C33.5469 129.016 34.0156 129.203 34.3906 129.578C34.7969 129.953 35 130.422 35 130.984C35 131.547 34.7969 132.031 34.3906 132.438C34.0156 132.812 33.5469 133 32.9844 133C32.4219 133 31.9531 132.812 31.5781 132.438Z" fill="#03A9F4"/> | ||||
| <path d="M84 4C84 1.79086 85.7909 0 88 0H100C102.209 0 104 1.79086 104 4V4C104 6.20914 102.209 8 100 8H88C85.7909 8 84 6.20914 84 4V4Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M84 20C84 15.5817 87.5817 12 92 12H152C156.418 12 160 15.5817 160 20V36C160 40.4183 156.418 44 152 44H92C87.5817 44 84 40.4183 84 36V20Z" fill="#1C1C1C"/> | ||||
| <path d="M92 12.5H152C156.142 12.5 159.5 15.8579 159.5 20V36C159.5 40.1421 156.142 43.5 152 43.5H92C87.8579 43.5 84.5 40.1421 84.5 36V20C84.5 15.8579 87.8579 12.5 92 12.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M131.984 30.0156C131.984 30.7656 131.797 31.4531 131.422 32.0781C131.047 32.6719 130.562 33.1406 129.969 33.4844C129.844 34.2031 129.5 34.8125 128.938 35.3125C128.406 35.7812 127.766 36.0156 127.016 36.0156C126.359 36.0156 125.766 35.8281 125.234 35.4531C124.734 35.0781 124.391 34.5938 124.203 34H119.797C119.609 34.5938 119.25 35.0781 118.719 35.4531C118.219 35.8281 117.641 36.0156 116.984 36.0156C116.234 36.0156 115.578 35.7812 115.016 35.3125C114.484 34.8125 114.156 34.2031 114.031 33.4844C113.438 33.1406 112.953 32.6719 112.578 32.0781C112.203 31.4531 112.016 30.7656 112.016 30.0156C112.016 29.1094 112.266 28.3125 112.766 27.625C113.297 26.9375 113.969 26.4688 114.781 26.2188L113 24.3906L112.719 24.7188C112.5 24.9062 112.25 25 111.969 25C111.719 25 111.5 24.9062 111.312 24.7188C111.094 24.5312 110.984 24.2969 110.984 24.0156C110.984 23.7344 111.094 23.5 111.312 23.3125L113.281 21.2969C113.469 21.1094 113.703 21.0156 113.984 21.0156C114.266 21.0156 114.5 21.1094 114.688 21.2969C114.906 21.4844 115.016 21.7188 115.016 22C115.016 22.2812 114.906 22.5156 114.688 22.7031L114.406 22.9844L115.812 24.3906L116.609 22.0469C116.797 21.4219 117.156 20.9219 117.688 20.5469C118.219 20.1719 118.797 19.9844 119.422 19.9844H124.578C125.203 19.9844 125.781 20.1719 126.312 20.5469C126.844 20.9219 127.203 21.4219 127.391 22.0469L128.75 26.0781C129.375 26.2031 129.922 26.4531 130.391 26.8281C130.891 27.2031 131.281 27.6719 131.562 28.2344C131.844 28.7656 131.984 29.3594 131.984 30.0156ZM116.984 34C117.266 34 117.5 33.9062 117.688 33.7188C117.906 33.5 118.016 33.2656 118.016 33.0156C118.016 32.7344 117.906 32.5 117.688 32.3125C117.5 32.0938 117.266 31.9844 116.984 31.9844C116.734 31.9844 116.5 32.0938 116.281 32.3125C116.094 32.5 116 32.7344 116 33.0156C116 33.2656 116.094 33.5 116.281 33.7188C116.5 33.9062 116.734 34 116.984 34ZM121.016 25.9844V22H119.422C118.953 22 118.641 22.2344 118.484 22.7031L117.406 25.9844H121.016ZM122.984 22V25.9844H126.594L125.516 22.7031C125.359 22.2344 125.047 22 124.578 22H122.984ZM127.016 34C127.266 34 127.484 33.9062 127.672 33.7188C127.891 33.5 128 33.2656 128 33.0156C128 32.7344 127.891 32.5 127.672 32.3125C127.484 32.0938 127.266 31.9844 127.016 31.9844C126.734 31.9844 126.484 32.0938 126.266 32.3125C126.078 32.5 125.984 32.7344 125.984 33.0156C125.984 33.2656 126.078 33.5 126.266 33.7188C126.484 33.9062 126.734 34 127.016 34Z" fill="#03A9F4"/> | ||||
| <path d="M84 56C84 51.5817 87.5817 48 92 48H152C156.418 48 160 51.5817 160 56V72C160 76.4183 156.418 80 152 80H92C87.5817 80 84 76.4183 84 72V56Z" fill="#1C1C1C"/> | ||||
| <path d="M92 48.5H152C156.142 48.5 159.5 51.8579 159.5 56V72C159.5 76.1421 156.142 79.5 152 79.5H92C87.8579 79.5 84.5 76.1421 84.5 72V56C84.5 51.8579 87.8579 48.5 92 48.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M128.984 58.9844C130.078 58.9844 131.016 59.3906 131.797 60.2031C132.609 60.9844 133.016 61.9219 133.016 63.0156V72.0156H131V69.0156H113V72.0156H110.984V57.0156H113V66.0156H121.016V58.9844H128.984ZM119.094 64.0938C118.5 64.6875 117.797 64.9844 116.984 64.9844C116.172 64.9844 115.469 64.6875 114.875 64.0938C114.281 63.5 113.984 62.7969 113.984 61.9844C113.984 61.1719 114.281 60.4688 114.875 59.875C115.469 59.2812 116.172 58.9844 116.984 58.9844C117.797 58.9844 118.5 59.2812 119.094 59.875C119.688 60.4688 119.984 61.1719 119.984 61.9844C119.984 62.7969 119.688 63.5 119.094 64.0938Z" fill="#03A9F4"/> | ||||
| <path d="M84 92C84 87.5817 87.5817 84 92 84H152C156.418 84 160 87.5817 160 92V108C160 112.418 156.418 116 152 116H92C87.5817 116 84 112.418 84 108V92Z" fill="#1C1C1C"/> | ||||
| <path d="M92 84.5H152C156.142 84.5 159.5 87.8579 159.5 92V108C159.5 112.142 156.142 115.5 152 115.5H92C87.8579 115.5 84.5 112.142 84.5 108V92C84.5 87.8579 87.8579 84.5 92 84.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M112.016 94H131.984V106H130.016V103.984H125.984V106H124.016V96.0156H113.984V106H112.016V94ZM130.016 96.0156H125.984V97.9844H130.016V96.0156ZM125.984 102.016H130.016V100H125.984V102.016Z" fill="#03A9F4"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_3969_57097"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 11 KiB | 
| @@ -1,76 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_4744_40067)"> | ||||
| <path d="M0 6C0 2.68629 2.68629 0 6 0H28C31.3137 0 34 2.68629 34 6C34 9.31371 31.3137 12 28 12H6C2.68629 12 0 9.31371 0 6Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M0 28C0 23.5817 3.58172 20 8 20H42.6667C47.0849 20 50.6667 23.5817 50.6667 28V36C50.6667 40.4183 47.0849 44 42.6667 44H8.00001C3.58173 44 0 40.4183 0 36V28Z" fill="#1C1C1C"/> | ||||
| <path d="M8 20.5H42.667C46.809 20.5002 50.167 23.858 50.167 28V36C50.167 40.142 46.809 43.4998 42.667 43.5H8C3.85787 43.5 0.5 40.1421 0.5 36V28C0.5 23.8579 3.85786 20.5 8 20.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M6 32C6 28.6863 8.68629 26 12 26C15.3137 26 18 28.6863 18 32C18 35.3137 15.3137 38 12 38C8.68629 38 6 35.3137 6 32Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M24 31C24 29.3431 25.3431 28 27 28H39.6667C41.3235 28 42.6667 29.3431 42.6667 31V33C42.6667 34.6569 41.3235 36 39.6667 36H27C25.3431 36 24 34.6569 24 33V31Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M54.6666 28C54.6666 23.5817 58.2483 20 62.6666 20H97.3333C101.752 20 105.333 23.5817 105.333 28V36C105.333 40.4183 101.752 44 97.3333 44H62.6666C58.2484 44 54.6666 40.4183 54.6666 36V28Z" fill="#1C1C1C"/> | ||||
| <path d="M62.6666 20.5H97.3336C101.476 20.5002 104.834 23.858 104.834 28V36C104.834 40.142 101.476 43.4998 97.3336 43.5H62.6666C58.5245 43.5 55.1666 40.1421 55.1666 36V28C55.1666 23.8579 58.5245 20.5 62.6666 20.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M60.6666 32C60.6666 28.6863 63.3529 26 66.6666 26C69.9803 26 72.6666 28.6863 72.6666 32C72.6666 35.3137 69.9803 38 66.6666 38C63.3529 38 60.6666 35.3137 60.6666 32Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M78.6666 31C78.6666 29.3431 80.0098 28 81.6666 28H94.3333C95.9901 28 97.3333 29.3431 97.3333 31V33C97.3333 34.6569 95.9901 36 94.3333 36H81.6666C80.0098 36 78.6666 34.6569 78.6666 33V31Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M109.333 28C109.333 23.5817 112.915 20 117.333 20H152C156.418 20 160 23.5817 160 28V36C160 40.4183 156.418 44 152 44H117.333C112.915 44 109.333 40.4183 109.333 36V28Z" fill="#1C1C1C"/> | ||||
| <path d="M117.333 20.5H152C156.142 20.5002 159.5 23.858 159.5 28V36C159.5 40.142 156.142 43.4998 152 43.5H117.333C113.191 43.5 109.833 40.1421 109.833 36V28C109.833 23.8579 113.191 20.5 117.333 20.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M115.333 32C115.333 28.6863 118.02 26 121.333 26C124.647 26 127.333 28.6863 127.333 32C127.333 35.3137 124.647 38 121.333 38C118.02 38 115.333 35.3137 115.333 32Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M133.333 31C133.333 29.3431 134.677 28 136.333 28H149C150.657 28 152 29.3431 152 31V33C152 34.6569 150.657 36 149 36H136.333C134.677 36 133.333 34.6569 133.333 33V31Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M0 56C0 53.7909 1.79086 52 4 52H29C31.2091 52 33 53.7909 33 56C33 58.2091 31.2091 60 29 60H4C1.79086 60 0 58.2091 0 56Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M0 72C0 67.5817 3.58172 64 8 64H29C33.4183 64 37 67.5817 37 72V96C37 100.418 33.4183 104 29 104H8C3.58172 104 0 100.418 0 96V72Z" fill="#1C1C1C"/> | ||||
| <path d="M8 64.5H29C33.1421 64.5 36.5 67.8579 36.5 72V96C36.5 100.142 33.1421 103.5 29 103.5H8C3.85786 103.5 0.5 100.142 0.5 96V72C0.5 67.8579 3.85786 64.5 8 64.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <mask id="mask0_4744_40067" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="6" y="72" width="25" height="24"> | ||||
| <path d="M18.5 74C16.6435 74 14.863 74.7375 13.5503 76.0503C12.2375 77.363 11.5 79.1435 11.5 81C11.5 83.38 12.69 85.47 14.5 86.74V89C14.5 89.2652 14.6054 89.5196 14.7929 89.7071C14.9804 89.8946 15.2348 90 15.5 90H21.5C21.7652 90 22.0196 89.8946 22.2071 89.7071C22.3946 89.5196 22.5 89.2652 22.5 89V86.74C24.31 85.47 25.5 83.38 25.5 81C25.5 79.1435 24.7625 77.363 23.4497 76.0503C22.137 74.7375 20.3565 74 18.5 74ZM15.5 93C15.5 93.2652 15.6054 93.5196 15.7929 93.7071C15.9804 93.8946 16.2348 94 16.5 94H20.5C20.7652 94 21.0196 93.8946 21.2071 93.7071C21.3946 93.5196 21.5 93.2652 21.5 93V92H15.5V93Z" fill="black"/> | ||||
| </mask> | ||||
| <g mask="url(#mask0_4744_40067)"> | ||||
| <rect x="6.5" y="72" width="24" height="24" fill="#03A9F4"/> | ||||
| </g> | ||||
| <path d="M41 72C41 67.5817 44.5817 64 49 64H70C74.4183 64 78 67.5817 78 72V96C78 100.418 74.4183 104 70 104H49C44.5817 104 41 100.418 41 96V72Z" fill="#1C1C1C"/> | ||||
| <path d="M49 64.5H70C74.1421 64.5 77.5 67.8579 77.5 72V96C77.5 100.142 74.1421 103.5 70 103.5H49C44.8579 103.5 41.5 100.142 41.5 96V72C41.5 67.8579 44.8579 64.5 49 64.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <mask id="mask1_4744_40067" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="47" y="72" width="25" height="24"> | ||||
| <path d="M66.5 80C67.61 80 68.5 80.9 68.5 82V88.76C69.11 89.31 69.5 90.11 69.5 91C69.5 92.66 68.16 94 66.5 94C64.84 94 63.5 92.66 63.5 91C63.5 90.11 63.89 89.31 64.5 88.76V82C64.5 80.9 65.4 80 66.5 80ZM66.5 81C65.95 81 65.5 81.45 65.5 82V83H67.5V82C67.5 81.45 67.05 81 66.5 81ZM52.5 92V84H49.5L59.5 75L63.9 78.96C63.04 79.69 62.5 80.78 62.5 82V88C61.87 88.83 61.5 89.87 61.5 91L61.6 92H52.5Z" fill="black"/> | ||||
| </mask> | ||||
| <g mask="url(#mask1_4744_40067)"> | ||||
| <rect x="47.5" y="72" width="24" height="24" fill="#03A9F4"/> | ||||
| </g> | ||||
| <path d="M82 72C82 67.5817 85.5817 64 90 64H111C115.418 64 119 67.5817 119 72V96C119 100.418 115.418 104 111 104H90C85.5817 104 82 100.418 82 96V72Z" fill="#1C1C1C"/> | ||||
| <path d="M90 64.5H111C115.142 64.5 118.5 67.8579 118.5 72V96C118.5 100.142 115.142 103.5 111 103.5H90C85.8579 103.5 82.5 100.142 82.5 96V72C82.5 67.8579 85.8579 64.5 90 64.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <mask id="mask2_4744_40067" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="88" y="72" width="25" height="24"> | ||||
| <path d="M100.5 84H107.5C106.97 88.11 104.22 91.78 100.5 92.92V84H93.5V78.3L100.5 75.19M100.5 73L91.5 77V83C91.5 88.55 95.34 93.73 100.5 95C105.66 93.73 109.5 88.55 109.5 83V77L100.5 73Z" fill="black"/> | ||||
| </mask> | ||||
| <g mask="url(#mask2_4744_40067)"> | ||||
| <rect x="88.5" y="72" width="24" height="24" fill="#03A9F4"/> | ||||
| </g> | ||||
| <path d="M123 72C123 67.5817 126.582 64 131 64H152C156.418 64 160 67.5817 160 72V96C160 100.418 156.418 104 152 104H131C126.582 104 123 100.418 123 96V72Z" fill="#1C1C1C"/> | ||||
| <path d="M131 64.5H152C156.142 64.5 159.5 67.8579 159.5 72V96C159.5 100.142 156.142 103.5 152 103.5H131C126.858 103.5 123.5 100.142 123.5 96V72C123.5 67.8579 126.858 64.5 131 64.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <mask id="mask3_4744_40067" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="129" y="72" width="25" height="24"> | ||||
| <path d="M145.5 84C145.5 83.4696 145.711 82.9609 146.086 82.5858C146.461 82.2107 146.97 82 147.5 82C148.03 82 148.539 82.2107 148.914 82.5858C149.289 82.9609 149.5 83.4696 149.5 84C149.5 84.5304 149.289 85.0391 148.914 85.4142C148.539 85.7893 148.03 86 147.5 86C146.97 86 146.461 85.7893 146.086 85.4142C145.711 85.0391 145.5 84.5304 145.5 84ZM139.5 84C139.5 83.4696 139.711 82.9609 140.086 82.5858C140.461 82.2107 140.97 82 141.5 82C142.03 82 142.539 82.2107 142.914 82.5858C143.289 82.9609 143.5 83.4696 143.5 84C143.5 84.5304 143.289 85.0391 142.914 85.4142C142.539 85.7893 142.03 86 141.5 86C140.97 86 140.461 85.7893 140.086 85.4142C139.711 85.0391 139.5 84.5304 139.5 84ZM133.5 84C133.5 83.4696 133.711 82.9609 134.086 82.5858C134.461 82.2107 134.97 82 135.5 82C136.03 82 136.539 82.2107 136.914 82.5858C137.289 82.9609 137.5 83.4696 137.5 84C137.5 84.5304 137.289 85.0391 136.914 85.4142C136.539 85.7893 136.03 86 135.5 86C134.97 86 134.461 85.7893 134.086 85.4142C133.711 85.0391 133.5 84.5304 133.5 84Z" fill="black"/> | ||||
| </mask> | ||||
| <g mask="url(#mask3_4744_40067)"> | ||||
| <rect x="129.5" y="72" width="24" height="24" fill="#03A9F4"/> | ||||
| </g> | ||||
| <path d="M0 116C0 113.791 1.79086 112 4 112H29C31.2091 112 33 113.791 33 116C33 118.209 31.2091 120 29 120H4C1.79086 120 0 118.209 0 116Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M0 132C0 127.582 3.58172 124 8 124H70C74.4183 124 78 127.582 78 132V160H0V132Z" fill="url(#paint0_linear_4744_40067)"/> | ||||
| <path d="M8 124.5H70C74.1421 124.5 77.5 127.858 77.5 132V159.5H0.5V132C0.5 127.858 3.85786 124.5 8 124.5Z" stroke="url(#paint1_linear_4744_40067)" stroke-opacity="0.12"/> | ||||
| <path d="M82 132C82 127.582 85.5817 124 90 124H152C156.418 124 160 127.582 160 132V160H82V132Z" fill="url(#paint2_linear_4744_40067)"/> | ||||
| <path d="M90 124.5H152C156.142 124.5 159.5 127.858 159.5 132V159.5H82.5V132C82.5 127.858 85.8579 124.5 90 124.5Z" stroke="url(#paint3_linear_4744_40067)" stroke-opacity="0.12"/> | ||||
| </g> | ||||
| <defs> | ||||
| <linearGradient id="paint0_linear_4744_40067" x1="39" y1="124" x2="39" y2="160" gradientUnits="userSpaceOnUse"> | ||||
| <stop offset="0.5" stop-color="#1C1C1C"/> | ||||
| <stop offset="1" stop-color="#1C1C1C" stop-opacity="0"/> | ||||
| </linearGradient> | ||||
| <linearGradient id="paint1_linear_4744_40067" x1="39" y1="124" x2="39" y2="160" gradientUnits="userSpaceOnUse"> | ||||
| <stop offset="0.5" stop-color="white" stop-opacity="0.24"/> | ||||
| <stop offset="1" stop-opacity="0"/> | ||||
| </linearGradient> | ||||
| <linearGradient id="paint2_linear_4744_40067" x1="121" y1="124" x2="121" y2="160" gradientUnits="userSpaceOnUse"> | ||||
| <stop offset="0.5" stop-color="#1C1C1C"/> | ||||
| <stop offset="1" stop-color="#1C1C1C" stop-opacity="0"/> | ||||
| </linearGradient> | ||||
| <linearGradient id="paint3_linear_4744_40067" x1="121" y1="124" x2="121" y2="160" gradientUnits="userSpaceOnUse"> | ||||
| <stop offset="0.5" stop-color="white" stop-opacity="0.24"/> | ||||
| <stop offset="1" stop-opacity="0"/> | ||||
| </linearGradient> | ||||
| <clipPath id="clip0_4744_40067"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 9.3 KiB | 
| @@ -1,16 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_1738_5540)"> | ||||
| <path d="M0 8C0 3.58172 3.58172 0 8 0H152C156.418 0 160 3.58172 160 8V152C160 156.418 156.418 160 152 160H8C3.58172 160 0 156.418 0 152V8Z" fill="#1C1C1C"/> | ||||
| <path d="M8 0.5H152C156.142 0.500001 159.5 3.85787 159.5 8V152C159.5 156.142 156.142 159.5 152 159.5H8C3.85787 159.5 0.5 156.142 0.5 152V8C0.500001 3.85787 3.85787 0.5 8 0.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M83 86.9844V75.125L77 73.0156V84.875L83 86.9844ZM88.4844 71C88.8281 71 89 71.1719 89 71.5156V86.6094C89 86.8594 88.875 87.0156 88.625 87.0781L83 89L77 86.8906L71.6562 88.9531L71.5156 89C71.1719 89 71 88.8281 71 88.4844V73.3906C71 73.1406 71.125 72.9844 71.375 72.9219L77 71L83 73.1094L88.3438 71.0469L88.4844 71Z" fill="#03A9F4"/> | ||||
| <path d="M148 77.0508L139.288 85.8983M139.288 85.8983L116.852 108.684C115.724 109.83 114.184 110.475 112.576 110.475H81.6939M139.288 85.8983V69.4322M21.1957 82.9492H12M21.1957 82.9492L32.3274 71.6441M21.1957 82.9492L44.427 106.542M81.6939 110.475V124.237M81.6939 110.475H54.5907M81.6939 138V124.237M81.6939 124.237H148M32.3274 71.6441L43.4591 60.339L54.5907 49.0339L70.4941 32.8828C74.2533 29.0651 79.3871 26.9153 84.7451 26.9153H128.641V26.9153C134.521 26.9153 139.288 31.6824 139.288 37.5629V37.7288V49.0339M32.3274 71.6441L24.8569 64.0572C22.5573 61.7218 22.5573 57.9732 24.8569 55.6377L57.9786 22M139.288 49.0339H126.313C124.706 49.0339 123.166 48.389 122.038 47.2436L116.057 41.1695M139.288 49.0339V69.4322M139.288 69.4322H126.313C124.706 69.4322 123.166 68.7873 122.038 67.6419L116.057 61.5678M54.5907 110.475V110.475C54.5907 107.488 52.17 105.068 49.184 105.068H49.0249C45.951 105.068 43.4591 107.56 43.4591 110.634V110.807C43.4591 113.881 45.951 116.373 49.0249 116.373V116.373C52.0988 116.373 54.5907 113.881 54.5907 110.807V110.475ZM44.427 115.39L22.1637 138" stroke="white" stroke-opacity="0.24" stroke-width="5" stroke-linecap="round"/> | ||||
| <circle cx="41" cy="39" r="14" fill="white" fill-opacity="0.48"/> | ||||
| <circle cx="89" cy="117" r="14" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M116 62.325C115.767 62.325 115.533 62.2833 115.3 62.2C115.067 62.1167 114.858 61.9917 114.675 61.825C113.592 60.825 112.633 59.85 111.8 58.9C110.967 57.95 110.267 57.0333 109.7 56.15C109.15 55.25 108.725 54.3917 108.425 53.575C108.142 52.7417 108 51.95 108 51.2C108 48.7 108.8 46.7083 110.4 45.225C112.017 43.7417 113.883 43 116 43C118.117 43 119.975 43.7417 121.575 45.225C123.192 46.7083 124 48.7 124 51.2C124 51.95 123.85 52.7417 123.55 53.575C123.267 54.3917 122.842 55.25 122.275 56.15C121.725 57.0333 121.033 57.95 120.2 58.9C119.367 59.85 118.408 60.825 117.325 61.825C117.142 61.9917 116.933 62.1167 116.7 62.2C116.467 62.2833 116.233 62.325 116 62.325Z" fill="white" fill-opacity="0.48"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_1738_5540"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 2.9 KiB | 
| @@ -1,13 +0,0 @@ | ||||
| <svg width="160" height="72" viewBox="0 0 160 72" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_1738_5536)"> | ||||
| <path d="M0 4C0 1.79086 1.79086 0 4 0H28C30.2091 0 32 1.79086 32 4V4C32 6.20914 30.2091 8 28 8H4C1.79086 8 0 6.20914 0 4V4Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M0 20C0 15.5817 3.58172 12 8 12H42.6667C47.0849 12 50.6667 15.5817 50.6667 20V64C50.6667 68.4183 47.0849 72 42.6667 72H8.00001C3.58173 72 0 68.4183 0 64V20Z" fill="#1C1C1C"/> | ||||
| <path d="M8 12.5H42.667C46.809 12.5002 50.167 15.858 50.167 20V64C50.167 68.142 46.809 71.4998 42.667 71.5H8C3.85787 71.5 0.5 68.1421 0.5 64V20L0.509766 19.6143C0.710536 15.6514 3.98724 12.5 8 12.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M32.3177 42.9844H26.3177V48.9844H24.349V42.9844H18.349V41.0156H24.349V35.0156H26.3177V41.0156H32.3177V42.9844Z" fill="#03A9F4"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_1738_5536"> | ||||
| <rect width="160" height="72" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 973 B | 
| @@ -1,63 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_1738_5534)"> | ||||
| <path d="M0 8C0 3.58172 3.58172 0 8 0H42.6667C47.0849 0 50.6667 3.58172 50.6667 8V64C50.6667 68.4183 47.0849 72 42.6667 72H8.00001C3.58173 72 0 68.4183 0 64V8Z" fill="#1C1C1C"/> | ||||
| <path d="M8 0.5H42.667C46.809 0.500178 50.167 3.85798 50.167 8V64C50.167 68.142 46.809 71.4998 42.667 71.5H8C3.85787 71.5 0.5 68.1421 0.5 64V8C0.5 3.85786 3.85786 0.5 8 0.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M8 12C8 9.79086 9.79086 8 12 8H38.6667C40.8758 8 42.6667 9.79086 42.6667 12V12C42.6667 14.2091 40.8758 16 38.6667 16H12C9.79086 16 8 14.2091 8 12V12Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M8 27C8 25.3431 9.34315 24 11 24H27.6667C29.3235 24 30.6667 25.3431 30.6667 27V29C30.6667 30.6569 29.3235 32 27.6667 32H11C9.34314 32 8 30.6569 8 29V27Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M34.6667 28C34.6667 25.7909 36.4575 24 38.6667 24V24C40.8758 24 42.6667 25.7909 42.6667 28V28C42.6667 30.2091 40.8758 32 38.6667 32V32C36.4575 32 34.6667 30.2091 34.6667 28V28Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M8 43C8 41.3431 9.34315 40 11 40H27.6667C29.3235 40 30.6667 41.3431 30.6667 43V45C30.6667 46.6569 29.3235 48 27.6667 48H11C9.34314 48 8 46.6569 8 45V43Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M34.6667 44C34.6667 41.7909 36.4575 40 38.6667 40V40C40.8758 40 42.6667 41.7909 42.6667 44V44C42.6667 46.2091 40.8758 48 38.6667 48V48C36.4575 48 34.6667 46.2091 34.6667 44V44Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M8 59C8 57.3431 9.34315 56 11 56H27.6667C29.3235 56 30.6667 57.3431 30.6667 59V61C30.6667 62.6569 29.3235 64 27.6667 64H11C9.34314 64 8 62.6569 8 61V59Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M34.6667 60C34.6667 57.7909 36.4575 56 38.6667 56V56C40.8758 56 42.6667 57.7909 42.6667 60V60C42.6667 62.2091 40.8758 64 38.6667 64V64C36.4575 64 34.6667 62.2091 34.6667 60V60Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M0 84C0 79.5817 3.58172 76 8 76H42.6667C47.0849 76 50.6667 79.5817 50.6667 84V124C50.6667 128.418 47.0849 132 42.6667 132H8.00001C3.58173 132 0 128.418 0 124V84Z" fill="#1C1C1C"/> | ||||
| <path d="M8 76.5H42.667C46.809 76.5002 50.167 79.858 50.167 84V124C50.167 128.142 46.809 131.5 42.667 131.5H8C3.85787 131.5 0.5 128.142 0.5 124V84C0.5 79.8579 3.85786 76.5 8 76.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M8 88C8 85.7909 9.79086 84 12 84H38.6667C40.8758 84 42.6667 85.7909 42.6667 88V88C42.6667 90.2091 40.8758 92 38.6667 92H12C9.79086 92 8 90.2091 8 88V88Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M8 103C8 101.343 9.34315 100 11 100H27.6667C29.3235 100 30.6667 101.343 30.6667 103V105C30.6667 106.657 29.3235 108 27.6667 108H11C9.34314 108 8 106.657 8 105V103Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M34.6667 104C34.6667 101.791 36.4575 100 38.6667 100V100C40.8758 100 42.6667 101.791 42.6667 104V104C42.6667 106.209 40.8758 108 38.6667 108V108C36.4575 108 34.6667 106.209 34.6667 104V104Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M8 119C8 117.343 9.34315 116 11 116H27.6667C29.3235 116 30.6667 117.343 30.6667 119V121C30.6667 122.657 29.3235 124 27.6667 124H11C9.34314 124 8 122.657 8 121V119Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M34.6667 120C34.6667 117.791 36.4575 116 38.6667 116V116C40.8758 116 42.6667 117.791 42.6667 120V120C42.6667 122.209 40.8758 124 38.6667 124V124C36.4575 124 34.6667 122.209 34.6667 120V120Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M0 144C0 139.582 3.58172 136 8 136H42.6667C47.0849 136 50.6667 139.582 50.6667 144V152C50.6667 156.418 47.0849 160 42.6667 160H8.00001C3.58173 160 0 156.418 0 152V144Z" fill="#1C1C1C"/> | ||||
| <path d="M8 136.5H42.667C46.809 136.5 50.167 139.858 50.167 144V152C50.167 156.142 46.809 159.5 42.667 159.5H8C3.85787 159.5 0.5 156.142 0.5 152V144C0.5 139.858 3.85786 136.5 8 136.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M8 148C8 145.791 9.79086 144 12 144H38.6667C40.8758 144 42.6667 145.791 42.6667 148V148C42.6667 150.209 40.8758 152 38.6667 152H12C9.79086 152 8 150.209 8 148V148Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M54.6667 8C54.6667 3.58172 58.2484 0 62.6667 0H97.3333C101.752 0 105.333 3.58172 105.333 8V48C105.333 52.4183 101.752 56 97.3334 56H62.6667C58.2484 56 54.6667 52.4183 54.6667 48V8Z" fill="#1C1C1C"/> | ||||
| <path d="M62.6667 0.5H97.3337C101.476 0.50018 104.834 3.85798 104.834 8V48C104.834 52.142 101.476 55.4998 97.3337 55.5H62.6667C58.5246 55.5 55.1667 52.1421 55.1667 48V8C55.1667 3.85786 58.5246 0.5 62.6667 0.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M62.6667 12C62.6667 9.79086 64.4575 8 66.6667 8H93.3334C95.5425 8 97.3334 9.79086 97.3334 12V12C97.3334 14.2091 95.5425 16 93.3334 16H66.6667C64.4576 16 62.6667 14.2091 62.6667 12V12Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M62.6667 27C62.6667 25.3431 64.0098 24 65.6667 24H82.3334C83.9902 24 85.3334 25.3431 85.3334 27V29C85.3334 30.6569 83.9902 32 82.3334 32H65.6667C64.0098 32 62.6667 30.6569 62.6667 29V27Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M89.3333 28C89.3333 25.7909 91.1242 24 93.3333 24V24C95.5425 24 97.3333 25.7909 97.3333 28V28C97.3333 30.2091 95.5425 32 93.3333 32V32C91.1242 32 89.3333 30.2091 89.3333 28V28Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M62.6667 43C62.6667 41.3431 64.0098 40 65.6667 40H82.3334C83.9902 40 85.3334 41.3431 85.3334 43V45C85.3334 46.6569 83.9902 48 82.3334 48H65.6667C64.0098 48 62.6667 46.6569 62.6667 45V43Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M89.3333 44C89.3333 41.7909 91.1242 40 93.3333 40V40C95.5425 40 97.3333 41.7909 97.3333 44V44C97.3333 46.2091 95.5425 48 93.3333 48V48C91.1242 48 89.3333 46.2091 89.3333 44V44Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M54.6667 68C54.6667 63.5817 58.2484 60 62.6667 60H97.3333C101.752 60 105.333 63.5817 105.333 68V76C105.333 80.4183 101.752 84 97.3334 84H62.6667C58.2484 84 54.6667 80.4183 54.6667 76V68Z" fill="#1C1C1C"/> | ||||
| <path d="M62.6667 60.5H97.3337C101.476 60.5002 104.834 63.858 104.834 68V76C104.834 80.142 101.476 83.4998 97.3337 83.5H62.6667C58.5246 83.5 55.1667 80.1421 55.1667 76V68C55.1667 63.8579 58.5246 60.5 62.6667 60.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M62.6667 72C62.6667 69.7909 64.4575 68 66.6667 68H93.3334C95.5425 68 97.3334 69.7909 97.3334 72V72C97.3334 74.2091 95.5425 76 93.3334 76H66.6667C64.4576 76 62.6667 74.2091 62.6667 72V72Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M54.6667 96C54.6667 91.5817 58.2484 88 62.6667 88H97.3333C101.752 88 105.333 91.5817 105.333 96V136C105.333 140.418 101.752 144 97.3334 144H62.6667C58.2484 144 54.6667 140.418 54.6667 136V96Z" fill="#1C1C1C"/> | ||||
| <path d="M62.6667 88.5H97.3337C101.476 88.5002 104.834 91.858 104.834 96V136C104.834 140.142 101.476 143.5 97.3337 143.5H62.6667C58.5246 143.5 55.1667 140.142 55.1667 136V96C55.1667 91.8579 58.5246 88.5 62.6667 88.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M62.6667 100C62.6667 97.7909 64.4575 96 66.6667 96H93.3334C95.5425 96 97.3334 97.7909 97.3334 100V100C97.3334 102.209 95.5425 104 93.3334 104H66.6667C64.4576 104 62.6667 102.209 62.6667 100V100Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M62.6667 115C62.6667 113.343 64.0098 112 65.6667 112H82.3334C83.9902 112 85.3334 113.343 85.3334 115V117C85.3334 118.657 83.9902 120 82.3334 120H65.6667C64.0098 120 62.6667 118.657 62.6667 117V115Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M89.3333 116C89.3333 113.791 91.1242 112 93.3333 112V112C95.5425 112 97.3333 113.791 97.3333 116V116C97.3333 118.209 95.5425 120 93.3333 120V120C91.1242 120 89.3333 118.209 89.3333 116V116Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M62.6667 131C62.6667 129.343 64.0098 128 65.6667 128H82.3334C83.9902 128 85.3334 129.343 85.3334 131V133C85.3334 134.657 83.9902 136 82.3334 136H65.6667C64.0098 136 62.6667 134.657 62.6667 133V131Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M89.3333 132C89.3333 129.791 91.1242 128 93.3333 128V128C95.5425 128 97.3333 129.791 97.3333 132V132C97.3333 134.209 95.5425 136 93.3333 136V136C91.1242 136 89.3333 134.209 89.3333 132V132Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M109.333 8C109.333 3.58172 112.915 0 117.333 0H152C156.418 0 160 3.58172 160 8V112C160 116.418 156.418 120 152 120H117.333C112.915 120 109.333 116.418 109.333 112V8Z" fill="#1C1C1C"/> | ||||
| <path d="M117.333 0.5H152C156.142 0.50018 159.5 3.85798 159.5 8V112C159.5 116.142 156.142 119.5 152 119.5H117.333C113.191 119.5 109.833 116.142 109.833 112V8C109.833 3.85786 113.191 0.5 117.333 0.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M117.333 12C117.333 9.79086 119.124 8 121.333 8H148C150.209 8 152 9.79086 152 12V12C152 14.2091 150.209 16 148 16H121.333C119.124 16 117.333 14.2091 117.333 12V12Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M117.333 27C117.333 25.3431 118.676 24 120.333 24H137C138.657 24 140 25.3431 140 27V29C140 30.6569 138.657 32 137 32H120.333C118.676 32 117.333 30.6569 117.333 29V27Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 28C144 25.7909 145.791 24 148 24V24C150.209 24 152 25.7909 152 28V28C152 30.2091 150.209 32 148 32V32C145.791 32 144 30.2091 144 28V28Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M117.333 43C117.333 41.3431 118.676 40 120.333 40H137C138.657 40 140 41.3431 140 43V45C140 46.6569 138.657 48 137 48H120.333C118.676 48 117.333 46.6569 117.333 45V43Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 44C144 41.7909 145.791 40 148 40V40C150.209 40 152 41.7909 152 44V44C152 46.2091 150.209 48 148 48V48C145.791 48 144 46.2091 144 44V44Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M117.333 59C117.333 57.3431 118.676 56 120.333 56H137C138.657 56 140 57.3431 140 59V61C140 62.6569 138.657 64 137 64H120.333C118.676 64 117.333 62.6569 117.333 61V59Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 60C144 57.7909 145.791 56 148 56V56C150.209 56 152 57.7909 152 60V60C152 62.2091 150.209 64 148 64V64C145.791 64 144 62.2091 144 60V60Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M117.333 75C117.333 73.3431 118.676 72 120.333 72H137C138.657 72 140 73.3431 140 75V77C140 78.6569 138.657 80 137 80H120.333C118.676 80 117.333 78.6569 117.333 77V75Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 76C144 73.7909 145.791 72 148 72V72C150.209 72 152 73.7909 152 76V76C152 78.2091 150.209 80 148 80V80C145.791 80 144 78.2091 144 76V76Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M117.333 91C117.333 89.3431 118.676 88 120.333 88H137C138.657 88 140 89.3431 140 91V93C140 94.6569 138.657 96 137 96H120.333C118.676 96 117.333 94.6569 117.333 93V91Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 92C144 89.7909 145.791 88 148 88V88C150.209 88 152 89.7909 152 92V92C152 94.2091 150.209 96 148 96V96C145.791 96 144 94.2091 144 92V92Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M117.333 107C117.333 105.343 118.676 104 120.333 104H137C138.657 104 140 105.343 140 107V109C140 110.657 138.657 112 137 112H120.333C118.676 112 117.333 110.657 117.333 109V107Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M144 108C144 105.791 145.791 104 148 104V104C150.209 104 152 105.791 152 108V108C152 110.209 150.209 112 148 112V112C145.791 112 144 110.209 144 108V108Z" fill="white" fill-opacity="0.24"/> | ||||
| <path d="M109.333 132C109.333 127.582 112.915 124 117.333 124H152C156.418 124 160 127.582 160 132V140C160 144.418 156.418 148 152 148H117.333C112.915 148 109.333 144.418 109.333 140V132Z" fill="#1C1C1C"/> | ||||
| <path d="M117.333 124.5H152C156.142 124.5 159.5 127.858 159.5 132V140C159.5 144.142 156.142 147.5 152 147.5H117.333C113.191 147.5 109.833 144.142 109.833 140V132C109.833 127.858 113.191 124.5 117.333 124.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M117.333 136C117.333 133.791 119.124 132 121.333 132H148C150.209 132 152 133.791 152 136V136C152 138.209 150.209 140 148 140H121.333C119.124 140 117.333 138.209 117.333 136V136Z" fill="white" fill-opacity="0.48"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_1738_5534"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 12 KiB | 
| @@ -1,19 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_1738_5538)"> | ||||
| <path d="M0 8C0 3.58172 3.58172 0 8 0H152C156.418 0 160 3.58172 160 8V152C160 156.418 156.418 160 152 160H8C3.58172 160 0 156.418 0 152V8Z" fill="#1C1C1C"/> | ||||
| <path d="M8 0.5H152C156.142 0.500001 159.5 3.85787 159.5 8V152C159.5 156.142 156.142 159.5 152 159.5H8C3.85787 159.5 0.5 156.142 0.5 152V8C0.500001 3.85787 3.85787 0.5 8 0.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M8 13.5C8 10.4624 10.4624 8 13.5 8H146.5C149.538 8 152 10.4624 152 13.5V13.5C152 16.5376 149.538 19 146.5 19H13.5C10.4624 19 8 16.5376 8 13.5V13.5Z" fill="white" fill-opacity="0.48"/> | ||||
| <path d="M8 35C8 30.5817 11.5817 27 16 27H45.3333C49.7516 27 53.3333 30.5817 53.3333 35V125C53.3333 129.418 49.7516 133 45.3333 133H16C11.5817 133 8 129.418 8 125V35Z" fill="#1C1C1C"/> | ||||
| <path d="M16 27.5H45.333C49.4751 27.5 52.833 30.8579 52.833 35V125C52.833 129.142 49.4751 132.5 45.333 132.5H16C11.8579 132.5 8.5 129.142 8.5 125V35C8.5 30.8579 11.8579 27.5 16 27.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M57.3333 35C57.3333 30.5817 60.9151 27 65.3333 27H94.6667C99.0849 27 102.667 30.5817 102.667 35V49C102.667 53.4183 99.0849 57 94.6667 57H65.3333C60.915 57 57.3333 53.4183 57.3333 49V35Z" fill="#1C1C1C"/> | ||||
| <path d="M65.3333 27.5H94.6663C98.8085 27.5 102.166 30.8579 102.166 35V49C102.166 53.1421 98.8085 56.5 94.6663 56.5H65.3333C61.1912 56.5 57.8333 53.1421 57.8333 49V35C57.8333 30.8579 61.1912 27.5 65.3333 27.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M106.667 35C106.667 30.5817 110.248 27 114.667 27H144C148.418 27 152 30.5817 152 35V87C152 91.4183 148.418 95 144 95H114.667C110.248 95 106.667 91.4183 106.667 87V35Z" fill="#1C1C1C"/> | ||||
| <path d="M114.667 27.5H144C148.142 27.5 151.5 30.8579 151.5 35V87C151.5 91.1421 148.142 94.5 144 94.5H114.667C110.525 94.5 107.167 91.1421 107.167 87V35C107.167 30.8579 110.525 27.5 114.667 27.5Z" stroke="white" stroke-opacity="0.24"/> | ||||
| <path d="M84.3594 82.0156H87.7344C87.9219 81.1406 88.0156 80.4688 88.0156 80C88.0156 79.5312 87.9219 78.8594 87.7344 77.9844H84.3594C84.4531 78.6406 84.5 79.3125 84.5 80C84.5 80.6875 84.4531 81.3594 84.3594 82.0156ZM82.5781 87.5469C83.3594 87.2969 84.1719 86.8281 85.0156 86.1406C85.8594 85.4219 86.5 84.7031 86.9375 83.9844H83.9844C83.6719 85.2344 83.2031 86.4219 82.5781 87.5469ZM82.3438 82.0156C82.4375 81.3594 82.4844 80.6875 82.4844 80C82.4844 79.3125 82.4375 78.6406 82.3438 77.9844H77.6562C77.5625 78.6406 77.5156 79.3125 77.5156 80C77.5156 80.6875 77.5625 81.3594 77.6562 82.0156H82.3438ZM80 87.9688C80.875 86.6875 81.5156 85.3594 81.9219 83.9844H78.0781C78.4844 85.3594 79.125 86.6875 80 87.9688ZM76.0156 76.0156C76.3906 74.6719 76.8594 73.4844 77.4219 72.4531C76.6406 72.7031 75.8125 73.1875 74.9375 73.9062C74.0938 74.5938 73.4688 75.2969 73.0625 76.0156H76.0156ZM73.0625 83.9844C73.4688 84.7031 74.0938 85.4219 74.9375 86.1406C75.8125 86.8281 76.6406 87.2969 77.4219 87.5469C76.7969 86.4219 76.3281 85.2344 76.0156 83.9844H73.0625ZM72.2656 82.0156H75.6406C75.5469 81.3594 75.5 80.6875 75.5 80C75.5 79.3125 75.5469 78.6406 75.6406 77.9844H72.2656C72.0781 78.8594 71.9844 79.5312 71.9844 80C71.9844 80.4688 72.0781 81.1406 72.2656 82.0156ZM80 72.0312C79.125 73.3125 78.4844 74.6406 78.0781 76.0156H81.9219C81.5156 74.6406 80.875 73.3125 80 72.0312ZM86.9375 76.0156C86.5 75.2969 85.8594 74.5938 85.0156 73.9062C84.1719 73.1875 83.3594 72.7031 82.5781 72.4531C83.1406 73.4844 83.6094 74.6719 83.9844 76.0156H86.9375ZM72.9219 72.9688C74.8906 71 77.25 70.0156 80 70.0156C82.75 70.0156 85.0938 71 87.0312 72.9688C89 74.9062 89.9844 77.25 89.9844 80C89.9844 82.75 89 85.1094 87.0312 87.0781C85.0938 89.0156 82.75 89.9844 80 89.9844C77.25 89.9844 74.8906 89.0156 72.9219 87.0781C70.9844 85.1094 70.0156 82.75 70.0156 80C70.0156 77.25 70.9844 74.9062 72.9219 72.9688Z" fill="#03A9F4"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_1738_5538"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 3.9 KiB | 
| @@ -1,32 +0,0 @@ | ||||
| <svg width="160" height="160" viewBox="0 0 160 160" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <g clip-path="url(#clip0_3969_50764)"> | ||||
| <path d="M0 4C0 1.79086 1.79086 0 4 0H16C18.2091 0 20 1.79086 20 4V4C20 6.20914 18.2091 8 16 8H4C1.79086 8 0 6.20914 0 4V4Z" fill="black" fill-opacity="0.32"/> | ||||
| <path d="M0 20C0 15.5817 3.58172 12 8 12H68C72.4183 12 76 15.5817 76 20V36C76 40.4183 72.4183 44 68 44H8C3.58172 44 0 40.4183 0 36V20Z" fill="white"/> | ||||
| <path d="M8 12.5H68C72.1421 12.5 75.5 15.8579 75.5 20V36C75.5 40.1421 72.1421 43.5 68 43.5H8C3.85786 43.5 0.5 40.1421 0.5 36V20C0.5 15.8579 3.85786 12.5 8 12.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M32.9844 27.0156C32.9844 26.0781 32.7031 25.2656 32.1406 24.5781C31.5781 23.8594 30.8594 23.375 29.9844 23.125V22C29.9844 21.4375 30.125 20.9375 30.4062 20.5C30.6875 20.0312 31.0469 19.6719 31.4844 19.4219C31.9531 19.1406 32.4531 19 32.9844 19H43.0156C43.5469 19 44.0312 19.1406 44.4688 19.4219C44.9375 19.6719 45.3125 20.0312 45.5938 20.5C45.875 20.9375 46.0156 21.4375 46.0156 22V23.125C45.1406 23.375 44.4219 23.8594 43.8594 24.5781C43.2969 25.2656 43.0156 26.0781 43.0156 27.0156V28.9844H32.9844V27.0156ZM47 25C47.5625 25 48.0312 25.2031 48.4062 25.6094C48.8125 25.9844 49.0156 26.4531 49.0156 27.0156V31.9844C49.0156 32.5469 48.875 33.0625 48.5938 33.5312C48.3125 33.9688 47.9375 34.3281 47.4688 34.6094C47.0312 34.8594 46.5469 34.9844 46.0156 34.9844V36.0156C46.0156 36.2656 45.9062 36.5 45.6875 36.7188C45.5 36.9062 45.2656 37 44.9844 37C44.7344 37 44.5 36.9062 44.2812 36.7188C44.0938 36.5 44 36.2656 44 36.0156V34.9844H32V36.0156C32 36.2656 31.8906 36.5 31.6719 36.7188C31.4844 36.9062 31.2656 37 31.0156 37C30.7344 37 30.4844 36.9062 30.2656 36.7188C30.0781 36.5 29.9844 36.2656 29.9844 36.0156V34.9844C29.4531 34.9844 28.9531 34.8594 28.4844 34.6094C28.0469 34.3281 27.6875 33.9688 27.4062 33.5312C27.125 33.0625 26.9844 32.5469 26.9844 31.9844V27.0156C26.9844 26.4531 27.1719 25.9844 27.5469 25.6094C27.9531 25.2031 28.4375 25 29 25C29.5625 25 30.0312 25.2031 30.4062 25.6094C30.8125 25.9844 31.0156 26.4531 31.0156 27.0156V31H44.9844V27.0156C44.9844 26.4531 45.1719 25.9844 45.5469 25.6094C45.9531 25.2031 46.4375 25 47 25Z" fill="#03A9F4"/> | ||||
| <path d="M0 56C0 51.5817 3.58172 48 8 48H68C72.4183 48 76 51.5817 76 56V72C76 76.4183 72.4183 80 68 80H8C3.58172 80 0 76.4183 0 72V56Z" fill="white"/> | ||||
| <path d="M8 48.5H68C72.1421 48.5 75.5 51.8579 75.5 56V72C75.5 76.1421 72.1421 79.5 68 79.5H8C3.85786 79.5 0.5 76.1421 0.5 72V56C0.5 51.8579 3.85786 48.5 8 48.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M44 61.9844H47.9844V64H46.0156V72.0156H29.9844V64H28.0156V61.9844H32C31.4375 61.9844 30.9531 61.7969 30.5469 61.4219C30.1719 61.0156 29.9844 60.5469 29.9844 60.0156V55.9844H35.9844V60.0156C35.9844 60.5469 35.7812 61.0156 35.375 61.4219C35 61.7969 34.5469 61.9844 34.0156 61.9844H41.9844V58.9844C41.9844 58.7344 41.8906 58.5156 41.7031 58.3281C41.5156 58.1094 41.2812 58 41 58C40.7188 58 40.4844 58.1094 40.2969 58.3281C40.1094 58.5156 40.0156 58.7344 40.0156 58.9844H38C38 58.4531 38.125 57.9688 38.375 57.5312C38.6562 57.0625 39.0156 56.6875 39.4531 56.4062C39.9219 56.125 40.4375 55.9844 41 55.9844C41.5625 55.9844 42.0625 56.125 42.5 56.4062C42.9688 56.6875 43.3281 57.0625 43.5781 57.5312C43.8594 57.9688 44 58.4531 44 58.9844V61.9844ZM38.9844 70V64H37.0156V70H38.9844Z" fill="#03A9F4"/> | ||||
| <path d="M0 92C0 87.5817 3.58172 84 8 84H68C72.4183 84 76 87.5817 76 92V108C76 112.418 72.4183 116 68 116H8C3.58172 116 0 112.418 0 108V92Z" fill="white"/> | ||||
| <path d="M8 84.5H68C72.1421 84.5 75.5 87.8579 75.5 92V108C75.5 112.142 72.1421 115.5 68 115.5H8C3.85786 115.5 0.5 112.142 0.5 108V92C0.5 87.8579 3.85786 84.5 8 84.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M44.9844 94.9844C46.0781 94.9844 47.0156 95.3906 47.7969 96.2031C48.6094 96.9844 49.0156 97.9219 49.0156 99.0156V108.016H47V105.016H29V108.016H26.9844V93.0156H29V102.016H37.0156V94.9844H44.9844ZM35.0938 100.094C34.5 100.688 33.7969 100.984 32.9844 100.984C32.1719 100.984 31.4688 100.688 30.875 100.094C30.2812 99.5 29.9844 98.7969 29.9844 97.9844C29.9844 97.1719 30.2812 96.4688 30.875 95.875C31.4688 95.2812 32.1719 94.9844 32.9844 94.9844C33.7969 94.9844 34.5 95.2812 35.0938 95.875C35.6875 96.4688 35.9844 97.1719 35.9844 97.9844C35.9844 98.7969 35.6875 99.5 35.0938 100.094Z" fill="#03A9F4"/> | ||||
| <path d="M0 128C0 123.582 3.58172 120 8 120H68C72.4183 120 76 123.582 76 128V144C76 148.418 72.4183 152 68 152H8C3.58172 152 0 148.418 0 144V128Z" fill="white"/> | ||||
| <path d="M8 120.5H68C72.1421 120.5 75.5 123.858 75.5 128V144C75.5 148.142 72.1421 151.5 68 151.5H8C3.85786 151.5 0.5 148.142 0.5 144V128C0.5 123.858 3.85786 120.5 8 120.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M46.0156 136.984H47.9844V142.984C47.9844 143.516 47.7812 143.984 47.375 144.391C47 144.797 46.5469 145 46.0156 145C46.0156 145.281 45.9062 145.516 45.6875 145.703C45.5 145.891 45.2656 145.984 44.9844 145.984H31.0156C30.7344 145.984 30.4844 145.891 30.2656 145.703C30.0781 145.516 29.9844 145.281 29.9844 145C29.4531 145 28.9844 144.797 28.5781 144.391C28.2031 143.984 28.0156 143.516 28.0156 142.984V136.984H31.0156V136.234C31.0156 135.641 31.2344 135.125 31.6719 134.688C32.1406 134.219 32.6719 133.984 33.2656 133.984C33.8906 133.984 34.4531 134.234 34.9531 134.734L36.3125 136.281C36.5 136.5 36.7812 136.734 37.1562 136.984H44V128.828C44 128.609 43.9219 128.422 43.7656 128.266C43.6094 128.078 43.4062 127.984 43.1562 127.984C42.9375 127.984 42.75 128.062 42.5938 128.219L41.3281 129.484C41.3906 129.734 41.4219 129.906 41.4219 130C41.4219 130.344 41.3125 130.703 41.0938 131.078L38.3281 128.312C38.7031 128.094 39.0625 127.984 39.4062 127.984C39.5625 127.984 39.7344 128.016 39.9219 128.078L41.1875 126.812C41.7188 126.281 42.375 126.016 43.1562 126.016C43.9375 126.016 44.6094 126.297 45.1719 126.859C45.7344 127.391 46.0156 128.047 46.0156 128.828V136.984ZM31.5781 132.438C31.2031 132.031 31.0156 131.547 31.0156 130.984C31.0156 130.422 31.2031 129.953 31.5781 129.578C31.9531 129.203 32.4219 129.016 32.9844 129.016C33.5469 129.016 34.0156 129.203 34.3906 129.578C34.7969 129.953 35 130.422 35 130.984C35 131.547 34.7969 132.031 34.3906 132.438C34.0156 132.812 33.5469 133 32.9844 133C32.4219 133 31.9531 132.812 31.5781 132.438Z" fill="#03A9F4"/> | ||||
| <path d="M84 4C84 1.79086 85.7909 0 88 0H100C102.209 0 104 1.79086 104 4V4C104 6.20914 102.209 8 100 8H88C85.7909 8 84 6.20914 84 4V4Z" fill="black" fill-opacity="0.32"/> | ||||
| <path d="M84 20C84 15.5817 87.5817 12 92 12H152C156.418 12 160 15.5817 160 20V36C160 40.4183 156.418 44 152 44H92C87.5817 44 84 40.4183 84 36V20Z" fill="white"/> | ||||
| <path d="M92 12.5H152C156.142 12.5 159.5 15.8579 159.5 20V36C159.5 40.1421 156.142 43.5 152 43.5H92C87.8579 43.5 84.5 40.1421 84.5 36V20C84.5 15.8579 87.8579 12.5 92 12.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M131.984 30.0156C131.984 30.7656 131.797 31.4531 131.422 32.0781C131.047 32.6719 130.562 33.1406 129.969 33.4844C129.844 34.2031 129.5 34.8125 128.938 35.3125C128.406 35.7812 127.766 36.0156 127.016 36.0156C126.359 36.0156 125.766 35.8281 125.234 35.4531C124.734 35.0781 124.391 34.5938 124.203 34H119.797C119.609 34.5938 119.25 35.0781 118.719 35.4531C118.219 35.8281 117.641 36.0156 116.984 36.0156C116.234 36.0156 115.578 35.7812 115.016 35.3125C114.484 34.8125 114.156 34.2031 114.031 33.4844C113.438 33.1406 112.953 32.6719 112.578 32.0781C112.203 31.4531 112.016 30.7656 112.016 30.0156C112.016 29.1094 112.266 28.3125 112.766 27.625C113.297 26.9375 113.969 26.4688 114.781 26.2188L113 24.3906L112.719 24.7188C112.5 24.9062 112.25 25 111.969 25C111.719 25 111.5 24.9062 111.312 24.7188C111.094 24.5312 110.984 24.2969 110.984 24.0156C110.984 23.7344 111.094 23.5 111.312 23.3125L113.281 21.2969C113.469 21.1094 113.703 21.0156 113.984 21.0156C114.266 21.0156 114.5 21.1094 114.688 21.2969C114.906 21.4844 115.016 21.7188 115.016 22C115.016 22.2812 114.906 22.5156 114.688 22.7031L114.406 22.9844L115.812 24.3906L116.609 22.0469C116.797 21.4219 117.156 20.9219 117.688 20.5469C118.219 20.1719 118.797 19.9844 119.422 19.9844H124.578C125.203 19.9844 125.781 20.1719 126.312 20.5469C126.844 20.9219 127.203 21.4219 127.391 22.0469L128.75 26.0781C129.375 26.2031 129.922 26.4531 130.391 26.8281C130.891 27.2031 131.281 27.6719 131.562 28.2344C131.844 28.7656 131.984 29.3594 131.984 30.0156ZM116.984 34C117.266 34 117.5 33.9062 117.688 33.7188C117.906 33.5 118.016 33.2656 118.016 33.0156C118.016 32.7344 117.906 32.5 117.688 32.3125C117.5 32.0938 117.266 31.9844 116.984 31.9844C116.734 31.9844 116.5 32.0938 116.281 32.3125C116.094 32.5 116 32.7344 116 33.0156C116 33.2656 116.094 33.5 116.281 33.7188C116.5 33.9062 116.734 34 116.984 34ZM121.016 25.9844V22H119.422C118.953 22 118.641 22.2344 118.484 22.7031L117.406 25.9844H121.016ZM122.984 22V25.9844H126.594L125.516 22.7031C125.359 22.2344 125.047 22 124.578 22H122.984ZM127.016 34C127.266 34 127.484 33.9062 127.672 33.7188C127.891 33.5 128 33.2656 128 33.0156C128 32.7344 127.891 32.5 127.672 32.3125C127.484 32.0938 127.266 31.9844 127.016 31.9844C126.734 31.9844 126.484 32.0938 126.266 32.3125C126.078 32.5 125.984 32.7344 125.984 33.0156C125.984 33.2656 126.078 33.5 126.266 33.7188C126.484 33.9062 126.734 34 127.016 34Z" fill="#03A9F4"/> | ||||
| <path d="M84 56C84 51.5817 87.5817 48 92 48H152C156.418 48 160 51.5817 160 56V72C160 76.4183 156.418 80 152 80H92C87.5817 80 84 76.4183 84 72V56Z" fill="white"/> | ||||
| <path d="M92 48.5H152C156.142 48.5 159.5 51.8579 159.5 56V72C159.5 76.1421 156.142 79.5 152 79.5H92C87.8579 79.5 84.5 76.1421 84.5 72V56C84.5 51.8579 87.8579 48.5 92 48.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M128.984 58.9844C130.078 58.9844 131.016 59.3906 131.797 60.2031C132.609 60.9844 133.016 61.9219 133.016 63.0156V72.0156H131V69.0156H113V72.0156H110.984V57.0156H113V66.0156H121.016V58.9844H128.984ZM119.094 64.0938C118.5 64.6875 117.797 64.9844 116.984 64.9844C116.172 64.9844 115.469 64.6875 114.875 64.0938C114.281 63.5 113.984 62.7969 113.984 61.9844C113.984 61.1719 114.281 60.4688 114.875 59.875C115.469 59.2812 116.172 58.9844 116.984 58.9844C117.797 58.9844 118.5 59.2812 119.094 59.875C119.688 60.4688 119.984 61.1719 119.984 61.9844C119.984 62.7969 119.688 63.5 119.094 64.0938Z" fill="#03A9F4"/> | ||||
| <path d="M84 92C84 87.5817 87.5817 84 92 84H152C156.418 84 160 87.5817 160 92V108C160 112.418 156.418 116 152 116H92C87.5817 116 84 112.418 84 108V92Z" fill="white"/> | ||||
| <path d="M92 84.5H152C156.142 84.5 159.5 87.8579 159.5 92V108C159.5 112.142 156.142 115.5 152 115.5H92C87.8579 115.5 84.5 112.142 84.5 108V92C84.5 87.8579 87.8579 84.5 92 84.5Z" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M112.016 94H131.984V106H130.016V103.984H125.984V106H124.016V96.0156H113.984V106H112.016V94ZM130.016 96.0156H125.984V97.9844H130.016V96.0156ZM125.984 102.016H130.016V100H125.984V102.016Z" fill="#03A9F4"/> | ||||
| </g> | ||||
| <defs> | ||||
| <clipPath id="clip0_3969_50764"> | ||||
| <rect width="160" height="160" fill="white"/> | ||||
| </clipPath> | ||||
| </defs> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 11 KiB |