mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 19:49:53 +00:00 
			
		
		
		
	Compare commits
	
		
			159 Commits
		
	
	
		
			20240719.0
			...
			boolean_se
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 932120869b | ||
|   | 061521a979 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d3f73baa36 | ||
|   | 3037bf494c | ||
|   | b4dd953128 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 430a28f350 | ||
|   | 0eacf3fdac | ||
|   | 7f9bf69a08 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 32cca9e30c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d7d62307b8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 12bfa5dab2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c0a728bc66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 19e8667349 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f3688b95d4 | ||
|   | 01f692f05c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d652f6382d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c0f96d9473 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f1a2af24b3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8501098bd1 | ||
|   | a235f76985 | ||
|   | 968c0de141 | ||
|   | 77d8aff1f4 | ||
|   | 5e486d9cf0 | ||
|   | f730761b3f | ||
|   | 46fc9c1a33 | ||
|   | 8ed68bf295 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5622180d42 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ef1f9b371d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0b79684cf1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bc68d8df11 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ebbade2fb7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2b5f778f2e | ||
|   | 91fc2383cb | ||
|   | 1080a8c961 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6144049f8c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4ad3ad6e93 | ||
|   | 16e84296da | ||
|   | 3e1ea8d236 | ||
|   | 8c9996fc81 | ||
|   | 336b5fb547 | ||
|   | 3f0f3affb6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6754b8893b | ||
|   | 6ec4323c76 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 20408392d2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b030a5d1f0 | ||
|   | e26e6d7c2d | ||
|   | e0c98e4524 | ||
|   | 2832f501d1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d2f2d682a9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 54b2121273 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c5ca731e05 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fd9c6c5449 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 39f4355c1a | ||
|   | 35fed0b0e2 | ||
|   | f0f0aefca1 | ||
|   | b60864086f | ||
|   | 6629f372fc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ec9a95dde0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3cd0c49c78 | ||
|   | e1ae95dd9f | ||
|   | 9c6aef033d | ||
|   | b23aacef84 | ||
|   | 7d218c89ae | ||
|   | ffa96d789f | ||
|   | 5eae163796 | ||
|   | 9e7f01a009 | ||
|   | eeffa3561c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f096eb2031 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3aad7431da | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a4f167559c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 44a0582e83 | ||
|   | 94c28ea534 | ||
|   | 5b7c20af33 | ||
|   | 4719775926 | ||
|   | 5c30c1647c | ||
|   | 3ba572ee37 | ||
|   | 0adee7d189 | ||
|   | 51b6d7758d | ||
|   | 0f21dfadf1 | ||
|   | 54e8a34f21 | ||
|   | dfbf4abd5d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 62d8434596 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d6df228e17 | ||
|   | 00faa16349 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3c92fe4170 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 02a9d67c7d | ||
|   | 88718bca23 | ||
|   | 33931b29a1 | ||
|   | a9310fdde0 | ||
|   | d634317438 | ||
|   | a41978f647 | ||
|   | 7e799bf639 | ||
|   | 73617711ff | ||
|   | edbfc22bf8 | ||
|   | 5e75f6a6c2 | ||
|   | 6dc80306e8 | ||
|   | acd1b04b0a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7e9d09e11c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 038f7ec000 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 08959cbabf | ||
|   | b611674ebd | ||
|   | 2f696bd511 | ||
|   | 9e3284a7db | ||
|   | 0c3471e0b7 | ||
|   | 1ca0b58aca | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | adceed689b | ||
|   | 574fc99889 | ||
|   | 3a83cb36a1 | ||
|   | a458ccf995 | ||
|   | 560e2c9438 | ||
|   | 78becb5440 | ||
|   | da2e530601 | ||
|   | a88a7c5236 | ||
|   | 0a095c6f21 | ||
|   | e8dd835eeb | ||
|   | e05c66444a | ||
|   | d0e61ca31a | ||
|   | 1f90a0c2e5 | ||
|   | dbd84901f8 | ||
|   | 0aa25dbed9 | ||
|   | dd74a35d3f | ||
|   | b1d8ec0fe4 | ||
|   | cd4af674a3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7a6491a901 | ||
|   | 8d20303d54 | ||
|   | a5786b4761 | ||
|   | 4ade39543d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a85dda3365 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e578904ff7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 09f0da1ead | ||
|   | 2faa8fec17 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 945c4a66b1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5d794e7e88 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 88a33bee14 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 35ec9af23f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dd331173ad | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 885ccb84cb | ||
|   | 0358fe5614 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 89d842c2a8 | ||
|   | 8911b55316 | ||
|   | 6791e85625 | ||
|   | 79618ce114 | ||
|   | 87ba0e73dd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 567a2ea019 | ||
|   | 811c34b489 | ||
|   | dd22ae446a | ||
|   | d96ddf968c | ||
|   | bbb64870a1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f05ddd3fcd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0612e25d9f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 559ecf3eae | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ddb31a8342 | ||
|   | 1c978b7cce | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9d9624c960 | ||
|   | 179245e1aa | ||
|   | 345000a0e9 | ||
|   | d94d5f96c3 | ||
|   | 0fa3538db5 | ||
|   | 1749725229 | 
| @@ -1,28 +1,25 @@ | ||||
| [modern] | ||||
| # Support for dynamic import is the main litmus test for serving modern builds. | ||||
| # Although officially a ES2020 feature, browsers implemented it early, so this | ||||
| # enables all of ES2017 and some features in ES2018. | ||||
| supports es6-module-dynamic-import | ||||
|  | ||||
| # Exclude Safari 11-12 because of a bug in tagged template literals | ||||
| # https://bugs.webkit.org/show_bug.cgi?id=190756 | ||||
| # Note: Dropping version 11 also enables several more ES2018 features | ||||
| not Safari < 13 | ||||
| not iOS < 13 | ||||
|  | ||||
| # Exclude KaiOS, QQ, and UC browsers due to lack of sufficient feature support data | ||||
| # Babel ignores these automatically, but we need here for Webpack to output ESM with dynamic imports | ||||
| # Modern builds target recent browsers supporting the latest features to minimize transpilation, polyfills, etc. | ||||
| # It is served to browsers meeting the following requirements: | ||||
| # - released in the last year + current alpha/beta versions | ||||
| # - Firefox extended support release (ESR) | ||||
| # - with global utilization at or above 0.5% | ||||
| # - must support dynamic import of ES modules | ||||
| # - exclude browsers no longer being maintained | ||||
| # - exclude KaiOS, QQ, and UC browsers due to lack of sufficient feature support data | ||||
| unreleased versions | ||||
| last 1 year | ||||
| Firefox ESR | ||||
| >= 0.5% and supports es6-module-dynamic-import | ||||
| not dead | ||||
| not KaiOS > 0 | ||||
| not QQAndroid > 0 | ||||
| not UCAndroid > 0 | ||||
|  | ||||
| # Exclude unsupported browsers | ||||
| not dead | ||||
|  | ||||
| [legacy] | ||||
| # Legacy builds are served when modern requirements are not met and support browsers: | ||||
| # - released in the last 7 years + current alpha/beta versionss | ||||
| # - with global utilization above 0.05% | ||||
| # - with global utilization at or above 0.05% | ||||
| # The lattermost query ensures that support for popular old browsers is not dropped too early | ||||
| # (e.g. IE 11, Android 4.4, or Samsung 4). | ||||
| # | ||||
| @@ -36,10 +33,10 @@ not dead | ||||
| # As of May 2023, only web sockets must be added to the query. | ||||
| unreleased versions | ||||
| last 7 years | ||||
| > 0.05% and supports websockets | ||||
| >= 0.05% and supports websockets | ||||
|  | ||||
| [legacy-sw] | ||||
| # Same as legacy plus supports service workers | ||||
| unreleased versions | ||||
| last 7 years | ||||
| > 0.05% and supports websockets and supports serviceworkers | ||||
| >= 0.05% and supports websockets and supports serviceworkers | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -89,7 +89,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@v4.3.4 | ||||
|         uses: actions/upload-artifact@v4.3.6 | ||||
|         with: | ||||
|           name: frontend-bundle-stats | ||||
|           path: build/stats/*.json | ||||
| @@ -113,7 +113,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@v4.3.4 | ||||
|         uses: actions/upload-artifact@v4.3.6 | ||||
|         with: | ||||
|           name: supervisor-bundle-stats | ||||
|           path: build/stats/*.json | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -57,14 +57,14 @@ jobs: | ||||
|         run: tar -czvf translations.tar.gz translations | ||||
|  | ||||
|       - name: Upload build artifacts | ||||
|         uses: actions/upload-artifact@v4.3.4 | ||||
|         uses: actions/upload-artifact@v4.3.6 | ||||
|         with: | ||||
|           name: wheels | ||||
|           path: dist/home_assistant_frontend*.whl | ||||
|           if-no-files-found: error | ||||
|  | ||||
|       - name: Upload translations | ||||
|         uses: actions/upload-artifact@v4.3.4 | ||||
|         uses: actions/upload-artifact@v4.3.6 | ||||
|         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@v2.1.11 | ||||
|         uses: relative-ci/agent-action@v2.1.12 | ||||
|         with: | ||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} | ||||
|           token: ${{ github.token }} | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -55,7 +55,7 @@ jobs: | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@v2.0.6 | ||||
|         uses: softprops/action-gh-release@v2.0.8 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
| @@ -76,7 +76,7 @@ jobs: | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2024.07.1 | ||||
|         with: | ||||
|           abi: cp311 | ||||
|           abi: cp312 | ||||
|           tag: musllinux_1_2 | ||||
|           arch: amd64 | ||||
|           wheels-key: ${{ secrets.WHEELS_KEY }} | ||||
|   | ||||
| @@ -1,4 +1 @@ | ||||
| #!/usr/bin/env sh | ||||
| . "$(dirname -- "$0")/_/husky.sh" | ||||
|  | ||||
| yarn run lint-staged --relative --shell "/bin/bash" | ||||
|   | ||||
							
								
								
									
										894
									
								
								.yarn/releases/yarn-4.3.1.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										894
									
								
								.yarn/releases/yarn-4.3.1.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										925
									
								
								.yarn/releases/yarn-4.4.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										925
									
								
								.yarn/releases/yarn-4.4.0.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.3.1.cjs | ||||
| yarnPath: .yarn/releases/yarn-4.4.0.cjs | ||||
|   | ||||
| @@ -140,8 +140,14 @@ module.exports.babelOptions = ({ | ||||
|       "@babel/plugin-transform-runtime", | ||||
|       { version: dependencies["@babel/runtime"] }, | ||||
|     ], | ||||
|     // Support  some proposals still in TC39 process | ||||
|     ["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }], | ||||
|     // 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), | ||||
|   exclude: [ | ||||
|     // \\ for Windows, / for Mac OS and Linux | ||||
|   | ||||
| @@ -1,19 +1,54 @@ | ||||
| // Tasks to compress | ||||
|  | ||||
| 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}"; | ||||
| const brotliOptions = { | ||||
|   skipLarger: true, | ||||
|   params: { | ||||
|     [constants.BROTLI_PARAM_QUALITY]: constants.BROTLI_MAX_QUALITY, | ||||
|   }, | ||||
| }; | ||||
| const zopfliOptions = { threshold: 150 }; | ||||
|  | ||||
| const compressDist = (rootDir) => | ||||
| const compressDistBrotli = (rootDir, modernDir) => | ||||
|   gulp | ||||
|     .src([ | ||||
|       `${rootDir}/**/*.{js,json,css,svg,xml}`, | ||||
|     .src([`${modernDir}/**/${filesGlob}`, `${rootDir}/sw-modern.js`], { | ||||
|       base: rootDir, | ||||
|     }) | ||||
|     .pipe(brotli(brotliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
|  | ||||
| const compressDistZopfli = (rootDir, modernDir) => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         `${rootDir}/**/${filesGlob}`, | ||||
|         `!${modernDir}/**/${filesGlob}`, | ||||
|         `!${rootDir}/{sw-modern,service_worker}.js`, | ||||
|         `${rootDir}/{authorize,onboarding}.html`, | ||||
|     ]) | ||||
|       ], | ||||
|       { base: rootDir } | ||||
|     ) | ||||
|     .pipe(zopfli(zopfliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
|  | ||||
| gulp.task("compress-app", () => compressDist(paths.app_output_root)); | ||||
| gulp.task("compress-hassio", () => compressDist(paths.hassio_output_root)); | ||||
| const compressAppBrotli = () => | ||||
|   compressDistBrotli(paths.app_output_root, paths.app_output_latest); | ||||
| const compressHassioBrotli = () => | ||||
|   compressDistBrotli(paths.hassio_output_root, paths.hassio_output_latest); | ||||
|  | ||||
| const compressAppZopfli = () => | ||||
|   compressDistZopfli(paths.app_output_root, paths.app_output_latest); | ||||
| const compressHassioZopfli = () => | ||||
|   compressDistZopfli(paths.hassio_output_root, paths.hassio_output_latest); | ||||
|  | ||||
| gulp.task("compress-app", gulp.parallel(compressAppBrotli, compressAppZopfli)); | ||||
| gulp.task( | ||||
|   "compress-hassio", | ||||
|   gulp.parallel(compressHassioBrotli, compressHassioZopfli) | ||||
| ); | ||||
|   | ||||
| @@ -1,28 +1,76 @@ | ||||
| // Tasks to generate entry HTML | ||||
|  | ||||
| import { | ||||
|   applyVersionsToRegexes, | ||||
|   compileRegex, | ||||
|   getPreUserAgentRegexes, | ||||
| } from "browserslist-useragent-regexp"; | ||||
| import fs from "fs-extra"; | ||||
| import gulp from "gulp"; | ||||
| import { minify } from "html-minifier-terser"; | ||||
| import template from "lodash.template"; | ||||
| import path from "path"; | ||||
| import { dirname, extname, resolve } from "node:path"; | ||||
| import { htmlMinifierOptions, terserOptions } from "../bundle.cjs"; | ||||
| import env from "../env.cjs"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| // macOS companion app has no way to obtain the Safari version used by WKWebView, | ||||
| // and it is not in the default user agent string. So we add an additional regex | ||||
| // to serve modern based on a minimum macOS version. We take the minimum Safari | ||||
| // major version from browserslist and manually map that to a supported macOS | ||||
| // version. Note this assumes the user has kept Safari updated. | ||||
| const HA_MACOS_REGEX = | ||||
|   /Home Assistant\/[\d.]+ \(.+; macOS (\d+)\.(\d+)(?:\.(\d+))?\)/; | ||||
| const SAFARI_TO_MACOS = { | ||||
|   15: [10, 15, 0], | ||||
|   16: [11, 0, 0], | ||||
|   17: [12, 0, 0], | ||||
|   18: [13, 0, 0], | ||||
| }; | ||||
|  | ||||
| const getCommonTemplateVars = () => { | ||||
|   const browserRegexes = getPreUserAgentRegexes({ | ||||
|     env: "modern", | ||||
|     allowHigherVersions: true, | ||||
|     mobileToDesktop: true, | ||||
|     throwOnMissing: true, | ||||
|   }); | ||||
|   const minSafariVersion = browserRegexes.find( | ||||
|     (regex) => regex.family === "safari" | ||||
|   )?.matchedVersions[0][0]; | ||||
|   const minMacOSVersion = SAFARI_TO_MACOS[minSafariVersion]; | ||||
|   if (!minMacOSVersion) { | ||||
|     throw Error( | ||||
|       `Could not find minimum MacOS version for Safari ${minSafariVersion}.` | ||||
|     ); | ||||
|   } | ||||
|   const haMacOSRegex = applyVersionsToRegexes( | ||||
|     [ | ||||
|       { | ||||
|         family: "ha_macos", | ||||
|         regex: HA_MACOS_REGEX, | ||||
|         matchedVersions: [minMacOSVersion], | ||||
|         requestVersions: [minMacOSVersion], | ||||
|       }, | ||||
|     ], | ||||
|     { ignorePatch: true, allowHigherVersions: true } | ||||
|   ); | ||||
|   return { | ||||
|     useRollup: env.useRollup(), | ||||
|     useWDS: env.useWDS(), | ||||
|     modernRegex: compileRegex(browserRegexes.concat(haMacOSRegex)).toString(), | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| const renderTemplate = (templateFile, data = {}) => { | ||||
|   const compiled = template( | ||||
|     fs.readFileSync(templateFile, { encoding: "utf-8" }) | ||||
|   ); | ||||
|   return compiled({ | ||||
|     ...data, | ||||
|     useRollup: env.useRollup(), | ||||
|     useWDS: env.useWDS(), | ||||
|     // Resolve any child/nested templates relative to the parent and pass the same data | ||||
|     renderTemplate: (childTemplate) => | ||||
|       renderTemplate( | ||||
|         path.resolve(path.dirname(templateFile), childTemplate), | ||||
|         data | ||||
|       ), | ||||
|       renderTemplate(resolve(dirname(templateFile), childTemplate), data), | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| @@ -56,10 +104,12 @@ const genPagesDevTask = | ||||
|     publicRoot = "" | ||||
|   ) => | ||||
|   async () => { | ||||
|     const commonVars = getCommonTemplateVars(); | ||||
|     for (const [page, entries] of Object.entries(pageEntries)) { | ||||
|       const content = renderTemplate( | ||||
|         path.resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         { | ||||
|           ...commonVars, | ||||
|           latestEntryJS: entries.map((entry) => | ||||
|             useWDS | ||||
|               ? `http://localhost:8000/src/entrypoints/${entry}.ts` | ||||
| @@ -74,7 +124,7 @@ const genPagesDevTask = | ||||
|           es5CustomPanelJS: `${publicRoot}/frontend_es5/custom-panel.js`, | ||||
|         } | ||||
|       ); | ||||
|       fs.outputFileSync(path.resolve(outputRoot, page), content); | ||||
|       fs.outputFileSync(resolve(outputRoot, page), content); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
| @@ -91,16 +141,18 @@ const genPagesProdTask = | ||||
|   ) => | ||||
|   async () => { | ||||
|     const latestManifest = fs.readJsonSync( | ||||
|       path.resolve(outputLatest, "manifest.json") | ||||
|       resolve(outputLatest, "manifest.json") | ||||
|     ); | ||||
|     const es5Manifest = outputES5 | ||||
|       ? fs.readJsonSync(path.resolve(outputES5, "manifest.json")) | ||||
|       ? fs.readJsonSync(resolve(outputES5, "manifest.json")) | ||||
|       : {}; | ||||
|     const commonVars = getCommonTemplateVars(); | ||||
|     const minifiedHTML = []; | ||||
|     for (const [page, entries] of Object.entries(pageEntries)) { | ||||
|       const content = renderTemplate( | ||||
|         path.resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         { | ||||
|           ...commonVars, | ||||
|           latestEntryJS: entries.map((entry) => latestManifest[`${entry}.js`]), | ||||
|           es5EntryJS: entries.map((entry) => es5Manifest[`${entry}.js`]), | ||||
|           latestCustomPanelJS: latestManifest["custom-panel.js"], | ||||
| @@ -108,8 +160,8 @@ const genPagesProdTask = | ||||
|         } | ||||
|       ); | ||||
|       minifiedHTML.push( | ||||
|         minifyHtml(content, path.extname(page)).then((minified) => | ||||
|           fs.outputFileSync(path.resolve(outputRoot, page), minified) | ||||
|         minifyHtml(content, extname(page)).then((minified) => | ||||
|           fs.outputFileSync(resolve(outputRoot, page), minified) | ||||
|         ) | ||||
|       ); | ||||
|     } | ||||
|   | ||||
| @@ -2,8 +2,8 @@ | ||||
|  | ||||
| import { deleteAsync } from "del"; | ||||
| import gulp from "gulp"; | ||||
| import { mkdir, readFile, writeFile } from "node:fs/promises"; | ||||
| import { join, relative } from "node:path"; | ||||
| import { mkdir, readFile, symlink, writeFile } from "node:fs/promises"; | ||||
| import { basename, join, relative } from "node:path"; | ||||
| import { injectManifest } from "workbox-build"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| @@ -41,10 +41,11 @@ gulp.task("gen-service-worker-app-prod", () => | ||||
|         await readFile(join(outPath, "manifest.json"), "utf-8") | ||||
|       ); | ||||
|       const swSrc = join(paths.app_output_root, manifest["service-worker.js"]); | ||||
|       const swDest = join(paths.app_output_root, `sw-${build}.js`); | ||||
|       const buildDir = relative(paths.app_output_root, outPath); | ||||
|       const { warnings } = await injectManifest({ | ||||
|         swSrc, | ||||
|         swDest: join(paths.app_output_root, `sw-${build}.js`), | ||||
|         swDest, | ||||
|         injectionPoint: "__WB_MANIFEST__", | ||||
|         // Files that mach this pattern will be considered unique and skip revision check | ||||
|         // ignore JS files + translation files | ||||
| @@ -76,6 +77,11 @@ gulp.task("gen-service-worker-app-prod", () => | ||||
|         ); | ||||
|       } | ||||
|       await deleteAsync(`${swSrc}?(.map)`); | ||||
|       // Needed to install new SW from a cached HTML | ||||
|       if (build === "modern") { | ||||
|         const swOld = join(paths.app_output_root, "service_worker.js"); | ||||
|         await symlink(basename(swDest), swOld); | ||||
|       } | ||||
|     }) | ||||
|   ) | ||||
| ); | ||||
|   | ||||
							
								
								
									
										5
									
								
								cast/public/sw-legacy.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								cast/public/sw-legacy.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| "use strict"; | ||||
|  | ||||
| self.addEventListener("fetch", (event) => { | ||||
|   event.respondWith(fetch(event.request)); | ||||
| }); | ||||
| @@ -36,13 +36,7 @@ | ||||
|   </head> | ||||
|   <body> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <script> | ||||
|       <% for (const entry of latestEntryJS) { %> | ||||
|         import("<%= entry %>"); | ||||
|       <% } %> | ||||
|       window.latestJS = true; | ||||
|     </script> | ||||
|     <%= renderTemplate("../../../src/html/_script_load_es5.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | ||||
|     <hc-layout subtitle="FAQ"> | ||||
|       <style> | ||||
|         a { | ||||
|   | ||||
| @@ -13,15 +13,9 @@ | ||||
|     <%= renderTemplate("_social_meta.html.template") %> | ||||
|   </head> | ||||
|   <body> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <hc-connect></hc-connect> | ||||
|     <script> | ||||
|       <% for (const entry of latestEntryJS) { %> | ||||
|         import("<%= entry %>"); | ||||
|       <% } %> | ||||
|       window.latestJS = true; | ||||
|     </script> | ||||
|     <%= renderTemplate("../../../src/html/_script_load_es5.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | ||||
|     <script> | ||||
|     (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){ | ||||
|     (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o), | ||||
|   | ||||
| @@ -16,14 +16,8 @@ | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <cast-media-player></cast-media-player> | ||||
|     <script> | ||||
|       <% for (const entry of latestEntryJS) { %> | ||||
|         import("<%= entry %>"); | ||||
|       <% } %> | ||||
|       window.latestJS = true; | ||||
|     </script> | ||||
|     <%= renderTemplate("../../../src/html/_script_load_es5.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										5
									
								
								demo/public/sw-legacy.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/public/sw-legacy.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| "use strict"; | ||||
|  | ||||
| self.addEventListener("fetch", (event) => { | ||||
|   event.respondWith(fetch(event.request)); | ||||
| }); | ||||
| @@ -82,6 +82,8 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         has_entity_name: false, | ||||
|         unique_id: "co2_intensity", | ||||
|         options: null, | ||||
|         created_at: 0, | ||||
|         modified_at: 0, | ||||
|       }, | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
| @@ -100,6 +102,8 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         has_entity_name: false, | ||||
|         unique_id: "grid_fossil_fuel_percentage", | ||||
|         options: null, | ||||
|         created_at: 0, | ||||
|         modified_at: 0, | ||||
|       }, | ||||
|     ]); | ||||
|  | ||||
|   | ||||
| @@ -63,35 +63,47 @@ | ||||
|         align-items: center; | ||||
|       } | ||||
|       #ha-launch-screen svg { | ||||
|         width: 170px; | ||||
|         width: 112px; | ||||
|         flex-shrink: 0; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer { | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-top { | ||||
|         flex: 1; | ||||
|         margin-top: calc( 2 * max(env(safe-area-inset-bottom), 48px) + 46px ); | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-bottom { | ||||
|         flex: 1; | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       .ohf-logo { | ||||
|         margin: max(env(safe-area-inset-bottom), 48px) 0; | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         opacity: .66; | ||||
|       } | ||||
|       @media (prefers-color-scheme: dark) { | ||||
|         .ohf-logo { | ||||
|           filter: invert(1); | ||||
|         } | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="ha-launch-screen"> | ||||
|       <div class="ha-launch-screen-spacer"></div> | ||||
|       <div class="ha-launch-screen-spacer-top"></div> | ||||
|       <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 240 240"> | ||||
|         <path fill="#18BCF2" d="M240 224.762a15 15 0 0 1-15 15H15a15 15 0 0 1-15-15v-90c0-8.25 4.77-19.769 10.61-25.609l98.78-98.7805c5.83-5.83 15.38-5.83 21.21 0l98.79 98.7895c5.83 5.83 10.61 17.36 10.61 25.61v90-.01Z"/> | ||||
|         <path fill="#F2F4F9" d="m107.27 239.762-40.63-40.63c-2.09.72-4.32 1.13-6.64 1.13-11.3 0-20.5-9.2-20.5-20.5s9.2-20.5 20.5-20.5 20.5 9.2 20.5 20.5c0 2.33-.41 4.56-1.13 6.65l31.63 31.63v-115.88c-6.8-3.3395-11.5-10.3195-11.5-18.3895 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5c0 8.07-4.7 15.05-11.5 18.3895v81.27l31.46-31.46c-.62-1.96-.96-4.04-.96-6.2 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5-9.2 20.5-20.5 20.5c-2.5 0-4.88-.47-7.09-1.29L129 208.892v30.88z"/> | ||||
|       </svg> | ||||
|       <div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer"></div> | ||||
|       <div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer-bottom"></div> | ||||
|       <div class="ohf-logo"> | ||||
|         <img src="/static/images/ohf-badge.svg" alt="Home Assistant is a project by the Open Home Foundation" height="46"> | ||||
|       </div> | ||||
|     </div> | ||||
|     <ha-demo></ha-demo> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_preload_roboto.html.template") %> | ||||
|     <script> | ||||
|       // Safari 12 and below does not have a compliant ES2015 implementation of template literals, so we ship ES5 | ||||
|       if (!isS11_12) { | ||||
|         <% for (const entry of latestEntryJS) { %> | ||||
|           import("<%= entry %>"); | ||||
|         <% } %> | ||||
|         window.latestJS = true; | ||||
|       } | ||||
|     </script> | ||||
|     <%= renderTemplate("../../../src/html/_script_load_es5.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/paulus.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/paulus.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 110 KiB | 
| @@ -532,15 +532,6 @@ export default { | ||||
|     last_changed: "2018-07-19T10:44:46.200946+00:00", | ||||
|     last_updated: "2018-07-19T10:44:46.200946+00:00", | ||||
|   }, | ||||
|   "mailbox.demomailbox": { | ||||
|     entity_id: "mailbox.demomailbox", | ||||
|     state: "10", | ||||
|     attributes: { | ||||
|       friendly_name: "DemoMailbox", | ||||
|     }, | ||||
|     last_changed: "2018-07-19T10:45:16.555210+00:00", | ||||
|     last_updated: "2018-07-19T10:45:16.555210+00:00", | ||||
|   }, | ||||
|   "input_select.living_room_preset": { | ||||
|     entity_id: "input_select.living_room_preset", | ||||
|     state: "Visitors", | ||||
|   | ||||
| @@ -15,6 +15,7 @@ import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { DeviceRegistryEntry } from "../../../../src/data/device_registry"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
| @@ -41,7 +42,7 @@ const ENTITIES = [ | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES = [ | ||||
| const DEVICES: DeviceRegistryEntry[] = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
| @@ -61,6 +62,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
| @@ -81,6 +84,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
| @@ -101,6 +106,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -113,6 +120,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
| @@ -122,6 +131,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
| @@ -131,6 +142,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -21,6 +21,7 @@ import { FloorRegistryEntry } from "../../../../src/data/floor_registry"; | ||||
| import { LabelRegistryEntry } from "../../../../src/data/label_registry"; | ||||
| import { mockFloorRegistry } from "../../../../demo/src/stubs/floor_registry"; | ||||
| import { mockLabelRegistry } from "../../../../demo/src/stubs/label_registry"; | ||||
| import { DeviceRegistryEntry } from "../../../../src/data/device_registry"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
| @@ -41,7 +42,7 @@ const ENTITIES = [ | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES = [ | ||||
| const DEVICES: DeviceRegistryEntry[] = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
| @@ -61,6 +62,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
| @@ -81,6 +84,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
| @@ -101,6 +106,8 @@ const DEVICES = [ | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -113,6 +120,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
| @@ -122,6 +131,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
| @@ -131,6 +142,8 @@ const AREAS: AreaRegistryEntry[] = [ | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -141,6 +154,8 @@ const FLOORS: FloorRegistryEntry[] = [ | ||||
|     level: 0, | ||||
|     icon: null, | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     floor_id: "first", | ||||
| @@ -148,6 +163,8 @@ const FLOORS: FloorRegistryEntry[] = [ | ||||
|     level: 1, | ||||
|     icon: "mdi:numeric-1", | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     floor_id: "second", | ||||
| @@ -155,6 +172,8 @@ const FLOORS: FloorRegistryEntry[] = [ | ||||
|     level: 2, | ||||
|     icon: "mdi:numeric-2", | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -165,6 +184,8 @@ const LABELS: LabelRegistryEntry[] = [ | ||||
|     icon: null, | ||||
|     color: "yellow", | ||||
|     description: null, | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     label_id: "entertainment", | ||||
| @@ -172,6 +193,8 @@ const LABELS: LabelRegistryEntry[] = [ | ||||
|     icon: "mdi:popcorn", | ||||
|     color: "blue", | ||||
|     description: null, | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -287,11 +287,11 @@ const CONFIGS = [ | ||||
|     config: ` | ||||
| - type: entities | ||||
|   entities: | ||||
|     - type: call-service | ||||
|     - type: perform-action | ||||
|       icon: mdi:power | ||||
|       name: Bed light | ||||
|       action_name: Toggle light | ||||
|       service: light.toggle | ||||
|       action: light.toggle | ||||
|       data: | ||||
|         entity_id: light.bed_light | ||||
|     - type: section | ||||
|   | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/picture-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/picture-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Picture Card | ||||
| --- | ||||
							
								
								
									
										61
									
								
								gallery/src/pages/lovelace/picture-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								gallery/src/pages/lovelace/picture-card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| import { mockIcons } from "../../../../demo/src/stubs/icons"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("person", "paulus", "home", { | ||||
|     friendly_name: "Paulus", | ||||
|     entity_picture: "/images/paulus.jpg", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
|   { | ||||
|     heading: "Image URL", | ||||
|     config: ` | ||||
| - type: picture | ||||
|   image: /images/living_room.png | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Person entity", | ||||
|     config: ` | ||||
| - type: picture | ||||
|   image_entity: person.paulus | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Error: Image required", | ||||
|     config: ` | ||||
| - type: picture | ||||
|   entity: person.paulus | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-picture-card") | ||||
| class DemoPicture extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("lovelace", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     mockIcons(hass); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-lovelace-picture-card": DemoPicture; | ||||
|   } | ||||
| } | ||||
| @@ -25,6 +25,15 @@ const ENTITIES = [ | ||||
|     friendly_name: "Movement Backyard", | ||||
|     device_class: "motion", | ||||
|   }), | ||||
|   getEntity("person", "paulus", "home", { | ||||
|     friendly_name: "Paulus", | ||||
|     entity_picture: "/images/paulus.jpg", | ||||
|   }), | ||||
|   getEntity("sensor", "battery", 35, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
| @@ -123,6 +132,19 @@ const CONFIGS = [ | ||||
|         left: 35% | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Person entity", | ||||
|     config: ` | ||||
| - type: picture-elements | ||||
|   image_entity: person.paulus | ||||
|   elements: | ||||
|   - type: state-icon | ||||
|     entity: sensor.battery | ||||
|     style: | ||||
|       top: 8% | ||||
|       left: 8% | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-picture-elements-card") | ||||
|   | ||||
| @@ -12,6 +12,10 @@ const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "off", { | ||||
|     friendly_name: "Bed Light", | ||||
|   }), | ||||
|   getEntity("person", "paulus", "home", { | ||||
|     friendly_name: "Paulus", | ||||
|     entity_picture: "/images/paulus.jpg", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
| @@ -50,6 +54,13 @@ const CONFIGS = [ | ||||
|   entity: camera.demo_camera | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Person entity", | ||||
|     config: ` | ||||
| - type: picture-entity | ||||
|   entity: person.paulus | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Hidden name", | ||||
|     config: ` | ||||
|   | ||||
| @@ -20,6 +20,15 @@ const ENTITIES = [ | ||||
|     friendly_name: "Basement Floor Wet", | ||||
|     device_class: "moisture", | ||||
|   }), | ||||
|   getEntity("person", "paulus", "home", { | ||||
|     friendly_name: "Paulus", | ||||
|     entity_picture: "/images/paulus.jpg", | ||||
|   }), | ||||
|   getEntity("sensor", "battery", 35, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
| @@ -90,6 +99,15 @@ const CONFIGS = [ | ||||
|     - light.ceiling_lights | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Person entity", | ||||
|     config: ` | ||||
| - type: picture-glance | ||||
|   image_entity: person.paulus | ||||
|   entities: | ||||
|     - sensor.battery | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Custom icon", | ||||
|     config: ` | ||||
|   | ||||
| @@ -8,6 +8,7 @@ import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| import { mockIcons } from "../../../../demo/src/stubs/icons"; | ||||
| import { ClimateEntityFeature } from "../../../../src/data/climate"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("switch", "tv_outlet", "on", { | ||||
| @@ -60,6 +61,36 @@ const ENTITIES = [ | ||||
|       CoverEntityFeature.OPEN_TILT + | ||||
|       CoverEntityFeature.STOP_TILT, | ||||
|   }), | ||||
|   getEntity("input_number", "counter", "1.0", { | ||||
|     friendly_name: "Counter", | ||||
|     initial: 0, | ||||
|     min: 0, | ||||
|     max: 100, | ||||
|     step: 1, | ||||
|     mode: "slider", | ||||
|   }), | ||||
|   getEntity("climate", "dual_thermostat", "heat/cool", { | ||||
|     friendly_name: "Dual thermostat", | ||||
|     hvac_modes: ["off", "cool", "heat_cool", "auto", "dry", "fan_only"], | ||||
|     min_temp: 7, | ||||
|     max_temp: 35, | ||||
|     fan_modes: ["on_low", "on_high", "auto_low", "auto_high", "off"], | ||||
|     preset_modes: ["home", "eco", "away"], | ||||
|     swing_modes: ["auto", "1", "2", "3", "off"], | ||||
|     current_temperature: 23, | ||||
|     target_temp_high: 24, | ||||
|     target_temp_low: 21, | ||||
|     fan_mode: "auto_low", | ||||
|     preset_mode: "home", | ||||
|     swing_mode: "auto", | ||||
|     supported_features: | ||||
|       ClimateEntityFeature.TURN_ON + | ||||
|       ClimateEntityFeature.TURN_OFF + | ||||
|       ClimateEntityFeature.SWING_MODE + | ||||
|       ClimateEntityFeature.PRESET_MODE + | ||||
|       ClimateEntityFeature.FAN_MODE + | ||||
|       ClimateEntityFeature.TARGET_TEMPERATURE_RANGE, | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
| @@ -193,6 +224,25 @@ const CONFIGS = [ | ||||
|   - type: "cover-tilt" | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Number buttons feature", | ||||
|     config: ` | ||||
| - type: tile | ||||
|   entity: input_number.counter | ||||
|   features: | ||||
|   - type: numeric-input | ||||
|     style: buttons | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Dual thermostat feature", | ||||
|     config: ` | ||||
| - type: tile | ||||
|   entity: climate.dual_thermostat | ||||
|   features: | ||||
|   - type: target-temperature | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-tile-card") | ||||
|   | ||||
| @@ -358,13 +358,11 @@ export class DemoEntityState extends LitElement { | ||||
|         }, | ||||
|         entity_id: { | ||||
|           title: "Entity ID", | ||||
|           width: "30%", | ||||
|           filterable: true, | ||||
|           sortable: true, | ||||
|         }, | ||||
|         state: { | ||||
|           title: "State", | ||||
|           width: "20%", | ||||
|           sortable: true, | ||||
|           template: (entry) => | ||||
|             html`${computeStateDisplay( | ||||
| @@ -379,14 +377,12 @@ export class DemoEntityState extends LitElement { | ||||
|         device_class: { | ||||
|           title: "Device class", | ||||
|           template: (entry) => html`${entry.device_class ?? "-"}`, | ||||
|           width: "20%", | ||||
|           filterable: true, | ||||
|           sortable: true, | ||||
|         }, | ||||
|         domain: { | ||||
|           title: "Domain", | ||||
|           template: (entry) => html`${computeDomain(entry.entity_id)}`, | ||||
|           width: "20%", | ||||
|           filterable: true, | ||||
|           sortable: true, | ||||
|         }, | ||||
|   | ||||
| @@ -203,6 +203,8 @@ const createEntityRegistryEntries = ( | ||||
|     options: null, | ||||
|     labels: [], | ||||
|     categories: {}, | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -228,6 +230,8 @@ const createDeviceRegistryEntries = ( | ||||
|     disabled_by: null, | ||||
|     configuration_url: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -127,14 +127,13 @@ export class HassioBackups extends LitElement { | ||||
|         main: true, | ||||
|         sortable: true, | ||||
|         filterable: true, | ||||
|         grows: true, | ||||
|         flex: 2, | ||||
|         template: (backup) => | ||||
|           html`${backup.name || backup.slug} | ||||
|             <div class="secondary">${backup.secondary}</div>`, | ||||
|       }, | ||||
|       size: { | ||||
|         title: this.supervisor.localize("backup.size"), | ||||
|         width: "15%", | ||||
|         hidden: narrow, | ||||
|         filterable: true, | ||||
|         sortable: true, | ||||
| @@ -142,7 +141,6 @@ export class HassioBackups extends LitElement { | ||||
|       }, | ||||
|       location: { | ||||
|         title: this.supervisor.localize("backup.location"), | ||||
|         width: "15%", | ||||
|         hidden: narrow, | ||||
|         filterable: true, | ||||
|         sortable: true, | ||||
| @@ -151,7 +149,6 @@ export class HassioBackups extends LitElement { | ||||
|       }, | ||||
|       date: { | ||||
|         title: this.supervisor.localize("backup.created"), | ||||
|         width: "15%", | ||||
|         direction: "desc", | ||||
|         hidden: narrow, | ||||
|         filterable: true, | ||||
|   | ||||
| @@ -66,7 +66,8 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|           repo.slug !== "core" && // The core add-ons repository | ||||
|           repo.slug !== "local" && // Locally managed add-ons | ||||
|           repo.slug !== "a0d7b954" && // Home Assistant Community Add-ons | ||||
|           repo.slug !== "5c53de3b" // The ESPHome repository | ||||
|           repo.slug !== "5c53de3b" && // The ESPHome repository | ||||
|           repo.slug !== "d5369777" // Music Assistant repository | ||||
|       ) | ||||
|       .sort((a, b) => | ||||
|         caseInsensitiveStringCompare(a.name, b.name, this.hass.locale.language) | ||||
|   | ||||
| @@ -4,11 +4,7 @@ | ||||
|     el.src = src; | ||||
|     document.body.appendChild(el); | ||||
|   } | ||||
|   if (/.*Version\/(?:11|12)(?:\.\d+)*.*Safari\//.test(navigator.userAgent)) { | ||||
|     <% for (const entry of es5EntryJS) { %> | ||||
|       loadES5("<%= entry %>"); | ||||
|     <% } %> | ||||
|   } else { | ||||
|   if (<%= modernRegex %>.test(navigator.userAgent)) { | ||||
|     try { | ||||
|         <% for (const entry of latestEntryJS) { %> | ||||
|           new Function("import('<%= entry %>')")(); | ||||
| @@ -17,6 +13,10 @@ | ||||
|       <% for (const entry of es5EntryJS) { %> | ||||
|         loadES5("<%= entry %>"); | ||||
|       <% } %> | ||||
|   } else { | ||||
|     <% for (const entry of es5EntryJS) { %> | ||||
|       loadES5("<%= entry %>"); | ||||
|     <% } %> | ||||
|   } | ||||
|   } | ||||
| })(); | ||||
|   | ||||
							
								
								
									
										72
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										72
									
								
								package.json
									
									
									
									
									
								
							| @@ -16,7 +16,7 @@ | ||||
|     "lint:lit": "lit-analyzer \"{.,*}/src/**/*.ts\"", | ||||
|     "lint": "yarn run lint:eslint && yarn run lint:prettier && yarn run lint:types && yarn run lint:lit", | ||||
|     "format": "yarn run format:eslint && yarn run format:prettier", | ||||
|     "postinstall": "husky install", | ||||
|     "postinstall": "husky", | ||||
|     "prepack": "pinst --disable", | ||||
|     "postpack": "pinst --enable", | ||||
|     "test": "instant-mocha --webpack-config ./test/webpack.config.js --require ./test/setup.cjs \"test/**/*.ts\"" | ||||
| @@ -25,15 +25,15 @@ | ||||
|   "license": "Apache-2.0", | ||||
|   "type": "module", | ||||
|   "dependencies": { | ||||
|     "@babel/runtime": "7.24.8", | ||||
|     "@braintree/sanitize-url": "7.0.4", | ||||
|     "@codemirror/autocomplete": "6.17.0", | ||||
|     "@babel/runtime": "7.25.4", | ||||
|     "@braintree/sanitize-url": "7.1.0", | ||||
|     "@codemirror/autocomplete": "6.18.0", | ||||
|     "@codemirror/commands": "6.6.0", | ||||
|     "@codemirror/language": "6.10.2", | ||||
|     "@codemirror/legacy-modes": "6.4.0", | ||||
|     "@codemirror/legacy-modes": "6.4.1", | ||||
|     "@codemirror/search": "6.5.6", | ||||
|     "@codemirror/state": "6.4.1", | ||||
|     "@codemirror/view": "6.28.4", | ||||
|     "@codemirror/view": "6.32.0", | ||||
|     "@egjs/hammerjs": "2.0.17", | ||||
|     "@formatjs/intl-datetimeformat": "6.12.5", | ||||
|     "@formatjs/intl-displaynames": "6.6.8", | ||||
| @@ -49,11 +49,11 @@ | ||||
|     "@fullcalendar/list": "6.1.15", | ||||
|     "@fullcalendar/luxon3": "6.1.15", | ||||
|     "@fullcalendar/timegrid": "6.1.15", | ||||
|     "@lezer/highlight": "1.2.0", | ||||
|     "@lezer/highlight": "1.2.1", | ||||
|     "@lit-labs/context": "0.4.1", | ||||
|     "@lit-labs/motion": "1.0.7", | ||||
|     "@lit-labs/observers": "2.0.2", | ||||
|     "@lit-labs/virtualizer": "2.0.13", | ||||
|     "@lit-labs/virtualizer": "2.0.14", | ||||
|     "@lrnwebcomponents/simple-tooltip": "8.0.2", | ||||
|     "@material/chips": "=14.0.0-canary.53b3cad2f.0", | ||||
|     "@material/data-table": "=14.0.0-canary.53b3cad2f.0", | ||||
| @@ -80,7 +80,7 @@ | ||||
|     "@material/mwc-top-app-bar": "0.27.0", | ||||
|     "@material/mwc-top-app-bar-fixed": "0.27.0", | ||||
|     "@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0", | ||||
|     "@material/web": "1.5.1", | ||||
|     "@material/web": "2.1.0", | ||||
|     "@mdi/js": "7.4.47", | ||||
|     "@mdi/svg": "7.4.47", | ||||
|     "@polymer/paper-item": "3.0.1", | ||||
| @@ -88,8 +88,8 @@ | ||||
|     "@polymer/paper-tabs": "3.1.0", | ||||
|     "@polymer/polymer": "3.5.1", | ||||
|     "@thomasloven/round-slider": "0.6.0", | ||||
|     "@vaadin/combo-box": "24.4.3", | ||||
|     "@vaadin/vaadin-themable-mixin": "24.4.3", | ||||
|     "@vaadin/combo-box": "24.4.6", | ||||
|     "@vaadin/vaadin-themable-mixin": "24.4.6", | ||||
|     "@vibrant/color": "3.2.1-alpha.1", | ||||
|     "@vibrant/core": "3.2.1-alpha.1", | ||||
|     "@vibrant/quantizer-mmcq": "3.2.1-alpha.1", | ||||
| @@ -97,10 +97,10 @@ | ||||
|     "@webcomponents/scoped-custom-element-registry": "0.0.9", | ||||
|     "@webcomponents/webcomponentsjs": "2.8.0", | ||||
|     "app-datepicker": "5.1.1", | ||||
|     "chart.js": "4.4.3", | ||||
|     "chart.js": "4.4.4", | ||||
|     "color-name": "2.0.0", | ||||
|     "comlink": "4.4.1", | ||||
|     "core-js": "3.37.1", | ||||
|     "core-js": "3.38.1", | ||||
|     "cropperjs": "1.6.2", | ||||
|     "date-fns": "3.6.0", | ||||
|     "date-fns-tz": "3.1.3", | ||||
| @@ -117,20 +117,20 @@ | ||||
|     "leaflet": "1.9.4", | ||||
|     "leaflet-draw": "1.0.4", | ||||
|     "lit": "2.8.0", | ||||
|     "luxon": "3.4.4", | ||||
|     "marked": "13.0.2", | ||||
|     "luxon": "3.5.0", | ||||
|     "marked": "14.0.0", | ||||
|     "memoize-one": "6.0.0", | ||||
|     "node-vibrant": "3.2.1-alpha.1", | ||||
|     "proxy-polyfill": "0.3.2", | ||||
|     "punycode": "2.3.1", | ||||
|     "qr-scanner": "1.4.2", | ||||
|     "qrcode": "1.5.3", | ||||
|     "qrcode": "1.5.4", | ||||
|     "roboto-fontface": "0.10.0", | ||||
|     "rrule": "2.8.1", | ||||
|     "sortablejs": "1.15.2", | ||||
|     "stacktrace-js": "2.0.2", | ||||
|     "superstruct": "2.0.2", | ||||
|     "tinykeys": "2.1.0", | ||||
|     "tinykeys": "3.0.0", | ||||
|     "tsparticles-engine": "2.12.0", | ||||
|     "tsparticles-preset-links": "2.12.0", | ||||
|     "ua-parser-js": "1.0.38", | ||||
| @@ -149,18 +149,18 @@ | ||||
|     "xss": "1.0.15" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@babel/core": "7.24.9", | ||||
|     "@babel/core": "7.25.2", | ||||
|     "@babel/helper-define-polyfill-provider": "0.6.2", | ||||
|     "@babel/plugin-proposal-decorators": "7.24.7", | ||||
|     "@babel/plugin-transform-runtime": "7.24.7", | ||||
|     "@babel/preset-env": "7.24.8", | ||||
|     "@babel/plugin-transform-runtime": "7.25.4", | ||||
|     "@babel/preset-env": "7.25.4", | ||||
|     "@babel/preset-typescript": "7.24.7", | ||||
|     "@bundle-stats/plugin-webpack-filter": "4.13.3", | ||||
|     "@bundle-stats/plugin-webpack-filter": "4.14.2", | ||||
|     "@koa/cors": "5.0.0", | ||||
|     "@lokalise/node-api": "12.6.0", | ||||
|     "@lokalise/node-api": "12.7.0", | ||||
|     "@octokit/auth-oauth-device": "7.1.1", | ||||
|     "@octokit/plugin-retry": "7.1.1", | ||||
|     "@octokit/rest": "21.0.0", | ||||
|     "@octokit/rest": "21.0.2", | ||||
|     "@open-wc/dev-server-hmr": "0.1.4", | ||||
|     "@rollup/plugin-babel": "6.0.4", | ||||
|     "@rollup/plugin-commonjs": "26.0.1", | ||||
| @@ -168,7 +168,7 @@ | ||||
|     "@rollup/plugin-node-resolve": "15.2.3", | ||||
|     "@rollup/plugin-replace": "5.0.7", | ||||
|     "@types/babel__plugin-transform-runtime": "7.9.5", | ||||
|     "@types/chromecast-caf-receiver": "6.0.16", | ||||
|     "@types/chromecast-caf-receiver": "6.0.17", | ||||
|     "@types/chromecast-caf-sender": "1.0.10", | ||||
|     "@types/color-name": "1.1.4", | ||||
|     "@types/glob": "8.1.0", | ||||
| @@ -185,12 +185,13 @@ | ||||
|     "@types/tar": "6.1.13", | ||||
|     "@types/ua-parser-js": "0.7.39", | ||||
|     "@types/webspeechapi": "0.0.29", | ||||
|     "@typescript-eslint/eslint-plugin": "7.16.1", | ||||
|     "@typescript-eslint/parser": "7.16.1", | ||||
|     "@typescript-eslint/eslint-plugin": "7.18.0", | ||||
|     "@typescript-eslint/parser": "7.18.0", | ||||
|     "@web/dev-server": "0.1.38", | ||||
|     "@web/dev-server-rollup": "0.4.1", | ||||
|     "babel-loader": "9.1.3", | ||||
|     "babel-plugin-template-html-minifier": "4.1.0", | ||||
|     "browserslist-useragent-regexp": "4.1.3", | ||||
|     "chai": "5.1.1", | ||||
|     "del": "7.1.0", | ||||
|     "eslint": "8.57.0", | ||||
| @@ -201,24 +202,25 @@ | ||||
|     "eslint-plugin-import": "2.29.1", | ||||
|     "eslint-plugin-lit": "1.14.0", | ||||
|     "eslint-plugin-lit-a11y": "4.1.4", | ||||
|     "eslint-plugin-unused-imports": "4.0.0", | ||||
|     "eslint-plugin-wc": "2.1.0", | ||||
|     "eslint-plugin-unused-imports": "4.1.3", | ||||
|     "eslint-plugin-wc": "2.1.1", | ||||
|     "fancy-log": "2.0.0", | ||||
|     "fs-extra": "11.2.0", | ||||
|     "glob": "11.0.0", | ||||
|     "gulp": "5.0.0", | ||||
|     "gulp-brotli": "3.0.0", | ||||
|     "gulp-json-transform": "0.5.0", | ||||
|     "gulp-rename": "2.0.0", | ||||
|     "gulp-zopfli-green": "6.0.2", | ||||
|     "html-minifier-terser": "7.2.0", | ||||
|     "husky": "9.0.11", | ||||
|     "husky": "9.1.5", | ||||
|     "instant-mocha": "1.5.2", | ||||
|     "jszip": "3.10.1", | ||||
|     "lint-staged": "15.2.7", | ||||
|     "lint-staged": "15.2.9", | ||||
|     "lit-analyzer": "2.0.3", | ||||
|     "lodash.merge": "4.6.2", | ||||
|     "lodash.template": "4.5.0", | ||||
|     "magic-string": "0.30.10", | ||||
|     "magic-string": "0.30.11", | ||||
|     "map-stream": "0.0.7", | ||||
|     "mocha": "10.5.0", | ||||
|     "object-hash": "3.0.0", | ||||
| @@ -232,12 +234,12 @@ | ||||
|     "serve-handler": "6.1.5", | ||||
|     "sinon": "18.0.0", | ||||
|     "systemjs": "6.15.1", | ||||
|     "tar": "7.4.0", | ||||
|     "tar": "7.4.3", | ||||
|     "terser-webpack-plugin": "5.3.10", | ||||
|     "transform-async-modules-webpack-plugin": "1.1.1", | ||||
|     "ts-lit-plugin": "2.0.2", | ||||
|     "typescript": "5.5.3", | ||||
|     "webpack": "5.93.0", | ||||
|     "typescript": "5.5.4", | ||||
|     "webpack": "5.94.0", | ||||
|     "webpack-cli": "5.1.4", | ||||
|     "webpack-dev-server": "5.0.4", | ||||
|     "webpack-manifest-plugin": "5.0.0", | ||||
| @@ -256,5 +258,5 @@ | ||||
|     "sortablejs@1.15.2": "patch:sortablejs@npm%3A1.15.2#~/.yarn/patches/sortablejs-npm-1.15.2-73347ae85a.patch", | ||||
|     "leaflet-draw@1.0.4": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch" | ||||
|   }, | ||||
|   "packageManager": "yarn@4.3.1" | ||||
|   "packageManager": "yarn@4.4.0" | ||||
| } | ||||
|   | ||||
							
								
								
									
										1
									
								
								public/static/icons/ohf.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								public/static/icons/ohf.svg
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| After Width: | Height: | Size: 5.2 KiB | 
							
								
								
									
										66
									
								
								public/static/images/ohf-badge.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								public/static/images/ohf-badge.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <svg id="b" data-name="Layer_2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 760.69 138.69"> | ||||
|   <g id="c" data-name="Layer_1"> | ||||
|     <g> | ||||
|       <g> | ||||
|         <path d="M136.22,68.35c3.31-.05,6.29.72,8.92,2.31s4.67,3.77,6.11,6.55,2.14,5.89,2.11,9.33c.03,3.44-.68,6.55-2.12,9.34s-3.49,4.98-6.13,6.56-5.62,2.36-8.93,2.31c-3.33,0-6.29-.78-8.88-2.33s-4.6-3.71-6.02-6.48-2.13-5.9-2.13-9.4c-.03-2.56.38-4.98,1.22-7.24s2.01-4.21,3.5-5.82,3.31-2.88,5.45-3.8,4.45-1.36,6.91-1.32ZM136.27,98.35c3.09,0,5.56-1.07,7.41-3.2s2.77-5,2.77-8.61-.91-6.52-2.73-8.65-4.3-3.19-7.44-3.19-5.63,1.06-7.46,3.19-2.75,5.01-2.75,8.65.91,6.52,2.74,8.64,4.32,3.18,7.48,3.18Z"/> | ||||
|         <path d="M184.16,80.53c0,3.47-1.06,6.27-3.18,8.41s-4.98,3.21-8.59,3.21h-7.45v12h-6.56v-35.18h14.06c3.64,0,6.5,1.04,8.59,3.11s3.13,4.89,3.13,8.45ZM177.25,80.39c0-1.64-.52-2.98-1.56-4.03s-2.52-1.57-4.44-1.57h-6.3v11.65h6.26c1.95,0,3.45-.55,4.49-1.65s1.56-2.57,1.56-4.39Z"/> | ||||
|         <path d="M210.82,98.02v6.14h-22.03v-35.18h21.98v6.19h-15.42v8.3h13.78v5.81h-13.78v8.74h15.47Z"/> | ||||
|         <path d="M246.95,68.98v35.18h-6.49l-16.08-24.77v24.77h-6.52v-35.18h6.52l16.08,24.77v-24.77h6.49Z"/> | ||||
|         <path d="M266.45,68.98h6.56v14.44l14.7.05v-14.48h6.63v35.18h-6.63v-14.84l-14.7-.09v14.93h-6.56v-35.18Z"/> | ||||
|         <path d="M316.41,68.35c3.31-.05,6.29.72,8.92,2.31s4.67,3.77,6.11,6.55,2.14,5.89,2.11,9.33c.03,3.44-.68,6.55-2.12,9.34s-3.49,4.98-6.13,6.56-5.62,2.36-8.93,2.31c-3.33,0-6.29-.78-8.88-2.33s-4.6-3.71-6.02-6.48-2.13-5.9-2.13-9.4c-.03-2.56.38-4.98,1.22-7.24s2.01-4.21,3.5-5.82,3.31-2.88,5.45-3.8,4.45-1.36,6.91-1.32ZM316.46,98.35c3.09,0,5.56-1.07,7.41-3.2s2.77-5,2.77-8.61-.91-6.52-2.73-8.65-4.3-3.19-7.44-3.19-5.63,1.06-7.46,3.19-2.75,5.01-2.75,8.65.91,6.52,2.74,8.64,4.32,3.18,7.48,3.18Z"/> | ||||
|         <path d="M373.66,68.98v35.18h-6.45v-20.55l-8.11,20.55h-6.23l-8.02-20.39v20.39h-6.28v-35.18h6.28l11.13,27.54,11.23-27.54h6.45Z"/> | ||||
|         <path d="M402.87,98.02v6.14h-22.03v-35.18h21.98v6.19h-15.42v8.3h13.78v5.81h-13.78v8.74h15.47Z"/> | ||||
|         <path d="M427.83,75.12v8.93h13.01l-.05,5.91h-12.96v14.2h-6.52v-35.18h21.98l-.05,6.14h-15.42Z"/> | ||||
|         <path d="M463.16,68.35c3.31-.05,6.29.72,8.92,2.31s4.67,3.77,6.11,6.55,2.14,5.89,2.11,9.33c.03,3.44-.68,6.55-2.12,9.34s-3.49,4.98-6.13,6.56-5.62,2.36-8.93,2.31c-3.33,0-6.29-.78-8.88-2.33s-4.6-3.71-6.02-6.48-2.13-5.9-2.13-9.4c-.03-2.56.38-4.98,1.22-7.24s2.01-4.21,3.5-5.82,3.31-2.88,5.45-3.8,4.45-1.36,6.91-1.32ZM463.21,98.35c3.09,0,5.56-1.07,7.41-3.2s2.77-5,2.77-8.61-.91-6.52-2.73-8.65-4.3-3.19-7.44-3.19-5.63,1.06-7.46,3.19-2.75,5.01-2.75,8.65.91,6.52,2.74,8.64,4.32,3.18,7.48,3.18Z"/> | ||||
|         <path d="M485,68.98h6.56v22.12c0,2.31.72,4.12,2.16,5.43s3.3,1.96,5.58,1.96,4.08-.67,5.58-2.02,2.25-3.13,2.25-5.37v-22.12h6.52v22.31c0,2.08-.38,3.98-1.14,5.7s-1.79,3.14-3.09,4.25-2.82,1.98-4.56,2.59-3.59.91-5.55.91c-2.59,0-4.96-.52-7.1-1.55s-3.88-2.58-5.2-4.65-1.99-4.49-1.99-7.25v-22.31Z"/> | ||||
|         <path d="M549.63,68.98v35.18h-6.49l-16.08-24.77v24.77h-6.52v-35.18h6.52l16.08,24.77v-24.77h6.49Z"/> | ||||
|         <path d="M586.9,86.58c.05,3.34-.71,6.37-2.27,9.08s-3.7,4.82-6.42,6.32-5.73,2.23-9.02,2.18h-12.42v-35.18h12.42c2.45-.03,4.78.39,6.98,1.28s4.1,2.1,5.68,3.66,2.84,3.43,3.75,5.64,1.35,4.55,1.3,7.03ZM579.99,86.58c0-3.39-1-6.16-3.01-8.3s-4.62-3.21-7.84-3.21h-5.81v23.04h5.81c3.27,0,5.89-1.06,7.88-3.19s2.98-4.91,2.98-8.34Z"/> | ||||
|         <path d="M609.16,96.19h-12.73l-2.79,7.97h-6.82l12.68-35.18h6.63l12.66,35.18h-6.96l-2.67-7.97ZM607.24,90.73l-4.43-12.87-4.45,12.87h8.88Z"/> | ||||
|         <path d="M642.87,75.17h-9.89v28.99h-6.56v-28.99h-9.94v-6.19h26.39v6.19Z"/> | ||||
|         <path d="M647.06,104.16v-35.18h6.56v35.18h-6.56Z"/> | ||||
|         <path d="M675.71,68.35c3.31-.05,6.29.72,8.92,2.31s4.67,3.77,6.11,6.55,2.14,5.89,2.11,9.33c.03,3.44-.68,6.55-2.12,9.34s-3.49,4.98-6.13,6.56-5.62,2.36-8.93,2.31c-3.33,0-6.29-.78-8.88-2.33s-4.6-3.71-6.02-6.48-2.13-5.9-2.13-9.4c-.03-2.56.38-4.98,1.22-7.24s2.01-4.21,3.5-5.82,3.31-2.88,5.45-3.8,4.45-1.36,6.91-1.32ZM675.76,98.35c3.09,0,5.56-1.07,7.41-3.2s2.77-5,2.77-8.61-.91-6.52-2.73-8.65-4.3-3.19-7.44-3.19-5.63,1.06-7.46,3.19-2.75,5.01-2.75,8.65.91,6.52,2.74,8.64,4.32,3.18,7.48,3.18Z"/> | ||||
|         <path d="M726.96,68.98v35.18h-6.49l-16.08-24.77v24.77h-6.52v-35.18h6.52l16.08,24.77v-24.77h6.49Z"/> | ||||
|       </g> | ||||
|       <g> | ||||
|         <path d="M94.34,79.34c0,2.75-2.25,5-5,5h-50c-2.75,0-5-2.25-5-5v-20c0-2.75,1.59-6.59,3.54-8.54l22.93-22.93c1.94-1.94,5.13-1.94,7.07,0l22.93,22.93c1.94,1.94,3.54,5.79,3.54,8.54v20Z"/> | ||||
|         <g> | ||||
|           <rect x="34.34" y="94.34" width="60" height="10" rx="2.5" ry="2.5"/> | ||||
|           <rect x="34.34" y="94.34" width="10" height="20" rx="1.56" ry="1.56"/> | ||||
|           <rect x="84.34" y="94.34" width="10" height="20" rx="1.56" ry="1.56"/> | ||||
|         </g> | ||||
|       </g> | ||||
|       <path d="M735.34,3c12.32,0,22.34,10.02,22.34,22.34v88c0,12.32-10.02,22.34-22.34,22.34H25.34c-12.32,0-22.34-10.02-22.34-22.34V25.34C3,13.02,13.02,3,25.34,3h710M735.34,0H25.34C11.37,0,0,11.37,0,25.34v88c0,13.98,11.37,25.34,25.34,25.34h710c13.97,0,25.34-11.37,25.34-25.34V25.34c0-13.98-11.37-25.34-25.34-25.34h0Z"/> | ||||
|       <g> | ||||
|         <path d="M120.98,36.79h2.95v7.26l7.66.02v-7.29h2.97v17.37h-2.97v-7.47l-7.66-.02v7.49h-2.95v-17.37Z"/> | ||||
|         <path d="M146.97,36.47c1.63,0,3.09.39,4.37,1.16s2.28,1.84,2.99,3.2,1.06,2.9,1.06,4.61c.02,1.7-.32,3.24-1.04,4.62s-1.72,2.47-3.02,3.25-2.75,1.16-4.36,1.14c-1.62.02-3.08-.36-4.37-1.14s-2.29-1.86-3-3.24-1.05-2.91-1.03-4.61c0-1.27.2-2.47.61-3.58s.99-2.08,1.72-2.88,1.63-1.42,2.68-1.88,2.18-.67,3.39-.66ZM146.99,51.57c1.6,0,2.89-.56,3.85-1.67s1.45-2.6,1.45-4.45-.48-3.32-1.45-4.43-2.25-1.66-3.85-1.66-2.89.55-3.86,1.66-1.45,2.58-1.45,4.43.48,3.34,1.44,4.46,2.25,1.67,3.88,1.67Z"/> | ||||
|         <path d="M176.51,36.79v17.37h-2.89v-10.78l-4.29,10.78h-2.81l-4.25-10.71v10.71h-2.84v-17.37h2.84l5.66,13.92,5.69-13.92h2.89Z"/> | ||||
|         <path d="M192.41,51.37v2.79h-10.78v-17.37h10.78v2.81h-7.83v4.5h7v2.61h-7v4.66h7.83Z"/> | ||||
|         <path d="M213.93,50.11h-6.61l-1.43,4.04h-3.04l6.27-17.37h3.04l6.29,17.37h-3.11l-1.41-4.04ZM213.07,47.62l-2.43-6.95-2.45,6.95h4.88Z"/> | ||||
|         <path d="M226.96,36.47c1.59,0,2.91.39,3.96,1.16s1.7,1.81,1.94,3.1l-2.78.76c-.16-.74-.52-1.32-1.09-1.72s-1.27-.6-2.11-.6c-.9,0-1.61.21-2.14.64s-.79,1-.79,1.71c0,1.12.7,1.85,2.09,2.18l2.84.71c1.46.38,2.56.98,3.29,1.81s1.09,1.84,1.09,3.05c0,1.55-.56,2.8-1.68,3.76s-2.63,1.44-4.51,1.44c-1.7,0-3.13-.4-4.3-1.2-1.15-.83-1.84-1.92-2.05-3.28l2.78-.72c.1.77.48,1.37,1.14,1.8s1.5.65,2.53.65,1.76-.21,2.32-.62.84-.98.84-1.69c0-1.12-.7-1.85-2.09-2.21l-2.84-.69c-1.46-.33-2.55-.92-3.28-1.77s-1.1-1.88-1.1-3.11c0-1.53.54-2.78,1.63-3.74s2.52-1.44,4.29-1.44Z"/> | ||||
|         <path d="M242.38,36.47c1.59,0,2.91.39,3.96,1.16s1.7,1.81,1.94,3.1l-2.78.76c-.16-.74-.52-1.32-1.09-1.72s-1.27-.6-2.11-.6c-.9,0-1.61.21-2.14.64s-.79,1-.79,1.71c0,1.12.7,1.85,2.09,2.18l2.84.71c1.46.38,2.56.98,3.29,1.81s1.09,1.84,1.09,3.05c0,1.55-.56,2.8-1.68,3.76s-2.63,1.44-4.51,1.44c-1.7,0-3.13-.4-4.3-1.2-1.15-.83-1.84-1.92-2.05-3.28l2.78-.72c.1.77.48,1.37,1.14,1.8s1.5.65,2.53.65,1.76-.21,2.32-.62.84-.98.84-1.69c0-1.12-.7-1.85-2.09-2.21l-2.84-.69c-1.46-.33-2.55-.92-3.28-1.77s-1.1-1.88-1.1-3.11c0-1.53.54-2.78,1.63-3.74s2.52-1.44,4.29-1.44Z"/> | ||||
|         <path d="M252.68,54.16v-17.37h2.95v17.37h-2.95Z"/> | ||||
|         <path d="M265.82,36.47c1.59,0,2.91.39,3.96,1.16s1.7,1.81,1.94,3.1l-2.78.76c-.16-.74-.52-1.32-1.09-1.72s-1.27-.6-2.11-.6c-.9,0-1.61.21-2.14.64s-.79,1-.79,1.71c0,1.12.7,1.85,2.09,2.18l2.84.71c1.46.38,2.56.98,3.29,1.81s1.09,1.84,1.09,3.05c0,1.55-.56,2.8-1.68,3.76s-2.63,1.44-4.51,1.44c-1.7,0-3.13-.4-4.3-1.2-1.15-.83-1.84-1.92-2.05-3.28l2.78-.72c.1.77.48,1.37,1.14,1.8s1.5.65,2.53.65,1.76-.21,2.32-.62.84-.98.84-1.69c0-1.12-.7-1.85-2.09-2.21l-2.84-.69c-1.46-.33-2.55-.92-3.28-1.77s-1.1-1.88-1.1-3.11c0-1.53.54-2.78,1.63-3.74s2.52-1.44,4.29-1.44Z"/> | ||||
|         <path d="M287.47,39.57h-4.97v14.58h-2.95v-14.58h-4.97v-2.79h12.9v2.79Z"/> | ||||
|         <path d="M298.87,50.11h-6.61l-1.43,4.04h-3.04l6.27-17.37h3.04l6.29,17.37h-3.11l-1.41-4.04ZM298.01,47.62l-2.43-6.95-2.45,6.95h4.88Z"/> | ||||
|         <path d="M320.89,36.79v17.37h-2.93l-8.25-12.67v12.67h-2.93v-17.37h2.93l8.25,12.65v-12.65h2.93Z"/> | ||||
|         <path d="M337.31,39.57h-4.97v14.58h-2.95v-14.58h-4.97v-2.79h12.9v2.79Z"/> | ||||
|         <path d="M348.75,54.16v-17.14h2.05v17.14h-2.05Z"/> | ||||
|         <path d="M360.95,36.72c1.55,0,2.82.38,3.81,1.14,1,.74,1.61,1.72,1.82,2.95l-1.95.52c-.16-.87-.56-1.54-1.23-2.02s-1.5-.71-2.5-.71c-1.08,0-1.95.27-2.6.8s-.97,1.24-.97,2.13c0,1.36.84,2.26,2.52,2.71l2.9.73c1.37.34,2.41.9,3.11,1.68s1.05,1.74,1.05,2.9c0,1.46-.53,2.64-1.6,3.54s-2.49,1.36-4.28,1.36c-1.61,0-2.95-.37-4.03-1.12s-1.72-1.76-1.95-3.06l1.98-.55c.13.88.55,1.56,1.25,2.06s1.63.75,2.77.75,2.12-.26,2.79-.77,1.02-1.22,1.02-2.14c0-1.44-.84-2.36-2.52-2.77l-2.88-.71c-1.38-.34-2.42-.9-3.12-1.69s-1.05-1.75-1.05-2.9c0-1.44.52-2.61,1.56-3.51s2.4-1.35,4.09-1.35Z"/> | ||||
|         <path d="M388.35,49.75h-7.54l-1.59,4.4h-2.07l6.25-17.14h2.36l6.31,17.14h-2.15l-1.57-4.4ZM387.73,48.05l-3.09-8.71-3.2,8.71h6.29Z"/> | ||||
|         <path d="M415.46,42.47c0,1.6-.5,2.91-1.5,3.95s-2.32,1.56-3.97,1.56h-4.53v6.18h-2.05v-17.14h6.6c1.67,0,3,.49,3.98,1.47s1.47,2.31,1.47,3.98ZM413.31,42.42c0-1.07-.32-1.92-.95-2.56s-1.51-.96-2.64-.96h-4.26v7.24h4.17c1.15,0,2.06-.34,2.71-1.02s.98-1.58.98-2.7Z"/> | ||||
|         <path d="M428.37,46.9l3.43,7.26h-2.31l-3.18-6.95h-4.76v6.95h-2.05v-17.14h6.54c1.81,0,3.22.45,4.24,1.35s1.53,2.14,1.53,3.72c0,1.22-.3,2.26-.9,3.1s-1.44,1.41-2.53,1.7ZM429.64,42.12c0-1.01-.32-1.81-.95-2.38s-1.52-.86-2.66-.86h-4.5v6.47h4.53c1.15,0,2.03-.28,2.64-.85s.92-1.36.92-2.38Z"/> | ||||
|         <path d="M443.34,36.74c1.18-.02,2.28.2,3.31.65s1.9,1.07,2.62,1.85,1.28,1.73,1.69,2.83.6,2.27.59,3.52c.02,1.67-.33,3.19-1.03,4.54s-1.68,2.42-2.95,3.19-2.68,1.14-4.25,1.12c-1.59,0-3-.38-4.25-1.13s-2.21-1.81-2.89-3.15-1.03-2.87-1.03-4.57c-.02-1.67.32-3.18,1.02-4.53s1.67-2.42,2.93-3.2,2.68-1.15,4.24-1.13ZM443.34,52.45c1.8,0,3.26-.64,4.38-1.91s1.68-2.92,1.68-4.95-.56-3.71-1.68-4.98-2.58-1.9-4.38-1.9-3.28.63-4.4,1.9-1.68,2.93-1.68,4.98.56,3.69,1.68,4.96,2.59,1.9,4.39,1.9Z"/> | ||||
|         <path d="M464.3,37.02v12.42c0,1.49-.47,2.71-1.41,3.64s-2.17,1.39-3.71,1.39c-1.56,0-2.76-.46-3.61-1.37s-1.27-2.13-1.27-3.69v-.55h1.98v.55c0,1.18.29,1.99.86,2.45.59.45,1.26.67,2.02.67.93,0,1.68-.26,2.24-.78.57-.52.85-1.32.85-2.39v-12.35h2.05Z"/> | ||||
|         <path d="M479.86,52.23v1.93h-10.35v-17.14h10.33v1.95h-8.31v5.67h7.53v1.8h-7.53v5.79h8.33Z"/> | ||||
|         <path d="M496.97,42.42c-.36-1.15-1.01-2.06-1.93-2.71s-2.02-.98-3.3-.98c-1.79,0-3.24.63-4.35,1.89s-1.66,2.92-1.66,4.96.55,3.72,1.66,4.96,2.55,1.86,4.33,1.86c1.27,0,2.39-.31,3.35-.94.95-.63,1.61-1.44,1.98-2.45l1.93.83c-.55,1.41-1.48,2.53-2.78,3.36-1.31.81-2.81,1.22-4.51,1.22-2.4,0-4.35-.81-5.84-2.43s-2.24-3.75-2.24-6.4c0-1.74.34-3.28,1.02-4.62s1.64-2.39,2.88-3.13,2.65-1.11,4.25-1.11c1.8,0,3.33.46,4.59,1.38,1.26.91,2.12,2.1,2.57,3.59l-1.93.71Z"/> | ||||
|         <path d="M512.92,38.95h-5.12v15.21h-2.05v-15.21h-5.16v-1.93h12.33v1.93Z"/> | ||||
|         <path d="M536.4,49.66c0,1.39-.49,2.48-1.46,3.29s-2.27,1.21-3.9,1.21h-6.72v-17.12h6.58c1.65,0,2.95.41,3.89,1.24s1.41,1.93,1.41,3.32c0,.92-.21,1.72-.64,2.4s-1.02,1.2-1.79,1.57c.81.37,1.45.91,1.92,1.62s.7,1.53.7,2.47ZM526.3,38.81v5.97h4.54c1.03,0,1.85-.27,2.46-.81s.92-1.24.92-2.09c0-.91-.31-1.65-.93-2.22s-1.45-.85-2.5-.85h-4.5ZM534.45,49.39c0-.91-.32-1.64-.95-2.17s-1.44-.8-2.46-.8h-4.74v5.95h4.74c1.04,0,1.86-.27,2.48-.81s.92-1.26.92-2.16Z"/> | ||||
|         <path d="M541.07,37.02l4.54,8.1,4.54-8.1h2.24l-5.76,10.05v7.09h-2.05v-7.09l-5.83-10.05h2.31Z"/> | ||||
|         <path d="M574.27,38.95h-5.12v15.21h-2.05v-15.21h-5.16v-1.93h12.33v1.93Z"/> | ||||
|         <path d="M577.95,37.02h2.05v7.55l8.74.02v-7.58h2.05v17.14h-2.05v-7.76l-8.74-.02v7.79h-2.05v-17.14Z"/> | ||||
|         <path d="M606.55,52.23v1.93h-10.35v-17.14h10.33v1.95h-8.31v5.67h7.53v1.8h-7.53v5.79h8.33Z"/> | ||||
|       </g> | ||||
|     </g> | ||||
|   </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 12 KiB | 
| @@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta" | ||||
|  | ||||
| [project] | ||||
| name         = "home-assistant-frontend" | ||||
| version      = "20240719.0" | ||||
| version      = "20240809.0" | ||||
| license      = {text = "Apache-2.0"} | ||||
| description  = "The Home Assistant frontend" | ||||
| readme       = "README.md" | ||||
|   | ||||
| @@ -37,8 +37,7 @@ | ||||
|     { | ||||
|       "description": "Group tsparticles engine and presets", | ||||
|       "groupName": "tsparticles", | ||||
|       "matchPackageNames": ["tsparticles-engine"], | ||||
|       "matchPackagePrefixes": ["tsparticles-preset-"] | ||||
|       "matchPackageNames": ["tsparticles-engine", "tsparticles-preset-{/,}**"] | ||||
|     }, | ||||
|     { | ||||
|       "description": "Group date-fns with dependent timezone package", | ||||
| @@ -48,8 +47,8 @@ | ||||
|     { | ||||
|       "description": "Group and temporarily disable WDS packages", | ||||
|       "groupName": "Web Dev Server", | ||||
|       "matchPackagePrefixes": ["@web/dev-server"], | ||||
|       "enabled": false | ||||
|       "enabled": false, | ||||
|       "matchPackageNames": ["@web/dev-server{/,}**"] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
|   | ||||
| @@ -40,7 +40,6 @@ import { | ||||
|   mdiImageFilterFrames, | ||||
|   mdiLightbulb, | ||||
|   mdiLightningBolt, | ||||
|   mdiMailbox, | ||||
|   mdiMapMarkerRadius, | ||||
|   mdiMeterGas, | ||||
|   mdiMicrophoneMessage, | ||||
| @@ -119,7 +118,6 @@ export const FIXED_DOMAIN_ICONS = { | ||||
|   input_text: mdiFormTextbox, | ||||
|   lawn_mower: mdiRobotMower, | ||||
|   light: mdiLightbulb, | ||||
|   mailbox: mdiMailbox, | ||||
|   notify: mdiCommentAlert, | ||||
|   number: mdiRayVertex, | ||||
|   persistent_notification: mdiBell, | ||||
|   | ||||
| @@ -26,7 +26,7 @@ export const FIXED_DOMAIN_STATES = { | ||||
|   humidifier: ["on", "off"], | ||||
|   input_boolean: ["on", "off"], | ||||
|   input_button: [], | ||||
|   lawn_mower: ["error", "paused", "mowing", "docked"], | ||||
|   lawn_mower: ["error", "paused", "mowing", "returning", "docked"], | ||||
|   light: ["on", "off"], | ||||
|   lock: [ | ||||
|     "jammed", | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { HassServiceTarget } from "home-assistant-js-websocket"; | ||||
| import { showConfirmationDialog } from "../../dialogs/generic/show-dialog-box"; | ||||
| import "./ha-progress-button"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| @@ -17,7 +18,9 @@ class HaCallServiceButton extends LitElement { | ||||
|  | ||||
|   @property() public service!: string; | ||||
|  | ||||
|   @property({ type: Object }) public serviceData = {}; | ||||
|   @property({ type: Object }) public target!: HassServiceTarget; | ||||
|  | ||||
|   @property({ type: Object }) public data = {}; | ||||
|  | ||||
|   @property() public confirmation?; | ||||
|  | ||||
| @@ -39,7 +42,8 @@ class HaCallServiceButton extends LitElement { | ||||
|     const eventData = { | ||||
|       domain: this.domain, | ||||
|       service: this.service, | ||||
|       serviceData: this.serviceData, | ||||
|       data: this.data, | ||||
|       target: this.target, | ||||
|       success: false, | ||||
|     }; | ||||
|  | ||||
| @@ -47,7 +51,12 @@ class HaCallServiceButton extends LitElement { | ||||
|       this.shadowRoot!.querySelector("ha-progress-button")!; | ||||
|  | ||||
|     try { | ||||
|       await this.hass.callService(this.domain, this.service, this.serviceData); | ||||
|       await this.hass.callService( | ||||
|         this.domain, | ||||
|         this.service, | ||||
|         this.data, | ||||
|         this.target | ||||
|       ); | ||||
|       this.progress = false; | ||||
|       progressElement.actionSuccess(); | ||||
|       eventData.success = true; | ||||
| @@ -85,7 +94,8 @@ declare global { | ||||
|     "hass-service-called": { | ||||
|       domain: string; | ||||
|       service: string; | ||||
|       serviceData: object; | ||||
|       target: HassServiceTarget; | ||||
|       data: object; | ||||
|       success: boolean; | ||||
|     }; | ||||
|   } | ||||
|   | ||||
| @@ -85,9 +85,9 @@ export interface DataTableColumnData<T = any> extends DataTableSortColumnData { | ||||
|     | "flex"; | ||||
|   template?: (row: T) => TemplateResult | string | typeof nothing; | ||||
|   extraTemplate?: (row: T) => TemplateResult | string | typeof nothing; | ||||
|   width?: string; | ||||
|   minWidth?: string; | ||||
|   maxWidth?: string; | ||||
|   grows?: boolean; | ||||
|   flex?: number; | ||||
|   forceLTR?: boolean; | ||||
|   hidden?: boolean; | ||||
| } | ||||
| @@ -216,6 +216,18 @@ export class HaDataTable extends LitElement { | ||||
|     this.updateComplete.then(() => this._calcTableHeight()); | ||||
|   } | ||||
|  | ||||
|   protected updated() { | ||||
|     const header = this.renderRoot.querySelector(".mdc-data-table__header-row"); | ||||
|     if (!header) { | ||||
|       return; | ||||
|     } | ||||
|     if (header.scrollWidth > header.clientWidth) { | ||||
|       this.style.setProperty("--table-row-width", `${header.scrollWidth}px`); | ||||
|     } else { | ||||
|       this.style.removeProperty("--table-row-width"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public willUpdate(properties: PropertyValues) { | ||||
|     super.willUpdate(properties); | ||||
|  | ||||
| @@ -355,7 +367,12 @@ export class HaDataTable extends LitElement { | ||||
|               : `calc(100% - ${this._headerHeight}px)`, | ||||
|           })} | ||||
|         > | ||||
|           <div class="mdc-data-table__header-row" role="row" aria-rowindex="1"> | ||||
|           <div | ||||
|             class="mdc-data-table__header-row" | ||||
|             role="row" | ||||
|             aria-rowindex="1" | ||||
|             @scroll=${this._scrollContent} | ||||
|           > | ||||
|             <slot name="header-row"> | ||||
|               ${this.selectable | ||||
|                 ? html` | ||||
| @@ -398,18 +415,16 @@ export class HaDataTable extends LitElement { | ||||
|                     column.type === "overflow", | ||||
|                   sortable: Boolean(column.sortable), | ||||
|                   "not-sorted": Boolean(column.sortable && !sorted), | ||||
|                   grows: Boolean(column.grows), | ||||
|                 }; | ||||
|                 return html` | ||||
|                   <div | ||||
|                     aria-label=${ifDefined(column.label)} | ||||
|                     class="mdc-data-table__header-cell ${classMap(classes)}" | ||||
|                     style=${column.width | ||||
|                       ? styleMap({ | ||||
|                           [column.grows ? "minWidth" : "width"]: column.width, | ||||
|                           maxWidth: column.maxWidth || "", | ||||
|                         }) | ||||
|                       : ""} | ||||
|                     style=${styleMap({ | ||||
|                       minWidth: column.minWidth, | ||||
|                       maxWidth: column.maxWidth, | ||||
|                       flex: column.flex || 1, | ||||
|                     })} | ||||
|                     role="columnheader" | ||||
|                     aria-sort=${ifDefined( | ||||
|                       sorted | ||||
| @@ -538,15 +553,13 @@ export class HaDataTable extends LitElement { | ||||
|                 "mdc-data-table__cell--overflow-menu": | ||||
|                   column.type === "overflow-menu", | ||||
|                 "mdc-data-table__cell--overflow": column.type === "overflow", | ||||
|                 grows: Boolean(column.grows), | ||||
|                 forceLTR: Boolean(column.forceLTR), | ||||
|               })}" | ||||
|               style=${column.width | ||||
|                 ? styleMap({ | ||||
|                     [column.grows ? "minWidth" : "width"]: column.width, | ||||
|                     maxWidth: column.maxWidth ? column.maxWidth : "", | ||||
|                   }) | ||||
|                 : ""} | ||||
|               style=${styleMap({ | ||||
|                 minWidth: column.minWidth, | ||||
|                 maxWidth: column.maxWidth, | ||||
|                 flex: column.flex || 1, | ||||
|               })} | ||||
|             > | ||||
|               ${column.template | ||||
|                 ? column.template(row) | ||||
| @@ -597,7 +610,7 @@ export class HaDataTable extends LitElement { | ||||
|       filteredData = await this._memFilterData( | ||||
|         this.data, | ||||
|         this._sortColumns, | ||||
|         this._filter | ||||
|         this._filter.trim() | ||||
|       ); | ||||
|     } | ||||
|  | ||||
| @@ -815,6 +828,17 @@ export class HaDataTable extends LitElement { | ||||
|   @eventOptions({ passive: true }) | ||||
|   private _saveScrollPos(e: Event) { | ||||
|     this._savedScrollPos = (e.target as HTMLDivElement).scrollTop; | ||||
|  | ||||
|     this.renderRoot.querySelector(".mdc-data-table__header-row")!.scrollLeft = ( | ||||
|       e.target as HTMLDivElement | ||||
|     ).scrollLeft; | ||||
|   } | ||||
|  | ||||
|   @eventOptions({ passive: true }) | ||||
|   private _scrollContent(e: Event) { | ||||
|     this.renderRoot.querySelector("lit-virtualizer")!.scrollLeft = ( | ||||
|       e.target as HTMLDivElement | ||||
|     ).scrollLeft; | ||||
|   } | ||||
|  | ||||
|   private _collapseGroup = (ev: Event) => { | ||||
| @@ -889,8 +913,8 @@ export class HaDataTable extends LitElement { | ||||
|  | ||||
|         .mdc-data-table__row { | ||||
|           display: flex; | ||||
|           width: 100%; | ||||
|           height: var(--data-table-row-height, 52px); | ||||
|           width: var(--table-row-width, 100%); | ||||
|         } | ||||
|  | ||||
|         .mdc-data-table__row ~ .mdc-data-table__row { | ||||
| @@ -914,18 +938,26 @@ export class HaDataTable extends LitElement { | ||||
|         .mdc-data-table__header-row { | ||||
|           height: 56px; | ||||
|           display: flex; | ||||
|           width: 100%; | ||||
|           border-bottom: 1px solid var(--divider-color); | ||||
|           overflow: auto; | ||||
|         } | ||||
|  | ||||
|         /* Hide scrollbar for Chrome, Safari and Opera */ | ||||
|         .mdc-data-table__header-row::-webkit-scrollbar { | ||||
|           display: none; | ||||
|         } | ||||
|  | ||||
|         /* Hide scrollbar for IE, Edge and Firefox */ | ||||
|         .mdc-data-table__header-row { | ||||
|           -ms-overflow-style: none; /* IE and Edge */ | ||||
|           scrollbar-width: none; /* Firefox */ | ||||
|         } | ||||
|  | ||||
|         .mdc-data-table__cell, | ||||
|         .mdc-data-table__header-cell { | ||||
|           padding-right: 16px; | ||||
|           padding-left: 16px; | ||||
|           min-width: 150px; | ||||
|           align-self: center; | ||||
|           overflow: hidden; | ||||
|           text-overflow: ellipsis; | ||||
| @@ -973,6 +1005,8 @@ export class HaDataTable extends LitElement { | ||||
|           letter-spacing: 0.0178571429em; | ||||
|           text-decoration: inherit; | ||||
|           text-transform: inherit; | ||||
|           flex-grow: 0; | ||||
|           flex-shrink: 0; | ||||
|         } | ||||
|  | ||||
|         .mdc-data-table__cell a { | ||||
| @@ -991,7 +1025,8 @@ export class HaDataTable extends LitElement { | ||||
|  | ||||
|         .mdc-data-table__header-cell--icon, | ||||
|         .mdc-data-table__cell--icon { | ||||
|           width: 54px; | ||||
|           min-width: 64px; | ||||
|           flex: 0 0 64px !important; | ||||
|         } | ||||
|  | ||||
|         .mdc-data-table__cell--icon img { | ||||
| @@ -1031,11 +1066,14 @@ export class HaDataTable extends LitElement { | ||||
|         .mdc-data-table__header-cell--overflow-menu, | ||||
|         .mdc-data-table__header-cell--icon-button, | ||||
|         .mdc-data-table__cell--icon-button { | ||||
|           min-width: 64px; | ||||
|           flex: 0 0 64px !important; | ||||
|           padding: 8px; | ||||
|         } | ||||
|  | ||||
|         .mdc-data-table__header-cell--icon-button, | ||||
|         .mdc-data-table__cell--icon-button { | ||||
|           min-width: 56px; | ||||
|           width: 56px; | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -81,6 +81,9 @@ class HaEntityStatePicker extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public required = false; | ||||
|  | ||||
|   @property({ type: Boolean, attribute: "allow-name" }) public allowName = | ||||
|     false; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property() public value?: string[] | string; | ||||
| @@ -95,13 +98,24 @@ class HaEntityStatePicker extends LitElement { | ||||
|     return !(!changedProps.has("_opened") && this._opened); | ||||
|   } | ||||
|  | ||||
|   private options = memoizeOne((entityId?: string, stateObj?: HassEntity) => { | ||||
|   private options = memoizeOne( | ||||
|     (entityId?: string, stateObj?: HassEntity, allowName?: boolean) => { | ||||
|       const domain = entityId ? computeDomain(entityId) : undefined; | ||||
|       return [ | ||||
|         { | ||||
|           label: this.hass.localize("ui.components.state-content-picker.state"), | ||||
|           value: "state", | ||||
|         }, | ||||
|         ...(allowName | ||||
|           ? [ | ||||
|               { | ||||
|                 label: this.hass.localize( | ||||
|                   "ui.components.state-content-picker.name" | ||||
|                 ), | ||||
|                 value: "name", | ||||
|               }, | ||||
|             ] | ||||
|           : []), | ||||
|         { | ||||
|           label: this.hass.localize( | ||||
|             "ui.components.state-content-picker.last_changed" | ||||
| @@ -131,7 +145,8 @@ class HaEntityStatePicker extends LitElement { | ||||
|             label: this.hass.formatEntityAttributeName(stateObj!, attribute), | ||||
|           })), | ||||
|       ]; | ||||
|   }); | ||||
|     } | ||||
|   ); | ||||
|  | ||||
|   private _filter = ""; | ||||
|  | ||||
| @@ -146,7 +161,7 @@ class HaEntityStatePicker extends LitElement { | ||||
|       ? this.hass.states[this.entityId] | ||||
|       : undefined; | ||||
|  | ||||
|     const options = this.options(this.entityId, stateObj); | ||||
|     const options = this.options(this.entityId, stateObj, this.allowName); | ||||
|     const optionItems = options.filter( | ||||
|       (option) => !this._value.includes(option.value) | ||||
|     ); | ||||
| @@ -214,6 +229,7 @@ class HaEntityStatePicker extends LitElement { | ||||
|  | ||||
|   private _openedChanged(ev: ValueChangedEvent<boolean>) { | ||||
|     this._opened = ev.detail.value; | ||||
|     this._comboBox.filteredItems = this._comboBox.items; | ||||
|   } | ||||
|  | ||||
|   private _filterChanged(ev?: CustomEvent): void { | ||||
|   | ||||
| @@ -279,6 +279,8 @@ export class HaAreaPicker extends LitElement { | ||||
|             icon: null, | ||||
|             aliases: [], | ||||
|             labels: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ]; | ||||
|       } | ||||
| @@ -295,6 +297,8 @@ export class HaAreaPicker extends LitElement { | ||||
|               icon: "mdi:plus", | ||||
|               aliases: [], | ||||
|               labels: [], | ||||
|               created_at: 0, | ||||
|               modified_at: 0, | ||||
|             }, | ||||
|           ]; | ||||
|     } | ||||
| @@ -377,6 +381,8 @@ export class HaAreaPicker extends LitElement { | ||||
|             picture: null, | ||||
|             labels: [], | ||||
|             aliases: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ] as AreaRegistryEntry[]; | ||||
|       } else { | ||||
| @@ -393,6 +399,8 @@ export class HaAreaPicker extends LitElement { | ||||
|             picture: null, | ||||
|             labels: [], | ||||
|             aliases: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ] as AreaRegistryEntry[]; | ||||
|       } | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement, TemplateResult, nothing } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { mdiClose } from "@mdi/js"; | ||||
| import { ifDefined } from "lit/directives/if-defined"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { stopPropagation } from "../common/dom/stop_propagation"; | ||||
| import "./ha-select"; | ||||
| import "./ha-icon-button"; | ||||
| import { HaTextField } from "./ha-textfield"; | ||||
| import "./ha-input-helper-text"; | ||||
|  | ||||
| @@ -124,11 +126,14 @@ export class HaBaseTimeInput extends LitElement { | ||||
|    */ | ||||
|   @property() amPm: "AM" | "PM" = "AM"; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public clearable?: boolean; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${this.label | ||||
|         ? html`<label>${this.label}${this.required ? " *" : ""}</label>` | ||||
|         : ""} | ||||
|       <div class="time-input-wrap-wrap"> | ||||
|         <div class="time-input-wrap"> | ||||
|           ${this.enableDay | ||||
|             ? html` | ||||
| @@ -234,6 +239,15 @@ export class HaBaseTimeInput extends LitElement { | ||||
|               > | ||||
|               </ha-textfield>` | ||||
|             : ""} | ||||
|           ${this.clearable && !this.required && !this.disabled | ||||
|             ? html`<ha-icon-button | ||||
|                 label="clear" | ||||
|                 @click=${this._clearValue} | ||||
|                 .path=${mdiClose} | ||||
|               ></ha-icon-button>` | ||||
|             : nothing} | ||||
|         </div> | ||||
|  | ||||
|         ${this.format === 24 | ||||
|           ? "" | ||||
|           : html`<ha-select | ||||
| @@ -249,13 +263,17 @@ export class HaBaseTimeInput extends LitElement { | ||||
|               <mwc-list-item value="AM">AM</mwc-list-item> | ||||
|               <mwc-list-item value="PM">PM</mwc-list-item> | ||||
|             </ha-select>`} | ||||
|       </div> | ||||
|         ${this.helper | ||||
|           ? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>` | ||||
|           : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _clearValue(): void { | ||||
|     fireEvent(this, "value-changed"); | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: InputEvent) { | ||||
|     const textField = ev.currentTarget as HaTextField; | ||||
|     this[textField.name] = | ||||
| @@ -302,18 +320,25 @@ export class HaBaseTimeInput extends LitElement { | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host([clearable]) { | ||||
|       position: relative; | ||||
|     } | ||||
|     :host { | ||||
|       display: block; | ||||
|     } | ||||
|     .time-input-wrap-wrap { | ||||
|       display: flex; | ||||
|     } | ||||
|     .time-input-wrap { | ||||
|       display: flex; | ||||
|       border-radius: var(--mdc-shape-small, 4px) var(--mdc-shape-small, 4px) 0 0; | ||||
|       overflow: hidden; | ||||
|       position: relative; | ||||
|       direction: ltr; | ||||
|       padding-right: 3px; | ||||
|     } | ||||
|     ha-textfield { | ||||
|       width: 40px; | ||||
|       width: 55px; | ||||
|       text-align: center; | ||||
|       --mdc-shape-small: 0; | ||||
|       --text-field-appearance: none; | ||||
| @@ -335,6 +360,21 @@ export class HaBaseTimeInput extends LitElement { | ||||
|       --mdc-shape-small: 0; | ||||
|       width: 85px; | ||||
|     } | ||||
|     :host([clearable]) .mdc-select__anchor { | ||||
|         padding-inline-end: var(--select-selected-text-padding-end, 12px); | ||||
|     } | ||||
|     ha-icon-button { | ||||
|       position: relative | ||||
|       --mdc-icon-button-size: 36px; | ||||
|       --mdc-icon-size: 20px; | ||||
|       color: var(--secondary-text-color); | ||||
|       direction: var(--direction); | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       background-color:var(--mdc-text-field-fill-color, whitesmoke); | ||||
|       border-bottom-style: solid; | ||||
|       border-bottom-width: 1px; | ||||
|     } | ||||
|     label { | ||||
|       -moz-osx-font-smoothing: grayscale; | ||||
|       -webkit-font-smoothing: antialiased; | ||||
|   | ||||
| @@ -196,8 +196,8 @@ export class HaControlNumberButton extends LitElement { | ||||
|         --control-number-buttons-background-opacity: 0.2; | ||||
|         --control-number-buttons-border-radius: 10px; | ||||
|         --mdc-icon-size: 16px; | ||||
|         height: 40px; | ||||
|         width: 200px; | ||||
|         height: var(--feature-height); | ||||
|         width: 100%; | ||||
|         color: var(--primary-text-color); | ||||
|         -webkit-tap-highlight-color: transparent; | ||||
|         font-style: normal; | ||||
|   | ||||
| @@ -295,6 +295,8 @@ export class HaFloorPicker extends SubscribeMixin(LitElement) { | ||||
|             icon: null, | ||||
|             level: null, | ||||
|             aliases: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ]; | ||||
|       } | ||||
| @@ -309,6 +311,8 @@ export class HaFloorPicker extends SubscribeMixin(LitElement) { | ||||
|               icon: "mdi:plus", | ||||
|               level: null, | ||||
|               aliases: [], | ||||
|               created_at: 0, | ||||
|               modified_at: 0, | ||||
|             }, | ||||
|           ]; | ||||
|     } | ||||
| @@ -391,6 +395,8 @@ export class HaFloorPicker extends SubscribeMixin(LitElement) { | ||||
|             icon: null, | ||||
|             level: null, | ||||
|             aliases: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ] as FloorRegistryEntry[]; | ||||
|       } else { | ||||
| @@ -405,6 +411,8 @@ export class HaFloorPicker extends SubscribeMixin(LitElement) { | ||||
|             icon: "mdi:plus", | ||||
|             level: null, | ||||
|             aliases: [], | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ] as FloorRegistryEntry[]; | ||||
|       } | ||||
|   | ||||
| @@ -303,6 +303,8 @@ export class HaLabelPicker extends SubscribeMixin(LitElement) { | ||||
|             icon: null, | ||||
|             color: null, | ||||
|             description: null, | ||||
|             created_at: 0, | ||||
|             modified_at: 0, | ||||
|           }, | ||||
|         ]; | ||||
|       } | ||||
| @@ -317,6 +319,8 @@ export class HaLabelPicker extends SubscribeMixin(LitElement) { | ||||
|               icon: "mdi:plus", | ||||
|               color: null, | ||||
|               description: null, | ||||
|               created_at: 0, | ||||
|               modified_at: 0, | ||||
|             }, | ||||
|           ]; | ||||
|     } | ||||
|   | ||||
| @@ -28,6 +28,11 @@ const LAWN_MOWER_ACTIONS: Partial< | ||||
|     service: "start_mowing", | ||||
|     feature: LawnMowerEntityFeature.START_MOWING, | ||||
|   }, | ||||
|   returning: { | ||||
|     action: "pause", | ||||
|     service: "pause", | ||||
|     feature: LawnMowerEntityFeature.PAUSE, | ||||
|   }, | ||||
|   paused: { | ||||
|     action: "resume_mowing", | ||||
|     service: "start_mowing", | ||||
|   | ||||
| @@ -6,12 +6,12 @@ import { | ||||
|   mdiSofa, | ||||
| } from "@mdi/js"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   nothing, | ||||
|   PropertyValues, | ||||
|   css, | ||||
|   html, | ||||
|   nothing, | ||||
| } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { styleMap } from "lit/directives/style-map"; | ||||
| @@ -20,7 +20,7 @@ import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { caseInsensitiveStringCompare } from "../common/string/compare"; | ||||
| import { Blueprints, fetchBlueprints } from "../data/blueprint"; | ||||
| import { ConfigEntry, getConfigEntries } from "../data/config_entries"; | ||||
| import { findRelated, ItemType, RelatedResult } from "../data/search"; | ||||
| import { ItemType, RelatedResult, findRelated } from "../data/search"; | ||||
| import { haStyle } from "../resources/styles"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { brandsUrl } from "../util/brands-url"; | ||||
| @@ -109,6 +109,26 @@ export class HaRelatedItems extends LitElement { | ||||
|         ) | ||||
|       ); | ||||
|  | ||||
|   private _getConfigEntries = memoizeOne( | ||||
|     ( | ||||
|       relatedConfigEntries: string[] | undefined, | ||||
|       entries: ConfigEntry[] | undefined | ||||
|     ) => { | ||||
|       const configEntries = | ||||
|         relatedConfigEntries && entries | ||||
|           ? relatedConfigEntries.map((entryId) => | ||||
|               entries!.find((configEntry) => configEntry.entry_id === entryId) | ||||
|             ) | ||||
|           : undefined; | ||||
|  | ||||
|       const configEntryDomains = new Set( | ||||
|         configEntries?.map((entry) => entry?.domain) | ||||
|       ); | ||||
|  | ||||
|       return { configEntries, configEntryDomains }; | ||||
|     } | ||||
|   ); | ||||
|  | ||||
|   protected render() { | ||||
|     if (!this._related) { | ||||
|       return nothing; | ||||
| @@ -128,22 +148,25 @@ export class HaRelatedItems extends LitElement { | ||||
|         </mwc-list> | ||||
|       `; | ||||
|     } | ||||
|  | ||||
|     const { configEntries, configEntryDomains } = this._getConfigEntries( | ||||
|       this._related.config_entry, | ||||
|       this._entries | ||||
|     ); | ||||
|  | ||||
|     return html` | ||||
|       ${this._related.config_entry && this._entries | ||||
|       ${configEntries || this._related.integration | ||||
|         ? html`<h3> | ||||
|               ${this.hass.localize("ui.components.related-items.integration")} | ||||
|             </h3> | ||||
|             <mwc-list | ||||
|               >${this._related.config_entry.map((relatedConfigEntryId) => { | ||||
|                 const entry: ConfigEntry | undefined = this._entries!.find( | ||||
|                   (configEntry) => configEntry.entry_id === relatedConfigEntryId | ||||
|                 ); | ||||
|               >${configEntries?.map((entry) => { | ||||
|                 if (!entry) { | ||||
|                   return nothing; | ||||
|                 } | ||||
|                 return html` | ||||
|                   <a | ||||
|                     href=${`/config/integrations/integration/${entry.domain}#config_entry=${relatedConfigEntryId}`} | ||||
|                     href=${`/config/integrations/integration/${entry.domain}#config_entry=${entry.entry_id}`} | ||||
|                     @click=${this._navigateAwayClose} | ||||
|                   > | ||||
|                     <ha-list-item hasMeta graphic="icon"> | ||||
| @@ -164,8 +187,34 @@ export class HaRelatedItems extends LitElement { | ||||
|                     </ha-list-item> | ||||
|                   </a> | ||||
|                 `; | ||||
|               })}</mwc-list | ||||
|             >` | ||||
|               })} | ||||
|               ${this._related.integration | ||||
|                 ?.filter((integration) => !configEntryDomains.has(integration)) | ||||
|                 .map( | ||||
|                   (integration) => | ||||
|                     html`<a | ||||
|                       href=${`/config/integrations/integration/${integration}`} | ||||
|                       @click=${this._navigateAwayClose} | ||||
|                     > | ||||
|                       <ha-list-item hasMeta graphic="icon"> | ||||
|                         <img | ||||
|                           .src=${brandsUrl({ | ||||
|                             domain: integration, | ||||
|                             type: "icon", | ||||
|                             useFallback: true, | ||||
|                             darkOptimized: this.hass.themes?.darkMode, | ||||
|                           })} | ||||
|                           crossorigin="anonymous" | ||||
|                           referrerpolicy="no-referrer" | ||||
|                           alt=${integration} | ||||
|                           slot="graphic" | ||||
|                         /> | ||||
|                         ${this.hass.localize(`component.${integration}.title`)} | ||||
|                         <ha-icon-next slot="meta"></ha-icon-next> | ||||
|                       </ha-list-item> | ||||
|                     </a>` | ||||
|                 )} | ||||
|             </mwc-list>` | ||||
|         : nothing} | ||||
|       ${this._related.device | ||||
|         ? html`<h3> | ||||
|   | ||||
| @@ -1,17 +1,23 @@ | ||||
| import { css, CSSResultGroup, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { BooleanSelector } from "../../data/selector"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import "../ha-checkbox"; | ||||
| import "../ha-formfield"; | ||||
| import "../ha-switch"; | ||||
| import "../ha-input-helper-text"; | ||||
| import "../ha-switch"; | ||||
|  | ||||
| @customElement("ha-selector-boolean") | ||||
| export class HaBooleanSelector extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public selector!: BooleanSelector; | ||||
|  | ||||
|   @property({ type: Boolean }) public value = false; | ||||
|  | ||||
|   @property() public placeholder?: any; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property() public helper?: string; | ||||
| @@ -19,13 +25,24 @@ export class HaBooleanSelector extends LitElement { | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     const checkbox = this.selector.boolean?.mode === "checkbox"; | ||||
|     return html` | ||||
|       <ha-formfield alignEnd spaceBetween .label=${this.label}> | ||||
|       <ha-formfield .alignEnd=${!checkbox} spaceBetween .label=${this.label}> | ||||
|         ${checkbox | ||||
|           ? html` | ||||
|               <ha-checkbox | ||||
|                 .checked=${this.value ?? this.placeholder === true} | ||||
|                 @change=${this._handleChange} | ||||
|                 .disabled=${this.disabled} | ||||
|               ></ha-checkbox> | ||||
|             ` | ||||
|           : html` | ||||
|               <ha-switch | ||||
|           .checked=${this.value} | ||||
|                 .checked=${this.value ?? this.placeholder === true} | ||||
|                 @change=${this._handleChange} | ||||
|                 .disabled=${this.disabled} | ||||
|               ></ha-switch> | ||||
|             `} | ||||
|       </ha-formfield> | ||||
|       ${this.helper | ||||
|         ? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>` | ||||
|   | ||||
| @@ -30,6 +30,7 @@ export class HaTimeDuration extends LitElement { | ||||
|         .disabled=${this.disabled} | ||||
|         .required=${this.required} | ||||
|         ?enableDay=${this.selector.duration?.enable_day} | ||||
|         ?enableMillisecond=${this.selector.duration?.enable_millisecond} | ||||
|       ></ha-duration-input> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -45,7 +45,14 @@ export class HaImageSelector extends LitElement { | ||||
|     return html` | ||||
|       <div> | ||||
|         <label> | ||||
|           ${this.hass.localize("ui.components.selectors.image.select_image")} | ||||
|           ${this.hass.localize( | ||||
|             "ui.components.selectors.image.select_image_with_label", | ||||
|             { | ||||
|               label: | ||||
|                 this.label || | ||||
|                 this.hass.localize("ui.components.selectors.image.image"), | ||||
|             } | ||||
|           )} | ||||
|           <ha-formfield | ||||
|             .label=${this.hass.localize("ui.components.selectors.image.upload")} | ||||
|           > | ||||
|   | ||||
| @@ -57,6 +57,10 @@ const SELECTOR_SCHEMAS = { | ||||
|       name: "enable_day", | ||||
|       selector: { boolean: {} }, | ||||
|     }, | ||||
|     { | ||||
|       name: "enable_millisecond", | ||||
|       selector: { boolean: {} }, | ||||
|     }, | ||||
|   ] as const, | ||||
|   entity: [ | ||||
|     { | ||||
|   | ||||
| @@ -81,7 +81,8 @@ export class HaTargetSelector extends LitElement { | ||||
|       return nothing; | ||||
|     } | ||||
|  | ||||
|     return html`<ha-target-picker | ||||
|     return html` ${this.label ? html`<label>${this.label}</label>` : nothing} | ||||
|       <ha-target-picker | ||||
|         .hass=${this.hass} | ||||
|         .value=${this.value} | ||||
|         .helper=${this.helper} | ||||
|   | ||||
| @@ -1,9 +1,18 @@ | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { css, html, nothing, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import { documentationUrl } from "../../util/documentation-url"; | ||||
| import "../ha-code-editor"; | ||||
| import "../ha-input-helper-text"; | ||||
| import "../ha-alert"; | ||||
|  | ||||
| const WARNING_STRINGS = [ | ||||
|   "template:", | ||||
|   "sensor:", | ||||
|   "state:", | ||||
|   "platform: template", | ||||
| ]; | ||||
|  | ||||
| @customElement("ha-selector-template") | ||||
| export class HaTemplateSelector extends LitElement { | ||||
| @@ -19,9 +28,33 @@ export class HaTemplateSelector extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public required = true; | ||||
|  | ||||
|   @state() private warn: string | undefined = undefined; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       ${this.label ? html`<p>${this.label}${this.required ? "*" : ""}</p>` : ""} | ||||
|       ${this.warn | ||||
|         ? html`<ha-alert alert-type="warning" | ||||
|             >${this.hass.localize( | ||||
|               "ui.components.selectors.template.yaml_warning", | ||||
|               { string: this.warn } | ||||
|             )} | ||||
|             <br /> | ||||
|             <a | ||||
|               target="_blank" | ||||
|               rel="noopener noreferrer" | ||||
|               href=${documentationUrl( | ||||
|                 this.hass, | ||||
|                 "/docs/configuration/templating/" | ||||
|               )} | ||||
|               >${this.hass.localize( | ||||
|                 "ui.components.selectors.template.learn_more" | ||||
|               )}</a | ||||
|             ></ha-alert | ||||
|           >` | ||||
|         : nothing} | ||||
|       ${this.label | ||||
|         ? html`<p>${this.label}${this.required ? "*" : ""}</p>` | ||||
|         : nothing} | ||||
|       <ha-code-editor | ||||
|         mode="jinja2" | ||||
|         .hass=${this.hass} | ||||
| @@ -36,7 +69,7 @@ export class HaTemplateSelector extends LitElement { | ||||
|       ></ha-code-editor> | ||||
|       ${this.helper | ||||
|         ? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>` | ||||
|         : ""} | ||||
|         : nothing} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
| @@ -45,6 +78,7 @@ export class HaTemplateSelector extends LitElement { | ||||
|     if (this.value === value) { | ||||
|       return; | ||||
|     } | ||||
|     this.warn = WARNING_STRINGS.find((str) => value.includes(str)); | ||||
|     fireEvent(this, "value-changed", { value }); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -27,6 +27,7 @@ export class HaTimeSelector extends LitElement { | ||||
|         .locale=${this.hass.locale} | ||||
|         .disabled=${this.disabled} | ||||
|         .required=${this.required} | ||||
|         clearable | ||||
|         .helper=${this.helper} | ||||
|         .label=${this.label} | ||||
|         enable-second | ||||
|   | ||||
| @@ -36,6 +36,7 @@ export class HaSelectorUiStateContent extends SubscribeMixin(LitElement) { | ||||
|         .helper=${this.helper} | ||||
|         .disabled=${this.disabled} | ||||
|         .required=${this.required} | ||||
|         .allowName=${this.selector.ui_state_content?.allow_name} | ||||
|       ></ha-entity-state-content-picker> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -77,7 +77,7 @@ export class HaServiceControl extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public value?: { | ||||
|     service: string; | ||||
|     action: string; | ||||
|     target?: HassServiceTarget; | ||||
|     data?: Record<string, any>; | ||||
|   }; | ||||
| @@ -112,23 +112,23 @@ export class HaServiceControl extends LitElement { | ||||
|       | undefined | ||||
|       | this["value"]; | ||||
|  | ||||
|     if (oldValue?.service !== this.value?.service) { | ||||
|     if (oldValue?.action !== this.value?.action) { | ||||
|       this._checkedKeys = new Set(); | ||||
|     } | ||||
|  | ||||
|     const serviceData = this._getServiceInfo( | ||||
|       this.value?.service, | ||||
|       this.value?.action, | ||||
|       this.hass.services | ||||
|     ); | ||||
|  | ||||
|     // Fetch the manifest if we have a service selected and the service domain changed. | ||||
|     // If no service is selected, clear the manifest. | ||||
|     if (this.value?.service) { | ||||
|     if (this.value?.action) { | ||||
|       if ( | ||||
|         !oldValue?.service || | ||||
|         computeDomain(this.value.service) !== computeDomain(oldValue.service) | ||||
|         !oldValue?.action || | ||||
|         computeDomain(this.value.action) !== computeDomain(oldValue.action) | ||||
|       ) { | ||||
|         this._fetchManifest(computeDomain(this.value?.service)); | ||||
|         this._fetchManifest(computeDomain(this.value?.action)); | ||||
|       } | ||||
|     } else { | ||||
|       this._manifest = undefined; | ||||
| @@ -168,7 +168,7 @@ export class HaServiceControl extends LitElement { | ||||
|       this._value = this.value; | ||||
|     } | ||||
|  | ||||
|     if (oldValue?.service !== this.value?.service) { | ||||
|     if (oldValue?.action !== this.value?.action) { | ||||
|       let updatedDefaultValue = false; | ||||
|       if (this._value && serviceData) { | ||||
|         const loadDefaults = this.value && !("data" in this.value); | ||||
| @@ -367,7 +367,7 @@ export class HaServiceControl extends LitElement { | ||||
|  | ||||
|   protected render() { | ||||
|     const serviceData = this._getServiceInfo( | ||||
|       this._value?.service, | ||||
|       this._value?.action, | ||||
|       this.hass.services | ||||
|     ); | ||||
|  | ||||
| @@ -392,11 +392,11 @@ export class HaServiceControl extends LitElement { | ||||
|       this._value | ||||
|     ); | ||||
|  | ||||
|     const domain = this._value?.service | ||||
|       ? computeDomain(this._value.service) | ||||
|     const domain = this._value?.action | ||||
|       ? computeDomain(this._value.action) | ||||
|       : undefined; | ||||
|     const serviceName = this._value?.service | ||||
|       ? computeObjectId(this._value.service) | ||||
|     const serviceName = this._value?.action | ||||
|       ? computeObjectId(this._value.action) | ||||
|       : undefined; | ||||
|  | ||||
|     const description = | ||||
| @@ -410,7 +410,7 @@ export class HaServiceControl extends LitElement { | ||||
|       ? nothing | ||||
|       : html`<ha-service-picker | ||||
|           .hass=${this.hass} | ||||
|           .value=${this._value?.service} | ||||
|           .value=${this._value?.action} | ||||
|           .disabled=${this.disabled} | ||||
|           @value-changed=${this._serviceChanged} | ||||
|         ></ha-service-picker>`} | ||||
| @@ -497,6 +497,11 @@ export class HaServiceControl extends LitElement { | ||||
|                 dataField.name || | ||||
|                 dataField.key} | ||||
|               > | ||||
|                 ${this._renderSectionDescription( | ||||
|                   dataField, | ||||
|                   domain, | ||||
|                   serviceName | ||||
|                 )} | ||||
|                 ${Object.entries(dataField.fields).map(([key, field]) => | ||||
|                   this._renderField( | ||||
|                     { key, ...field }, | ||||
| @@ -517,6 +522,22 @@ export class HaServiceControl extends LitElement { | ||||
|         )} `; | ||||
|   } | ||||
|  | ||||
|   private _renderSectionDescription( | ||||
|     dataField: ExtHassService["fields"][number], | ||||
|     domain: string | undefined, | ||||
|     serviceName: string | undefined | ||||
|   ) { | ||||
|     const description = this.hass!.localize( | ||||
|       `component.${domain}.services.${serviceName}.sections.${dataField.key}.description` | ||||
|     ); | ||||
|  | ||||
|     if (!description) { | ||||
|       return nothing; | ||||
|     } | ||||
|  | ||||
|     return html`<p>${description}</p>`; | ||||
|   } | ||||
|  | ||||
|   private _renderField = ( | ||||
|     dataField: ExtHassService["fields"][number], | ||||
|     hasOptional: boolean, | ||||
| @@ -596,11 +617,11 @@ export class HaServiceControl extends LitElement { | ||||
|   }; | ||||
|  | ||||
|   private _localizeValueCallback = (key: string) => { | ||||
|     if (!this._value?.service) { | ||||
|     if (!this._value?.action) { | ||||
|       return ""; | ||||
|     } | ||||
|     return this.hass.localize( | ||||
|       `component.${computeDomain(this._value.service)}.selector.${key}` | ||||
|       `component.${computeDomain(this._value.action)}.selector.${key}` | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
| @@ -612,7 +633,7 @@ export class HaServiceControl extends LitElement { | ||||
|     if (checked) { | ||||
|       this._checkedKeys.add(key); | ||||
|       const field = this._getServiceInfo( | ||||
|         this._value?.service, | ||||
|         this._value?.action, | ||||
|         this.hass.services | ||||
|       )?.fields.find((_field) => _field.key === key); | ||||
|  | ||||
| @@ -658,7 +679,7 @@ export class HaServiceControl extends LitElement { | ||||
|  | ||||
|   private _serviceChanged(ev: ValueChangedEvent<string>) { | ||||
|     ev.stopPropagation(); | ||||
|     if (ev.detail.value === this._value?.service) { | ||||
|     if (ev.detail.value === this._value?.action) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
| @@ -717,7 +738,7 @@ export class HaServiceControl extends LitElement { | ||||
|     } | ||||
|  | ||||
|     const value = { | ||||
|       service: newService, | ||||
|       action: newService, | ||||
|       target, | ||||
|     }; | ||||
|  | ||||
|   | ||||
| @@ -23,6 +23,8 @@ export class HaTimeInput extends LitElement { | ||||
|   @property({ type: Boolean, attribute: "enable-second" }) | ||||
|   public enableSecond = false; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public clearable?: boolean; | ||||
|  | ||||
|   protected render() { | ||||
|     const useAMPM = useAmPm(this.locale); | ||||
|  | ||||
| @@ -48,22 +50,26 @@ export class HaTimeInput extends LitElement { | ||||
|         @value-changed=${this._timeChanged} | ||||
|         .enableSecond=${this.enableSecond} | ||||
|         .required=${this.required} | ||||
|         .clearable=${this.clearable && this.value !== undefined} | ||||
|         .helper=${this.helper} | ||||
|       ></ha-base-time-input> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _timeChanged(ev: CustomEvent<{ value: TimeChangedEvent }>) { | ||||
|   private _timeChanged(ev: CustomEvent<{ value?: TimeChangedEvent }>) { | ||||
|     ev.stopPropagation(); | ||||
|     const eventValue = ev.detail.value; | ||||
|  | ||||
|     const useAMPM = useAmPm(this.locale); | ||||
|     let value; | ||||
|     let value: string | undefined; | ||||
|  | ||||
|     // An undefined eventValue means the time selector is being cleared, | ||||
|     // the `value` variable will (intentionally) be left undefined. | ||||
|     if ( | ||||
|       !isNaN(eventValue.hours) || | ||||
|       eventValue !== undefined && | ||||
|       (!isNaN(eventValue.hours) || | ||||
|         !isNaN(eventValue.minutes) || | ||||
|       !isNaN(eventValue.seconds) | ||||
|         !isNaN(eventValue.seconds)) | ||||
|     ) { | ||||
|       let hours = eventValue.hours || 0; | ||||
|       if (eventValue && useAMPM) { | ||||
|   | ||||
| @@ -49,6 +49,8 @@ export class HaYamlEditor extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public copyClipboard = false; | ||||
|  | ||||
|   @property({ type: Boolean }) public hasExtraActions = false; | ||||
|  | ||||
|   @state() private _yaml = ""; | ||||
|  | ||||
|   public setValue(value): void { | ||||
| @@ -100,13 +102,16 @@ export class HaYamlEditor extends LitElement { | ||||
|         @value-changed=${this._onChange} | ||||
|         dir="ltr" | ||||
|       ></ha-code-editor> | ||||
|       ${this.copyClipboard | ||||
|       ${this.copyClipboard || this.hasExtraActions | ||||
|         ? html`<div class="card-actions"> | ||||
|             <mwc-button @click=${this._copyYaml}> | ||||
|             ${this.copyClipboard | ||||
|               ? html` <mwc-button @click=${this._copyYaml}> | ||||
|                   ${this.hass.localize( | ||||
|                     "ui.components.yaml-editor.copy_to_clipboard" | ||||
|                   )} | ||||
|             </mwc-button> | ||||
|                 </mwc-button>` | ||||
|               : nothing} | ||||
|             <slot name="extra-actions"></slot> | ||||
|           </div>` | ||||
|         : nothing} | ||||
|     `; | ||||
|   | ||||
| @@ -79,7 +79,7 @@ class SearchInputOutlined extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private async _filterInputChanged(e) { | ||||
|     this._filterChanged(e.target.value?.trim()); | ||||
|     this._filterChanged(e.target.value); | ||||
|   } | ||||
|  | ||||
|   private async _clearSearch() { | ||||
|   | ||||
| @@ -67,7 +67,7 @@ class SearchInput extends LitElement { | ||||
|   } | ||||
|  | ||||
|   private async _filterInputChanged(e) { | ||||
|     this._filterChanged(e.target.value?.trim()); | ||||
|     this._filterChanged(e.target.value); | ||||
|   } | ||||
|  | ||||
|   private async _clearSearch() { | ||||
|   | ||||
| @@ -424,7 +424,7 @@ export class HatScriptGraph extends LitElement { | ||||
|     return html` | ||||
|       <hat-graph-node | ||||
|         .graphStart=${graphStart} | ||||
|         .iconPath=${node.service ? undefined : mdiRoomService} | ||||
|         .iconPath=${node.action ? undefined : mdiRoomService} | ||||
|         @focus=${this.selectNode(node, path)} | ||||
|         ?track=${path in this.trace.trace} | ||||
|         ?active=${this.selected === path} | ||||
| @@ -432,11 +432,11 @@ export class HatScriptGraph extends LitElement { | ||||
|         .error=${this.trace.trace[path]?.some((tr) => tr.error)} | ||||
|         tabindex=${this.trace && path in this.trace.trace ? "0" : "-1"} | ||||
|       > | ||||
|         ${node.service | ||||
|         ${node.action | ||||
|           ? html`<ha-service-icon | ||||
|               slot="icon" | ||||
|               .hass=${this.hass} | ||||
|               .service=${node.service} | ||||
|               .service=${node.action} | ||||
|             ></ha-service-icon>` | ||||
|           : nothing} | ||||
|       </hat-graph-node> | ||||
|   | ||||
| @@ -2,10 +2,11 @@ import { stringCompare } from "../common/string/compare"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { DeviceRegistryEntry } from "./device_registry"; | ||||
| import { EntityRegistryEntry } from "./entity_registry"; | ||||
| import { RegistryEntry } from "./registry"; | ||||
|  | ||||
| export { subscribeAreaRegistry } from "./ws-area_registry"; | ||||
|  | ||||
| export interface AreaRegistryEntry { | ||||
| export interface AreaRegistryEntry extends RegistryEntry { | ||||
|   area_id: string; | ||||
|   floor_id: string | null; | ||||
|   name: string; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import { navigate } from "../common/navigate"; | ||||
| import { Context, HomeAssistant } from "../types"; | ||||
| import { BlueprintInput } from "./blueprint"; | ||||
| import { DeviceCondition, DeviceTrigger } from "./device_automation"; | ||||
| import { Action, MODES } from "./script"; | ||||
| import { Action, MODES, migrateAutomationAction } from "./script"; | ||||
|  | ||||
| export const AUTOMATION_DEFAULT_MODE: (typeof MODES)[number] = "single"; | ||||
| export const AUTOMATION_DEFAULT_MAX = 10; | ||||
| @@ -28,7 +28,7 @@ export interface ManualAutomationConfig { | ||||
|   description?: string; | ||||
|   trigger: Trigger | Trigger[]; | ||||
|   condition?: Condition | Condition[]; | ||||
|   action: Action | Action[]; | ||||
|   action?: Action | Action[]; | ||||
|   mode?: (typeof MODES)[number]; | ||||
|   max?: number; | ||||
|   max_exceeded?: | ||||
| @@ -357,7 +357,7 @@ export const normalizeAutomationConfig = < | ||||
| >( | ||||
|   config: T | ||||
| ): T => { | ||||
|   // Normalize data: ensure trigger, action and condition are lists | ||||
|   // Normalize data: ensure triggers, actions and conditions are lists | ||||
|   // Happens when people copy paste their automations into the config | ||||
|   for (const key of ["trigger", "condition", "action"]) { | ||||
|     const value = config[key]; | ||||
| @@ -365,6 +365,11 @@ export const normalizeAutomationConfig = < | ||||
|       config[key] = [value]; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (config.action) { | ||||
|     config.action = migrateAutomationAction(config.action); | ||||
|   } | ||||
|  | ||||
|   return config; | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,10 @@ | ||||
| import { ensureArray } from "../common/array/ensure-array"; | ||||
| import { HomeAssistant } from "../types"; | ||||
|  | ||||
| export const enum ConversationEntityFeature { | ||||
|   CONTROL = 1, | ||||
| } | ||||
|  | ||||
| interface IntentTarget { | ||||
|   type: "area" | "device" | "entity" | "domain" | "device_class" | "custom"; | ||||
|   name: string; | ||||
|   | ||||
| @@ -178,7 +178,11 @@ const getEntityName = ( | ||||
|   entityId: string | undefined | ||||
| ): string => { | ||||
|   if (!entityId) { | ||||
|     return "<unknown entity>"; | ||||
|     return ( | ||||
|       "<" + | ||||
|       hass.localize("ui.panel.config.automation.editor.unknown_entity") + | ||||
|       ">" | ||||
|     ); | ||||
|   } | ||||
|   if (entityId.includes(".")) { | ||||
|     const state = hass.states[entityId]; | ||||
| @@ -191,7 +195,11 @@ const getEntityName = ( | ||||
|   if (entityReg) { | ||||
|     return computeEntityRegistryName(hass, entityReg) || entityId; | ||||
|   } | ||||
|   return "<unknown entity>"; | ||||
|   return ( | ||||
|     "<" + | ||||
|     hass.localize("ui.panel.config.automation.editor.unknown_entity") + | ||||
|     ">" | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export const localizeDeviceAutomationAction = ( | ||||
|   | ||||
| @@ -1,19 +1,20 @@ | ||||
| import { computeStateName } from "../common/entity/compute_state_name"; | ||||
| import { caseInsensitiveStringCompare } from "../common/string/compare"; | ||||
| import type { HomeAssistant } from "../types"; | ||||
| import { ConfigEntry } from "./config_entries"; | ||||
| import type { | ||||
|   EntityRegistryDisplayEntry, | ||||
|   EntityRegistryEntry, | ||||
| } from "./entity_registry"; | ||||
| import { ConfigEntry } from "./config_entries"; | ||||
| import type { EntitySources } from "./entity_sources"; | ||||
| import { RegistryEntry } from "./registry"; | ||||
|  | ||||
| export { | ||||
|   fetchDeviceRegistry, | ||||
|   subscribeDeviceRegistry, | ||||
| } from "./ws-device_registry"; | ||||
|  | ||||
| export interface DeviceRegistryEntry { | ||||
| export interface DeviceRegistryEntry extends RegistryEntry { | ||||
|   id: string; | ||||
|   config_entries: string[]; | ||||
|   connections: Array<[string, string]>; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import { debounce } from "../common/util/debounce"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { LightColor } from "./light"; | ||||
| import { computeDomain } from "../common/entity/compute_domain"; | ||||
| import { RegistryEntry } from "./registry"; | ||||
|  | ||||
| export { subscribeEntityRegistryDisplay } from "./ws-entity_registry_display"; | ||||
|  | ||||
| @@ -43,7 +44,7 @@ export interface EntityRegistryDisplayEntryResponse { | ||||
|   entity_categories: Record<number, EntityCategory>; | ||||
| } | ||||
|  | ||||
| export interface EntityRegistryEntry { | ||||
| export interface EntityRegistryEntry extends RegistryEntry { | ||||
|   id: string; | ||||
|   entity_id: string; | ||||
|   name: string | null; | ||||
|   | ||||
| @@ -4,10 +4,11 @@ import { stringCompare } from "../common/string/compare"; | ||||
| import { debounce } from "../common/util/debounce"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { AreaRegistryEntry } from "./area_registry"; | ||||
| import { RegistryEntry } from "./registry"; | ||||
|  | ||||
| export { subscribeAreaRegistry } from "./ws-area_registry"; | ||||
|  | ||||
| export interface FloorRegistryEntry { | ||||
| export interface FloorRegistryEntry extends RegistryEntry { | ||||
|   floor_id: string; | ||||
|   name: string; | ||||
|   level: number | null; | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| import { Connection, createCollection } from "home-assistant-js-websocket"; | ||||
| import { Store } from "home-assistant-js-websocket/dist/store"; | ||||
| import { stringCompare } from "../common/string/compare"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { debounce } from "../common/util/debounce"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { RegistryEntry } from "./registry"; | ||||
|  | ||||
| export interface LabelRegistryEntry { | ||||
| export interface LabelRegistryEntry extends RegistryEntry { | ||||
|   label_id: string; | ||||
|   name: string; | ||||
|   icon: string | null; | ||||
|   | ||||
| @@ -4,7 +4,12 @@ import { | ||||
| } from "home-assistant-js-websocket"; | ||||
| import { UNAVAILABLE } from "./entity"; | ||||
|  | ||||
| export type LawnMowerEntityState = "paused" | "mowing" | "docked" | "error"; | ||||
| export type LawnMowerEntityState = | ||||
|   | "paused" | ||||
|   | "mowing" | ||||
|   | "returning" | ||||
|   | "docked" | ||||
|   | "error"; | ||||
|  | ||||
| export const enum LawnMowerEntityFeature { | ||||
|   START_MOWING = 1, | ||||
|   | ||||
| @@ -5,10 +5,12 @@ export interface ToggleActionConfig extends BaseActionConfig { | ||||
| } | ||||
|  | ||||
| export interface CallServiceActionConfig extends BaseActionConfig { | ||||
|   action: "call-service"; | ||||
|   service: string; | ||||
|   action: "call-service" | "perform-action"; | ||||
|   /** @deprecated "service" is kept for backwards compatibility. Replaced by "perform_action". */ | ||||
|   service?: string; | ||||
|   perform_action: string; | ||||
|   target?: HassServiceTarget; | ||||
|   // "service_data" is kept for backwards compatibility. Replaced by "data". | ||||
|   /** @deprecated "service_data" is kept for backwards compatibility. Replaced by "data". */ | ||||
|   service_data?: Record<string, unknown>; | ||||
|   data?: Record<string, unknown>; | ||||
| } | ||||
|   | ||||
| @@ -1,12 +1,26 @@ | ||||
| import { Condition } from "../../../panels/lovelace/common/validate-condition"; | ||||
|  | ||||
| export interface LovelaceBadgeConfig { | ||||
|   type?: string; | ||||
|   type: string; | ||||
|   [key: string]: any; | ||||
|   visibility?: Condition[]; | ||||
| } | ||||
|  | ||||
| export const defaultBadgeConfig = (entity_id: string): LovelaceBadgeConfig => ({ | ||||
| export const ensureBadgeConfig = ( | ||||
|   config: Partial<LovelaceBadgeConfig> | string | ||||
| ): LovelaceBadgeConfig => { | ||||
|   if (typeof config === "string") { | ||||
|     return { | ||||
|       type: "entity", | ||||
|   entity: entity_id, | ||||
| }); | ||||
|       entity: config, | ||||
|       display_type: "complete", | ||||
|     }; | ||||
|   } | ||||
|   if ("type" in config && config.type) { | ||||
|     return config as LovelaceBadgeConfig; | ||||
|   } | ||||
|   return { | ||||
|     type: "entity", | ||||
|     ...config, | ||||
|   }; | ||||
| }; | ||||
|   | ||||
| @@ -27,7 +27,7 @@ export interface LovelaceBaseViewConfig { | ||||
|  | ||||
| export interface LovelaceViewConfig extends LovelaceBaseViewConfig { | ||||
|   type?: string; | ||||
|   badges?: (string | LovelaceBadgeConfig)[]; // Badge can be just an entity_id | ||||
|   badges?: (string | Partial<LovelaceBadgeConfig>)[]; // Badge can be just an entity_id or without type | ||||
|   cards?: LovelaceCardConfig[]; | ||||
|   sections?: LovelaceSectionRawConfig[]; | ||||
| } | ||||
|   | ||||
| @@ -1,3 +1,7 @@ | ||||
| import { | ||||
|   HassEntityAttributeBase, | ||||
|   HassEntityBase, | ||||
| } from "home-assistant-js-websocket"; | ||||
| import { HomeAssistant } from "../types"; | ||||
|  | ||||
| export interface BasePerson { | ||||
| @@ -18,6 +22,20 @@ export interface PersonMutableParams { | ||||
|   picture: string | null; | ||||
| } | ||||
|  | ||||
| interface PersonEntityAttributes extends HassEntityAttributeBase { | ||||
|   id?: string; | ||||
|   user_id?: string; | ||||
|   device_trackers?: string[]; | ||||
|   editable?: boolean; | ||||
|   gps_accuracy?: number; | ||||
|   latitude?: number; | ||||
|   longitude?: number; | ||||
| } | ||||
|  | ||||
| export interface PersonEntity extends HassEntityBase { | ||||
|   attributes: PersonEntityAttributes; | ||||
| } | ||||
|  | ||||
| export const fetchPersons = (hass: HomeAssistant) => | ||||
|   hass.callWS<{ | ||||
|     storage: Person[]; | ||||
|   | ||||
							
								
								
									
										4
									
								
								src/data/registry.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/data/registry.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| export interface RegistryEntry { | ||||
|   created_at: number; | ||||
|   modified_at: number; | ||||
| } | ||||
| @@ -32,7 +32,11 @@ export const fetchRepairsIssues = (conn: Connection) => | ||||
|     type: "repairs/list_issues", | ||||
|   }); | ||||
|  | ||||
| export const fetchRepairsIssueData = (conn: Connection, domain, issue_id) => | ||||
| export const fetchRepairsIssueData = ( | ||||
|   conn: Connection, | ||||
|   domain: string, | ||||
|   issue_id: string | ||||
| ) => | ||||
|   conn.sendMessagePromise<{ issue_data: { string: any } }>({ | ||||
|     type: "repairs/get_issue_data", | ||||
|     domain, | ||||
|   | ||||
| @@ -49,7 +49,7 @@ const targetStruct = object({ | ||||
| export const serviceActionStruct: Describe<ServiceAction> = assign( | ||||
|   baseActionStruct, | ||||
|   object({ | ||||
|     service: optional(string()), | ||||
|     action: optional(string()), | ||||
|     service_template: optional(string()), | ||||
|     entity_id: optional(string()), | ||||
|     target: optional(targetStruct), | ||||
| @@ -62,7 +62,7 @@ export const serviceActionStruct: Describe<ServiceAction> = assign( | ||||
| const playMediaActionStruct: Describe<PlayMediaAction> = assign( | ||||
|   baseActionStruct, | ||||
|   object({ | ||||
|     service: literal("media_player.play_media"), | ||||
|     action: literal("media_player.play_media"), | ||||
|     target: optional(object({ entity_id: optional(string()) })), | ||||
|     entity_id: optional(string()), | ||||
|     data: object({ media_content_id: string(), media_content_type: string() }), | ||||
| @@ -73,7 +73,7 @@ const playMediaActionStruct: Describe<PlayMediaAction> = assign( | ||||
| const activateSceneActionStruct: Describe<ServiceSceneAction> = assign( | ||||
|   baseActionStruct, | ||||
|   object({ | ||||
|     service: literal("scene.turn_on"), | ||||
|     action: literal("scene.turn_on"), | ||||
|     target: optional(object({ entity_id: optional(string()) })), | ||||
|     entity_id: optional(string()), | ||||
|     metadata: object(), | ||||
| @@ -132,7 +132,7 @@ export interface EventAction extends BaseAction { | ||||
| } | ||||
|  | ||||
| export interface ServiceAction extends BaseAction { | ||||
|   service?: string; | ||||
|   action?: string; | ||||
|   service_template?: string; | ||||
|   entity_id?: string; | ||||
|   target?: HassServiceTarget; | ||||
| @@ -160,7 +160,7 @@ export interface DelayAction extends BaseAction { | ||||
| } | ||||
|  | ||||
| export interface ServiceSceneAction extends BaseAction { | ||||
|   service: "scene.turn_on"; | ||||
|   action: "scene.turn_on"; | ||||
|   target?: { entity_id?: string }; | ||||
|   entity_id?: string; | ||||
|   metadata: Record<string, unknown>; | ||||
| @@ -191,7 +191,7 @@ export interface WaitForTriggerAction extends BaseAction { | ||||
| } | ||||
|  | ||||
| export interface PlayMediaAction extends BaseAction { | ||||
|   service: "media_player.play_media"; | ||||
|   action: "media_player.play_media"; | ||||
|   target?: { entity_id?: string }; | ||||
|   entity_id?: string; | ||||
|   data: { media_content_id: string; media_content_type: string }; | ||||
| @@ -404,7 +404,7 @@ export const getActionType = (action: Action): ActionType => { | ||||
|   if ("set_conversation_response" in action) { | ||||
|     return "set_conversation_response"; | ||||
|   } | ||||
|   if ("service" in action) { | ||||
|   if ("action" in action) { | ||||
|     if ("metadata" in action) { | ||||
|       if (is(action, activateSceneActionStruct)) { | ||||
|         return "activate_scene"; | ||||
| @@ -425,3 +425,60 @@ export const hasScriptFields = ( | ||||
|   const fields = hass.services.script[computeObjectId(entityId)]?.fields; | ||||
|   return fields !== undefined && Object.keys(fields).length > 0; | ||||
| }; | ||||
|  | ||||
| export const migrateAutomationAction = ( | ||||
|   action: Action | Action[] | ||||
| ): Action | Action[] => { | ||||
|   if (Array.isArray(action)) { | ||||
|     return action.map(migrateAutomationAction) as Action[]; | ||||
|   } | ||||
|  | ||||
|   if ("service" in action) { | ||||
|     if (!("action" in action)) { | ||||
|       action.action = action.service; | ||||
|     } | ||||
|     delete action.service; | ||||
|   } | ||||
|  | ||||
|   if ("sequence" in action) { | ||||
|     for (const sequenceAction of (action as SequenceAction).sequence) { | ||||
|       migrateAutomationAction(sequenceAction); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   const actionType = getActionType(action); | ||||
|  | ||||
|   if (actionType === "parallel") { | ||||
|     const _action = action as ParallelAction; | ||||
|     migrateAutomationAction(_action.parallel); | ||||
|   } | ||||
|  | ||||
|   if (actionType === "choose") { | ||||
|     const _action = action as ChooseAction; | ||||
|     if (Array.isArray(_action.choose)) { | ||||
|       for (const choice of _action.choose) { | ||||
|         migrateAutomationAction(choice.sequence); | ||||
|       } | ||||
|     } else if (_action.choose) { | ||||
|       migrateAutomationAction(_action.choose.sequence); | ||||
|     } | ||||
|     if (_action.default) { | ||||
|       migrateAutomationAction(_action.default); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (actionType === "repeat") { | ||||
|     const _action = action as RepeatAction; | ||||
|     migrateAutomationAction(_action.repeat.sequence); | ||||
|   } | ||||
|  | ||||
|   if (actionType === "if") { | ||||
|     const _action = action as IfAction; | ||||
|     migrateAutomationAction(_action.then); | ||||
|     if (_action.else) { | ||||
|       migrateAutomationAction(_action.else); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return action; | ||||
| }; | ||||
|   | ||||
| @@ -192,7 +192,7 @@ const tryDescribeAction = <T extends ActionType>( | ||||
|  | ||||
|     if ( | ||||
|       config.service_template || | ||||
|       (config.service && isTemplate(config.service)) | ||||
|       (config.action && isTemplate(config.action)) | ||||
|     ) { | ||||
|       return hass.localize( | ||||
|         targets.length | ||||
| @@ -204,8 +204,8 @@ const tryDescribeAction = <T extends ActionType>( | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (config.service) { | ||||
|       const [domain, serviceName] = config.service.split(".", 2); | ||||
|     if (config.action) { | ||||
|       const [domain, serviceName] = config.action.split(".", 2); | ||||
|       const service = | ||||
|         hass.localize(`component.${domain}.services.${serviceName}.name`) || | ||||
|         hass.services[domain][serviceName]?.name; | ||||
| @@ -217,7 +217,7 @@ const tryDescribeAction = <T extends ActionType>( | ||||
|             : `${actionTranslationBaseKey}.service.description.service_name_no_targets`, | ||||
|           { | ||||
|             domain: domainToName(hass.localize, domain), | ||||
|             name: service || config.service, | ||||
|             name: service || config.action, | ||||
|             targets: formatListWithAnds(hass.locale, targets), | ||||
|           } | ||||
|         ); | ||||
| @@ -230,7 +230,7 @@ const tryDescribeAction = <T extends ActionType>( | ||||
|         { | ||||
|           name: service | ||||
|             ? `${domainToName(hass.localize, domain)}: ${service}` | ||||
|             : config.service, | ||||
|             : config.action, | ||||
|           targets: formatListWithAnds(hass.locale, targets), | ||||
|         } | ||||
|       ); | ||||
|   | ||||
| @@ -8,6 +8,7 @@ export interface RelatedResult { | ||||
|   device?: string[]; | ||||
|   entity?: string[]; | ||||
|   group?: string[]; | ||||
|   integration?: string[]; | ||||
|   scene?: string[]; | ||||
|   script?: string[]; | ||||
|   script_blueprint?: string[]; | ||||
|   | ||||
| @@ -101,8 +101,9 @@ export interface AttributeSelector { | ||||
| } | ||||
|  | ||||
| export interface BooleanSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   boolean: {} | null; | ||||
|   boolean: { | ||||
|     mode?: "checkbox" | "switch"; | ||||
|   } | null; | ||||
| } | ||||
|  | ||||
| export interface ColorRGBSelector { | ||||
| @@ -203,6 +204,7 @@ export interface LegacyDeviceSelector { | ||||
| export interface DurationSelector { | ||||
|   duration: { | ||||
|     enable_day?: boolean; | ||||
|     enable_millisecond?: boolean; | ||||
|   } | null; | ||||
| } | ||||
|  | ||||
| @@ -460,6 +462,7 @@ export interface UiStateContentSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   ui_state_content: { | ||||
|     entity_id?: string; | ||||
|     allow_name?: boolean; | ||||
|   } | null; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,14 @@ import { | ||||
|   mdiTuneVariant, | ||||
|   mdiWaterPercent, | ||||
| } from "@mdi/js"; | ||||
| import { CSSResultGroup, LitElement, css, html, nothing } from "lit"; | ||||
| import { | ||||
|   CSSResultGroup, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   css, | ||||
|   html, | ||||
|   nothing, | ||||
| } from "lit"; | ||||
| import { property, state } from "lit/decorators"; | ||||
| import { stopPropagation } from "../../../common/dom/stop_propagation"; | ||||
| import { supportsFeature } from "../../../common/entity/supports-feature"; | ||||
| @@ -39,6 +46,17 @@ class MoreInfoClimate extends LitElement { | ||||
|  | ||||
|   @state() private _mainControl: MainControl = "temperature"; | ||||
|  | ||||
|   protected willUpdate(changedProps: PropertyValues): void { | ||||
|     if ( | ||||
|       changedProps.has("stateObj") && | ||||
|       this.stateObj && | ||||
|       this._mainControl === "humidity" && | ||||
|       !supportsFeature(this.stateObj, ClimateEntityFeature.TARGET_HUMIDITY) | ||||
|     ) { | ||||
|       this._mainControl = "temperature"; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|     if (!this.stateObj) { | ||||
|       return nothing; | ||||
|   | ||||
| @@ -148,7 +148,7 @@ class MoreInfoScript extends LitElement { | ||||
|     const newState = this.stateObj; | ||||
|  | ||||
|     if (newState && (!oldState || oldState.entity_id !== newState.entity_id)) { | ||||
|       this._scriptData = { service: newState.entity_id, data: {} }; | ||||
|       this._scriptData = { action: newState.entity_id, data: {} }; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -41,6 +41,8 @@ import { AudioRecorder } from "../../util/audio-recorder"; | ||||
| import { documentationUrl } from "../../util/documentation-url"; | ||||
| import { showAlertDialog } from "../generic/show-dialog-box"; | ||||
| import { VoiceCommandDialogParams } from "./show-ha-voice-command-dialog"; | ||||
| import { supportsFeature } from "../../common/entity/supports-feature"; | ||||
| import { ConversationEntityFeature } from "../../data/conversation"; | ||||
|  | ||||
| interface Message { | ||||
|   who: string; | ||||
| @@ -136,6 +138,14 @@ export class HaVoiceCommandDialog extends LitElement { | ||||
|       return nothing; | ||||
|     } | ||||
|  | ||||
|     const controlHA = !this._pipeline | ||||
|       ? false | ||||
|       : this.hass.states[this._pipeline?.conversation_engine] | ||||
|         ? supportsFeature( | ||||
|             this.hass.states[this._pipeline?.conversation_engine], | ||||
|             ConversationEntityFeature.CONTROL | ||||
|           ) | ||||
|         : true; | ||||
|     const supportsMicrophone = AudioRecorder.isSupported; | ||||
|     const supportsSTT = this._pipeline?.stt_engine; | ||||
|  | ||||
| @@ -212,6 +222,15 @@ export class HaVoiceCommandDialog extends LitElement { | ||||
|             ></ha-icon-button> | ||||
|           </a> | ||||
|         </ha-dialog-header> | ||||
|         ${controlHA | ||||
|           ? nothing | ||||
|           : html` | ||||
|               <ha-alert> | ||||
|                 ${this.hass.localize( | ||||
|                   "ui.dialogs.voice_command.conversation_no_control" | ||||
|                 )} | ||||
|               </ha-alert> | ||||
|             `} | ||||
|         <div class="messages"> | ||||
|           <div class="messages-container" id="scroll-container"> | ||||
|             ${this._conversation!.map( | ||||
| @@ -469,10 +488,11 @@ export class HaVoiceCommandDialog extends LitElement { | ||||
|       who: "user", | ||||
|       text: "…", | ||||
|     }; | ||||
|     this._audioRecorder.start().then(() => { | ||||
|     await this._audioRecorder.start(); | ||||
|  | ||||
|     this._addMessage(userMessage); | ||||
|     this.requestUpdate("_audioRecorder"); | ||||
|     }); | ||||
|  | ||||
|     const hassMessage: Message = { | ||||
|       who: "hass", | ||||
|       text: "…", | ||||
|   | ||||
| @@ -226,6 +226,7 @@ export const provideHass = ( | ||||
|       }, | ||||
|       suspendReconnectUntil: noop, | ||||
|       suspend: noop, | ||||
|       ping: noop, | ||||
|       socket: { | ||||
|         readyState: WebSocket.OPEN, | ||||
|       }, | ||||
|   | ||||
| @@ -16,8 +16,9 @@ | ||||
|   ) { | ||||
|     _ls("/static/polyfills/webcomponents-bundle.js", true); | ||||
|   } | ||||
|   var isS11_12 = | ||||
|     /(?:.*(?:iPhone|iPad).*OS (?:11|12)_\d)|(?:.*Version\/(?:11|12)(?:\.\d+)*.*Safari\/)/.test( | ||||
|       navigator.userAgent | ||||
|     ); | ||||
|   // Modern browsers are detected primarily using the user agent string. | ||||
|   // A feature detection which roughly lines up with the modern targets is used | ||||
|   // as a fallback to guard against spoofs. It should be updated periodically. | ||||
|   var isModern = <%= modernRegex %>.test(navigator.userAgent) && | ||||
|     "findLast" in Array.prototype; | ||||
| </script> | ||||
|   | ||||
| @@ -1,3 +1,11 @@ | ||||
| <script <% if (!useWDS) { %>crossorigin="use-credentials"<% } %>> | ||||
|   if (isModern) { | ||||
|     <% for (const entry of latestEntryJS) { %> | ||||
|       import("<%= entry %>"); | ||||
|     <% } %> | ||||
|     window.latestJS = true; | ||||
|   } | ||||
| </script> | ||||
| <script> | ||||
|   (function() { | ||||
|     if (!window.latestJS) { | ||||
| @@ -52,18 +52,13 @@ | ||||
|     </div> | ||||
|     <%= renderTemplate("_js_base.html.template") %> | ||||
|     <%= renderTemplate("_preload_roboto.html.template") %> | ||||
|     <%= renderTemplate("_script_loader.html.template") %> | ||||
|     <script crossorigin="use-credentials"> | ||||
|       // Safari 12 and below does not have a compliant ES2015 implementation of template literals, so we ship ES5 | ||||
|       if (!isS11_12) { | ||||
|         <% for (const entry of latestEntryJS) { %> | ||||
|           import("<%= entry %>"); | ||||
|         <% } %> | ||||
|         window.latestJS = true; | ||||
|       if (window.latestJS) { | ||||
|         window.providersPromise = fetch("/auth/providers", { | ||||
|           credentials: "same-origin", | ||||
|         }); | ||||
|       } | ||||
|     </script> | ||||
|     <%= renderTemplate("_script_load_es5.html.template") %> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -42,26 +42,46 @@ | ||||
|         width: 112px; | ||||
|         flex-shrink: 0; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer { | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-top { | ||||
|         flex: 1; | ||||
|         margin-top: calc( 2 * max(env(safe-area-inset-bottom), 48px) + 46px ); | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-bottom { | ||||
|         flex: 1; | ||||
|         padding-top: 48px; | ||||
|       } | ||||
|       .ohf-logo { | ||||
|         margin: max(env(safe-area-inset-bottom), 48px) 0; | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         align-items: center; | ||||
|         opacity: .66; | ||||
|       } | ||||
|       @media (prefers-color-scheme: dark) { | ||||
|         .ohf-logo { | ||||
|           filter: invert(1); | ||||
|         } | ||||
|       } | ||||
|     </style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div id="ha-launch-screen"> | ||||
|       <div class="ha-launch-screen-spacer"></div> | ||||
|       <div class="ha-launch-screen-spacer-top"></div> | ||||
|       <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 240 240"> | ||||
|         <path fill="#18BCF2" d="M240 224.762a15 15 0 0 1-15 15H15a15 15 0 0 1-15-15v-90c0-8.25 4.77-19.769 10.61-25.609l98.78-98.7805c5.83-5.83 15.38-5.83 21.21 0l98.79 98.7895c5.83 5.83 10.61 17.36 10.61 25.61v90-.01Z"/> | ||||
|         <path fill="#F2F4F9" d="m107.27 239.762-40.63-40.63c-2.09.72-4.32 1.13-6.64 1.13-11.3 0-20.5-9.2-20.5-20.5s9.2-20.5 20.5-20.5 20.5 9.2 20.5 20.5c0 2.33-.41 4.56-1.13 6.65l31.63 31.63v-115.88c-6.8-3.3395-11.5-10.3195-11.5-18.3895 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5c0 8.07-4.7 15.05-11.5 18.3895v81.27l31.46-31.46c-.62-1.96-.96-4.04-.96-6.2 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5-9.2 20.5-20.5 20.5c-2.5 0-4.88-.47-7.09-1.29L129 208.892v30.88z"/> | ||||
|       </svg> | ||||
|       <div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer"></div> | ||||
|       <div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer-bottom"></div> | ||||
|       <div class="ohf-logo"> | ||||
|         <img src="/static/images/ohf-badge.svg" alt="Home Assistant is a project by the Open Home Foundation" height="46"> | ||||
|       </div> | ||||
|     </div> | ||||
|     <home-assistant></home-assistant> | ||||
|     <%= renderTemplate("_js_base.html.template") %> | ||||
|     <%= renderTemplate("_preload_roboto.html.template") %> | ||||
|     <script <% if (!useWDS) { %>crossorigin="use-credentials"<% } %>> | ||||
|       // Safari 12 and below does not have a compliant ES2015 implementation of template literals, so we ship ES5 | ||||
|       if (!isS11_12) { | ||||
|       if (isModern) { | ||||
|         <% for (const entry of latestEntryJS) { %> | ||||
|           import("<%= entry %>"); | ||||
|         <% } %> | ||||
|   | ||||
| @@ -48,18 +48,13 @@ | ||||
|     </div> | ||||
|     <%= renderTemplate("_js_base.html.template") %> | ||||
|     <%= renderTemplate("_preload_roboto.html.template") %> | ||||
|     <%= renderTemplate("_script_loader.html.template") %> | ||||
|     <script crossorigin="use-credentials"> | ||||
|       // Safari 12 and below does not have a compliant ES2015 implementation of template literals, so we ship ES5 | ||||
|       if (!isS11_12) { | ||||
|         <% for (const entry of latestEntryJS) { %> | ||||
|           import("<%= entry %>"); | ||||
|         <% } %> | ||||
|         window.latestJS = true; | ||||
|       if (window.latestJS) { | ||||
|         window.stepsPromise = fetch("/api/onboarding", { | ||||
|           credentials: "same-origin", | ||||
|         }); | ||||
|       } | ||||
|     </script> | ||||
|     <%= renderTemplate("_script_load_es5.html.template") %> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -39,7 +39,13 @@ class HaInitPage extends LitElement { | ||||
|           </div> | ||||
|           <div id="loading-text"> | ||||
|             ${this.migration | ||||
|               ? "Database migration in progress, please wait this might take some time" | ||||
|               ? html` | ||||
|                   Database upgrade is in progress, Home Assistant will not start | ||||
|                   until the upgrade is completed. | ||||
|                   <br /><br /> | ||||
|                   The upgrade may need a long time to complete, please be | ||||
|                   patient. | ||||
|                 ` | ||||
|               : "Loading data"} | ||||
|           </div> | ||||
|         `; | ||||
|   | ||||
| @@ -266,7 +266,8 @@ export class HaTabsSubpageDataTable extends LitElement { | ||||
|           <ha-assist-chip | ||||
|             .label=${localize("ui.components.subpage-data-table.sort_by", { | ||||
|               sortColumn: this._sortColumn | ||||
|                 ? ` ${this.columns[this._sortColumn].title || this.columns[this._sortColumn].label}` | ||||
|                 ? ` ${this.columns[this._sortColumn]?.title || this.columns[this._sortColumn]?.label}` || | ||||
|                   "" | ||||
|                 : "", | ||||
|             })} | ||||
|             id="sort-by-anchor" | ||||
|   | ||||
| @@ -200,7 +200,11 @@ export class HomeAssistantAppEl extends QuickBarMixin(HassElement) { | ||||
|   } | ||||
|  | ||||
|   protected async checkDataBaseMigration() { | ||||
|     if (this.hass?.config?.components.includes("recorder")) { | ||||
|     if (__DEMO__) { | ||||
|       this._databaseMigration = false; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     let recorderInfoProm: Promise<RecorderInfo> | undefined; | ||||
|     const preloadWindow = window as WindowWithPreloads; | ||||
|     // On first load, we speed up loading page by having recorderInfoProm ready | ||||
| @@ -208,17 +212,20 @@ export class HomeAssistantAppEl extends QuickBarMixin(HassElement) { | ||||
|       recorderInfoProm = preloadWindow.recorderInfoProm; | ||||
|       preloadWindow.recorderInfoProm = undefined; | ||||
|     } | ||||
|       const info = await (recorderInfoProm || | ||||
|         getRecorderInfo(this.hass.connection)); | ||||
|     const info = await ( | ||||
|       recorderInfoProm || getRecorderInfo(this.hass!.connection) | ||||
|     ).catch((err) => { | ||||
|       // If the command failed with code unknown_command, recorder is not enabled, | ||||
|       // otherwise re-throw the error | ||||
|       if (err.code !== "unknown_command") throw err; | ||||
|       return { migration_in_progress: false, migration_is_live: false }; | ||||
|     }); | ||||
|     this._databaseMigration = | ||||
|       info.migration_in_progress && !info.migration_is_live; | ||||
|     if (this._databaseMigration) { | ||||
|       // check every 5 seconds if the migration is done | ||||
|       setTimeout(() => this.checkDataBaseMigration(), 5000); | ||||
|     } | ||||
|     } else { | ||||
|       this._databaseMigration = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected async _initializeHass() { | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user