mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-26 12:09:47 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			card_edito
			...
			remove-unc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | f4bfcc6a69 | 
| @@ -1,25 +1,28 @@ | ||||
| [modern] | ||||
| # 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 | ||||
| # 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 | ||||
| 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 at or above 0.05% | ||||
| # - with global utilization 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). | ||||
| # | ||||
| @@ -33,10 +36,4 @@ not UCAndroid > 0 | ||||
| # As of May 2023, only web sockets must be added to the query. | ||||
| unreleased versions | ||||
| last 7 years | ||||
| >= 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 | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | ||||
| FROM mcr.microsoft.com/devcontainers/python:3.12 | ||||
| FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.11 | ||||
|  | ||||
| ENV \ | ||||
|   DEBIAN_FRONTEND=noninteractive \ | ||||
|   | ||||
| @@ -8,7 +8,6 @@ | ||||
|   "postCreateCommand": "sudo apt update && sudo apt upgrade -y && sudo apt install -y libpcap-dev", | ||||
|   "postStartCommand": "script/bootstrap", | ||||
|   "containerEnv": { | ||||
|     "DEV_CONTAINER": "1", | ||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" | ||||
|   }, | ||||
|   "customizations": { | ||||
|   | ||||
| @@ -115,7 +115,6 @@ | ||||
|       } | ||||
|     ], | ||||
|     "unused-imports/no-unused-imports": "error", | ||||
|     "lit/attribute-names": "warn", | ||||
|     "lit/attribute-value-entities": "off", | ||||
|     "lit/no-template-map": "off", | ||||
|     "lit/no-native-attributes": "warn", | ||||
| @@ -126,5 +125,6 @@ | ||||
|     "lit-a11y/anchor-is-valid": "warn", | ||||
|     "lit-a11y/role-has-required-aria-attrs": "warn" | ||||
|   }, | ||||
|   "plugins": ["unused-imports"] | ||||
|   "plugins": ["disable", "unused-imports"], | ||||
|   "processor": "disable/disable" | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -21,12 +21,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -57,12 +57,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|   | ||||
							
								
								
									
										22
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -24,9 +24,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -37,7 +37,7 @@ jobs: | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages | ||||
|       - name: Setup lint cache | ||||
|         uses: actions/cache@v4.0.2 | ||||
|         uses: actions/cache@v4.0.0 | ||||
|         with: | ||||
|           path: | | ||||
|             node_modules/.cache/prettier | ||||
| @@ -58,9 +58,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -76,9 +76,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -89,7 +89,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@v4.4.0 | ||||
|         uses: actions/upload-artifact@v4.3.1 | ||||
|         with: | ||||
|           name: frontend-bundle-stats | ||||
|           path: build/stats/*.json | ||||
| @@ -100,9 +100,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -113,7 +113,7 @@ jobs: | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|       - name: Upload bundle stats | ||||
|         uses: actions/upload-artifact@v4.4.0 | ||||
|         uses: actions/upload-artifact@v4.3.1 | ||||
|         with: | ||||
|           name: supervisor-bundle-stats | ||||
|           path: build/stats/*.json | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,7 +23,7 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         with: | ||||
|           # We must fetch at least the immediate parents so that if this is | ||||
|           # a pull request then we can checkout the head. | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -22,12 +22,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -58,12 +58,12 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -16,10 +16,10 @@ jobs: | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -21,10 +21,10 @@ jobs: | ||||
|     if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview') | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | ||||
|     - cron: "0 1 * * *" | ||||
|  | ||||
| env: | ||||
|   PYTHON_VERSION: "3.12" | ||||
|   PYTHON_VERSION: "3.11" | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| permissions: | ||||
| @@ -20,7 +20,7 @@ jobs: | ||||
|       contents: write | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/setup-python@v5 | ||||
| @@ -28,7 +28,7 @@ jobs: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -57,14 +57,14 @@ jobs: | ||||
|         run: tar -czvf translations.tar.gz translations | ||||
|  | ||||
|       - name: Upload build artifacts | ||||
|         uses: actions/upload-artifact@v4.4.0 | ||||
|         uses: actions/upload-artifact@v4.3.1 | ||||
|         with: | ||||
|           name: wheels | ||||
|           path: dist/home_assistant_frontend*.whl | ||||
|           if-no-files-found: error | ||||
|  | ||||
|       - name: Upload translations | ||||
|         uses: actions/upload-artifact@v4.4.0 | ||||
|         uses: actions/upload-artifact@v4.3.1 | ||||
|         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.12 | ||||
|         uses: relative-ci/agent-action@v2.1.10 | ||||
|         with: | ||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} | ||||
|           token: ${{ github.token }} | ||||
|   | ||||
							
								
								
									
										12
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | ||||
|       - published | ||||
|  | ||||
| env: | ||||
|   PYTHON_VERSION: "3.12" | ||||
|   PYTHON_VERSION: "3.11" | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| # Set default workflow permissions | ||||
| @@ -23,7 +23,7 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|  | ||||
|       - name: Verify version | ||||
|         uses: home-assistant/actions/helpers/verify-version@master | ||||
| @@ -34,7 +34,7 @@ jobs: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
|       - name: Setup Node | ||||
|         uses: actions/setup-node@v4.0.3 | ||||
|         uses: actions/setup-node@v4.0.2 | ||||
|         with: | ||||
|           node-version-file: ".nvmrc" | ||||
|           cache: yarn | ||||
| @@ -55,7 +55,7 @@ jobs: | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@v2.0.8 | ||||
|         uses: softprops/action-gh-release@v0.1.15 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
| @@ -74,9 +74,9 @@ jobs: | ||||
|           echo "home-assistant-frontend==$version" > ./requirements.txt | ||||
|  | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2024.07.1 | ||||
|         uses: home-assistant/wheels@2024.01.0 | ||||
|         with: | ||||
|           abi: cp312 | ||||
|           abi: cp311 | ||||
|           tag: musllinux_1_2 | ||||
|           arch: amd64 | ||||
|           wheels-key: ${{ secrets.WHEELS_KEY }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -13,7 +13,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.7 | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|  | ||||
|       - name: Upload Translations | ||||
|         run: | | ||||
|   | ||||
| @@ -1 +1,4 @@ | ||||
| #!/usr/bin/env sh | ||||
| . "$(dirname -- "$0")/_/husky.sh" | ||||
|  | ||||
| yarn run lint-staged --relative --shell "/bin/bash" | ||||
|   | ||||
| @@ -0,0 +1,13 @@ | ||||
| diff --git a/simple-tooltip.js b/simple-tooltip.js | ||||
| index 78a87f6a223925f0e29fbedb268c85a142ec6985..3d686dd6a3d5a93342b4b01408089fc316b408ca 100644 | ||||
| --- a/simple-tooltip.js | ||||
| +++ b/simple-tooltip.js | ||||
| @@ -195,6 +195,8 @@ class SimpleTooltip extends LitElement { | ||||
|          .hidden { | ||||
|            position: absolute; | ||||
|            left: -10000px; | ||||
| +          inset-inline-start: -10000px; | ||||
| +          inset-inline-end: initial; | ||||
|            top: auto; | ||||
|            width: 1px; | ||||
|            height: 1px; | ||||
| @@ -1,18 +0,0 @@ | ||||
| diff --git a/dist/hls.light.mjs b/dist/hls.light.mjs | ||||
| index eed9d788fafdb159975e1a2eb08ac88ba9c9ac33..ace881935e6665946f1c8110ebd2f739cde4427e 100644 | ||||
| --- a/dist/hls.light.mjs | ||||
| +++ b/dist/hls.light.mjs | ||||
| @@ -20523,9 +20523,9 @@ class Hls { | ||||
|  } | ||||
|  Hls.defaultConfig = void 0; | ||||
|   | ||||
| -var KeySystemFormats = empty.KeySystemFormats; | ||||
| -var KeySystems = empty.KeySystems; | ||||
| -var SubtitleStreamController = empty.SubtitleStreamController; | ||||
| -var TimelineController = empty.TimelineController; | ||||
| +var KeySystemFormats = empty; | ||||
| +var KeySystems = empty; | ||||
| +var SubtitleStreamController = empty; | ||||
| +var TimelineController = empty; | ||||
|  export { AbrController, AttrList, Cues as AudioStreamController, Cues as AudioTrackController, BasePlaylistController, BaseSegment, BaseStreamController, BufferController, Cues as CMCDController, CapLevelController, ChunkMetadata, ContentSteeringController, DateRange, Cues as EMEController, ErrorActionFlags, ErrorController, ErrorDetails, ErrorTypes, Events, FPSController, Fragment, Hls, HlsSkip, HlsUrlParameters, KeySystemFormats, KeySystems, Level, LevelDetails, LevelKey, LoadStats, MetadataSchema, NetworkErrorAction, Part, PlaylistLevelType, SubtitleStreamController, Cues as SubtitleTrackController, TimelineController, Hls as default, getMediaSource, isMSESupported, isSupported }; | ||||
|  //# sourceMappingURL=hls.light.mjs.map | ||||
| @@ -1,7 +1,16 @@ | ||||
| diff --git a/modular/sortable.core.esm.js b/modular/sortable.core.esm.js
 | ||||
| index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa526090a00 100644
 | ||||
| index 93ba17509e2e8583ab241fea6845fbe714c584a2..de0651ddb5dced30d36f7d764da0dd0b441f523f 100644
 | ||||
| --- a/modular/sortable.core.esm.js
 | ||||
| +++ b/modular/sortable.core.esm.js
 | ||||
| @@ -1461,7 +1461,7 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
|            } | ||||
|            target = parent; // store last element | ||||
|          } | ||||
| -        /* jshint boss:true */ while (parent = parent.parentNode);
 | ||||
| +        /* jshint boss:true */ while (parent = parent.parentNode || parent.getRootNode().host);
 | ||||
|        } | ||||
|        _unhideGhostForTarget(); | ||||
|      } | ||||
| @@ -1781,11 +1781,16 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
|          } | ||||
|          if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, !!target) !== false) { | ||||
| @@ -24,7 +33,7 @@ index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa5 | ||||
|            } | ||||
|            parentEl = el; // actualization | ||||
|   | ||||
| @@ -1802,7 +1807,12 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
| @@ -1802,7 +1807,13 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
|          targetRect = getRect(target); | ||||
|          if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, false) !== false) { | ||||
|            capture(); | ||||
| @@ -35,10 +44,11 @@ index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa5 | ||||
| +          catch(err) {
 | ||||
| +            return completed(false);
 | ||||
| +          }
 | ||||
| +          
 | ||||
|            parentEl = el; // actualization | ||||
|   | ||||
|            changed(); | ||||
| @@ -1849,10 +1859,15 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
| @@ -1849,12 +1860,17 @@ Sortable.prototype = /** @lends Sortable.prototype */{
 | ||||
|            _silent = true; | ||||
|            setTimeout(_unsilent, 30); | ||||
|            capture(); | ||||
| @@ -46,6 +56,8 @@ index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa5 | ||||
| -            el.appendChild(dragEl);
 | ||||
| -          } else {
 | ||||
| -            target.parentNode.insertBefore(dragEl, after ? nextSibling : target);
 | ||||
| -          }
 | ||||
|   | ||||
| +          try {
 | ||||
| +            if (after && !nextSibling) {
 | ||||
| +              el.appendChild(dragEl);
 | ||||
| @@ -55,6 +67,7 @@ index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa5 | ||||
| +          }
 | ||||
| +          catch(err) {
 | ||||
| +            return completed(false);
 | ||||
|            } | ||||
|   | ||||
| +          }
 | ||||
|            // Undo chrome's scroll adjustment (has no effect on other browsers) | ||||
|            if (scrolledPastTop) { | ||||
|              scrollBy(scrolledPastTop, 0, scrollBefore - scrolledPastTop.scrollTop); | ||||
| @@ -1,55 +0,0 @@ | ||||
| diff --git a/build/inject-manifest.js b/build/inject-manifest.js | ||||
| index 60e3d2bb51c11a19fbbedbad65e101082ec41c36..fed6026630f43f86e25446383982cf6fb694313b 100644 | ||||
| --- a/build/inject-manifest.js | ||||
| +++ b/build/inject-manifest.js | ||||
| @@ -104,7 +104,7 @@ async function injectManifest(config) { | ||||
|              replaceString: manifestString, | ||||
|              searchString: options.injectionPoint, | ||||
|          }); | ||||
| -        filesToWrite[options.swDest] = source; | ||||
| +        filesToWrite[options.swDest] = source.replace(url, encodeURI(upath_1.default.basename(destPath))); | ||||
|          filesToWrite[destPath] = map; | ||||
|      } | ||||
|      else { | ||||
| diff --git a/build/lib/translate-url-to-sourcemap-paths.js b/build/lib/translate-url-to-sourcemap-paths.js | ||||
| index 3220c5474eeac6e8a56ca9b2ac2bd9be48529e43..5f003879a904d4840529a42dd056d288fd213771 100644 | ||||
| --- a/build/lib/translate-url-to-sourcemap-paths.js | ||||
| +++ b/build/lib/translate-url-to-sourcemap-paths.js | ||||
| @@ -22,7 +22,7 @@ function translateURLToSourcemapPaths(url, swSrc, swDest) { | ||||
|          const possibleSrcPath = upath_1.default.resolve(upath_1.default.dirname(swSrc), url); | ||||
|          if (fs_extra_1.default.existsSync(possibleSrcPath)) { | ||||
|              srcPath = possibleSrcPath; | ||||
| -            destPath = upath_1.default.resolve(upath_1.default.dirname(swDest), url); | ||||
| +            destPath = `${swDest}.map`; | ||||
|          } | ||||
|          else { | ||||
|              warning = `${errors_1.errors['cant-find-sourcemap']} ${possibleSrcPath}`; | ||||
| diff --git a/src/inject-manifest.ts b/src/inject-manifest.ts | ||||
| index 8795ddcaa77aea7b0356417e4bc4b19e2b3f860c..fcdc68342d9ac53936c9ed40a9ccfc2f5070cad3 100644 | ||||
| --- a/src/inject-manifest.ts | ||||
| +++ b/src/inject-manifest.ts | ||||
| @@ -129,7 +129,10 @@ export async function injectManifest( | ||||
|        searchString: options.injectionPoint!, | ||||
|      }); | ||||
|   | ||||
| -    filesToWrite[options.swDest] = source; | ||||
| +    filesToWrite[options.swDest] = source.replace( | ||||
| +      url!, | ||||
| +      encodeURI(upath.basename(destPath)), | ||||
| +    ); | ||||
|      filesToWrite[destPath] = map; | ||||
|    } else { | ||||
|      // If there's no sourcemap associated with swSrc, a simple string | ||||
| diff --git a/src/lib/translate-url-to-sourcemap-paths.ts b/src/lib/translate-url-to-sourcemap-paths.ts | ||||
| index 072eac40d4ef5d095a01cb7f7e392a9e034853bd..f0bbe69e88ef3a415de18a7e9cb264daea273d71 100644 | ||||
| --- a/src/lib/translate-url-to-sourcemap-paths.ts | ||||
| +++ b/src/lib/translate-url-to-sourcemap-paths.ts | ||||
| @@ -28,7 +28,7 @@ export function translateURLToSourcemapPaths( | ||||
|      const possibleSrcPath = upath.resolve(upath.dirname(swSrc), url); | ||||
|      if (fse.existsSync(possibleSrcPath)) { | ||||
|        srcPath = possibleSrcPath; | ||||
| -      destPath = upath.resolve(upath.dirname(swDest), url); | ||||
| +      destPath = `${swDest}.map`; | ||||
|      } else { | ||||
|        warning = `${errors['cant-find-sourcemap']} ${possibleSrcPath}`; | ||||
|      } | ||||
							
								
								
									
										893
									
								
								.yarn/releases/yarn-4.1.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										893
									
								
								.yarn/releases/yarn-4.1.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										925
									
								
								.yarn/releases/yarn-4.4.1.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										925
									
								
								.yarn/releases/yarn-4.4.1.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												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.4.1.cjs | ||||
| yarnPath: .yarn/releases/yarn-4.1.0.cjs | ||||
|   | ||||
| @@ -1,56 +1,7 @@ | ||||
| import defineProvider from "@babel/helper-define-polyfill-provider"; | ||||
| import { join } from "node:path"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const POLYFILL_DIR = join(paths.polymer_dir, "src/resources/polyfills"); | ||||
|  | ||||
| // List of polyfill keys with supported browser targets for the functionality | ||||
| const PolyfillSupport = { | ||||
|   // Note states and shadowRoot properties should be supported. | ||||
|   "element-internals": { | ||||
|     android: 90, | ||||
|     chrome: 90, | ||||
|     edge: 90, | ||||
|     firefox: 126, | ||||
|     ios: 17.4, | ||||
|     opera: 76, | ||||
|     opera_mobile: 64, | ||||
|     safari: 17.4, | ||||
|     samsung: 15.0, | ||||
|   }, | ||||
|   "element-append": { | ||||
|     android: 54, | ||||
|     chrome: 54, | ||||
|     edge: 17, | ||||
|     firefox: 49, | ||||
|     ios: 10.0, | ||||
|     opera: 41, | ||||
|     opera_mobile: 41, | ||||
|     safari: 10.0, | ||||
|     samsung: 6.0, | ||||
|   }, | ||||
|   "element-getattributenames": { | ||||
|     android: 61, | ||||
|     chrome: 61, | ||||
|     edge: 18, | ||||
|     firefox: 45, | ||||
|     ios: 10.3, | ||||
|     opera: 48, | ||||
|     opera_mobile: 45, | ||||
|     safari: 10.1, | ||||
|     samsung: 8.0, | ||||
|   }, | ||||
|   "element-toggleattribute": { | ||||
|     android: 69, | ||||
|     chrome: 69, | ||||
|     edge: 18, | ||||
|     firefox: 63, | ||||
|     ios: 12.0, | ||||
|     opera: 56, | ||||
|     opera_mobile: 48, | ||||
|     safari: 12.0, | ||||
|     samsung: 10.0, | ||||
|   }, | ||||
|   fetch: { | ||||
|     android: 42, | ||||
|     chrome: 42, | ||||
| @@ -62,31 +13,6 @@ const PolyfillSupport = { | ||||
|     safari: 10.1, | ||||
|     samsung: 4.0, | ||||
|   }, | ||||
|   "intl-getcanonicallocales": { | ||||
|     android: 54, | ||||
|     chrome: 54, | ||||
|     edge: 16, | ||||
|     firefox: 48, | ||||
|     ios: 10.3, | ||||
|     opera: 41, | ||||
|     opera_mobile: 41, | ||||
|     safari: 10.1, | ||||
|     samsung: 6.0, | ||||
|   }, | ||||
|   "intl-locale": { | ||||
|     android: 74, | ||||
|     chrome: 74, | ||||
|     edge: 79, | ||||
|     firefox: 75, | ||||
|     ios: 14.0, | ||||
|     opera: 62, | ||||
|     opera_mobile: 53, | ||||
|     safari: 14.0, | ||||
|     samsung: 11.0, | ||||
|   }, | ||||
|   "intl-other": { | ||||
|     // Not specified (i.e. always try polyfill) since compatibility depends on supported locales | ||||
|   }, | ||||
|   proxy: { | ||||
|     android: 49, | ||||
|     chrome: 49, | ||||
| @@ -98,67 +24,17 @@ const PolyfillSupport = { | ||||
|     safari: 10.0, | ||||
|     samsung: 5.0, | ||||
|   }, | ||||
|   "resize-observer": { | ||||
|     android: 64, | ||||
|     chrome: 64, | ||||
|     edge: 79, | ||||
|     firefox: 69, | ||||
|     ios: 13.4, | ||||
|     opera: 51, | ||||
|     opera_mobile: 47, | ||||
|     safari: 13.1, | ||||
|     samsung: 9.0, | ||||
|   }, | ||||
| }; | ||||
|  | ||||
| // Map of global variables and/or instance and static properties to the | ||||
| // corresponding polyfill key and actual module to import | ||||
| const polyfillMap = { | ||||
|   global: { | ||||
|     fetch: { key: "fetch", module: "unfetch/polyfill" }, | ||||
|     Proxy: { key: "proxy", module: "proxy-polyfill" }, | ||||
|     ResizeObserver: { | ||||
|       key: "resize-observer", | ||||
|       module: join(POLYFILL_DIR, "resize-observer.ts"), | ||||
|     }, | ||||
|   }, | ||||
|   instance: { | ||||
|     attachInternals: { | ||||
|       key: "element-internals", | ||||
|       module: "element-internals-polyfill", | ||||
|     }, | ||||
|     ...Object.fromEntries( | ||||
|       ["append", "getAttributeNames", "toggleAttribute"].map((prop) => { | ||||
|         const key = `element-${prop.toLowerCase()}`; | ||||
|         return [prop, { key, module: join(POLYFILL_DIR, `${key}.ts`) }]; | ||||
|       }) | ||||
|     ), | ||||
|   }, | ||||
|   static: { | ||||
|     Intl: { | ||||
|       getCanonicalLocales: { | ||||
|         key: "intl-getcanonicallocales", | ||||
|         module: join(POLYFILL_DIR, "intl-polyfill.ts"), | ||||
|       }, | ||||
|       Locale: { | ||||
|         key: "intl-locale", | ||||
|         module: join(POLYFILL_DIR, "intl-polyfill.ts"), | ||||
|       }, | ||||
|       ...Object.fromEntries( | ||||
|         [ | ||||
|           "DateTimeFormat", | ||||
|           "DisplayNames", | ||||
|           "ListFormat", | ||||
|           "NumberFormat", | ||||
|           "PluralRules", | ||||
|           "RelativeTimeFormat", | ||||
|         ].map((obj) => [ | ||||
|           obj, | ||||
|           { key: "intl-other", module: join(POLYFILL_DIR, "intl-polyfill.ts") }, | ||||
|         ]) | ||||
|       ), | ||||
|     }, | ||||
|     fetch: { key: "fetch", module: "unfetch/polyfill" }, | ||||
|   }, | ||||
|   instance: {}, | ||||
|   static: {}, | ||||
| }; | ||||
|  | ||||
| // Create plugin using the same factory as for CoreJS | ||||
| @@ -166,16 +42,14 @@ export default defineProvider( | ||||
|   ({ createMetaResolver, debug, shouldInjectPolyfill }) => { | ||||
|     const resolvePolyfill = createMetaResolver(polyfillMap); | ||||
|     return { | ||||
|       name: "custom-polyfill", | ||||
|       name: "HA Custom", | ||||
|       polyfills: PolyfillSupport, | ||||
|       usageGlobal(meta, utils) { | ||||
|         const polyfill = resolvePolyfill(meta); | ||||
|         if (polyfill && shouldInjectPolyfill(polyfill.desc.key)) { | ||||
|           debug(polyfill.desc.key); | ||||
|           utils.injectGlobalImport(polyfill.desc.module); | ||||
|           return true; | ||||
|         } | ||||
|         return false; | ||||
|       }, | ||||
|     }; | ||||
|   } | ||||
|   | ||||
| @@ -3,8 +3,6 @@ const env = require("./env.cjs"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const { dependencies } = require("../package.json"); | ||||
|  | ||||
| const BABEL_PLUGINS = path.join(__dirname, "babel-plugins"); | ||||
|  | ||||
| // GitHub base URL to use for production source maps | ||||
| // Nightly builds use the commit SHA, otherwise assumes there is a tag that matches the version | ||||
| module.exports.sourceMapURL = () => { | ||||
| @@ -47,7 +45,7 @@ module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) => | ||||
|  | ||||
| module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
|   __DEV__: !isProdBuild, | ||||
|   __BUILD__: JSON.stringify(latestBuild ? "modern" : "legacy"), | ||||
|   __BUILD__: JSON.stringify(latestBuild ? "latest" : "es5"), | ||||
|   __VERSION__: JSON.stringify(env.version()), | ||||
|   __DEMO__: false, | ||||
|   __SUPERVISOR__: false, | ||||
| @@ -79,12 +77,7 @@ module.exports.terserOptions = ({ latestBuild, isTestBuild }) => ({ | ||||
|   sourceMap: !isTestBuild, | ||||
| }); | ||||
|  | ||||
| module.exports.babelOptions = ({ | ||||
|   latestBuild, | ||||
|   isProdBuild, | ||||
|   isTestBuild, | ||||
|   sw, | ||||
| }) => ({ | ||||
| module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({ | ||||
|   babelrc: false, | ||||
|   compact: false, | ||||
|   assumptions: { | ||||
| @@ -92,13 +85,13 @@ module.exports.babelOptions = ({ | ||||
|     setPublicClassFields: true, | ||||
|     setSpreadProperties: true, | ||||
|   }, | ||||
|   browserslistEnv: latestBuild ? "modern" : `legacy${sw ? "-sw" : ""}`, | ||||
|   browserslistEnv: latestBuild ? "modern" : "legacy", | ||||
|   presets: [ | ||||
|     [ | ||||
|       "@babel/preset-env", | ||||
|       { | ||||
|         useBuiltIns: "usage", | ||||
|         corejs: dependencies["core-js"], | ||||
|         useBuiltIns: latestBuild ? false : "usage", | ||||
|         corejs: latestBuild ? false : dependencies["core-js"], | ||||
|         bugfixes: true, | ||||
|         shippedProposals: true, | ||||
|       }, | ||||
| @@ -107,12 +100,22 @@ module.exports.babelOptions = ({ | ||||
|   ], | ||||
|   plugins: [ | ||||
|     [ | ||||
|       path.join(BABEL_PLUGINS, "inline-constants-plugin.cjs"), | ||||
|       path.resolve( | ||||
|         paths.polymer_dir, | ||||
|         "build-scripts/babel-plugins/inline-constants-plugin.cjs" | ||||
|       ), | ||||
|       { | ||||
|         modules: ["@mdi/js"], | ||||
|         ignoreModuleNotFound: true, | ||||
|       }, | ||||
|     ], | ||||
|     [ | ||||
|       path.resolve( | ||||
|         paths.polymer_dir, | ||||
|         "build-scripts/babel-plugins/custom-polyfill-plugin.js" | ||||
|       ), | ||||
|       { method: "usage-global" }, | ||||
|     ], | ||||
|     // Minify template literals for production | ||||
|     isProdBuild && [ | ||||
|       "template-html-minifier", | ||||
| @@ -140,14 +143,8 @@ module.exports.babelOptions = ({ | ||||
|       "@babel/plugin-transform-runtime", | ||||
|       { version: dependencies["@babel/runtime"] }, | ||||
|     ], | ||||
|     // Transpile decorators (still in TC39 process) | ||||
|     // Modern browsers support class fields and private methods, but transform is required with the older decorator version dictated by Lit | ||||
|     [ | ||||
|       "@babel/plugin-proposal-decorators", | ||||
|       { version: "2018-09", decoratorsBeforeExport: true }, | ||||
|     ], | ||||
|     "@babel/plugin-transform-class-properties", | ||||
|     "@babel/plugin-transform-private-methods", | ||||
|     // Support  some proposals still in TC39 process | ||||
|     ["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }], | ||||
|   ].filter(Boolean), | ||||
|   exclude: [ | ||||
|     // \\ for Windows, / for Mac OS and Linux | ||||
| @@ -156,27 +153,6 @@ module.exports.babelOptions = ({ | ||||
|   ], | ||||
|   sourceMaps: !isTestBuild, | ||||
|   overrides: [ | ||||
|     { | ||||
|       // Add plugin to inject various polyfills, excluding the polyfills | ||||
|       // themselves to prevent self-injection. | ||||
|       plugins: [ | ||||
|         [ | ||||
|           path.join(BABEL_PLUGINS, "custom-polyfill-plugin.js"), | ||||
|           { method: "usage-global" }, | ||||
|         ], | ||||
|       ], | ||||
|       exclude: [ | ||||
|         path.join(paths.polymer_dir, "src/resources/polyfills"), | ||||
|         ...[ | ||||
|           "@formatjs/(?:ecma402-abstract|intl-\\w+)", | ||||
|           "@lit-labs/virtualizer/polyfills", | ||||
|           "@webcomponents/scoped-custom-element-registry", | ||||
|           "element-internals-polyfill", | ||||
|           "proxy-polyfill", | ||||
|           "unfetch", | ||||
|         ].map((p) => new RegExp(`/node_modules/${p}/`)), | ||||
|       ], | ||||
|     }, | ||||
|     { | ||||
|       // Use unambiguous for dependencies so that require() is correctly injected into CommonJS files | ||||
|       // Exclusions are needed in some cases where ES modules have no static imports or exports, such as polyfills | ||||
| @@ -226,13 +202,7 @@ module.exports.config = { | ||||
|     return { | ||||
|       name: "frontend" + nameSuffix(latestBuild), | ||||
|       entry: { | ||||
|         "service-worker": | ||||
|           !env.useRollup() && !latestBuild | ||||
|             ? { | ||||
|                 import: "./src/entrypoints/service-worker.ts", | ||||
|                 layer: "sw", | ||||
|               } | ||||
|             : "./src/entrypoints/service-worker.ts", | ||||
|         service_worker: "./src/entrypoints/service_worker.ts", | ||||
|         app: "./src/entrypoints/app.ts", | ||||
|         authorize: "./src/entrypoints/authorize.ts", | ||||
|         onboarding: "./src/entrypoints/onboarding.ts", | ||||
|   | ||||
| @@ -32,7 +32,4 @@ module.exports = { | ||||
|     } | ||||
|     return version[1]; | ||||
|   }, | ||||
|   isDevContainer() { | ||||
|     return process.env.DEV_CONTAINER === "1"; | ||||
|   }, | ||||
| }; | ||||
|   | ||||
| @@ -1,68 +1,26 @@ | ||||
| // Tasks to compress | ||||
|  | ||||
| import { constants } from "node:zlib"; | ||||
| import { deleteAsync } from "del"; | ||||
| import gulp from "gulp"; | ||||
| import brotli from "gulp-brotli"; | ||||
| import gulpIf from "gulp-if"; | ||||
| import vinylPaths from "vinyl-paths"; | ||||
| 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 compressDistBrotli = (rootDir, modernDir, compressServiceWorker = true) => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         `${modernDir}/**/${filesGlob}`, | ||||
|         compressServiceWorker ? `${rootDir}/sw-modern.js` : undefined, | ||||
|       ].filter(Boolean), | ||||
|       { | ||||
|         base: rootDir, | ||||
|       } | ||||
|     ) | ||||
|     .pipe(brotli(brotliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
| const compressedExt = /\.gz$/; | ||||
| const deleteUncompressed = (p) => deleteAsync(p.replace(compressedExt, "")); | ||||
|  | ||||
| const compressDistZopfli = (rootDir, modernDir, compressModern = false) => | ||||
| const compressDist = (rootDir) => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         `${rootDir}/**/${filesGlob}`, | ||||
|         compressModern ? undefined : `!${modernDir}/**/${filesGlob}`, | ||||
|         `!${rootDir}/{sw-modern,service_worker}.js`, | ||||
|         `${rootDir}/{authorize,onboarding}.html`, | ||||
|       ].filter(Boolean), | ||||
|       { base: rootDir } | ||||
|     ) | ||||
|     .src([ | ||||
|       `${rootDir}/**/*.{js?(.map),json,css,svg,xml}`, | ||||
|       `${rootDir}/{authorize,onboarding}.html`, | ||||
|     ]) | ||||
|     .pipe(zopfli(zopfliOptions)) | ||||
|     .pipe(gulp.dest(rootDir)); | ||||
|     .pipe(gulp.dest(rootDir)) | ||||
|     .pipe(gulpIf(compressedExt, vinylPaths(deleteUncompressed))); | ||||
|  | ||||
| const compressAppBrotli = () => | ||||
|   compressDistBrotli(paths.app_output_root, paths.app_output_latest); | ||||
| const compressHassioBrotli = () => | ||||
|   compressDistBrotli( | ||||
|     paths.hassio_output_root, | ||||
|     paths.hassio_output_latest, | ||||
|     false | ||||
|   ); | ||||
|  | ||||
| const compressAppZopfli = () => | ||||
|   compressDistZopfli(paths.app_output_root, paths.app_output_latest); | ||||
| const compressHassioZopfli = () => | ||||
|   compressDistZopfli( | ||||
|     paths.hassio_output_root, | ||||
|     paths.hassio_output_latest, | ||||
|     true | ||||
|   ); | ||||
|  | ||||
| gulp.task("compress-app", gulp.parallel(compressAppBrotli, compressAppZopfli)); | ||||
| gulp.task( | ||||
|   "compress-hassio", | ||||
|   gulp.parallel(compressHassioBrotli, compressHassioZopfli) | ||||
| ); | ||||
| gulp.task("compress-app", () => compressDist(paths.app_output_root)); | ||||
| gulp.task("compress-hassio", () => compressDist(paths.hassio_output_root)); | ||||
|   | ||||
| @@ -1,76 +1,28 @@ | ||||
| // 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 { dirname, extname, resolve } from "node:path"; | ||||
| import path from "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(resolve(dirname(templateFile), childTemplate), data), | ||||
|       renderTemplate( | ||||
|         path.resolve(path.dirname(templateFile), childTemplate), | ||||
|         data | ||||
|       ), | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| @@ -104,12 +56,10 @@ const genPagesDevTask = | ||||
|     publicRoot = "" | ||||
|   ) => | ||||
|   async () => { | ||||
|     const commonVars = getCommonTemplateVars(); | ||||
|     for (const [page, entries] of Object.entries(pageEntries)) { | ||||
|       const content = renderTemplate( | ||||
|         resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         path.resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         { | ||||
|           ...commonVars, | ||||
|           latestEntryJS: entries.map((entry) => | ||||
|             useWDS | ||||
|               ? `http://localhost:8000/src/entrypoints/${entry}.ts` | ||||
| @@ -124,7 +74,7 @@ const genPagesDevTask = | ||||
|           es5CustomPanelJS: `${publicRoot}/frontend_es5/custom-panel.js`, | ||||
|         } | ||||
|       ); | ||||
|       fs.outputFileSync(resolve(outputRoot, page), content); | ||||
|       fs.outputFileSync(path.resolve(outputRoot, page), content); | ||||
|     } | ||||
|   }; | ||||
|  | ||||
| @@ -141,18 +91,16 @@ const genPagesProdTask = | ||||
|   ) => | ||||
|   async () => { | ||||
|     const latestManifest = fs.readJsonSync( | ||||
|       resolve(outputLatest, "manifest.json") | ||||
|       path.resolve(outputLatest, "manifest.json") | ||||
|     ); | ||||
|     const es5Manifest = outputES5 | ||||
|       ? fs.readJsonSync(resolve(outputES5, "manifest.json")) | ||||
|       ? fs.readJsonSync(path.resolve(outputES5, "manifest.json")) | ||||
|       : {}; | ||||
|     const commonVars = getCommonTemplateVars(); | ||||
|     const minifiedHTML = []; | ||||
|     for (const [page, entries] of Object.entries(pageEntries)) { | ||||
|       const content = renderTemplate( | ||||
|         resolve(inputRoot, inputSub, `${page}.template`), | ||||
|         path.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"], | ||||
| @@ -160,8 +108,8 @@ const genPagesProdTask = | ||||
|         } | ||||
|       ); | ||||
|       minifiedHTML.push( | ||||
|         minifyHtml(content, extname(page)).then((minified) => | ||||
|           fs.outputFileSync(resolve(outputRoot, page), minified) | ||||
|         minifyHtml(content, path.extname(page)).then((minified) => | ||||
|           fs.outputFileSync(path.resolve(outputRoot, page), minified) | ||||
|         ) | ||||
|       ); | ||||
|     } | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import gulp from "gulp"; | ||||
| import jszip from "jszip"; | ||||
| import path from "path"; | ||||
| import process from "process"; | ||||
| import { extract } from "tar"; | ||||
| import tar from "tar"; | ||||
|  | ||||
| const MAX_AGE = 24; // hours | ||||
| const OWNER = "home-assistant"; | ||||
| @@ -156,7 +156,7 @@ gulp.task("fetch-nightly-translations", async function () { | ||||
|   console.log("Unpacking downloaded translations..."); | ||||
|   const zip = await jszip.loadAsync(downloadResponse.data); | ||||
|   await deleteCurrent; | ||||
|   const extractStream = zip.file(/.*/)[0].nodeStream().pipe(extract()); | ||||
|   const extractStream = zip.file(/.*/)[0].nodeStream().pipe(tar.extract()); | ||||
|   await new Promise((resolve, reject) => { | ||||
|     extractStream.on("close", resolve).on("error", reject); | ||||
|   }); | ||||
|   | ||||
| @@ -1,19 +1,20 @@ | ||||
| // Generate service workers | ||||
| // Generate service worker. | ||||
| // Based on manifest, create a file with the content as service_worker.js | ||||
|  | ||||
| import { deleteAsync } from "del"; | ||||
| import fs from "fs-extra"; | ||||
| import gulp from "gulp"; | ||||
| import { mkdir, readFile, symlink, writeFile } from "node:fs/promises"; | ||||
| import { basename, join, relative } from "node:path"; | ||||
| import { injectManifest } from "workbox-build"; | ||||
| import path from "path"; | ||||
| import sourceMapUrl from "source-map-url"; | ||||
| import workboxBuild from "workbox-build"; | ||||
| import paths from "../paths.cjs"; | ||||
|  | ||||
| const SW_MAP = { | ||||
|   [paths.app_output_latest]: "modern", | ||||
|   [paths.app_output_es5]: "legacy", | ||||
| }; | ||||
| const swDest = path.resolve(paths.app_output_root, "service_worker.js"); | ||||
|  | ||||
| const SW_DEV = | ||||
|   ` | ||||
| const writeSW = (content) => fs.outputFileSync(swDest, content.trim() + "\n"); | ||||
|  | ||||
| gulp.task("gen-service-worker-app-dev", (done) => { | ||||
|   writeSW( | ||||
|     ` | ||||
| console.debug('Service worker disabled in development'); | ||||
|  | ||||
| self.addEventListener('install', (event) => { | ||||
| @@ -21,67 +22,72 @@ self.addEventListener('install', (event) => { | ||||
|   // removing any prod service worker the dev might have running | ||||
|   self.skipWaiting(); | ||||
| }); | ||||
|   `.trim() + "\n"; | ||||
|  | ||||
| gulp.task("gen-service-worker-app-dev", async () => { | ||||
|   await mkdir(paths.app_output_root, { recursive: true }); | ||||
|   await Promise.all( | ||||
|     Object.values(SW_MAP).map((build) => | ||||
|       writeFile(join(paths.app_output_root, `sw-${build}.js`), SW_DEV, { | ||||
|         encoding: "utf-8", | ||||
|       }) | ||||
|     ) | ||||
|   ` | ||||
|   ); | ||||
|   done(); | ||||
| }); | ||||
|  | ||||
| gulp.task("gen-service-worker-app-prod", () => | ||||
|   Promise.all( | ||||
|     Object.entries(SW_MAP).map(async ([outPath, build]) => { | ||||
|       const manifest = JSON.parse( | ||||
|         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, | ||||
|         injectionPoint: "__WB_MANIFEST__", | ||||
|         // Files that mach this pattern will be considered unique and skip revision check | ||||
|         // ignore JS files + translation files | ||||
|         dontCacheBustURLsMatching: new RegExp( | ||||
|           `(?:${buildDir}/.+|static/translations/.+)` | ||||
|         ), | ||||
|         globDirectory: paths.app_output_root, | ||||
|         globPatterns: [ | ||||
|           `${buildDir}/*.js`, | ||||
|           // Cache all English translations because we catch them as fallback | ||||
|           // Using pattern to match hash instead of * to avoid caching en-GB | ||||
|           // 'v' added as valid hash letter because in dev we hash with 'dev' | ||||
|           "static/translations/**/en-+([a-fv0-9]).json", | ||||
|           // Icon shown on splash screen | ||||
|           "static/icons/favicon-192x192.png", | ||||
|           "static/icons/favicon.ico", | ||||
|           // Common fonts | ||||
|           "static/fonts/roboto/Roboto-Light.woff2", | ||||
|           "static/fonts/roboto/Roboto-Medium.woff2", | ||||
|           "static/fonts/roboto/Roboto-Regular.woff2", | ||||
|           "static/fonts/roboto/Roboto-Bold.woff2", | ||||
|         ], | ||||
|         globIgnores: [`${buildDir}/service-worker*`], | ||||
|       }); | ||||
|       if (warnings.length > 0) { | ||||
|         console.warn( | ||||
|           `Problems while injecting ${build} service worker:\n`, | ||||
|           warnings.join("\n") | ||||
|         ); | ||||
|       } | ||||
|       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); | ||||
|       } | ||||
|     }) | ||||
|   ) | ||||
| ); | ||||
| gulp.task("gen-service-worker-app-prod", async () => { | ||||
|   // Read bundled source file | ||||
|   const bundleManifestLatest = fs.readJsonSync( | ||||
|     path.resolve(paths.app_output_latest, "manifest.json") | ||||
|   ); | ||||
|   let serviceWorkerContent = fs.readFileSync( | ||||
|     paths.app_output_root + bundleManifestLatest["service_worker.js"], | ||||
|     "utf-8" | ||||
|   ); | ||||
|  | ||||
|   // Delete old file from frontend_latest so manifest won't pick it up | ||||
|   fs.removeSync( | ||||
|     paths.app_output_root + bundleManifestLatest["service_worker.js"] | ||||
|   ); | ||||
|   fs.removeSync( | ||||
|     paths.app_output_root + bundleManifestLatest["service_worker.js.map"] | ||||
|   ); | ||||
|  | ||||
|   // Remove ES5 | ||||
|   const bundleManifestES5 = fs.readJsonSync( | ||||
|     path.resolve(paths.app_output_es5, "manifest.json") | ||||
|   ); | ||||
|   fs.removeSync(paths.app_output_root + bundleManifestES5["service_worker.js"]); | ||||
|   fs.removeSync( | ||||
|     paths.app_output_root + bundleManifestES5["service_worker.js.map"] | ||||
|   ); | ||||
|  | ||||
|   const workboxManifest = await workboxBuild.getManifest({ | ||||
|     // Files that mach this pattern will be considered unique and skip revision check | ||||
|     // ignore JS files + translation files | ||||
|     dontCacheBustURLsMatching: /(frontend_latest\/.+|static\/translations\/.+)/, | ||||
|  | ||||
|     globDirectory: paths.app_output_root, | ||||
|     globPatterns: [ | ||||
|       "frontend_latest/*.js", | ||||
|       // Cache all English translations because we catch them as fallback | ||||
|       // Using pattern to match hash instead of * to avoid caching en-GB | ||||
|       // 'v' added as valid hash letter because in dev we hash with 'dev' | ||||
|       "static/translations/**/en-+([a-fv0-9]).json", | ||||
|       // Icon shown on splash screen | ||||
|       "static/icons/favicon-192x192.png", | ||||
|       "static/icons/favicon.ico", | ||||
|       // Common fonts | ||||
|       "static/fonts/roboto/Roboto-Light.woff2", | ||||
|       "static/fonts/roboto/Roboto-Medium.woff2", | ||||
|       "static/fonts/roboto/Roboto-Regular.woff2", | ||||
|       "static/fonts/roboto/Roboto-Bold.woff2", | ||||
|     ], | ||||
|   }); | ||||
|  | ||||
|   for (const warning of workboxManifest.warnings) { | ||||
|     console.warn(warning); | ||||
|   } | ||||
|  | ||||
|   // remove source map and add WB manifest | ||||
|   serviceWorkerContent = sourceMapUrl.removeFrom(serviceWorkerContent); | ||||
|   serviceWorkerContent = serviceWorkerContent.replace( | ||||
|     "WB_MANIFEST", | ||||
|     JSON.stringify(workboxManifest.manifestEntries) | ||||
|   ); | ||||
|  | ||||
|   // Write new file to root | ||||
|   fs.writeFileSync(swDest, serviceWorkerContent); | ||||
| }); | ||||
|   | ||||
| @@ -1,112 +1,92 @@ | ||||
| /* eslint-disable max-classes-per-file */ | ||||
|  | ||||
| import { deleteAsync } from "del"; | ||||
| import { glob } from "glob"; | ||||
| import { createHash } from "crypto"; | ||||
| import { deleteSync } from "del"; | ||||
| import { mkdirSync, readdirSync, readFileSync, renameSync } from "fs"; | ||||
| import { writeFile } from "node:fs/promises"; | ||||
| import gulp from "gulp"; | ||||
| import flatmap from "gulp-flatmap"; | ||||
| import transform from "gulp-json-transform"; | ||||
| import merge from "gulp-merge-json"; | ||||
| import rename from "gulp-rename"; | ||||
| import merge from "lodash.merge"; | ||||
| import { createHash } from "node:crypto"; | ||||
| import { mkdir, readFile } from "node:fs/promises"; | ||||
| import { basename, join } from "node:path"; | ||||
| import { PassThrough, Transform } from "node:stream"; | ||||
| import { finished } from "node:stream/promises"; | ||||
| import path from "path"; | ||||
| import vinylBuffer from "vinyl-buffer"; | ||||
| import source from "vinyl-source-stream"; | ||||
| import env from "../env.cjs"; | ||||
| import paths from "../paths.cjs"; | ||||
| import { mapFiles } from "../util.cjs"; | ||||
| import "./fetch-nightly-translations.js"; | ||||
|  | ||||
| const inFrontendDir = "translations/frontend"; | ||||
| const inBackendDir = "translations/backend"; | ||||
| const workDir = "build/translations"; | ||||
| const outDir = join(workDir, "output"); | ||||
| const EN_SRC = join(paths.translations_src, "en.json"); | ||||
| const TEST_LOCALE = "en-x-test"; | ||||
|  | ||||
| const fullDir = workDir + "/full"; | ||||
| const coreDir = workDir + "/core"; | ||||
| const outDir = workDir + "/output"; | ||||
| let mergeBackend = false; | ||||
|  | ||||
| gulp.task( | ||||
|   "translations-enable-merge-backend", | ||||
|   gulp.parallel(async () => { | ||||
|   gulp.parallel((done) => { | ||||
|     mergeBackend = true; | ||||
|     done(); | ||||
|   }, "allow-setup-fetch-nightly-translations") | ||||
| ); | ||||
|  | ||||
| // Transform stream to apply a function on Vinyl JSON files (buffer mode only). | ||||
| // The provided function can either return a new object, or an array of | ||||
| // [object, subdirectory] pairs for fragmentizing the JSON. | ||||
| class CustomJSON extends Transform { | ||||
|   constructor(func, reviver = null) { | ||||
|     super({ objectMode: true }); | ||||
|     this._func = func; | ||||
|     this._reviver = reviver; | ||||
|   } | ||||
| // Panel translations which should be split from the core translations. | ||||
| const TRANSLATION_FRAGMENTS = Object.keys( | ||||
|   JSON.parse( | ||||
|     readFileSync( | ||||
|       path.resolve(paths.polymer_dir, "src/translations/en.json"), | ||||
|       "utf-8" | ||||
|     ) | ||||
|   ).ui.panel | ||||
| ); | ||||
|  | ||||
|   async _transform(file, _, callback) { | ||||
|     try { | ||||
|       let obj = JSON.parse(file.contents.toString(), this._reviver); | ||||
|       if (this._func) obj = this._func(obj, file.path); | ||||
|       for (const [outObj, dir] of Array.isArray(obj) ? obj : [[obj, ""]]) { | ||||
|         const outFile = file.clone({ contents: false }); | ||||
|         outFile.contents = Buffer.from(JSON.stringify(outObj)); | ||||
|         outFile.dirname += `/${dir}`; | ||||
|         this.push(outFile); | ||||
|       } | ||||
|       callback(null); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Transform stream to merge Vinyl JSON files (buffer mode only). | ||||
| class MergeJSON extends Transform { | ||||
|   _objects = []; | ||||
|  | ||||
|   constructor(stem, startObj = {}, reviver = null) { | ||||
|     super({ objectMode: true, allowHalfOpen: false }); | ||||
|     this._stem = stem; | ||||
|     this._startObj = structuredClone(startObj); | ||||
|     this._reviver = reviver; | ||||
|   } | ||||
|  | ||||
|   async _transform(file, _, callback) { | ||||
|     try { | ||||
|       this._objects.push(JSON.parse(file.contents.toString(), this._reviver)); | ||||
|       if (!this._outFile) this._outFile = file.clone({ contents: false }); | ||||
|       callback(null); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   async _flush(callback) { | ||||
|     try { | ||||
|       const mergedObj = merge(this._startObj, ...this._objects); | ||||
|       this._outFile.contents = Buffer.from(JSON.stringify(mergedObj)); | ||||
|       this._outFile.stem = this._stem; | ||||
|       callback(null, this._outFile); | ||||
|     } catch (err) { | ||||
|       callback(err); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // Utility to flatten object keys to single level using separator | ||||
| const flatten = (data, prefix = "", sep = ".") => { | ||||
|   const output = {}; | ||||
|   for (const [key, value] of Object.entries(data)) { | ||||
|     if (typeof value === "object") { | ||||
|       Object.assign(output, flatten(value, prefix + key + sep, sep)); | ||||
| function recursiveFlatten(prefix, data) { | ||||
|   let output = {}; | ||||
|   Object.keys(data).forEach((key) => { | ||||
|     if (typeof data[key] === "object") { | ||||
|       output = { | ||||
|         ...output, | ||||
|         ...recursiveFlatten(prefix + key + ".", data[key]), | ||||
|       }; | ||||
|     } else { | ||||
|       output[prefix + key] = value; | ||||
|       output[prefix + key] = data[key]; | ||||
|     } | ||||
|   } | ||||
|   }); | ||||
|   return output; | ||||
| }; | ||||
| } | ||||
|  | ||||
| // Filter functions that can be passed directly to JSON.parse() | ||||
| const emptyReviver = (_key, value) => value || undefined; | ||||
| const testReviver = (_key, value) => | ||||
|   value && typeof value === "string" ? "TRANSLATED" : value; | ||||
| function flatten(data) { | ||||
|   return recursiveFlatten("", data); | ||||
| } | ||||
|  | ||||
| function emptyFilter(data) { | ||||
|   const newData = {}; | ||||
|   Object.keys(data).forEach((key) => { | ||||
|     if (data[key]) { | ||||
|       if (typeof data[key] === "object") { | ||||
|         newData[key] = emptyFilter(data[key]); | ||||
|       } else { | ||||
|         newData[key] = data[key]; | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|   return newData; | ||||
| } | ||||
|  | ||||
| function recursiveEmpty(data) { | ||||
|   const newData = {}; | ||||
|   Object.keys(data).forEach((key) => { | ||||
|     if (data[key]) { | ||||
|       if (typeof data[key] === "object") { | ||||
|         newData[key] = recursiveEmpty(data[key]); | ||||
|       } else { | ||||
|         newData[key] = "TRANSLATED"; | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|   return newData; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Replace Lokalise key placeholders with their actual values. | ||||
| @@ -115,44 +95,60 @@ const testReviver = (_key, value) => | ||||
|  * be included in src/translations/en.json, but still be usable while | ||||
|  * developing locally. | ||||
|  * | ||||
|  * @link https://docs.lokalise.com/en/articles/1400528-key-referencing | ||||
|  * @link https://docs.lokalise.co/article/KO5SZWLLsy-key-referencing | ||||
|  */ | ||||
| const KEY_REFERENCE = /\[%key:([^%]+)%\]/; | ||||
| const lokaliseTransform = (data, path, original = data) => { | ||||
| const re_key_reference = /\[%key:([^%]+)%\]/; | ||||
| function lokaliseTransform(data, original, file) { | ||||
|   const output = {}; | ||||
|   for (const [key, value] of Object.entries(data)) { | ||||
|     if (typeof value === "object") { | ||||
|       output[key] = lokaliseTransform(value, path, original); | ||||
|   Object.entries(data).forEach(([key, value]) => { | ||||
|     if (value instanceof Object) { | ||||
|       output[key] = lokaliseTransform(value, original, file); | ||||
|     } else { | ||||
|       output[key] = value.replace(KEY_REFERENCE, (_match, lokalise_key) => { | ||||
|       output[key] = value.replace(re_key_reference, (_match, lokalise_key) => { | ||||
|         const replace = lokalise_key.split("::").reduce((tr, k) => { | ||||
|           if (!tr) { | ||||
|             throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`); | ||||
|             throw Error( | ||||
|               `Invalid key placeholder ${lokalise_key} in ${file.path}` | ||||
|             ); | ||||
|           } | ||||
|           return tr[k]; | ||||
|         }, original); | ||||
|         if (typeof replace !== "string") { | ||||
|           throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`); | ||||
|           throw Error( | ||||
|             `Invalid key placeholder ${lokalise_key} in ${file.path}` | ||||
|           ); | ||||
|         } | ||||
|         return replace; | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
|   }); | ||||
|   return output; | ||||
| }; | ||||
| } | ||||
|  | ||||
| gulp.task("clean-translations", () => deleteAsync([workDir])); | ||||
| gulp.task("clean-translations", async () => deleteSync([workDir])); | ||||
|  | ||||
| const makeWorkDir = () => mkdir(workDir, { recursive: true }); | ||||
| gulp.task("ensure-translations-build-dir", async () => { | ||||
|   mkdirSync(workDir, { recursive: true }); | ||||
| }); | ||||
|  | ||||
| const createTestTranslation = () => | ||||
| gulp.task("create-test-metadata", () => | ||||
|   env.isProdBuild() | ||||
|     ? Promise.resolve() | ||||
|     : writeFile( | ||||
|         workDir + "/testMetadata.json", | ||||
|         JSON.stringify({ test: { nativeName: "Test" } }) | ||||
|       ) | ||||
| ); | ||||
|  | ||||
| gulp.task("create-test-translation", () => | ||||
|   env.isProdBuild() | ||||
|     ? Promise.resolve() | ||||
|     : gulp | ||||
|         .src(EN_SRC) | ||||
|         .pipe(new CustomJSON(null, testReviver)) | ||||
|         .pipe(rename(`${TEST_LOCALE}.json`)) | ||||
|         .pipe(gulp.dest(workDir)); | ||||
|         .src(path.join(paths.translations_src, "en.json")) | ||||
|         .pipe(transform((data, _file) => recursiveEmpty(data))) | ||||
|         .pipe(rename("test.json")) | ||||
|         .pipe(gulp.dest(workDir)) | ||||
| ); | ||||
|  | ||||
| /** | ||||
|  * This task will build a master translation file, to be used as the base for | ||||
| @@ -163,164 +159,279 @@ const createTestTranslation = () => | ||||
|  * project is buildable immediately after merging new translation keys, since | ||||
|  * the Lokalise update to translations/en.json will not happen immediately. | ||||
|  */ | ||||
| const createMasterTranslation = () => | ||||
|   gulp | ||||
|     .src([EN_SRC, ...(mergeBackend ? [`${inBackendDir}/en.json`] : [])]) | ||||
|     .pipe(new CustomJSON(lokaliseTransform)) | ||||
|     .pipe(new MergeJSON("en")) | ||||
|     .pipe(gulp.dest(workDir)); | ||||
| gulp.task("build-master-translation", () => { | ||||
|   const src = [path.join(paths.translations_src, "en.json")]; | ||||
|  | ||||
| const FRAGMENTS = ["base"]; | ||||
|  | ||||
| const toggleSupervisorFragment = async () => { | ||||
|   FRAGMENTS[0] = "supervisor"; | ||||
| }; | ||||
|  | ||||
| const panelFragment = (fragment) => | ||||
|   fragment !== "base" && fragment !== "supervisor"; | ||||
|  | ||||
| const HASHES = new Map(); | ||||
|  | ||||
| const createTranslations = async () => { | ||||
|   // Parse and store the master to avoid repeating this for each locale, then | ||||
|   // add the panel fragments when processing the app. | ||||
|   const enMaster = JSON.parse(await readFile(`${workDir}/en.json`, "utf-8")); | ||||
|   if (FRAGMENTS[0] === "base") { | ||||
|     FRAGMENTS.push(...Object.keys(enMaster.ui.panel)); | ||||
|   if (mergeBackend) { | ||||
|     src.push(path.join(inBackendDir, "en.json")); | ||||
|   } | ||||
|  | ||||
|   // The downstream pipeline is setup first.  It hashes the merged data for | ||||
|   // each locale, then fragmentizes and flattens the data for final output. | ||||
|   const translationFiles = await glob([ | ||||
|     `${inFrontendDir}/!(en).json`, | ||||
|     ...(env.isProdBuild() ? [] : [`${workDir}/${TEST_LOCALE}.json`]), | ||||
|   ]); | ||||
|   const hashStream = new Transform({ | ||||
|     objectMode: true, | ||||
|     transform: async (file, _, callback) => { | ||||
|       const hash = env.isProdBuild() | ||||
|         ? createHash("md5").update(file.contents).digest("hex") | ||||
|         : "dev"; | ||||
|       HASHES.set(file.stem, hash); | ||||
|       file.stem += `-${hash}`; | ||||
|       callback(null, file); | ||||
|     }, | ||||
|   }).setMaxListeners(translationFiles.length + 1); | ||||
|   const fragmentsStream = hashStream | ||||
|   return gulp | ||||
|     .src(src) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       new CustomJSON((data) => | ||||
|         FRAGMENTS.map((fragment) => { | ||||
|           switch (fragment) { | ||||
|             case "base": | ||||
|               // Remove the panels and supervisor to create the base translations | ||||
|               return [ | ||||
|                 flatten({ | ||||
|                   ...data, | ||||
|                   ui: { ...data.ui, panel: undefined }, | ||||
|                   supervisor: undefined, | ||||
|                 }), | ||||
|                 "", | ||||
|               ]; | ||||
|             case "supervisor": | ||||
|               // Supervisor key is at the top level | ||||
|               return [flatten(data.supervisor), ""]; | ||||
|             default: | ||||
|               // Create a fragment with only the given panel | ||||
|               return [ | ||||
|                 flatten(data.ui.panel[fragment], `ui.panel.${fragment}.`), | ||||
|                 fragment, | ||||
|               ]; | ||||
|           } | ||||
|         }) | ||||
|       ) | ||||
|     ) | ||||
|     .pipe(gulp.dest(outDir)); | ||||
|  | ||||
|   // Send the English master downstream first, then for each other locale | ||||
|   // generate merged JSON data to continue piping. It begins with the master | ||||
|   // translation as a failsafe for untranslated strings, and merges all parent | ||||
|   // tags into one file for each specific subtag | ||||
|   // | ||||
|   // TODO: This is a naive interpretation of BCP47 that should be improved. | ||||
|   //       Will be OK for now as long as we don't have anything more complicated | ||||
|   // than a base translation + region. | ||||
|   const masterStream = gulp | ||||
|     .src(`${workDir}/en.json`) | ||||
|     .pipe(new PassThrough({ objectMode: true })); | ||||
|   masterStream.pipe(hashStream, { end: false }); | ||||
|   const mergesFinished = [finished(masterStream)]; | ||||
|   for (const translationFile of translationFiles) { | ||||
|     const locale = basename(translationFile, ".json"); | ||||
|     const subtags = locale.split("-"); | ||||
|     const mergeFiles = []; | ||||
|     for (let i = 1; i <= subtags.length; i++) { | ||||
|       const lang = subtags.slice(0, i).join("-"); | ||||
|       if (lang === TEST_LOCALE) { | ||||
|         mergeFiles.push(`${workDir}/${TEST_LOCALE}.json`); | ||||
|       } else if (lang !== "en") { | ||||
|         mergeFiles.push(`${inFrontendDir}/${lang}.json`); | ||||
|         if (mergeBackend) { | ||||
|           mergeFiles.push(`${inBackendDir}/${lang}.json`); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     const mergeStream = gulp | ||||
|       .src(mergeFiles, { allowEmpty: true }) | ||||
|       .pipe(new MergeJSON(locale, enMaster, emptyReviver)); | ||||
|     mergesFinished.push(finished(mergeStream)); | ||||
|     mergeStream.pipe(hashStream, { end: false }); | ||||
|   } | ||||
|  | ||||
|   // Wait for all merges to finish, then it's safe to end writing to the | ||||
|   // downstream pipeline and wait for all fragments to finish writing. | ||||
|   await Promise.all(mergesFinished); | ||||
|   hashStream.end(); | ||||
|   await finished(fragmentsStream); | ||||
| }; | ||||
|  | ||||
| const writeTranslationMetaData = () => | ||||
|   gulp | ||||
|     .src([`${paths.translations_src}/translationMetadata.json`]) | ||||
|     .pipe( | ||||
|       new CustomJSON((meta) => { | ||||
|         // Add the test translation in development. | ||||
|         if (!env.isProdBuild()) { | ||||
|           meta[TEST_LOCALE] = { nativeName: "Translation Test" }; | ||||
|         } | ||||
|         // Filter out locales without a native name, and add the hashes. | ||||
|         for (const locale of Object.keys(meta)) { | ||||
|           if (!meta[locale].nativeName) { | ||||
|             meta[locale] = undefined; | ||||
|             console.warn( | ||||
|               `Skipping locale ${locale} because native name is not translated.` | ||||
|             ); | ||||
|           } else { | ||||
|             meta[locale].hash = HASHES.get(locale); | ||||
|           } | ||||
|         } | ||||
|         return { | ||||
|           fragments: FRAGMENTS.filter(panelFragment), | ||||
|           translations: meta, | ||||
|         }; | ||||
|       merge({ | ||||
|         fileName: "en.json", | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(workDir)); | ||||
|     .pipe(gulp.dest(fullDir)); | ||||
| }); | ||||
|  | ||||
| gulp.task("build-merged-translations", () => | ||||
|   gulp | ||||
|     .src([ | ||||
|       inFrontendDir + "/*.json", | ||||
|       "!" + inFrontendDir + "/en.json", | ||||
|       ...(env.isProdBuild() ? [] : [workDir + "/test.json"]), | ||||
|     ]) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       flatmap((stream, file) => { | ||||
|         // For each language generate a merged json file. It begins with the master | ||||
|         // translation as a failsafe for untranslated strings, and merges all parent | ||||
|         // tags into one file for each specific subtag | ||||
|         // | ||||
|         // TODO: This is a naive interpretation of BCP47 that should be improved. | ||||
|         //       Will be OK for now as long as we don't have anything more complicated | ||||
|         //       than a base translation + region. | ||||
|         const tr = path.basename(file.history[0], ".json"); | ||||
|         const subtags = tr.split("-"); | ||||
|         const src = [fullDir + "/en.json"]; | ||||
|         for (let i = 1; i <= subtags.length; i++) { | ||||
|           const lang = subtags.slice(0, i).join("-"); | ||||
|           if (lang === "test") { | ||||
|             src.push(workDir + "/test.json"); | ||||
|           } else if (lang !== "en") { | ||||
|             src.push(inFrontendDir + "/" + lang + ".json"); | ||||
|             if (mergeBackend) { | ||||
|               src.push(inBackendDir + "/" + lang + ".json"); | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|         return gulp | ||||
|           .src(src, { allowEmpty: true }) | ||||
|           .pipe(transform((data) => emptyFilter(data))) | ||||
|           .pipe( | ||||
|             merge({ | ||||
|               fileName: tr + ".json", | ||||
|             }) | ||||
|           ) | ||||
|           .pipe(gulp.dest(fullDir)); | ||||
|       }) | ||||
|     ) | ||||
| ); | ||||
|  | ||||
| let taskName; | ||||
|  | ||||
| const splitTasks = []; | ||||
| TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||
|   taskName = "build-translation-fragment-" + fragment; | ||||
|   gulp.task(taskName, () => | ||||
|     // Return only the translations for this fragment. | ||||
|     gulp | ||||
|       .src(fullDir + "/*.json") | ||||
|       .pipe( | ||||
|         transform((data) => ({ | ||||
|           ui: { | ||||
|             panel: { | ||||
|               [fragment]: data.ui.panel[fragment], | ||||
|             }, | ||||
|           }, | ||||
|         })) | ||||
|       ) | ||||
|       .pipe(gulp.dest(workDir + "/" + fragment)) | ||||
|   ); | ||||
|   splitTasks.push(taskName); | ||||
| }); | ||||
|  | ||||
| taskName = "build-translation-core"; | ||||
| gulp.task(taskName, () => | ||||
|   // Remove the fragment translations from the core translation. | ||||
|   gulp | ||||
|     .src(fullDir + "/*.json") | ||||
|     .pipe( | ||||
|       transform((data, _file) => { | ||||
|         TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||
|           delete data.ui.panel[fragment]; | ||||
|         }); | ||||
|         delete data.supervisor; | ||||
|         return data; | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(coreDir)) | ||||
| ); | ||||
|  | ||||
| splitTasks.push(taskName); | ||||
|  | ||||
| gulp.task("build-flattened-translations", () => | ||||
|   // Flatten the split versions of our translations, and move them into outDir | ||||
|   gulp | ||||
|     .src( | ||||
|       TRANSLATION_FRAGMENTS.map( | ||||
|         (fragment) => workDir + "/" + fragment + "/*.json" | ||||
|       ).concat(coreDir + "/*.json"), | ||||
|       { base: workDir } | ||||
|     ) | ||||
|     .pipe( | ||||
|       transform((data) => | ||||
|         // Polymer.AppLocalizeBehavior requires flattened json | ||||
|         flatten(data) | ||||
|       ) | ||||
|     ) | ||||
|     .pipe( | ||||
|       rename((filePath) => { | ||||
|         if (filePath.dirname === "core") { | ||||
|           filePath.dirname = ""; | ||||
|         } | ||||
|         // In dev we create the file with the fake hash in the filename | ||||
|         if (!env.isProdBuild()) { | ||||
|           filePath.basename += "-dev"; | ||||
|         } | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(outDir)) | ||||
| ); | ||||
|  | ||||
| const fingerprints = {}; | ||||
|  | ||||
| gulp.task("build-translation-fingerprints", () => { | ||||
|   // Fingerprint full file of each language | ||||
|   const files = readdirSync(fullDir); | ||||
|  | ||||
|   for (let i = 0; i < files.length; i++) { | ||||
|     fingerprints[files[i].split(".")[0]] = { | ||||
|       // In dev we create fake hashes | ||||
|       hash: env.isProdBuild() | ||||
|         ? createHash("md5") | ||||
|             .update(readFileSync(path.join(fullDir, files[i]), "utf-8")) | ||||
|             .digest("hex") | ||||
|         : "dev", | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   // In dev we create the file with the fake hash in the filename | ||||
|   if (env.isProdBuild()) { | ||||
|     mapFiles(outDir, ".json", (filename) => { | ||||
|       const parsed = path.parse(filename); | ||||
|  | ||||
|       // nl.json -> nl-<hash>.json | ||||
|       if (!(parsed.name in fingerprints)) { | ||||
|         throw new Error(`Unable to find hash for ${filename}`); | ||||
|       } | ||||
|  | ||||
|       renameSync( | ||||
|         filename, | ||||
|         `${parsed.dir}/${parsed.name}-${fingerprints[parsed.name].hash}${ | ||||
|           parsed.ext | ||||
|         }` | ||||
|       ); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   const stream = source("translationFingerprints.json"); | ||||
|   stream.write(JSON.stringify(fingerprints)); | ||||
|   process.nextTick(() => stream.end()); | ||||
|   return stream.pipe(vinylBuffer()).pipe(gulp.dest(workDir)); | ||||
| }); | ||||
|  | ||||
| gulp.task("build-translation-fragment-supervisor", () => | ||||
|   gulp | ||||
|     .src(fullDir + "/*.json") | ||||
|     .pipe(transform((data) => data.supervisor)) | ||||
|     .pipe( | ||||
|       rename((filePath) => { | ||||
|         // In dev we create the file with the fake hash in the filename | ||||
|         if (!env.isProdBuild()) { | ||||
|           filePath.basename += "-dev"; | ||||
|         } | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(workDir + "/supervisor")) | ||||
| ); | ||||
|  | ||||
| gulp.task("build-translation-flatten-supervisor", () => | ||||
|   gulp | ||||
|     .src(workDir + "/supervisor/*.json") | ||||
|     .pipe( | ||||
|       transform((data) => | ||||
|         // Polymer.AppLocalizeBehavior requires flattened json | ||||
|         flatten(data) | ||||
|       ) | ||||
|     ) | ||||
|     .pipe(gulp.dest(outDir)) | ||||
| ); | ||||
|  | ||||
| gulp.task("build-translation-write-metadata", () => | ||||
|   gulp | ||||
|     .src([ | ||||
|       path.join(paths.translations_src, "translationMetadata.json"), | ||||
|       ...(env.isProdBuild() ? [] : [workDir + "/testMetadata.json"]), | ||||
|       workDir + "/translationFingerprints.json", | ||||
|     ]) | ||||
|     .pipe(merge({})) | ||||
|     .pipe( | ||||
|       transform((data) => { | ||||
|         const newData = {}; | ||||
|         Object.entries(data).forEach(([key, value]) => { | ||||
|           // Filter out translations without native name. | ||||
|           if (value.nativeName) { | ||||
|             newData[key] = value; | ||||
|           } else { | ||||
|             console.warn( | ||||
|               `Skipping language ${key}. Native name was not translated.` | ||||
|             ); | ||||
|           } | ||||
|         }); | ||||
|         return newData; | ||||
|       }) | ||||
|     ) | ||||
|     .pipe( | ||||
|       transform((data) => ({ | ||||
|         fragments: TRANSLATION_FRAGMENTS, | ||||
|         translations: data, | ||||
|       })) | ||||
|     ) | ||||
|     .pipe(rename("translationMetadata.json")) | ||||
|     .pipe(gulp.dest(workDir)) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "create-translations", | ||||
|   gulp.series( | ||||
|     gulp.parallel("create-test-metadata", "create-test-translation"), | ||||
|     "build-master-translation", | ||||
|     "build-merged-translations", | ||||
|     gulp.parallel(...splitTasks), | ||||
|     "build-flattened-translations" | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "build-translations", | ||||
|   gulp.series( | ||||
|     gulp.parallel( | ||||
|       "fetch-nightly-translations", | ||||
|       gulp.series("clean-translations", makeWorkDir) | ||||
|       gulp.series("clean-translations", "ensure-translations-build-dir") | ||||
|     ), | ||||
|     createTestTranslation, | ||||
|     createMasterTranslation, | ||||
|     createTranslations, | ||||
|     writeTranslationMetaData | ||||
|     "create-translations", | ||||
|     "build-translation-fingerprints", | ||||
|     "build-translation-write-metadata" | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "build-supervisor-translations", | ||||
|   gulp.series(toggleSupervisorFragment, "build-translations") | ||||
|   gulp.series( | ||||
|     gulp.parallel( | ||||
|       "fetch-nightly-translations", | ||||
|       gulp.series("clean-translations", "ensure-translations-build-dir") | ||||
|     ), | ||||
|     gulp.parallel("create-test-metadata", "create-test-translation"), | ||||
|     "build-master-translation", | ||||
|     "build-merged-translations", | ||||
|     "build-translation-fragment-supervisor", | ||||
|     "build-translation-flatten-supervisor", | ||||
|     "build-translation-fingerprints", | ||||
|     "build-translation-write-metadata" | ||||
|   ) | ||||
| ); | ||||
|   | ||||
| @@ -40,12 +40,8 @@ const runDevServer = async ({ | ||||
|   compiler, | ||||
|   contentBase, | ||||
|   port, | ||||
|   listenHost = undefined, | ||||
|   listenHost = "localhost", | ||||
| }) => { | ||||
|   if (listenHost === undefined) { | ||||
|     // For dev container, we need to listen on all hosts | ||||
|     listenHost = env.isDevContainer() ? "0.0.0.0" : "localhost"; | ||||
|   } | ||||
|   const server = new WebpackDevServer( | ||||
|     { | ||||
|       hot: false, | ||||
| @@ -103,7 +99,7 @@ gulp.task("webpack-watch-app", () => { | ||||
|   ).watch({ poll: isWsl }, doneHandler()); | ||||
|   gulp.watch( | ||||
|     path.join(paths.translations_src, "en.json"), | ||||
|     gulp.series("build-translations", "copy-translations-app") | ||||
|     gulp.series("create-translations", "copy-translations-app") | ||||
|   ); | ||||
| }); | ||||
|  | ||||
|   | ||||
							
								
								
									
										16
									
								
								build-scripts/util.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								build-scripts/util.cjs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| const path = require("path"); | ||||
| const fs = require("fs"); | ||||
|  | ||||
| // Helper function to map recursively over files in a folder and it's subfolders | ||||
| module.exports.mapFiles = function mapFiles(startPath, filter, mapFunc) { | ||||
|   const files = fs.readdirSync(startPath); | ||||
|   for (let i = 0; i < files.length; i++) { | ||||
|     const filename = path.join(startPath, files[i]); | ||||
|     const stat = fs.lstatSync(filename); | ||||
|     if (stat.isDirectory()) { | ||||
|       mapFiles(filename, filter, mapFunc); | ||||
|     } else if (filename.indexOf(filter) >= 0) { | ||||
|       mapFunc(filename); | ||||
|     } | ||||
|   } | ||||
| }; | ||||
| @@ -10,7 +10,6 @@ const WebpackBar = require("webpackbar"); | ||||
| const { | ||||
|   TransformAsyncModulesPlugin, | ||||
| } = require("transform-async-modules-webpack-plugin"); | ||||
| const { dependencies } = require("../package.json"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const bundle = require("./bundle.cjs"); | ||||
|  | ||||
| @@ -63,25 +62,17 @@ const createWebpackConfig = ({ | ||||
|       rules: [ | ||||
|         { | ||||
|           test: /\.m?js$|\.ts$/, | ||||
|           use: (info) => ({ | ||||
|           use: { | ||||
|             loader: "babel-loader", | ||||
|             options: { | ||||
|               ...bundle.babelOptions({ | ||||
|                 latestBuild, | ||||
|                 isProdBuild, | ||||
|                 isTestBuild, | ||||
|                 sw: info.issuerLayer === "sw", | ||||
|               }), | ||||
|               ...bundle.babelOptions({ latestBuild, isProdBuild, isTestBuild }), | ||||
|               cacheDirectory: !isProdBuild, | ||||
|               cacheCompression: false, | ||||
|             }, | ||||
|           }), | ||||
|           }, | ||||
|           resolve: { | ||||
|             fullySpecified: false, | ||||
|           }, | ||||
|           parser: { | ||||
|             worker: ["*context.audioWorklet.addModule()", "..."], | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           test: /\.css$/, | ||||
| @@ -100,15 +91,11 @@ const createWebpackConfig = ({ | ||||
|       moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", | ||||
|       chunkIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", | ||||
|       splitChunks: { | ||||
|         // Disable splitting for web workers and worklets because imports of | ||||
|         // external chunks are broken for: | ||||
|         // - ESM output: https://github.com/webpack/webpack/issues/17014 | ||||
|         // - Worklets use `importScripts`: https://github.com/webpack/webpack/issues/11543 | ||||
|         chunks: (chunk) => | ||||
|           !chunk.canBeInitial() && | ||||
|           !new RegExp(`^.+-work${latestBuild ? "(?:let|er)" : "let"}$`).test( | ||||
|             chunk.name | ||||
|           ), | ||||
|         // Disable splitting for web workers with ESM output | ||||
|         // Imports of external chunks are broken | ||||
|         chunks: latestBuild | ||||
|           ? (chunk) => !chunk.canBeInitial() && !/^.+-worker$/.test(chunk.name) | ||||
|           : undefined, | ||||
|       }, | ||||
|     }, | ||||
|     plugins: [ | ||||
| @@ -169,15 +156,11 @@ const createWebpackConfig = ({ | ||||
|           transform: (stats) => JSON.stringify(filterStats(stats)), | ||||
|         }), | ||||
|       !latestBuild && | ||||
|         new TransformAsyncModulesPlugin({ | ||||
|           browserslistEnv: "legacy", | ||||
|           runtime: { version: dependencies["@babel/runtime"] }, | ||||
|         }), | ||||
|         new TransformAsyncModulesPlugin({ browserslistEnv: "legacy" }), | ||||
|     ].filter(Boolean), | ||||
|     resolve: { | ||||
|       extensions: [".ts", ".js", ".json"], | ||||
|       alias: { | ||||
|         "lit/static-html$": "lit/static-html.js", | ||||
|         "lit/decorators$": "lit/decorators.js", | ||||
|         "lit/directive$": "lit/directive.js", | ||||
|         "lit/directives/until$": "lit/directives/until.js", | ||||
| @@ -240,7 +223,6 @@ const createWebpackConfig = ({ | ||||
|       ), | ||||
|     }, | ||||
|     experiments: { | ||||
|       layers: true, | ||||
|       outputModule: true, | ||||
|     }, | ||||
|   }; | ||||
|   | ||||
| @@ -1,5 +0,0 @@ | ||||
| "use strict"; | ||||
|  | ||||
| self.addEventListener("fetch", (event) => { | ||||
|   event.respondWith(fetch(event.request)); | ||||
| }); | ||||
| @@ -36,7 +36,13 @@ | ||||
|   </head> | ||||
|   <body> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | ||||
|     <script> | ||||
|       <% for (const entry of latestEntryJS) { %> | ||||
|         import("<%= entry %>"); | ||||
|       <% } %> | ||||
|       window.latestJS = true; | ||||
|     </script> | ||||
|     <%= renderTemplate("../../../src/html/_script_load_es5.html.template") %> | ||||
|     <hc-layout subtitle="FAQ"> | ||||
|       <style> | ||||
|         a { | ||||
| @@ -226,5 +232,17 @@ http: | ||||
|         </p> | ||||
|       </div> | ||||
|     </hc-layout> | ||||
|  | ||||
|     <script> | ||||
|       var _gaq = [["_setAccount", "UA-57927901-9"], ["_trackPageview"]]; | ||||
|       (function (d, t) { | ||||
|         var g = d.createElement(t), | ||||
|           s = d.getElementsByTagName(t)[0]; | ||||
|         g.src = | ||||
|           ("https:" == location.protocol ? "//ssl" : "//www") + | ||||
|           ".google-analytics.com/ga.js"; | ||||
|         s.parentNode.insertBefore(g, s); | ||||
|       })(document, "script"); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -13,9 +13,15 @@ | ||||
|     <%= renderTemplate("_social_meta.html.template") %> | ||||
|   </head> | ||||
|   <body> | ||||
|     <hc-connect></hc-connect> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.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") %> | ||||
|     <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), | ||||
|   | ||||
| @@ -14,10 +14,22 @@ | ||||
|         --background-color: #41bdf5; | ||||
|       } | ||||
|     </style> | ||||
|     <script> | ||||
|       var _gaq=[['_setAccount','UA-57927901-10'],['_trackPageview']]; | ||||
|       (function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0]; | ||||
|       g.src=('https:'==location.protocol?'//ssl':'//www')+'.google-analytics.com/ga.js'; | ||||
|       s.parentNode.insertBefore(g,s)}(document,'script')); | ||||
|     </script> | ||||
|   </head> | ||||
|   <body> | ||||
|     <cast-media-player></cast-media-player> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.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") %> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -11,4 +11,10 @@ | ||||
|       font-size: initial; | ||||
|     } | ||||
|   </style> | ||||
|   <script> | ||||
|   var _gaq=[['_setAccount','UA-57927901-10'],['_trackPageview']]; | ||||
|   (function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0]; | ||||
|   g.src=('https:'==location.protocol?'//ssl':'//www')+'.google-analytics.com/ga.js'; | ||||
|   s.parentNode.insertBefore(g,s)}(document,'script')); | ||||
|   </script> | ||||
| </html> | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "../../../src/resources/safari-14-attachshadow-patch"; | ||||
| import "./layout/hc-connect"; | ||||
|  | ||||
| import("../../../src/resources/ha-style"); | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { ActionDetail } from "@material/mwc-list/mwc-list"; | ||||
| import { mdiCast, mdiCastConnected, mdiViewDashboard } from "@mdi/js"; | ||||
| import { mdiCast, mdiCastConnected } from "@mdi/js"; | ||||
| import "@polymer/paper-item/paper-icon-item"; | ||||
| import "@polymer/paper-listbox/paper-listbox"; | ||||
| import { Auth, Connection } from "home-assistant-js-websocket"; | ||||
| import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| @@ -27,7 +28,6 @@ import { LovelaceViewConfig } from "../../../../src/data/lovelace/config/view"; | ||||
| import "../../../../src/layouts/hass-loading-screen"; | ||||
| import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | ||||
| import "./hc-layout"; | ||||
| import "../../../../src/components/ha-list-item"; | ||||
|  | ||||
| @customElement("hc-cast") | ||||
| class HcCast extends LitElement { | ||||
| @@ -83,37 +83,34 @@ class HcCast extends LitElement { | ||||
|               ` | ||||
|             : html` | ||||
|                 <div class="section-header">PICK A VIEW</div> | ||||
|                 <mwc-list @action=${this._handlePickView} activatable> | ||||
|                 <paper-listbox | ||||
|                   attr-for-selected="data-path" | ||||
|                   .selected=${this.castManager.status.lovelacePath || ""} | ||||
|                 > | ||||
|                   ${( | ||||
|                     this.lovelaceViews ?? [ | ||||
|                       generateDefaultViewConfig({}, {}, {}, {}, () => ""), | ||||
|                     ] | ||||
|                   ).map( | ||||
|                     (view, idx) => | ||||
|                       html`<ha-list-item | ||||
|                         graphic="avatar" | ||||
|                         .activated=${this.castManager.status?.lovelacePath === | ||||
|                         (view.path ?? idx)} | ||||
|                         .selected=${this.castManager.status?.lovelacePath === | ||||
|                         (view.path ?? idx)} | ||||
|                     (view, idx) => html` | ||||
|                       <paper-icon-item | ||||
|                         @click=${this._handlePickView} | ||||
|                         data-path=${view.path || idx} | ||||
|                       > | ||||
|                         ${view.title || view.path || "Unnamed view"} | ||||
|                         ${view.icon | ||||
|                           ? html` | ||||
|                               <ha-icon | ||||
|                                 .icon=${view.icon} | ||||
|                                 slot="graphic" | ||||
|                                 slot="item-icon" | ||||
|                               ></ha-icon> | ||||
|                             ` | ||||
|                           : html`<ha-svg-icon | ||||
|                               slot="item-icon" | ||||
|                               .path=${mdiViewDashboard} | ||||
|                             ></ha-svg-icon>`}</ha-list-item | ||||
|                       > ` | ||||
|                   )}</mwc-list | ||||
|                 > | ||||
|                           : ""} | ||||
|                         ${view.title || view.path} | ||||
|                       </paper-icon-item> | ||||
|                     ` | ||||
|                   )} | ||||
|                 </paper-listbox> | ||||
|               `} | ||||
|  | ||||
|         <div class="card-actions"> | ||||
|           ${this.castManager.status | ||||
|             ? html` | ||||
| @@ -185,8 +182,8 @@ class HcCast extends LitElement { | ||||
|     this.castManager.requestSession(); | ||||
|   } | ||||
|  | ||||
|   private async _handlePickView(ev: CustomEvent<ActionDetail>) { | ||||
|     const path = this.lovelaceViews![ev.detail.index].path ?? ev.detail.index; | ||||
|   private async _handlePickView(ev: Event) { | ||||
|     const path = (ev.currentTarget as any).getAttribute("data-path"); | ||||
|     await ensureConnectedCastSession(this.castManager!, this.auth!); | ||||
|     castSendShowLovelaceView(this.castManager, this.auth.data.hassUrl, path); | ||||
|   } | ||||
| @@ -249,14 +246,25 @@ class HcCast extends LitElement { | ||||
|         height: 18px; | ||||
|       } | ||||
|  | ||||
|       ha-list-item ha-icon, | ||||
|       ha-list-item ha-svg-icon { | ||||
|       paper-listbox { | ||||
|         padding-top: 0; | ||||
|       } | ||||
|  | ||||
|       paper-listbox ha-icon { | ||||
|         padding: 12px; | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|  | ||||
|       :host([hide-icons]) ha-icon { | ||||
|         display: none; | ||||
|       paper-icon-item { | ||||
|         cursor: pointer; | ||||
|       } | ||||
|  | ||||
|       paper-icon-item[disabled] { | ||||
|         cursor: initial; | ||||
|       } | ||||
|  | ||||
|       :host([hide-icons]) paper-icon-item { | ||||
|         --paper-item-icon-width: 0px; | ||||
|       } | ||||
|  | ||||
|       .spacer { | ||||
|   | ||||
| @@ -28,7 +28,7 @@ class HcLaunchScreen extends LitElement { | ||||
|       :host { | ||||
|         display: block; | ||||
|         height: 100vh; | ||||
|         background-color: #f2f4f9; | ||||
|         background-color: white; | ||||
|         font-size: 24px; | ||||
|       } | ||||
|       .container { | ||||
| @@ -43,9 +43,6 @@ class HcLaunchScreen extends LitElement { | ||||
|         max-width: 80%; | ||||
|         object-fit: cover; | ||||
|       } | ||||
|       .status { | ||||
|         color: #1d2126; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -2,7 +2,6 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { LovelaceConfig } from "../../../../src/data/lovelace/config/types"; | ||||
| import { getPanelTitleFromUrlPath } from "../../../../src/data/panel"; | ||||
| import { Lovelace } from "../../../../src/panels/lovelace/types"; | ||||
| import "../../../../src/panels/lovelace/views/hui-view"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -62,12 +61,7 @@ class HcLovelace extends LitElement { | ||||
|       const index = this._viewIndex; | ||||
|  | ||||
|       if (index !== undefined) { | ||||
|         const title = getPanelTitleFromUrlPath( | ||||
|           this.hass, | ||||
|           this.urlPath || "lovelace" | ||||
|         ); | ||||
|  | ||||
|         const dashboardTitle = title || this.urlPath; | ||||
|         const dashboardTitle = this.lovelaceConfig.title || this.urlPath; | ||||
|  | ||||
|         const viewTitle = | ||||
|           this.lovelaceConfig.views[index].title || | ||||
| @@ -86,17 +80,10 @@ class HcLovelace extends LitElement { | ||||
|           this.lovelaceConfig.views[index].background || | ||||
|           this.lovelaceConfig.background; | ||||
|  | ||||
|         const backgroundStyle = | ||||
|           typeof configBackground === "string" | ||||
|             ? configBackground | ||||
|             : configBackground?.image | ||||
|               ? `center / cover no-repeat url('${configBackground.image}')` | ||||
|               : undefined; | ||||
|  | ||||
|         if (backgroundStyle) { | ||||
|         if (configBackground) { | ||||
|           this._huiView!.style.setProperty( | ||||
|             "--lovelace-background", | ||||
|             backgroundStyle | ||||
|             configBackground | ||||
|           ); | ||||
|         } else { | ||||
|           this._huiView!.style.removeProperty("--lovelace-background"); | ||||
|   | ||||
| @@ -35,7 +35,6 @@ import { loadLovelaceResources } from "../../../../src/panels/lovelace/common/lo | ||||
| import { HassElement } from "../../../../src/state/hass-element"; | ||||
| import { castContext } from "../cast_context"; | ||||
| import "./hc-launch-screen"; | ||||
| import { getPanelTitleFromUrlPath } from "../../../../src/data/panel"; | ||||
|  | ||||
| const DEFAULT_CONFIG: LovelaceDashboardStrategyConfig = { | ||||
|   strategy: { | ||||
| @@ -271,7 +270,7 @@ export class HcMain extends HassElement { | ||||
|     } | ||||
|  | ||||
|     this._error = undefined; | ||||
|     if (msg.urlPath === "lovelace" || msg.urlPath === undefined) { | ||||
|     if (msg.urlPath === "lovelace") { | ||||
|       msg.urlPath = null; | ||||
|     } | ||||
|     this._lovelacePath = msg.viewPath; | ||||
| @@ -360,11 +359,7 @@ export class HcMain extends HassElement { | ||||
|   } | ||||
|  | ||||
|   private _handleNewLovelaceConfig(lovelaceConfig: LovelaceConfig) { | ||||
|     const title = getPanelTitleFromUrlPath( | ||||
|       this.hass!, | ||||
|       this._urlPath || "lovelace" | ||||
|     ); | ||||
|     castContext.setApplicationState(title || ""); | ||||
|     castContext.setApplicationState(lovelaceConfig.title || ""); | ||||
|     this._lovelaceConfig = lovelaceConfig; | ||||
|   } | ||||
|  | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 30 KiB | 
| @@ -1,5 +0,0 @@ | ||||
| "use strict"; | ||||
|  | ||||
| self.addEventListener("fetch", (event) => { | ||||
|   event.respondWith(fetch(event.request)); | ||||
| }); | ||||
| @@ -4,7 +4,6 @@ import { energyEntities } from "../stubs/entities"; | ||||
| import { DemoConfig } from "./types"; | ||||
|  | ||||
| export const demoConfigs: Array<() => Promise<DemoConfig>> = [ | ||||
|   () => import("./sections").then((mod) => mod.demoSections), | ||||
|   () => import("./arsaboo").then((mod) => mod.demoArsaboo), | ||||
|   () => import("./teachingbirds").then((mod) => mod.demoTeachingbirds), | ||||
|   () => import("./kernehed").then((mod) => mod.demoKernehed), | ||||
|   | ||||
| @@ -1,16 +0,0 @@ | ||||
| import { html } from "lit"; | ||||
| import { DemoConfig } from "../types"; | ||||
|  | ||||
| export const demoLovelaceDescription: DemoConfig["description"] = ( | ||||
|   localize | ||||
| ) => html` | ||||
|   <p> | ||||
|     ${localize("ui.panel.page-demo.config.sections.description", { | ||||
|       blog_post: html`<a | ||||
|         href="https://www.home-assistant.io/blog/2024/03/04/dashboard-chapter-1/" | ||||
|         target="_blank" | ||||
|         >${localize("ui.panel.page-demo.config.sections.description_blog_post")} | ||||
|       </a>`, | ||||
|     })} | ||||
|   </p> | ||||
| `; | ||||
| @@ -1,532 +0,0 @@ | ||||
| import { convertEntities } from "../../../../src/fake_data/entity"; | ||||
| import { DemoConfig } from "../types"; | ||||
|  | ||||
| export const demoEntitiesSections: DemoConfig["entities"] = (localize) => | ||||
|   convertEntities({ | ||||
|     "cover.living_room_garden_shutter": { | ||||
|       entity_id: "cover.living_room_garden_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Living room garden shutter", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "cover.living_room_graveyard_shutter": { | ||||
|       entity_id: "cover.living_room_graveyard_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Living room graveyard shutter", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "cover.living_room_left_shutter": { | ||||
|       entity_id: "cover.living_room_left_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Living room left shutter", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "cover.living_room_right_shutter": { | ||||
|       entity_id: "cover.living_room_right_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Living room right shutter", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "light.floor_lamp": { | ||||
|       entity_id: "light.floor_lamp", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         min_color_temp_kelvin: 2000, | ||||
|         max_color_temp_kelvin: 6535, | ||||
|         min_mireds: 153, | ||||
|         max_mireds: 500, | ||||
|         supported_color_modes: ["color_temp", "xy"], | ||||
|         color_mode: "color_temp", | ||||
|         brightness: 178, | ||||
|         color_temp_kelvin: 2583, | ||||
|         color_temp: 387, | ||||
|         hs_color: [28.664, 69.597], | ||||
|         rgb_color: [255, 162, 77], | ||||
|         xy_color: [0.538, 0.389], | ||||
|         icon: "mdi:floor-lamp", | ||||
|         friendly_name: "Floor lamp", | ||||
|         supported_features: 44, | ||||
|       }, | ||||
|     }, | ||||
|     "light.living_room_spotlights": { | ||||
|       entity_id: "light.living_room_spotlights", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         supported_color_modes: ["brightness"], | ||||
|         color_mode: "brightness", | ||||
|         brightness: 126, | ||||
|         icon: "mdi:ceiling-light-multiple", | ||||
|         friendly_name: "Living room spotlights", | ||||
|         supported_features: 32, | ||||
|       }, | ||||
|     }, | ||||
|     "light.bar_lamp": { | ||||
|       entity_id: "light.bar_lamp", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         min_color_temp_kelvin: 2202, | ||||
|         max_color_temp_kelvin: 4504, | ||||
|         min_mireds: 222, | ||||
|         max_mireds: 454, | ||||
|         effect_list: ["None", "candle"], | ||||
|         supported_color_modes: ["color_temp"], | ||||
|         effect: null, | ||||
|         color_mode: null, | ||||
|         brightness: null, | ||||
|         color_temp_kelvin: null, | ||||
|         color_temp: null, | ||||
|         hs_color: null, | ||||
|         rgb_color: null, | ||||
|         xy_color: null, | ||||
|         mode: "normal", | ||||
|         dynamics: "none", | ||||
|         icon: "mdi:lightbulb-variant", | ||||
|         friendly_name: "Bar lamp", | ||||
|         supported_features: 44, | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.living_room_temperature": { | ||||
|       entity_id: "sensor.living_room_temperature", | ||||
|       state: "22.8", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         unit_of_measurement: "°C", | ||||
|         device_class: "temperature", | ||||
|         friendly_name: "Living room Temperature", | ||||
|       }, | ||||
|     }, | ||||
|     "media_player.living_room_nest_mini": { | ||||
|       entity_id: "media_player.living_room_nest_mini", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         device_class: "speaker", | ||||
|         volume_level: 0.18, | ||||
|         is_volume_muted: false, | ||||
|         media_content_type: "music", | ||||
|         media_duration: 300, | ||||
|         media_position: 0, | ||||
|         media_position_updated_at: new Date( | ||||
|           // 23 seconds in | ||||
|           new Date().getTime() - 23000 | ||||
|         ).toISOString(), | ||||
|         media_title: "I Wasn't Born To Follow", | ||||
|         media_artist: "The Byrds", | ||||
|         media_album_name: "The Notorious Byrd Brothers", | ||||
|         source_list: ["It's A Party", "Radio HSL", "Retro 70s and 80s"], | ||||
|         shuffle: false, | ||||
|         night_sound: false, | ||||
|         speech_enhance: false, | ||||
|         friendly_name: localize( | ||||
|           "ui.panel.page-demo.config.sections.entities.media_player.living_room_nest_mini" | ||||
|         ), | ||||
|         entity_picture: "/assets/sections/images/media_player_family_room.jpg", | ||||
|         supported_features: 64063, | ||||
|       }, | ||||
|     }, | ||||
|     "cover.kitchen_shutter": { | ||||
|       entity_id: "cover.kitchen_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Kitchen shutter ", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "light.kitchen_spotlights": { | ||||
|       entity_id: "light.kitchen_spotlights", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         supported_color_modes: ["brightness"], | ||||
|         color_mode: null, | ||||
|         brightness: null, | ||||
|         icon: "mdi:ceiling-light-multiple", | ||||
|         friendly_name: "Kitchen spotlights ", | ||||
|         supported_features: 32, | ||||
|       }, | ||||
|     }, | ||||
|     "light.worktop_spotlights": { | ||||
|       entity_id: "light.worktop_spotlights", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         supported_color_modes: ["brightness"], | ||||
|         color_mode: null, | ||||
|         brightness: null, | ||||
|         icon: "mdi:ceiling-light-multiple", | ||||
|         friendly_name: "Worktop spotlights ", | ||||
|         supported_features: 32, | ||||
|       }, | ||||
|     }, | ||||
|     "binary_sensor.fridge_door": { | ||||
|       entity_id: "binary_sensor.fridge_door", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         device_class: "door", | ||||
|         icon: "mdi:fridge", | ||||
|         friendly_name: "Fridge door", | ||||
|       }, | ||||
|     }, | ||||
|     "media_player.kitchen_nest_audio": { | ||||
|       entity_id: "media_player.kitchen_nest_audio", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         device_class: "speaker", | ||||
|         volume_level: 0.18, | ||||
|         is_volume_muted: false, | ||||
|         media_content_type: "music", | ||||
|         media_duration: 300, | ||||
|         media_position: 0, | ||||
|         media_position_updated_at: new Date( | ||||
|           // 23 seconds in | ||||
|           new Date().getTime() - 23000 | ||||
|         ).toISOString(), | ||||
|         media_title: "I Wasn't Born To Follow", | ||||
|         media_artist: "The Byrds", | ||||
|         media_album_name: "The Notorious Byrd Brothers", | ||||
|         source_list: ["It's A Party", "Radio HSL", "Retro 70s and 80s"], | ||||
|         shuffle: false, | ||||
|         night_sound: false, | ||||
|         speech_enhance: false, | ||||
|         friendly_name: localize( | ||||
|           "ui.panel.page-demo.config.sections.entities.media_player.kitchen_nest_audio" | ||||
|         ), | ||||
|         entity_picture: "/assets/sections/images/media_player_family_room.jpg", | ||||
|         supported_features: 64063, | ||||
|       }, | ||||
|     }, | ||||
|     "binary_sensor.tesla_wall_connector_vehicle_connected": { | ||||
|       entity_id: "binary_sensor.tesla_wall_connector_vehicle_connected", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         device_class: "plug", | ||||
|         friendly_name: "Wall Connector Vehicle connected", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.tesla_wall_connector_session_energy": { | ||||
|       entity_id: "sensor.tesla_wall_connector_session_energy", | ||||
|       state: "16.3", | ||||
|       attributes: { | ||||
|         state_class: "total_increasing", | ||||
|         unit_of_measurement: "kWh", | ||||
|         device_class: "energy", | ||||
|         friendly_name: "Tesla Wall Connector Session energy", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.electric_meter_power": { | ||||
|       entity_id: "sensor.electric_meter_power", | ||||
|       state: "797.86", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         unit_of_measurement: "W", | ||||
|         device_class: "power", | ||||
|         icon: "mdi:meter-electric", | ||||
|         friendly_name: "Electric meter Power", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.eletric_meter_voltage": { | ||||
|       entity_id: "sensor.eletric_meter_voltage", | ||||
|       state: "232.19", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         unit_of_measurement: "V", | ||||
|         device_class: "voltage", | ||||
|         friendly_name: "Electric meter voltage", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.electricity_maps_grid_fossil_fuel_percentage": { | ||||
|       entity_id: "sensor.electricity_maps_grid_fossil_fuel_percentage", | ||||
|       state: "9.84", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         country_code: "FR", | ||||
|         unit_of_measurement: "%", | ||||
|         attribution: "Data provided by Electricity Maps", | ||||
|         icon: "mdi:barrel", | ||||
|         friendly_name: "Electricity Maps Grid fossil fuel percentage", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.electricity_maps_co2_intensity": { | ||||
|       entity_id: "sensor.electricity_maps_co2_intensity", | ||||
|       state: "62.0", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         country_code: "FR", | ||||
|         unit_of_measurement: "gCO2eq/kWh", | ||||
|         attribution: "Data provided by Electricity Maps", | ||||
|         friendly_name: "Electricity Maps CO2 intensity", | ||||
|         icon: "mdi:molecule-co2", | ||||
|       }, | ||||
|     }, | ||||
|     "sun.sun": { | ||||
|       entity_id: "sun.sun", | ||||
|       state: "above_horizon", | ||||
|       attributes: { | ||||
|         next_dawn: "2024-03-05T05:50:21.964405+00:00", | ||||
|         next_dusk: "2024-03-04T18:08:54.311334+00:00", | ||||
|         next_midnight: "2024-03-05T00:00:00+00:00", | ||||
|         next_noon: "2024-03-05T12:00:05+00:00", | ||||
|         next_rising: "2024-03-05T06:23:42.739159+00:00", | ||||
|         next_setting: "2024-03-04T17:35:26.271171+00:00", | ||||
|         elevation: 30.38, | ||||
|         azimuth: 204.42, | ||||
|         rising: false, | ||||
|         friendly_name: "Sun", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.rain": { | ||||
|       entity_id: "sensor.moon_phase", | ||||
|       state: "7.2", | ||||
|       attributes: { | ||||
|         state_class: "total_increasing", | ||||
|         unit_of_measurement: "mm", | ||||
|         device_class: "precipitation", | ||||
|         friendly_name: "Rain", | ||||
|       }, | ||||
|     }, | ||||
|     "climate.ground_floor": { | ||||
|       entity_id: "climate.ground_floor", | ||||
|       state: "heat", | ||||
|       attributes: { | ||||
|         hvac_modes: ["auto", "heat", "off"], | ||||
|         min_temp: 7, | ||||
|         max_temp: 35, | ||||
|         preset_modes: [ | ||||
|           "comfort", | ||||
|           "away", | ||||
|           "eco", | ||||
|           "frost_protection", | ||||
|           "external", | ||||
|           "home", | ||||
|         ], | ||||
|         current_temperature: 20.8, | ||||
|         temperature: 21, | ||||
|         preset_mode: "comfort", | ||||
|         icon: "mdi:home-floor-0", | ||||
|         friendly_name: "Ground floor Thermostat", | ||||
|         supported_features: 401, | ||||
|       }, | ||||
|     }, | ||||
|     "climate.first_floor": { | ||||
|       entity_id: "climate.first_floor", | ||||
|       state: "heat", | ||||
|       attributes: { | ||||
|         hvac_modes: ["auto", "heat", "off"], | ||||
|         min_temp: 7, | ||||
|         max_temp: 35, | ||||
|         preset_modes: [ | ||||
|           "comfort", | ||||
|           "away", | ||||
|           "eco", | ||||
|           "frost_protection", | ||||
|           "external", | ||||
|           "home", | ||||
|         ], | ||||
|         current_temperature: 21.7, | ||||
|         temperature: 21, | ||||
|         preset_mode: "comfort", | ||||
|         icon: "mdi:home-floor-1", | ||||
|         friendly_name: "First floor Thermostat", | ||||
|         supported_features: 401, | ||||
|       }, | ||||
|     }, | ||||
|     "cover.study_shutter": { | ||||
|       entity_id: "cover.study_shutter", | ||||
|       state: "open", | ||||
|       attributes: { | ||||
|         current_position: 100, | ||||
|         device_class: "shutter", | ||||
|         friendly_name: "Study shutter", | ||||
|         supported_features: 15, | ||||
|       }, | ||||
|     }, | ||||
|     "light.study_spotlights": { | ||||
|       entity_id: "light.study_spotlights", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         supported_color_modes: ["brightness"], | ||||
|         color_mode: null, | ||||
|         brightness: null, | ||||
|         icon: "mdi:ceiling-light-multiple", | ||||
|         friendly_name: "Study spotlights", | ||||
|         supported_features: 32, | ||||
|       }, | ||||
|     }, | ||||
|     "media_player.study_nest_hub": { | ||||
|       entity_id: "media_player.study_nest_hub", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         device_class: "speaker", | ||||
|         volume_level: 0.18, | ||||
|         is_volume_muted: false, | ||||
|         media_content_type: "music", | ||||
|         media_duration: 300, | ||||
|         media_position: 0, | ||||
|         media_position_updated_at: new Date( | ||||
|           // 23 seconds in | ||||
|           new Date().getTime() - 23000 | ||||
|         ).toISOString(), | ||||
|         media_title: "I Wasn't Born To Follow", | ||||
|         media_artist: "The Byrds", | ||||
|         media_album_name: "The Notorious Byrd Brothers", | ||||
|         source_list: ["It's A Party", "Radio HSL", "Retro 70s and 80s"], | ||||
|         shuffle: false, | ||||
|         night_sound: false, | ||||
|         speech_enhance: false, | ||||
|         friendly_name: localize( | ||||
|           "ui.panel.page-demo.config.sections.entities.media_player.study_nest_hub" | ||||
|         ), | ||||
|         entity_picture: "/assets/sections/images/media_player_family_room.jpg", | ||||
|         supported_features: 64063, | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.standing_desk_height": { | ||||
|       entity_id: "sensor.standing_desk_height", | ||||
|       state: "72", | ||||
|       attributes: { | ||||
|         unit_of_measurement: "cm", | ||||
|         icon: "mdi:tape-measure", | ||||
|         friendly_name: "Standing desk Height", | ||||
|       }, | ||||
|     }, | ||||
|     "light.outdoor_light": { | ||||
|       entity_id: "light.outdoor_light", | ||||
|       state: "on", | ||||
|       attributes: { | ||||
|         supported_color_modes: ["brightness"], | ||||
|         color_mode: null, | ||||
|         brightness: 255, | ||||
|         icon: "mdi:outdoor-lamp", | ||||
|         friendly_name: "Outdoor light", | ||||
|         supported_features: 32, | ||||
|       }, | ||||
|     }, | ||||
|     "light.flood_light": { | ||||
|       entity_id: "light.flood_light", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         effect_list: ["None", "candle"], | ||||
|         supported_color_modes: ["brightness"], | ||||
|         effect: null, | ||||
|         color_mode: null, | ||||
|         brightness: null, | ||||
|         mode: "normal", | ||||
|         dynamics: "none", | ||||
|         icon: "mdi:light-flood-down", | ||||
|         friendly_name: "Flood light", | ||||
|         supported_features: 44, | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.outdoor_motion_sensor_temperature": { | ||||
|       entity_id: "sensor.outdoor_motion_sensor_temperature", | ||||
|       state: "10.2", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         unit_of_measurement: "°C", | ||||
|         device_class: "temperature", | ||||
|         friendly_name: "Outdoor motion sensor Temperature", | ||||
|       }, | ||||
|     }, | ||||
|     "binary_sensor.outdoor_motion_sensor_motion": { | ||||
|       entity_id: "binary_sensor.outdoor_motion_sensor_motion", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         device_class: "motion", | ||||
|         friendly_name: "Outdoor motion sensor Motion", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.outdoor_motion_sensor_illuminance": { | ||||
|       entity_id: "sensor.outdoor_motion_sensor_illuminance", | ||||
|       state: "555", | ||||
|       attributes: { | ||||
|         state_class: "measurement", | ||||
|         light_level: 27444, | ||||
|         unit_of_measurement: "lx", | ||||
|         device_class: "illuminance", | ||||
|         friendly_name: "Outdoor motion sensor Illuminance", | ||||
|       }, | ||||
|     }, | ||||
|     "automation.home_assistant_auto_update": { | ||||
|       entity_id: "automation.home_assistant_auto_update", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         id: "1700669321947", | ||||
|         last_triggered: "2024-02-29T18:02:05.343139+00:00", | ||||
|         mode: "queued", | ||||
|         current: 0, | ||||
|         max: 50, | ||||
|         icon: "mdi:auto-mode", | ||||
|         friendly_name: "Home Assistant Auto-update", | ||||
|       }, | ||||
|     }, | ||||
|     "update.home_assistant_operating_system_update": { | ||||
|       entity_id: "update.home_assistant_operating_system_update", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         auto_update: false, | ||||
|         installed_version: "12.1", | ||||
|         in_progress: false, | ||||
|         latest_version: "12.1", | ||||
|         release_summary: null, | ||||
|         release_url: | ||||
|           "https://github.com/home-assistant/operating-system/commits/dev", | ||||
|         skipped_version: null, | ||||
|         title: "Home Assistant Operating System", | ||||
|         entity_picture: | ||||
|           "https://brands.home-assistant.io/homeassistant/icon.png", | ||||
|         friendly_name: "Home Assistant Operating System Update", | ||||
|         supported_features: 3, | ||||
|       }, | ||||
|     }, | ||||
|     "update.home_assistant_supervisor_update": { | ||||
|       entity_id: "update.home_assistant_supervisor_update", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         auto_update: true, | ||||
|         installed_version: "2024.02.2", | ||||
|         in_progress: false, | ||||
|         latest_version: "2024.02.2", | ||||
|         release_summary: null, | ||||
|         release_url: | ||||
|           "https://github.com/home-assistant/supervisor/commits/main", | ||||
|         skipped_version: null, | ||||
|         title: "Home Assistant Supervisor", | ||||
|         entity_picture: "https://brands.home-assistant.io/hassio/icon.png", | ||||
|         friendly_name: "Home Assistant Supervisor Update", | ||||
|         supported_features: 1, | ||||
|       }, | ||||
|     }, | ||||
|     "update.home_assistant_core_update": { | ||||
|       entity_id: "update.home_assistant_supervisor_update", | ||||
|       state: "off", | ||||
|       attributes: { | ||||
|         auto_update: false, | ||||
|         installed_version: "2024.4.0", | ||||
|         in_progress: false, | ||||
|         latest_version: "2024.4.0", | ||||
|         release_summary: null, | ||||
|         release_url: "https://github.com/home-assistant/core/commits/dev", | ||||
|         skipped_version: null, | ||||
|         title: "Home Assistant Core", | ||||
|         entity_picture: | ||||
|           "https://brands.home-assistant.io/homeassistant/icon.png", | ||||
|         friendly_name: "Home Assistant Core Update", | ||||
|         supported_features: 11, | ||||
|       }, | ||||
|     }, | ||||
|   }); | ||||
| @@ -1,14 +0,0 @@ | ||||
| import { DemoConfig } from "../types"; | ||||
| import { demoLovelaceDescription } from "./description"; | ||||
| import { demoEntitiesSections } from "./entities"; | ||||
| import { demoLovelaceSections } from "./lovelace"; | ||||
|  | ||||
| export const demoSections: DemoConfig = { | ||||
|   authorName: "Home Assistant", | ||||
|   authorUrl: "https://github.com/home-assistant/frontend/", | ||||
|   name: "Home Demo", | ||||
|   description: demoLovelaceDescription, | ||||
|   lovelace: demoLovelaceSections, | ||||
|   entities: demoEntitiesSections, | ||||
|   theme: () => ({}), | ||||
| }; | ||||
| @@ -1,268 +0,0 @@ | ||||
| import { isFrontpageEmbed } from "../../util/is_frontpage"; | ||||
| import { DemoConfig } from "../types"; | ||||
|  | ||||
| export const demoLovelaceSections: DemoConfig["lovelace"] = (localize) => ({ | ||||
|   title: "Home Assistant Demo", | ||||
|   views: [ | ||||
|     { | ||||
|       type: "sections", | ||||
|       title: isFrontpageEmbed ? "Home Assistant" : "Demo", | ||||
|       path: "home", | ||||
|       icon: "mdi:home-assistant", | ||||
|       sections: [ | ||||
|         ...(isFrontpageEmbed | ||||
|           ? [] | ||||
|           : [ | ||||
|               { | ||||
|                 title: `${localize("ui.panel.page-demo.config.sections.titles.welcome")} 👋`, | ||||
|                 cards: [{ type: "custom:ha-demo-card" }], | ||||
|               }, | ||||
|             ]), | ||||
|         { | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.floor_lamp", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.living_room_spotlights", | ||||
|               name: "Spotlights", | ||||
|               features: [ | ||||
|                 { | ||||
|                   type: "light-brightness", | ||||
|                 }, | ||||
|               ], | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.bar_lamp", | ||||
|             }, | ||||
|             { | ||||
|               graph: "line", | ||||
|               type: "sensor", | ||||
|               entity: "sensor.living_room_temperature", | ||||
|               detail: 1, | ||||
|               name: "Temperature", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "cover.living_room_garden_shutter", | ||||
|               name: "Blinds", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "media_player.living_room_nest_mini", | ||||
|             }, | ||||
|           ], | ||||
|           title: `🛋️ ${localize("ui.panel.page-demo.config.sections.titles.living_room")} `, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "cover.kitchen_shutter", | ||||
|               name: "Shutter", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.kitchen_spotlights", | ||||
|               name: "Spotlights", | ||||
|               features: [ | ||||
|                 { | ||||
|                   type: "light-brightness", | ||||
|                 }, | ||||
|               ], | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.worktop_spotlights", | ||||
|               name: "Worktop", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "binary_sensor.fridge_door", | ||||
|               name: "Fridge", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "media_player.kitchen_nest_audio", | ||||
|             }, | ||||
|           ], | ||||
|           title: `👩🍳 ${localize("ui.panel.page-demo.config.sections.titles.kitchen")}`, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "binary_sensor.tesla_wall_connector_vehicle_connected", | ||||
|               name: "EV", | ||||
|               icon: "mdi:car", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.tesla_wall_connector_session_energy", | ||||
|               name: "Last charge", | ||||
|               color: "green", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.electric_meter_power", | ||||
|               color: "deep-orange", | ||||
|               name: "Home power", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.eletric_meter_voltage", | ||||
|               name: "Voltage", | ||||
|               color: "deep-orange", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.electricity_maps_grid_fossil_fuel_percentage", | ||||
|               name: "Fossil fuel", | ||||
|               color: "brown", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.electricity_maps_co2_intensity", | ||||
|               name: "CO2 Intensity", | ||||
|               color: "dark-grey", | ||||
|             }, | ||||
|           ], | ||||
|           title: `⚡️ ${localize("ui.panel.page-demo.config.sections.titles.energy")}`, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sun.sun", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.rain", | ||||
|               color: "blue", | ||||
|             }, | ||||
|             { | ||||
|               features: [ | ||||
|                 { | ||||
|                   type: "target-temperature", | ||||
|                 }, | ||||
|               ], | ||||
|               type: "tile", | ||||
|               name: "Downstairs", | ||||
|               entity: "climate.ground_floor", | ||||
|               state_content: ["preset_mode", "current_temperature"], | ||||
|             }, | ||||
|             { | ||||
|               features: [ | ||||
|                 { | ||||
|                   type: "target-temperature", | ||||
|                 }, | ||||
|               ], | ||||
|               type: "tile", | ||||
|               name: "Upstairs", | ||||
|               entity: "climate.first_floor", | ||||
|               state_content: ["preset_mode", "current_temperature"], | ||||
|             }, | ||||
|           ], | ||||
|           title: `🌤️ ${localize("ui.panel.page-demo.config.sections.titles.climate")}`, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "cover.study_shutter", | ||||
|               name: "Shutter", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.study_spotlights", | ||||
|               name: "Spotlights", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "media_player.study_nest_hub", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.standing_desk_height", | ||||
|               name: "Desk", | ||||
|               color: "brown", | ||||
|               icon: "mdi:desk", | ||||
|             }, | ||||
|           ], | ||||
|           title: `🧑💻 ${localize("ui.panel.page-demo.config.sections.titles.study")}`, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.outdoor_light", | ||||
|               name: "Door light", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "light.flood_light", | ||||
|             }, | ||||
|             { | ||||
|               graph: "line", | ||||
|               type: "sensor", | ||||
|               entity: "sensor.outdoor_motion_sensor_temperature", | ||||
|               detail: 1, | ||||
|               name: "Temperature", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "binary_sensor.outdoor_motion_sensor_motion", | ||||
|               name: "Motion", | ||||
|               color: "blue", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "sensor.outdoor_motion_sensor_illuminance", | ||||
|               color: "amber", | ||||
|               name: "Illuminance", | ||||
|             }, | ||||
|           ], | ||||
|           title: `🌳 ${localize("ui.panel.page-demo.config.sections.titles.outdoor")}`, | ||||
|         }, | ||||
|         { | ||||
|           type: "grid", | ||||
|           cards: [ | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "automation.home_assistant_auto_update", | ||||
|               name: "Auto-update", | ||||
|               color: "green", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "update.home_assistant_operating_system_update", | ||||
|               name: "OS", | ||||
|               icon: "mdi:home-assistant", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "update.home_assistant_supervisor_update", | ||||
|               icon: "mdi:home-assistant", | ||||
|               name: "Supervisor", | ||||
|             }, | ||||
|             { | ||||
|               type: "tile", | ||||
|               entity: "update.home_assistant_core_update", | ||||
|               name: "Core", | ||||
|               icon: "mdi:home-assistant", | ||||
|             }, | ||||
|           ], | ||||
|           title: `🎉 ${localize("ui.panel.page-demo.config.sections.titles.updates")}`, | ||||
|         }, | ||||
|       ], | ||||
|     }, | ||||
|   ], | ||||
| }); | ||||
| @@ -1,4 +1,3 @@ | ||||
| import { TemplateResult } from "lit"; | ||||
| import { LocalizeFunc } from "../../../src/common/translations/localize"; | ||||
| import { LovelaceConfig } from "../../../src/data/lovelace/config/types"; | ||||
| import { Entity } from "../../../src/fake_data/entity"; | ||||
| @@ -8,9 +7,6 @@ export interface DemoConfig { | ||||
|   name: string; | ||||
|   authorName: string; | ||||
|   authorUrl: string; | ||||
|   description?: | ||||
|     | string | ||||
|     | ((localize: LocalizeFunc) => string | TemplateResult<1>); | ||||
|   lovelace: (localize: LocalizeFunc) => LovelaceConfig; | ||||
|   entities: (localize: LocalizeFunc) => Entity[]; | ||||
|   theme: () => Record<string, string> | null; | ||||
|   | ||||
| @@ -1,9 +1,8 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { until } from "lit/directives/until"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-button"; | ||||
| import "../../../src/components/ha-circular-progress"; | ||||
| import { LovelaceCardConfig } from "../../../src/data/lovelace/config/card"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| @@ -12,6 +11,7 @@ import { | ||||
|   demoConfigs, | ||||
|   selectedDemoConfig, | ||||
|   selectedDemoConfigIndex, | ||||
|   setDemoConfig, | ||||
| } from "../configs/demo-configs"; | ||||
|  | ||||
| @customElement("ha-demo-card") | ||||
| @@ -39,57 +39,38 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|         <div class="picker"> | ||||
|           <div class="label"> | ||||
|             ${this._switching | ||||
|               ? html` | ||||
|                   <ha-circular-progress indeterminate></ha-circular-progress> | ||||
|                 ` | ||||
|               ? html`<ha-circular-progress | ||||
|                   indeterminate | ||||
|                 ></ha-circular-progress>` | ||||
|               : until( | ||||
|                   selectedDemoConfig.then( | ||||
|                     (conf) => html` | ||||
|                       ${conf.name} | ||||
|                       <small> | ||||
|                         ${this.hass.localize( | ||||
|                           "ui.panel.page-demo.cards.demo.demo_by", | ||||
|                           { | ||||
|                             name: html` | ||||
|                               <a target="_blank" href=${conf.authorUrl}> | ||||
|                                 ${conf.authorName} | ||||
|                               </a> | ||||
|                             `, | ||||
|                           } | ||||
|                         )} | ||||
|                         <a target="_blank" href=${conf.authorUrl}> | ||||
|                           ${this.hass.localize( | ||||
|                             "ui.panel.page-demo.cards.demo.demo_by", | ||||
|                             { name: conf.authorName } | ||||
|                           )} | ||||
|                         </a> | ||||
|                       </small> | ||||
|                     ` | ||||
|                   ), | ||||
|                   "" | ||||
|                 )} | ||||
|           </div> | ||||
|  | ||||
|           <ha-button @click=${this._nextConfig} .disabled=${this._switching}> | ||||
|           <mwc-button @click=${this._nextConfig} .disabled=${this._switching}> | ||||
|             ${this.hass.localize("ui.panel.page-demo.cards.demo.next_demo")} | ||||
|           </ha-button> | ||||
|           </mwc-button> | ||||
|         </div> | ||||
|         <div class="content"> | ||||
|           <p class="small-hidden"> | ||||
|             ${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")} | ||||
|           </p> | ||||
|           ${until( | ||||
|             selectedDemoConfig.then((conf) => { | ||||
|               if (typeof conf.description === "function") { | ||||
|                 return conf.description(this.hass.localize); | ||||
|               } | ||||
|               if (conf.description) { | ||||
|                 return html`<p>${conf.description}</p>`; | ||||
|               } | ||||
|               return nothing; | ||||
|             }), | ||||
|             nothing | ||||
|           )} | ||||
|         <div class="content small-hidden"> | ||||
|           ${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")} | ||||
|         </div> | ||||
|         <div class="actions small-hidden"> | ||||
|           <a href="https://www.home-assistant.io" target="_blank"> | ||||
|             <ha-button> | ||||
|             <mwc-button> | ||||
|               ${this.hass.localize("ui.panel.page-demo.cards.demo.learn_more")} | ||||
|             </ha-button> | ||||
|             </mwc-button> | ||||
|           </a> | ||||
|         </div> | ||||
|       </ha-card> | ||||
| @@ -113,7 +94,13 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|  | ||||
|   private async _updateConfig(index: number) { | ||||
|     this._switching = true; | ||||
|     fireEvent(this, "set-demo-config" as any, { index }); | ||||
|     try { | ||||
|       await setDemoConfig(this.hass, this.lovelace!, index); | ||||
|     } catch (err: any) { | ||||
|       alert("Failed to switch config :-("); | ||||
|     } finally { | ||||
|       this._switching = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
| @@ -121,7 +108,6 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|       css` | ||||
|         a { | ||||
|           color: var(--primary-color); | ||||
|           display: inline-block; | ||||
|         } | ||||
|  | ||||
|         .actions a { | ||||
| @@ -129,11 +115,7 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|         } | ||||
|  | ||||
|         .content { | ||||
|           padding: 0 16px; | ||||
|         } | ||||
|  | ||||
|         .content p { | ||||
|           margin: 16px 0; | ||||
|           padding: 16px; | ||||
|         } | ||||
|  | ||||
|         .picker { | ||||
| @@ -143,7 +125,7 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|           height: 60px; | ||||
|         } | ||||
|  | ||||
|         .picker ha-button { | ||||
|         .picker mwc-button { | ||||
|           margin-right: 8px; | ||||
|         } | ||||
|  | ||||
| @@ -156,8 +138,9 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|         } | ||||
|  | ||||
|         .actions { | ||||
|           padding: 0px 8px 4px 8px; | ||||
|           padding-left: 8px; | ||||
|         } | ||||
|  | ||||
|         @media only screen and (max-width: 500px) { | ||||
|           .small-hidden { | ||||
|             display: none; | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import "./util/is_frontpage"; | ||||
| import "../../src/resources/safari-14-attachshadow-patch"; | ||||
| import "./ha-demo"; | ||||
|  | ||||
| import("../../src/resources/ha-style"); | ||||
|   | ||||
| @@ -10,7 +10,6 @@ import { | ||||
| import { HomeAssistantAppEl } from "../../src/layouts/home-assistant"; | ||||
| import { HomeAssistant } from "../../src/types"; | ||||
| import { selectedDemoConfig } from "./configs/demo-configs"; | ||||
| import { mockAreaRegistry } from "./stubs/area_registry"; | ||||
| import { mockAuth } from "./stubs/auth"; | ||||
| import { mockConfigEntries } from "./stubs/config_entries"; | ||||
| import { mockEnergy } from "./stubs/energy"; | ||||
| @@ -24,10 +23,10 @@ import { mockLovelace } from "./stubs/lovelace"; | ||||
| import { mockMediaPlayer } from "./stubs/media_player"; | ||||
| import { mockPersistentNotification } from "./stubs/persistent_notification"; | ||||
| import { mockRecorder } from "./stubs/recorder"; | ||||
| import { mockTodo } from "./stubs/todo"; | ||||
| import { mockSensor } from "./stubs/sensor"; | ||||
| import { mockSystemLog } from "./stubs/system_log"; | ||||
| import { mockTemplate } from "./stubs/template"; | ||||
| import { mockTodo } from "./stubs/todo"; | ||||
| import { mockTranslations } from "./stubs/translations"; | ||||
|  | ||||
| @customElement("ha-demo") | ||||
| @@ -63,7 +62,6 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|     mockEnergy(hass); | ||||
|     mockPersistentNotification(hass); | ||||
|     mockConfigEntries(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockEntityRegistry(hass, [ | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
| @@ -74,16 +72,12 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         id: "sensor.co2_intensity", | ||||
|         name: null, | ||||
|         icon: null, | ||||
|         labels: [], | ||||
|         categories: {}, | ||||
|         platform: "co2signal", | ||||
|         hidden_by: null, | ||||
|         entity_category: null, | ||||
|         has_entity_name: false, | ||||
|         unique_id: "co2_intensity", | ||||
|         options: null, | ||||
|         created_at: 0, | ||||
|         modified_at: 0, | ||||
|       }, | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
| @@ -94,16 +88,12 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         id: "sensor.co2_intensity", | ||||
|         name: null, | ||||
|         icon: null, | ||||
|         labels: [], | ||||
|         categories: {}, | ||||
|         platform: "co2signal", | ||||
|         hidden_by: null, | ||||
|         entity_category: null, | ||||
|         has_entity_name: false, | ||||
|         unique_id: "grid_fossil_fuel_percentage", | ||||
|         options: null, | ||||
|         created_at: 0, | ||||
|         modified_at: 0, | ||||
|       }, | ||||
|     ]); | ||||
|  | ||||
|   | ||||
| @@ -63,47 +63,46 @@ | ||||
|         align-items: center; | ||||
|       } | ||||
|       #ha-launch-screen svg { | ||||
|         width: 112px; | ||||
|         width: 170px; | ||||
|         flex-shrink: 0; | ||||
|       } | ||||
|       #ha-launch-screen .ha-launch-screen-spacer-top { | ||||
|       #ha-launch-screen .ha-launch-screen-spacer { | ||||
|         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-top"></div> | ||||
|       <div class="ha-launch-screen-spacer"></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-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 id="ha-launch-screen-info-box" class="ha-launch-screen-spacer"></div> | ||||
|     </div> | ||||
|     <ha-demo></ha-demo> | ||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_preload_roboto.html.template") %> | ||||
|     <%= renderTemplate("../../../src/html/_script_loader.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") %> | ||||
|     <script> | ||||
|       var _gaq = [["_setAccount", "UA-57927901-5"], ["_trackPageview"]]; | ||||
|       (function (d, t) { | ||||
|         var g = d.createElement(t), | ||||
|           s = d.getElementsByTagName(t)[0]; | ||||
|         g.src = | ||||
|           ("https:" == location.protocol ? "//ssl" : "//www") + | ||||
|           ".google-analytics.com/ga.js"; | ||||
|         s.parentNode.insertBefore(g, s); | ||||
|       })(document, "script"); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
|   | ||||
| @@ -4,11 +4,4 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| export const mockAreaRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: AreaRegistryEntry[] = [] | ||||
| ) => { | ||||
|   hass.mockWS("config/area_registry/list", () => data); | ||||
|   const areas = {}; | ||||
|   data.forEach((area) => { | ||||
|     areas[area.area_id] = area; | ||||
|   }); | ||||
|   hass.updateHass({ areas }); | ||||
| }; | ||||
| ) => hass.mockWS("config/area_registry/list", () => data); | ||||
|   | ||||
| @@ -10,7 +10,6 @@ export const mockConfigEntries = (hass: MockHomeAssistant) => { | ||||
|     supports_options: false, | ||||
|     supports_remove_device: false, | ||||
|     supports_unload: true, | ||||
|     supports_reconfigure: true, | ||||
|     pref_disable_new_entities: false, | ||||
|     pref_disable_polling: false, | ||||
|     disabled_by: null, | ||||
|   | ||||
| @@ -4,11 +4,4 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| export const mockDeviceRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: DeviceRegistryEntry[] = [] | ||||
| ) => { | ||||
|   hass.mockWS("config/device_registry/list", () => data); | ||||
|   const devices = {}; | ||||
|   data.forEach((device) => { | ||||
|     devices[device.id] = device; | ||||
|   }); | ||||
|   hass.updateHass({ devices }); | ||||
| }; | ||||
| ) => hass.mockWS("config/device_registry/list", () => data); | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { format, startOfToday, startOfTomorrow } from "date-fns"; | ||||
| import { format, startOfToday, startOfTomorrow } from "date-fns/esm"; | ||||
| import { | ||||
|   EnergyInfo, | ||||
|   EnergyPreferences, | ||||
|   | ||||
| @@ -1,55 +1,5 @@ | ||||
| import { convertEntities } from "../../../src/fake_data/entity"; | ||||
|  | ||||
| export const mapEntities = () => | ||||
|   convertEntities({ | ||||
|     "zone.home": { | ||||
|       entity_id: "zone.home", | ||||
|       state: "zoning", | ||||
|       attributes: { | ||||
|         hidden: true, | ||||
|         latitude: 52.3631339, | ||||
|         longitude: 4.8903147, | ||||
|         radius: 200, | ||||
|         friendly_name: "Home", | ||||
|         icon: "hademo:home", | ||||
|       }, | ||||
|     }, | ||||
|     "zone.uva": { | ||||
|       entity_id: "zone.buckhead", | ||||
|       state: "zoning", | ||||
|       attributes: { | ||||
|         hidden: true, | ||||
|         radius: 400, | ||||
|         friendly_name: "UvA", | ||||
|         icon: "hademo:school", | ||||
|         latitude: 52.3558182, | ||||
|         longitude: 4.9535376, | ||||
|       }, | ||||
|     }, | ||||
|     "person.arsaboo": { | ||||
|       entity_id: "person.arsaboo", | ||||
|       state: "not_home", | ||||
|       attributes: { | ||||
|         radius: 50, | ||||
|         friendly_name: "Arsaboo", | ||||
|         latitude: 52.3579946, | ||||
|         longitude: 4.8664597, | ||||
|         entity_picture: "/assets/arsaboo/images/arsaboo.jpg", | ||||
|       }, | ||||
|     }, | ||||
|     "person.melody": { | ||||
|       entity_id: "person.melody", | ||||
|       state: "not_home", | ||||
|       attributes: { | ||||
|         radius: 50, | ||||
|         friendly_name: "Melody", | ||||
|         latitude: 52.3408927, | ||||
|         longitude: 4.8711073, | ||||
|         entity_picture: "/assets/arsaboo/images/melody.jpg", | ||||
|       }, | ||||
|     }, | ||||
|   }); | ||||
|  | ||||
| export const energyEntities = () => | ||||
|   convertEntities({ | ||||
|     "sensor.grid_fossil_fuel_percentage": { | ||||
|   | ||||
| @@ -1,7 +0,0 @@ | ||||
| import { FloorRegistryEntry } from "../../../src/data/floor_registry"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockFloorRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: FloorRegistryEntry[] = [] | ||||
| ) => hass.mockWS("config/floor_registry/list", () => data); | ||||
| @@ -1,7 +0,0 @@ | ||||
| import { LabelRegistryEntry } from "../../../src/data/label_registry"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockLabelRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: LabelRegistryEntry[] = [] | ||||
| ) => hass.mockWS("config/label_registry/list", () => data); | ||||
| @@ -1,52 +1,35 @@ | ||||
| import type { LocalizeFunc } from "../../../src/common/translations/localize"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| import { | ||||
|   selectedDemoConfig, | ||||
|   selectedDemoConfigIndex, | ||||
|   setDemoConfig, | ||||
| } from "../configs/demo-configs"; | ||||
| import { selectedDemoConfig } from "../configs/demo-configs"; | ||||
| import "../custom-cards/cast-demo-row"; | ||||
| import "../custom-cards/ha-demo-card"; | ||||
| import { mapEntities } from "./entities"; | ||||
| import type { HADemoCard } from "../custom-cards/ha-demo-card"; | ||||
|  | ||||
| export const mockLovelace = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   localizePromise: Promise<LocalizeFunc> | ||||
| ) => { | ||||
|   hass.mockWS("lovelace/config", ({ url_path }) => { | ||||
|     if (url_path === "map") { | ||||
|       hass.addEntities(mapEntities()); | ||||
|       return { | ||||
|         strategy: { | ||||
|           type: "map", | ||||
|         }, | ||||
|       }; | ||||
|     } | ||||
|     return Promise.all([selectedDemoConfig, localizePromise]).then( | ||||
|   hass.mockWS("lovelace/config", () => | ||||
|     Promise.all([selectedDemoConfig, localizePromise]).then( | ||||
|       ([config, localize]) => config.lovelace(localize) | ||||
|     ); | ||||
|   }); | ||||
|     ) | ||||
|   ); | ||||
|  | ||||
|   hass.mockWS("lovelace/config/save", () => Promise.resolve()); | ||||
|   hass.mockWS("lovelace/resources", () => Promise.resolve([])); | ||||
| }; | ||||
|  | ||||
| customElements.whenDefined("hui-root").then(() => { | ||||
| customElements.whenDefined("hui-view").then(() => { | ||||
|   // eslint-disable-next-line | ||||
|   const HUIRoot = customElements.get("hui-root")!; | ||||
|   const HUIView = customElements.get("hui-view"); | ||||
|   // Patch HUI-VIEW to make the lovelace object available to the demo card | ||||
|   const oldCreateCard = HUIView!.prototype.createCardElement; | ||||
|  | ||||
|   const oldFirstUpdated = HUIRoot.prototype.firstUpdated; | ||||
|  | ||||
|   HUIRoot.prototype.firstUpdated = function (changedProperties) { | ||||
|     oldFirstUpdated.call(this, changedProperties); | ||||
|     this.addEventListener("set-demo-config", async (ev) => { | ||||
|       const index = (ev as CustomEvent).detail.index; | ||||
|       try { | ||||
|         await setDemoConfig(this.hass, this.lovelace!, index); | ||||
|       } catch (err: any) { | ||||
|         setDemoConfig(this.hass, this.lovelace!, selectedDemoConfigIndex); | ||||
|         alert("Failed to switch config :-("); | ||||
|       } | ||||
|     }); | ||||
|   HUIView!.prototype.createCardElement = function (config) { | ||||
|     const el = oldCreateCard.call(this, config); | ||||
|     if (el.tagName === "HA-DEMO-CARD") { | ||||
|       (el as HADemoCard).lovelace = this.lovelace; | ||||
|     } | ||||
|     return el; | ||||
|   }; | ||||
| }); | ||||
|   | ||||
| @@ -1 +0,0 @@ | ||||
| export const isFrontpageEmbed = document.location.search === "?frontpage"; | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 110 KiB | 
| @@ -1,9 +1,7 @@ | ||||
| import { load } from "js-yaml"; | ||||
| import { LitElement, PropertyValueMap, css, html, nothing } from "lit"; | ||||
| import { html, css, LitElement, PropertyValues } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import "../../../src/panels/lovelace/cards/hui-card"; | ||||
| import type { HuiCard } from "../../../src/panels/lovelace/cards/hui-card"; | ||||
| import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
|  | ||||
| export interface DemoCardConfig { | ||||
| @@ -21,12 +19,7 @@ class DemoCard extends LitElement { | ||||
|  | ||||
|   @state() private _size?: number; | ||||
|  | ||||
|   @query("hui-card", false) private _card?: HuiCard; | ||||
|  | ||||
|   private _config = memoizeOne((config: string) => { | ||||
|     const c = (load(config) as any)[0]; | ||||
|     return c; | ||||
|   }); | ||||
|   @query("#card") private _card!: HTMLElement; | ||||
|  | ||||
|   render() { | ||||
|     return html` | ||||
| @@ -37,32 +30,63 @@ class DemoCard extends LitElement { | ||||
|           : ""} | ||||
|       </h2> | ||||
|       <div class="root"> | ||||
|         <hui-card | ||||
|           .config=${this._config(this.config.config)} | ||||
|           .hass=${this.hass} | ||||
|           @card-updated=${this._cardUpdated} | ||||
|         ></hui-card> | ||||
|         ${this.showConfig | ||||
|           ? html`<pre>${this.config.config.trim()}</pre>` | ||||
|           : nothing} | ||||
|         <div id="card"></div> | ||||
|         ${this.showConfig ? html`<pre>${this.config.config.trim()}</pre>` : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private async _cardUpdated(ev) { | ||||
|     ev.stopPropagation(); | ||||
|     this._updateSize(); | ||||
|   updated(changedProps: PropertyValues) { | ||||
|     super.updated(changedProps); | ||||
|  | ||||
|     if (changedProps.has("config")) { | ||||
|       const card = this._card; | ||||
|       while (card.lastChild) { | ||||
|         card.removeChild(card.lastChild); | ||||
|       } | ||||
|  | ||||
|       const el = this._createCardElement((load(this.config.config) as any)[0]); | ||||
|       card.appendChild(el); | ||||
|       this._getSize(el); | ||||
|     } | ||||
|  | ||||
|     if (changedProps.has("hass")) { | ||||
|       const card = this._card.lastChild; | ||||
|       if (card) { | ||||
|         (card as any).hass = this.hass; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _updateSize() { | ||||
|     this._size = await this._card?.getCardSize(); | ||||
|   async _getSize(el) { | ||||
|     await customElements.whenDefined(el.localName); | ||||
|  | ||||
|     if (!("getCardSize" in el)) { | ||||
|       this._size = undefined; | ||||
|       return; | ||||
|     } | ||||
|     this._size = await el.getCardSize(); | ||||
|   } | ||||
|  | ||||
|   protected update( | ||||
|     _changedProperties: PropertyValueMap<any> | Map<PropertyKey, unknown> | ||||
|   ): void { | ||||
|     super.update(_changedProperties); | ||||
|     this._updateSize(); | ||||
|   _createCardElement(cardConfig) { | ||||
|     const element = createCardElement(cardConfig); | ||||
|     if (this.hass) { | ||||
|       element.hass = this.hass; | ||||
|     } | ||||
|     element.addEventListener( | ||||
|       "ll-rebuild", | ||||
|       (ev) => { | ||||
|         ev.stopPropagation(); | ||||
|         this._rebuildCard(element, cardConfig); | ||||
|       }, | ||||
|       { once: true } | ||||
|     ); | ||||
|     return element; | ||||
|   } | ||||
|  | ||||
|   _rebuildCard(cardElToReplace, config) { | ||||
|     const newCardEl = this._createCardElement(config); | ||||
|     cardElToReplace.parentElement.replaceChild(newCardEl, cardElToReplace); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
| @@ -77,7 +101,7 @@ class DemoCard extends LitElement { | ||||
|       font-size: 0.5em; | ||||
|       color: var(--primary-text-color); | ||||
|     } | ||||
|     hui-card { | ||||
|     #card { | ||||
|       max-width: 400px; | ||||
|       width: 100vw; | ||||
|     } | ||||
|   | ||||
| @@ -532,6 +532,15 @@ 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", | ||||
|   | ||||
| @@ -17,7 +17,6 @@ export const basicTrace: DemoTrace = { | ||||
|         { | ||||
|           path: "trigger/0", | ||||
|           timestamp: "2021-03-25T04:36:51.223693+00:00", | ||||
|           changed_variables: {}, | ||||
|         }, | ||||
|       ], | ||||
|       "condition/0": [ | ||||
|   | ||||
| @@ -17,7 +17,6 @@ export const motionLightTrace: DemoTrace = { | ||||
|         { | ||||
|           path: "trigger/0", | ||||
|           timestamp: "2021-03-25T04:36:51.223693+00:00", | ||||
|           changed_variables: {}, | ||||
|         }, | ||||
|       ], | ||||
|       "action/0": [ | ||||
|   | ||||
| @@ -3,16 +3,13 @@ title: When to use remove, delete, add and create | ||||
| subtitle: The difference between remove/delete and add/create. | ||||
| --- | ||||
|  | ||||
| # Removing or deleting content | ||||
| # Remove vs Delete | ||||
|  | ||||
| _Remove_ and _Delete_ are quite similar, but can be frustrating if used inconsistently. | ||||
|  | ||||
| - Remove refers to an action that can be restored or reapplied. | ||||
| - Delete refers to a permanent, non-recoverable action. | ||||
| Remove and Delete are quite similar, but can be frustrating if used inconsistently. | ||||
|  | ||||
| ## Remove | ||||
|  | ||||
| The term _Remove_ should always be used when an item/setting or content is to be removed or disassociated, but the action can be reversed or reapplied. | ||||
| Take away and set aside, but kept in existence. | ||||
|  | ||||
| For example: | ||||
|  | ||||
| @@ -25,7 +22,7 @@ For example: | ||||
|  | ||||
| ## Delete | ||||
|  | ||||
| The term _Delete_ should always be used to refer to any action that will cause the permanent deletion of an item/setting or content. | ||||
| Erase, rendered nonexistent or nonrecoverable. | ||||
|  | ||||
| For example: | ||||
|  | ||||
|   | ||||
| @@ -64,12 +64,6 @@ const ACTIONS = [ | ||||
|       entity_id: "input_boolean.toggle_4", | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     sequence: [ | ||||
|       { scene: "scene.kitchen_morning" }, | ||||
|       { service: "light.turn_off", target: { entity_id: "light.kitchen" } }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     parallel: [ | ||||
|       { scene: "scene.kitchen_morning" }, | ||||
| @@ -142,7 +136,7 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|         <div class="action"> | ||||
|           <span> | ||||
|             ${this._action | ||||
|               ? describeAction(this.hass, [], [], [], this._action) | ||||
|               ? describeAction(this.hass, [], this._action) | ||||
|               : "<invalid YAML>"} | ||||
|           </span> | ||||
|           <ha-yaml-editor | ||||
| @@ -155,7 +149,7 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|         ${ACTIONS.map( | ||||
|           (conf) => html` | ||||
|             <div class="action"> | ||||
|               <span>${describeAction(this.hass, [], [], [], conf as any)}</span> | ||||
|               <span>${describeAction(this.hass, [], conf as any)}</span> | ||||
|               <pre>${dump(conf)}</pre> | ||||
|             </div> | ||||
|           ` | ||||
|   | ||||
| @@ -21,10 +21,10 @@ const ENTITIES = [ | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const conditions: Condition[] = [ | ||||
|   { condition: "and", conditions: [] }, | ||||
|   { condition: "not", conditions: [] }, | ||||
|   { condition: "or", conditions: [] }, | ||||
| const conditions = [ | ||||
|   { condition: "and" }, | ||||
|   { condition: "not" }, | ||||
|   { condition: "or" }, | ||||
|   { condition: "state", entity_id: "light.kitchen", state: "on" }, | ||||
|   { | ||||
|     condition: "numeric_state", | ||||
| @@ -34,11 +34,11 @@ const conditions: Condition[] = [ | ||||
|     above: 20, | ||||
|   }, | ||||
|   { condition: "sun", after: "sunset" }, | ||||
|   { condition: "sun", after: "sunrise", before_offset: 3600 }, | ||||
|   { condition: "sun", after: "sunrise", offset: "-01:00" }, | ||||
|   { condition: "zone", entity_id: "device_tracker.person", zone: "zone.home" }, | ||||
|   { condition: "trigger", id: "motion" }, | ||||
|   { condition: "time" }, | ||||
|   { condition: "template", value_template: "" }, | ||||
|   { condition: "template" }, | ||||
| ]; | ||||
|  | ||||
| const initialCondition: Condition = { | ||||
|   | ||||
| @@ -20,7 +20,6 @@ import { HaWaitForTriggerAction } from "../../../../src/panels/config/automation | ||||
| import { HaWaitAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | ||||
| import { Action } from "../../../../src/data/script"; | ||||
| import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
| import { HaSequenceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-sequence"; | ||||
| import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | ||||
| import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | ||||
| import { HaStopAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-stop"; | ||||
| @@ -40,7 +39,6 @@ const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
|   { name: "If-Then", actions: [HaIfAction.defaultConfig] }, | ||||
|   { name: "Choose", actions: [HaChooseAction.defaultConfig] }, | ||||
|   { name: "Variables", actions: [{ variables: { hello: "1" } }] }, | ||||
|   { name: "Sequence", actions: [HaSequenceAction.defaultConfig] }, | ||||
|   { name: "Parallel", actions: [HaParallelAction.defaultConfig] }, | ||||
|   { name: "Stop", actions: [HaStopAction.defaultConfig] }, | ||||
| ]; | ||||
|   | ||||
| @@ -11,6 +11,7 @@ import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervis | ||||
| import type { ConditionWithShorthand } from "../../../../src/data/automation"; | ||||
| import "../../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||
| import { HaDeviceCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||
| import { HaLogicalCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | ||||
| import HaNumericStateCondition from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-numeric_state"; | ||||
| import { HaStateCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-state"; | ||||
| import { HaSunCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-sun"; | ||||
| @@ -18,67 +19,62 @@ import { HaTemplateCondition } from "../../../../src/panels/config/automation/co | ||||
| import { HaTimeCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-time"; | ||||
| import { HaTriggerCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||
| import { HaZoneCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||
| import { HaAndCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-and"; | ||||
| import { HaOrCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-or"; | ||||
| import { HaNotCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-not"; | ||||
|  | ||||
| const SCHEMAS: { name: string; conditions: ConditionWithShorthand[] }[] = [ | ||||
|   { | ||||
|     name: "State", | ||||
|     conditions: [{ ...HaStateCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "state", ...HaStateCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Numeric State", | ||||
|     conditions: [{ ...HaNumericStateCondition.defaultConfig }], | ||||
|     conditions: [ | ||||
|       { condition: "numeric_state", ...HaNumericStateCondition.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     name: "Sun", | ||||
|     conditions: [{ ...HaSunCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "sun", ...HaSunCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Zone", | ||||
|     conditions: [{ ...HaZoneCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "zone", ...HaZoneCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Time", | ||||
|     conditions: [{ ...HaTimeCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "time", ...HaTimeCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Template", | ||||
|     conditions: [{ ...HaTemplateCondition.defaultConfig }], | ||||
|     conditions: [ | ||||
|       { condition: "template", ...HaTemplateCondition.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     name: "Device", | ||||
|     conditions: [{ ...HaDeviceCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "device", ...HaDeviceCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "And", | ||||
|     conditions: [{ ...HaAndCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "and", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Or", | ||||
|     conditions: [{ ...HaOrCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "or", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Not", | ||||
|     conditions: [{ ...HaNotCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "not", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Trigger", | ||||
|     conditions: [{ ...HaTriggerCondition.defaultConfig }], | ||||
|     conditions: [{ condition: "trigger", ...HaTriggerCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Shorthand", | ||||
|     conditions: [ | ||||
|       { | ||||
|         ...HaAndCondition.defaultConfig, | ||||
|       }, | ||||
|       { | ||||
|         ...HaOrCondition.defaultConfig, | ||||
|       }, | ||||
|       { | ||||
|         ...HaNotCondition.defaultConfig, | ||||
|       }, | ||||
|       { and: HaLogicalCondition.defaultConfig.conditions }, | ||||
|       { or: HaLogicalCondition.defaultConfig.conditions }, | ||||
|       { not: HaLogicalCondition.defaultConfig.conditions }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -30,48 +30,55 @@ import { HaConversationTrigger } from "../../../../src/panels/config/automation/ | ||||
| const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
|   { | ||||
|     name: "State", | ||||
|     triggers: [{ ...HaStateTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "state", ...HaStateTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "MQTT", | ||||
|     triggers: [{ ...HaMQTTTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "mqtt", ...HaMQTTTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "GeoLocation", | ||||
|     triggers: [{ ...HaGeolocationTrigger.defaultConfig }], | ||||
|     triggers: [ | ||||
|       { platform: "geo_location", ...HaGeolocationTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Home Assistant", | ||||
|     triggers: [{ ...HaHassTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "homeassistant", ...HaHassTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Numeric State", | ||||
|     triggers: [{ ...HaNumericStateTrigger.defaultConfig }], | ||||
|     triggers: [ | ||||
|       { platform: "numeric_state", ...HaNumericStateTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Sun", | ||||
|     triggers: [{ ...HaSunTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "sun", ...HaSunTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Time Pattern", | ||||
|     triggers: [{ ...HaTimePatternTrigger.defaultConfig }], | ||||
|     triggers: [ | ||||
|       { platform: "time_pattern", ...HaTimePatternTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Webhook", | ||||
|     triggers: [{ ...HaWebhookTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "webhook", ...HaWebhookTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Persistent Notification", | ||||
|     triggers: [ | ||||
|       { | ||||
|         platform: "persistent_notification", | ||||
|         ...HaPersistentNotificationTrigger.defaultConfig, | ||||
|       }, | ||||
|     ], | ||||
| @@ -79,37 +86,37 @@ const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
|  | ||||
|   { | ||||
|     name: "Zone", | ||||
|     triggers: [{ ...HaZoneTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "zone", ...HaZoneTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Tag", | ||||
|     triggers: [{ ...HaTagTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "tag", ...HaTagTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Time", | ||||
|     triggers: [{ ...HaTimeTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "time", ...HaTimeTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Template", | ||||
|     triggers: [{ ...HaTemplateTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "template", ...HaTemplateTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Event", | ||||
|     triggers: [{ ...HaEventTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "event", ...HaEventTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Device Trigger", | ||||
|     triggers: [{ ...HaDeviceTrigger.defaultConfig }], | ||||
|     triggers: [{ platform: "device", ...HaDeviceTrigger.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Sentence", | ||||
|     triggers: [ | ||||
|       { ...HaConversationTrigger.defaultConfig }, | ||||
|       { platform: "conversation", ...HaConversationTrigger.defaultConfig }, | ||||
|       { | ||||
|         platform: "conversation", | ||||
|         command: ["Turn on the lights", "Turn the lights on"], | ||||
|   | ||||
| @@ -55,7 +55,6 @@ export class DemoAutomationTraceTimeline extends LitElement { | ||||
|     super.firstUpdated(changedProps); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|   | ||||
| @@ -60,7 +60,6 @@ export class DemoAutomationTrace extends LitElement { | ||||
|     super.firstUpdated(changedProps); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|   | ||||
| @@ -162,7 +162,7 @@ export class DemoHaBarButton extends LitElement { | ||||
|       } | ||||
|       .custom-group { | ||||
|         --control-button-group-thickness: 100px; | ||||
|         --control-button-group-border-radius: 36px; | ||||
|         --control-button-group-border-radius: 18px; | ||||
|         --control-button-group-spacing: 20px; | ||||
|       } | ||||
|       .custom-group ha-control-button { | ||||
|   | ||||
| @@ -94,7 +94,7 @@ export class DemoHarControlNumberButtons extends LitElement { | ||||
|         --control-number-buttons-background-color: #2196f3; | ||||
|         --control-number-buttons-background-opacity: 0.1; | ||||
|         --control-number-buttons-thickness: 100px; | ||||
|         --control-number-buttons-border-radius: 36px; | ||||
|         --control-number-buttons-border-radius: 24px; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -186,8 +186,8 @@ export class DemoHaControlSelect extends LitElement { | ||||
|       .custom { | ||||
|         --mdc-icon-size: 24px; | ||||
|         --control-select-color: var(--state-fan-active-color); | ||||
|         --control-select-thickness: 130px; | ||||
|         --control-select-border-radius: 36px; | ||||
|         --control-select-thickness: 100px; | ||||
|         --control-select-border-radius: 24px; | ||||
|       } | ||||
|       .vertical-selects { | ||||
|         height: 300px; | ||||
|   | ||||
| @@ -150,8 +150,8 @@ export class DemoHaBarSlider extends LitElement { | ||||
|         --control-slider-color: #ffcf4c; | ||||
|         --control-slider-background: #ffcf4c; | ||||
|         --control-slider-background-opacity: 0.2; | ||||
|         --control-slider-thickness: 130px; | ||||
|         --control-slider-border-radius: 36px; | ||||
|         --control-slider-thickness: 100px; | ||||
|         --control-slider-border-radius: 24px; | ||||
|       } | ||||
|       .vertical-sliders { | ||||
|         height: 300px; | ||||
|   | ||||
| @@ -117,8 +117,8 @@ export class DemoHaControlSwitch extends LitElement { | ||||
|       .custom { | ||||
|         --control-switch-on-color: var(--green-color); | ||||
|         --control-switch-off-color: var(--red-color); | ||||
|         --control-switch-thickness: 130px; | ||||
|         --control-switch-border-radius: 36px; | ||||
|         --control-switch-thickness: 100px; | ||||
|         --control-switch-border-radius: 24px; | ||||
|         --control-switch-padding: 6px; | ||||
|         --mdc-icon-size: 24px; | ||||
|       } | ||||
|   | ||||
| @@ -15,7 +15,6 @@ 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", { | ||||
| @@ -42,7 +41,7 @@ const ENTITIES = [ | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES: DeviceRegistryEntry[] = [ | ||||
| const DEVICES = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
| @@ -54,17 +53,12 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "volume1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: null, | ||||
|     name: "Dishwasher", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
| @@ -77,17 +71,12 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: null, | ||||
|     name: "Lamp", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
| @@ -100,53 +89,36 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: "User name", | ||||
|     name: "Technical name", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const AREAS: AreaRegistryEntry[] = [ | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     floor_id: null, | ||||
|     name: "Backyard", | ||||
|     icon: null, | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     floor_id: null, | ||||
|     name: "Bedroom", | ||||
|     icon: "mdi:bed", | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
|     floor_id: null, | ||||
|     name: "Livingroom", | ||||
|     icon: "mdi:sofa", | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -17,11 +17,6 @@ import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| 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", { | ||||
| @@ -42,7 +37,7 @@ const ENTITIES = [ | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES: DeviceRegistryEntry[] = [ | ||||
| const DEVICES = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
| @@ -54,17 +49,12 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "volume1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: null, | ||||
|     name: "Dishwasher", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
| @@ -77,17 +67,12 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: null, | ||||
|     name: "Lamp", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
| @@ -100,104 +85,36 @@ const DEVICES: DeviceRegistryEntry[] = [ | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     model_id: null, | ||||
|     name_by_user: "User name", | ||||
|     name: "Technical name", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|     serial_number: null, | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|     primary_config_entry: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const AREAS: AreaRegistryEntry[] = [ | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     floor_id: "ground", | ||||
|     name: "Backyard", | ||||
|     icon: null, | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     floor_id: "first", | ||||
|     name: "Bedroom", | ||||
|     icon: "mdi:bed", | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
|     floor_id: "ground", | ||||
|     name: "Livingroom", | ||||
|     icon: "mdi:sofa", | ||||
|     picture: null, | ||||
|     aliases: [], | ||||
|     labels: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const FLOORS: FloorRegistryEntry[] = [ | ||||
|   { | ||||
|     floor_id: "ground", | ||||
|     name: "Ground floor", | ||||
|     level: 0, | ||||
|     icon: null, | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     floor_id: "first", | ||||
|     name: "First floor", | ||||
|     level: 1, | ||||
|     icon: "mdi:numeric-1", | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     floor_id: "second", | ||||
|     name: "Second floor", | ||||
|     level: 2, | ||||
|     icon: "mdi:numeric-2", | ||||
|     aliases: [], | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const LABELS: LabelRegistryEntry[] = [ | ||||
|   { | ||||
|     label_id: "energy", | ||||
|     name: "Energy", | ||||
|     icon: null, | ||||
|     color: "yellow", | ||||
|     description: null, | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
|   { | ||||
|     label_id: "entertainment", | ||||
|     name: "Entertainment", | ||||
|     icon: "mdi:popcorn", | ||||
|     color: "blue", | ||||
|     description: null, | ||||
|     created_at: 0, | ||||
|     modified_at: 0, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @@ -208,12 +125,7 @@ const SCHEMAS: { | ||||
|   { | ||||
|     name: "One of each", | ||||
|     input: { | ||||
|       label: { name: "Label", selector: { label: {} } }, | ||||
|       floor: { name: "Floor", selector: { floor: {} } }, | ||||
|       area: { name: "Area", selector: { area: {} } }, | ||||
|       device: { name: "Device", selector: { device: {} } }, | ||||
|       entity: { name: "Entity", selector: { entity: {} } }, | ||||
|       target: { name: "Target", selector: { target: {} } }, | ||||
|       state: { | ||||
|         name: "State", | ||||
|         selector: { state: { entity_id: "alarm_control_panel.alarm" } }, | ||||
| @@ -222,12 +134,15 @@ const SCHEMAS: { | ||||
|         name: "Attribute", | ||||
|         selector: { attribute: { entity_id: "" } }, | ||||
|       }, | ||||
|       device: { name: "Device", selector: { device: {} } }, | ||||
|       config_entry: { | ||||
|         name: "Integration", | ||||
|         selector: { config_entry: {} }, | ||||
|       }, | ||||
|       duration: { name: "Duration", selector: { duration: {} } }, | ||||
|       addon: { name: "Addon", selector: { addon: {} } }, | ||||
|       area: { name: "Area", selector: { area: {} } }, | ||||
|       target: { name: "Target", selector: { target: {} } }, | ||||
|       number_box: { | ||||
|         name: "Number Box", | ||||
|         selector: { | ||||
| @@ -376,8 +291,6 @@ const SCHEMAS: { | ||||
|       entity: { name: "Entity", selector: { entity: { multiple: true } } }, | ||||
|       device: { name: "Device", selector: { device: { multiple: true } } }, | ||||
|       area: { name: "Area", selector: { area: { multiple: true } } }, | ||||
|       floor: { name: "Floor", selector: { floor: { multiple: true } } }, | ||||
|       label: { name: "Label", selector: { label: { multiple: true } } }, | ||||
|       select: { | ||||
|         name: "Select Multiple", | ||||
|         selector: { | ||||
| @@ -434,8 +347,6 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | ||||
|     mockDeviceRegistry(hass, DEVICES); | ||||
|     mockConfigEntries(hass); | ||||
|     mockAreaRegistry(hass, AREAS); | ||||
|     mockFloorRegistry(hass, FLOORS); | ||||
|     mockLabelRegistry(hass, LABELS); | ||||
|     mockHassioSupervisor(hass); | ||||
|     hass.mockWS("auth/sign_path", (params) => params); | ||||
|     hass.mockWS("media_player/browse_media", this._browseMedia); | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeDateTimeNumeric extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeNumeric( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeNumeric( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeNumeric( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeNumeric( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeNumeric( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeNumeric( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeDateTimeSeconds extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeDateTimeShortYear extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTimeWithYear( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTimeWithYear( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTimeWithYear( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTimeWithYear( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTimeWithYear( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTimeWithYear( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeDateTimeShort extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatShortDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatShortDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeDateTime extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -35,57 +35,59 @@ export class DemoDateTimeDate extends LitElement { | ||||
|           <div class="center">Month-Day-Year</div> | ||||
|           <div class="center">Year-Month-Day</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateNumeric( | ||||
|                   date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     date_format: DateFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateNumeric( | ||||
|                     date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       date_format: DateFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateNumeric( | ||||
|                     date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       date_format: DateFormat.DMY, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateNumeric( | ||||
|                     date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       date_format: DateFormat.MDY, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatDateNumeric( | ||||
|                     date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       date_format: DateFormat.YMD, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateNumeric( | ||||
|                   date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     date_format: DateFormat.DMY, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateNumeric( | ||||
|                   date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     date_format: DateFormat.MDY, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatDateNumeric( | ||||
|                   date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     date_format: DateFormat.YMD, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeTimeSeconds extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWithSeconds( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWithSeconds( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeTimeWeekday extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWeekday( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWeekday( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWeekday( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTimeWeekday( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWeekday( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTimeWeekday( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -56,46 +56,48 @@ export class DemoDateTimeTime extends LitElement { | ||||
|           <div class="center">12 Hours</div> | ||||
|           <div class="center">24 Hours</div> | ||||
|         </div> | ||||
|         ${Object.entries(translationMetadata.translations).map( | ||||
|           ([key, value]) => html` | ||||
|             <div class="container"> | ||||
|               <div>${value.nativeName}</div> | ||||
|               <div class="center"> | ||||
|                 ${formatTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.language, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|         ${Object.entries(translationMetadata.translations) | ||||
|           .filter(([key, _]) => key !== "test") | ||||
|           .map( | ||||
|             ([key, value]) => html` | ||||
|               <div class="container"> | ||||
|                 <div>${value.nativeName}</div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.language, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.am_pm, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|                 <div class="center"> | ||||
|                   ${formatTime( | ||||
|                     this.date, | ||||
|                     { | ||||
|                       ...defaultLocale, | ||||
|                       language: key, | ||||
|                       time_format: TimeFormat.twenty_four, | ||||
|                     }, | ||||
|                     demoConfig | ||||
|                   )} | ||||
|                 </div> | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.am_pm, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|               <div class="center"> | ||||
|                 ${formatTime( | ||||
|                   this.date, | ||||
|                   { | ||||
|                     ...defaultLocale, | ||||
|                     language: key, | ||||
|                     time_format: TimeFormat.twenty_four, | ||||
|                   }, | ||||
|                   demoConfig | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ` | ||||
|         )} | ||||
|             ` | ||||
|           )} | ||||
|       </mwc-list> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -287,11 +287,11 @@ const CONFIGS = [ | ||||
|     config: ` | ||||
| - type: entities | ||||
|   entities: | ||||
|     - type: perform-action | ||||
|     - type: call-service | ||||
|       icon: mdi:power | ||||
|       name: Bed light | ||||
|       action_name: Toggle light | ||||
|       action: light.toggle | ||||
|       service: light.toggle | ||||
|       data: | ||||
|         entity_id: light.bed_light | ||||
|     - type: section | ||||
|   | ||||
| @@ -11,7 +11,7 @@ const ENTITIES = [ | ||||
|     latitude: 32.877105, | ||||
|     longitude: 117.232185, | ||||
|     gps_accuracy: 91, | ||||
|     battery: 25, | ||||
|     battery: 71, | ||||
|     friendly_name: "Paulus", | ||||
|   }), | ||||
|   getEntity("device_tracker", "demo_anne_therese", "school", { | ||||
| @@ -19,7 +19,7 @@ const ENTITIES = [ | ||||
|     latitude: 32.877105, | ||||
|     longitude: 117.232185, | ||||
|     gps_accuracy: 91, | ||||
|     battery: 50, | ||||
|     battery: 71, | ||||
|     friendly_name: "Anne Therese", | ||||
|   }), | ||||
|   getEntity("device_tracker", "demo_home_boy", "home", { | ||||
| @@ -27,7 +27,7 @@ const ENTITIES = [ | ||||
|     latitude: 32.877105, | ||||
|     longitude: 117.232185, | ||||
|     gps_accuracy: 91, | ||||
|     battery: 75, | ||||
|     battery: 71, | ||||
|     friendly_name: "Home Boy", | ||||
|   }), | ||||
|   getEntity("light", "bed_light", "on", { | ||||
| @@ -39,53 +39,21 @@ const ENTITIES = [ | ||||
|   getEntity("light", "ceiling_lights", "off", { | ||||
|     friendly_name: "Ceiling Lights", | ||||
|   }), | ||||
|   getEntity("sensor", "battery_1", 20, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery 1", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("sensor", "battery_2", 35, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery 2", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("sensor", "battery_3", 40, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery 3", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("sensor", "battery_4", 80, { | ||||
|     device_class: "battery", | ||||
|     friendly_name: "Battery 4", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("input_number", "min_battery_level", 30, { | ||||
|     mode: "slider", | ||||
|     step: 10, | ||||
|     min: 0, | ||||
|     max: 100, | ||||
|     icon: "mdi:battery-alert-variant", | ||||
|     friendly_name: "Minimum Battery Level", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const CONFIGS = [ | ||||
|   { | ||||
|     heading: "Unfiltered entities", | ||||
|     heading: "Unfiltered controller", | ||||
|     config: ` | ||||
| - type: entities | ||||
|   entities: | ||||
|     - device_tracker.demo_anne_therese | ||||
|     - device_tracker.demo_home_boy | ||||
|     - device_tracker.demo_paulus | ||||
|     - light.bed_light | ||||
|     - light.ceiling_lights | ||||
|     - light.kitchen_lights | ||||
|   - light.bed_light | ||||
|   - light.ceiling_lights | ||||
|   - light.kitchen_lights | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "On and home entities", | ||||
|     heading: "Filtered entities card", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
| @@ -95,28 +63,9 @@ const CONFIGS = [ | ||||
|     - light.bed_light | ||||
|     - light.ceiling_lights | ||||
|     - light.kitchen_lights | ||||
|   conditions: | ||||
|     - condition: state | ||||
|       state: | ||||
|         - "on" | ||||
|         - home | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Same state as Bed Light", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - device_tracker.demo_anne_therese | ||||
|     - device_tracker.demo_home_boy | ||||
|     - device_tracker.demo_paulus | ||||
|     - light.bed_light | ||||
|     - light.ceiling_lights | ||||
|     - light.kitchen_lights | ||||
|   conditions: | ||||
|     - condition: state | ||||
|       state: | ||||
|         - light.bed_light | ||||
|   state_filter: | ||||
|     - "on" | ||||
|     - home | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
| @@ -130,11 +79,9 @@ const CONFIGS = [ | ||||
|     - light.bed_light | ||||
|     - light.ceiling_lights | ||||
|     - light.kitchen_lights | ||||
|   conditions: | ||||
|     - condition: state | ||||
|       state: | ||||
|         - "on" | ||||
|         - home | ||||
|   state_filter: | ||||
|     - "on" | ||||
|     - not_home | ||||
|   card: | ||||
|     type: entities | ||||
|     title: Custom Title | ||||
| @@ -152,101 +99,15 @@ const CONFIGS = [ | ||||
|     - light.bed_light | ||||
|     - light.ceiling_lights | ||||
|     - light.kitchen_lights | ||||
|   conditions: | ||||
|     - condition: state | ||||
|       state: | ||||
|         - "on" | ||||
|         - home | ||||
|   state_filter: | ||||
|     - "on" | ||||
|     - not_home | ||||
|   card: | ||||
|     type: glance | ||||
|     show_state: true | ||||
|     title: Custom Title | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: | ||||
|       "Filtered entities by battery attribute (< '30') using state filter", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - device_tracker.demo_anne_therese | ||||
|     - device_tracker.demo_home_boy | ||||
|     - device_tracker.demo_paulus | ||||
|   state_filter: | ||||
|     - operator: < | ||||
|       attribute: battery | ||||
|       value: "30" | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Unfiltered number entities", | ||||
|     config: ` | ||||
| - type: entities | ||||
|   entities: | ||||
|     - input_number.min_battery_level | ||||
|     - sensor.battery_1 | ||||
|     - sensor.battery_3 | ||||
|     - sensor.battery_2 | ||||
|     - sensor.battery_4 | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Battery lower than 50%", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - sensor.battery_1 | ||||
|     - sensor.battery_3 | ||||
|     - sensor.battery_2 | ||||
|     - sensor.battery_4 | ||||
|   conditions: | ||||
|     - condition: numeric_state | ||||
|       below: 50 | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Battery lower than min battery level", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - sensor.battery_1 | ||||
|     - sensor.battery_3 | ||||
|     - sensor.battery_2 | ||||
|     - sensor.battery_4 | ||||
|   conditions: | ||||
|     - condition: numeric_state | ||||
|       below: input_number.min_battery_level | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Battery between min battery level and 70%", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - sensor.battery_1 | ||||
|     - sensor.battery_3 | ||||
|     - sensor.battery_2 | ||||
|     - sensor.battery_4 | ||||
|   conditions: | ||||
|     - condition: numeric_state | ||||
|       above: input_number.min_battery_level | ||||
|       below: 70 | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Error: Entities must be specified", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Error: Incorrect filter config", | ||||
|     config: ` | ||||
| - type: entity-filter | ||||
|   entities: | ||||
|     - sensor.gas_station_lowest_price | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-entity-filter-card") | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| --- | ||||
| title: Picture Card | ||||
| --- | ||||
| @@ -1,61 +0,0 @@ | ||||
| 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,15 +25,6 @@ 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 = [ | ||||
| @@ -132,19 +123,6 @@ 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") | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user