mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 19:49:53 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			20230406.1
			...
			fix-entity
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | be6fef1824 | 
| @@ -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/vscode/devcontainers/python:0-3.10 | ||||
| FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.9 | ||||
|  | ||||
| ENV \ | ||||
|   DEBIAN_FRONTEND=noninteractive \ | ||||
|   | ||||
| @@ -5,39 +5,33 @@ | ||||
|     "context": ".." | ||||
|   }, | ||||
|   "appPort": "8124:8123", | ||||
|   "context": "..", | ||||
|   "postCreateCommand": "script/bootstrap", | ||||
|   "extensions": [ | ||||
|     "github.vscode-pull-request-github", | ||||
|     "dbaeumer.vscode-eslint", | ||||
|     "ms-vscode.vscode-typescript-tslint-plugin", | ||||
|     "esbenp.prettier-vscode", | ||||
|     "bierner.lit-html", | ||||
|     "runem.lit-plugin", | ||||
|     "ms-python.vscode-pylance" | ||||
|   ], | ||||
|   "containerEnv": { | ||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" | ||||
|   }, | ||||
|   "customizations": { | ||||
|     "vscode": { | ||||
|       "extensions": [ | ||||
|         "dbaeumer.vscode-eslint", | ||||
|         "esbenp.prettier-vscode", | ||||
|         "runem.lit-plugin", | ||||
|         "github.vscode-pull-request-github", | ||||
|         "eamodio.gitlens" | ||||
|       ], | ||||
|   "settings": { | ||||
|     "terminal.integrated.shell.linux": "/bin/bash", | ||||
|     "files.eol": "\n", | ||||
|     "editor.tabSize": 2, | ||||
|     "editor.formatOnPaste": false, | ||||
|     "editor.formatOnSave": true, | ||||
|     "editor.formatOnType": true, | ||||
|         "editor.renderWhitespace": "boundary", | ||||
|         "editor.rulers": [80], | ||||
|     "[typescript]": { | ||||
|       "editor.defaultFormatter": "esbenp.prettier-vscode" | ||||
|     }, | ||||
|     "[javascript]": { | ||||
|       "editor.defaultFormatter": "esbenp.prettier-vscode" | ||||
|     }, | ||||
|         "files.trimTrailingWhitespace": true, | ||||
|         "terminal.integrated.shell.linux": "/usr/bin/zsh", | ||||
|         "gitlens.showWelcomeOnInstall": false, | ||||
|         "gitlens.showWhatsNewAfterUpgrades": false, | ||||
|         "workbench.startupEditor": "none" | ||||
|       } | ||||
|     } | ||||
|     "files.trimTrailingWhitespace": true | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -5,7 +5,6 @@ | ||||
|     "plugin:@typescript-eslint/recommended", | ||||
|     "plugin:wc/recommended", | ||||
|     "plugin:lit/all", | ||||
|     "plugin:lit-a11y/recommended", | ||||
|     "prettier" | ||||
|   ], | ||||
|   "parser": "@typescript-eslint/parser", | ||||
| @@ -20,7 +19,7 @@ | ||||
|   "settings": { | ||||
|     "import/resolver": { | ||||
|       "webpack": { | ||||
|         "config": "./webpack.config.cjs" | ||||
|         "config": "./webpack.config.js" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
| @@ -66,10 +65,7 @@ | ||||
|     "import/extensions": [ | ||||
|       "error", | ||||
|       "ignorePackages", | ||||
|       { | ||||
|         "ts": "never", | ||||
|         "js": "never" | ||||
|       } | ||||
|       { "ts": "never", "js": "never" } | ||||
|     ], | ||||
|     "no-restricted-syntax": ["error", "LabeledStatement", "WithStatement"], | ||||
|     "object-curly-newline": "off", | ||||
| @@ -116,14 +112,7 @@ | ||||
|     ], | ||||
|     "unused-imports/no-unused-imports": "error", | ||||
|     "lit/attribute-value-entities": "off", | ||||
|     "lit/no-template-map": "off", | ||||
|     "lit/no-native-attributes": "warn", | ||||
|     "lit/no-this-assign-in-render": "warn", | ||||
|     "lit-a11y/click-events-have-key-events": ["off"], | ||||
|     "lit-a11y/no-autofocus": "off", | ||||
|     "lit-a11y/alt-text": "warn", | ||||
|     "lit-a11y/anchor-is-valid": "warn", | ||||
|     "lit-a11y/role-has-required-aria-attrs": "warn" | ||||
|     "lit/no-template-map": "off" | ||||
|   }, | ||||
|   "plugins": ["disable", "unused-imports"], | ||||
|   "processor": "disable/disable" | ||||
|   | ||||
							
								
								
									
										86
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										86
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,86 +0,0 @@ | ||||
| name: Cast deployment | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: "0 0 * * *" | ||||
|   push: | ||||
|     branches: | ||||
|       - master | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   deploy_dev: | ||||
|     runs-on: ubuntu-latest | ||||
|     name: Deploy Development | ||||
|     if: github.event_name != 'push' | ||||
|     environment: | ||||
|       name: Cast Development | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Cast | ||||
|         run: ./node_modules/.bin/gulp build-cast | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=cast/dist --alias dev | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
|  | ||||
|   deploy_master: | ||||
|     runs-on: ubuntu-latest | ||||
|     name: Deploy Production | ||||
|     if: github.event_name == 'push' | ||||
|     environment: | ||||
|       name: Cast Production | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Cast | ||||
|         run: ./node_modules/.bin/gulp build-cast | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=cast/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||
							
								
								
									
										51
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -13,85 +13,84 @@ on: | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|   GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} | ||||
|   cancel-in-progress: true | ||||
|  | ||||
| jobs: | ||||
|   lint: | ||||
|     name: Lint and check format | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|       - name: Check for duplicate dependencies | ||||
|         run: yarn dedupe --check | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages | ||||
|       - name: Run eslint | ||||
|         run: yarn run lint:eslint --quiet | ||||
|         run: yarn run lint:eslint | ||||
|       - name: Run tsc | ||||
|         run: yarn run lint:types | ||||
|       - name: Run prettier | ||||
|         run: yarn run lint:prettier | ||||
|       - name: Check for duplicate dependencies | ||||
|         run: yarn dedupe --check | ||||
|   test: | ||||
|     name: Run tests | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp build-translations build-locale-data | ||||
|       - name: Run Tests | ||||
|         run: yarn run test | ||||
|   build: | ||||
|     name: Build frontend | ||||
|     needs: [lint, test] | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: [lint, test] | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build Application | ||||
|         run: ./node_modules/.bin/gulp build-app | ||||
|         env: | ||||
|           IS_TEST: "true" | ||||
|   supervisor: | ||||
|     name: Build supervisor | ||||
|     needs: [lint, test] | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: [lint, test] | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build Application | ||||
|         run: ./node_modules/.bin/gulp build-hassio | ||||
|         env: | ||||
|   | ||||
							
								
								
									
										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@v3.5.0 | ||||
|       uses: actions/checkout@v3 | ||||
|       with: | ||||
|         # We must fetch at least the immediate parents so that if this is | ||||
|         # a pull request then we can checkout the head. | ||||
|   | ||||
							
								
								
									
										35
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| name: Demo | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build Demo | ||||
|         run: ./node_modules/.bin/gulp build-demo | ||||
|       - name: Deploy to Netlify | ||||
|         uses: netlify/actions/cli@master | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} | ||||
|         with: | ||||
|           args: deploy --dir=demo/dist --prod | ||||
							
								
								
									
										87
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										87
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,87 +0,0 @@ | ||||
| name: Demo deployment | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: "0 0 * * *" | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
|       - master | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   deploy_dev: | ||||
|     runs-on: ubuntu-latest | ||||
|     name: Demo Development | ||||
|     if: github.event_name != 'push' || github.ref != 'master' | ||||
|     environment: | ||||
|       name: Demo Development | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         with: | ||||
|           ref: dev | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Demo | ||||
|         run: ./node_modules/.bin/gulp build-demo | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} | ||||
|  | ||||
|   deploy_master: | ||||
|     runs-on: ubuntu-latest | ||||
|     name: Demo Production | ||||
|     if: github.event_name == 'push' && github.ref == 'master' | ||||
|     environment: | ||||
|       name: Demo Production | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         with: | ||||
|           ref: master | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Demo | ||||
|         run: ./node_modules/.bin/gulp build-demo | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=demo/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} | ||||
							
								
								
									
										43
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										43
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,43 +0,0 @@ | ||||
| name: Design deployment | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: "0 0 * * *" | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     environment: | ||||
|       name: Design | ||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Gallery | ||||
|         run: ./node_modules/.bin/gulp build-gallery | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=gallery/dist --prod | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||
							
								
								
									
										52
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										52
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,52 +0,0 @@ | ||||
| name: Design preview | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|     types: | ||||
|       - opened | ||||
|       - synchronize | ||||
|       - reopened | ||||
|       - labeled | ||||
|     branches: | ||||
|       - dev | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 16 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   preview: | ||||
|     runs-on: ubuntu-latest | ||||
|     # Skip running on forks since it won't have access to secrets | ||||
|     # Skip running PRs without 'needs design preview' label | ||||
|     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@v3.5.0 | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install --immutable | ||||
|  | ||||
|       - name: Build Gallery | ||||
|         run: ./node_modules/.bin/gulp build-gallery | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Deploy preview to Netlify | ||||
|         id: deploy | ||||
|         uses: netlify/actions/cli@master | ||||
|         with: | ||||
|           args: deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" | ||||
|         env: | ||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||
|  | ||||
|       - name: Generate summary | ||||
|         run: | | ||||
|           echo "${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}" >> "$GITHUB_STEP_SUMMARY" | ||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | ||||
|   lock: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@v4.0.0 | ||||
|       - uses: dessant/lock-threads@v3.0.0 | ||||
|         with: | ||||
|           github-token: ${{ github.token }} | ||||
|           issue-lock-inactive-days: "30" | ||||
|   | ||||
							
								
								
									
										19
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| name: Netlify | ||||
|  | ||||
| on: | ||||
|   schedule: | ||||
|     - cron: "0 0 * * *" | ||||
|  | ||||
| jobs: | ||||
|   trigger_builds: | ||||
|     name: Trigger netlify build preview | ||||
|     runs-on: "ubuntu-latest" | ||||
|     steps: | ||||
|       - name: Trigger Cast build | ||||
|         run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_CAST_DEV_BUILD_HOOK }} | ||||
|  | ||||
|       - name: Trigger Demo build | ||||
|         run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_DEMO_DEV_BUILD_HOOK }} | ||||
|  | ||||
|       - name: Trigger Design build | ||||
|         run: curl -X POST -d "NIGHTLY" https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_GALLERY_DEV_BUILD_HOOK }} | ||||
							
								
								
									
										9
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -21,7 +21,7 @@ jobs: | ||||
|       contents: write | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||
|         uses: actions/setup-python@v4 | ||||
| @@ -29,7 +29,7 @@ jobs: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
| @@ -43,14 +43,15 @@ jobs: | ||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} | ||||
|  | ||||
|       - name: Bump version | ||||
|         run: script/version_bump.cjs nightly | ||||
|         run: script/version_bump.js nightly | ||||
|  | ||||
|       - name: Build nightly Python wheels | ||||
|         run: | | ||||
|           pip install build | ||||
|           yarn install | ||||
|           export SKIP_FETCH_NIGHTLY_TRANSLATIONS=1 | ||||
|  | ||||
|           script/build_frontend | ||||
|  | ||||
|           rm -rf dist home_assistant_frontend.egg-info | ||||
|           python3 -m build | ||||
|  | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -24,7 +24,7 @@ jobs: | ||||
|       contents: write # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Verify version | ||||
|         uses: home-assistant/actions/helpers/verify-version@master | ||||
| @@ -35,7 +35,7 @@ jobs: | ||||
|           python-version: ${{ env.PYTHON_VERSION }} | ||||
|  | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v3.6.0 | ||||
|         uses: actions/setup-node@v3 | ||||
|         with: | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
| @@ -52,11 +52,11 @@ jobs: | ||||
|           python3 -m pip install twine build | ||||
|           export TWINE_USERNAME="__token__" | ||||
|           export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}" | ||||
|           export SKIP_FETCH_NIGHTLY_TRANSLATIONS=1 | ||||
|  | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@v0.1.15 | ||||
|         uses: softprops/action-gh-release@v0.1.14 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
| @@ -75,7 +75,7 @@ jobs: | ||||
|           echo "home-assistant-frontend==$version" > ./requirements.txt | ||||
|  | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2022.10.1 | ||||
|         uses: home-assistant/wheels@2022.06.7 | ||||
|         with: | ||||
|           abi: cp310 | ||||
|           tag: musllinux_1_2 | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: 90 days stale policy | ||||
|         uses: actions/stale@v8.0.0 | ||||
|         uses: actions/stale@v5.1.1 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 90 | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -16,7 +16,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v3.5.0 | ||||
|         uses: actions/checkout@v3 | ||||
|  | ||||
|       - name: Upload Translations | ||||
|         run: | | ||||
|   | ||||
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,10 +2,9 @@ | ||||
| .reify-cache | ||||
|  | ||||
| # build | ||||
| build/ | ||||
| dist/ | ||||
| /hass_frontend/ | ||||
| /translations/ | ||||
| build | ||||
| hass_frontend/* | ||||
| dist | ||||
|  | ||||
| # yarn | ||||
| .yarn/* | ||||
| @@ -15,7 +14,7 @@ dist/ | ||||
| !.yarn/sdks | ||||
| !.yarn/versions | ||||
| .pnp.* | ||||
| /node_modules/ | ||||
| node_modules/* | ||||
| yarn-error.log | ||||
| npm-debug.log | ||||
|  | ||||
| @@ -27,7 +26,7 @@ npm-debug.log | ||||
| # venv stuff | ||||
| pyvenv.cfg | ||||
| pip-selfcheck.json | ||||
| /venv/ | ||||
| venv/* | ||||
| .venv | ||||
|  | ||||
| # vscode | ||||
| @@ -46,4 +45,4 @@ src/cast/dev_const.ts | ||||
| .tool-versions | ||||
|  | ||||
| # Home Assistant config | ||||
| /config/ | ||||
| /config | ||||
|   | ||||
							
								
								
									
										5
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -2,8 +2,7 @@ | ||||
|   "recommendations": [ | ||||
|     "dbaeumer.vscode-eslint", | ||||
|     "esbenp.prettier-vscode", | ||||
|     "runem.lit-plugin", | ||||
|     "github.vscode-pull-request-github", | ||||
|     "eamodio.gitlens" | ||||
|     "bierner.lit-html", | ||||
|     "runem.lit-plugin" | ||||
|   ] | ||||
| } | ||||
|   | ||||
							
								
								
									
										6
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -191,12 +191,6 @@ | ||||
|       "runOptions": { | ||||
|         "instanceLimit": 1 | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       "label": "Setup and fetch nightly translations", | ||||
|       "type": "gulp", | ||||
|       "task": "setup-and-fetch-nightly-translations", | ||||
|       "problemMatcher": [] | ||||
|     } | ||||
|   ], | ||||
|   "inputs": [ | ||||
|   | ||||
							
								
								
									
										29
									
								
								.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| diff --git a/polyfillLoaders/EventTarget.js b/polyfillLoaders/EventTarget.js | ||||
| index 4e18ade7ba485849f17f28c94c42f0e0e01ac387..8f34f4f646c7f7becc208fb5a546c96034fc74dc 100644 | ||||
| --- a/polyfillLoaders/EventTarget.js | ||||
| +++ b/polyfillLoaders/EventTarget.js | ||||
| @@ -6,16 +6,15 @@ | ||||
|  let _ET; | ||||
|  let ET; | ||||
|  export default async function EventTarget() { | ||||
| -    return ET || init(); | ||||
| +  return ET || init(); | ||||
|  } | ||||
|  async function init() { | ||||
| -    _ET = window.EventTarget; | ||||
| -    try { | ||||
| -        new _ET(); | ||||
| -    } | ||||
| -    catch (_a) { | ||||
| -        _ET = (await import('event-target-shim')).EventTarget; | ||||
| -    } | ||||
| -    return (ET = _ET); | ||||
| +  _ET = window.EventTarget; | ||||
| +  try { | ||||
| +    new _ET(); | ||||
| +  } catch (_a) { | ||||
| +    _ET = (await import("event-target-shim")).default.EventTarget; | ||||
| +  } | ||||
| +  return (ET = _ET); | ||||
|  } | ||||
|  //# sourceMappingURL=EventTarget.js.map | ||||
							
								
								
									
										12
									
								
								.yarn/patches/@material/mwc-icon-button/remove-icon.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.yarn/patches/@material/mwc-icon-button/remove-icon.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| diff --git a/mwc-icon-button-base.js b/mwc-icon-button-base.js | ||||
| index 45cdaab93ccc0a6daaaaabc01266dcdc32e46bfd..b3ea5b541597308d85f86ce6c23fd00785fda835 100644 | ||||
| --- a/mwc-icon-button-base.js | ||||
| +++ b/mwc-icon-button-base.js | ||||
| @@ -63,7 +63,6 @@ export class IconButtonBase extends LitElement { | ||||
|          @touchend="${this.handleRippleDeactivate}" | ||||
|          @touchcancel="${this.handleRippleDeactivate}" | ||||
|      >${this.renderRipple()} | ||||
| -    <i class="material-icons">${this.icon}</i> | ||||
|      <span | ||||
|        ><slot></slot | ||||
|      ></span> | ||||
							
								
								
									
										564
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										564
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										3
									
								
								.yarn/plugins/@yarnpkg/plugin-typescript.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.yarn/plugins/@yarnpkg/plugin-typescript.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										873
									
								
								.yarn/releases/yarn-3.5.0.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										873
									
								
								.yarn/releases/yarn-3.5.0.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,5 +1,3 @@ | ||||
| defaultSemverRangePrefix: "" | ||||
|  | ||||
| nodeLinker: node-modules | ||||
|  | ||||
| plugins: | ||||
| @@ -8,4 +6,4 @@ plugins: | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||
|     spec: "@yarnpkg/plugin-interactive-tools" | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-3.5.0.cjs | ||||
| yarnPath: .yarn/releases/yarn-3.2.0.cjs | ||||
|   | ||||
							
								
								
									
										7
									
								
								build-scripts/.eslintrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								build-scripts/.eslintrc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| { | ||||
|   "rules": { | ||||
|     "import/no-extraneous-dependencies": 0, | ||||
|     "no-restricted-syntax": 0, | ||||
|     "no-console": 0 | ||||
|   } | ||||
| } | ||||
| @@ -1,12 +1,7 @@ | ||||
| { | ||||
|   "extends": "../.eslintrc.json", | ||||
|   "rules": { | ||||
|     "no-console": "off", | ||||
|     "import/no-extraneous-dependencies": "off", | ||||
|     "import/extensions": "off", | ||||
|     "import/no-dynamic-require": "off", | ||||
|     "global-require": "off", | ||||
|     "@typescript-eslint/no-var-requires": "off", | ||||
|     "prefer-arrow-callback": "off" | ||||
|     "import/no-extraneous-dependencies": 0, | ||||
|     "global-require": 0 | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| 
 | ||||
| // Currently only supports CommonJS modules, as require is synchronous. `import` would need babel running asynchronous.
 | ||||
| @@ -28,6 +29,7 @@ module.exports = function inlineConstants(babel, options, cwd) { | ||||
|       const absolute = module.startsWith(".") | ||||
|         ? require.resolve(module, { paths: [cwd] }) | ||||
|         : module; | ||||
|       // eslint-disable-next-line import/no-dynamic-require
 | ||||
|       return [absolute, require(absolute)]; | ||||
|     }) | ||||
|   ); | ||||
| @@ -1,18 +1,9 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| const env = require("./env.cjs"); | ||||
| const paths = require("./paths.cjs"); | ||||
| 
 | ||||
| // 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 = () => { | ||||
|   const ref = env.version().endsWith("dev") | ||||
|     ? process.env.GITHUB_SHA || "dev" | ||||
|     : env.version(); | ||||
|   return `https://raw.githubusercontent.com/home-assistant/frontend/${ref}`; | ||||
| }; | ||||
| const env = require("./env.js"); | ||||
| const paths = require("./paths.js"); | ||||
| 
 | ||||
| // Files from NPM Packages that should not be imported
 | ||||
| // eslint-disable-next-line unused-imports/no-unused-vars
 | ||||
| module.exports.ignorePackages = ({ latestBuild }) => [ | ||||
|   // Part of yaml.js and only used for !!js functions that we don't use
 | ||||
|   require.resolve("esprima"), | ||||
| @@ -62,26 +53,13 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
|   ...defineOverlay, | ||||
| }); | ||||
| 
 | ||||
| module.exports.htmlMinifierOptions = { | ||||
|   caseSensitive: true, | ||||
|   collapseWhitespace: true, | ||||
|   conservativeCollapse: true, | ||||
|   decodeEntities: true, | ||||
|   removeComments: true, | ||||
|   removeRedundantAttributes: true, | ||||
|   minifyCSS: { | ||||
|     compatibility: "*,-properties.zeroUnits", | ||||
|   }, | ||||
| }; | ||||
| 
 | ||||
| module.exports.terserOptions = ({ latestBuild, isTestBuild }) => ({ | ||||
| module.exports.terserOptions = (latestBuild) => ({ | ||||
|   safari10: !latestBuild, | ||||
|   ecma: latestBuild ? undefined : 5, | ||||
|   format: { comments: false }, | ||||
|   sourceMap: !isTestBuild, | ||||
|   output: { comments: false }, | ||||
| }); | ||||
| 
 | ||||
| module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({ | ||||
| module.exports.babelOptions = ({ latestBuild }) => ({ | ||||
|   babelrc: false, | ||||
|   compact: false, | ||||
|   presets: [ | ||||
| @@ -89,7 +67,7 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({ | ||||
|       "@babel/preset-env", | ||||
|       { | ||||
|         useBuiltIns: "entry", | ||||
|         corejs: { version: "3.29", proposals: true }, | ||||
|         corejs: "3.15", | ||||
|         bugfixes: true, | ||||
|       }, | ||||
|     ], | ||||
| @@ -99,7 +77,7 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({ | ||||
|     [ | ||||
|       path.resolve( | ||||
|         paths.polymer_dir, | ||||
|         "build-scripts/babel-plugins/inline-constants-plugin.cjs" | ||||
|         "build-scripts/babel-plugins/inline-constants-plugin.js" | ||||
|       ), | ||||
|       { | ||||
|         modules: ["@mdi/js"], | ||||
| @@ -115,41 +93,20 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({ | ||||
|     "@babel/plugin-syntax-import-meta", | ||||
|     "@babel/plugin-syntax-dynamic-import", | ||||
|     "@babel/plugin-syntax-top-level-await", | ||||
|     // Support  various proposals
 | ||||
|     "@babel/plugin-proposal-optional-chaining", | ||||
|     "@babel/plugin-proposal-nullish-coalescing-operator", | ||||
|     ["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }], | ||||
|     ["@babel/plugin-proposal-private-methods", { loose: true }], | ||||
|     ["@babel/plugin-proposal-private-property-in-object", { loose: true }], | ||||
|     ["@babel/plugin-proposal-class-properties", { loose: true }], | ||||
|     // Minify template literals for production
 | ||||
|     isProdBuild && [ | ||||
|       "template-html-minifier", | ||||
|       { | ||||
|         modules: { | ||||
|           lit: [ | ||||
|             "html", | ||||
|             { name: "svg", encapsulation: "svg" }, | ||||
|             { name: "css", encapsulation: "style" }, | ||||
|           ], | ||||
|           "@polymer/polymer/lib/utils/html-tag": ["html"], | ||||
|         }, | ||||
|         strictCSS: true, | ||||
|         htmlMinifier: module.exports.htmlMinifierOptions, | ||||
|         failOnError: true, // we can turn this off in case of false positives
 | ||||
|       }, | ||||
|     ], | ||||
|   ].filter(Boolean), | ||||
|   exclude: [ | ||||
|     // \\ for Windows, / for Mac OS and Linux
 | ||||
|     /node_modules[\\/]core-js/, | ||||
|     /node_modules[\\/]webpack[\\/]buildin/, | ||||
|   ], | ||||
|   sourceMaps: !isTestBuild, | ||||
| }); | ||||
| 
 | ||||
| const nameSuffix = (latestBuild) => (latestBuild ? "-latest" : "-es5"); | ||||
| 
 | ||||
| const outputPath = (outputRoot, latestBuild) => | ||||
|   path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5"); | ||||
| 
 | ||||
| @@ -172,17 +129,14 @@ BundleConfig { | ||||
|   latestBuild: boolean, | ||||
|   // If we're doing a stats build (create nice chunk names)
 | ||||
|   isStatsBuild: boolean, | ||||
|   // If it's just a test build in CI, skip time on source map generation
 | ||||
|   isTestBuild: boolean, | ||||
|   // Names of entrypoints that should not be hashed
 | ||||
|   dontHash: Set<string> | ||||
| } | ||||
| */ | ||||
| 
 | ||||
| module.exports.config = { | ||||
|   app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild, isWDS }) { | ||||
|   app({ isProdBuild, latestBuild, isStatsBuild, isWDS }) { | ||||
|     return { | ||||
|       name: "app" + nameSuffix(latestBuild), | ||||
|       entry: { | ||||
|         service_worker: "./src/entrypoints/service_worker.ts", | ||||
|         app: "./src/entrypoints/app.ts", | ||||
| @@ -196,14 +150,12 @@ module.exports.config = { | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       isStatsBuild, | ||||
|       isTestBuild, | ||||
|       isWDS, | ||||
|     }; | ||||
|   }, | ||||
| 
 | ||||
|   demo({ isProdBuild, latestBuild, isStatsBuild }) { | ||||
|     return { | ||||
|       name: "demo" + nameSuffix(latestBuild), | ||||
|       entry: { | ||||
|         main: path.resolve(paths.demo_dir, "src/entrypoint.ts"), | ||||
|       }, | ||||
| @@ -233,7 +185,6 @@ module.exports.config = { | ||||
|     } | ||||
| 
 | ||||
|     return { | ||||
|       name: "cast" + nameSuffix(latestBuild), | ||||
|       entry, | ||||
|       outputPath: outputPath(paths.cast_output_root, latestBuild), | ||||
|       publicPath: publicPath(latestBuild), | ||||
| @@ -245,9 +196,8 @@ module.exports.config = { | ||||
|     }; | ||||
|   }, | ||||
| 
 | ||||
|   hassio({ isProdBuild, latestBuild, isStatsBuild, isTestBuild }) { | ||||
|   hassio({ isProdBuild, latestBuild }) { | ||||
|     return { | ||||
|       name: "supervisor" + nameSuffix(latestBuild), | ||||
|       entry: { | ||||
|         entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"), | ||||
|       }, | ||||
| @@ -255,8 +205,6 @@ module.exports.config = { | ||||
|       publicPath: publicPath(latestBuild, paths.hassio_publicPath), | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       isStatsBuild, | ||||
|       isTestBuild, | ||||
|       isHassioBuild: true, | ||||
|       defineOverlay: { | ||||
|         __SUPERVISOR__: true, | ||||
| @@ -266,7 +214,6 @@ module.exports.config = { | ||||
| 
 | ||||
|   gallery({ isProdBuild, latestBuild }) { | ||||
|     return { | ||||
|       name: "gallery" + nameSuffix(latestBuild), | ||||
|       entry: { | ||||
|         entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"), | ||||
|       }, | ||||
| @@ -1,6 +1,7 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const fs = require("fs"); | ||||
| const path = require("path"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const paths = require("./paths.js"); | ||||
| 
 | ||||
| module.exports = { | ||||
|   useRollup() { | ||||
| @@ -17,7 +18,7 @@ module.exports = { | ||||
|   isStatsBuild() { | ||||
|     return process.env.STATS === "1"; | ||||
|   }, | ||||
|   isTestBuild() { | ||||
|   isTest() { | ||||
|     return process.env.IS_TEST === "true"; | ||||
|   }, | ||||
|   isNetlify() { | ||||
| @@ -1,18 +1,19 @@ | ||||
| // Run HA develop mode
 | ||||
| 
 | ||||
| const gulp = require("gulp"); | ||||
| const env = require("../env.cjs"); | ||||
| require("./clean.cjs"); | ||||
| require("./translations.cjs"); | ||||
| require("./locale-data.cjs"); | ||||
| require("./gen-icons-json.cjs"); | ||||
| require("./gather-static.cjs"); | ||||
| require("./compress.cjs"); | ||||
| require("./webpack.cjs"); | ||||
| require("./service-worker.cjs"); | ||||
| require("./entry-html.cjs"); | ||||
| require("./rollup.cjs"); | ||||
| require("./wds.cjs"); | ||||
| 
 | ||||
| const env = require("../env"); | ||||
| 
 | ||||
| require("./clean.js"); | ||||
| require("./translations.js"); | ||||
| require("./locale-data.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| require("./gather-static.js"); | ||||
| require("./compress.js"); | ||||
| require("./webpack.js"); | ||||
| require("./service-worker.js"); | ||||
| require("./entry-html.js"); | ||||
| require("./rollup.js"); | ||||
| require("./wds.js"); | ||||
| 
 | ||||
| gulp.task( | ||||
|   "develop-app", | ||||
| @@ -49,7 +50,7 @@ gulp.task( | ||||
|     "copy-static-app", | ||||
|     env.useRollup() ? "rollup-prod-app" : "webpack-prod-app", | ||||
|     // Don't compress running tests
 | ||||
|     ...(env.isTestBuild() ? [] : ["compress-app"]), | ||||
|     ...(env.isTest() ? [] : ["compress-app"]), | ||||
|     gulp.parallel( | ||||
|       "gen-pages-prod", | ||||
|       "gen-index-app-prod", | ||||
| @@ -1,13 +1,14 @@ | ||||
| const gulp = require("gulp"); | ||||
| const env = require("../env.cjs"); | ||||
| 
 | ||||
| require("./clean.cjs"); | ||||
| require("./translations.cjs"); | ||||
| require("./gather-static.cjs"); | ||||
| require("./webpack.cjs"); | ||||
| require("./service-worker.cjs"); | ||||
| require("./entry-html.cjs"); | ||||
| require("./rollup.cjs"); | ||||
| const env = require("../env"); | ||||
| 
 | ||||
| require("./clean.js"); | ||||
| require("./translations.js"); | ||||
| require("./gather-static.js"); | ||||
| require("./webpack.js"); | ||||
| require("./service-worker.js"); | ||||
| require("./entry-html.js"); | ||||
| require("./rollup.js"); | ||||
| 
 | ||||
| gulp.task( | ||||
|   "develop-cast", | ||||
| @@ -1,40 +0,0 @@ | ||||
| const del = import("del"); | ||||
| const gulp = require("gulp"); | ||||
| const paths = require("../paths.cjs"); | ||||
| require("./translations.cjs"); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean", | ||||
|   gulp.parallel("clean-translations", async () => | ||||
|     (await del).deleteSync([paths.app_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-demo", | ||||
|   gulp.parallel("clean-translations", async () => | ||||
|     (await del).deleteSync([paths.demo_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-cast", | ||||
|   gulp.parallel("clean-translations", async () => | ||||
|     (await del).deleteSync([paths.cast_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task("clean-hassio", async () => | ||||
|   (await del).deleteSync([paths.hassio_output_root, paths.build_dir]) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-gallery", | ||||
|   gulp.parallel("clean-translations", async () => | ||||
|     (await del).deleteSync([ | ||||
|       paths.gallery_output_root, | ||||
|       paths.gallery_build, | ||||
|       paths.build_dir, | ||||
|     ]) | ||||
|   ) | ||||
| ); | ||||
							
								
								
									
										36
									
								
								build-scripts/gulp/clean.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								build-scripts/gulp/clean.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| const del = require("del"); | ||||
| const gulp = require("gulp"); | ||||
| const paths = require("../paths"); | ||||
| require("./translations"); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean", | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.app_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-demo", | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.demo_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-cast", | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.cast_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task("clean-hassio", () => | ||||
|   del([paths.hassio_output_root, paths.build_dir]) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-gallery", | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.gallery_output_root, paths.gallery_build, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
| @@ -4,7 +4,7 @@ const gulp = require("gulp"); | ||||
| const zopfli = require("gulp-zopfli-green"); | ||||
| const merge = require("merge-stream"); | ||||
| const path = require("path"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const paths = require("../paths"); | ||||
| 
 | ||||
| const zopfliOptions = { threshold: 150 }; | ||||
| 
 | ||||
| @@ -1,15 +1,16 @@ | ||||
| // Run demo develop mode
 | ||||
| const gulp = require("gulp"); | ||||
| const env = require("../env.cjs"); | ||||
| 
 | ||||
| require("./clean.cjs"); | ||||
| require("./translations.cjs"); | ||||
| require("./gen-icons-json.cjs"); | ||||
| require("./gather-static.cjs"); | ||||
| require("./webpack.cjs"); | ||||
| require("./service-worker.cjs"); | ||||
| require("./entry-html.cjs"); | ||||
| require("./rollup.cjs"); | ||||
| const env = require("../env"); | ||||
| 
 | ||||
| require("./clean.js"); | ||||
| require("./translations.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| require("./gather-static.js"); | ||||
| require("./webpack.js"); | ||||
| require("./service-worker.js"); | ||||
| require("./entry-html.js"); | ||||
| require("./rollup.js"); | ||||
| 
 | ||||
| gulp.task( | ||||
|   "develop-demo", | ||||
| @@ -1,69 +0,0 @@ | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs/promises"); | ||||
| const mapStream = require("map-stream"); | ||||
|  | ||||
| const inDirFrontend = "translations/frontend"; | ||||
| const inDirBackend = "translations/backend"; | ||||
| const srcMeta = "src/translations/translationMetadata.json"; | ||||
| const encoding = "utf8"; | ||||
|  | ||||
| function hasHtml(data) { | ||||
|   return /<[a-z][\s\S]*>/i.test(data); | ||||
| } | ||||
|  | ||||
| function recursiveCheckHasHtml(file, data, errors, recKey) { | ||||
|   Object.keys(data).forEach(function (key) { | ||||
|     if (typeof data[key] === "object") { | ||||
|       const nextRecKey = recKey ? `${recKey}.${key}` : key; | ||||
|       recursiveCheckHasHtml(file, data[key], errors, nextRecKey); | ||||
|     } else if (hasHtml(data[key])) { | ||||
|       errors.push(`HTML found in ${file.path} at key ${recKey}.${key}`); | ||||
|     } | ||||
|   }); | ||||
| } | ||||
|  | ||||
| function checkHtml() { | ||||
|   const errors = []; | ||||
|  | ||||
|   return mapStream(function (file, cb) { | ||||
|     const content = file.contents; | ||||
|     let error; | ||||
|     if (content) { | ||||
|       if (hasHtml(String(content))) { | ||||
|         const data = JSON.parse(String(content)); | ||||
|         recursiveCheckHasHtml(file, data, errors); | ||||
|         if (errors.length > 0) { | ||||
|           error = errors.join("\r\n"); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     cb(error, file); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| // Backend translations do not currently pass HTML check so are excluded here for now | ||||
| gulp.task("check-translations-html", function () { | ||||
|   return gulp.src([`${inDirFrontend}/*.json`]).pipe(checkHtml()); | ||||
| }); | ||||
|  | ||||
| gulp.task("check-all-files-exist", async function () { | ||||
|   const file = await fs.readFile(srcMeta, { encoding }); | ||||
|   const meta = JSON.parse(file); | ||||
|   const writings = []; | ||||
|   Object.keys(meta).forEach((lang) => { | ||||
|     writings.push( | ||||
|       fs.writeFile(`${inDirFrontend}/${lang}.json`, JSON.stringify({}), { | ||||
|         flag: "wx", | ||||
|       }), | ||||
|       fs.writeFile(`${inDirBackend}/${lang}.json`, JSON.stringify({}), { | ||||
|         flag: "wx", | ||||
|       }) | ||||
|     ); | ||||
|   }); | ||||
|   await Promise.allSettled(writings); | ||||
| }); | ||||
|  | ||||
| gulp.task( | ||||
|   "check-downloaded-translations", | ||||
|   gulp.series("check-translations-html", "check-all-files-exist") | ||||
| ); | ||||
							
								
								
									
										95
									
								
								build-scripts/gulp/download_translations.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								build-scripts/gulp/download_translations.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
| const del = require("del"); | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const mapStream = require("map-stream"); | ||||
|  | ||||
| const inDirFrontend = "translations/frontend"; | ||||
| const inDirBackend = "translations/backend"; | ||||
| const downloadDir = "translations/downloads"; | ||||
| const srcMeta = "src/translations/translationMetadata.json"; | ||||
|  | ||||
| const encoding = "utf8"; | ||||
|  | ||||
| const tasks = []; | ||||
|  | ||||
| function hasHtml(data) { | ||||
|   return /<[a-z][\s\S]*>/i.test(data); | ||||
| } | ||||
|  | ||||
| function recursiveCheckHasHtml(file, data, errors, recKey) { | ||||
|   Object.keys(data).forEach(function (key) { | ||||
|     if (typeof data[key] === "object") { | ||||
|       const nextRecKey = recKey ? `${recKey}.${key}` : key; | ||||
|       recursiveCheckHasHtml(file, data[key], errors, nextRecKey); | ||||
|     } else if (hasHtml(data[key])) { | ||||
|       errors.push(`HTML found in ${file.path} at key ${recKey}.${key}`); | ||||
|     } | ||||
|   }); | ||||
| } | ||||
|  | ||||
| function checkHtml() { | ||||
|   const errors = []; | ||||
|  | ||||
|   return mapStream(function (file, cb) { | ||||
|     const content = file.contents; | ||||
|     let error; | ||||
|     if (content) { | ||||
|       if (hasHtml(String(content))) { | ||||
|         const data = JSON.parse(String(content)); | ||||
|         recursiveCheckHasHtml(file, data, errors); | ||||
|         if (errors.length > 0) { | ||||
|           error = errors.join("\r\n"); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     cb(error, file); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| let taskName = "clean-downloaded-translations"; | ||||
| gulp.task(taskName, function () { | ||||
|   return del([`${downloadDir}/**`]); | ||||
| }); | ||||
| tasks.push(taskName); | ||||
|  | ||||
| taskName = "check-translations-html"; | ||||
| gulp.task(taskName, function () { | ||||
|   return gulp.src(`${downloadDir}/*.json`).pipe(checkHtml()); | ||||
| }); | ||||
| tasks.push(taskName); | ||||
|  | ||||
| taskName = "check-all-files-exist"; | ||||
| gulp.task(taskName, function () { | ||||
|   const file = fs.readFileSync(srcMeta, { encoding }); | ||||
|   const meta = JSON.parse(file); | ||||
|   Object.keys(meta).forEach((lang) => { | ||||
|     if (!fs.existsSync(`${inDirFrontend}/${lang}.json`)) { | ||||
|       fs.writeFileSync(`${inDirFrontend}/${lang}.json`, JSON.stringify({})); | ||||
|     } | ||||
|     if (!fs.existsSync(`${inDirBackend}/${lang}.json`)) { | ||||
|       fs.writeFileSync(`${inDirBackend}/${lang}.json`, JSON.stringify({})); | ||||
|     } | ||||
|   }); | ||||
|   return Promise.resolve(); | ||||
| }); | ||||
| tasks.push(taskName); | ||||
|  | ||||
| taskName = "move-downloaded-translations"; | ||||
| gulp.task(taskName, function () { | ||||
|   return gulp.src(`${downloadDir}/*.json`).pipe(gulp.dest(inDirFrontend)); | ||||
| }); | ||||
| tasks.push(taskName); | ||||
|  | ||||
| taskName = "check-downloaded-translations"; | ||||
| gulp.task( | ||||
|   taskName, | ||||
|   gulp.series( | ||||
|     "check-translations-html", | ||||
|     "move-downloaded-translations", | ||||
|     "check-all-files-exist", | ||||
|     "clean-downloaded-translations" | ||||
|   ) | ||||
| ); | ||||
| tasks.push(taskName); | ||||
|  | ||||
| module.exports = tasks; | ||||
| @@ -1,12 +1,13 @@ | ||||
| // Tasks to generate entry HTML
 | ||||
| /* eslint-disable import/no-dynamic-require */ | ||||
| /* eslint-disable global-require */ | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs-extra"); | ||||
| const path = require("path"); | ||||
| const template = require("lodash.template"); | ||||
| const { minify } = require("html-minifier-terser"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const env = require("../env.cjs"); | ||||
| const { htmlMinifierOptions, terserOptions } = require("../bundle.cjs"); | ||||
| const minify = require("html-minifier").minify; | ||||
| const paths = require("../paths.js"); | ||||
| const env = require("../env.js"); | ||||
| 
 | ||||
| const templatePath = (tpl) => | ||||
|   path.resolve(paths.polymer_dir, "src/html/", `${tpl}.html.template`); | ||||
| @@ -40,12 +41,10 @@ const renderGalleryTemplate = (pth, data = {}) => | ||||
| 
 | ||||
| const minifyHtml = (content) => | ||||
|   minify(content, { | ||||
|     ...htmlMinifierOptions, | ||||
|     conservativeCollapse: false, | ||||
|     minifyJS: terserOptions({ | ||||
|       latestBuild: false, // Shared scripts should be ES5
 | ||||
|       isTestBuild: true, // Don't need source maps
 | ||||
|     }), | ||||
|     collapseWhitespace: true, | ||||
|     minifyJS: true, | ||||
|     minifyCSS: true, | ||||
|     removeComments: true, | ||||
|   }); | ||||
| 
 | ||||
| const PAGES = ["onboarding", "authorize"]; | ||||
| @@ -66,7 +65,7 @@ gulp.task("gen-pages-dev", (done) => { | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-pages-prod", async () => { | ||||
| gulp.task("gen-pages-prod", (done) => { | ||||
|   const latestManifest = require(path.resolve( | ||||
|     paths.app_output_latest, | ||||
|     "manifest.json" | ||||
| @@ -76,29 +75,23 @@ gulp.task("gen-pages-prod", async () => { | ||||
|     "manifest.json" | ||||
|   )); | ||||
| 
 | ||||
|   const minifiedHTML = []; | ||||
|   for (const page of PAGES) { | ||||
|     const content = renderTemplate(page, { | ||||
|       latestPageJS: latestManifest[`${page}.js`], | ||||
| 
 | ||||
|       es5PageJS: es5Manifest[`${page}.js`], | ||||
|     }); | ||||
| 
 | ||||
|     minifiedHTML.push( | ||||
|       minifyHtml(content).then((minified) => | ||||
|     fs.outputFileSync( | ||||
|       path.resolve(paths.app_output_root, `${page}.html`), | ||||
|           minified | ||||
|         ) | ||||
|       ) | ||||
|       minifyHtml(content) | ||||
|     ); | ||||
|   } | ||||
|   await Promise.all(minifiedHTML); | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-app-dev", (done) => { | ||||
|   let latestAppJS; | ||||
|   let latestCoreJS; | ||||
|   let latestCustomPanelJS; | ||||
|   let latestAppJS, latestCoreJS, latestCustomPanelJS; | ||||
| 
 | ||||
|   if (env.useWDS()) { | ||||
|     latestAppJS = "http://localhost:8000/src/entrypoints/app.ts"; | ||||
| @@ -125,7 +118,7 @@ gulp.task("gen-index-app-dev", (done) => { | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-app-prod", async () => { | ||||
| gulp.task("gen-index-app-prod", (done) => { | ||||
|   const latestManifest = require(path.resolve( | ||||
|     paths.app_output_latest, | ||||
|     "manifest.json" | ||||
| @@ -143,15 +136,13 @@ gulp.task("gen-index-app-prod", async () => { | ||||
|     es5CoreJS: es5Manifest["core.js"], | ||||
|     es5CustomPanelJS: es5Manifest["custom-panel.js"], | ||||
|   }); | ||||
|   const minified = (await minifyHtml(content)).replace( | ||||
|     /#THEMEC/g, | ||||
|     "{{ theme_color }}" | ||||
|   ); | ||||
|   const minified = minifyHtml(content).replace(/#THEMEC/g, "{{ theme_color }}"); | ||||
| 
 | ||||
|   fs.outputFileSync( | ||||
|     path.resolve(paths.app_output_root, "index.html"), | ||||
|     minified | ||||
|   ); | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-cast-dev", (done) => { | ||||
| @@ -253,7 +244,7 @@ gulp.task("gen-index-demo-dev", (done) => { | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-demo-prod", async () => { | ||||
| gulp.task("gen-index-demo-prod", (done) => { | ||||
|   const latestManifest = require(path.resolve( | ||||
|     paths.demo_output_latest, | ||||
|     "manifest.json" | ||||
| @@ -267,12 +258,13 @@ gulp.task("gen-index-demo-prod", async () => { | ||||
| 
 | ||||
|     es5DemoJS: es5Manifest["main.js"], | ||||
|   }); | ||||
|   const minified = await minifyHtml(content); | ||||
|   const minified = minifyHtml(content); | ||||
| 
 | ||||
|   fs.outputFileSync( | ||||
|     path.resolve(paths.demo_output_root, "index.html"), | ||||
|     minified | ||||
|   ); | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-gallery-dev", (done) => { | ||||
| @@ -287,7 +279,7 @@ gulp.task("gen-index-gallery-dev", (done) => { | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-gallery-prod", async () => { | ||||
| gulp.task("gen-index-gallery-prod", (done) => { | ||||
|   const latestManifest = require(path.resolve( | ||||
|     paths.gallery_output_latest, | ||||
|     "manifest.json" | ||||
| @@ -295,12 +287,13 @@ gulp.task("gen-index-gallery-prod", async () => { | ||||
|   const content = renderGalleryTemplate("index", { | ||||
|     latestGalleryJS: latestManifest["entrypoint.js"], | ||||
|   }); | ||||
|   const minified = await minifyHtml(content); | ||||
|   const minified = minifyHtml(content); | ||||
| 
 | ||||
|   fs.outputFileSync( | ||||
|     path.resolve(paths.gallery_output_root, "index.html"), | ||||
|     minified | ||||
|   ); | ||||
|   done(); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("gen-index-hassio-dev", async () => { | ||||
| @@ -1,175 +0,0 @@ | ||||
| // Task to download the latest Lokalise translations from the nightly workflow artifacts | ||||
|  | ||||
| const del = import("del"); | ||||
| const fs = require("fs/promises"); | ||||
| const path = require("path"); | ||||
| const process = require("process"); | ||||
| const gulp = require("gulp"); | ||||
| const jszip = require("jszip"); | ||||
| const tar = require("tar"); | ||||
| const { Octokit } = require("@octokit/rest"); | ||||
| const { retry } = require("@octokit/plugin-retry"); | ||||
| const { createOAuthDeviceAuth } = require("@octokit/auth-oauth-device"); | ||||
|  | ||||
| const MAX_AGE = 24; // hours | ||||
| const OWNER = "home-assistant"; | ||||
| const REPO = "frontend"; | ||||
| const WORKFLOW_NAME = "nightly.yaml"; | ||||
| const ARTIFACT_NAME = "translations"; | ||||
| const CLIENT_ID = "Iv1.3914e28cb27834d1"; | ||||
| const EXTRACT_DIR = "translations"; | ||||
| const TOKEN_FILE = path.posix.join(EXTRACT_DIR, "token.json"); | ||||
| const ARTIFACT_FILE = path.posix.join(EXTRACT_DIR, "artifact.json"); | ||||
|  | ||||
| let allowTokenSetup = false; | ||||
| gulp.task("allow-setup-fetch-nightly-translations", (done) => { | ||||
|   allowTokenSetup = true; | ||||
|   done(); | ||||
| }); | ||||
|  | ||||
| gulp.task("fetch-nightly-translations", async function () { | ||||
|   // Skip all when environment flag is set (assumes translations are already in place) | ||||
|   if (process.env?.SKIP_FETCH_NIGHTLY_TRANSLATIONS) { | ||||
|     console.log("Skipping fetch due to environment signal"); | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   // Read current translations artifact info if it exists, | ||||
|   // and stop if they are not old enough | ||||
|   let currentArtifact; | ||||
|   try { | ||||
|     currentArtifact = JSON.parse(await fs.readFile(ARTIFACT_FILE, "utf-8")); | ||||
|     const currentAge = | ||||
|       (Date.now() - Date.parse(currentArtifact.created_at)) / 3600000; | ||||
|     if (currentAge < MAX_AGE) { | ||||
|       console.log( | ||||
|         "Keeping current translations (only %s hours old)", | ||||
|         currentAge.toFixed(1) | ||||
|       ); | ||||
|       return; | ||||
|     } | ||||
|   } catch { | ||||
|     currentArtifact = null; | ||||
|   } | ||||
|  | ||||
|   // To store file writing promises | ||||
|   const createExtractDir = fs.mkdir(EXTRACT_DIR, { recursive: true }); | ||||
|   const writings = []; | ||||
|  | ||||
|   // Authenticate to GitHub using GitHub action token if it exists, | ||||
|   // otherwise look for a saved user token or generate a new one if none | ||||
|   let tokenAuth; | ||||
|   if (process.env.GITHUB_TOKEN) { | ||||
|     tokenAuth = { token: process.env.GITHUB_TOKEN }; | ||||
|   } else { | ||||
|     try { | ||||
|       tokenAuth = JSON.parse(await fs.readFile(TOKEN_FILE, "utf-8")); | ||||
|     } catch { | ||||
|       if (!allowTokenSetup) { | ||||
|         console.log("No token found so  build wil continue with English only"); | ||||
|         return; | ||||
|       } | ||||
|       const auth = createOAuthDeviceAuth({ | ||||
|         clientType: "github-app", | ||||
|         clientId: CLIENT_ID, | ||||
|         onVerification: (verification) => { | ||||
|           console.log( | ||||
|             "Task needs to authenticate to GitHub to fetch the translations from nightly workflow\n" + | ||||
|               "Please go to %s to authorize this task\n" + | ||||
|               "\nEnter user code: %s\n\n" + | ||||
|               "This code will expire in %s minutes\n" + | ||||
|               "Task will automatically continue after authorization and token will be saved for future use", | ||||
|             verification.verification_uri, | ||||
|             verification.user_code, | ||||
|             (verification.expires_in / 60).toFixed(0) | ||||
|           ); | ||||
|         }, | ||||
|       }); | ||||
|       tokenAuth = await auth({ type: "oauth" }); | ||||
|       writings.push( | ||||
|         createExtractDir.then( | ||||
|           fs.writeFile(TOKEN_FILE, JSON.stringify(tokenAuth, null, 2)) | ||||
|         ) | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // Authenticate with token and request workflow runs from GitHub | ||||
|   console.log("Fetching new translations..."); | ||||
|   const octokit = new (Octokit.plugin(retry))({ | ||||
|     userAgent: "Fetch Nightly Translations", | ||||
|     auth: tokenAuth.token, | ||||
|   }); | ||||
|  | ||||
|   const workflowRunsResponse = await octokit.rest.actions.listWorkflowRuns({ | ||||
|     owner: OWNER, | ||||
|     repo: REPO, | ||||
|     workflow_id: WORKFLOW_NAME, | ||||
|     status: "success", | ||||
|     event: "schedule", | ||||
|     per_page: 1, | ||||
|     exclude_pull_requests: true, | ||||
|   }); | ||||
|   if (workflowRunsResponse.data.total_count === 0) { | ||||
|     throw Error("No successful nightly workflow runs found"); | ||||
|   } | ||||
|   const latestNightlyRun = workflowRunsResponse.data.workflow_runs[0]; | ||||
|  | ||||
|   // Stop if current is already the latest, otherwise Find the translations artifact | ||||
|   if (currentArtifact?.workflow_run.id === latestNightlyRun.id) { | ||||
|     console.log("Stopping because current translations are still the latest"); | ||||
|     return; | ||||
|   } | ||||
|   const latestArtifact = ( | ||||
|     await octokit.actions.listWorkflowRunArtifacts({ | ||||
|       owner: OWNER, | ||||
|       repo: REPO, | ||||
|       run_id: latestNightlyRun.id, | ||||
|     }) | ||||
|   ).data.artifacts.find((artifact) => artifact.name === ARTIFACT_NAME); | ||||
|   if (!latestArtifact) { | ||||
|     throw Error("Latest nightly workflow run has no translations artifact"); | ||||
|   } | ||||
|   writings.push( | ||||
|     createExtractDir.then( | ||||
|       fs.writeFile(ARTIFACT_FILE, JSON.stringify(latestArtifact, null, 2)) | ||||
|     ) | ||||
|   ); | ||||
|  | ||||
|   // Remove the current translations | ||||
|   const deleteCurrent = Promise.all(writings).then( | ||||
|     (await del).deleteAsync([ | ||||
|       `${EXTRACT_DIR}/*`, | ||||
|       `!${ARTIFACT_FILE}`, | ||||
|       `!${TOKEN_FILE}`, | ||||
|     ]) | ||||
|   ); | ||||
|  | ||||
|   // Get the download URL and follow the redirect to download (stored as ArrayBuffer) | ||||
|   const downloadResponse = await octokit.actions.downloadArtifact({ | ||||
|     owner: OWNER, | ||||
|     repo: REPO, | ||||
|     artifact_id: latestArtifact.id, | ||||
|     archive_format: "zip", | ||||
|   }); | ||||
|   if (downloadResponse.status !== 200) { | ||||
|     throw Error("Failure downloading translations artifact"); | ||||
|   } | ||||
|  | ||||
|   // Artifact is a tarball, but GitHub adds it to a zip file | ||||
|   console.log("Unpacking downloaded translations..."); | ||||
|   const zip = await jszip.loadAsync(downloadResponse.data); | ||||
|   await deleteCurrent; | ||||
|   const extractStream = zip.file(/.*/)[0].nodeStream().pipe(tar.extract()); | ||||
|   await new Promise((resolve, reject) => { | ||||
|     extractStream.on("close", resolve).on("error", reject); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| gulp.task( | ||||
|   "setup-and-fetch-nightly-translations", | ||||
|   gulp.series( | ||||
|     "allow-setup-fetch-nightly-translations", | ||||
|     "fetch-nightly-translations" | ||||
|   ) | ||||
| ); | ||||
| @@ -1,3 +1,4 @@ | ||||
| /* eslint-disable */ | ||||
| // Run demo develop mode
 | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| @@ -6,21 +7,21 @@ const { marked } = require("marked"); | ||||
| const glob = require("glob"); | ||||
| const yaml = require("js-yaml"); | ||||
| 
 | ||||
| const env = require("../env.cjs"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const env = require("../env"); | ||||
| const paths = require("../paths"); | ||||
| 
 | ||||
| require("./clean.cjs"); | ||||
| require("./translations.cjs"); | ||||
| require("./gen-icons-json.cjs"); | ||||
| require("./gather-static.cjs"); | ||||
| require("./webpack.cjs"); | ||||
| require("./service-worker.cjs"); | ||||
| require("./entry-html.cjs"); | ||||
| require("./rollup.cjs"); | ||||
| require("./clean.js"); | ||||
| require("./translations.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| require("./gather-static.js"); | ||||
| require("./webpack.js"); | ||||
| require("./service-worker.js"); | ||||
| require("./entry-html.js"); | ||||
| require("./rollup.js"); | ||||
| 
 | ||||
| gulp.task("gather-gallery-pages", async function gatherPages() { | ||||
|   const pageDir = path.resolve(paths.gallery_dir, "src/pages"); | ||||
|   const files = await glob(path.resolve(pageDir, "**/*")); | ||||
|   const files = glob.sync(path.resolve(pageDir, "**/*")); | ||||
| 
 | ||||
|   const galleryBuild = path.resolve(paths.gallery_dir, "build"); | ||||
|   fs.mkdirSync(galleryBuild, { recursive: true }); | ||||
| @@ -40,7 +41,7 @@ gulp.task("gather-gallery-pages", async function gatherPages() { | ||||
|     } | ||||
|     processed.add(pageId); | ||||
| 
 | ||||
|     const [category] = pageId.split("/", 2); | ||||
|     const [category, name] = pageId.split("/", 2); | ||||
| 
 | ||||
|     const demoFile = path.resolve(pageDir, `${pageId}.ts`); | ||||
|     const descriptionFile = path.resolve(pageDir, `${pageId}.markdown`); | ||||
| @@ -89,7 +90,9 @@ gulp.task("gather-gallery-pages", async function gatherPages() { | ||||
| 
 | ||||
|   // Generate sidebar
 | ||||
|   const sidebarPath = path.resolve(paths.gallery_dir, "sidebar.js"); | ||||
|   const sidebar = (await import(sidebarPath)).default; | ||||
|   // To make watch work during development
 | ||||
|   delete require.cache[sidebarPath]; | ||||
|   const sidebar = require(sidebarPath); | ||||
| 
 | ||||
|   const pagesToProcess = {}; | ||||
|   for (const key of processed) { | ||||
| @@ -3,7 +3,7 @@ | ||||
| const gulp = require("gulp"); | ||||
| const path = require("path"); | ||||
| const fs = require("fs-extra"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const paths = require("../paths"); | ||||
| 
 | ||||
| const npmPath = (...parts) => | ||||
|   path.resolve(paths.polymer_dir, "node_modules", ...parts); | ||||
| @@ -134,11 +134,11 @@ gulp.task("gen-icons-json", (done) => { | ||||
|   }); | ||||
| 
 | ||||
|   const file = fs.readFileSync(PACKAGE_PATH, { encoding }); | ||||
|   const packageMeta = JSON.parse(file); | ||||
|   const package = JSON.parse(file); | ||||
| 
 | ||||
|   fs.writeFileSync( | ||||
|     path.resolve(OUTPUT_DIR, "iconMetadata.json"), | ||||
|     JSON.stringify({ version: packageMeta.version, parts }) | ||||
|     JSON.stringify({ version: package.version, parts }) | ||||
|   ); | ||||
| 
 | ||||
|   fs.writeFileSync( | ||||
| @@ -1,13 +1,15 @@ | ||||
| const gulp = require("gulp"); | ||||
| const env = require("../env.cjs"); | ||||
| require("./clean.cjs"); | ||||
| require("./compress.cjs"); | ||||
| require("./entry-html.cjs"); | ||||
| require("./gather-static.cjs"); | ||||
| require("./gen-icons-json.cjs"); | ||||
| require("./rollup.cjs"); | ||||
| require("./translations.cjs"); | ||||
| require("./webpack.cjs"); | ||||
| 
 | ||||
| const env = require("../env"); | ||||
| 
 | ||||
| require("./clean.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| require("./webpack.js"); | ||||
| require("./compress.js"); | ||||
| require("./rollup.js"); | ||||
| require("./gather-static.js"); | ||||
| require("./translations.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| 
 | ||||
| gulp.task( | ||||
|   "develop-hassio", | ||||
| @@ -41,6 +43,6 @@ gulp.task( | ||||
|     env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio", | ||||
|     "gen-index-hassio-prod", | ||||
|     ...// Don't compress running tests
 | ||||
|     (env.isTestBuild() ? [] : ["compress-hassio"]) | ||||
|     (env.isTest() ? [] : ["compress-hassio"]) | ||||
|   ) | ||||
| ); | ||||
| @@ -1,12 +1,14 @@ | ||||
| const del = import("del"); | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| 
 | ||||
| const del = require("del"); | ||||
| const path = require("path"); | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const paths = require("../paths"); | ||||
| 
 | ||||
| const outDir = "build/locale-data"; | ||||
| 
 | ||||
| gulp.task("clean-locale-data", async () => (await del).deleteSync([outDir])); | ||||
| gulp.task("clean-locale-data", () => del([outDir])); | ||||
| 
 | ||||
| gulp.task("ensure-locale-data-build-dir", (done) => { | ||||
|   if (!fs.existsSync(outDir)) { | ||||
| @@ -5,9 +5,9 @@ const rollup = require("rollup"); | ||||
| const handler = require("serve-handler"); | ||||
| const http = require("http"); | ||||
| const log = require("fancy-log"); | ||||
| const rollupConfig = require("../rollup"); | ||||
| const paths = require("../paths"); | ||||
| const open = require("open"); | ||||
| const rollupConfig = require("../rollup.cjs"); | ||||
| const paths = require("../paths.cjs"); | ||||
| 
 | ||||
| const bothBuilds = (createConfigFunc, params) => | ||||
|   gulp.series( | ||||
| @@ -30,11 +30,11 @@ const bothBuilds = (createConfigFunc, params) => | ||||
|   ); | ||||
| 
 | ||||
| function createServer(serveOptions) { | ||||
|   const server = http.createServer((request, response) => | ||||
|     handler(request, response, { | ||||
|   const server = http.createServer((request, response) => { | ||||
|     return handler(request, response, { | ||||
|       public: serveOptions.root, | ||||
|     }) | ||||
|   ); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   server.listen( | ||||
|     serveOptions.port, | ||||
| @@ -46,7 +46,7 @@ function createServer(serveOptions) { | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| function watchRollup(createConfig, extraWatchSrc = [], serveOptions = null) { | ||||
| function watchRollup(createConfig, extraWatchSrc = [], serveOptions) { | ||||
|   const { inputOptions, outputOptions } = createConfig({ | ||||
|     isProdBuild: false, | ||||
|     latestBuild: true, | ||||
| @@ -1,11 +1,13 @@ | ||||
| // Generate service worker.
 | ||||
| // Based on manifest, create a file with the content as service_worker.js
 | ||||
| /* eslint-disable import/no-dynamic-require */ | ||||
| /* eslint-disable global-require */ | ||||
| const gulp = require("gulp"); | ||||
| const path = require("path"); | ||||
| const fs = require("fs-extra"); | ||||
| const workboxBuild = require("workbox-build"); | ||||
| const sourceMapUrl = require("source-map-url"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const paths = require("../paths.js"); | ||||
| 
 | ||||
| const swDest = path.resolve(paths.app_output_root, "service_worker.js"); | ||||
| 
 | ||||
| @@ -1,5 +1,7 @@ | ||||
| const del = import("del"); | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| 
 | ||||
| const crypto = require("crypto"); | ||||
| const del = require("del"); | ||||
| const path = require("path"); | ||||
| const source = require("vinyl-source-stream"); | ||||
| const vinylBuffer = require("vinyl-buffer"); | ||||
| @@ -9,11 +11,9 @@ const flatmap = require("gulp-flatmap"); | ||||
| const merge = require("gulp-merge-json"); | ||||
| const rename = require("gulp-rename"); | ||||
| const transform = require("gulp-json-transform"); | ||||
| const { mapFiles } = require("../util.cjs"); | ||||
| const env = require("../env.cjs"); | ||||
| const paths = require("../paths.cjs"); | ||||
| 
 | ||||
| require("./fetch-nightly-translations.cjs"); | ||||
| const { mapFiles } = require("../util"); | ||||
| const env = require("../env"); | ||||
| const paths = require("../paths"); | ||||
| 
 | ||||
| const inFrontendDir = "translations/frontend"; | ||||
| const inBackendDir = "translations/backend"; | ||||
| @@ -23,13 +23,10 @@ const coreDir = workDir + "/core"; | ||||
| const outDir = workDir + "/output"; | ||||
| let mergeBackend = false; | ||||
| 
 | ||||
| gulp.task( | ||||
|   "translations-enable-merge-backend", | ||||
|   gulp.parallel((done) => { | ||||
| gulp.task("translations-enable-merge-backend", (done) => { | ||||
|   mergeBackend = true; | ||||
|   done(); | ||||
|   }, "allow-setup-fetch-nightly-translations") | ||||
| ); | ||||
| }); | ||||
| 
 | ||||
| // Panel translations which should be split from the core translations.
 | ||||
| const TRANSLATION_FRAGMENTS = Object.keys( | ||||
| @@ -120,7 +117,7 @@ function lokaliseTransform(data, original, file) { | ||||
|   return output; | ||||
| } | ||||
| 
 | ||||
| gulp.task("clean-translations", async () => (await del).deleteSync([workDir])); | ||||
| gulp.task("clean-translations", () => del([workDir])); | ||||
| 
 | ||||
| gulp.task("ensure-translations-build-dir", (done) => { | ||||
|   if (!fs.existsSync(workDir)) { | ||||
| @@ -173,24 +170,17 @@ gulp.task("build-master-translation", () => { | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       merge({ | ||||
|         fileName: "en.json", | ||||
|         fileName: "translationMaster.json", | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(fullDir)); | ||||
|     .pipe(gulp.dest(workDir)); | ||||
| }); | ||||
| 
 | ||||
| gulp.task("build-merged-translations", () => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         inFrontendDir + "/*.json", | ||||
|         "!" + inFrontendDir + "/en.json", | ||||
|         workDir + "/test.json", | ||||
|       ], | ||||
|       { | ||||
|     .src([inFrontendDir + "/*.json", workDir + "/test.json"], { | ||||
|       allowEmpty: true, | ||||
|       } | ||||
|     ) | ||||
|     }) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       flatmap((stream, file) => { | ||||
| @@ -203,7 +193,7 @@ gulp.task("build-merged-translations", () => | ||||
|         //       than a base translation + region.
 | ||||
|         const tr = path.basename(file.history[0], ".json"); | ||||
|         const subtags = tr.split("-"); | ||||
|         const src = [fullDir + "/en.json"]; | ||||
|         const src = [workDir + "/translationMaster.json"]; | ||||
|         for (let i = 1; i <= subtags.length; i++) { | ||||
|           const lang = subtags.slice(0, i).join("-"); | ||||
|           if (lang === "test") { | ||||
| @@ -388,6 +378,7 @@ gulp.task("build-translation-write-metadata", () => | ||||
|           if (value.nativeName) { | ||||
|             newData[key] = value; | ||||
|           } else { | ||||
|             // eslint-disable-next-line no-console
 | ||||
|             console.warn( | ||||
|               `Skipping language ${key}. Native name was not translated.` | ||||
|             ); | ||||
| @@ -420,10 +411,8 @@ gulp.task( | ||||
| gulp.task( | ||||
|   "build-translations", | ||||
|   gulp.series( | ||||
|     gulp.parallel( | ||||
|       "fetch-nightly-translations", | ||||
|       gulp.series("clean-translations", "ensure-translations-build-dir") | ||||
|     ), | ||||
|     "clean-translations", | ||||
|     "ensure-translations-build-dir", | ||||
|     "create-translations", | ||||
|     "build-translation-fingerprints", | ||||
|     "build-translation-write-metadata" | ||||
| @@ -433,10 +422,8 @@ gulp.task( | ||||
| gulp.task( | ||||
|   "build-supervisor-translations", | ||||
|   gulp.series( | ||||
|     gulp.parallel( | ||||
|       "fetch-nightly-translations", | ||||
|       gulp.series("clean-translations", "ensure-translations-build-dir") | ||||
|     ), | ||||
|     "clean-translations", | ||||
|     "ensure-translations-build-dir", | ||||
|     "build-master-translation", | ||||
|     "build-merged-translations", | ||||
|     "build-translation-fragment-supervisor", | ||||
| @@ -1,3 +1,4 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| // Tasks to run webpack.
 | ||||
| const fs = require("fs"); | ||||
| const gulp = require("gulp"); | ||||
| @@ -5,15 +6,14 @@ const webpack = require("webpack"); | ||||
| const WebpackDevServer = require("webpack-dev-server"); | ||||
| const log = require("fancy-log"); | ||||
| const path = require("path"); | ||||
| const env = require("../env.cjs"); | ||||
| const paths = require("../paths.cjs"); | ||||
| const paths = require("../paths"); | ||||
| const { | ||||
|   createAppConfig, | ||||
|   createDemoConfig, | ||||
|   createCastConfig, | ||||
|   createHassioConfig, | ||||
|   createGalleryConfig, | ||||
| } = require("../webpack.cjs"); | ||||
| } = require("../webpack"); | ||||
| 
 | ||||
| const bothBuilds = (createConfigFunc, params) => [ | ||||
|   createConfigFunc({ ...params, latestBuild: true }), | ||||
| @@ -69,6 +69,7 @@ const doneHandler = (done) => (err, stats) => { | ||||
|   } | ||||
| 
 | ||||
|   if (stats.hasErrors() || stats.hasWarnings()) { | ||||
|     // eslint-disable-next-line no-console
 | ||||
|     console.log(stats.toString("minimal")); | ||||
|   } | ||||
| 
 | ||||
| @@ -105,8 +106,6 @@ gulp.task("webpack-prod-app", () => | ||||
|   prodBuild( | ||||
|     bothBuilds(createAppConfig, { | ||||
|       isProdBuild: true, | ||||
|       isStatsBuild: env.isStatsBuild(), | ||||
|       isTestBuild: env.isTestBuild(), | ||||
|     }) | ||||
|   ) | ||||
| ); | ||||
| @@ -164,8 +163,6 @@ gulp.task("webpack-prod-hassio", () => | ||||
|   prodBuild( | ||||
|     bothBuilds(createHassioConfig, { | ||||
|       isProdBuild: true, | ||||
|       isStatsBuild: env.isStatsBuild(), | ||||
|       isTestBuild: env.isTestBuild(), | ||||
|     }) | ||||
|   ) | ||||
| ); | ||||
| @@ -1,3 +1,4 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| 
 | ||||
| module.exports = { | ||||
| @@ -1 +1,30 @@ | ||||
| [] | ||||
| [ | ||||
|   { | ||||
|     "path": "M20,20H7A2,2 0 0,1 5,18V8.94L2.23,5.64C2.09,5.47 2,5.24 2,5A1,1 0 0,1 3,4H20A2,2 0 0,1 22,6V18A2,2 0 0,1 20,20M8.5,7A0.5,0.5 0 0,0 8,7.5V8.5A0.5,0.5 0 0,0 8.5,9H18.5A0.5,0.5 0 0,0 19,8.5V7.5A0.5,0.5 0 0,0 18.5,7H8.5M8.5,11A0.5,0.5 0 0,0 8,11.5V12.5A0.5,0.5 0 0,0 8.5,13H18.5A0.5,0.5 0 0,0 19,12.5V11.5A0.5,0.5 0 0,0 18.5,11H8.5M8.5,15A0.5,0.5 0 0,0 8,15.5V16.5A0.5,0.5 0 0,0 8.5,17H13.5A0.5,0.5 0 0,0 14,16.5V15.5A0.5,0.5 0 0,0 13.5,15H8.5Z", | ||||
|     "name": "android-messages" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M4,6H2V20A2,2 0 0,0 4,22H18V20H4V6M20,2H8A2,2 0 0,0 6,4V16A2,2 0 0,0 8,18H20A2,2 0 0,0 22,16V4A2,2 0 0,0 20,2M20,12L17.5,10.5L15,12V4H20V12Z", | ||||
|     "name": "book-variant-multiple" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M21,14H3V4H21M21,2H3C1.89,2 1,2.89 1,4V16A2,2 0 0,0 3,18H10L8,21V22H16V21L14,18H21A2,2 0 0,0 23,16V4C23,2.89 22.1,2 21,2Z", | ||||
|     "name": "desktop-mac" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M21,14V4H3V14H21M21,2A2,2 0 0,1 23,4V16A2,2 0 0,1 21,18H14L16,21V22H8V21L10,18H3C1.89,18 1,17.1 1,16V4C1,2.89 1.89,2 3,2H21M4,5H15V10H4V5M16,5H20V7H16V5M20,8V13H16V8H20M4,11H9V13H4V11M10,11H15V13H10V11Z", | ||||
|     "name": "desktop-mac-dashboard" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M22,24L16.75,19L17.38,21H4.5A2.5,2.5 0 0,1 2,18.5V3.5A2.5,2.5 0 0,1 4.5,1H19.5A2.5,2.5 0 0,1 22,3.5V24M12,6.8C9.32,6.8 7.44,7.95 7.44,7.95C8.47,7.03 10.27,6.5 10.27,6.5L10.1,6.33C8.41,6.36 6.88,7.53 6.88,7.53C5.16,11.12 5.27,14.22 5.27,14.22C6.67,16.03 8.75,15.9 8.75,15.9L9.46,15C8.21,14.73 7.42,13.62 7.42,13.62C7.42,13.62 9.3,14.9 12,14.9C14.7,14.9 16.58,13.62 16.58,13.62C16.58,13.62 15.79,14.73 14.54,15L15.25,15.9C15.25,15.9 17.33,16.03 18.73,14.22C18.73,14.22 18.84,11.12 17.12,7.53C17.12,7.53 15.59,6.36 13.9,6.33L13.73,6.5C13.73,6.5 15.53,7.03 16.56,7.95C16.56,7.95 14.68,6.8 12,6.8M9.93,10.59C10.58,10.59 11.11,11.16 11.1,11.86C11.1,12.55 10.58,13.13 9.93,13.13C9.29,13.13 8.77,12.55 8.77,11.86C8.77,11.16 9.28,10.59 9.93,10.59M14.1,10.59C14.75,10.59 15.27,11.16 15.27,11.86C15.27,12.55 14.75,13.13 14.1,13.13C13.46,13.13 12.94,12.55 12.94,11.86C12.94,11.16 13.45,10.59 14.1,10.59Z", | ||||
|     "name": "discord" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M8.06,7.78C7.5,7.78 7.17,7.73 7.08,7.64L6.66,13.73C7.19,14.05 7.88,14.3 8.72,14.5C9.56,14.71 10.78,14.77 12.38,14.67C13.97,14.58 15.63,14.23 17.34,13.64L16.55,4.22C15.67,5.09 14.38,5.91 12.66,6.66C11.13,7.31 9.81,7.69 8.72,7.78H8.06M7.97,5.34C7.28,5.94 7,6.34 7.13,6.56C7.22,6.78 7.7,6.84 8.58,6.75C9.67,6.66 10.91,6.31 12.28,5.72C13.22,5.31 14.03,4.88 14.72,4.41C15.41,3.94 15.88,3.55 16.13,3.23C16.38,2.92 16.47,2.7 16.41,2.58C16.34,2.42 16.03,2.34 15.47,2.34C14.34,2.34 12.94,2.7 11.25,3.42C9.81,4.05 8.72,4.69 7.97,5.34M17.34,2.2C17.41,2.33 17.44,2.47 17.44,2.63L18.61,17C18.61,18.73 18,20.09 16.83,21.07C15.64,22.05 14.03,22.55 12,22.55C10,22.55 8.4,22.04 7.2,21C6,20 5.39,18.64 5.39,16.92L6.09,6.47C6.09,6.22 6.2,5.94 6.42,5.63C6.64,5.31 6.84,5.06 7.03,4.88L7.36,4.59C8.33,3.78 9.5,3.08 10.88,2.5C11.81,2.08 12.73,1.77 13.62,1.57C14.5,1.37 15.3,1.3 16,1.38C16.71,1.46 17.16,1.73 17.34,2.2Z", | ||||
|     "name": "google-home" | ||||
|   }, | ||||
|   { | ||||
|     "path": "M19.25,19H4.75V3H19.25M14,22H10V21H14M18,0H6A3,3 0 0,0 3,3V21A3,3 0 0,0 6,24H18A3,3 0 0,0 21,21V3A3,3 0 0,0 18,0Z", | ||||
|     "name": "tablet-android" | ||||
|   } | ||||
| ] | ||||
|   | ||||
| @@ -81,13 +81,13 @@ module.exports = function (opts = {}) { | ||||
|         opts.workerRegexp.flags | ||||
|       ); | ||||
|       if (!workerRegexp.test(code)) { | ||||
|         return undefined; | ||||
|         return; | ||||
|       } | ||||
| 
 | ||||
|       const ms = new MagicString(code); | ||||
|       // Reset the regexp
 | ||||
|       workerRegexp.lastIndex = 0; | ||||
|       for (;;) { | ||||
|       while (true) { | ||||
|         const match = workerRegexp.exec(code); | ||||
|         if (!match) { | ||||
|           break; | ||||
| @@ -98,12 +98,11 @@ module.exports = function (opts = {}) { | ||||
|         // Parse the optional options object
 | ||||
|         if (match[3] && match[3].length > 0) { | ||||
|           // FIXME: ooooof!
 | ||||
|           // eslint-disable-next-line @typescript-eslint/no-implied-eval
 | ||||
|           optionsObject = new Function(`return ${match[3].slice(1)};`)(); | ||||
|         } | ||||
|         delete optionsObject.type; | ||||
| 
 | ||||
|         if (!/^.*\//.test(workerFile)) { | ||||
|         if (!new RegExp("^.*/").test(workerFile)) { | ||||
|           this.warn( | ||||
|             `Paths passed to the Worker constructor must be relative or absolute, i.e. start with /, ./ or ../ (just like dynamic import!). Ignoring "${workerFile}".` | ||||
|           ); | ||||
| @@ -111,14 +110,12 @@ module.exports = function (opts = {}) { | ||||
|         } | ||||
| 
 | ||||
|         // Find worker file and store it as a chunk with ID prefixed for our loader
 | ||||
|         // eslint-disable-next-line no-await-in-loop
 | ||||
|         const resolvedWorkerFile = (await this.resolve(workerFile, id)).id; | ||||
|         let chunkRefId; | ||||
|         if (resolvedWorkerFile in refIds) { | ||||
|           chunkRefId = refIds[resolvedWorkerFile]; | ||||
|         } else { | ||||
|           this.addWatchFile(resolvedWorkerFile); | ||||
|           // eslint-disable-next-line no-await-in-loop
 | ||||
|           const source = await getBundledWorker( | ||||
|             resolvedWorkerFile, | ||||
|             rollupOptions | ||||
| @@ -1,20 +1,21 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| 
 | ||||
| const commonjs = require("@rollup/plugin-commonjs"); | ||||
| const resolve = require("@rollup/plugin-node-resolve"); | ||||
| const json = require("@rollup/plugin-json"); | ||||
| const { babel } = require("@rollup/plugin-babel"); | ||||
| const babel = require("@rollup/plugin-babel").babel; | ||||
| const replace = require("@rollup/plugin-replace"); | ||||
| const visualizer = require("rollup-plugin-visualizer"); | ||||
| const { string } = require("rollup-plugin-string"); | ||||
| const { terser } = require("rollup-plugin-terser"); | ||||
| const manifest = require("./rollup-plugins/manifest-plugin.cjs"); | ||||
| const worker = require("./rollup-plugins/worker-plugin.cjs"); | ||||
| const dontHashPlugin = require("./rollup-plugins/dont-hash-plugin.cjs"); | ||||
| const ignore = require("./rollup-plugins/ignore-plugin.cjs"); | ||||
| const manifest = require("./rollup-plugins/manifest-plugin"); | ||||
| const worker = require("./rollup-plugins/worker-plugin"); | ||||
| const dontHashPlugin = require("./rollup-plugins/dont-hash-plugin"); | ||||
| const ignore = require("./rollup-plugins/ignore-plugin"); | ||||
| 
 | ||||
| const bundle = require("./bundle.cjs"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const bundle = require("./bundle"); | ||||
| const paths = require("./paths"); | ||||
| 
 | ||||
| const extensions = [".js", ".ts"]; | ||||
| 
 | ||||
| @@ -39,18 +40,11 @@ const createRollupConfig = ({ | ||||
|   inputOptions: { | ||||
|     input: entry, | ||||
|     // Some entry points contain no JavaScript. This setting silences a warning about that.
 | ||||
|     // https://rollupjs.org/configuration-options/#preserveentrysignatures
 | ||||
|     // https://rollupjs.org/guide/en/#preserveentrysignatures
 | ||||
|     preserveEntrySignatures: false, | ||||
|     plugins: [ | ||||
|       ignore({ | ||||
|         files: bundle | ||||
|           .emptyPackages({ latestBuild }) | ||||
|           // TEMP HACK: Makes Rollup build work again
 | ||||
|           .concat( | ||||
|             require.resolve( | ||||
|               "@webcomponents/scoped-custom-element-registry/scoped-custom-element-registry.min" | ||||
|             ) | ||||
|           ), | ||||
|         files: bundle.emptyPackages({ latestBuild }), | ||||
|       }), | ||||
|       resolve({ | ||||
|         extensions, | ||||
| @@ -61,7 +55,7 @@ const createRollupConfig = ({ | ||||
|       commonjs(), | ||||
|       json(), | ||||
|       babel({ | ||||
|         ...bundle.babelOptions({ latestBuild, isProdBuild }), | ||||
|         ...bundle.babelOptions({ latestBuild }), | ||||
|         extensions, | ||||
|         babelHelpers: isWDS ? "inline" : "bundled", | ||||
|       }), | ||||
| @@ -76,7 +70,7 @@ const createRollupConfig = ({ | ||||
|         }), | ||||
|       !isWDS && worker(), | ||||
|       !isWDS && dontHashPlugin({ dontHash }), | ||||
|       !isWDS && isProdBuild && terser(bundle.terserOptions({ latestBuild })), | ||||
|       !isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)), | ||||
|       !isWDS && | ||||
|         isStatsBuild && | ||||
|         visualizer({ | ||||
| @@ -90,20 +84,20 @@ const createRollupConfig = ({ | ||||
|    * @type { import("rollup").OutputOptions } | ||||
|    */ | ||||
|   outputOptions: { | ||||
|     // https://rollupjs.org/configuration-options/#output-dir
 | ||||
|     // https://rollupjs.org/guide/en/#outputdir
 | ||||
|     dir: outputPath, | ||||
|     // https://rollupjs.org/configuration-options/#output-format
 | ||||
|     // https://rollupjs.org/guide/en/#outputformat
 | ||||
|     format: latestBuild ? "es" : "systemjs", | ||||
|     // https://rollupjs.org/configuration-options/#output-externallivebindings
 | ||||
|     // https://rollupjs.org/guide/en/#outputexternallivebindings
 | ||||
|     externalLiveBindings: false, | ||||
|     // https://rollupjs.org/configuration-options/#output-entryfilenames
 | ||||
|     // https://rollupjs.org/configuration-options/#output-chunkfilenames
 | ||||
|     // https://rollupjs.org/configuration-options/#output-assetfilenames
 | ||||
|     // https://rollupjs.org/guide/en/#outputentryfilenames
 | ||||
|     // https://rollupjs.org/guide/en/#outputchunkfilenames
 | ||||
|     // https://rollupjs.org/guide/en/#outputassetfilenames
 | ||||
|     entryFileNames: | ||||
|       isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js", | ||||
|     chunkFileNames: isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js", | ||||
|     assetFileNames: isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js", | ||||
|     // https://rollupjs.org/configuration-options/#output-sourcemap
 | ||||
|     // https://rollupjs.org/guide/en/#outputsourcemap
 | ||||
|     sourcemap: isProdBuild ? true : "inline", | ||||
|   }, | ||||
| }); | ||||
| @@ -1,3 +1,4 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| const fs = require("fs"); | ||||
| 
 | ||||
| @@ -1,11 +1,12 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const webpack = require("webpack"); | ||||
| const path = require("path"); | ||||
| const TerserPlugin = require("terser-webpack-plugin"); | ||||
| const { WebpackManifestPlugin } = require("webpack-manifest-plugin"); | ||||
| const log = require("fancy-log"); | ||||
| const WebpackBar = require("webpackbar"); | ||||
| const paths = require("./paths.cjs"); | ||||
| const bundle = require("./bundle.cjs"); | ||||
| const paths = require("./paths.js"); | ||||
| const bundle = require("./bundle.js"); | ||||
| 
 | ||||
| class LogStartCompilePlugin { | ||||
|   ignoredFirst = false; | ||||
| @@ -22,7 +23,6 @@ class LogStartCompilePlugin { | ||||
| } | ||||
| 
 | ||||
| const createWebpackConfig = ({ | ||||
|   name, | ||||
|   entry, | ||||
|   outputPath, | ||||
|   publicPath, | ||||
| @@ -30,7 +30,6 @@ const createWebpackConfig = ({ | ||||
|   isProdBuild, | ||||
|   latestBuild, | ||||
|   isStatsBuild, | ||||
|   isTestBuild, | ||||
|   isHassioBuild, | ||||
|   dontHash, | ||||
| }) => { | ||||
| @@ -39,16 +38,10 @@ const createWebpackConfig = ({ | ||||
|   } | ||||
|   const ignorePackages = bundle.ignorePackages({ latestBuild }); | ||||
|   return { | ||||
|     name, | ||||
|     mode: isProdBuild ? "production" : "development", | ||||
|     target: ["web", latestBuild ? "es2017" : "es5"], | ||||
|     // For tests/CI, source maps are skipped to gain build speed
 | ||||
|     // For production, generate source maps for accurate stack traces without source code
 | ||||
|     // For development, generate "cheap" versions that can map to original line numbers
 | ||||
|     devtool: isTestBuild | ||||
|       ? false | ||||
|       : isProdBuild | ||||
|       ? "nosources-source-map" | ||||
|     devtool: isProdBuild | ||||
|       ? "cheap-module-source-map" | ||||
|       : "eval-cheap-module-source-map", | ||||
|     entry, | ||||
|     node: false, | ||||
| @@ -59,14 +52,11 @@ const createWebpackConfig = ({ | ||||
|           use: { | ||||
|             loader: "babel-loader", | ||||
|             options: { | ||||
|               ...bundle.babelOptions({ latestBuild, isProdBuild, isTestBuild }), | ||||
|               ...bundle.babelOptions({ latestBuild }), | ||||
|               cacheDirectory: !isProdBuild, | ||||
|               cacheCompression: false, | ||||
|             }, | ||||
|           }, | ||||
|           resolve: { | ||||
|             fullySpecified: false, | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           test: /\.css$/, | ||||
| @@ -79,7 +69,7 @@ const createWebpackConfig = ({ | ||||
|         new TerserPlugin({ | ||||
|           parallel: true, | ||||
|           extractComments: true, | ||||
|           terserOptions: bundle.terserOptions({ latestBuild, isTestBuild }), | ||||
|           terserOptions: bundle.terserOptions(latestBuild), | ||||
|         }), | ||||
|       ], | ||||
|       moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", | ||||
| @@ -113,6 +103,7 @@ const createWebpackConfig = ({ | ||||
|               ? path.resolve(context, resource) | ||||
|               : require.resolve(resource); | ||||
|           } catch (err) { | ||||
|             // eslint-disable-next-line no-console
 | ||||
|             console.error( | ||||
|               "Error in Home Assistant ignore plugin", | ||||
|               resource, | ||||
| @@ -152,37 +143,18 @@ const createWebpackConfig = ({ | ||||
|       }, | ||||
|     }, | ||||
|     output: { | ||||
|       filename: ({ chunk }) => | ||||
|         !isProdBuild || isStatsBuild || dontHash.has(chunk.name) | ||||
|           ? "[name].js" | ||||
|           : "[name]-[contenthash].js", | ||||
|       filename: ({ chunk }) => { | ||||
|         if (!isProdBuild || isStatsBuild || dontHash.has(chunk.name)) { | ||||
|           return `${chunk.name}.js`; | ||||
|         } | ||||
|         return `${chunk.name}.${chunk.hash.substr(0, 8)}.js`; | ||||
|       }, | ||||
|       chunkFilename: | ||||
|         isProdBuild && !isStatsBuild ? "[id]-[contenthash].js" : "[name].js", | ||||
|       assetModuleFilename: | ||||
|         isProdBuild && !isStatsBuild ? "[id]-[contenthash][ext]" : "[id][ext]", | ||||
|       hashFunction: "xxhash64", | ||||
|       hashDigest: "base64url", | ||||
|       hashDigestLength: 11, // full length of 64 bit base64url
 | ||||
|         isProdBuild && !isStatsBuild ? "[chunkhash:8].js" : "[id].chunk.js", | ||||
|       path: outputPath, | ||||
|       publicPath, | ||||
|       // To silence warning in worker plugin
 | ||||
|       globalObject: "self", | ||||
|       // Since production source maps don't include sources, we need to point to them elsewhere
 | ||||
|       // For dependencies, just provide the path (no source in browser)
 | ||||
|       // Otherwise, point to the raw code on GitHub for browser to load
 | ||||
|       devtoolModuleFilenameTemplate: | ||||
|         !isTestBuild && isProdBuild | ||||
|           ? (info) => { | ||||
|               const sourcePath = info.resourcePath.replace(/^\.\//, ""); | ||||
|               if ( | ||||
|                 sourcePath.startsWith("node_modules") || | ||||
|                 sourcePath.startsWith("webpack") | ||||
|               ) { | ||||
|                 return `no-source/${sourcePath}`; | ||||
|               } | ||||
|               return `${bundle.sourceMapURL()}/${sourcePath}`; | ||||
|             } | ||||
|           : undefined, | ||||
|     }, | ||||
|     experiments: { | ||||
|       topLevelAwait: true, | ||||
| @@ -190,14 +162,9 @@ const createWebpackConfig = ({ | ||||
|   }; | ||||
| }; | ||||
| 
 | ||||
| const createAppConfig = ({ | ||||
|   isProdBuild, | ||||
|   latestBuild, | ||||
|   isStatsBuild, | ||||
|   isTestBuild, | ||||
| }) => | ||||
| const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => | ||||
|   createWebpackConfig( | ||||
|     bundle.config.app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild }) | ||||
|     bundle.config.app({ isProdBuild, latestBuild, isStatsBuild }) | ||||
|   ); | ||||
| 
 | ||||
| const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => | ||||
| @@ -208,20 +175,8 @@ const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => | ||||
| const createCastConfig = ({ isProdBuild, latestBuild }) => | ||||
|   createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild })); | ||||
| 
 | ||||
| const createHassioConfig = ({ | ||||
|   isProdBuild, | ||||
|   latestBuild, | ||||
|   isStatsBuild, | ||||
|   isTestBuild, | ||||
| }) => | ||||
|   createWebpackConfig( | ||||
|     bundle.config.hassio({ | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       isStatsBuild, | ||||
|       isTestBuild, | ||||
|     }) | ||||
|   ); | ||||
| const createHassioConfig = ({ isProdBuild, latestBuild }) => | ||||
|   createWebpackConfig(bundle.config.hassio({ isProdBuild, latestBuild })); | ||||
| 
 | ||||
| const createGalleryConfig = ({ isProdBuild, latestBuild }) => | ||||
|   createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild })); | ||||
| @@ -1,5 +1,5 @@ | ||||
| import rollup from "../build-scripts/rollup.cjs"; | ||||
| import env from "../build-scripts/env.cjs"; | ||||
| const rollup = require("../build-scripts/rollup.js"); | ||||
| const env = require("../build-scripts/env.js"); | ||||
|  | ||||
| const config = rollup.createCastConfig({ | ||||
|   isProdBuild: env.isProdBuild(), | ||||
| @@ -7,4 +7,4 @@ const config = rollup.createCastConfig({ | ||||
|   isStatsBuild: env.isStatsBuild(), | ||||
| }); | ||||
|  | ||||
| export default { ...config.inputOptions, output: config.outputOptions }; | ||||
| module.exports = { ...config.inputOptions, output: config.outputOptions }; | ||||
|   | ||||
| @@ -213,7 +213,7 @@ | ||||
|         </p> | ||||
|         <ul> | ||||
|           <li>Google Chrome (all platforms except iOS)</li> | ||||
|           <li>Microsoft Edge (all platforms except iOS)</li> | ||||
|           <li>Microsoft Edge (all platforms)</li> | ||||
|         </ul> | ||||
|       </div> | ||||
|  | ||||
|   | ||||
| @@ -88,7 +88,7 @@ class HcCast extends LitElement { | ||||
|               > | ||||
|                 ${(this.lovelaceConfig | ||||
|                   ? this.lovelaceConfig.views | ||||
|                   : [generateDefaultViewConfig({}, {}, {}, {}, () => "")] | ||||
|                   : [generateDefaultViewConfig([], [], [], {}, () => "")] | ||||
|                 ).map( | ||||
|                   (view, idx) => html` | ||||
|                     <paper-icon-item | ||||
| @@ -181,7 +181,7 @@ class HcCast extends LitElement { | ||||
|   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); | ||||
|     castSendShowLovelaceView(this.castManager, path); | ||||
|   } | ||||
|  | ||||
|   private async _handleLogout() { | ||||
|   | ||||
| @@ -22,11 +22,7 @@ class HcLayout extends LitElement { | ||||
|     return html` | ||||
|       <ha-card> | ||||
|         <div class="layout"> | ||||
|           <img | ||||
|             class="hero" | ||||
|             alt="A Google Nest Hub with a Home Assistant dashboard on its screen" | ||||
|             src="/images/google-nest-hub.png" | ||||
|           /> | ||||
|           <img class="hero" src="/images/google-nest-hub.png" /> | ||||
|           <h1 class="card-header"> | ||||
|             Home Assistant Cast${this.subtitle ? ` – ${this.subtitle}` : ""} | ||||
|             ${this.auth | ||||
| @@ -48,7 +44,7 @@ class HcLayout extends LitElement { | ||||
|       <div class="footer"> | ||||
|         <a href="./faq.html">Frequently Asked Questions</a> – Found a bug? | ||||
|         <a | ||||
|           href="https://github.com/home-assistant/frontend/issues" | ||||
|           href="https://github.com/home-assistant/home-assistant-polymer/issues" | ||||
|           target="_blank" | ||||
|           >Let us know!</a | ||||
|         > | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { html, nothing } from "lit"; | ||||
| import { html, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { mockHistory } from "../../../../demo/src/stubs/history"; | ||||
| import { LovelaceConfig } from "../../../../src/data/lovelace"; | ||||
| @@ -18,9 +18,9 @@ class HcDemo extends HassElement { | ||||
|  | ||||
|   @state() private _lovelaceConfig?: LovelaceConfig; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this._lovelaceConfig) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <hc-lovelace | ||||
|   | ||||
| @@ -12,7 +12,6 @@ class HcLaunchScreen extends LitElement { | ||||
|     return html` | ||||
|       <div class="container"> | ||||
|         <img | ||||
|           alt="Home Assistant logo on left, Nabu Casa logo on right, and red heart in center" | ||||
|           src="https://www.home-assistant.io/images/blog/2018-09-thinking-big/social.png" | ||||
|         /> | ||||
|         <div class="status"> | ||||
|   | ||||
| @@ -33,6 +33,7 @@ import { castContext } from "../cast_context"; | ||||
| import "./hc-launch-screen"; | ||||
|  | ||||
| let resourcesLoaded = false; | ||||
|  | ||||
| @customElement("hc-main") | ||||
| export class HcMain extends HassElement { | ||||
|   @state() private _showDemo = false; | ||||
| @@ -45,8 +46,6 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|   @state() private _urlPath?: string | null; | ||||
|  | ||||
|   private _hassUUID?: string; | ||||
|  | ||||
|   private _unsubLovelace?: UnsubscribeFunc; | ||||
|  | ||||
|   public processIncomingMessage(msg: HassMessage) { | ||||
| @@ -126,7 +125,6 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|     if (this.hass) { | ||||
|       status.hassUrl = this.hass.auth.data.hassUrl; | ||||
|       status.hassUUID = this._hassUUID; | ||||
|       status.lovelacePath = this._lovelacePath; | ||||
|       status.urlPath = this._urlPath; | ||||
|     } | ||||
| @@ -165,18 +163,6 @@ export class HcMain extends HassElement { | ||||
|   }; | ||||
|  | ||||
|   private async _handleGetStatusMessage(msg: GetStatusMessage) { | ||||
|     if ( | ||||
|       (this.hass && msg.hassUUID && msg.hassUUID !== this._hassUUID) || | ||||
|       (this.hass && msg.hassUrl && msg.hassUrl !== this.hass.auth.data.hassUrl) | ||||
|     ) { | ||||
|       this._error = "Not connected to the same Home Assistant instance."; | ||||
|       this._sendError( | ||||
|         ReceiverErrorCode.WRONG_INSTANCE, | ||||
|         this._error, | ||||
|         msg.senderId! | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     this._sendStatus(msg.senderId!); | ||||
|   } | ||||
|  | ||||
| @@ -193,7 +179,6 @@ export class HcMain extends HassElement { | ||||
|           expires_in: 0, | ||||
|         }), | ||||
|       }); | ||||
|       this._hassUUID = msg.hassUUID; | ||||
|     } catch (err: any) { | ||||
|       const errorMessage = this._getErrorMessage(err); | ||||
|       this._error = errorMessage; | ||||
| @@ -224,50 +209,14 @@ export class HcMain extends HassElement { | ||||
|     if (!this.hass) { | ||||
|       this._sendStatus(msg.senderId!); | ||||
|       this._error = "Cannot show Lovelace because we're not connected."; | ||||
|       this._sendError( | ||||
|         ReceiverErrorCode.NOT_CONNECTED, | ||||
|         this._error, | ||||
|         msg.senderId! | ||||
|       ); | ||||
|       this._sendError(ReceiverErrorCode.NOT_CONNECTED, this._error); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if ( | ||||
|       (msg.hassUUID && msg.hassUUID !== this._hassUUID) || | ||||
|       (msg.hassUrl && msg.hassUrl !== this.hass.auth.data.hassUrl) | ||||
|     ) { | ||||
|       this._sendStatus(msg.senderId!); | ||||
|       this._error = | ||||
|         "Cannot show Lovelace because we're not connected to the same Home Assistant instance."; | ||||
|       this._sendError( | ||||
|         ReceiverErrorCode.WRONG_INSTANCE, | ||||
|         this._error, | ||||
|         msg.senderId! | ||||
|       ); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     this._error = undefined; | ||||
|     if (msg.urlPath === "lovelace") { | ||||
|       msg.urlPath = null; | ||||
|     } | ||||
|     this._lovelacePath = msg.viewPath; | ||||
|     if (msg.urlPath === "energy") { | ||||
|       this._lovelaceConfig = { | ||||
|         views: [ | ||||
|           { | ||||
|             strategy: { | ||||
|               type: "energy", | ||||
|               options: { show_date_selection: true }, | ||||
|             }, | ||||
|           }, | ||||
|         ], | ||||
|       }; | ||||
|       this._urlPath = "energy"; | ||||
|       this._lovelacePath = 0; | ||||
|       this._sendStatus(); | ||||
|       return; | ||||
|     } | ||||
|     if (!this._unsubLovelace || this._urlPath !== msg.urlPath) { | ||||
|       this._urlPath = msg.urlPath; | ||||
|       this._lovelaceConfig = undefined; | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import webpack from "../build-scripts/webpack.cjs"; | ||||
| import env from "../build-scripts/env.cjs"; | ||||
| const { createCastConfig } = require("../build-scripts/webpack.js"); | ||||
| const { isProdBuild, isStatsBuild } = require("../build-scripts/env.js"); | ||||
|  | ||||
| export default webpack.createCastConfig({ | ||||
|   isProdBuild: env.isProdBuild(), | ||||
|   isStatsBuild: env.isStatsBuild(), | ||||
| module.exports = createCastConfig({ | ||||
|   isProdBuild: isProdBuild(), | ||||
|   isStatsBuild: isStatsBuild(), | ||||
|   latestBuild: true, | ||||
| }); | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import rollup from "../build-scripts/rollup.cjs"; | ||||
| import env from "../build-scripts/env.cjs"; | ||||
| const rollup = require("../build-scripts/rollup.js"); | ||||
| const env = require("../build-scripts/env.js"); | ||||
|  | ||||
| const config = rollup.createDemoConfig({ | ||||
|   isProdBuild: env.isProdBuild(), | ||||
| @@ -7,4 +7,4 @@ const config = rollup.createDemoConfig({ | ||||
|   isStatsBuild: env.isStatsBuild(), | ||||
| }); | ||||
|  | ||||
| export default { ...config.inputOptions, output: config.outputOptions }; | ||||
| module.exports = { ...config.inputOptions, output: config.outputOptions }; | ||||
|   | ||||
| @@ -6,9 +6,6 @@ set -e | ||||
|  | ||||
| cd "$(dirname "$0")/.." | ||||
|  | ||||
| export STATS=1 | ||||
| statsfile="compilation-stats-demo.json" | ||||
|  | ||||
| ./node_modules/.bin/webpack-cli --profile --node-env=production --json=$statsfile | ||||
| npx webpack-bundle-analyzer $statsfile dist/frontend_latest | ||||
| rm -f $statsfile | ||||
| STATS=1 NODE_ENV=production ../node_modules/.bin/webpack --profile --json > compilation-stats.json | ||||
| npx webpack-bundle-analyzer compilation-stats.json dist/frontend_latest | ||||
| rm compilation-stats.json | ||||
|   | ||||
| @@ -508,7 +508,7 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) => | ||||
|         origin_addresses: ["XYZ"], | ||||
|         status: "OK", | ||||
|         mode: "driving", | ||||
|         units: "us_customary", | ||||
|         units: "imperial", | ||||
|         duration_in_traffic: "41 mins", | ||||
|         duration: "44 mins", | ||||
|         distance: "34.3 mi", | ||||
| @@ -527,7 +527,7 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) => | ||||
|         origin_addresses: ["XYZ"], | ||||
|         status: "OK", | ||||
|         mode: "driving", | ||||
|         units: "us_customary", | ||||
|         units: "imperial", | ||||
|         duration_in_traffic: "37 mins", | ||||
|         duration: "37 mins", | ||||
|         distance: "30.2 mi", | ||||
|   | ||||
| @@ -1196,7 +1196,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | ||||
|                       left: "15%", | ||||
|                     }, | ||||
|                     type: "state-icon", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d00026e26dc", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d0002338651", | ||||
|                   }, | ||||
|                   { | ||||
|                     prefix: "Kitchen: ", | ||||
| @@ -1206,7 +1206,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | ||||
|                       top: "89%", | ||||
|                       left: "32%", | ||||
|                     }, | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d00026e26dc", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d0002338651", | ||||
|                   }, | ||||
|                   { | ||||
|                     style: { | ||||
| @@ -1215,7 +1215,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | ||||
|                       left: "60%", | ||||
|                     }, | ||||
|                     type: "state-icon", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d0002338651", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d00026e26dc", | ||||
|                   }, | ||||
|                   { | ||||
|                     prefix: "Bathroom: ", | ||||
| @@ -1225,7 +1225,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | ||||
|                       top: "89%", | ||||
|                       left: "77%", | ||||
|                     }, | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d0002338651", | ||||
|                     entity: "binary_sensor.water_leak_sensor_158d00026e26dc", | ||||
|                   }, | ||||
|                 ], | ||||
|                 type: "picture-elements", | ||||
|   | ||||
| @@ -138,7 +138,7 @@ if (!window.cardTools) { | ||||
|       return cardTools.createThing("row", config); | ||||
|  | ||||
|     const domain = config.entity.split(".", 1)[0]; | ||||
|     Object.assign(config, { type: DEFAULT_ROWS[domain] || "simple" }); | ||||
|     Object.assign(config, { type: DEFAULT_ROWS[domain] || "text" }); | ||||
|     return cardTools.createThing("entity-row", config); | ||||
|   }; | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { mdiTelevision } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { CastManager } from "../../../src/cast/cast_manager"; | ||||
| import { castSendShowDemo } from "../../../src/cast/receiver_messages"; | ||||
| @@ -20,12 +20,12 @@ class CastDemoRow extends LitElement implements LovelaceRow { | ||||
|     // No config possible. | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if ( | ||||
|       !this._castManager || | ||||
|       this._castManager.castState === "NO_DEVICES_AVAILABLE" | ||||
|     ) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <ha-svg-icon .path=${mdiTelevision}></ha-svg-icon> | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { css, CSSResultGroup, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { property, state } from "lit/decorators"; | ||||
| import { until } from "lit/directives/until"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-circular-progress"; | ||||
| @@ -14,7 +14,6 @@ import { | ||||
|   setDemoConfig, | ||||
| } from "../configs/demo-configs"; | ||||
|  | ||||
| @customElement("ha-demo-card") | ||||
| export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|   @property({ attribute: false }) public lovelace?: Lovelace; | ||||
|  | ||||
| @@ -30,9 +29,9 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|  | ||||
|   public setConfig(_config: LovelaceCardConfig) {} | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (this._hidden) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <ha-card> | ||||
| @@ -155,3 +154,5 @@ declare global { | ||||
|     "ha-demo-card": HADemoCard; | ||||
|   } | ||||
| } | ||||
|  | ||||
| customElements.define("ha-demo-card", HADemoCard); | ||||
|   | ||||
| @@ -1,6 +1,4 @@ | ||||
| // Compat needs to be first import | ||||
| import "../../src/resources/compatibility"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { isNavigationClick } from "../../src/common/dom/is-navigation-click"; | ||||
| import { navigate } from "../../src/common/navigate"; | ||||
| import { | ||||
| @@ -8,6 +6,7 @@ import { | ||||
|   provideHass, | ||||
| } from "../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistantAppEl } from "../../src/layouts/home-assistant"; | ||||
| import "../../src/resources/compatibility"; | ||||
| import { HomeAssistant } from "../../src/types"; | ||||
| import { selectedDemoConfig } from "./configs/demo-configs"; | ||||
| import { mockAuth } from "./stubs/auth"; | ||||
| @@ -21,14 +20,12 @@ import { mockHistory } from "./stubs/history"; | ||||
| import { mockLovelace } from "./stubs/lovelace"; | ||||
| import { mockMediaPlayer } from "./stubs/media_player"; | ||||
| import { mockPersistentNotification } from "./stubs/persistent_notification"; | ||||
| import { mockRecorder } from "./stubs/recorder"; | ||||
| import { mockShoppingList } from "./stubs/shopping_list"; | ||||
| import { mockSystemLog } from "./stubs/system_log"; | ||||
| import { mockTemplate } from "./stubs/template"; | ||||
| import { mockTranslations } from "./stubs/translations"; | ||||
|  | ||||
| @customElement("ha-demo") | ||||
| export class HaDemo extends HomeAssistantAppEl { | ||||
| class HaDemo extends HomeAssistantAppEl { | ||||
|   protected async _initializeHass() { | ||||
|     const initial: Partial<MockHomeAssistant> = { | ||||
|       panelUrl: (this as any)._panelUrl, | ||||
| @@ -48,7 +45,6 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|     mockAuth(hass); | ||||
|     mockTranslations(hass); | ||||
|     mockHistory(hass); | ||||
|     mockRecorder(hass); | ||||
|     mockShoppingList(hass); | ||||
|     mockSystemLog(hass); | ||||
|     mockTemplate(hass); | ||||
| @@ -65,15 +61,12 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         area_id: null, | ||||
|         disabled_by: null, | ||||
|         entity_id: "sensor.co2_intensity", | ||||
|         id: "sensor.co2_intensity", | ||||
|         name: null, | ||||
|         icon: null, | ||||
|         platform: "co2signal", | ||||
|         hidden_by: null, | ||||
|         entity_category: null, | ||||
|         has_entity_name: false, | ||||
|         unique_id: "co2_intensity", | ||||
|         options: null, | ||||
|       }, | ||||
|       { | ||||
|         config_entry_id: "co2signal", | ||||
| @@ -81,15 +74,12 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|         area_id: null, | ||||
|         disabled_by: null, | ||||
|         entity_id: "sensor.grid_fossil_fuel_percentage", | ||||
|         id: "sensor.co2_intensity", | ||||
|         name: null, | ||||
|         icon: null, | ||||
|         platform: "co2signal", | ||||
|         hidden_by: null, | ||||
|         entity_category: null, | ||||
|         has_entity_name: false, | ||||
|         unique_id: "grid_fossil_fuel_percentage", | ||||
|         options: null, | ||||
|       }, | ||||
|     ]); | ||||
|  | ||||
| @@ -125,8 +115,4 @@ export class HaDemo extends HomeAssistantAppEl { | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-demo": HaDemo; | ||||
|   } | ||||
| } | ||||
| customElements.define("ha-demo", HaDemo); | ||||
|   | ||||
| @@ -1,16 +1,9 @@ | ||||
| import { format, startOfToday, startOfTomorrow } from "date-fns/esm"; | ||||
| import { | ||||
|   EnergyInfo, | ||||
|   EnergyPreferences, | ||||
|   EnergySolarForecasts, | ||||
|   FossilEnergyConsumption, | ||||
| } from "../../../src/data/energy"; | ||||
| import { EnergySolarForecasts } from "../../../src/data/energy"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS( | ||||
|     "energy/get_prefs", | ||||
|     (): EnergyPreferences => ({ | ||||
|   hass.mockWS("energy/get_prefs", () => ({ | ||||
|     energy_sources: [ | ||||
|       { | ||||
|         type: "grid", | ||||
| @@ -18,12 +11,14 @@ export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|           { | ||||
|             stat_energy_from: "sensor.energy_consumption_tarif_1", | ||||
|             stat_cost: "sensor.energy_consumption_tarif_1_cost", | ||||
|             entity_energy_from: "sensor.energy_consumption_tarif_1", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|           { | ||||
|             stat_energy_from: "sensor.energy_consumption_tarif_2", | ||||
|             stat_cost: "sensor.energy_consumption_tarif_2_cost", | ||||
|             entity_energy_from: "sensor.energy_consumption_tarif_2", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
| @@ -31,15 +26,15 @@ export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|         flow_to: [ | ||||
|           { | ||||
|             stat_energy_to: "sensor.energy_production_tarif_1", | ||||
|               stat_compensation: | ||||
|                 "sensor.energy_production_tarif_1_compensation", | ||||
|             stat_compensation: "sensor.energy_production_tarif_1_compensation", | ||||
|             entity_energy_to: "sensor.energy_production_tarif_1", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|           { | ||||
|             stat_energy_to: "sensor.energy_production_tarif_2", | ||||
|               stat_compensation: | ||||
|                 "sensor.energy_production_tarif_2_compensation", | ||||
|             stat_compensation: "sensor.energy_production_tarif_2_compensation", | ||||
|             entity_energy_to: "sensor.energy_production_tarif_2", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
| @@ -60,6 +55,7 @@ export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|         type: "gas", | ||||
|         stat_energy_from: "sensor.energy_gas", | ||||
|         stat_cost: "sensor.energy_gas_cost", | ||||
|         entity_energy_from: "sensor.energy_gas", | ||||
|         entity_energy_price: null, | ||||
|         number_energy_price: null, | ||||
|       }, | ||||
| @@ -84,18 +80,11 @@ export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|         stat_consumption: "sensor.energy_boiler", | ||||
|       }, | ||||
|     ], | ||||
|     }) | ||||
|   ); | ||||
|   hass.mockWS( | ||||
|     "energy/info", | ||||
|     (): EnergyInfo => ({ cost_sensors: {}, solar_forecast_domains: [] }) | ||||
|   ); | ||||
|   hass.mockWS( | ||||
|     "energy/fossil_energy_consumption", | ||||
|     ({ period }): FossilEnergyConsumption => ({ | ||||
|   })); | ||||
|   hass.mockWS("energy/info", () => ({ cost_sensors: [] })); | ||||
|   hass.mockWS("energy/fossil_energy_consumption", ({ period }) => ({ | ||||
|     start: period === "month" ? 250 : period === "day" ? 10 : 2, | ||||
|     }) | ||||
|   ); | ||||
|   })); | ||||
|   const todayString = format(startOfToday(), "yyyy-MM-dd"); | ||||
|   const tomorrowString = format(startOfTomorrow(), "yyyy-MM-dd"); | ||||
|   hass.mockWS( | ||||
|   | ||||
| @@ -1,19 +1,47 @@ | ||||
| import { | ||||
|   addDays, | ||||
|   addHours, | ||||
|   addMonths, | ||||
|   differenceInHours, | ||||
|   endOfDay, | ||||
| } from "date-fns/esm"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { HistoryStates } from "../../../src/data/history"; | ||||
| import { StatisticValue } from "../../../src/data/history"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| const generateStateHistory = ( | ||||
|   state: HassEntity, | ||||
|   deltas, | ||||
|   start_date: Date, | ||||
|   end_date: Date | ||||
| ) => { | ||||
| interface HistoryQueryParams { | ||||
|   filter_entity_id: string; | ||||
|   end_time: string; | ||||
| } | ||||
|  | ||||
| const parseQuery = <T>(queryString: string) => { | ||||
|   const query: any = {}; | ||||
|   const items = queryString.split("&"); | ||||
|   for (const item of items) { | ||||
|     const parts = item.split("="); | ||||
|     const key = decodeURIComponent(parts[0]); | ||||
|     const value = parts.length > 1 ? decodeURIComponent(parts[1]) : undefined; | ||||
|     query[key] = value; | ||||
|   } | ||||
|   return query as T; | ||||
| }; | ||||
|  | ||||
| const getTime = (minutesAgo) => { | ||||
|   const ts = new Date(Date.now() - minutesAgo * 60 * 1000); | ||||
|   return ts.toISOString(); | ||||
| }; | ||||
|  | ||||
| const randomTimeAdjustment = (diff) => Math.random() * diff - diff / 2; | ||||
|  | ||||
| const maxTime = 1440; | ||||
|  | ||||
| const generateHistory = (state, deltas) => { | ||||
|   const changes = | ||||
|     typeof deltas[0] === "object" | ||||
|       ? deltas | ||||
|       : deltas.map((st) => ({ state: st })); | ||||
|  | ||||
|   const timeDiff = (end_date.getTime() - start_date.getTime()) / changes.length; | ||||
|   const timeDiff = 900 / changes.length; | ||||
|  | ||||
|   return changes.map((change, index) => { | ||||
|     let attributes; | ||||
| @@ -27,43 +55,358 @@ const generateStateHistory = ( | ||||
|       attributes = { ...state.attributes, ...change.attributes }; | ||||
|     } | ||||
|  | ||||
|     const time = start_date.getTime() + timeDiff * index; | ||||
|     const time = | ||||
|       index === 0 | ||||
|         ? getTime(maxTime) | ||||
|         : getTime(maxTime - index * timeDiff + randomTimeAdjustment(timeDiff)); | ||||
|  | ||||
|     return { | ||||
|       a: attributes, | ||||
|       s: change.state || state.state, | ||||
|       lc: time / 1000, | ||||
|       lu: time / 1000, | ||||
|       attributes, | ||||
|       entity_id: state.entity_id, | ||||
|       state: change.state || state.state, | ||||
|       last_changed: time, | ||||
|       last_updated: time, | ||||
|     }; | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| const incrementalUnits = ["clients", "queries", "ads"]; | ||||
|  | ||||
| export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|   mockHass.mockWS( | ||||
|     "history/stream", | ||||
| const generateMeanStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let lastVal = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const delta = Math.random() * maxDiff; | ||||
|     const mean = lastVal + delta; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean, | ||||
|       min: mean - Math.random() * maxDiff, | ||||
|       max: mean + Math.random() * maxDiff, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: mean, | ||||
|       sum: null, | ||||
|     }); | ||||
|     lastVal = mean; | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateSumStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += add; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: initValue + sum, | ||||
|       sum, | ||||
|     }); | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateCurvedStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   _period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number, | ||||
|   metered: boolean | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const hours = differenceInHours(end, start) - 1; | ||||
|   let i = 0; | ||||
|   let half = false; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += i * add; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: initValue + sum, | ||||
|       sum: metered ? sum : null, | ||||
|     }); | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|     if (!half && i > hours / 2) { | ||||
|       half = true; | ||||
|     } | ||||
|     i += half ? -1 : 1; | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const statisticsFunctions: Record< | ||||
|   string, | ||||
|   ( | ||||
|       { | ||||
|         entity_ids, | ||||
|         start_time, | ||||
|         end_time, | ||||
|       }: { | ||||
|         entity_ids: string[]; | ||||
|         start_time: string; | ||||
|         end_time?: string; | ||||
|       }, | ||||
|       hass, | ||||
|       onChange | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period: "5minute" | "hour" | "day" | "month" | ||||
|   ) => StatisticValue[] | ||||
| > = { | ||||
|   "sensor.energy_consumption_tarif_1": ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|       const states: HistoryStates = {}; | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 10 * 60 * 60 * 1000); | ||||
|     const morningLow = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       morningEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.7 | ||||
|     ); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const morningFinalVal = morningLow.length | ||||
|       ? morningLow[morningLow.length - 1].sum! | ||||
|       : 0; | ||||
|     const empty = generateSumStatistics( | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const eveningLow = generateSumStatistics( | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0.7 | ||||
|     ); | ||||
|     return [...morningLow, ...empty, ...eveningLow]; | ||||
|   }, | ||||
|   "sensor.energy_consumption_tarif_2": ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const highTarif = generateSumStatistics( | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       0, | ||||
|       0.3 | ||||
|     ); | ||||
|     const highTarifFinalVal = highTarif.length | ||||
|       ? highTarif[highTarif.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, morningEnd, period, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       highTarifFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     return [...morning, ...highTarif, ...evening]; | ||||
|   }, | ||||
|   "sensor.energy_production_tarif_1": (id, start, end, period = "hour") => | ||||
|     generateSumStatistics(id, start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_1_compensation": ( | ||||
|     id, | ||||
|     start, | ||||
|     end, | ||||
|     period = "hour" | ||||
|   ) => generateSumStatistics(id, start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_2": (id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 21 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.15, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       productionStart, | ||||
|       period, | ||||
|       0, | ||||
|       0 | ||||
|     ); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics( | ||||
|       id, | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       1 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.solar_production": (id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 7 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 23 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.3, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       productionStart, | ||||
|       period, | ||||
|       0, | ||||
|       0 | ||||
|     ); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics( | ||||
|       id, | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       2 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
| }; | ||||
|  | ||||
|       const start = new Date(start_time); | ||||
|       const end = end_time ? new Date(end_time) : new Date(); | ||||
| export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|   mockHass.mockAPI( | ||||
|     new RegExp("history/period/.+"), | ||||
|     (hass, _method, path, _parameters) => { | ||||
|       const params = parseQuery<HistoryQueryParams>(path.split("?")[1]); | ||||
|       const entities = params.filter_entity_id.split(","); | ||||
|  | ||||
|       for (const entityId of entity_ids) { | ||||
|         states[entityId] = []; | ||||
|       const results: HassEntity[][] = []; | ||||
|  | ||||
|       for (const entityId of entities) { | ||||
|         const state = hass.states[entityId]; | ||||
|  | ||||
|         if (!state) { | ||||
| @@ -71,12 +414,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|         } | ||||
|  | ||||
|         if (!state.attributes.unit_of_measurement) { | ||||
|           states[entityId] = generateStateHistory( | ||||
|             state, | ||||
|             [state.state], | ||||
|             start, | ||||
|             end | ||||
|           ); | ||||
|           results.push(generateHistory(state, [state.state])); | ||||
|           continue; | ||||
|         } | ||||
|  | ||||
| @@ -115,23 +453,56 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|             numberState - diff + Math.floor(Math.random() * 2 * diff); | ||||
|         } | ||||
|  | ||||
|         states[entityId] = generateStateHistory( | ||||
|           state, | ||||
|           Array.from({ length: statesToGenerate }, genFunc), | ||||
|           start, | ||||
|           end | ||||
|         results.push( | ||||
|           generateHistory( | ||||
|             { | ||||
|               entity_id: state.entity_id, | ||||
|               attributes: state.attributes, | ||||
|             }, | ||||
|             Array.from({ length: statesToGenerate }, genFunc) | ||||
|           ) | ||||
|         ); | ||||
|       } | ||||
|       return results; | ||||
|     } | ||||
|   ); | ||||
|   mockHass.mockWS("recorder/get_statistics_metadata", () => []); | ||||
|   mockHass.mockWS("history/list_statistic_ids", () => []); | ||||
|   mockHass.mockWS( | ||||
|     "history/statistics_during_period", | ||||
|     ({ statistic_ids, start_time, end_time, period }, hass) => { | ||||
|       const start = new Date(start_time); | ||||
|       const end = end_time ? new Date(end_time) : new Date(); | ||||
|  | ||||
|       setTimeout(() => { | ||||
|         onChange?.({ | ||||
|           states, | ||||
|           start_time: start, | ||||
|           end_time: end, | ||||
|       const statistics: Record<string, StatisticValue[]> = {}; | ||||
|  | ||||
|       statistic_ids.forEach((id: string) => { | ||||
|         if (id in statisticsFunctions) { | ||||
|           statistics[id] = statisticsFunctions[id](id, start, end, period); | ||||
|         } else { | ||||
|           const entityState = hass.states[id]; | ||||
|           const state = entityState ? Number(entityState.state) : 1; | ||||
|           statistics[id] = | ||||
|             entityState && "last_reset" in entityState.attributes | ||||
|               ? generateSumStatistics( | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.01 : 0.05) | ||||
|                 ) | ||||
|               : generateMeanStatistics( | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.05 : 0.1) | ||||
|                 ); | ||||
|         } | ||||
|       }); | ||||
|       }, 1); | ||||
|  | ||||
|       return () => {}; | ||||
|       return statistics; | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -1,345 +0,0 @@ | ||||
| import { | ||||
|   addDays, | ||||
|   addHours, | ||||
|   addMonths, | ||||
|   differenceInHours, | ||||
|   endOfDay, | ||||
| } from "date-fns"; | ||||
| import { | ||||
|   Statistics, | ||||
|   StatisticsMetaData, | ||||
|   StatisticValue, | ||||
| } from "../../../src/data/recorder"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| const generateMeanStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ): StatisticValue[] => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let lastVal = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const delta = Math.random() * maxDiff; | ||||
|     const mean = lastVal + delta; | ||||
|     statistics.push({ | ||||
|       start: currentDate.getTime(), | ||||
|       end: currentDate.getTime(), | ||||
|       mean, | ||||
|       min: mean - Math.random() * maxDiff, | ||||
|       max: mean + Math.random() * maxDiff, | ||||
|       last_reset: 0, | ||||
|       state: mean, | ||||
|       sum: null, | ||||
|     }); | ||||
|     lastVal = mean; | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateSumStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ): StatisticValue[] => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += add; | ||||
|     statistics.push({ | ||||
|       start: currentDate.getTime(), | ||||
|       end: currentDate.getTime(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: 0, | ||||
|       state: initValue + sum, | ||||
|       sum, | ||||
|     }); | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateCurvedStatistics = ( | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   // eslint-disable-next-line @typescript-eslint/default-param-last | ||||
|   _period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number, | ||||
|   metered: boolean | ||||
| ): StatisticValue[] => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const hours = differenceInHours(end, start) - 1; | ||||
|   let i = 0; | ||||
|   let half = false; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += i * add; | ||||
|     statistics.push({ | ||||
|       start: currentDate.getTime(), | ||||
|       end: currentDate.getTime(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: 0, | ||||
|       state: initValue + sum, | ||||
|       sum: metered ? sum : null, | ||||
|     }); | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|     if (!half && i > hours / 2) { | ||||
|       half = true; | ||||
|     } | ||||
|     i += half ? -1 : 1; | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const statisticsFunctions: Record< | ||||
|   string, | ||||
|   ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period: "5minute" | "hour" | "day" | "month" | ||||
|   ) => StatisticValue[] | ||||
| > = { | ||||
|   "sensor.energy_consumption_tarif_1": ( | ||||
|     _id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 10 * 60 * 60 * 1000); | ||||
|     const morningLow = generateSumStatistics(start, morningEnd, period, 0, 0.7); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const morningFinalVal = morningLow.length | ||||
|       ? morningLow[morningLow.length - 1].sum! | ||||
|       : 0; | ||||
|     const empty = generateSumStatistics( | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const eveningLow = generateSumStatistics( | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0.7 | ||||
|     ); | ||||
|     return [...morningLow, ...empty, ...eveningLow]; | ||||
|   }, | ||||
|   "sensor.energy_consumption_tarif_2": ( | ||||
|     _id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const highTarif = generateSumStatistics( | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       0, | ||||
|       0.3 | ||||
|     ); | ||||
|     const highTarifFinalVal = highTarif.length | ||||
|       ? highTarif[highTarif.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(start, morningEnd, period, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       highTarifFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     return [...morning, ...highTarif, ...evening]; | ||||
|   }, | ||||
|   "sensor.energy_production_tarif_1": (_id, start, end, period = "hour") => | ||||
|     generateSumStatistics(start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_1_compensation": ( | ||||
|     _id, | ||||
|     start, | ||||
|     end, | ||||
|     period = "hour" | ||||
|   ) => generateSumStatistics(start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_2": (_id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 21 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.15, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(start, productionStart, period, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics( | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       1 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.solar_production": (_id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 7 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 23 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.3, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(start, productionStart, period, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics( | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       2 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
| }; | ||||
| export const mockRecorder = (mockHass: MockHomeAssistant) => { | ||||
|   mockHass.mockWS( | ||||
|     "recorder/get_statistics_metadata", | ||||
|     (): StatisticsMetaData[] => [] | ||||
|   ); | ||||
|   mockHass.mockWS( | ||||
|     "recorder/list_statistic_ids", | ||||
|     (): StatisticsMetaData[] => [] | ||||
|   ); | ||||
|   mockHass.mockWS( | ||||
|     "recorder/statistics_during_period", | ||||
|     ({ statistic_ids, start_time, end_time, period }, hass): Statistics => { | ||||
|       const start = new Date(start_time); | ||||
|       const end = end_time ? new Date(end_time) : new Date(); | ||||
|  | ||||
|       const statistics: Record<string, StatisticValue[]> = {}; | ||||
|  | ||||
|       statistic_ids.forEach((id: string) => { | ||||
|         if (id in statisticsFunctions) { | ||||
|           statistics[id] = statisticsFunctions[id](id, start, end, period); | ||||
|         } else { | ||||
|           const entityState = hass.states[id]; | ||||
|           const state = entityState ? Number(entityState.state) : 1; | ||||
|           statistics[id] = | ||||
|             entityState && "last_reset" in entityState.attributes | ||||
|               ? generateSumStatistics( | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.01 : 0.05) | ||||
|                 ) | ||||
|               : generateMeanStatistics( | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.05 : 0.1) | ||||
|                 ); | ||||
|         } | ||||
|       }); | ||||
|       return statistics; | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
| @@ -1,11 +1,12 @@ | ||||
| import webpack from "../build-scripts/webpack.cjs"; | ||||
| import env from "../build-scripts/env.cjs"; | ||||
| const { createDemoConfig } = require("../build-scripts/webpack.js"); | ||||
| const { isProdBuild, isStatsBuild } = require("../build-scripts/env.js"); | ||||
|  | ||||
| // File just used for stats builds | ||||
|  | ||||
| const latestBuild = true; | ||||
|  | ||||
| export default webpack.createDemoConfig({ | ||||
|   isProdBuild: env.isProdBuild(), | ||||
|   isStatsBuild: env.isStatsBuild(), | ||||
| module.exports = createDemoConfig({ | ||||
|   isProdBuild: isProdBuild(), | ||||
|   isStatsBuild: isStatsBuild(), | ||||
|   latestBuild, | ||||
| }); | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 25 KiB | 
| @@ -1,5 +1,5 @@ | ||||
| import rollup from "../build-scripts/rollup.cjs"; | ||||
| import env from "../build-scripts/env.cjs"; | ||||
| const rollup = require("../build-scripts/rollup.js"); | ||||
| const env = require("../build-scripts/env.js"); | ||||
|  | ||||
| const config = rollup.createGalleryConfig({ | ||||
|   isProdBuild: env.isProdBuild(), | ||||
| @@ -7,4 +7,4 @@ const config = rollup.createGalleryConfig({ | ||||
|   isStatsBuild: env.isStatsBuild(), | ||||
| }); | ||||
|  | ||||
| export default { ...config.inputOptions, output: config.outputOptions }; | ||||
| module.exports = { ...config.inputOptions, output: config.outputOptions }; | ||||
|   | ||||
							
								
								
									
										35
									
								
								gallery/script/netlify_build_gallery
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										35
									
								
								gallery/script/netlify_build_gallery
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| TARGET_LABEL="needs design preview" | ||||
|  | ||||
| if [[ "$NETLIFY" != "true" ]]; then | ||||
|   echo "This script can only be run on Netlify" | ||||
|   exit 1 | ||||
| fi | ||||
|  | ||||
| function createStatus() { | ||||
|   state="$1" | ||||
|   description="$2" | ||||
|   target_url="$3" | ||||
|   curl -X POST -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/statuses/$COMMIT_REF" \ | ||||
|     -d '{"state": "'"${state}"'", "context": "Netlify/Design Preview Build", "description": "'"$description"'", "target_url":  "'"$target_url"'"}' | ||||
| } | ||||
|  | ||||
|  | ||||
| if [[ "${PULL_REQUEST}" == "true" ]]; then | ||||
|   if [[ "$(curl -sSLf -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/pulls/${REVIEW_ID}" | jq '.labels[].name' -r)" =~ "$TARGET_LABEL" ]]; then | ||||
|     createStatus "pending" "Building design preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     gulp build-gallery | ||||
|     if [ $? -eq 0 ]; then | ||||
|       createStatus "success" "Build complete" "$DEPLOY_PRIME_URL" | ||||
|     else | ||||
|       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     fi | ||||
|   else | ||||
|     createStatus "success" "Build was not requested by PR label" | ||||
|   fi | ||||
| elif [[ "$INCOMING_HOOK_BODY" == "NIGHTLY" ]]; then | ||||
|   gulp build-gallery | ||||
| fi | ||||
| @@ -1,4 +1,4 @@ | ||||
| export default [ | ||||
| module.exports = [ | ||||
|   { | ||||
|     // This section has no header and so all page links are shown directly in the sidebar | ||||
|     category: "concepts", | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { css, html, nothing } from "lit"; | ||||
| import { html, css } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { until } from "lit/directives/until"; | ||||
| import { HaMarkdown } from "../../../src/components/ha-markdown"; | ||||
| @@ -10,7 +10,7 @@ class PageDescription extends HaMarkdown { | ||||
|  | ||||
|   render() { | ||||
|     if (!PAGES[this.page].description) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|   | ||||
| @@ -1,56 +0,0 @@ | ||||
| --- | ||||
| title: When to use remove, delete, add and create | ||||
| subtitle: The difference between remove/delete and add/create. | ||||
| --- | ||||
|  | ||||
| # Remove vs Delete | ||||
| Remove and Delete are quite similar, but can be frustrating if used inconsistently. | ||||
|  | ||||
| ## Remove | ||||
| Take away and set aside, but kept in existence. | ||||
|  | ||||
| For example: | ||||
| * Removing a user's permission | ||||
| * Removing a user from a group | ||||
| * Removing links between items | ||||
| * Removing a widget | ||||
| * Removing a link | ||||
| * Removing an item from a cart | ||||
|  | ||||
| ## Delete | ||||
| Erase, rendered nonexistent or nonrecoverable. | ||||
|  | ||||
| For example: | ||||
| * Deleting a field | ||||
| * Deleting a value in a field | ||||
| * Deleting a task | ||||
| * Deleting a group | ||||
| * Deleting a permission | ||||
| * Deleting a calendar event | ||||
|  | ||||
| # Add vs Create | ||||
| In most cases, Create can be paired with Delete, and Add can be paired with Remove. | ||||
|  | ||||
| ## Add | ||||
| An already-exisiting item. | ||||
|  | ||||
| For example: | ||||
| * Adding a permission to a user | ||||
| * Adding a user to a group | ||||
| * Adding links between items | ||||
| * Adding a widget | ||||
| * Adding a link | ||||
| * Adding an item to a cart | ||||
|  | ||||
| ## Create | ||||
| Something made from scratch. | ||||
|  | ||||
| For example: | ||||
| * Creating a new field | ||||
| * Creating a new value in a field | ||||
| * Creating a new task | ||||
| * Creating a new group | ||||
| * Creating a new permission | ||||
| * Creating a new calendar event | ||||
|  | ||||
| Based on this is [UX magazine article](https://uxmag.com/articles/ui-copy-remove-vs-delete2-banner). | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-yaml-editor"; | ||||
| @@ -127,16 +127,16 @@ export class DemoAutomationDescribeAction extends LitElement { | ||||
|  | ||||
|   @state() _action = initialAction; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <ha-card header="Actions"> | ||||
|         <div class="action"> | ||||
|           <span> | ||||
|             ${this._action | ||||
|               ? describeAction(this.hass, [], this._action) | ||||
|               ? describeAction(this.hass, this._action) | ||||
|               : "<invalid YAML>"} | ||||
|           </span> | ||||
|           <ha-yaml-editor | ||||
| @@ -149,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> | ||||
|           ` | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-yaml-editor"; | ||||
| @@ -36,7 +36,6 @@ const conditions = [ | ||||
|   { condition: "sun", after: "sunset" }, | ||||
|   { 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" }, | ||||
| ]; | ||||
| @@ -53,9 +52,9 @@ export class DemoAutomationDescribeCondition extends LitElement { | ||||
|  | ||||
|   @state() _condition = initialCondition; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-yaml-editor"; | ||||
| @@ -64,9 +64,9 @@ export class DemoAutomationDescribeTrigger extends LitElement { | ||||
|  | ||||
|   @state() _trigger = initialTrigger; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html, css } from "lit"; | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -47,8 +47,6 @@ const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
| class DemoHaAutomationEditorAction extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   @state() private _disabled = false; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.actions); | ||||
|  | ||||
|   constructor() { | ||||
| @@ -69,15 +67,6 @@ class DemoHaAutomationEditorAction extends LitElement { | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       <div class="options"> | ||||
|         <ha-formfield label="Disabled"> | ||||
|           <ha-switch | ||||
|             .name=${"disabled"} | ||||
|             .checked=${this._disabled} | ||||
|             @change=${this._handleOptionChange} | ||||
|           ></ha-switch> | ||||
|         </ha-formfield> | ||||
|       </div> | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
| @@ -92,7 +81,6 @@ class DemoHaAutomationEditorAction extends LitElement { | ||||
|                     .hass=${this.hass} | ||||
|                     .actions=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     .disabled=${this._disabled} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-action> | ||||
|                 ` | ||||
| @@ -102,20 +90,6 @@ class DemoHaAutomationEditorAction extends LitElement { | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleOptionChange(ev) { | ||||
|     this[`_${ev.target.name}`] = ev.target.checked; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .options { | ||||
|       max-width: 800px; | ||||
|       margin: 16px auto; | ||||
|     } | ||||
|     .options ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html, css } from "lit"; | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -83,8 +83,6 @@ const SCHEMAS: { name: string; conditions: ConditionWithShorthand[] }[] = [ | ||||
| class DemoHaAutomationEditorCondition extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   @state() private _disabled = false; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.conditions); | ||||
|  | ||||
|   constructor() { | ||||
| @@ -105,15 +103,6 @@ class DemoHaAutomationEditorCondition extends LitElement { | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       <div class="options"> | ||||
|         <ha-formfield label="Disabled"> | ||||
|           <ha-switch | ||||
|             .name=${"disabled"} | ||||
|             .checked=${this._disabled} | ||||
|             @change=${this._handleOptionChange} | ||||
|           ></ha-switch> | ||||
|         </ha-formfield> | ||||
|       </div> | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
| @@ -128,7 +117,6 @@ class DemoHaAutomationEditorCondition extends LitElement { | ||||
|                     .hass=${this.hass} | ||||
|                     .conditions=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     .disabled=${this._disabled} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-condition> | ||||
|                 ` | ||||
| @@ -138,20 +126,6 @@ class DemoHaAutomationEditorCondition extends LitElement { | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleOptionChange(ev) { | ||||
|     this[`_${ev.target.name}`] = ev.target.checked; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .options { | ||||
|       max-width: 800px; | ||||
|       margin: 16px auto; | ||||
|     } | ||||
|     .options ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html, css } from "lit"; | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -107,8 +107,6 @@ const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
| class DemoHaAutomationEditorTrigger extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   @state() private _disabled = false; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.triggers); | ||||
|  | ||||
|   constructor() { | ||||
| @@ -129,15 +127,6 @@ class DemoHaAutomationEditorTrigger extends LitElement { | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       <div class="options"> | ||||
|         <ha-formfield label="Disabled"> | ||||
|           <ha-switch | ||||
|             .name=${"disabled"} | ||||
|             .checked=${this._disabled} | ||||
|             @change=${this._handleOptionChange} | ||||
|           ></ha-switch> | ||||
|         </ha-formfield> | ||||
|       </div> | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
| @@ -152,7 +141,6 @@ class DemoHaAutomationEditorTrigger extends LitElement { | ||||
|                     .hass=${this.hass} | ||||
|                     .triggers=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     .disabled=${this._disabled} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-trigger> | ||||
|                 ` | ||||
| @@ -162,20 +150,6 @@ class DemoHaAutomationEditorTrigger extends LitElement { | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleOptionChange(ev) { | ||||
|     this[`_${ev.target.name}`] = ev.target.checked; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .options { | ||||
|       max-width: 800px; | ||||
|       margin: 16px auto; | ||||
|     } | ||||
|     .options ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
|  | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/trace/hat-script-graph"; | ||||
| @@ -30,9 +29,9 @@ const traces: DemoTrace[] = [ | ||||
| export class DemoAutomationTraceTimeline extends LitElement { | ||||
|   @property({ attribute: false }) hass?: HomeAssistant; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       ${traces.map( | ||||
|   | ||||
| @@ -1,15 +1,14 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
|  | ||||
| import { css, html, LitElement, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../../src/components/trace/hat-trace-timeline"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { DemoTrace } from "../../data/traces/types"; | ||||
| import { basicTrace } from "../../data/traces/basic_trace"; | ||||
| import { motionLightTrace } from "../../data/traces/motion-light-trace"; | ||||
| import { DemoTrace } from "../../data/traces/types"; | ||||
|  | ||||
| const traces: DemoTrace[] = [basicTrace, motionLightTrace]; | ||||
|  | ||||
| @@ -19,9 +18,9 @@ export class DemoAutomationTrace extends LitElement { | ||||
|  | ||||
|   @state() private _selected = {}; | ||||
|  | ||||
|   protected render() { | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return nothing; | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       ${traces.map( | ||||
|   | ||||
| @@ -2,6 +2,8 @@ | ||||
| title: "Logo" | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| # Using our logo | ||||
|  | ||||
| As a community, we are proud of our logo. Follow these guidelines to ensure it always looks its best. Our logo follows Google's material design spec and uses the blue interface color. | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| --- | ||||
| title: Dialog | ||||
| title: Dialgos | ||||
| subtitle: Dialogs provide important prompts in a user flow. | ||||
| --- | ||||
| 
 | ||||
| # Material Design 3 | ||||
| # Material Desing 3 | ||||
| 
 | ||||
| Our dialogs are based on the latest version of Material Design. Specs and guidelines can be found on its [website](https://m3.material.io/components/dialogs/overview).  | ||||
| Our dialogs are based on the latest version of Material Design. Specs and guidelines can be found on it's [website](https://m3.material.io/components/dialogs/overview).  | ||||
| 
 | ||||
| # Highlighted guidelines | ||||
| 
 | ||||
| @@ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: Alert | ||||
| title: Alerts | ||||
| subtitle: An alert displays a short, important message in a way that attracts the user's attention without interrupting the user's task. | ||||
| --- | ||||
|  | ||||
| @@ -156,6 +156,18 @@ The `title ` option should not be used without a description. | ||||
|  | ||||
| *Documentation coming soon* | ||||
|  | ||||
| **Right to left** | ||||
|  | ||||
| <ha-alert alert-type="success" rtl> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" rtl> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| ### API | ||||
| **Properties/Attributes** | ||||
|  | ||||
|   | ||||
| @@ -98,9 +98,7 @@ const alerts: { | ||||
|     description: "Alert with slotted image", | ||||
|     type: "warning", | ||||
|     iconSlot: html`<span slot="icon" class="image" | ||||
|       ><img | ||||
|         alt="Home Assistant logo" | ||||
|         src="https://www.home-assistant.io/images/home-assistant-logo.svg" | ||||
|       ><img src="https://www.home-assistant.io/images/home-assistant-logo.svg" | ||||
|     /></span>`, | ||||
|   }, | ||||
|   { | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| --- | ||||
| title: Bar | ||||
| subtitle: Can be used to communicate progress of a task. | ||||
| title: Progress Bars | ||||
| --- | ||||
|   | ||||
| @@ -1,3 +1,3 @@ | ||||
| --- | ||||
| title: Chip | ||||
| title: Chips | ||||
| --- | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user