mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 03:29:41 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			20251001.3
			...
			fix_new_sc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 637eb6e894 | 
| @@ -1,4 +1,5 @@ | ||||
| FROM mcr.microsoft.com/devcontainers/python:1-3.13 | ||||
| # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | ||||
| FROM mcr.microsoft.com/devcontainers/python:3.12 | ||||
|  | ||||
| ENV \ | ||||
|   DEBIAN_FRONTEND=noninteractive \ | ||||
|   | ||||
| @@ -5,15 +5,12 @@ | ||||
|     "context": ".." | ||||
|   }, | ||||
|   "appPort": "8124:8123", | ||||
|   "postCreateCommand": "./.devcontainer/post_create.sh", | ||||
|   "postCreateCommand": "sudo apt update && sudo apt upgrade -y && sudo apt install -y libpcap-dev", | ||||
|   "postStartCommand": "script/bootstrap", | ||||
|   "containerEnv": { | ||||
|     "DEV_CONTAINER": "1", | ||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" | ||||
|   }, | ||||
|   "remoteEnv": { | ||||
|     "NODE_OPTIONS": "--max_old_space_size=8192" | ||||
|   }, | ||||
|   "customizations": { | ||||
|     "vscode": { | ||||
|       "extensions": [ | ||||
| @@ -21,8 +18,7 @@ | ||||
|         "esbenp.prettier-vscode", | ||||
|         "runem.lit-plugin", | ||||
|         "github.vscode-pull-request-github", | ||||
|         "eamodio.gitlens", | ||||
|         "yeion7.styled-global-variables-autocomplete" | ||||
|         "eamodio.gitlens" | ||||
|       ], | ||||
|       "settings": { | ||||
|         "files.eol": "\n", | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| # This script will run after the container is created | ||||
|  | ||||
| # add github cli | ||||
| (type -p wget >/dev/null || (sudo apt update && sudo apt-get install wget -y)) \ | ||||
| 	&& sudo mkdir -p -m 755 /etc/apt/keyrings \ | ||||
|         && out=$(mktemp) && wget -nv -O$out https://cli.github.com/packages/githubcli-archive-keyring.gpg \ | ||||
|         && cat $out | sudo tee /etc/apt/keyrings/githubcli-archive-keyring.gpg > /dev/null \ | ||||
| 	&& sudo chmod go+r /etc/apt/keyrings/githubcli-archive-keyring.gpg \ | ||||
| 	&& echo "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/githubcli-archive-keyring.gpg] https://cli.github.com/packages stable main" | sudo tee /etc/apt/sources.list.d/github-cli.list > /dev/null | ||||
|  | ||||
| # Update package lists | ||||
| sudo apt-get update | ||||
|  | ||||
| sudo apt upgrade -y | ||||
|  | ||||
| # Install necessary packages | ||||
| sudo apt-get install -y libpcap-dev gh | ||||
|  | ||||
| # Display a message | ||||
| echo "Post-create script has been executed successfully." | ||||
							
								
								
									
										10
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -7,11 +7,11 @@ body: | ||||
|       value: | | ||||
|         Make sure you are running the [latest version of Home Assistant][releases] before reporting an issue. | ||||
|  | ||||
|         If you have a feature or enhancement request for the frontend, please [start a discussion][fr] instead of creating an issue. | ||||
|         If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue. | ||||
|  | ||||
|         **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: | ||||
| @@ -74,7 +74,7 @@ body: | ||||
|         If known, otherwise leave blank. | ||||
|   - type: input | ||||
|     attributes: | ||||
|       label: In which browser are you experiencing the issue? | ||||
|       label: In which browser are you experiencing the issue with? | ||||
|       placeholder: Google Chrome 88.0.4324.150 | ||||
|       description: > | ||||
|         Provide the full name and don't forget to add the version! | ||||
| @@ -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 | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,8 +1,8 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
|   - name: Request a feature for the UI / Dashboards | ||||
|     url: https://github.com/orgs/home-assistant/discussions | ||||
|     about: Request a new feature for the Home Assistant frontend. | ||||
|     url: https://github.com/home-assistant/frontend/discussions/category_choices | ||||
|     about: Request an 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 | ||||
|     about: This is the issue tracker for our frontend. Please report other issues in the backend ("core") repository. | ||||
|   | ||||
							
								
								
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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
									
									
								
							
							
						
						
									
										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 | ||||
							
								
								
									
										18
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -41,8 +41,9 @@ jobs: | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --alias dev | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=cast/dist --alias dev | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
| @@ -56,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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -76,8 +77,9 @@ jobs: | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --prod | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=cast/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										56
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										56
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -24,20 +24,26 @@ 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 | ||||
|         id: setup-node | ||||
|         uses: actions/setup-node@v4.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|       - uses: actions/cache@v4.2.0 | ||||
|         id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`) | ||||
|         with: | ||||
|           path: "node_modules" | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }}-${{ hashFiles('yarn.lock') }} | ||||
|           restore-keys: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }} | ||||
|       - name: Install dependencies | ||||
|         if: steps.yarn-cache.outputs.cache-hit != 'true' | ||||
|         run: yarn install --immutable | ||||
|       - name: Check for duplicate dependencies | ||||
|         run: yarn dedupe --check | ||||
|       - 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.0 | ||||
|         with: | ||||
|           path: | | ||||
|             node_modules/.cache/prettier | ||||
| @@ -58,13 +64,21 @@ 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 | ||||
|         id: setup-node | ||||
|         uses: actions/setup-node@v4.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|       - uses: actions/cache@v4.2.0 | ||||
|         id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`) | ||||
|         with: | ||||
|           path: "node_modules" | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }}-${{ hashFiles('yarn.lock') }} | ||||
|           restore-keys: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }} | ||||
|       - name: Install dependencies | ||||
|         if: steps.yarn-cache.outputs.cache-hit != 'true' | ||||
|         run: yarn install --immutable | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data | ||||
| @@ -76,20 +90,28 @@ 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 | ||||
|         id: setup-node | ||||
|         uses: actions/setup-node@v4.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|       - uses: actions/cache@v4.2.0 | ||||
|         id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`) | ||||
|         with: | ||||
|           path: "node_modules" | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }}-${{ hashFiles('yarn.lock') }} | ||||
|           restore-keys: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }} | ||||
|       - name: Install dependencies | ||||
|         if: steps.yarn-cache.outputs.cache-hit != 'true' | ||||
|         run: yarn install --immutable | ||||
|       - name: Build Application | ||||
|         run: ./node_modules/.bin/gulp build-app | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.4.3 | ||||
|         with: | ||||
|           name: frontend-bundle-stats | ||||
|           path: build/stats/*.json | ||||
| @@ -100,20 +122,28 @@ 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 | ||||
|         id: setup-node | ||||
|         uses: actions/setup-node@v4.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|       - uses: actions/cache@v4.2.0 | ||||
|         id: yarn-cache # use this to check for `cache-hit` (`steps.yarn-cache.outputs.cache-hit != 'true'`) | ||||
|         with: | ||||
|           path: "node_modules" | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }}-${{ hashFiles('yarn.lock') }} | ||||
|           restore-keys: ${{ runner.os }}-yarn-${{ hashFiles('.yarnrc.yml') }}-${{ steps.setup-node.outputs.node-version }} | ||||
|       - name: Install dependencies | ||||
|         if: steps.yarn-cache.outputs.cache-hit != 'true' | ||||
|         run: yarn install --immutable | ||||
|       - name: Build Application | ||||
|         run: ./node_modules/.bin/gulp build-hassio | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.4.3 | ||||
|         with: | ||||
|           name: supervisor-bundle-stats | ||||
|           path: build/stats/*.json | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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 | ||||
|   | ||||
							
								
								
									
										18
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -42,8 +42,9 @@ jobs: | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} | ||||
| @@ -57,12 +58,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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -77,8 +78,9 @@ jobs: | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										9
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -34,8 +34,9 @@ jobs: | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         run: | | ||||
|           npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --prod | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=gallery/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||
|   | ||||
							
								
								
									
										13
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.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.1.0 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -39,14 +39,13 @@ 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 }}" \ | ||||
|             --json > deploy_output.json | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||
|  | ||||
|       - name: Generate summary | ||||
|         run: | | ||||
|           NETLIFY_LIVE_URL=$(jq -r '.deploy_url' deploy_output.json) | ||||
|           echo "$NETLIFY_LIVE_URL" >> "$GITHUB_STEP_SUMMARY" | ||||
|           echo "${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}" >> "$GITHUB_STEP_SUMMARY" | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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
									
									
								
							
							
						
						
									
										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" | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | ||||
|     - cron: "0 1 * * *" | ||||
|  | ||||
| env: | ||||
|   PYTHON_VERSION: "3.13" | ||||
|   PYTHON_VERSION: "3.12" | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| permissions: | ||||
| @@ -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.1.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.4.3 | ||||
|         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.4.3 | ||||
|         with: | ||||
|           name: translations | ||||
|           path: translations.tar.gz | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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.1.14 | ||||
|         with: | ||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} | ||||
|           token: ${{ github.token }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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.0.0 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|   | ||||
							
								
								
									
										35
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										35
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | ||||
|       - published | ||||
|  | ||||
| env: | ||||
|   PYTHON_VERSION: "3.13" | ||||
|   PYTHON_VERSION: "3.12" | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| # Set default workflow permissions | ||||
| @@ -23,18 +23,18 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         with: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Verify version | ||||
|         uses: home-assistant/actions/helpers/verify-version@master | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         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.1.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.1.0 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
| @@ -73,11 +73,10 @@ 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@2024.11.0 | ||||
|         with: | ||||
|           abi: cp313 | ||||
|           abi: cp312 | ||||
|           tag: musllinux_1_2 | ||||
|           arch: amd64 | ||||
|           wheels-key: ${{ secrets.WHEELS_KEY }} | ||||
| @@ -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.1.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.1.0 | ||||
|         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.1.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.1.0 | ||||
|         with: | ||||
|           files: hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz | ||||
|   | ||||
							
								
								
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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
									
									
								
							
							
						
						
									
										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.0.0 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 90 | ||||
|   | ||||
							
								
								
									
										3
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										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
									
									
								
							
							
						
						
									
										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" | ||||
|   | ||||
							
								
								
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -5,7 +5,6 @@ | ||||
|     "runem.lit-plugin", | ||||
|     "github.vscode-pull-request-github", | ||||
|     "eamodio.gitlens", | ||||
|     "vitest.explorer", | ||||
|     "yeion7.styled-global-variables-autocomplete" | ||||
|     "vitest.explorer" | ||||
|   ] | ||||
| } | ||||
|   | ||||
							
								
								
									
										42
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,42 +1,6 @@ | ||||
| { | ||||
|   "version": "2.0.0", | ||||
|   "tasks": [ | ||||
|     { | ||||
|       "label": "Develop and serve Frontend", | ||||
|       "type": "shell", | ||||
|       "command": "script/develop_and_serve -c ${input:coreUrl}", | ||||
|       // Sync changes here to other tasks until issue resolved | ||||
|       // https://github.com/Microsoft/vscode/issues/61497 | ||||
|       "problemMatcher": { | ||||
|         "owner": "ha-build", | ||||
|         "source": "ha-build", | ||||
|         "fileLocation": "absolute", | ||||
|         "severity": "error", | ||||
|         "pattern": [ | ||||
|           { | ||||
|             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||
|             "severity": 1, | ||||
|             "file": 2, | ||||
|             "message": 3, | ||||
|             "line": 4, | ||||
|             "column": 5 | ||||
|           } | ||||
|         ], | ||||
|         "background": { | ||||
|           "activeOnStart": true, | ||||
|           "beginsPattern": "Changes detected. Starting compilation", | ||||
|           "endsPattern": "Build done @" | ||||
|         } | ||||
|       }, | ||||
|       "isBackground": true, | ||||
|       "group": { | ||||
|         "kind": "build", | ||||
|         "isDefault": true | ||||
|       }, | ||||
|       "runOptions": { | ||||
|         "instanceLimit": 1 | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "label": "Develop Frontend", | ||||
|       "type": "gulp", | ||||
| @@ -277,12 +241,6 @@ | ||||
|       "id": "supervisorToken", | ||||
|       "type": "promptString", | ||||
|       "description": "The token for the Remote API proxy add-on" | ||||
|     }, | ||||
|     { | ||||
|       "id": "coreUrl", | ||||
|       "type": "promptString", | ||||
|       "description": "The URL of the Home Assistant Core instance", | ||||
|       "default": "http://127.0.0.1:8123" | ||||
|     } | ||||
|   ] | ||||
| } | ||||
|   | ||||
| @@ -1,22 +0,0 @@ | ||||
| diff --git a/mwc-formfield-base.js b/mwc-formfield-base.js | ||||
| index 7b763326d7d51835ad52646bfbc80fe21989abd3..f2baa8224e6d03df1fdb0b9fd03f5c6d77fc8747 100644 | ||||
| --- a/mwc-formfield-base.js | ||||
| +++ b/mwc-formfield-base.js | ||||
| @@ -9,7 +9,7 @@ import { BaseElement } from '@material/mwc-base/base-element.js'; | ||||
|  import { FormElement } from '@material/mwc-base/form-element.js'; | ||||
|  import { observer } from '@material/mwc-base/observer.js'; | ||||
|  import { html } from 'lit'; | ||||
| -import { property, query, queryAssignedNodes } from 'lit/decorators.js'; | ||||
| +import { property, query, queryAssignedElements } from 'lit/decorators.js'; | ||||
|  import { classMap } from 'lit/directives/class-map.js'; | ||||
|  export class FormfieldBase extends BaseElement { | ||||
|      constructor() { | ||||
| @@ -96,7 +96,7 @@ __decorate([ | ||||
|      query('.mdc-form-field') | ||||
|  ], FormfieldBase.prototype, "mdcRoot", void 0); | ||||
|  __decorate([ | ||||
| -    queryAssignedNodes('', true, '*') | ||||
| +    queryAssignedElements({ slot: "", flatten: true, selector: "*" }) | ||||
|  ], FormfieldBase.prototype, "slottedInputs", void 0); | ||||
|  __decorate([ | ||||
|      query('label') | ||||
| @@ -1,26 +0,0 @@ | ||||
| diff --git a/mwc-list-base.js b/mwc-list-base.js | ||||
| index 1ba95b6a01dcecea4d85b5cbbbcc3dfb04c40d5f..dced13fdb7929c490d6661b1bbe7e9f96dcd2285 100644 | ||||
| --- a/mwc-list-base.js | ||||
| +++ b/mwc-list-base.js | ||||
| @@ -11,7 +11,7 @@ import { BaseElement } from '@material/mwc-base/base-element.js'; | ||||
|  import { observer } from '@material/mwc-base/observer.js'; | ||||
|  import { deepActiveElementPath, doesElementContainFocus, isNodeElement } from '@material/mwc-base/utils.js'; | ||||
|  import { html } from 'lit'; | ||||
| -import { property, query, queryAssignedNodes } from 'lit/decorators.js'; | ||||
| +import { property, query, queryAssignedElements } from 'lit/decorators.js'; | ||||
|  import { ifDefined } from 'lit/directives/if-defined.js'; | ||||
|  import MDCListFoundation, { isIndexSet } from './mwc-list-foundation.js'; | ||||
|  export { createSetFromIndex, isEventMulti, isIndexSet } from './mwc-list-foundation.js'; | ||||
| @@ -425,10 +425,10 @@ __decorate([ | ||||
|      query('.mdc-deprecated-list') | ||||
|  ], ListBase.prototype, "mdcRoot", void 0); | ||||
|  __decorate([ | ||||
| -    queryAssignedNodes('', true, '*') | ||||
| +    queryAssignedElements({ flatten: true, selector: "*" }) | ||||
|  ], ListBase.prototype, "assignedElements", void 0); | ||||
|  __decorate([ | ||||
| -    queryAssignedNodes('', true, '[tabindex="0"]') | ||||
| +    queryAssignedElements({ flatten: true, selector: '[tabindex="0"]' }) | ||||
|  ], ListBase.prototype, "tabbableElements", void 0); | ||||
|  __decorate([ | ||||
|      property({ type: Boolean }), | ||||
							
								
								
									
										34
									
								
								.yarn/patches/@polymer/polymer/pr-5569.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.yarn/patches/@polymer/polymer/pr-5569.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| diff --git a/lib/legacy/class.js b/lib/legacy/class.js | ||||
| index aee2511be1cd9bf900ee552bc98190c1631c57c0..f2f499d68bf52034cac9c28307c99e8ce6b8417d 100644 | ||||
| --- a/lib/legacy/class.js | ||||
| +++ b/lib/legacy/class.js | ||||
| @@ -304,17 +304,23 @@ function GenerateClassFromInfo(info, Base, behaviors) { | ||||
|        // only proceed if the generated class' prototype has not been registered. | ||||
|        const generatedProto = PolymerGenerated.prototype; | ||||
|        if (!generatedProto.hasOwnProperty(JSCompiler_renameProperty('__hasRegisterFinished', generatedProto))) { | ||||
| -        generatedProto.__hasRegisterFinished = true; | ||||
| +        // make sure legacy lifecycle is called on the *element*'s prototype | ||||
| +        // and not the generated class prototype; if the element has been | ||||
| +        // extended, these are *not* the same. | ||||
| +        const proto = Object.getPrototypeOf(this); | ||||
| +        // Only set flag when generated prototype itself is registered, | ||||
| +        // as this element may be extended from, and needs to run `registered` | ||||
| +        // on all behaviors on the subclass as well. | ||||
| +        if (proto === generatedProto) { | ||||
| +          generatedProto.__hasRegisterFinished = true; | ||||
| +        } | ||||
|          // ensure superclass is registered first. | ||||
|          super._registered(); | ||||
|          // copy properties onto the generated class lazily if we're optimizing, | ||||
| -        if (legacyOptimizations) { | ||||
| +        if (legacyOptimizations && !Object.hasOwnProperty(generatedProto, '__hasCopiedProperties')) { | ||||
| +          generatedProto.__hasCopiedProperties = true; | ||||
|            copyPropertiesToProto(generatedProto); | ||||
|          } | ||||
| -        // make sure legacy lifecycle is called on the *element*'s prototype | ||||
| -        // and not the generated class prototype; if the element has been | ||||
| -        // extended, these are *not* the same. | ||||
| -        const proto = Object.getPrototypeOf(this); | ||||
|          let list = lifecycle.beforeRegister; | ||||
|          if (list) { | ||||
|            for (let i=0; i < list.length; i++) { | ||||
							
								
								
									
										18
									
								
								.yarn/patches/hls.js-npm-1.5.7-f5bbd3d060.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.yarn/patches/hls.js-npm-1.5.7-f5bbd3d060.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| diff --git a/dist/hls.light.mjs b/dist/hls.light.mjs | ||||
| index eed9d788fafdb159975e1a2eb08ac88ba9c9ac33..ace881935e6665946f1c8110ebd2f739cde4427e 100644 | ||||
| --- a/dist/hls.light.mjs | ||||
| +++ b/dist/hls.light.mjs | ||||
| @@ -20523,9 +20523,9 @@ class Hls { | ||||
|  } | ||||
|  Hls.defaultConfig = void 0; | ||||
|   | ||||
| -var KeySystemFormats = empty.KeySystemFormats; | ||||
| -var KeySystems = empty.KeySystems; | ||||
| -var SubtitleStreamController = empty.SubtitleStreamController; | ||||
| -var TimelineController = empty.TimelineController; | ||||
| +var KeySystemFormats = empty; | ||||
| +var KeySystems = empty; | ||||
| +var SubtitleStreamController = empty; | ||||
| +var TimelineController = empty; | ||||
|  export { AbrController, AttrList, Cues as AudioStreamController, Cues as AudioTrackController, BasePlaylistController, BaseSegment, BaseStreamController, BufferController, Cues as CMCDController, CapLevelController, ChunkMetadata, ContentSteeringController, DateRange, Cues as EMEController, ErrorActionFlags, ErrorController, ErrorDetails, ErrorTypes, Events, FPSController, Fragment, Hls, HlsSkip, HlsUrlParameters, KeySystemFormats, KeySystems, Level, LevelDetails, LevelKey, LoadStats, MetadataSchema, NetworkErrorAction, Part, PlaylistLevelType, SubtitleStreamController, Cues as SubtitleTrackController, TimelineController, Hls as default, getMediaSource, isMSESupported, isSupported }; | ||||
|  //# sourceMappingURL=hls.light.mjs.map | ||||
							
								
								
									
										942
									
								
								.yarn/releases/yarn-4.10.2.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										942
									
								
								.yarn/releases/yarn-4.10.2.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										934
									
								
								.yarn/releases/yarn-4.5.3.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										934
									
								
								.yarn/releases/yarn-4.5.3.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -6,4 +6,4 @@ enableGlobalCache: false | ||||
|  | ||||
| nodeLinker: node-modules | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-4.10.2.cjs | ||||
| yarnPath: .yarn/releases/yarn-4.5.3.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 | ||||
| @@ -2,10 +2,10 @@ import defineProvider from "@babel/helper-define-polyfill-provider"; | ||||
| import { join } from "node:path"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const POLYFILL_DIR = join(paths.root_dir, "src/resources/polyfills"); | ||||
| const POLYFILL_DIR = join(paths.polymer_dir, "src/resources/polyfills"); | ||||
|  | ||||
| // List of polyfill keys with supported browser targets for the functionality | ||||
| const polyfillSupport = { | ||||
| const PolyfillSupport = { | ||||
|   // Note states and shadowRoot properties should be supported. | ||||
|   "element-internals": { | ||||
|     android: 90, | ||||
| @@ -18,6 +18,17 @@ const polyfillSupport = { | ||||
|     safari: 17.4, | ||||
|     samsung: 15.0, | ||||
|   }, | ||||
|   "element-append": { | ||||
|     android: 54, | ||||
|     chrome: 54, | ||||
|     edge: 17, | ||||
|     firefox: 49, | ||||
|     ios: 10.0, | ||||
|     opera: 41, | ||||
|     opera_mobile: 41, | ||||
|     safari: 10.0, | ||||
|     samsung: 6.0, | ||||
|   }, | ||||
|   "element-getattributenames": { | ||||
|     android: 61, | ||||
|     chrome: 61, | ||||
| @@ -40,18 +51,27 @@ const polyfillSupport = { | ||||
|     safari: 12.0, | ||||
|     samsung: 10.0, | ||||
|   }, | ||||
|   // FormatJS polyfill detects fix for https://bugs.chromium.org/p/v8/issues/detail?id=10682, | ||||
|   // so adjusted to several months after that was marked fixed | ||||
|   fetch: { | ||||
|     android: 42, | ||||
|     chrome: 42, | ||||
|     edge: 14, | ||||
|     firefox: 39, | ||||
|     ios: 10.3, | ||||
|     opera: 29, | ||||
|     opera_mobile: 29, | ||||
|     safari: 10.1, | ||||
|     samsung: 4.0, | ||||
|   }, | ||||
|   "intl-getcanonicallocales": { | ||||
|     android: 90, | ||||
|     chrome: 90, | ||||
|     edge: 90, | ||||
|     android: 54, | ||||
|     chrome: 54, | ||||
|     edge: 16, | ||||
|     firefox: 48, | ||||
|     ios: 10.3, | ||||
|     opera: 76, | ||||
|     opera_mobile: 64, | ||||
|     opera: 41, | ||||
|     opera_mobile: 41, | ||||
|     safari: 10.1, | ||||
|     samsung: 15.0, | ||||
|     samsung: 6.0, | ||||
|   }, | ||||
|   "intl-locale": { | ||||
|     android: 74, | ||||
| @@ -67,6 +87,17 @@ const polyfillSupport = { | ||||
|   "intl-other": { | ||||
|     // Not specified (i.e. always try polyfill) since compatibility depends on supported locales | ||||
|   }, | ||||
|   proxy: { | ||||
|     android: 49, | ||||
|     chrome: 49, | ||||
|     edge: 12, | ||||
|     firefox: 18, | ||||
|     ios: 10.0, | ||||
|     opera: 36, | ||||
|     opera_mobile: 36, | ||||
|     safari: 10.0, | ||||
|     samsung: 5.0, | ||||
|   }, | ||||
|   "resize-observer": { | ||||
|     android: 64, | ||||
|     chrome: 64, | ||||
| @@ -84,6 +115,8 @@ const polyfillSupport = { | ||||
| // corresponding polyfill key and actual module to import | ||||
| const polyfillMap = { | ||||
|   global: { | ||||
|     fetch: { key: "fetch", module: "unfetch/polyfill" }, | ||||
|     Proxy: { key: "proxy", module: "proxy-polyfill" }, | ||||
|     ResizeObserver: { | ||||
|       key: "resize-observer", | ||||
|       module: join(POLYFILL_DIR, "resize-observer.ts"), | ||||
| @@ -95,7 +128,7 @@ const polyfillMap = { | ||||
|       module: "element-internals-polyfill", | ||||
|     }, | ||||
|     ...Object.fromEntries( | ||||
|       ["getAttributeNames", "toggleAttribute"].map((prop) => { | ||||
|       ["append", "getAttributeNames", "toggleAttribute"].map((prop) => { | ||||
|         const key = `element-${prop.toLowerCase()}`; | ||||
|         return [prop, { key, module: join(POLYFILL_DIR, `${key}.ts`) }]; | ||||
|       }) | ||||
| @@ -135,7 +168,7 @@ export default defineProvider( | ||||
|     const resolvePolyfill = createMetaResolver(polyfillMap); | ||||
|     return { | ||||
|       name: "custom-polyfill", | ||||
|       polyfills: polyfillSupport, | ||||
|       polyfills: PolyfillSupport, | ||||
|       usageGlobal(meta, utils) { | ||||
|         const polyfill = resolvePolyfill(meta); | ||||
|         if (polyfill && shouldInjectPolyfill(polyfill.desc.key)) { | ||||
|   | ||||
| @@ -18,18 +18,30 @@ module.exports.sourceMapURL = () => { | ||||
| module.exports.ignorePackages = () => []; | ||||
|  | ||||
| // Files from NPM packages that we should replace with empty file | ||||
| module.exports.emptyPackages = ({ isHassioBuild }) => | ||||
| module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) => | ||||
|   [ | ||||
|     // Contains all color definitions for all material color sets. | ||||
|     // We don't use it | ||||
|     require.resolve("@polymer/paper-styles/color.js"), | ||||
|     require.resolve("@polymer/paper-styles/default-theme.js"), | ||||
|     // Loads stuff from a CDN | ||||
|     require.resolve("@polymer/font-roboto/roboto.js"), | ||||
|     require.resolve("@vaadin/vaadin-material-styles/typography.js"), | ||||
|     require.resolve("@vaadin/vaadin-material-styles/font-icons.js"), | ||||
|     // Compatibility not needed for latest builds | ||||
|     latestBuild && | ||||
|       // wrapped in require.resolve so it blows up if file no longer exists | ||||
|       require.resolve( | ||||
|         path.resolve(paths.polymer_dir, "src/resources/compatibility.ts") | ||||
|       ), | ||||
|     // Icons in supervisor conflict with icons in HA so we don't load. | ||||
|     isHassioBuild && | ||||
|       require.resolve( | ||||
|         path.resolve(paths.root_dir, "src/components/ha-icon.ts") | ||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon.ts") | ||||
|       ), | ||||
|     isHassioBuild && | ||||
|       require.resolve( | ||||
|         path.resolve(paths.root_dir, "src/components/ha-icon-picker.ts") | ||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon-picker.ts") | ||||
|       ), | ||||
|   ].filter(Boolean); | ||||
|  | ||||
| @@ -43,9 +55,8 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
|   __STATIC_PATH__: "/static/", | ||||
|   __HASS_URL__: `\`${ | ||||
|     "HASS_URL" in process.env | ||||
|       ? process.env.HASS_URL | ||||
|       : // eslint-disable-next-line no-template-curly-in-string | ||||
|         "${location.protocol}//${location.host}" | ||||
|       ? process.env["HASS_URL"] | ||||
|       : "${location.protocol}//${location.host}" | ||||
|   }\``, | ||||
|   "process.env.NODE_ENV": JSON.stringify( | ||||
|     isProdBuild ? "production" : "development" | ||||
| @@ -73,19 +84,6 @@ module.exports.terserOptions = ({ latestBuild, isTestBuild }) => ({ | ||||
|   sourceMap: !isTestBuild, | ||||
| }); | ||||
|  | ||||
| /** @type {import('@rspack/core').SwcLoaderOptions} */ | ||||
| module.exports.swcOptions = () => ({ | ||||
|   jsc: { | ||||
|     loose: true, | ||||
|     externalHelpers: true, | ||||
|     target: "ES2021", | ||||
|     parser: { | ||||
|       syntax: "typescript", | ||||
|       decorators: true, | ||||
|     }, | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| module.exports.babelOptions = ({ | ||||
|   latestBuild, | ||||
|   isProdBuild, | ||||
| @@ -110,6 +108,7 @@ module.exports.babelOptions = ({ | ||||
|         shippedProposals: true, | ||||
|       }, | ||||
|     ], | ||||
|     "@babel/preset-typescript", | ||||
|   ], | ||||
|   plugins: [ | ||||
|     [ | ||||
| @@ -146,6 +145,12 @@ module.exports.babelOptions = ({ | ||||
|       "@babel/plugin-transform-runtime", | ||||
|       { version: dependencies["@babel/runtime"] }, | ||||
|     ], | ||||
|     // Transpile decorators (still in TC39 process) | ||||
|     // Modern browsers support class fields and private methods, but transform is required with the older decorator version dictated by Lit | ||||
|     [ | ||||
|       "@babel/plugin-proposal-decorators", | ||||
|       { version: "2018-09", decoratorsBeforeExport: true }, | ||||
|     ], | ||||
|     "@babel/plugin-transform-class-properties", | ||||
|     "@babel/plugin-transform-private-methods", | ||||
|   ].filter(Boolean), | ||||
| @@ -165,7 +170,7 @@ module.exports.babelOptions = ({ | ||||
|         ], | ||||
|       ], | ||||
|       exclude: [ | ||||
|         path.join(paths.root_dir, "src/resources/polyfills"), | ||||
|         path.join(paths.polymer_dir, "src/resources/polyfills"), | ||||
|         ...[ | ||||
|           "@formatjs/(?:ecma402-abstract|intl-\\w+)", | ||||
|           "@lit-labs/virtualizer/polyfills", | ||||
| @@ -183,7 +188,6 @@ module.exports.babelOptions = ({ | ||||
|       include: /\/node_modules\//, | ||||
|       exclude: [ | ||||
|         "element-internals-polyfill", | ||||
|         "@shoelace-style", | ||||
|         "@?lit(?:-labs|-element|-html)?", | ||||
|       ].map((p) => new RegExp(`/node_modules/${p}/`)), | ||||
|     }, | ||||
|   | ||||
| @@ -21,7 +21,7 @@ module.exports = { | ||||
|   }, | ||||
|   version() { | ||||
|     const version = fs | ||||
|       .readFileSync(path.resolve(paths.root_dir, "pyproject.toml"), "utf8") | ||||
|       .readFileSync(path.resolve(paths.polymer_dir, "pyproject.toml"), "utf8") | ||||
|       .match(/version\W+=\W"(\d{8}\.\d(?:\.dev)?)"/); | ||||
|     if (!version) { | ||||
|       throw Error("Version not found"); | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| // @ts-check | ||||
|  | ||||
| import tseslint from "typescript-eslint"; | ||||
| import rootConfig from "../eslint.config.mjs"; | ||||
|  | ||||
| export default tseslint.config(...rootConfig, { | ||||
| export default [ | ||||
|   ...rootConfig, | ||||
|   { | ||||
|     rules: { | ||||
|       "no-console": "off", | ||||
|       "import/no-extraneous-dependencies": "off", | ||||
|       "import/extensions": "off", | ||||
|       "import/no-dynamic-require": "off", | ||||
|       "global-require": "off", | ||||
|     "@typescript-eslint/no-require-imports": "off", | ||||
|       "@typescript-eslint/no-var-requires": "off", | ||||
|       "prefer-arrow-callback": "off", | ||||
|     }, | ||||
| }); | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -3,7 +3,6 @@ | ||||
| import { constants } from "node:zlib"; | ||||
| import gulp from "gulp"; | ||||
| import brotli from "gulp-brotli"; | ||||
| import zopfli from "gulp-zopfli-green"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const filesGlob = "*.{js,json,css,svg,xml}"; | ||||
| @@ -13,74 +12,29 @@ const brotliOptions = { | ||||
|     [constants.BROTLI_PARAM_QUALITY]: constants.BROTLI_MAX_QUALITY, | ||||
|   }, | ||||
| }; | ||||
| const zopfliOptions = { threshold: 150 }; | ||||
|  | ||||
| const compressModern = (rootDir, modernDir, compress) => | ||||
|   gulp | ||||
|     .src([`${modernDir}/**/${filesGlob}`, `${rootDir}/sw-modern.js`], { | ||||
|       base: rootDir, | ||||
|       allowEmpty: true, | ||||
|     }) | ||||
|     .pipe(compress === "zopfli" ? zopfli(zopfliOptions) : brotli(brotliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
|  | ||||
| const compressOther = (rootDir, modernDir, compress) => | ||||
| const compressDistBrotli = (rootDir, modernDir, compressServiceWorker = true) => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         `${rootDir}/**/${filesGlob}`, | ||||
|         `!${modernDir}/**/${filesGlob}`, | ||||
|         `!${rootDir}/{sw-modern,service_worker}.js`, | ||||
|         `${rootDir}/{authorize,onboarding}.html`, | ||||
|       ], | ||||
|       { base: rootDir, allowEmpty: true } | ||||
|         `${modernDir}/**/${filesGlob}`, | ||||
|         compressServiceWorker ? `${rootDir}/sw-modern.js` : undefined, | ||||
|       ].filter(Boolean), | ||||
|       { | ||||
|         base: rootDir, | ||||
|       } | ||||
|     ) | ||||
|     .pipe(compress === "zopfli" ? zopfli(zopfliOptions) : brotli(brotliOptions)) | ||||
|     .pipe(brotli(brotliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
|  | ||||
| const compressAppModernBrotli = () => | ||||
|   compressModern(paths.app_output_root, paths.app_output_latest, "brotli"); | ||||
| const compressAppModernZopfli = () => | ||||
|   compressModern(paths.app_output_root, paths.app_output_latest, "zopfli"); | ||||
|  | ||||
| const compressHassioModernBrotli = () => | ||||
|   compressModern( | ||||
| const compressAppBrotli = () => | ||||
|   compressDistBrotli(paths.app_output_root, paths.app_output_latest); | ||||
| const compressHassioBrotli = () => | ||||
|   compressDistBrotli( | ||||
|     paths.hassio_output_root, | ||||
|     paths.hassio_output_latest, | ||||
|     "brotli" | ||||
|   ); | ||||
| const compressHassioModernZopfli = () => | ||||
|   compressModern( | ||||
|     paths.hassio_output_root, | ||||
|     paths.hassio_output_latest, | ||||
|     "zopfli" | ||||
|     false | ||||
|   ); | ||||
|  | ||||
| const compressAppOtherBrotli = () => | ||||
|   compressOther(paths.app_output_root, paths.app_output_latest, "brotli"); | ||||
| const compressAppOtherZopfli = () => | ||||
|   compressOther(paths.app_output_root, paths.app_output_latest, "zopfli"); | ||||
|  | ||||
| const compressHassioOtherBrotli = () => | ||||
|   compressOther(paths.hassio_output_root, paths.hassio_output_latest, "brotli"); | ||||
| const compressHassioOtherZopfli = () => | ||||
|   compressOther(paths.hassio_output_root, paths.hassio_output_latest, "zopfli"); | ||||
|  | ||||
| gulp.task( | ||||
|   "compress-app", | ||||
|   gulp.parallel( | ||||
|     compressAppModernBrotli, | ||||
|     compressAppOtherBrotli, | ||||
|     compressAppModernZopfli, | ||||
|     compressAppOtherZopfli | ||||
|   ) | ||||
| ); | ||||
| gulp.task( | ||||
|   "compress-hassio", | ||||
|   gulp.parallel( | ||||
|     compressHassioModernBrotli, | ||||
|     compressHassioOtherBrotli, | ||||
|     compressHassioModernZopfli, | ||||
|     compressHassioOtherZopfli | ||||
|   ) | ||||
| ); | ||||
| gulp.task("compress-app", compressAppBrotli); | ||||
| gulp.task("compress-hassio", compressHassioBrotli); | ||||
|   | ||||
| @@ -56,7 +56,6 @@ const getCommonTemplateVars = () => { | ||||
|   ); | ||||
|   return { | ||||
|     modernRegex: compileRegex(browserRegexes.concat(haMacOSRegex)).toString(), | ||||
|     hassUrl: process.env.HASS_URL || "", | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| @@ -169,14 +168,14 @@ const APP_PAGE_ENTRIES = { | ||||
|  | ||||
| gulp.task( | ||||
|   "gen-pages-app-dev", | ||||
|   genPagesDevTask(APP_PAGE_ENTRIES, paths.root_dir, paths.app_output_root) | ||||
|   genPagesDevTask(APP_PAGE_ENTRIES, paths.polymer_dir, paths.app_output_root) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "gen-pages-app-prod", | ||||
|   genPagesProdTask( | ||||
|     APP_PAGE_ENTRIES, | ||||
|     paths.root_dir, | ||||
|     paths.polymer_dir, | ||||
|     paths.app_output_root, | ||||
|     paths.app_output_latest, | ||||
|     paths.app_output_es5 | ||||
|   | ||||
| @@ -6,8 +6,8 @@ import path from "path"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const npmPath = (...parts) => | ||||
|   path.resolve(paths.root_dir, "node_modules", ...parts); | ||||
| const polyPath = (...parts) => path.resolve(paths.root_dir, ...parts); | ||||
|   path.resolve(paths.polymer_dir, "node_modules", ...parts); | ||||
| const polyPath = (...parts) => path.resolve(paths.polymer_dir, ...parts); | ||||
|  | ||||
| const copyFileDir = (fromFile, toDir) => | ||||
|   fs.copySync(fromFile, path.join(toDir, path.basename(fromFile))); | ||||
| @@ -59,11 +59,6 @@ function copyPolyfills(staticDir) { | ||||
|     npmPath("@webcomponents/webcomponentsjs/webcomponents-bundle.js.map"), | ||||
|     staticPath("polyfills/") | ||||
|   ); | ||||
|   // Lit polyfill support | ||||
|   fs.copySync( | ||||
|     npmPath("lit/polyfill-support.js"), | ||||
|     path.join(staticPath("polyfills/"), "lit-polyfill-support.js") | ||||
|   ); | ||||
|  | ||||
|   // dialog-polyfill css | ||||
|   copyFileDir( | ||||
| @@ -72,6 +67,12 @@ function copyPolyfills(staticDir) { | ||||
|   ); | ||||
| } | ||||
|  | ||||
| function copyLoaderJS(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
|   copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js")); | ||||
|   copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js")); | ||||
| } | ||||
|  | ||||
| function copyFonts(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
|   // Local fonts | ||||
| @@ -95,10 +96,6 @@ function copyMapPanel(staticDir) { | ||||
|     npmPath("leaflet/dist/leaflet.css"), | ||||
|     staticPath("images/leaflet/") | ||||
|   ); | ||||
|   copyFileDir( | ||||
|     npmPath("leaflet.markercluster/dist/MarkerCluster.css"), | ||||
|     staticPath("images/leaflet/") | ||||
|   ); | ||||
|   fs.copySync( | ||||
|     npmPath("leaflet/dist/images"), | ||||
|     staticPath("images/leaflet/images/") | ||||
| @@ -143,6 +140,8 @@ gulp.task("copy-static-app", async () => { | ||||
|   const staticDir = paths.app_output_static; | ||||
|   // Basic static files | ||||
|   fs.copySync(polyPath("public"), paths.app_output_root); | ||||
|  | ||||
|   copyLoaderJS(staticDir); | ||||
|   copyPolyfills(staticDir); | ||||
|   copyFonts(staticDir); | ||||
|   copyTranslations(staticDir); | ||||
| @@ -165,6 +164,8 @@ gulp.task("copy-static-demo", async () => { | ||||
|   ); | ||||
|   // Copy demo static files | ||||
|   fs.copySync(path.resolve(paths.demo_dir, "public"), paths.demo_output_root); | ||||
|  | ||||
|   copyLoaderJS(paths.demo_output_static); | ||||
|   copyPolyfills(paths.demo_output_static); | ||||
|   copyMapPanel(paths.demo_output_static); | ||||
|   copyFonts(paths.demo_output_static); | ||||
| @@ -178,6 +179,8 @@ gulp.task("copy-static-cast", async () => { | ||||
|   fs.copySync(polyPath("public/static"), paths.cast_output_static); | ||||
|   // Copy cast static files | ||||
|   fs.copySync(path.resolve(paths.cast_dir, "public"), paths.cast_output_root); | ||||
|  | ||||
|   copyLoaderJS(paths.cast_output_static); | ||||
|   copyPolyfills(paths.cast_output_static); | ||||
|   copyMapPanel(paths.cast_output_static); | ||||
|   copyFonts(paths.cast_output_static); | ||||
|   | ||||
| @@ -1,17 +0,0 @@ | ||||
| import "./app.js"; | ||||
| import "./cast.js"; | ||||
| import "./clean.js"; | ||||
| import "./compress.js"; | ||||
| import "./demo.js"; | ||||
| import "./download-translations.js"; | ||||
| import "./entry-html.js"; | ||||
| import "./fetch-nightly-translations.js"; | ||||
| import "./gallery.js"; | ||||
| import "./gather-static.js"; | ||||
| import "./gen-icons-json.js"; | ||||
| import "./hassio.js"; | ||||
| import "./landing-page.js"; | ||||
| import "./locale-data.js"; | ||||
| import "./rspack.js"; | ||||
| import "./service-worker.js"; | ||||
| import "./translations.js"; | ||||
| @@ -4,7 +4,7 @@ import gulp from "gulp"; | ||||
| import { join, resolve } from "node:path"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const formatjsDir = join(paths.root_dir, "node_modules", "@formatjs"); | ||||
| const formatjsDir = join(paths.polymer_dir, "node_modules", "@formatjs"); | ||||
| const outDir = join(paths.build_dir, "locale-data"); | ||||
|  | ||||
| const INTL_POLYFILLS = { | ||||
|   | ||||
| @@ -40,8 +40,8 @@ class CustomJSON extends Transform { | ||||
|     this._reviver = reviver; | ||||
|   } | ||||
|  | ||||
|   // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
|   async _transform(file, _, callback) { | ||||
|     try { | ||||
|       let obj = JSON.parse(file.contents.toString(), this._reviver); | ||||
|       if (this._func) obj = this._func(obj, file.path); | ||||
|       for (const [outObj, dir] of Array.isArray(obj) ? obj : [[obj, ""]]) { | ||||
| @@ -51,6 +51,9 @@ class CustomJSON extends Transform { | ||||
|         this.push(outFile); | ||||
|       } | ||||
|       callback(null); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| @@ -65,19 +68,25 @@ class MergeJSON extends Transform { | ||||
|     this._reviver = reviver; | ||||
|   } | ||||
|  | ||||
|   // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
|   async _transform(file, _, callback) { | ||||
|     try { | ||||
|       this._objects.push(JSON.parse(file.contents.toString(), this._reviver)); | ||||
|       if (!this._outFile) this._outFile = file.clone({ contents: false }); | ||||
|       callback(null); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
|   async _flush(callback) { | ||||
|     try { | ||||
|       const mergedObj = merge(this._startObj, ...this._objects); | ||||
|       this._outFile.contents = Buffer.from(JSON.stringify(mergedObj)); | ||||
|       this._outFile.stem = this._stem; | ||||
|       callback(null, this._outFile); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -16,7 +16,6 @@ const detailsClose = "</details>\n"; | ||||
|  | ||||
| const dummyAPI = { | ||||
|   version: babelVersion, | ||||
|   // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|   assertVersion: () => {}, | ||||
|   caller: (callback) => | ||||
|     callback({ | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| const path = require("path"); | ||||
|  | ||||
| module.exports = { | ||||
|   root_dir: path.resolve(__dirname, ".."), | ||||
|   polymer_dir: path.resolve(__dirname, ".."), | ||||
|  | ||||
|   build_dir: path.resolve(__dirname, "../build"), | ||||
|   app_output_root: path.resolve(__dirname, "../hass_frontend"), | ||||
|   | ||||
| @@ -1,17 +1,12 @@ | ||||
| const { existsSync } = require("fs"); | ||||
| const path = require("path"); | ||||
| const rspack = require("@rspack/core"); | ||||
| // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
| const { RsdoctorRspackPlugin } = require("@rsdoctor/rspack-plugin"); | ||||
| // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
| const { StatsWriterPlugin } = require("webpack-stats-plugin"); | ||||
| const filterStats = require("@bundle-stats/plugin-webpack-filter"); | ||||
| // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
| const filterStats = require("@bundle-stats/plugin-webpack-filter").default; | ||||
| const TerserPlugin = require("terser-webpack-plugin"); | ||||
| // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
| const { WebpackManifestPlugin } = require("rspack-manifest-plugin"); | ||||
| const log = require("fancy-log"); | ||||
| // eslint-disable-next-line @typescript-eslint/naming-convention | ||||
| const WebpackBar = require("webpackbar/rspack"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const bundle = require("./bundle.cjs"); | ||||
| @@ -65,9 +60,7 @@ const createRspackConfig = ({ | ||||
|       rules: [ | ||||
|         { | ||||
|           test: /\.m?js$|\.ts$/, | ||||
|           exclude: /node_modules[\\/]core-js/, | ||||
|           use: (info) => [ | ||||
|             { | ||||
|           use: (info) => ({ | ||||
|             loader: "babel-loader", | ||||
|             options: { | ||||
|               ...bundle.babelOptions({ | ||||
| @@ -79,12 +72,7 @@ const createRspackConfig = ({ | ||||
|               cacheDirectory: !isProdBuild, | ||||
|               cacheCompression: false, | ||||
|             }, | ||||
|             }, | ||||
|             { | ||||
|               loader: "builtin:swc-loader", | ||||
|               options: bundle.swcOptions(), | ||||
|             }, | ||||
|           ], | ||||
|           }), | ||||
|           resolve: { | ||||
|             fullySpecified: false, | ||||
|           }, | ||||
| @@ -143,8 +131,7 @@ const createRspackConfig = ({ | ||||
|             // calling define.amd will call require("!!webpack amd options") | ||||
|             resource.startsWith("!!webpack") || | ||||
|             // loaded by webpack dev server but doesn't exist. | ||||
|             resource === "webpack/hot" || | ||||
|             resource.startsWith("@swc/helpers") | ||||
|             resource === "webpack/hot" | ||||
|           ) { | ||||
|             return false; | ||||
|           } | ||||
| @@ -168,8 +155,10 @@ const createRspackConfig = ({ | ||||
|         }, | ||||
|       }), | ||||
|       new rspack.NormalModuleReplacementPlugin( | ||||
|         new RegExp(bundle.emptyPackages({ isHassioBuild }).join("|")), | ||||
|         path.resolve(paths.root_dir, "src/util/empty.js") | ||||
|         new RegExp( | ||||
|           bundle.emptyPackages({ latestBuild, isHassioBuild }).join("|") | ||||
|         ), | ||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") | ||||
|       ), | ||||
|       !isProdBuild && new LogStartCompilePlugin(), | ||||
|       isProdBuild && | ||||
| @@ -203,7 +192,6 @@ const createRspackConfig = ({ | ||||
|         "lit/directives/if-defined$": "lit/directives/if-defined.js", | ||||
|         "lit/directives/guard$": "lit/directives/guard.js", | ||||
|         "lit/directives/cache$": "lit/directives/cache.js", | ||||
|         "lit/directives/join$": "lit/directives/join.js", | ||||
|         "lit/directives/repeat$": "lit/directives/repeat.js", | ||||
|         "lit/directives/live$": "lit/directives/live.js", | ||||
|         "lit/directives/keyed$": "lit/directives/keyed.js", | ||||
|   | ||||
| @@ -25,7 +25,7 @@ Home Assistant Cast is made up of two separate applications: | ||||
|  | ||||
| ### Setting dev variables | ||||
|  | ||||
| Open `src/cast/dev_const.ts` and change `CAST_DEV_APP_ID` to the ID of the app you just created. And set the `CAST_DEV_HASS_URL` to the url of your development machine. | ||||
| Open `src/cast/dev_const.ts` and change `CAST_DEV_APP_ID` to the ID of the app you just created. And set the `CAST_DEV_HASS_URL` to the url of you development machine. | ||||
|  | ||||
| ### Changing configuration | ||||
|  | ||||
|   | ||||
| @@ -14,5 +14,5 @@ | ||||
|   "name": "Home Assistant Cast", | ||||
|   "short_name": "HA Cast", | ||||
|   "start_url": "/?homescreen=1", | ||||
|   "theme_color": "#009ac7" | ||||
|   "theme_color": "#03A9F4" | ||||
| } | ||||
|   | ||||
| @@ -7,6 +7,7 @@ | ||||
|   <%= renderTemplate("../../../src/html/_style_base.html.template") %> | ||||
|   <style> | ||||
|     body { | ||||
|       background-color: white; | ||||
|       font-size: initial; | ||||
|     } | ||||
|   </style> | ||||
|   | ||||
| @@ -1,3 +1,3 @@ | ||||
| import "./layout/hc-connect"; | ||||
|  | ||||
| import("../../../src/resources/append-ha-style"); | ||||
| import("../../../src/resources/ha-style"); | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import "@material/mwc-list/mwc-list"; | ||||
| 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"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { LitElement, css, html } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import type { CastManager } from "../../../../src/cast/cast_manager"; | ||||
| @@ -17,9 +19,6 @@ import { | ||||
| 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 { | ||||
|   getLegacyLovelaceCollection, | ||||
| @@ -30,6 +29,7 @@ import type { LovelaceViewConfig } from "../../../../src/data/lovelace/config/vi | ||||
| import "../../../../src/layouts/hass-loading-screen"; | ||||
| import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | ||||
| import "./hc-layout"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
|  | ||||
| @customElement("hc-cast") | ||||
| class HcCast extends LitElement { | ||||
| @@ -62,20 +62,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,15 +77,15 @@ 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` | ||||
|                 <div class="section-header">PICK A VIEW</div> | ||||
|                 <ha-list @action=${this._handlePickView} activatable> | ||||
|                 <mwc-list @action=${this._handlePickView} activatable> | ||||
|                   ${( | ||||
|                     this.lovelaceViews ?? [ | ||||
|                       generateDefaultViewConfig({}, {}, {}, {}, () => ""), | ||||
| @@ -121,29 +113,21 @@ class HcCast extends LitElement { | ||||
|                             ></ha-svg-icon>`} | ||||
|                       </ha-list-item> | ||||
|                     ` | ||||
|                   )}</ha-list | ||||
|                   )}</mwc-list | ||||
|                 > | ||||
|               `} | ||||
|  | ||||
|         <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> | ||||
|     `; | ||||
| @@ -219,12 +203,13 @@ class HcCast extends LitElement { | ||||
|       } | ||||
|       this.connection.close(); | ||||
|       location.reload(); | ||||
|     } catch (_err: any) { | ||||
|     } catch (err: any) { | ||||
|       alert("Unable to log out!"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       .center-item { | ||||
|         display: flex; | ||||
|         justify-content: space-around; | ||||
| @@ -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; | ||||
| @@ -279,6 +271,7 @@ class HcCast extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { mdiCastConnected, mdiCast } from "@mdi/js"; | ||||
| import type { | ||||
|   Auth, | ||||
| @@ -12,7 +13,7 @@ import { | ||||
|   ERR_INVALID_HTTPS_TO_HTTP, | ||||
|   getAuth, | ||||
| } from "home-assistant-js-websocket"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import type { CastManager } from "../../../../src/cast/cast_manager"; | ||||
| @@ -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> | ||||
|       `; | ||||
| @@ -221,7 +215,7 @@ export class HcConnect extends LitElement { | ||||
|     let url: URL; | ||||
|     try { | ||||
|       url = new URL(value); | ||||
|     } catch (_err: any) { | ||||
|     } catch (err: any) { | ||||
|       this.error = "Invalid URL"; | ||||
|       return; | ||||
|     } | ||||
| @@ -258,7 +252,7 @@ export class HcConnect extends LitElement { | ||||
|       this.loading = false; | ||||
|       return; | ||||
|     } finally { | ||||
|       // Clear url if we have an auth callback in url. | ||||
|       // Clear url if we have a auth callback in url. | ||||
|       if (location.search.includes("auth_callback=1")) { | ||||
|         history.replaceState(null, "", location.pathname); | ||||
|       } | ||||
| @@ -294,12 +288,13 @@ export class HcConnect extends LitElement { | ||||
|     try { | ||||
|       saveTokens(null); | ||||
|       location.reload(); | ||||
|     } catch (_err: any) { | ||||
|     } catch (err: any) { | ||||
|       alert("Unable to log out!"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       .card-content a { | ||||
|         color: var(--primary-color); | ||||
|       } | ||||
| @@ -308,13 +303,17 @@ export class HcConnect extends LitElement { | ||||
|       } | ||||
|       .error { | ||||
|         color: red; | ||||
|       font-weight: var(--ha-font-weight-bold); | ||||
|         font-weight: bold; | ||||
|       } | ||||
|  | ||||
|       .error a { | ||||
|         color: darkred; | ||||
|       } | ||||
|  | ||||
|       mwc-button ha-svg-icon { | ||||
|         margin-left: 8px; | ||||
|       } | ||||
|  | ||||
|       .spacer { | ||||
|         flex: 1; | ||||
|       } | ||||
| @@ -324,6 +323,7 @@ export class HcConnect extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import type { Auth, Connection, HassUser } from "home-assistant-js-websocket"; | ||||
| import { getUser } from "home-assistant-js-websocket"; | ||||
| import type { TemplateResult } from "lit"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| @@ -63,7 +63,8 @@ class HcLayout extends LitElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: flex; | ||||
|         min-height: 100%; | ||||
| @@ -86,9 +87,9 @@ class HcLayout extends LitElement { | ||||
|       .card-header { | ||||
|         color: var(--ha-card-header-color, var(--primary-text-color)); | ||||
|         font-family: var(--ha-card-header-font-family, inherit); | ||||
|       font-size: var(--ha-card-header-font-size, var(--ha-font-size-2xl)); | ||||
|         font-size: var(--ha-card-header-font-size, 24px); | ||||
|         letter-spacing: -0.012em; | ||||
|       line-height: var(--ha-line-height-condensed); | ||||
|         line-height: 32px; | ||||
|         padding: 24px 16px 16px; | ||||
|         display: block; | ||||
|         margin: 0; | ||||
| @@ -98,7 +99,7 @@ class HcLayout extends LitElement { | ||||
|         border-radius: 4px 4px 0 0; | ||||
|       } | ||||
|       .subtitle { | ||||
|       font-size: var(--ha-font-size-m); | ||||
|         font-size: 14px; | ||||
|         color: var(--secondary-text-color); | ||||
|         line-height: initial; | ||||
|       } | ||||
| @@ -113,7 +114,7 @@ class HcLayout extends LitElement { | ||||
|       } | ||||
|  | ||||
|       :host ::slotted(.section-header) { | ||||
|       font-weight: var(--ha-font-weight-medium); | ||||
|         font-weight: 500; | ||||
|         padding: 4px 16px; | ||||
|         text-transform: uppercase; | ||||
|       } | ||||
| @@ -135,7 +136,7 @@ class HcLayout extends LitElement { | ||||
|  | ||||
|       .footer { | ||||
|         text-align: center; | ||||
|       font-size: var(--ha-font-size-s); | ||||
|         font-size: 12px; | ||||
|         padding: 8px 0 24px; | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
| @@ -152,6 +153,7 @@ class HcLayout extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -24,12 +24,13 @@ class HcLaunchScreen extends LitElement { | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: block; | ||||
|         height: 100vh; | ||||
|         background-color: #f2f4f9; | ||||
|       font-size: var(--ha-font-size-2xl); | ||||
|         font-size: 24px; | ||||
|       } | ||||
|       .container { | ||||
|         display: flex; | ||||
| @@ -48,6 +49,7 @@ class HcLaunchScreen extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -1,4 +1,10 @@ | ||||
| import { css, html, LitElement, type TemplateResult } from "lit"; | ||||
| import { | ||||
|   css, | ||||
|   type CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   type TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import type { LovelaceConfig } from "../../../../src/data/lovelace/config/types"; | ||||
| @@ -8,7 +14,6 @@ import "../../../../src/panels/lovelace/views/hui-view"; | ||||
| import "../../../../src/panels/lovelace/views/hui-view-container"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "./hc-launch-screen"; | ||||
| import "../../../../src/panels/lovelace/views/hui-view-background"; | ||||
|  | ||||
| (window as any).loadCardHelpers = () => | ||||
|   import("../../../../src/panels/lovelace/custom-card-helpers"); | ||||
| @@ -52,9 +57,11 @@ class HcLovelace extends LitElement { | ||||
|     const background = viewConfig.background || this.lovelaceConfig.background; | ||||
|  | ||||
|     return html` | ||||
|       <hui-view-container .hass=${this.hass} .theme=${viewConfig.theme}> | ||||
|         <hui-view-background .hass=${this.hass} .background=${background}> | ||||
|         </hui-view-background> | ||||
|       <hui-view-container | ||||
|         .hass=${this.hass} | ||||
|         .background=${background} | ||||
|         .theme=${viewConfig.theme} | ||||
|       > | ||||
|         <hui-view | ||||
|           .hass=${this.hass} | ||||
|           .lovelace=${lovelace} | ||||
| @@ -111,19 +118,21 @@ class HcLovelace extends LitElement { | ||||
|     return undefined; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       hui-view-container { | ||||
|         display: flex; | ||||
|         position: relative; | ||||
|         min-height: 100vh; | ||||
|         box-sizing: border-box; | ||||
|       } | ||||
|     hui-view-container > * { | ||||
|       hui-view { | ||||
|         flex: 1 1 100%; | ||||
|         max-width: 100%; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| export interface CastViewChanged { | ||||
|   title: string | undefined; | ||||
|   | ||||
| @@ -109,7 +109,7 @@ export class HcMain extends HassElement { | ||||
|   protected firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     import("./hc-lovelace"); | ||||
|     import("../../../../src/resources/append-ha-style"); | ||||
|     import("../../../../src/resources/ha-style"); | ||||
|  | ||||
|     window.addEventListener("location-changed", () => { | ||||
|       const panelPath = `/${this._urlPath || "lovelace"}/`; | ||||
| @@ -309,7 +309,7 @@ export class HcMain extends HassElement { | ||||
|               "../../../../src/panels/lovelace/strategies/get-strategy" | ||||
|             ); | ||||
|             const config = await generateLovelaceDashboardStrategy( | ||||
|               rawConfig, | ||||
|               rawConfig.strategy, | ||||
|               this.hass! | ||||
|             ); | ||||
|             this._handleNewLovelaceConfig(config); | ||||
| @@ -351,7 +351,10 @@ export class HcMain extends HassElement { | ||||
|       "../../../../src/panels/lovelace/strategies/get-strategy" | ||||
|     ); | ||||
|     this._handleNewLovelaceConfig( | ||||
|       await generateLovelaceDashboardStrategy(DEFAULT_CONFIG, this.hass!) | ||||
|       await generateLovelaceDashboardStrategy( | ||||
|         DEFAULT_CONFIG.strategy, | ||||
|         this.hass! | ||||
|       ) | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -75,5 +75,5 @@ | ||||
|   "name": "Home Assistant Demo", | ||||
|   "short_name": "HA Demo", | ||||
|   "start_url": "/?homescreen=1", | ||||
|   "theme_color": "#009ac7" | ||||
|   "theme_color": "#03A9F4" | ||||
| } | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import type { Lovelace } from "../../../src/panels/lovelace/types"; | ||||
| import { energyEntities } from "../stubs/entities"; | ||||
| import type { DemoConfig } from "./types"; | ||||
|  | ||||
| export const demoConfigs: (() => Promise<DemoConfig>)[] = [ | ||||
| export const demoConfigs: Array<() => Promise<DemoConfig>> = [ | ||||
|   () => import("./sections").then((mod) => mod.demoSections), | ||||
|   () => import("./arsaboo").then((mod) => mod.demoArsaboo), | ||||
|   () => import("./teachingbirds").then((mod) => mod.demoTeachingbirds), | ||||
|   | ||||
| @@ -1,28 +1,37 @@ | ||||
| export const demoThemeJimpower = () => ({ | ||||
|   "text-primary-color": "var(--primary-text-color)", | ||||
|   "paper-item-icon-color": "var(--primary-text-color)", | ||||
|   "primary-color": "#5294E2", | ||||
|   "label-badge-red": "var(--accent-color)", | ||||
|   "paper-tabs-selection-bar-color": "green", | ||||
|   "light-primary-color": "var(--accent-color)", | ||||
|   "primary-background-color": "#383C45", | ||||
|   "primary-text-color": "#FFFFFF", | ||||
|   "paper-item-selected_-_background-color": "#434954", | ||||
|   "secondary-background-color": "#383C45", | ||||
|   "disabled-text-color": "#7F848E", | ||||
|   "paper-item-icon_-_color": "green", | ||||
|   "paper-grey-200": "#414A59", | ||||
|   "label-badge-background-color": "#2E333A", | ||||
|   "sidebar-icon-color": "var(--state-icon-color)", | ||||
|   "paper-card-header-color": "var(--accent-color)", | ||||
|   "sidebar-icon-color": "var(--paper-item-icon-color)", | ||||
|   "paper-listbox-background-color": "#2E333A", | ||||
|   "table-row-background-color": "#353840", | ||||
|   "paper-grey-50": "var(--primary-text-color)", | ||||
|   "switch-checked-color": "var(--accent-color)", | ||||
|   "paper-dialog-background-color": "#434954", | ||||
|   "secondary-text-color": "#5294E2", | ||||
|   "error-color": "#E45E65", | ||||
|   "divider-color": "rgba(0, 0, 0, .12)", | ||||
|   "success-color": "#39E949", | ||||
|   "switch-unchecked-button-color": "var(--disabled-text-color)", | ||||
|   "label-badge-border-color": "green", | ||||
|   "paper-listbox-color": "var(--primary-color)", | ||||
|   "card-background-color": "#434954", | ||||
|   "label-badge-text-color": "var(--primary-text-color)", | ||||
|   "switch-unchecked-track-color": "var(--disabled-text-color)", | ||||
|   "dark-primary-color": "var(--accent-color)", | ||||
|   "paper-item-icon-active-color": "#F9C536", | ||||
|   "accent-color": "#E45E65", | ||||
|   "table-row-alternative-background-color": "#3E424B", | ||||
| }); | ||||
|   | ||||
| @@ -1,29 +1,38 @@ | ||||
| // https://community.home-assistant.io/t/slate-a-new-dark-theme/86410 | ||||
| export const demoThemeKernehed = () => ({ | ||||
|   "text-primary-color": "var(--primary-text-color)", | ||||
|   "paper-item-icon-color": "var(--primary-text-color)", | ||||
|   "primary-color": "#2980b9", | ||||
|   "label-badge-red": "var(--accent-color)", | ||||
|   "paper-tabs-selection-bar-color": "green", | ||||
|   "primary-text-color": "#FFFFFF", | ||||
|   "light-primary-color": "var(--accent-color)", | ||||
|   "primary-background-color": "#222222", | ||||
|   "sidebar-icon-color": "#777777", | ||||
|   "paper-item-selected_-_background-color": "#292929", | ||||
|   "secondary-background-color": "#222222", | ||||
|   "disabled-text-color": "#777777", | ||||
|   "paper-item-icon_-_color": "green", | ||||
|   "paper-grey-200": "#222222", | ||||
|   "label-badge-background-color": "#222222", | ||||
|   "paper-card-header-color": "var(--accent-color)", | ||||
|   "paper-listbox-background-color": "#141414", | ||||
|   "table-row-background-color": "#292929", | ||||
|   "paper-grey-50": "var(--primary-text-color)", | ||||
|   "switch-checked-color": "var(--accent-color)", | ||||
|   "paper-dialog-background-color": "#292929", | ||||
|   "secondary-text-color": "#b58e31", | ||||
|   "error-color": "#b58e31", | ||||
|   "divider-color": "rgba(0, 0, 0, .12)", | ||||
|   "success-color": "#2980b9", | ||||
|   "switch-unchecked-button-color": "var(--disabled-text-color)", | ||||
|   "label-badge-border-color": "green", | ||||
|   "paper-listbox-color": "#777777", | ||||
|   "card-background-color": "#292929", | ||||
|   "label-badge-text-color": "var(--primary-text-color)", | ||||
|   "switch-unchecked-track-color": "var(--disabled-text-color)", | ||||
|   "dark-primary-color": "var(--accent-color)", | ||||
|   "paper-item-icon-active-color": "#b58e31", | ||||
|   "accent-color": "#2980b9", | ||||
|   "table-row-alternative-background-color": "#292929", | ||||
| }); | ||||
|   | ||||
| @@ -1,18 +1,26 @@ | ||||
| export const demoThemeTeachingbirds = () => ({ | ||||
|   "paper-card-header-color": "var(--paper-item-icon-color)", | ||||
|   "paper-listbox-background-color": "#202020", | ||||
|   "paper-grey-50": "var(--primary-text-color)", | ||||
|   "paper-item-icon-color": "#d3d3d3", | ||||
|   "divider-color": "rgba(255, 255, 255, 0.12)", | ||||
|   "primary-color": "#389638", | ||||
|   "light-primary-color": "#6f956f", | ||||
|   "label-badge-red": "var(--primary-color)", | ||||
|   "paper-listbox-color": "#FFFFFF", | ||||
|   "paper-toggle-button-checked-bar-color": "var(--light-primary-color)", | ||||
|   "switch-unchecked-track-color": "var(--primary-text-color)", | ||||
|   "card-background-color": "#4e4e4e", | ||||
|   "label-badge-text-color": "var(--text-primary-color)", | ||||
|   "primary-background-color": "#303030", | ||||
|   "sidebar-icon-color": "#d3d3d3", | ||||
|   "sidebar-icon-color": "var(--paper-item-icon-color)", | ||||
|   "secondary-background-color": "#2b2b2b", | ||||
|   "paper-item-icon-active-color": "#d8bf50", | ||||
|   "switch-checked-color": "var(--primary-color)", | ||||
|   "secondary-text-color": "#389638", | ||||
|   "disabled-text-color": "#545454", | ||||
|   "paper-item-icon_-_color": "var(--primary-text-color)", | ||||
|   "paper-grey-200": "#191919", | ||||
|   "primary-text-color": "#cfcfcf", | ||||
|   "label-badge-background-color": "var(--secondary-background-color)", | ||||
| }); | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { mdiTelevision } from "@mdi/js"; | ||||
| import type { CSSResultGroup } from "lit"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import type { CastManager } from "../../../src/cast/cast_manager"; | ||||
| @@ -66,14 +67,15 @@ class CastDemoRow extends LitElement implements LovelaceRow { | ||||
|     this.style.display = this._castManager ? "" : "none"; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|       } | ||||
|       ha-svg-icon { | ||||
|         padding: 8px; | ||||
|       color: var(--state-icon-color); | ||||
|         color: var(--paper-item-icon-color); | ||||
|       } | ||||
|       .flex { | ||||
|         flex: 1; | ||||
| @@ -96,6 +98,7 @@ class CastDemoRow extends LitElement implements LovelaceRow { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import { until } from "lit/directives/until"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-spinner"; | ||||
| import "../../../src/components/ha-circular-progress"; | ||||
| import type { LovelaceCardConfig } from "../../../src/data/lovelace/config/card"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| import type { | ||||
| @@ -32,7 +32,6 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|     return this._hidden ? 0 : 2; | ||||
|   } | ||||
|  | ||||
|   // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|   public setConfig(_config: LovelaceCardConfig) {} | ||||
|  | ||||
|   protected render() { | ||||
| @@ -44,7 +43,9 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|         <div class="picker"> | ||||
|           <div class="label"> | ||||
|             ${this._switching | ||||
|               ? html`<ha-spinner></ha-spinner>` | ||||
|               ? html` | ||||
|                   <ha-circular-progress indeterminate></ha-circular-progress> | ||||
|                 ` | ||||
|               : until( | ||||
|                   selectedDemoConfig.then( | ||||
|                     (conf) => html` | ||||
| @@ -89,14 +90,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" | ||||
|           > | ||||
|           <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> | ||||
|     `; | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import "./util/is_frontpage"; | ||||
| import "./ha-demo"; | ||||
|  | ||||
| import("../../src/resources/append-ha-style"); | ||||
| import("../../src/resources/ha-style"); | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| // Compat needs to be first import | ||||
| import "../../src/resources/compatibility"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { isNavigationClick } from "../../src/common/dom/is-navigation-click"; | ||||
| import { navigate } from "../../src/common/navigate"; | ||||
| @@ -63,7 +65,6 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|     mockEntityRegistry(hass, [ | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
|         config_subentry_id: null, | ||||
|         device_id: "co2signal", | ||||
|         area_id: null, | ||||
|         disabled_by: null, | ||||
| @@ -84,7 +85,6 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|       }, | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
|         config_subentry_id: null, | ||||
|         device_id: "co2signal", | ||||
|         area_id: null, | ||||
|         disabled_by: null, | ||||
|   | ||||
| @@ -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,10 +1,9 @@ | ||||
| import type { validateConfig } from "../../../src/data/config"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockConfig = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS<typeof validateConfig>("validate_config", () => ({ | ||||
|     actions: { valid: true, error: null }, | ||||
|     conditions: { valid: true, error: null }, | ||||
|     triggers: { valid: true, error: null }, | ||||
|   hass.mockWS("validate_config", () => ({ | ||||
|     actions: { valid: true }, | ||||
|     conditions: { valid: true }, | ||||
|     triggers: { valid: true }, | ||||
|   })); | ||||
| }; | ||||
|   | ||||
| @@ -1,10 +1,8 @@ | ||||
| import type { getConfigEntries } from "../../../src/data/config_entries"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockConfigEntries = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS<typeof getConfigEntries>("config_entries/get", () => [ | ||||
|     { | ||||
|       entry_id: "mock-entry-co2signal", | ||||
|   hass.mockWS("config_entries/get", () => ({ | ||||
|     entry_id: "co2signal", | ||||
|     domain: "co2signal", | ||||
|     title: "Electricity Maps", | ||||
|     source: "user", | ||||
| @@ -13,14 +11,9 @@ export const mockConfigEntries = (hass: MockHomeAssistant) => { | ||||
|     supports_remove_device: false, | ||||
|     supports_unload: true, | ||||
|     supports_reconfigure: true, | ||||
|       supported_subentry_types: {}, | ||||
|     pref_disable_new_entities: false, | ||||
|     pref_disable_polling: false, | ||||
|     disabled_by: null, | ||||
|     reason: null, | ||||
|       num_subentries: 0, | ||||
|       error_reason_translation_key: null, | ||||
|       error_reason_translation_placeholders: null, | ||||
|     }, | ||||
|   ]); | ||||
|   })); | ||||
| }; | ||||
|   | ||||
| @@ -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 () => {}; | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -131,7 +131,6 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|         }); | ||||
|       }, 1); | ||||
|  | ||||
|       // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|       return () => {}; | ||||
|     } | ||||
|   ); | ||||
|   | ||||
| @@ -43,7 +43,7 @@ customElements.whenDefined("hui-root").then(() => { | ||||
|       const index = (ev as CustomEvent).detail.index; | ||||
|       try { | ||||
|         await setDemoConfig(this.hass, this.lovelace!, index); | ||||
|       } catch (_err: any) { | ||||
|       } catch (err: any) { | ||||
|         setDemoConfig(this.hass, this.lovelace!, selectedDemoConfigIndex); | ||||
|         alert("Failed to switch config :-("); | ||||
|       } | ||||
|   | ||||
| @@ -15,7 +15,6 @@ export const mockPersistentNotification = (hass: MockHomeAssistant) => { | ||||
|         }, | ||||
|       }, | ||||
|     } as PersistentNotificationMessage); | ||||
|     // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|     return () => {}; | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -15,7 +15,7 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| const generateMeanStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line default-param-last | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| @@ -52,7 +52,7 @@ const generateMeanStatistics = ( | ||||
| const generateSumStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line default-param-last | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| @@ -89,7 +89,7 @@ const generateSumStatistics = ( | ||||
| const generateCurvedStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line default-param-last | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   _period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number, | ||||
|   | ||||
| @@ -11,7 +11,6 @@ export const mockTemplate = (hass: MockHomeAssistant) => { | ||||
|       result: msg.template, | ||||
|       listeners: { all: false, domains: [], entities: [], time: false }, | ||||
|     }); | ||||
|     // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|     return () => {}; | ||||
|   }); | ||||
| }; | ||||
|   | ||||
| @@ -2,7 +2,8 @@ import type { TodoItem } from "../../../src/data/todo"; | ||||
| import { TodoItemStatus } from "../../../src/data/todo"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| const items = { | ||||
| export const mockTodo = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS("todo/item/list", () => ({ | ||||
|     items: [ | ||||
|       { | ||||
|         uid: "12", | ||||
| @@ -19,19 +20,7 @@ const items = { | ||||
|         summary: "Oranges", | ||||
|         status: TodoItemStatus.Completed, | ||||
|       }, | ||||
|     { | ||||
|       uid: "15", | ||||
|       summary: "Beer", | ||||
|     }, | ||||
|     ] as TodoItem[], | ||||
| }; | ||||
|  | ||||
| export const mockTodo = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS("todo/item/list", () => items); | ||||
|   hass.mockWS("todo/item/move", () => undefined); | ||||
|   hass.mockWS("todo/item/subscribe", (_msg, _hass, onChange) => { | ||||
|     onChange!(items); | ||||
|     // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|     return () => {}; | ||||
|   }); | ||||
|   })); | ||||
|   hass.mockWS("todo/item/subscribe", (_msg, _hass) => () => {}); | ||||
| }; | ||||
|   | ||||
| @@ -1,17 +1,11 @@ | ||||
| // @ts-check | ||||
|  | ||||
| /* eslint-disable import/no-extraneous-dependencies */ | ||||
| import unusedImports from "eslint-plugin-unused-imports"; | ||||
| import globals from "globals"; | ||||
| import tsParser from "@typescript-eslint/parser"; | ||||
| import path from "node:path"; | ||||
| import { fileURLToPath } from "node:url"; | ||||
| import js from "@eslint/js"; | ||||
| import { FlatCompat } from "@eslint/eslintrc"; | ||||
| 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); | ||||
| @@ -21,15 +15,16 @@ const compat = new FlatCompat({ | ||||
|   allConfig: js.configs.all, | ||||
| }); | ||||
|  | ||||
| export default tseslint.config( | ||||
|   ...compat.extends("airbnb-base"), | ||||
|   eslintConfigPrettier, | ||||
|   litConfigs["flat/all"], | ||||
|   tseslint.configs.recommended, | ||||
|   tseslint.configs.strict, | ||||
|   tseslint.configs.stylistic, | ||||
|   wcConfigs["flat/recommended"], | ||||
|   a11yConfigs.recommended, | ||||
| export default [ | ||||
|   ...compat.extends( | ||||
|     "airbnb-base", | ||||
|     "airbnb-typescript/base", | ||||
|     "plugin:@typescript-eslint/recommended", | ||||
|     "plugin:wc/recommended", | ||||
|     "plugin:lit/all", | ||||
|     "plugin:lit-a11y/recommended", | ||||
|     "prettier" | ||||
|   ), | ||||
|   { | ||||
|     plugins: { | ||||
|       "unused-imports": unusedImports, | ||||
| @@ -44,9 +39,10 @@ export default tseslint.config( | ||||
|         __VERSION__: false, | ||||
|         __STATIC_PATH__: false, | ||||
|         __SUPERVISOR__: false, | ||||
|         Polymer: true, | ||||
|       }, | ||||
|  | ||||
|       parser: tseslint.parser, | ||||
|       parser: tsParser, | ||||
|       ecmaVersion: 2020, | ||||
|       sourceType: "module", | ||||
|  | ||||
| @@ -54,6 +50,8 @@ export default tseslint.config( | ||||
|         ecmaFeatures: { | ||||
|           modules: true, | ||||
|         }, | ||||
|  | ||||
|         project: "./tsconfig.json", | ||||
|       }, | ||||
|     }, | ||||
|  | ||||
| @@ -150,15 +148,15 @@ export default tseslint.config( | ||||
|         }, | ||||
|       ], | ||||
|  | ||||
|       "@typescript-eslint/no-unused-vars": [ | ||||
|       "@typescript-eslint/no-unused-vars": "off", | ||||
|  | ||||
|       "unused-imports/no-unused-vars": [ | ||||
|         "error", | ||||
|         { | ||||
|           args: "all", | ||||
|           argsIgnorePattern: "^_", | ||||
|           caughtErrors: "all", | ||||
|           caughtErrorsIgnorePattern: "^_", | ||||
|           destructuredArrayIgnorePattern: "^_", | ||||
|           vars: "all", | ||||
|           varsIgnorePattern: "^_", | ||||
|           args: "after-used", | ||||
|           argsIgnorePattern: "^_", | ||||
|           ignoreRestSiblings: true, | ||||
|         }, | ||||
|       ], | ||||
| @@ -176,16 +174,6 @@ export default tseslint.config( | ||||
|       "lit-a11y/role-has-required-aria-attrs": "error", | ||||
|       "@typescript-eslint/consistent-type-imports": "error", | ||||
|       "@typescript-eslint/no-import-type-side-effects": "error", | ||||
|       camelcase: "off", | ||||
|       "@typescript-eslint/no-dynamic-delete": "off", | ||||
|       "@typescript-eslint/no-empty-object-type": [ | ||||
|         "error", | ||||
|         { | ||||
|           allowInterfaces: "always", | ||||
|           allowObjectTypes: "always", | ||||
|     }, | ||||
|       ], | ||||
|       "no-use-before-define": "off", | ||||
|   }, | ||||
|   } | ||||
| ); | ||||
| ]; | ||||
|   | ||||
| @@ -1,10 +1,10 @@ | ||||
| // @ts-check | ||||
|  | ||||
| import tseslint from "typescript-eslint"; | ||||
| import rootConfig from "../eslint.config.mjs"; | ||||
|  | ||||
| export default tseslint.config(...rootConfig, { | ||||
| export default [ | ||||
|   ...rootConfig, | ||||
|   { | ||||
|     rules: { | ||||
|       "no-console": "off", | ||||
|     }, | ||||
| }); | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -1,10 +0,0 @@ | ||||
| <svg width="94" height="64" viewBox="0 0 94 64" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <rect width="94" height="64" rx="8" fill="white"/> | ||||
| <rect x="0.5" y="0.5" width="93" height="63" rx="7.5" stroke="black" stroke-opacity="0.12"/> | ||||
| <path d="M8 14C8 10.6863 10.6863 8 14 8H33C36.3137 8 39 10.6863 39 14C39 17.3137 36.3137 20 33 20H14C10.6863 20 8 17.3137 8 14Z" fill="black" fill-opacity="0.32"/> | ||||
| <path d="M8 27C8 25.3431 9.34315 24 11 24H31C32.6569 24 34 25.3431 34 27V29C34 30.6569 32.6569 32 31 32H11C9.34315 32 8 30.6569 8 29V27Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M38 27C38 25.3431 39.3431 24 41 24H83C84.6569 24 86 25.3431 86 27V29C86 30.6569 84.6569 32 83 32H41C39.3431 32 38 30.6569 38 29V27Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M8 39C8 37.3431 9.34315 36 11 36H53C54.6569 36 56 37.3431 56 39V41C56 42.6569 54.6569 44 53 44H11C9.34315 44 8 42.6569 8 41V39Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M60 39C60 37.3431 61.3431 36 63 36H83C84.6569 36 86 37.3431 86 39V41C86 42.6569 84.6569 44 83 44H63C61.3431 44 60 42.6569 60 41V39Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M8 51C8 49.3431 9.34315 48 11 48H31C32.6569 48 34 49.3431 34 51V53C34 54.6569 32.6569 56 31 56H11C9.34315 56 8 54.6569 8 53V51Z" fill="black" fill-opacity="0.12"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 1.3 KiB | 
| @@ -1,7 +0,0 @@ | ||||
| <svg width="94" height="48" viewBox="0 0 94 48" fill="none" xmlns="http://www.w3.org/2000/svg"> | ||||
| <path d="M0 11C0 9.34315 1.34315 8 3 8H23C24.6569 8 26 9.34315 26 11V13C26 14.6569 24.6569 16 23 16H3C1.34315 16 0 14.6569 0 13V11Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M30 11C30 9.34315 31.3431 8 33 8H91C92.6569 8 94 9.34315 94 11V13C94 14.6569 92.6569 16 91 16H33C31.3431 16 30 14.6569 30 13V11Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M0 23C0 21.3431 1.34315 20 3 20H61C62.6569 20 64 21.3431 64 23V25C64 26.6569 62.6569 28 61 28H3C1.34315 28 0 26.6569 0 25V23Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M68 23C68 21.3431 69.3431 20 71 20H91C92.6569 20 94 21.3431 94 23V25C94 26.6569 92.6569 28 91 28H71C69.3431 28 68 26.6569 68 25V23Z" fill="black" fill-opacity="0.12"/> | ||||
| <path d="M0 35C0 33.3431 1.34315 32 3 32H23C24.6569 32 26 33.3431 26 35V37C26 38.6569 24.6569 40 23 40H3C1.34315 40 0 38.6569 0 37V35Z" fill="black" fill-opacity="0.12"/> | ||||
| </svg> | ||||
| Before Width: | Height: | Size: 964 B | 
| @@ -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 | ||||
|             <state-card-content | ||||
|               .stateObj=${state} | ||||
|               .hass=${this.hass} | ||||
|               in-dialog | ||||
|                 ></state-card-content>` | ||||
|               : nothing} | ||||
|             ></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); | ||||
|         font-size: 18px; | ||||
|         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, | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import "./ha-gallery"; | ||||
|  | ||||
| import("../../src/resources/append-ha-style"); | ||||
| import("../../src/resources/ha-style"); | ||||
|  | ||||
| document.body.appendChild(document.createElement("ha-gallery")); | ||||
|   | ||||
| @@ -34,7 +34,7 @@ class HaDemoOptions extends LitElement { | ||||
|         height: 64px; | ||||
|         padding: 0 16px; | ||||
|         pointer-events: none; | ||||
|         font-size: var(--ha-font-size-xl); | ||||
|         font-size: 20px; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
|   | ||||
| @@ -250,14 +250,14 @@ 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); | ||||
|         font-size: 16px; | ||||
|         font-weight: 500; | ||||
|         line-height: 28px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer .secondary { | ||||
|         line-height: var(--ha-line-height-normal); | ||||
|         line-height: 23px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|   | ||||
| @@ -42,7 +42,7 @@ In most cases, Create can be paired with Delete, and Add can be paired with Remo | ||||
|  | ||||
| ## Add | ||||
|  | ||||
| An already-existing item. | ||||
| An already-exisiting item. | ||||
|  | ||||
| For example: | ||||
|  | ||||
|   | ||||
| @@ -177,7 +177,8 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|     this._action = ev.detail.isValid ? ev.detail.value : undefined; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
| @@ -196,6 +197,7 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -98,7 +98,8 @@ export class DemoAutomationDescribeCondition extends LitElement { | ||||
|     this._condition = ev.detail.isValid ? ev.detail.value : undefined; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
| @@ -117,6 +118,7 @@ export class DemoAutomationDescribeCondition extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -121,7 +121,8 @@ export class DemoAutomationDescribeTrigger extends LitElement { | ||||
|     this._trigger = ev.detail.isValid ? ev.detail.value : undefined; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
| @@ -140,6 +141,7 @@ export class DemoAutomationDescribeTrigger extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -1,29 +1,29 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { LitElement, css, html } from "lit"; | ||||
| import { LitElement, html, css } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import type { Action } from "../../../../src/data/script"; | ||||
| import "../../../../src/panels/config/automation/action/ha-automation-action"; | ||||
| import { HaChooseAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-choose"; | ||||
| import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
| import { HaDelayAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-delay"; | ||||
| import { HaDeviceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-device_id"; | ||||
| 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 { 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"; | ||||
| import { HaStopAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-stop"; | ||||
| import { HaWaitForTriggerAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_for_trigger"; | ||||
| import { HaWaitAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | ||||
| import type { Action } from "../../../../src/data/script"; | ||||
| import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
| import { HaSequenceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-sequence"; | ||||
| import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | ||||
| import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | ||||
| import { HaStopAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-stop"; | ||||
| import { HaPlayMediaAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-play_media"; | ||||
|  | ||||
| const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
|   { name: "Event", actions: [HaEventAction.defaultConfig] }, | ||||
| @@ -31,6 +31,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] }, | ||||
|   | ||||
| @@ -1,27 +1,26 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { LitElement, css, html } from "lit"; | ||||
| import { LitElement, html, css } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import type { ConditionWithShorthand } from "../../../../src/data/automation"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import type { ConditionWithShorthand } from "../../../../src/data/automation"; | ||||
| import "../../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||
| import { HaAndCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-and"; | ||||
| import { HaDeviceCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||
| import { HaNotCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-not"; | ||||
| import HaNumericStateCondition from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-numeric_state"; | ||||
| import { HaOrCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-or"; | ||||
| import { HaStateCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-state"; | ||||
| import { HaSunCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-sun"; | ||||
| import { HaTemplateCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-template"; | ||||
| import { HaTimeCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-time"; | ||||
| import { HaTriggerCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||
| import { HaZoneCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { HaAndCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-and"; | ||||
| import { HaOrCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-or"; | ||||
| import { HaNotCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-not"; | ||||
|  | ||||
| const SCHEMAS: { name: string; conditions: ConditionWithShorthand[] }[] = [ | ||||
|   { | ||||
|   | ||||
| @@ -1,36 +1,35 @@ | ||||
| import type { TemplateResult } from "lit"; | ||||
| import { LitElement, css, html } from "lit"; | ||||
| import { LitElement, html, css } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockAuth } from "../../../../demo/src/stubs/auth"; | ||||
| import { mockConfig } from "../../../../demo/src/stubs/config"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import { mockTags } from "../../../../demo/src/stubs/tags"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import type { Trigger } from "../../../../src/data/automation"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
| import { HaConversationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-conversation"; | ||||
| import { HaDeviceTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-device"; | ||||
| import { HaEventTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-event"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import { mockConfig } from "../../../../demo/src/stubs/config"; | ||||
| import { mockTags } from "../../../../demo/src/stubs/tags"; | ||||
| import { mockAuth } from "../../../../demo/src/stubs/auth"; | ||||
| import type { Trigger } from "../../../../src/data/automation"; | ||||
| import { HaGeolocationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-geo_location"; | ||||
| import { HaEventTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-event"; | ||||
| import { HaHassTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-homeassistant"; | ||||
| import { HaTriggerList } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-list"; | ||||
| import { HaMQTTTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-mqtt"; | ||||
| import { HaNumericStateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-numeric_state"; | ||||
| import { HaPersistentNotificationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-persistent_notification"; | ||||
| import { HaStateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-state"; | ||||
| import { HaSunTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-sun"; | ||||
| import { HaTagTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-tag"; | ||||
| import { HaTemplateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-template"; | ||||
| import { HaTimeTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time"; | ||||
| import { HaTimePatternTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time_pattern"; | ||||
| import { HaWebhookTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-webhook"; | ||||
| import { HaPersistentNotificationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-persistent_notification"; | ||||
| import { HaZoneTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-zone"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { HaDeviceTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-device"; | ||||
| import { HaStateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-state"; | ||||
| import { HaMQTTTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-mqtt"; | ||||
| import "../../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
| import { HaConversationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-conversation"; | ||||
| import { HaTriggerList } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-list"; | ||||
|  | ||||
| const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
|   { | ||||
|   | ||||
| @@ -58,7 +58,8 @@ export class DemoAutomationTraceTimeline extends LitElement { | ||||
|     hass.updateTranslations("config", "en"); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px; | ||||
| @@ -73,6 +74,7 @@ export class DemoAutomationTraceTimeline extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -68,7 +68,8 @@ export class DemoAutomationTrace extends LitElement { | ||||
|     this._selected = { ...this._selected, [sampleIdx]: ev.detail.path }; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px; | ||||
| @@ -89,6 +90,7 @@ export class DemoAutomationTrace extends LitElement { | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -19,7 +19,7 @@ The alert offers four severity levels that set a distinctive icon and color. | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="warning"> | ||||
|   This is a warning alert — check it out! | ||||
|   This is an warning alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="info"> | ||||
| @@ -27,7 +27,7 @@ The alert offers four severity levels that set a distinctive icon and color. | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| **Note:** This component is by <a href="https://mui.com/components/alert/" rel="noopener noreferrer" target="_blank">MUI</a> and is not documented in the <a href="https://material.io" rel="noopener noreferrer" target="_blank">Material Design guidelines</a>. | ||||
| @@ -95,7 +95,7 @@ Actions must have a tab index of 0 so that they can be reached by keyboard-only | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="warning"> | ||||
|   This is a warning alert — check it out! | ||||
|   This is an warning alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="info"> | ||||
| @@ -103,7 +103,7 @@ Actions must have a tab index of 0 so that they can be reached by keyboard-only | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| @@ -122,38 +122,38 @@ Actions must have a tab index of 0 so that they can be reached by keyboard-only | ||||
| The `title ` option should not be used without a description. | ||||
|  | ||||
| <ha-alert alert-type="success" title="Success"> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" title="Success"> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Dismissable** | ||||
|  | ||||
| <ha-alert alert-type="success" dismissable> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" dismissable> | ||||
|   This is a success alert — check it out! | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Slotted action** | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
|   <ha-button slot="action">Undo</ha-button> | ||||
|   This is an success alert — check it out! | ||||
|   <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> | ||||
|   This is an success alert — check it out! | ||||
|   <mwc-button slot="action" label="Undo"></mwc-button> | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user