mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-20 17:20:07 +00:00 
			
		
		
		
	Compare commits
	
		
			3 Commits
		
	
	
		
			partial_te
			...
			fix-number
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | a63e788ced | ||
|   | c377c01c65 | ||
|   | aaf65e0599 | 
| @@ -1,42 +0,0 @@ | |||||||
| [modern] |  | ||||||
| # Modern builds target recent browsers supporting the latest features to minimize transpilation, polyfills, etc. |  | ||||||
| # It is served to browsers meeting the following requirements: |  | ||||||
| # - released in the last year + current alpha/beta versions |  | ||||||
| # - Firefox extended support release (ESR) |  | ||||||
| # - with global utilization at or above 0.5% |  | ||||||
| # - must support dynamic import of ES modules |  | ||||||
| # - exclude browsers no longer being maintained |  | ||||||
| # - exclude KaiOS, QQ, and UC browsers due to lack of sufficient feature support data |  | ||||||
| unreleased versions |  | ||||||
| last 1 year |  | ||||||
| Firefox ESR |  | ||||||
| >= 0.5% and supports es6-module-dynamic-import |  | ||||||
| not dead |  | ||||||
| not KaiOS > 0 |  | ||||||
| not QQAndroid > 0 |  | ||||||
| not UCAndroid > 0 |  | ||||||
|  |  | ||||||
| [legacy] |  | ||||||
| # Legacy builds are served when modern requirements are not met and support browsers: |  | ||||||
| # - released in the last 7 years + current alpha/beta versionss |  | ||||||
| # - with global utilization at or above 0.05% |  | ||||||
| # The lattermost query ensures that support for popular old browsers is not dropped too early |  | ||||||
| # (e.g. IE 11, Android 4.4, or Samsung 4). |  | ||||||
| # |  | ||||||
| # In addition, legacy browsers must support some minimum features that cannot be polyfilled: |  | ||||||
| # - ES5 (strict mode) |  | ||||||
| # - web sockets to communicate with backend |  | ||||||
| # - inline SVG used widely in buttons, widgets, etc. |  | ||||||
| # - custom events used for most user interactions |  | ||||||
| # - CSS flexbox used in the majority of the layout |  | ||||||
| # Nearly all of these are redundant with the above rules. |  | ||||||
| # As of May 2023, only web sockets must be added to the query. |  | ||||||
| unreleased versions |  | ||||||
| last 7 years |  | ||||||
| >= 0.05% and supports websockets |  | ||||||
|  |  | ||||||
| [legacy-sw] |  | ||||||
| # Same as legacy plus supports service workers |  | ||||||
| unreleased versions |  | ||||||
| last 7 years |  | ||||||
| >= 0.05% and supports websockets and supports serviceworkers |  | ||||||
| @@ -1,7 +1,13 @@ | |||||||
| # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | ||||||
| FROM mcr.microsoft.com/devcontainers/python:3.12 | FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.9 | ||||||
|  |  | ||||||
| ENV \ | ENV \ | ||||||
|   DEBIAN_FRONTEND=noninteractive \ |   DEBIAN_FRONTEND=noninteractive \ | ||||||
|   DEVCONTAINER=true \ |   DEVCONTAINER=true \ | ||||||
|   PATH=$PATH:./node_modules/.bin |   PATH=$PATH:./node_modules/.bin | ||||||
|  |  | ||||||
|  | # Install nvm | ||||||
|  | COPY .nvmrc /tmp/.nvmrc | ||||||
|  | RUN \ | ||||||
|  |   su vscode -c \ | ||||||
|  |     "source /usr/local/share/nvm/nvm.sh && nvm install $(cat /tmp/.nvmrc) 2>&1" | ||||||
| @@ -4,42 +4,31 @@ | |||||||
|     "dockerfile": "Dockerfile", |     "dockerfile": "Dockerfile", | ||||||
|     "context": ".." |     "context": ".." | ||||||
|   }, |   }, | ||||||
|   "appPort": "8124:8123", |   "appPort": 8123, | ||||||
|   "postCreateCommand": "sudo apt update && sudo apt upgrade -y && sudo apt install -y libpcap-dev", |   "context": "..", | ||||||
|   "postStartCommand": "script/bootstrap", |   "postCreateCommand": "script/bootstrap", | ||||||
|   "containerEnv": { |   "extensions": [ | ||||||
|     "DEV_CONTAINER": "1", |     "github.vscode-pull-request-github", | ||||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" |     "dbaeumer.vscode-eslint", | ||||||
|   }, |     "ms-vscode.vscode-typescript-tslint-plugin", | ||||||
|   "customizations": { |     "esbenp.prettier-vscode", | ||||||
|     "vscode": { |     "bierner.lit-html", | ||||||
|       "extensions": [ |     "runem.lit-plugin", | ||||||
|         "dbaeumer.vscode-eslint", |     "ms-python.vscode-pylance" | ||||||
|         "esbenp.prettier-vscode", |   ], | ||||||
|         "runem.lit-plugin", |   "settings": { | ||||||
|         "github.vscode-pull-request-github", |     "terminal.integrated.shell.linux": "/bin/bash", | ||||||
|         "eamodio.gitlens" |     "files.eol": "\n", | ||||||
|       ], |     "editor.tabSize": 2, | ||||||
|       "settings": { |     "editor.formatOnPaste": false, | ||||||
|         "files.eol": "\n", |     "editor.formatOnSave": true, | ||||||
|         "editor.tabSize": 2, |     "editor.formatOnType": true, | ||||||
|         "editor.formatOnPaste": false, |     "[typescript]": { | ||||||
|         "editor.formatOnSave": true, |       "editor.defaultFormatter": "esbenp.prettier-vscode" | ||||||
|         "editor.formatOnType": true, |     }, | ||||||
|         "editor.renderWhitespace": "boundary", |     "[javascript]": { | ||||||
|         "editor.rulers": [80], |       "editor.defaultFormatter": "esbenp.prettier-vscode" | ||||||
|         "[typescript]": { |     }, | ||||||
|           "editor.defaultFormatter": "esbenp.prettier-vscode" |     "files.trimTrailingWhitespace": true | ||||||
|         }, |  | ||||||
|         "[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" |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										141
									
								
								.eslintrc.json
									
									
									
									
									
								
							
							
						
						
									
										141
									
								
								.eslintrc.json
									
									
									
									
									
								
							| @@ -1,12 +1,11 @@ | |||||||
| { | { | ||||||
|   "extends": [ |   "extends": [ | ||||||
|     "airbnb-base", |  | ||||||
|     "airbnb-typescript/base", |     "airbnb-typescript/base", | ||||||
|     "plugin:@typescript-eslint/recommended", |     "plugin:@typescript-eslint/recommended", | ||||||
|     "plugin:wc/recommended", |     "plugin:wc/recommended", | ||||||
|     "plugin:lit/all", |     "plugin:lit/recommended", | ||||||
|     "plugin:lit-a11y/recommended", |     "prettier", | ||||||
|     "prettier" |     "prettier/@typescript-eslint" | ||||||
|   ], |   ], | ||||||
|   "parser": "@typescript-eslint/parser", |   "parser": "@typescript-eslint/parser", | ||||||
|   "parserOptions": { |   "parserOptions": { | ||||||
| @@ -20,7 +19,7 @@ | |||||||
|   "settings": { |   "settings": { | ||||||
|     "import/resolver": { |     "import/resolver": { | ||||||
|       "webpack": { |       "webpack": { | ||||||
|         "config": "./webpack.config.cjs" |         "config": "./webpack.config.js" | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| @@ -30,101 +29,63 @@ | |||||||
|     "__BUILD__": false, |     "__BUILD__": false, | ||||||
|     "__VERSION__": false, |     "__VERSION__": false, | ||||||
|     "__STATIC_PATH__": false, |     "__STATIC_PATH__": false, | ||||||
|     "__SUPERVISOR__": false, |     "Polymer": true, | ||||||
|     "Polymer": true |     "webkitSpeechRecognition": false, | ||||||
|  |     "ResizeObserver": false | ||||||
|   }, |   }, | ||||||
|   "env": { |   "env": { | ||||||
|     "browser": true, |     "browser": true, | ||||||
|     "es6": true |     "es6": true | ||||||
|   }, |   }, | ||||||
|   "rules": { |   "rules": { | ||||||
|     "class-methods-use-this": "off", |     "class-methods-use-this": 0, | ||||||
|     "new-cap": "off", |     "new-cap": 0, | ||||||
|     "prefer-template": "off", |     "prefer-template": 0, | ||||||
|     "object-shorthand": "off", |     "object-shorthand": 0, | ||||||
|     "func-names": "off", |     "func-names": 0, | ||||||
|     "no-underscore-dangle": "off", |     "prefer-arrow-callback": 0, | ||||||
|     "strict": "off", |     "no-underscore-dangle": 0, | ||||||
|     "no-plusplus": "off", |     "strict": 0, | ||||||
|     "no-bitwise": "error", |     "prefer-spread": 0, | ||||||
|     "comma-dangle": "off", |     "no-plusplus": 0, | ||||||
|     "vars-on-top": "off", |     "no-bitwise": 2, | ||||||
|     "no-continue": "off", |     "comma-dangle": 0, | ||||||
|     "no-param-reassign": "off", |     "vars-on-top": 0, | ||||||
|     "no-multi-assign": "off", |     "no-continue": 0, | ||||||
|     "no-console": "error", |     "no-param-reassign": 0, | ||||||
|     "radix": "off", |     "no-multi-assign": 0, | ||||||
|     "no-alert": "off", |     "no-console": 2, | ||||||
|     "no-nested-ternary": "off", |     "radix": 0, | ||||||
|     "prefer-destructuring": "off", |     "no-alert": 0, | ||||||
|  |     "no-return-await": 0, | ||||||
|  |     "no-nested-ternary": 0, | ||||||
|  |     "prefer-destructuring": 0, | ||||||
|     "no-restricted-globals": [2, "event"], |     "no-restricted-globals": [2, "event"], | ||||||
|     "prefer-promise-reject-errors": "off", |     "prefer-promise-reject-errors": 0, | ||||||
|     "import/prefer-default-export": "off", |     "import/order": 0, | ||||||
|     "import/no-default-export": "off", |     "import/prefer-default-export": 0, | ||||||
|     "import/no-unresolved": "off", |     "import/no-unresolved": 0, | ||||||
|     "import/no-cycle": "off", |     "import/no-cycle": 0, | ||||||
|     "import/extensions": [ |     "import/extensions": [ | ||||||
|       "error", |       2, | ||||||
|       "ignorePackages", |       "ignorePackages", | ||||||
|       { |       { "ts": "never", "js": "never" } | ||||||
|         "ts": "never", |  | ||||||
|         "js": "never" |  | ||||||
|       } |  | ||||||
|     ], |     ], | ||||||
|     "no-restricted-syntax": ["error", "LabeledStatement", "WithStatement"], |     "no-restricted-syntax": ["error", "LabeledStatement", "WithStatement"], | ||||||
|     "object-curly-newline": "off", |     "object-curly-newline": 0, | ||||||
|     "default-case": "off", |     "default-case": 0, | ||||||
|     "wc/no-self-class": "off", |     "wc/no-self-class": 0, | ||||||
|     "no-shadow": "off", |     "no-shadow": 0, | ||||||
|     "@typescript-eslint/camelcase": "off", |     "@typescript-eslint/camelcase": 0, | ||||||
|     "@typescript-eslint/ban-ts-comment": "off", |     "@typescript-eslint/ban-ts-comment": 0, | ||||||
|     "@typescript-eslint/no-use-before-define": "off", |     "@typescript-eslint/no-use-before-define": 0, | ||||||
|     "@typescript-eslint/no-non-null-assertion": "off", |     "@typescript-eslint/no-non-null-assertion": 0, | ||||||
|     "@typescript-eslint/no-explicit-any": "off", |     "@typescript-eslint/no-explicit-any": 0, | ||||||
|     "@typescript-eslint/explicit-function-return-type": "off", |     "@typescript-eslint/no-unused-vars": 0, | ||||||
|     "@typescript-eslint/explicit-module-boundary-types": "off", |     "@typescript-eslint/explicit-function-return-type": 0, | ||||||
|     "@typescript-eslint/no-shadow": ["error"], |     "@typescript-eslint/explicit-module-boundary-types": 0, | ||||||
|     "@typescript-eslint/naming-convention": [ |     "@typescript-eslint/no-shadow": ["error"] | ||||||
|       "off", |  | ||||||
|       { |  | ||||||
|         "selector": "default", |  | ||||||
|         "format": ["camelCase", "snake_case"], |  | ||||||
|         "leadingUnderscore": "allow", |  | ||||||
|         "trailingUnderscore": "allow" |  | ||||||
|       }, |  | ||||||
|       { |  | ||||||
|         "selector": ["variable"], |  | ||||||
|         "format": ["camelCase", "snake_case", "UPPER_CASE"], |  | ||||||
|         "leadingUnderscore": "allow", |  | ||||||
|         "trailingUnderscore": "allow" |  | ||||||
|       }, |  | ||||||
|       { |  | ||||||
|         "selector": "typeLike", |  | ||||||
|         "format": ["PascalCase"] |  | ||||||
|       } |  | ||||||
|     ], |  | ||||||
|     "@typescript-eslint/no-unused-vars": "off", |  | ||||||
|     "unused-imports/no-unused-vars": [ |  | ||||||
|       "error", |  | ||||||
|       { |  | ||||||
|         "vars": "all", |  | ||||||
|         "varsIgnorePattern": "^_", |  | ||||||
|         "args": "after-used", |  | ||||||
|         "argsIgnorePattern": "^_", |  | ||||||
|         "ignoreRestSiblings": true |  | ||||||
|       } |  | ||||||
|     ], |  | ||||||
|     "unused-imports/no-unused-imports": "error", |  | ||||||
|     "lit/attribute-names": "warn", |  | ||||||
|     "lit/attribute-value-entities": "off", |  | ||||||
|     "lit/no-template-map": "off", |  | ||||||
|     "lit/no-native-attributes": "warn", |  | ||||||
|     "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" |  | ||||||
|   }, |   }, | ||||||
|   "plugins": ["unused-imports"] |   "plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"], | ||||||
|  |   "processor": "disable/disable" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ DO NOT DELETE ANY TEXT from this template! Otherwise, your issue may be closed w | |||||||
| <!-- | <!-- | ||||||
|   Provide details about the versions you are using, which helps us reproducing |   Provide details about the versions you are using, which helps us reproducing | ||||||
|   and finding the issue quicker. Version information is found in the |   and finding the issue quicker. Version information is found in the | ||||||
|   Home Assistant frontend: Settings -> About. |   Home Assistant frontend: Configuration -> Info. | ||||||
| 
 | 
 | ||||||
|   Browser version and operating system is important! Please try to replicate |   Browser version and operating system is important! Please try to replicate | ||||||
|   your issue in a different browser and be sure to include your findings. |   your issue in a different browser and be sure to include your findings. | ||||||
							
								
								
									
										122
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										122
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,122 +0,0 @@ | |||||||
| name: Report a bug with the UI / Dashboards |  | ||||||
| description: Report an issue related to the Home Assistant frontend. |  | ||||||
| labels: bug |  | ||||||
| body: |  | ||||||
|   - type: markdown |  | ||||||
|     attributes: |  | ||||||
|       value: | |  | ||||||
|         Make sure you are running the [latest version of Home Assistant][releases] before reporting an issue. |  | ||||||
|  |  | ||||||
|         If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue. |  | ||||||
|  |  | ||||||
|         **Please do not report issues for custom cards.** |  | ||||||
|  |  | ||||||
|         [fr]: https://github.com/home-assistant/frontend/discussions |  | ||||||
|         [releases]: https://github.com/home-assistant/home-assistant/releases |  | ||||||
|   - type: checkboxes |  | ||||||
|     attributes: |  | ||||||
|       label: Checklist |  | ||||||
|       description: Please verify that you've followed these steps |  | ||||||
|       options: |  | ||||||
|         - label: I have updated to the latest available Home Assistant version. |  | ||||||
|           required: true |  | ||||||
|         - label: I have cleared the cache of my browser. |  | ||||||
|           required: true |  | ||||||
|         - label: I have tried a different browser to see if it is related to my browser. |  | ||||||
|           required: true |  | ||||||
|         - label: I have tried reproducing the issue in [safe mode](https://www.home-assistant.io/blog/2023/11/01/release-202311/#restarting-into-safe-mode) to rule out problems with unsupported custom resources. |  | ||||||
|   - type: markdown |  | ||||||
|     attributes: |  | ||||||
|       value: | |  | ||||||
|         ## The problem |  | ||||||
|   - type: textarea |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|     attributes: |  | ||||||
|       label: Describe the issue you are experiencing |  | ||||||
|       description: Provide a clear and concise description of what the bug is. |  | ||||||
|   - type: textarea |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|     attributes: |  | ||||||
|       label: Describe the behavior you expected |  | ||||||
|       description: Describe what you expected to happen or it should look/behave. |  | ||||||
|   - type: textarea |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|     attributes: |  | ||||||
|       label: Steps to reproduce the issue |  | ||||||
|       description: | |  | ||||||
|         Please tell us exactly how to reproduce your issue. |  | ||||||
|         Provide clear and concise step by step instructions and add code snippets if needed. |  | ||||||
|       value: | |  | ||||||
|         1. |  | ||||||
|         2. |  | ||||||
|         3. |  | ||||||
|         ... |  | ||||||
|   - type: markdown |  | ||||||
|     attributes: |  | ||||||
|       value: | |  | ||||||
|         ## Environment |  | ||||||
|   - type: input |  | ||||||
|     validations: |  | ||||||
|       required: true |  | ||||||
|     attributes: |  | ||||||
|       label: What version of Home Assistant Core has the issue? |  | ||||||
|       placeholder: core- |  | ||||||
|       description: > |  | ||||||
|         Can be found in: [Settings -> About](https://my.home-assistant.io/redirect/info/). |  | ||||||
|   - type: input |  | ||||||
|     attributes: |  | ||||||
|       label: What was the last working version of Home Assistant Core? |  | ||||||
|       placeholder: core- |  | ||||||
|       description: > |  | ||||||
|         If known, otherwise leave blank. |  | ||||||
|   - type: input |  | ||||||
|     attributes: |  | ||||||
|       label: In which browser are you experiencing the issue with? |  | ||||||
|       placeholder: Google Chrome 88.0.4324.150 |  | ||||||
|       description: > |  | ||||||
|         Provide the full name and don't forget to add the version! |  | ||||||
|   - type: input |  | ||||||
|     attributes: |  | ||||||
|       label: Which operating system are you using to run this browser? |  | ||||||
|       placeholder: macOS Big Sur (1.11) |  | ||||||
|       description: > |  | ||||||
|         Don't forget to add the version! |  | ||||||
|   - type: markdown |  | ||||||
|     attributes: |  | ||||||
|       value: | |  | ||||||
|         # Details |  | ||||||
|  |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: State of relevant entities |  | ||||||
|       description: > |  | ||||||
|         If your issue is about how an entity is shown in the UI, please add the |  | ||||||
|         state and attributes for all situations. You can find this information |  | ||||||
|         at Developer Tools -> States. |  | ||||||
|       render: txt |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: Problem-relevant frontend configuration |  | ||||||
|       description: > |  | ||||||
|         An example configuration that caused the problem for you, e.g., the YAML |  | ||||||
|         configuration of the used cards. Fill this out even if it seems |  | ||||||
|         unimportant to you. Please be sure to remove personal information like |  | ||||||
|         passwords, private URLs and other credentials. |  | ||||||
|       render: yaml |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: Javascript errors shown in your browser console/inspector |  | ||||||
|       description: > |  | ||||||
|         If you come across any Javascript or other error logs, e.g., in your |  | ||||||
|         browser console/inspector please provide them. |  | ||||||
|       render: txt |  | ||||||
|   - type: textarea |  | ||||||
|     attributes: |  | ||||||
|       label: Additional information |  | ||||||
|       description: > |  | ||||||
|         If you have any additional information for us, use the field below. |  | ||||||
|         Please note, you can attach screenshots or screen recordings here, by |  | ||||||
|         dragging and dropping files in the field below. |  | ||||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,17 +1,17 @@ | |||||||
| blank_issues_enabled: false | blank_issues_enabled: false | ||||||
| contact_links: | contact_links: | ||||||
|   - name: Request a feature for the UI / Dashboards |   - name: Request a feature for the UI, Frontend or Lovelace | ||||||
|     url: https://github.com/home-assistant/frontend/discussions/category_choices |     url: https://github.com/home-assistant/frontend/discussions/category_choices | ||||||
|     about: Request an new feature for the Home Assistant frontend. |     about: Request an new feature for the Home Assistant frontend. | ||||||
|   - name: Report a bug that is NOT related to the UI / Dashboards |   - name: Report a bug that is NOT related to the UI, Frontend or Lovelace | ||||||
|     url: https://github.com/home-assistant/core/issues |     url: https://github.com/home-assistant/core/issues | ||||||
|     about: This is the issue tracker for our frontend. Please report other issues in the backend ("core") repository. |     about: This is the issue tracker for our frontend. Please report other issues with the backend repository. | ||||||
|   - name: Report incorrect or missing information on our website |   - name: Report incorrect or missing information on our website | ||||||
|     url: https://github.com/home-assistant/home-assistant.io/issues |     url: https://github.com/home-assistant/home-assistant.io/issues | ||||||
|     about: Our documentation has its own issue tracker. Please report issues with the website there. |     about: Our documentation has its own issue tracker. Please report issues with the website there. | ||||||
|   - name: I have a question or need support |   - name: I have a question or need support | ||||||
|     url: https://www.home-assistant.io/help |     url: https://www.home-assistant.io/help | ||||||
|     about: We use GitHub for tracking bugs. Check our website for resources on getting help. |     about: We use GitHub for tracking bugs, check our website for resources on getting help. | ||||||
|   - name: I'm unsure where to go |   - name: I'm unsure where to go | ||||||
|     url: https://www.home-assistant.io/join-chat |     url: https://www.home-assistant.io/join-chat | ||||||
|     about: If you are unsure where to go, then joining our chat is recommended; Just ask! |     about: If you are unsure where to go, then joining our chat is recommended; Just ask! | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -2,7 +2,9 @@ | |||||||
|   You are amazing! Thanks for contributing to our project! |   You are amazing! Thanks for contributing to our project! | ||||||
|   Please, DO NOT DELETE ANY TEXT from this template! (unless instructed). |   Please, DO NOT DELETE ANY TEXT from this template! (unless instructed). | ||||||
| --> | --> | ||||||
|  |  | ||||||
| ## Breaking change | ## Breaking change | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   If your PR contains a breaking change for existing users, it is important |   If your PR contains a breaking change for existing users, it is important | ||||||
|   to tell them what breaks, how to make it work again and why we did this. |   to tell them what breaks, how to make it work again and why we did this. | ||||||
| @@ -11,8 +13,8 @@ | |||||||
|   Note: Remove this section if this PR is NOT a breaking change. |   Note: Remove this section if this PR is NOT a breaking change. | ||||||
| --> | --> | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Proposed change | ## Proposed change | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   Describe the big picture of your changes here to communicate to the |   Describe the big picture of your changes here to communicate to the | ||||||
|   maintainers why we should accept this pull request. If it fixes a bug |   maintainers why we should accept this pull request. If it fixes a bug | ||||||
| @@ -20,8 +22,8 @@ | |||||||
|   in the additional information section. |   in the additional information section. | ||||||
| --> | --> | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Type of change | ## Type of change | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   What type of change does your PR introduce to the Home Assistant frontend? |   What type of change does your PR introduce to the Home Assistant frontend? | ||||||
|   NOTE: Please, check only 1! box! |   NOTE: Please, check only 1! box! | ||||||
| @@ -36,6 +38,7 @@ | |||||||
| - [ ] Code quality improvements to existing code or addition of tests | - [ ] Code quality improvements to existing code or addition of tests | ||||||
|  |  | ||||||
| ## Example configuration | ## Example configuration | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   Supplying a configuration snippet, makes it easier for a maintainer to test |   Supplying a configuration snippet, makes it easier for a maintainer to test | ||||||
|   your PR. |   your PR. | ||||||
| @@ -46,6 +49,7 @@ | |||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Additional information | ## Additional information | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   Details are important, and help maintainers processing your PR. |   Details are important, and help maintainers processing your PR. | ||||||
|   Please be sure to fill out additional details, if applicable. |   Please be sure to fill out additional details, if applicable. | ||||||
| @@ -56,6 +60,7 @@ | |||||||
| - Link to documentation pull request: | - Link to documentation pull request: | ||||||
|  |  | ||||||
| ## Checklist | ## Checklist | ||||||
|  |  | ||||||
| <!-- | <!-- | ||||||
|   Put an `x` in the boxes that apply. You can also fill these out after |   Put an `x` in the boxes that apply. You can also fill these out after | ||||||
|   creating the PR. If you're unsure about any of them, don't hesitate to ask. |   creating the PR. If you're unsure about any of them, don't hesitate to ask. | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +0,0 @@ | |||||||
| version: 2 |  | ||||||
| updates: |  | ||||||
|   - package-ecosystem: "github-actions" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: weekly |  | ||||||
|       time: "06:00" |  | ||||||
|     open-pull-requests-limit: 10 |  | ||||||
|     labels: |  | ||||||
|       - Dependencies |  | ||||||
|       - GitHub Actions |  | ||||||
							
								
								
									
										51
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,51 +0,0 @@ | |||||||
| Build: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - build-scripts/** |  | ||||||
|           - .browserslistrc |  | ||||||
|           - gulpfile.js |  | ||||||
|  |  | ||||||
| Cast: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - cast/src/** |  | ||||||
|           - src/cast/** |  | ||||||
|  |  | ||||||
| Demo: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - demo/src/** |  | ||||||
|           - src/fake_data/** |  | ||||||
|  |  | ||||||
| Design: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - gallery/src/** |  | ||||||
|           - src/fake_data/** |  | ||||||
|  |  | ||||||
| Dependencies: |  | ||||||
|   - any: |  | ||||||
|       - changed-files: |  | ||||||
|           # Match when only these files are changed (i.e. don't match PRs that happen to add or remove packages) |  | ||||||
|           - any-glob-to-all-files: |  | ||||||
|               - package.json |  | ||||||
|               - renovate.json |  | ||||||
|               - yarn.lock |  | ||||||
|               - .yarn/** |  | ||||||
|               - .yarnrc.yml |  | ||||||
|               - .nvmrc |  | ||||||
|       # Dependabot and Renovate branches always match (i.e. compatibility tweaks by members considered minor) |  | ||||||
|       - head-branch: |  | ||||||
|           - "^renovate/" |  | ||||||
|           - "^dependabot/" |  | ||||||
|  |  | ||||||
| GitHub Actions: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - .github/workflows/** |  | ||||||
|           - .github/*.yml |  | ||||||
|  |  | ||||||
| Supervisor: |  | ||||||
|   - changed-files: |  | ||||||
|       - any-glob-to-any-file: |  | ||||||
|           - hassio/src/** |  | ||||||
							
								
								
									
										5
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,8 +1,3 @@ | |||||||
| categories: |  | ||||||
|   - title: "Dependency updates" |  | ||||||
|     collapse-after: 3 |  | ||||||
|     labels: |  | ||||||
|       - "Dependencies" |  | ||||||
| template: | | template: | | ||||||
|   ## What's Changed |   ## What's Changed | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										85
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										85
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,85 +0,0 @@ | |||||||
| name: Cast deployment |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   schedule: |  | ||||||
|     - cron: "0 0 * * *" |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - master |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   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@v4.2.2 |  | ||||||
|         with: |  | ||||||
|           ref: dev |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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@v4.2.2 |  | ||||||
|         with: |  | ||||||
|           ref: master |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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 }} |  | ||||||
							
								
								
									
										147
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										147
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,111 +10,114 @@ on: | |||||||
|       - dev |       - dev | ||||||
|       - master |       - master | ||||||
|  |  | ||||||
| env: |  | ||||||
|   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: | jobs: | ||||||
|   lint: |   lint: | ||||||
|     name: Lint and check format |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|       - name: Setup Node |       - name: Setting up Node.js | ||||||
|         uses: actions/setup-node@v4.1.0 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version: 12.x | ||||||
|           cache: yarn |       - name: Get yarn cache path | ||||||
|  |         id: yarn-cache-dir-path | ||||||
|  |         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||||
|  |       - name: Fetching Yarn cache | ||||||
|  |         uses: actions/cache@v1 | ||||||
|  |         with: | ||||||
|  |           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||||
|  |           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-yarn- | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: yarn install --immutable |         run: yarn install | ||||||
|       - name: Check for duplicate dependencies |         env: | ||||||
|         run: yarn dedupe --check |           CI: true | ||||||
|       - name: Build resources |       - name: Build resources | ||||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages |         run: ./node_modules/.bin/gulp gen-icons-json build-translations gather-gallery-demos | ||||||
|       - name: Setup lint cache |  | ||||||
|         uses: actions/cache@v4.1.2 |  | ||||||
|         with: |  | ||||||
|           path: | |  | ||||||
|             node_modules/.cache/prettier |  | ||||||
|             node_modules/.cache/eslint |  | ||||||
|             node_modules/.cache/typescript |  | ||||||
|           key: lint-${{ github.sha }} |  | ||||||
|           restore-keys: lint- |  | ||||||
|       - name: Run eslint |       - name: Run eslint | ||||||
|         run: yarn run lint:eslint --quiet |         run: ./node_modules/.bin/eslint '{**/src,src}/**/*.{js,ts,html}' --ignore-path .gitignore | ||||||
|       - name: Run tsc |       - name: Run tsc | ||||||
|         run: yarn run lint:types |         run: ./node_modules/.bin/tsc | ||||||
|       - name: Run lit-analyzer |  | ||||||
|         run: yarn run lint:lit --quiet |  | ||||||
|       - name: Run prettier |  | ||||||
|         run: yarn run lint:prettier |  | ||||||
|   test: |   test: | ||||||
|     name: Run tests |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|       - name: Setup Node |       - name: Setting up Node.js | ||||||
|         uses: actions/setup-node@v4.1.0 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version: 12.x | ||||||
|           cache: yarn |       - name: Get yarn cache path | ||||||
|  |         id: yarn-cache-dir-path | ||||||
|  |         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||||
|  |       - name: Fetching Yarn cache | ||||||
|  |         uses: actions/cache@v1 | ||||||
|  |         with: | ||||||
|  |           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||||
|  |           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-yarn- | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: yarn install --immutable |         run: yarn install | ||||||
|       - name: Build resources |         env: | ||||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data |           CI: true | ||||||
|       - name: Run Tests |       - name: Run Mocha | ||||||
|         run: yarn run test |         run: npm run mocha | ||||||
|   build: |   build: | ||||||
|     name: Build frontend |  | ||||||
|     needs: [lint, test] |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |     needs: [lint, test] | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|       - name: Setup Node |       - name: Setting up Node.js | ||||||
|         uses: actions/setup-node@v4.1.0 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version: 12.x | ||||||
|           cache: yarn |       - name: Get yarn cache path | ||||||
|  |         id: yarn-cache-dir-path | ||||||
|  |         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||||
|  |       - name: Fetching Yarn cache | ||||||
|  |         uses: actions/cache@v1 | ||||||
|  |         with: | ||||||
|  |           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||||
|  |           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-yarn- | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: yarn install --immutable |         run: yarn install | ||||||
|  |         env: | ||||||
|  |           CI: true | ||||||
|       - name: Build Application |       - name: Build Application | ||||||
|         run: ./node_modules/.bin/gulp build-app |         run: ./node_modules/.bin/gulp build-app | ||||||
|         env: |         env: | ||||||
|           IS_TEST: "true" |           IS_TEST: "true" | ||||||
|       - name: Upload bundle stats |  | ||||||
|         uses: actions/upload-artifact@v4.4.3 |  | ||||||
|         with: |  | ||||||
|           name: frontend-bundle-stats |  | ||||||
|           path: build/stats/*.json |  | ||||||
|           if-no-files-found: error |  | ||||||
|   supervisor: |   supervisor: | ||||||
|     name: Build supervisor |  | ||||||
|     needs: [lint, test] |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |     needs: [lint, test] | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|       - name: Setup Node |       - name: Setting up Node.js | ||||||
|         uses: actions/setup-node@v4.1.0 |         uses: actions/setup-node@v1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version: 12.x | ||||||
|           cache: yarn |       - name: Get yarn cache path | ||||||
|  |         id: yarn-cache-dir-path | ||||||
|  |         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||||
|  |       - name: Fetching Yarn cache | ||||||
|  |         uses: actions/cache@v1 | ||||||
|  |         with: | ||||||
|  |           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||||
|  |           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-yarn- | ||||||
|       - name: Install dependencies |       - name: Install dependencies | ||||||
|         run: yarn install --immutable |         run: yarn install | ||||||
|  |         env: | ||||||
|  |           CI: true | ||||||
|       - name: Build Application |       - name: Build Application | ||||||
|         run: ./node_modules/.bin/gulp build-hassio |         run: ./node_modules/.bin/gulp build-hassio | ||||||
|         env: |         env: | ||||||
|           IS_TEST: "true" |           IS_TEST: "true" | ||||||
|       - name: Upload bundle stats |  | ||||||
|         uses: actions/upload-artifact@v4.4.3 |  | ||||||
|         with: |  | ||||||
|           name: supervisor-bundle-stats |  | ||||||
|           path: build/stats/*.json |  | ||||||
|           if-no-files-found: error |  | ||||||
|   | |||||||
							
								
								
									
										60
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										60
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,44 +17,44 @@ jobs: | |||||||
|       matrix: |       matrix: | ||||||
|         # Override automatic language detection by changing the below list |         # Override automatic language detection by changing the below list | ||||||
|         # Supported options are ['csharp', 'cpp', 'go', 'java', 'javascript', 'python'] |         # Supported options are ['csharp', 'cpp', 'go', 'java', 'javascript', 'python'] | ||||||
|         language: ["javascript"] |         language: ['javascript'] | ||||||
|         # Learn more... |         # Learn more... | ||||||
|         # https://docs.github.com/en/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#overriding-automatic-language-detection |         # https://docs.github.com/en/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#overriding-automatic-language-detection | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |     - name: Checkout repository | ||||||
|         uses: actions/checkout@v4.2.2 |       uses: actions/checkout@v2 | ||||||
|         with: |       with: | ||||||
|           # We must fetch at least the immediate parents so that if this is |         # We must fetch at least the immediate parents so that if this is | ||||||
|           # a pull request then we can checkout the head. |         # a pull request then we can checkout the head. | ||||||
|           fetch-depth: 2 |         fetch-depth: 2 | ||||||
|  |  | ||||||
|       # If this run was triggered by a pull request event, then checkout |     # If this run was triggered by a pull request event, then checkout | ||||||
|       # the head of the pull request instead of the merge commit. |     # the head of the pull request instead of the merge commit. | ||||||
|       - run: git checkout HEAD^2 |     - run: git checkout HEAD^2 | ||||||
|         if: ${{ github.event_name == 'pull_request' }} |       if: ${{ github.event_name == 'pull_request' }} | ||||||
|  |  | ||||||
|       # Initializes the CodeQL tools for scanning. |     # Initializes the CodeQL tools for scanning. | ||||||
|       - name: Initialize CodeQL |     - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v3 |       uses: github/codeql-action/init@v1 | ||||||
|         with: |       with: | ||||||
|           languages: ${{ matrix.language }} |         languages: ${{ matrix.language }} | ||||||
|  |  | ||||||
|       # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). |     # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). | ||||||
|       # If this step fails, then you should remove it and run the build manually (see below) |     # If this step fails, then you should remove it and run the build manually (see below) | ||||||
|       - name: Autobuild |     - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v3 |       uses: github/codeql-action/autobuild@v1 | ||||||
|  |  | ||||||
|       # ℹ️ Command-line programs to run using the OS shell. |     # ℹ️ Command-line programs to run using the OS shell. | ||||||
|       # 📚 https://git.io/JvXDl |     # 📚 https://git.io/JvXDl | ||||||
|  |  | ||||||
|       # ✏️ If the Autobuild fails above, remove it and uncomment the following three lines |     # ✏️ If the Autobuild fails above, remove it and uncomment the following three lines | ||||||
|       #    and modify them (or add more) to build your code if your project |     #    and modify them (or add more) to build your code if your project | ||||||
|       #    uses a compiled language |     #    uses a compiled language | ||||||
|  |  | ||||||
|       #- run: | |     #- run: | | ||||||
|       #   make bootstrap |     #   make bootstrap | ||||||
|       #   make release |     #   make release | ||||||
|  |  | ||||||
|       - name: Perform CodeQL Analysis |     - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v3 |       uses: github/codeql-action/analyze@v1 | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | name: Demo | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - dev | ||||||
|  | jobs: | ||||||
|  |   deploy: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Check out files from GitHub | ||||||
|  |         uses: actions/checkout@v2 | ||||||
|  |       - name: Setting up Node.js | ||||||
|  |         uses: actions/setup-node@v1 | ||||||
|  |         with: | ||||||
|  |           node-version: 12.x | ||||||
|  |       - name: Get yarn cache path | ||||||
|  |         id: yarn-cache-dir-path | ||||||
|  |         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||||
|  |       - name: Fetching Yarn cache | ||||||
|  |         uses: actions/cache@v1 | ||||||
|  |         with: | ||||||
|  |           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||||
|  |           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-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 | ||||||
							
								
								
									
										86
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										86
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,86 +0,0 @@ | |||||||
| name: Demo deployment |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   schedule: |  | ||||||
|     - cron: "0 0 * * *" |  | ||||||
|   push: |  | ||||||
|     branches: |  | ||||||
|       - dev |  | ||||||
|       - master |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   NODE_OPTIONS: --max_old_space_size=6144 |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   deploy_dev: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     name: Demo Development |  | ||||||
|     if: github.event_name != 'push' || github.ref_name != '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@v4.2.2 |  | ||||||
|         with: |  | ||||||
|           ref: dev |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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_name == '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@v4.2.2 |  | ||||||
|         with: |  | ||||||
|           ref: master |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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 }} |  | ||||||
							
								
								
									
										42
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										42
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,42 +0,0 @@ | |||||||
| name: Design deployment |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   schedule: |  | ||||||
|     - cron: "0 0 * * *" |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   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@v4.2.2 |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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 }} |  | ||||||
							
								
								
									
										51
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										51
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,51 +0,0 @@ | |||||||
| name: Design preview |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|     types: |  | ||||||
|       - opened |  | ||||||
|       - synchronize |  | ||||||
|       - reopened |  | ||||||
|       - labeled |  | ||||||
|     branches: |  | ||||||
|       - dev |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   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@v4.2.2 |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           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" |  | ||||||
							
								
								
									
										15
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,15 +0,0 @@ | |||||||
| name: "Pull Request Labeler" |  | ||||||
|  |  | ||||||
| on: pull_request_target |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   triage: |  | ||||||
|     permissions: |  | ||||||
|       contents: read |  | ||||||
|       pull-requests: write |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Apply labels |  | ||||||
|         uses: actions/labeler@v5.0.0 |  | ||||||
|         with: |  | ||||||
|           sync-labels: true |  | ||||||
							
								
								
									
										3
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,10 +9,9 @@ jobs: | |||||||
|   lock: |   lock: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: dessant/lock-threads@v5.0.1 |       - uses: dessant/lock-threads@v2.0.1 | ||||||
|         with: |         with: | ||||||
|           github-token: ${{ github.token }} |           github-token: ${{ github.token }} | ||||||
|           process-only: "issues, prs" |  | ||||||
|           issue-lock-inactive-days: "30" |           issue-lock-inactive-days: "30" | ||||||
|           issue-exclude-created-before: "2020-10-01T00:00:00Z" |           issue-exclude-created-before: "2020-10-01T00:00:00Z" | ||||||
|           issue-lock-reason: "" |           issue-lock-reason: "" | ||||||
|   | |||||||
							
								
								
									
										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 Gallery build | ||||||
|  |         run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_GALLERY_DEV_BUILD_HOOK }} | ||||||
							
								
								
									
										71
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										71
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,71 +0,0 @@ | |||||||
| name: Nightly |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   schedule: |  | ||||||
|     - cron: "0 1 * * *" |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   PYTHON_VERSION: "3.12" |  | ||||||
|   NODE_OPTIONS: --max_old_space_size=6144 |  | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   actions: none |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   nightly: |  | ||||||
|     name: Nightly |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     permissions: |  | ||||||
|       contents: write |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout the repository |  | ||||||
|         uses: actions/checkout@v4.2.2 |  | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} |  | ||||||
|         uses: actions/setup-python@v5 |  | ||||||
|         with: |  | ||||||
|           python-version: ${{ env.PYTHON_VERSION }} |  | ||||||
|  |  | ||||||
|       - name: Setup Node |  | ||||||
|         uses: actions/setup-node@v4.1.0 |  | ||||||
|         with: |  | ||||||
|           node-version-file: ".nvmrc" |  | ||||||
|           cache: yarn |  | ||||||
|  |  | ||||||
|       - name: Install dependencies |  | ||||||
|         run: yarn install |  | ||||||
|  |  | ||||||
|       - name: Download translations |  | ||||||
|         run: ./script/translations_download |  | ||||||
|         env: |  | ||||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} |  | ||||||
|  |  | ||||||
|       - name: Bump version |  | ||||||
|         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 |  | ||||||
|  |  | ||||||
|       - name: Archive translations |  | ||||||
|         run: tar -czvf translations.tar.gz translations |  | ||||||
|  |  | ||||||
|       - name: Upload build artifacts |  | ||||||
|         uses: actions/upload-artifact@v4.4.3 |  | ||||||
|         with: |  | ||||||
|           name: wheels |  | ||||||
|           path: dist/home_assistant_frontend*.whl |  | ||||||
|           if-no-files-found: error |  | ||||||
|  |  | ||||||
|       - name: Upload translations |  | ||||||
|         uses: actions/upload-artifact@v4.4.3 |  | ||||||
|         with: |  | ||||||
|           name: translations |  | ||||||
|           path: translations.tar.gz |  | ||||||
|           if-no-files-found: error |  | ||||||
							
								
								
									
										25
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,25 +0,0 @@ | |||||||
| name: RelativeCI |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_run: |  | ||||||
|     workflows: [CI] |  | ||||||
|     types: |  | ||||||
|       - completed |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   upload: |  | ||||||
|     name: Upload stats |  | ||||||
|     if: ${{ github.event.workflow_run.conclusion == 'success' }} |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         bundle: [frontend, supervisor] |  | ||||||
|         build: [modern, legacy] |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Send bundle stats and build information to RelativeCI |  | ||||||
|         uses: relative-ci/agent-action@v2.1.12 |  | ||||||
|         with: |  | ||||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} |  | ||||||
|           token: ${{ github.token }} |  | ||||||
|           artifactName: ${{ format('{0}-bundle-stats', matrix.bundle) }} |  | ||||||
|           webpackStatsFile: ${{ format('{0}-{1}.json', matrix.bundle, matrix.build) }} |  | ||||||
							
								
								
									
										11
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -5,19 +5,10 @@ on: | |||||||
|     branches: |     branches: | ||||||
|       - dev |       - dev | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   contents: read |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   update_release_draft: |   update_release_draft: | ||||||
|     permissions: |  | ||||||
|       # write permission for contents is required to create a github release |  | ||||||
|       contents: write |  | ||||||
|       # write permission for pull-requests is required for autolabeler |  | ||||||
|       # otherwise, read permission is required at least |  | ||||||
|       pull-requests: read |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: release-drafter/release-drafter@v6.0.0 |       - uses: release-drafter/release-drafter@v5 | ||||||
|         env: |         env: | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|   | |||||||
							
								
								
									
										72
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										72
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -6,60 +6,38 @@ on: | |||||||
|       - published |       - published | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   PYTHON_VERSION: "3.12" |   WHEELS_TAG: 3.7-alpine3.11 | ||||||
|   NODE_OPTIONS: --max_old_space_size=6144 |   PYTHON_VERSION: 3.7 | ||||||
|  |   NODE_VERSION: 12.1 | ||||||
| # Set default workflow permissions |  | ||||||
| # All scopes not mentioned here are set to no access |  | ||||||
| # https://docs.github.com/en/actions/security-guides/automatic-token-authentication#permissions-for-the-github_token |  | ||||||
| permissions: |  | ||||||
|   actions: none |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   release: |   release: | ||||||
|     name: Release |     name: Release | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       contents: write # Required to upload release assets |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|       - name: Verify version |       - name: Verify version | ||||||
|         uses: home-assistant/actions/helpers/verify-version@master |         uses: home-assistant/actions/helpers/verify-version@master | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} |       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@v2 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.PYTHON_VERSION }} |           python-version: ${{ env.PYTHON_VERSION }} | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Set up Node ${{ env.NODE_VERSION }} | ||||||
|         uses: actions/setup-node@v4.1.0 |         uses: actions/setup-node@v2 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version: ${{ env.NODE_VERSION }} | ||||||
|           cache: yarn |  | ||||||
|  |  | ||||||
|       - name: Install dependencies |  | ||||||
|         run: yarn install |  | ||||||
|  |  | ||||||
|       - name: Download Translations |  | ||||||
|         run: ./script/translations_download |  | ||||||
|         env: |  | ||||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} |  | ||||||
|       - name: Build and release package |       - name: Build and release package | ||||||
|         run: | |         run: | | ||||||
|           python3 -m pip install twine build |           python3 -m pip install twine | ||||||
|           export TWINE_USERNAME="__token__" |           export TWINE_USERNAME="__token__" | ||||||
|           export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}" |           export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}" | ||||||
|           export SKIP_FETCH_NIGHTLY_TRANSLATIONS=1 |  | ||||||
|           script/release |  | ||||||
|  |  | ||||||
|       - name: Upload release assets |           script/release | ||||||
|         uses: softprops/action-gh-release@v2.0.8 |  | ||||||
|         with: |  | ||||||
|           files: | |  | ||||||
|             dist/*.whl |  | ||||||
|             dist/*.tar.gz |  | ||||||
|  |  | ||||||
|   wheels-init: |   wheels-init: | ||||||
|     name: Init wheels build |     name: Init wheels build | ||||||
| @@ -73,11 +51,31 @@ jobs: | |||||||
|           version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' ) |           version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' ) | ||||||
|           echo "home-assistant-frontend==$version" > ./requirements.txt |           echo "home-assistant-frontend==$version" > ./requirements.txt | ||||||
|  |  | ||||||
|       - name: Build wheels |       - name: Upload requirements.txt | ||||||
|         uses: home-assistant/wheels@2024.07.1 |         uses: actions/upload-artifact@v2 | ||||||
|         with: |         with: | ||||||
|           abi: cp312 |           name: requirements | ||||||
|           tag: musllinux_1_2 |           path: ./requirements.txt | ||||||
|           arch: amd64 |  | ||||||
|  |   build-wheels: | ||||||
|  |     name: Build wheels for ${{ matrix.arch }} | ||||||
|  |     needs: wheels-init | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         arch: ["aarch64", "armhf", "armv7", "amd64", "i386"] | ||||||
|  |     steps: | ||||||
|  |       - name: Download requirements.txt | ||||||
|  |         uses: actions/download-artifact@v2 | ||||||
|  |         with: | ||||||
|  |           name: requirements | ||||||
|  |  | ||||||
|  |       - name: Build wheels | ||||||
|  |         uses: home-assistant/wheels@master | ||||||
|  |         with: | ||||||
|  |           tag: ${{ env.WHEELS_TAG }} | ||||||
|  |           arch: ${{ matrix.arch }} | ||||||
|  |           wheels-host: ${{ secrets.WHEELS_HOST }} | ||||||
|           wheels-key: ${{ secrets.WHEELS_KEY }} |           wheels-key: ${{ secrets.WHEELS_KEY }} | ||||||
|  |           wheels-user: wheels | ||||||
|           requirements: "requirements.txt" |           requirements: "requirements.txt" | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: 90 days stale policy |       - name: 90 days stale policy | ||||||
|         uses: actions/stale@v9.0.0 |         uses: actions/stale@v3.0.13 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} |           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           days-before-stale: 90 |           days-before-stale: 90 | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										48
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,16 @@ | |||||||
| name: Translations | name: Translations | ||||||
|  |  | ||||||
| on: | on: | ||||||
|  |   schedule: | ||||||
|  |     - cron: "30 0 * * *" | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - dev |       - dev | ||||||
|     paths: |     paths: | ||||||
|       - src/translations/en.json |       - translations/en.json | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   NODE_VERSION: 12 | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   upload: |   upload: | ||||||
| @@ -13,9 +18,48 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|  |       - name: Set up Node ${{ env.NODE_VERSION }} | ||||||
|  |         uses: actions/setup-node@v2 | ||||||
|  |         with: | ||||||
|  |           node-version: ${{ env.NODE_VERSION }} | ||||||
|  |  | ||||||
|       - name: Upload Translations |       - name: Upload Translations | ||||||
|         run: | |         run: | | ||||||
|           export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}" |           export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}" | ||||||
|  |  | ||||||
|           ./script/translations_upload_base |           ./script/translations_upload_base | ||||||
|  |  | ||||||
|  |   download: | ||||||
|  |     name: Download | ||||||
|  |     needs: upload | ||||||
|  |     if: github.event_name == 'schedule' | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: Checkout the repository | ||||||
|  |         uses: actions/checkout@v2 | ||||||
|  |  | ||||||
|  |       - name: Set up Node ${{ env.NODE_VERSION }} | ||||||
|  |         uses: actions/setup-node@v2 | ||||||
|  |         with: | ||||||
|  |           node-version: ${{ env.NODE_VERSION }} | ||||||
|  |  | ||||||
|  |       - name: Download Translations | ||||||
|  |         run: | | ||||||
|  |           export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}" | ||||||
|  |  | ||||||
|  |           npm install | ||||||
|  |           ./script/translations_download | ||||||
|  |  | ||||||
|  |       - name: Initialize git | ||||||
|  |         uses: home-assistant/actions/helpers/git-init@master | ||||||
|  |         with: | ||||||
|  |           name: GitHub Action | ||||||
|  |           email: github-action@users.noreply.github.com | ||||||
|  |  | ||||||
|  |       - name: Update translation | ||||||
|  |         run: | | ||||||
|  |           git add translations | ||||||
|  |           git commit -am "Translation update" | ||||||
|  |           git push | ||||||
|   | |||||||
							
								
								
									
										38
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										38
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,23 +1,10 @@ | |||||||
| .DS_Store | build | ||||||
| .reify-cache | build-translations/* | ||||||
|  | node_modules/* | ||||||
| # build |  | ||||||
| build/ |  | ||||||
| dist/ |  | ||||||
| /hass_frontend/ |  | ||||||
| /translations/ |  | ||||||
|  |  | ||||||
| # yarn |  | ||||||
| .yarn/* |  | ||||||
| !.yarn/patches |  | ||||||
| !.yarn/releases |  | ||||||
| !.yarn/plugins |  | ||||||
| !.yarn/sdks |  | ||||||
| !.yarn/versions |  | ||||||
| .pnp.* |  | ||||||
| /node_modules/ |  | ||||||
| yarn-error.log |  | ||||||
| npm-debug.log | npm-debug.log | ||||||
|  | .DS_Store | ||||||
|  | hass_frontend/* | ||||||
|  | .reify-cache | ||||||
|  |  | ||||||
| # Python stuff | # Python stuff | ||||||
| *.py[cod] | *.py[cod] | ||||||
| @@ -27,8 +14,11 @@ npm-debug.log | |||||||
| # venv stuff | # venv stuff | ||||||
| pyvenv.cfg | pyvenv.cfg | ||||||
| pip-selfcheck.json | pip-selfcheck.json | ||||||
| /venv/ | venv | ||||||
| .venv | .venv | ||||||
|  | lib | ||||||
|  | bin | ||||||
|  | dist | ||||||
|  |  | ||||||
| # vscode | # vscode | ||||||
| .vscode/* | .vscode/* | ||||||
| @@ -41,12 +31,10 @@ src/cast/dev_const.ts | |||||||
|  |  | ||||||
| # Secrets | # Secrets | ||||||
| .lokalise_token | .lokalise_token | ||||||
|  | yarn-error.log | ||||||
|  |  | ||||||
| # asdf | #asdf | ||||||
| .tool-versions | .tool-versions | ||||||
|  |  | ||||||
| # Home Assistant config | # Home Assistant config | ||||||
| /config/ | /config | ||||||
|  |  | ||||||
| # Jetbrains |  | ||||||
| /.idea/ |  | ||||||
|   | |||||||
| @@ -1 +0,0 @@ | |||||||
| yarn run lint-staged --relative --shell "/bin/bash" |  | ||||||
| @@ -1,4 +1,10 @@ | |||||||
| CLA.md | build | ||||||
| CODE_OF_CONDUCT.md | build-translations/* | ||||||
| LICENSE.md | translations/* | ||||||
| PULL_REQUEST_TEMPLATE.md | node_modules/* | ||||||
|  | hass_frontend/* | ||||||
|  | pip-selfcheck.json | ||||||
|  |  | ||||||
|  | # vscode | ||||||
|  | .vscode/* | ||||||
|  | !.vscode/extensions.json | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @@ -2,8 +2,7 @@ | |||||||
|   "recommendations": [ |   "recommendations": [ | ||||||
|     "dbaeumer.vscode-eslint", |     "dbaeumer.vscode-eslint", | ||||||
|     "esbenp.prettier-vscode", |     "esbenp.prettier-vscode", | ||||||
|     "runem.lit-plugin", |     "bierner.lit-html", | ||||||
|     "github.vscode-pull-request-github", |     "runem.lit-plugin" | ||||||
|     "eamodio.gitlens" |  | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,9 @@ | |||||||
|       "webRoot": "${workspaceFolder}/hass_frontend", |       "webRoot": "${workspaceFolder}/hass_frontend", | ||||||
|       "disableNetworkCache": true, |       "disableNetworkCache": true, | ||||||
|       "preLaunchTask": "Develop Frontend", |       "preLaunchTask": "Develop Frontend", | ||||||
|       "outFiles": ["${workspaceFolder}/hass_frontend/frontend_latest/*.js"] |       "outFiles": [ | ||||||
|  |         "${workspaceFolder}/hass_frontend/frontend_latest/*.js" | ||||||
|  |       ] | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "name": "Debug Gallery", |       "name": "Debug Gallery", | ||||||
| @@ -37,6 +39,6 @@ | |||||||
|       "webRoot": "${workspaceFolder}/cast/dist", |       "webRoot": "${workspaceFolder}/cast/dist", | ||||||
|       "disableNetworkCache": true, |       "disableNetworkCache": true, | ||||||
|       "preLaunchTask": "Develop Cast" |       "preLaunchTask": "Develop Cast" | ||||||
|     } |     }, | ||||||
|   ] |   ] | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -181,7 +181,7 @@ | |||||||
|     { |     { | ||||||
|       "label": "Run HA Core for Supervisor in devcontainer", |       "label": "Run HA Core for Supervisor in devcontainer", | ||||||
|       "type": "shell", |       "type": "shell", | ||||||
|       "command": "SUPERVISOR=${input:supervisorHost} SUPERVISOR_TOKEN=${input:supervisorToken} script/core", |       "command": "HASSIO=${input:supervisorHost} HASSIO_TOKEN=${input:supervisorToken} script/core", | ||||||
|       "isBackground": true, |       "isBackground": true, | ||||||
|       "group": { |       "group": { | ||||||
|         "kind": "build", |         "kind": "build", | ||||||
| @@ -191,12 +191,6 @@ | |||||||
|       "runOptions": { |       "runOptions": { | ||||||
|         "instanceLimit": 1 |         "instanceLimit": 1 | ||||||
|       } |       } | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       "label": "Setup and fetch nightly translations", |  | ||||||
|       "type": "gulp", |  | ||||||
|       "task": "setup-and-fetch-nightly-translations", |  | ||||||
|       "problemMatcher": [] |  | ||||||
|     } |     } | ||||||
|   ], |   ], | ||||||
|   "inputs": [ |   "inputs": [ | ||||||
|   | |||||||
| @@ -1,34 +0,0 @@ | |||||||
| diff --git a/lib/legacy/class.js b/lib/legacy/class.js |  | ||||||
| index aee2511be1cd9bf900ee552bc98190c1631c57c0..f2f499d68bf52034cac9c28307c99e8ce6b8417d 100644 |  | ||||||
| --- a/lib/legacy/class.js |  | ||||||
| +++ b/lib/legacy/class.js |  | ||||||
| @@ -304,17 +304,23 @@ function GenerateClassFromInfo(info, Base, behaviors) { |  | ||||||
|        // only proceed if the generated class' prototype has not been registered. |  | ||||||
|        const generatedProto = PolymerGenerated.prototype; |  | ||||||
|        if (!generatedProto.hasOwnProperty(JSCompiler_renameProperty('__hasRegisterFinished', generatedProto))) { |  | ||||||
| -        generatedProto.__hasRegisterFinished = true; |  | ||||||
| +        // make sure legacy lifecycle is called on the *element*'s prototype |  | ||||||
| +        // and not the generated class prototype; if the element has been |  | ||||||
| +        // extended, these are *not* the same. |  | ||||||
| +        const proto = Object.getPrototypeOf(this); |  | ||||||
| +        // Only set flag when generated prototype itself is registered, |  | ||||||
| +        // as this element may be extended from, and needs to run `registered` |  | ||||||
| +        // on all behaviors on the subclass as well. |  | ||||||
| +        if (proto === generatedProto) { |  | ||||||
| +          generatedProto.__hasRegisterFinished = true; |  | ||||||
| +        } |  | ||||||
|          // ensure superclass is registered first. |  | ||||||
|          super._registered(); |  | ||||||
|          // copy properties onto the generated class lazily if we're optimizing, |  | ||||||
| -        if (legacyOptimizations) { |  | ||||||
| +        if (legacyOptimizations && !Object.hasOwnProperty(generatedProto, '__hasCopiedProperties')) { |  | ||||||
| +          generatedProto.__hasCopiedProperties = true; |  | ||||||
|            copyPropertiesToProto(generatedProto); |  | ||||||
|          } |  | ||||||
| -        // make sure legacy lifecycle is called on the *element*'s prototype |  | ||||||
| -        // and not the generated class prototype; if the element has been |  | ||||||
| -        // extended, these are *not* the same. |  | ||||||
| -        const proto = Object.getPrototypeOf(this); |  | ||||||
|          let list = lifecycle.beforeRegister; |  | ||||||
|          if (list) { |  | ||||||
|            for (let i=0; i < list.length; i++) { |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| diff --git a/dist/hls.light.mjs b/dist/hls.light.mjs |  | ||||||
| index eed9d788fafdb159975e1a2eb08ac88ba9c9ac33..ace881935e6665946f1c8110ebd2f739cde4427e 100644 |  | ||||||
| --- a/dist/hls.light.mjs |  | ||||||
| +++ b/dist/hls.light.mjs |  | ||||||
| @@ -20523,9 +20523,9 @@ class Hls { |  | ||||||
|  } |  | ||||||
|  Hls.defaultConfig = void 0; |  | ||||||
|   |  | ||||||
| -var KeySystemFormats = empty.KeySystemFormats; |  | ||||||
| -var KeySystems = empty.KeySystems; |  | ||||||
| -var SubtitleStreamController = empty.SubtitleStreamController; |  | ||||||
| -var TimelineController = empty.TimelineController; |  | ||||||
| +var KeySystemFormats = empty; |  | ||||||
| +var KeySystems = empty; |  | ||||||
| +var SubtitleStreamController = empty; |  | ||||||
| +var TimelineController = empty; |  | ||||||
|  export { AbrController, AttrList, Cues as AudioStreamController, Cues as AudioTrackController, BasePlaylistController, BaseSegment, BaseStreamController, BufferController, Cues as CMCDController, CapLevelController, ChunkMetadata, ContentSteeringController, DateRange, Cues as EMEController, ErrorActionFlags, ErrorController, ErrorDetails, ErrorTypes, Events, FPSController, Fragment, Hls, HlsSkip, HlsUrlParameters, KeySystemFormats, KeySystems, Level, LevelDetails, LevelKey, LoadStats, MetadataSchema, NetworkErrorAction, Part, PlaylistLevelType, SubtitleStreamController, Cues as SubtitleTrackController, TimelineController, Hls as default, getMediaSource, isMSESupported, isSupported }; |  | ||||||
|  //# sourceMappingURL=hls.light.mjs.map |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,60 +0,0 @@ | |||||||
| diff --git a/modular/sortable.core.esm.js b/modular/sortable.core.esm.js |  | ||||||
| index 8b5e49b011713c8859c669069fbe85ce53974e1d..6a0afc92787157b8a31c38cc5f67dfa526090a00 100644 |  | ||||||
| --- a/modular/sortable.core.esm.js |  | ||||||
| +++ b/modular/sortable.core.esm.js |  | ||||||
| @@ -1781,11 +1781,16 @@ Sortable.prototype = /** @lends Sortable.prototype */{ |  | ||||||
|          } |  | ||||||
|          if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, !!target) !== false) { |  | ||||||
|            capture(); |  | ||||||
| -          if (elLastChild && elLastChild.nextSibling) { |  | ||||||
| -            // the last draggable element is not the last node |  | ||||||
| -            el.insertBefore(dragEl, elLastChild.nextSibling); |  | ||||||
| -          } else { |  | ||||||
| -            el.appendChild(dragEl); |  | ||||||
| +          try { |  | ||||||
| +            if (elLastChild && elLastChild.nextSibling) { |  | ||||||
| +              // the last draggable element is not the last node |  | ||||||
| +              el.insertBefore(dragEl, elLastChild.nextSibling); |  | ||||||
| +            } else { |  | ||||||
| +              el.appendChild(dragEl); |  | ||||||
| +            } |  | ||||||
| +          } |  | ||||||
| +          catch(err) { |  | ||||||
| +            return completed(false); |  | ||||||
|            } |  | ||||||
|            parentEl = el; // actualization |  | ||||||
|   |  | ||||||
| @@ -1802,7 +1807,12 @@ Sortable.prototype = /** @lends Sortable.prototype */{ |  | ||||||
|          targetRect = getRect(target); |  | ||||||
|          if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, false) !== false) { |  | ||||||
|            capture(); |  | ||||||
| -          el.insertBefore(dragEl, firstChild); |  | ||||||
| +          try { |  | ||||||
| +            el.insertBefore(dragEl, firstChild); |  | ||||||
| +          } |  | ||||||
| +          catch(err) { |  | ||||||
| +            return completed(false); |  | ||||||
| +          } |  | ||||||
|            parentEl = el; // actualization |  | ||||||
|   |  | ||||||
|            changed(); |  | ||||||
| @@ -1849,10 +1859,15 @@ Sortable.prototype = /** @lends Sortable.prototype */{ |  | ||||||
|            _silent = true; |  | ||||||
|            setTimeout(_unsilent, 30); |  | ||||||
|            capture(); |  | ||||||
| -          if (after && !nextSibling) { |  | ||||||
| -            el.appendChild(dragEl); |  | ||||||
| -          } else { |  | ||||||
| -            target.parentNode.insertBefore(dragEl, after ? nextSibling : target); |  | ||||||
| +          try { |  | ||||||
| +            if (after && !nextSibling) { |  | ||||||
| +              el.appendChild(dragEl); |  | ||||||
| +            } else { |  | ||||||
| +              target.parentNode.insertBefore(dragEl, after ? nextSibling : target); |  | ||||||
| +            } |  | ||||||
| +          } |  | ||||||
| +          catch(err) { |  | ||||||
| +            return completed(false); |  | ||||||
|            } |  | ||||||
|   |  | ||||||
|            // Undo chrome's scroll adjustment (has no effect on other browsers) |  | ||||||
| @@ -1,55 +0,0 @@ | |||||||
| diff --git a/build/inject-manifest.js b/build/inject-manifest.js |  | ||||||
| index 60e3d2bb51c11a19fbbedbad65e101082ec41c36..fed6026630f43f86e25446383982cf6fb694313b 100644 |  | ||||||
| --- a/build/inject-manifest.js |  | ||||||
| +++ b/build/inject-manifest.js |  | ||||||
| @@ -104,7 +104,7 @@ async function injectManifest(config) { |  | ||||||
|              replaceString: manifestString, |  | ||||||
|              searchString: options.injectionPoint, |  | ||||||
|          }); |  | ||||||
| -        filesToWrite[options.swDest] = source; |  | ||||||
| +        filesToWrite[options.swDest] = source.replace(url, encodeURI(upath_1.default.basename(destPath))); |  | ||||||
|          filesToWrite[destPath] = map; |  | ||||||
|      } |  | ||||||
|      else { |  | ||||||
| diff --git a/build/lib/translate-url-to-sourcemap-paths.js b/build/lib/translate-url-to-sourcemap-paths.js |  | ||||||
| index 3220c5474eeac6e8a56ca9b2ac2bd9be48529e43..5f003879a904d4840529a42dd056d288fd213771 100644 |  | ||||||
| --- a/build/lib/translate-url-to-sourcemap-paths.js |  | ||||||
| +++ b/build/lib/translate-url-to-sourcemap-paths.js |  | ||||||
| @@ -22,7 +22,7 @@ function translateURLToSourcemapPaths(url, swSrc, swDest) { |  | ||||||
|          const possibleSrcPath = upath_1.default.resolve(upath_1.default.dirname(swSrc), url); |  | ||||||
|          if (fs_extra_1.default.existsSync(possibleSrcPath)) { |  | ||||||
|              srcPath = possibleSrcPath; |  | ||||||
| -            destPath = upath_1.default.resolve(upath_1.default.dirname(swDest), url); |  | ||||||
| +            destPath = `${swDest}.map`; |  | ||||||
|          } |  | ||||||
|          else { |  | ||||||
|              warning = `${errors_1.errors['cant-find-sourcemap']} ${possibleSrcPath}`; |  | ||||||
| diff --git a/src/inject-manifest.ts b/src/inject-manifest.ts |  | ||||||
| index 8795ddcaa77aea7b0356417e4bc4b19e2b3f860c..fcdc68342d9ac53936c9ed40a9ccfc2f5070cad3 100644 |  | ||||||
| --- a/src/inject-manifest.ts |  | ||||||
| +++ b/src/inject-manifest.ts |  | ||||||
| @@ -129,7 +129,10 @@ export async function injectManifest( |  | ||||||
|        searchString: options.injectionPoint!, |  | ||||||
|      }); |  | ||||||
|   |  | ||||||
| -    filesToWrite[options.swDest] = source; |  | ||||||
| +    filesToWrite[options.swDest] = source.replace( |  | ||||||
| +      url!, |  | ||||||
| +      encodeURI(upath.basename(destPath)), |  | ||||||
| +    ); |  | ||||||
|      filesToWrite[destPath] = map; |  | ||||||
|    } else { |  | ||||||
|      // If there's no sourcemap associated with swSrc, a simple string |  | ||||||
| diff --git a/src/lib/translate-url-to-sourcemap-paths.ts b/src/lib/translate-url-to-sourcemap-paths.ts |  | ||||||
| index 072eac40d4ef5d095a01cb7f7e392a9e034853bd..f0bbe69e88ef3a415de18a7e9cb264daea273d71 100644 |  | ||||||
| --- a/src/lib/translate-url-to-sourcemap-paths.ts |  | ||||||
| +++ b/src/lib/translate-url-to-sourcemap-paths.ts |  | ||||||
| @@ -28,7 +28,7 @@ export function translateURLToSourcemapPaths( |  | ||||||
|      const possibleSrcPath = upath.resolve(upath.dirname(swSrc), url); |  | ||||||
|      if (fse.existsSync(possibleSrcPath)) { |  | ||||||
|        srcPath = possibleSrcPath; |  | ||||||
| -      destPath = upath.resolve(upath.dirname(swDest), url); |  | ||||||
| +      destPath = `${swDest}.map`; |  | ||||||
|      } else { |  | ||||||
|        warning = `${errors['cant-find-sourcemap']} ${possibleSrcPath}`; |  | ||||||
|      } |  | ||||||
							
								
								
									
										934
									
								
								.yarn/releases/yarn-4.5.1.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										934
									
								
								.yarn/releases/yarn-4.5.1.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,9 +0,0 @@ | |||||||
| compressionLevel: mixed |  | ||||||
|  |  | ||||||
| defaultSemverRangePrefix: "" |  | ||||||
|  |  | ||||||
| enableGlobalCache: false |  | ||||||
|  |  | ||||||
| nodeLinker: node-modules |  | ||||||
|  |  | ||||||
| yarnPath: .yarn/releases/yarn-4.5.1.cjs |  | ||||||
| @@ -1,4 +1,5 @@ | |||||||
| include README.md | include README.md | ||||||
|  | include LICENSE.md | ||||||
| graft hass_frontend | graft hass_frontend | ||||||
| graft hass_frontend_es5 | graft hass_frontend_es5 | ||||||
| recursive-exclude * *.py[co] | recursive-exclude * *.py[co] | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| This is the repository for the official [Home Assistant](https://home-assistant.io) frontend. | This is the repository for the official [Home Assistant](https://home-assistant.io) frontend. | ||||||
|  |  | ||||||
| [](https://demo.home-assistant.io/) | [](https://demo.home-assistant.io/) | ||||||
|  |  | ||||||
| - [View demo of Home Assistant](https://demo.home-assistant.io/) | - [View demo of Home Assistant](https://demo.home-assistant.io/) | ||||||
| - [More information about Home Assistant](https://home-assistant.io) | - [More information about Home Assistant](https://home-assistant.io) | ||||||
| @@ -27,5 +27,3 @@ A complete guide can be found at the following [link](https://www.home-assistant | |||||||
| Home Assistant is open-source and Apache 2 licensed. Feel free to browse the repository, learn and reuse parts in your own projects. | Home Assistant is open-source and Apache 2 licensed. Feel free to browse the repository, learn and reuse parts in your own projects. | ||||||
|  |  | ||||||
| We use [BrowserStack](https://www.browserstack.com) to test Home Assistant on a large variety of devices. | We use [BrowserStack](https://www.browserstack.com) to test Home Assistant on a large variety of devices. | ||||||
|  |  | ||||||
| [](https://www.openhomefoundation.org/) |  | ||||||
|   | |||||||
							
								
								
									
										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", |   "extends": "../.eslintrc.json", | ||||||
|   "rules": { |   "rules": { | ||||||
|     "no-console": "off", |     "import/no-extraneous-dependencies": 0, | ||||||
|     "import/no-extraneous-dependencies": "off", |     "global-require": 0 | ||||||
|     "import/extensions": "off", |  | ||||||
|     "import/no-dynamic-require": "off", |  | ||||||
|     "global-require": "off", |  | ||||||
|     "@typescript-eslint/no-var-requires": "off", |  | ||||||
|     "prefer-arrow-callback": "off" |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,182 +0,0 @@ | |||||||
| import defineProvider from "@babel/helper-define-polyfill-provider"; |  | ||||||
| import { join } from "node:path"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
|  |  | ||||||
| const POLYFILL_DIR = join(paths.polymer_dir, "src/resources/polyfills"); |  | ||||||
|  |  | ||||||
| // List of polyfill keys with supported browser targets for the functionality |  | ||||||
| const PolyfillSupport = { |  | ||||||
|   // Note states and shadowRoot properties should be supported. |  | ||||||
|   "element-internals": { |  | ||||||
|     android: 90, |  | ||||||
|     chrome: 90, |  | ||||||
|     edge: 90, |  | ||||||
|     firefox: 126, |  | ||||||
|     ios: 17.4, |  | ||||||
|     opera: 76, |  | ||||||
|     opera_mobile: 64, |  | ||||||
|     safari: 17.4, |  | ||||||
|     samsung: 15.0, |  | ||||||
|   }, |  | ||||||
|   "element-append": { |  | ||||||
|     android: 54, |  | ||||||
|     chrome: 54, |  | ||||||
|     edge: 17, |  | ||||||
|     firefox: 49, |  | ||||||
|     ios: 10.0, |  | ||||||
|     opera: 41, |  | ||||||
|     opera_mobile: 41, |  | ||||||
|     safari: 10.0, |  | ||||||
|     samsung: 6.0, |  | ||||||
|   }, |  | ||||||
|   "element-getattributenames": { |  | ||||||
|     android: 61, |  | ||||||
|     chrome: 61, |  | ||||||
|     edge: 18, |  | ||||||
|     firefox: 45, |  | ||||||
|     ios: 10.3, |  | ||||||
|     opera: 48, |  | ||||||
|     opera_mobile: 45, |  | ||||||
|     safari: 10.1, |  | ||||||
|     samsung: 8.0, |  | ||||||
|   }, |  | ||||||
|   "element-toggleattribute": { |  | ||||||
|     android: 69, |  | ||||||
|     chrome: 69, |  | ||||||
|     edge: 18, |  | ||||||
|     firefox: 63, |  | ||||||
|     ios: 12.0, |  | ||||||
|     opera: 56, |  | ||||||
|     opera_mobile: 48, |  | ||||||
|     safari: 12.0, |  | ||||||
|     samsung: 10.0, |  | ||||||
|   }, |  | ||||||
|   fetch: { |  | ||||||
|     android: 42, |  | ||||||
|     chrome: 42, |  | ||||||
|     edge: 14, |  | ||||||
|     firefox: 39, |  | ||||||
|     ios: 10.3, |  | ||||||
|     opera: 29, |  | ||||||
|     opera_mobile: 29, |  | ||||||
|     safari: 10.1, |  | ||||||
|     samsung: 4.0, |  | ||||||
|   }, |  | ||||||
|   "intl-getcanonicallocales": { |  | ||||||
|     android: 54, |  | ||||||
|     chrome: 54, |  | ||||||
|     edge: 16, |  | ||||||
|     firefox: 48, |  | ||||||
|     ios: 10.3, |  | ||||||
|     opera: 41, |  | ||||||
|     opera_mobile: 41, |  | ||||||
|     safari: 10.1, |  | ||||||
|     samsung: 6.0, |  | ||||||
|   }, |  | ||||||
|   "intl-locale": { |  | ||||||
|     android: 74, |  | ||||||
|     chrome: 74, |  | ||||||
|     edge: 79, |  | ||||||
|     firefox: 75, |  | ||||||
|     ios: 14.0, |  | ||||||
|     opera: 62, |  | ||||||
|     opera_mobile: 53, |  | ||||||
|     safari: 14.0, |  | ||||||
|     samsung: 11.0, |  | ||||||
|   }, |  | ||||||
|   "intl-other": { |  | ||||||
|     // Not specified (i.e. always try polyfill) since compatibility depends on supported locales |  | ||||||
|   }, |  | ||||||
|   proxy: { |  | ||||||
|     android: 49, |  | ||||||
|     chrome: 49, |  | ||||||
|     edge: 12, |  | ||||||
|     firefox: 18, |  | ||||||
|     ios: 10.0, |  | ||||||
|     opera: 36, |  | ||||||
|     opera_mobile: 36, |  | ||||||
|     safari: 10.0, |  | ||||||
|     samsung: 5.0, |  | ||||||
|   }, |  | ||||||
|   "resize-observer": { |  | ||||||
|     android: 64, |  | ||||||
|     chrome: 64, |  | ||||||
|     edge: 79, |  | ||||||
|     firefox: 69, |  | ||||||
|     ios: 13.4, |  | ||||||
|     opera: 51, |  | ||||||
|     opera_mobile: 47, |  | ||||||
|     safari: 13.1, |  | ||||||
|     samsung: 9.0, |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Map of global variables and/or instance and static properties to the |  | ||||||
| // corresponding polyfill key and actual module to import |  | ||||||
| const polyfillMap = { |  | ||||||
|   global: { |  | ||||||
|     fetch: { key: "fetch", module: "unfetch/polyfill" }, |  | ||||||
|     Proxy: { key: "proxy", module: "proxy-polyfill" }, |  | ||||||
|     ResizeObserver: { |  | ||||||
|       key: "resize-observer", |  | ||||||
|       module: join(POLYFILL_DIR, "resize-observer.ts"), |  | ||||||
|     }, |  | ||||||
|   }, |  | ||||||
|   instance: { |  | ||||||
|     attachInternals: { |  | ||||||
|       key: "element-internals", |  | ||||||
|       module: "element-internals-polyfill", |  | ||||||
|     }, |  | ||||||
|     ...Object.fromEntries( |  | ||||||
|       ["append", "getAttributeNames", "toggleAttribute"].map((prop) => { |  | ||||||
|         const key = `element-${prop.toLowerCase()}`; |  | ||||||
|         return [prop, { key, module: join(POLYFILL_DIR, `${key}.ts`) }]; |  | ||||||
|       }) |  | ||||||
|     ), |  | ||||||
|   }, |  | ||||||
|   static: { |  | ||||||
|     Intl: { |  | ||||||
|       getCanonicalLocales: { |  | ||||||
|         key: "intl-getcanonicallocales", |  | ||||||
|         module: join(POLYFILL_DIR, "intl-polyfill.ts"), |  | ||||||
|       }, |  | ||||||
|       Locale: { |  | ||||||
|         key: "intl-locale", |  | ||||||
|         module: join(POLYFILL_DIR, "intl-polyfill.ts"), |  | ||||||
|       }, |  | ||||||
|       ...Object.fromEntries( |  | ||||||
|         [ |  | ||||||
|           "DateTimeFormat", |  | ||||||
|           "DisplayNames", |  | ||||||
|           "ListFormat", |  | ||||||
|           "NumberFormat", |  | ||||||
|           "PluralRules", |  | ||||||
|           "RelativeTimeFormat", |  | ||||||
|         ].map((obj) => [ |  | ||||||
|           obj, |  | ||||||
|           { key: "intl-other", module: join(POLYFILL_DIR, "intl-polyfill.ts") }, |  | ||||||
|         ]) |  | ||||||
|       ), |  | ||||||
|     }, |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Create plugin using the same factory as for CoreJS |  | ||||||
| export default defineProvider( |  | ||||||
|   ({ createMetaResolver, debug, shouldInjectPolyfill }) => { |  | ||||||
|     const resolvePolyfill = createMetaResolver(polyfillMap); |  | ||||||
|     return { |  | ||||||
|       name: "custom-polyfill", |  | ||||||
|       polyfills: PolyfillSupport, |  | ||||||
|       usageGlobal(meta, utils) { |  | ||||||
|         const polyfill = resolvePolyfill(meta); |  | ||||||
|         if (polyfill && shouldInjectPolyfill(polyfill.desc.key)) { |  | ||||||
|           debug(polyfill.desc.key); |  | ||||||
|           utils.injectGlobalImport(polyfill.desc.module); |  | ||||||
|           return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|   } |  | ||||||
| ); |  | ||||||
| @@ -1,168 +0,0 @@ | |||||||
| const path = require("path"); |  | ||||||
|  |  | ||||||
| // Currently only supports CommonJS modules, as require is synchronous. `import` would need babel running asynchronous. |  | ||||||
| module.exports = function inlineConstants(babel, options, cwd) { |  | ||||||
|   const t = babel.types; |  | ||||||
|  |  | ||||||
|   if (!Array.isArray(options.modules)) { |  | ||||||
|     throw new TypeError( |  | ||||||
|       "babel-plugin-inline-constants: expected a `modules` array to be passed" |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (options.resolveExtensions && !Array.isArray(options.resolveExtensions)) { |  | ||||||
|     throw new TypeError( |  | ||||||
|       "babel-plugin-inline-constants: expected `resolveExtensions` to be an array" |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   const ignoreModuleNotFound = options.ignoreModuleNotFound; |  | ||||||
|   const resolveExtensions = options.resolveExtensions; |  | ||||||
|  |  | ||||||
|   const hasRelativeModules = options.modules.some( |  | ||||||
|     (module) => module.startsWith(".") || module.startsWith("/") |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const modules = Object.fromEntries( |  | ||||||
|     options.modules.map((module) => { |  | ||||||
|       const absolute = module.startsWith(".") |  | ||||||
|         ? require.resolve(module, { paths: [cwd] }) |  | ||||||
|         : module; |  | ||||||
|       return [absolute, require(absolute)]; |  | ||||||
|     }) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   const toLiteral = (value) => { |  | ||||||
|     if (typeof value === "string") { |  | ||||||
|       return t.stringLiteral(value); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (typeof value === "number") { |  | ||||||
|       return t.numericLiteral(value); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (typeof value === "boolean") { |  | ||||||
|       return t.booleanLiteral(value); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (value === null) { |  | ||||||
|       return t.nullLiteral(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     throw new Error( |  | ||||||
|       "babel-plugin-inline-constants: cannot handle non-literal `" + value + "`" |  | ||||||
|     ); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   const resolveAbsolute = (value, state, resolveExtensionIndex) => { |  | ||||||
|     if (!state.filename) { |  | ||||||
|       throw new TypeError( |  | ||||||
|         "babel-plugin-inline-constants: expected a `filename` to be set for files" |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (resolveExtensions && resolveExtensionIndex !== undefined) { |  | ||||||
|       value += resolveExtensions[resolveExtensionIndex]; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     try { |  | ||||||
|       return require.resolve(value, { paths: [path.dirname(state.filename)] }); |  | ||||||
|     } catch (error) { |  | ||||||
|       if ( |  | ||||||
|         error.code === "MODULE_NOT_FOUND" && |  | ||||||
|         resolveExtensions && |  | ||||||
|         (resolveExtensionIndex === undefined || |  | ||||||
|           resolveExtensionIndex < resolveExtensions.length - 1) |  | ||||||
|       ) { |  | ||||||
|         const resolveExtensionIdx = (resolveExtensionIndex || -1) + 1; |  | ||||||
|         return resolveAbsolute(value, state, resolveExtensionIdx); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (error.code === "MODULE_NOT_FOUND" && ignoreModuleNotFound) { |  | ||||||
|         return undefined; |  | ||||||
|       } |  | ||||||
|       throw error; |  | ||||||
|     } |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   const importDeclaration = (p, state) => { |  | ||||||
|     if (p.node.type !== "ImportDeclaration") { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     const absolute = |  | ||||||
|       hasRelativeModules && p.node.source.value.startsWith(".") |  | ||||||
|         ? resolveAbsolute(p.node.source.value, state) |  | ||||||
|         : p.node.source.value; |  | ||||||
|  |  | ||||||
|     if (!absolute || !(absolute in modules)) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const module = modules[absolute]; |  | ||||||
|  |  | ||||||
|     for (const specifier of p.node.specifiers) { |  | ||||||
|       if ( |  | ||||||
|         specifier.type === "ImportDefaultSpecifier" && |  | ||||||
|         specifier.local && |  | ||||||
|         specifier.local.type === "Identifier" |  | ||||||
|       ) { |  | ||||||
|         if (!("default" in module)) { |  | ||||||
|           throw new Error( |  | ||||||
|             "babel-plugin-inline-constants: cannot access default export from `" + |  | ||||||
|               p.node.source.value + |  | ||||||
|               "`" |  | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const variableValue = toLiteral(module.default); |  | ||||||
|         const variable = t.variableDeclarator( |  | ||||||
|           t.identifier(specifier.local.name), |  | ||||||
|           variableValue |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         p.insertBefore({ |  | ||||||
|           type: "VariableDeclaration", |  | ||||||
|           kind: "const", |  | ||||||
|           declarations: [variable], |  | ||||||
|         }); |  | ||||||
|       } else if ( |  | ||||||
|         specifier.type === "ImportSpecifier" && |  | ||||||
|         specifier.imported && |  | ||||||
|         specifier.imported.type === "Identifier" && |  | ||||||
|         specifier.local && |  | ||||||
|         specifier.local.type === "Identifier" |  | ||||||
|       ) { |  | ||||||
|         if (!(specifier.imported.name in module)) { |  | ||||||
|           throw new Error( |  | ||||||
|             "babel-plugin-inline-constants: cannot access `" + |  | ||||||
|               specifier.imported.name + |  | ||||||
|               "` from `" + |  | ||||||
|               p.node.source.value + |  | ||||||
|               "`" |  | ||||||
|           ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const variableValue = toLiteral(module[specifier.imported.name]); |  | ||||||
|         const variable = t.variableDeclarator( |  | ||||||
|           t.identifier(specifier.local.name), |  | ||||||
|           variableValue |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         p.insertBefore({ |  | ||||||
|           type: "VariableDeclaration", |  | ||||||
|           kind: "const", |  | ||||||
|           declarations: [variable], |  | ||||||
|         }); |  | ||||||
|       } else { |  | ||||||
|         throw new Error("Cannot handle specifier `" + specifier.type + "`"); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     p.remove(); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   return { |  | ||||||
|     visitor: { |  | ||||||
|       ImportDeclaration: importDeclaration, |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
| @@ -1,331 +0,0 @@ | |||||||
| const path = require("path"); |  | ||||||
| const env = require("./env.cjs"); |  | ||||||
| const paths = require("./paths.cjs"); |  | ||||||
| const { dependencies } = require("../package.json"); |  | ||||||
|  |  | ||||||
| const BABEL_PLUGINS = path.join(__dirname, "babel-plugins"); |  | ||||||
|  |  | ||||||
| // GitHub base URL to use for production source maps |  | ||||||
| // Nightly builds use the commit SHA, otherwise assumes there is a tag that matches the version |  | ||||||
| module.exports.sourceMapURL = () => { |  | ||||||
|   const ref = env.version().endsWith("dev") |  | ||||||
|     ? process.env.GITHUB_SHA || "dev" |  | ||||||
|     : env.version(); |  | ||||||
|   return `https://raw.githubusercontent.com/home-assistant/frontend/${ref}/`; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Files from NPM Packages that should not be imported |  | ||||||
| module.exports.ignorePackages = () => []; |  | ||||||
|  |  | ||||||
| // Files from NPM packages that we should replace with empty file |  | ||||||
| module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) => |  | ||||||
|   [ |  | ||||||
|     // Contains all color definitions for all material color sets. |  | ||||||
|     // We don't use it |  | ||||||
|     require.resolve("@polymer/paper-styles/color.js"), |  | ||||||
|     require.resolve("@polymer/paper-styles/default-theme.js"), |  | ||||||
|     // Loads stuff from a CDN |  | ||||||
|     require.resolve("@polymer/font-roboto/roboto.js"), |  | ||||||
|     require.resolve("@vaadin/vaadin-material-styles/typography.js"), |  | ||||||
|     require.resolve("@vaadin/vaadin-material-styles/font-icons.js"), |  | ||||||
|     // Compatibility not needed for latest builds |  | ||||||
|     latestBuild && |  | ||||||
|       // wrapped in require.resolve so it blows up if file no longer exists |  | ||||||
|       require.resolve( |  | ||||||
|         path.resolve(paths.polymer_dir, "src/resources/compatibility.ts") |  | ||||||
|       ), |  | ||||||
|     // Icons in supervisor conflict with icons in HA so we don't load. |  | ||||||
|     isHassioBuild && |  | ||||||
|       require.resolve( |  | ||||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon.ts") |  | ||||||
|       ), |  | ||||||
|     isHassioBuild && |  | ||||||
|       require.resolve( |  | ||||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon-picker.ts") |  | ||||||
|       ), |  | ||||||
|   ].filter(Boolean); |  | ||||||
|  |  | ||||||
| module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ |  | ||||||
|   __DEV__: !isProdBuild, |  | ||||||
|   __BUILD__: JSON.stringify(latestBuild ? "modern" : "legacy"), |  | ||||||
|   __VERSION__: JSON.stringify(env.version()), |  | ||||||
|   __DEMO__: false, |  | ||||||
|   __SUPERVISOR__: false, |  | ||||||
|   __BACKWARDS_COMPAT__: false, |  | ||||||
|   __STATIC_PATH__: "/static/", |  | ||||||
|   "process.env.NODE_ENV": JSON.stringify( |  | ||||||
|     isProdBuild ? "production" : "development" |  | ||||||
|   ), |  | ||||||
|   ...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 }) => ({ |  | ||||||
|   safari10: !latestBuild, |  | ||||||
|   ecma: latestBuild ? 2015 : 5, |  | ||||||
|   module: latestBuild, |  | ||||||
|   format: { comments: false }, |  | ||||||
|   sourceMap: !isTestBuild, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| module.exports.babelOptions = ({ |  | ||||||
|   latestBuild, |  | ||||||
|   isProdBuild, |  | ||||||
|   isTestBuild, |  | ||||||
|   sw, |  | ||||||
| }) => ({ |  | ||||||
|   babelrc: false, |  | ||||||
|   compact: false, |  | ||||||
|   assumptions: { |  | ||||||
|     privateFieldsAsProperties: true, |  | ||||||
|     setPublicClassFields: true, |  | ||||||
|     setSpreadProperties: true, |  | ||||||
|   }, |  | ||||||
|   browserslistEnv: latestBuild ? "modern" : `legacy${sw ? "-sw" : ""}`, |  | ||||||
|   presets: [ |  | ||||||
|     [ |  | ||||||
|       "@babel/preset-env", |  | ||||||
|       { |  | ||||||
|         useBuiltIns: "usage", |  | ||||||
|         corejs: dependencies["core-js"], |  | ||||||
|         bugfixes: true, |  | ||||||
|         shippedProposals: true, |  | ||||||
|       }, |  | ||||||
|     ], |  | ||||||
|     "@babel/preset-typescript", |  | ||||||
|   ], |  | ||||||
|   plugins: [ |  | ||||||
|     [ |  | ||||||
|       path.join(BABEL_PLUGINS, "inline-constants-plugin.cjs"), |  | ||||||
|       { |  | ||||||
|         modules: ["@mdi/js"], |  | ||||||
|         ignoreModuleNotFound: true, |  | ||||||
|       }, |  | ||||||
|     ], |  | ||||||
|     // Minify template literals for production |  | ||||||
|     isProdBuild && [ |  | ||||||
|       "template-html-minifier", |  | ||||||
|       { |  | ||||||
|         modules: { |  | ||||||
|           ...Object.fromEntries( |  | ||||||
|             ["lit", "lit-element", "lit-html"].map((m) => [ |  | ||||||
|               m, |  | ||||||
|               [ |  | ||||||
|                 "html", |  | ||||||
|                 { name: "svg", encapsulation: "svg" }, |  | ||||||
|                 { name: "css", encapsulation: "style" }, |  | ||||||
|               ], |  | ||||||
|             ]) |  | ||||||
|           ), |  | ||||||
|           "@polymer/polymer/lib/utils/html-tag.js": ["html"], |  | ||||||
|         }, |  | ||||||
|         strictCSS: true, |  | ||||||
|         htmlMinifier: module.exports.htmlMinifierOptions, |  | ||||||
|         failOnError: false, // we can turn this off in case of false positives |  | ||||||
|       }, |  | ||||||
|     ], |  | ||||||
|     // Import helpers and regenerator from runtime package |  | ||||||
|     [ |  | ||||||
|       "@babel/plugin-transform-runtime", |  | ||||||
|       { version: dependencies["@babel/runtime"] }, |  | ||||||
|     ], |  | ||||||
|     // Transpile decorators (still in TC39 process) |  | ||||||
|     // Modern browsers support class fields and private methods, but transform is required with the older decorator version dictated by Lit |  | ||||||
|     [ |  | ||||||
|       "@babel/plugin-proposal-decorators", |  | ||||||
|       { version: "2018-09", decoratorsBeforeExport: true }, |  | ||||||
|     ], |  | ||||||
|     "@babel/plugin-transform-class-properties", |  | ||||||
|     "@babel/plugin-transform-private-methods", |  | ||||||
|   ].filter(Boolean), |  | ||||||
|   exclude: [ |  | ||||||
|     // \\ for Windows, / for Mac OS and Linux |  | ||||||
|     /node_modules[\\/]core-js/, |  | ||||||
|     /node_modules[\\/]webpack[\\/]buildin/, |  | ||||||
|   ], |  | ||||||
|   sourceMaps: !isTestBuild, |  | ||||||
|   overrides: [ |  | ||||||
|     { |  | ||||||
|       // Add plugin to inject various polyfills, excluding the polyfills |  | ||||||
|       // themselves to prevent self-injection. |  | ||||||
|       plugins: [ |  | ||||||
|         [ |  | ||||||
|           path.join(BABEL_PLUGINS, "custom-polyfill-plugin.js"), |  | ||||||
|           { method: "usage-global" }, |  | ||||||
|         ], |  | ||||||
|       ], |  | ||||||
|       exclude: [ |  | ||||||
|         path.join(paths.polymer_dir, "src/resources/polyfills"), |  | ||||||
|         ...[ |  | ||||||
|           "@formatjs/(?:ecma402-abstract|intl-\\w+)", |  | ||||||
|           "@lit-labs/virtualizer/polyfills", |  | ||||||
|           "@webcomponents/scoped-custom-element-registry", |  | ||||||
|           "element-internals-polyfill", |  | ||||||
|           "proxy-polyfill", |  | ||||||
|           "unfetch", |  | ||||||
|         ].map((p) => new RegExp(`/node_modules/${p}/`)), |  | ||||||
|       ], |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|       // Use unambiguous for dependencies so that require() is correctly injected into CommonJS files |  | ||||||
|       // Exclusions are needed in some cases where ES modules have no static imports or exports, such as polyfills |  | ||||||
|       sourceType: "unambiguous", |  | ||||||
|       include: /\/node_modules\//, |  | ||||||
|       exclude: [ |  | ||||||
|         "element-internals-polyfill", |  | ||||||
|         "@?lit(?:-labs|-element|-html)?", |  | ||||||
|       ].map((p) => new RegExp(`/node_modules/${p}/`)), |  | ||||||
|     }, |  | ||||||
|   ], |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const nameSuffix = (latestBuild) => (latestBuild ? "-modern" : "-legacy"); |  | ||||||
|  |  | ||||||
| const outputPath = (outputRoot, latestBuild) => |  | ||||||
|   path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5"); |  | ||||||
|  |  | ||||||
| const publicPath = (latestBuild, root = "") => |  | ||||||
|   latestBuild ? `${root}/frontend_latest/` : `${root}/frontend_es5/`; |  | ||||||
|  |  | ||||||
| /* |  | ||||||
|   BundleConfig { |  | ||||||
|     // Object with entrypoints that need to be bundled |  | ||||||
|     entry: { [name: string]: pathToFile }, |  | ||||||
|     // Folder where bundled files need to be written |  | ||||||
|     outputPath: string, |  | ||||||
|     // absolute url-path where bundled files can be found |  | ||||||
|     publicPath: string, |  | ||||||
|     // extra definitions that we need to replace in source |  | ||||||
|     defineOverlay: {[name: string]: value }, |  | ||||||
|     // if this is a production build |  | ||||||
|     isProdBuild: boolean, |  | ||||||
|     // If we're targeting latest browsers |  | ||||||
|     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 }) { |  | ||||||
|     return { |  | ||||||
|       name: "frontend" + nameSuffix(latestBuild), |  | ||||||
|       entry: { |  | ||||||
|         "service-worker": |  | ||||||
|           !env.useRollup() && !latestBuild |  | ||||||
|             ? { |  | ||||||
|                 import: "./src/entrypoints/service-worker.ts", |  | ||||||
|                 layer: "sw", |  | ||||||
|               } |  | ||||||
|             : "./src/entrypoints/service-worker.ts", |  | ||||||
|         app: "./src/entrypoints/app.ts", |  | ||||||
|         authorize: "./src/entrypoints/authorize.ts", |  | ||||||
|         onboarding: "./src/entrypoints/onboarding.ts", |  | ||||||
|         core: "./src/entrypoints/core.ts", |  | ||||||
|         "custom-panel": "./src/entrypoints/custom-panel.ts", |  | ||||||
|       }, |  | ||||||
|       outputPath: outputPath(paths.app_output_root, latestBuild), |  | ||||||
|       publicPath: publicPath(latestBuild), |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       isStatsBuild, |  | ||||||
|       isTestBuild, |  | ||||||
|       isWDS, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   demo({ isProdBuild, latestBuild, isStatsBuild }) { |  | ||||||
|     return { |  | ||||||
|       name: "demo" + nameSuffix(latestBuild), |  | ||||||
|       entry: { |  | ||||||
|         main: path.resolve(paths.demo_dir, "src/entrypoint.ts"), |  | ||||||
|       }, |  | ||||||
|       outputPath: outputPath(paths.demo_output_root, latestBuild), |  | ||||||
|       publicPath: publicPath(latestBuild), |  | ||||||
|       defineOverlay: { |  | ||||||
|         __VERSION__: JSON.stringify(`DEMO-${env.version()}`), |  | ||||||
|         __DEMO__: true, |  | ||||||
|       }, |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       isStatsBuild, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   cast({ isProdBuild, latestBuild }) { |  | ||||||
|     const entry = { |  | ||||||
|       launcher: path.resolve(paths.cast_dir, "src/launcher/entrypoint.ts"), |  | ||||||
|       media: path.resolve(paths.cast_dir, "src/media/entrypoint.ts"), |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     if (latestBuild) { |  | ||||||
|       entry.receiver = path.resolve( |  | ||||||
|         paths.cast_dir, |  | ||||||
|         "src/receiver/entrypoint.ts" |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|       name: "cast" + nameSuffix(latestBuild), |  | ||||||
|       entry, |  | ||||||
|       outputPath: outputPath(paths.cast_output_root, latestBuild), |  | ||||||
|       publicPath: publicPath(latestBuild), |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       defineOverlay: { |  | ||||||
|         __BACKWARDS_COMPAT__: true, |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   hassio({ isProdBuild, latestBuild, isStatsBuild, isTestBuild }) { |  | ||||||
|     return { |  | ||||||
|       name: "supervisor" + nameSuffix(latestBuild), |  | ||||||
|       entry: { |  | ||||||
|         entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"), |  | ||||||
|       }, |  | ||||||
|       outputPath: outputPath(paths.hassio_output_root, latestBuild), |  | ||||||
|       publicPath: publicPath(latestBuild, paths.hassio_publicPath), |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       isStatsBuild, |  | ||||||
|       isTestBuild, |  | ||||||
|       isHassioBuild: true, |  | ||||||
|       defineOverlay: { |  | ||||||
|         __SUPERVISOR__: true, |  | ||||||
|         __STATIC_PATH__: `"${paths.hassio_publicPath}/static/"`, |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
|  |  | ||||||
|   gallery({ isProdBuild, latestBuild }) { |  | ||||||
|     return { |  | ||||||
|       name: "gallery" + nameSuffix(latestBuild), |  | ||||||
|       entry: { |  | ||||||
|         entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"), |  | ||||||
|       }, |  | ||||||
|       outputPath: outputPath(paths.gallery_output_root, latestBuild), |  | ||||||
|       publicPath: publicPath(latestBuild), |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       defineOverlay: { |  | ||||||
|         __DEMO__: true, |  | ||||||
|       }, |  | ||||||
|     }; |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
							
								
								
									
										203
									
								
								build-scripts/bundle.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								build-scripts/bundle.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,203 @@ | |||||||
|  | const path = require("path"); | ||||||
|  | const env = require("./env.js"); | ||||||
|  | const paths = require("./paths.js"); | ||||||
|  |  | ||||||
|  | // Files from NPM Packages that should not be imported | ||||||
|  | module.exports.ignorePackages = ({ latestBuild }) => [ | ||||||
|  |   // Bloats bundle and it's not used. | ||||||
|  |   path.resolve(require.resolve("moment"), "../locale"), | ||||||
|  |   // Part of yaml.js and only used for !!js functions that we don't use | ||||||
|  |   require.resolve("esprima"), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | // Files from NPM packages that we should replace with empty file | ||||||
|  | module.exports.emptyPackages = ({ latestBuild }) => | ||||||
|  |   [ | ||||||
|  |     // Contains all color definitions for all material color sets. | ||||||
|  |     // We don't use it | ||||||
|  |     require.resolve("@polymer/paper-styles/color.js"), | ||||||
|  |     require.resolve("@polymer/paper-styles/default-theme.js"), | ||||||
|  |     // Loads stuff from a CDN | ||||||
|  |     require.resolve("@polymer/font-roboto/roboto.js"), | ||||||
|  |     require.resolve("@vaadin/vaadin-material-styles/font-roboto.js"), | ||||||
|  |     // Compatibility not needed for latest builds | ||||||
|  |     latestBuild && | ||||||
|  |       // wrapped in require.resolve so it blows up if file no longer exists | ||||||
|  |       require.resolve( | ||||||
|  |         path.resolve(paths.polymer_dir, "src/resources/compatibility.ts") | ||||||
|  |       ), | ||||||
|  |     // This polyfill is loaded in workers to support ES5, filter it out. | ||||||
|  |     latestBuild && require.resolve("proxy-polyfill/src/index.js"), | ||||||
|  |   ].filter(Boolean); | ||||||
|  |  | ||||||
|  | module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||||
|  |   __DEV__: !isProdBuild, | ||||||
|  |   __BUILD__: JSON.stringify(latestBuild ? "latest" : "es5"), | ||||||
|  |   __VERSION__: JSON.stringify(env.version()), | ||||||
|  |   __DEMO__: false, | ||||||
|  |   __BACKWARDS_COMPAT__: false, | ||||||
|  |   __STATIC_PATH__: "/static/", | ||||||
|  |   "process.env.NODE_ENV": JSON.stringify( | ||||||
|  |     isProdBuild ? "production" : "development" | ||||||
|  |   ), | ||||||
|  |   ...defineOverlay, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports.terserOptions = (latestBuild) => ({ | ||||||
|  |   safari10: !latestBuild, | ||||||
|  |   ecma: latestBuild ? undefined : 5, | ||||||
|  |   output: { comments: false }, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports.babelOptions = ({ latestBuild }) => ({ | ||||||
|  |   babelrc: false, | ||||||
|  |   presets: [ | ||||||
|  |     !latestBuild && [ | ||||||
|  |       require("@babel/preset-env").default, | ||||||
|  |       { | ||||||
|  |         useBuiltIns: "entry", | ||||||
|  |         corejs: "3.6", | ||||||
|  |       }, | ||||||
|  |     ], | ||||||
|  |     require("@babel/preset-typescript").default, | ||||||
|  |   ].filter(Boolean), | ||||||
|  |   plugins: [ | ||||||
|  |     // Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2}) | ||||||
|  |     !latestBuild && [ | ||||||
|  |       "@babel/plugin-proposal-object-rest-spread", | ||||||
|  |       { loose: true, useBuiltIns: true }, | ||||||
|  |     ], | ||||||
|  |     // Only support the syntax, Webpack will handle it. | ||||||
|  |     "@babel/plugin-syntax-import-meta", | ||||||
|  |     "@babel/plugin-syntax-dynamic-import", | ||||||
|  |     "@babel/plugin-proposal-optional-chaining", | ||||||
|  |     "@babel/plugin-proposal-nullish-coalescing-operator", | ||||||
|  |     [ | ||||||
|  |       require("@babel/plugin-proposal-decorators").default, | ||||||
|  |       { decoratorsBeforeExport: true }, | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       require("@babel/plugin-proposal-class-properties").default, | ||||||
|  |       { loose: true }, | ||||||
|  |     ], | ||||||
|  |   ].filter(Boolean), | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Are already ES5, cause warnings when babelified. | ||||||
|  | module.exports.babelExclude = () => [ | ||||||
|  |   require.resolve("@mdi/js/mdi.js"), | ||||||
|  |   require.resolve("hls.js"), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | const outputPath = (outputRoot, latestBuild) => | ||||||
|  |   path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5"); | ||||||
|  |  | ||||||
|  | const publicPath = (latestBuild, root = "") => | ||||||
|  |   latestBuild ? `${root}/frontend_latest/` : `${root}/frontend_es5/`; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | BundleConfig { | ||||||
|  |   // Object with entrypoints that need to be bundled | ||||||
|  |   entry: { [name: string]: pathToFile }, | ||||||
|  |   // Folder where bundled files need to be written | ||||||
|  |   outputPath: string, | ||||||
|  |   // absolute url-path where bundled files can be found | ||||||
|  |   publicPath: string, | ||||||
|  |   // extra definitions that we need to replace in source | ||||||
|  |   defineOverlay: {[name: string]: value }, | ||||||
|  |   // if this is a production build | ||||||
|  |   isProdBuild: boolean, | ||||||
|  |   // If we're targeting latest browsers | ||||||
|  |   latestBuild: boolean, | ||||||
|  |   // If we're doing a stats build (create nice chunk names) | ||||||
|  |   isStatsBuild: boolean, | ||||||
|  |   // Names of entrypoints that should not be hashed | ||||||
|  |   dontHash: Set<string> | ||||||
|  | } | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | module.exports.config = { | ||||||
|  |   app({ isProdBuild, latestBuild, isStatsBuild, isWDS }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         service_worker: "./src/entrypoints/service_worker.ts", | ||||||
|  |         app: "./src/entrypoints/app.ts", | ||||||
|  |         authorize: "./src/entrypoints/authorize.ts", | ||||||
|  |         onboarding: "./src/entrypoints/onboarding.ts", | ||||||
|  |         core: "./src/entrypoints/core.ts", | ||||||
|  |         "custom-panel": "./src/entrypoints/custom-panel.ts", | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.app_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |       isWDS, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   demo({ isProdBuild, latestBuild, isStatsBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         main: path.resolve(paths.demo_dir, "src/entrypoint.ts"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.demo_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       defineOverlay: { | ||||||
|  |         __VERSION__: JSON.stringify(`DEMO-${env.version()}`), | ||||||
|  |         __DEMO__: true, | ||||||
|  |       }, | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   cast({ isProdBuild, latestBuild }) { | ||||||
|  |     const entry = { | ||||||
|  |       launcher: path.resolve(paths.cast_dir, "src/launcher/entrypoint.ts"), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (latestBuild) { | ||||||
|  |       entry.receiver = path.resolve( | ||||||
|  |         paths.cast_dir, | ||||||
|  |         "src/receiver/entrypoint.ts" | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return { | ||||||
|  |       entry, | ||||||
|  |       outputPath: outputPath(paths.cast_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       defineOverlay: { | ||||||
|  |         __BACKWARDS_COMPAT__: true, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   hassio({ isProdBuild, latestBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.hassio_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild, paths.hassio_publicPath), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   gallery({ isProdBuild, latestBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.gallery_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -1,6 +1,6 @@ | |||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const paths = require("./paths.cjs"); | const paths = require("./paths.js"); | ||||||
| 
 | 
 | ||||||
| module.exports = { | module.exports = { | ||||||
|   useRollup() { |   useRollup() { | ||||||
| @@ -17,7 +17,7 @@ module.exports = { | |||||||
|   isStatsBuild() { |   isStatsBuild() { | ||||||
|     return process.env.STATS === "1"; |     return process.env.STATS === "1"; | ||||||
|   }, |   }, | ||||||
|   isTestBuild() { |   isTest() { | ||||||
|     return process.env.IS_TEST === "true"; |     return process.env.IS_TEST === "true"; | ||||||
|   }, |   }, | ||||||
|   isNetlify() { |   isNetlify() { | ||||||
| @@ -25,14 +25,11 @@ module.exports = { | |||||||
|   }, |   }, | ||||||
|   version() { |   version() { | ||||||
|     const version = fs |     const version = fs | ||||||
|       .readFileSync(path.resolve(paths.polymer_dir, "pyproject.toml"), "utf8") |       .readFileSync(path.resolve(paths.polymer_dir, "setup.py"), "utf8") | ||||||
|       .match(/version\W+=\W"(\d{8}\.\d(?:\.dev)?)"/); |       .match(/\d{8}\.\d+/); | ||||||
|     if (!version) { |     if (!version) { | ||||||
|       throw Error("Version not found"); |       throw Error("Version not found"); | ||||||
|     } |     } | ||||||
|     return version[1]; |     return version[0]; | ||||||
|   }, |  | ||||||
|   isDevContainer() { |  | ||||||
|     return process.env.DEV_CONTAINER === "1"; |  | ||||||
|   }, |   }, | ||||||
| }; | }; | ||||||
| @@ -1,16 +1,18 @@ | |||||||
| import gulp from "gulp"; | // Run HA develop mode | ||||||
| import env from "../env.cjs"; | const gulp = require("gulp"); | ||||||
| import "./clean.js"; |  | ||||||
| import "./compress.js"; | const env = require("../env"); | ||||||
| import "./entry-html.js"; |  | ||||||
| import "./gather-static.js"; | require("./clean.js"); | ||||||
| import "./gen-icons-json.js"; | require("./translations.js"); | ||||||
| import "./locale-data.js"; | require("./gen-icons-json.js"); | ||||||
| import "./rollup.js"; | require("./gather-static.js"); | ||||||
| import "./service-worker.js"; | require("./compress.js"); | ||||||
| import "./translations.js"; | require("./webpack.js"); | ||||||
| import "./wds.js"; | require("./service-worker.js"); | ||||||
| import "./webpack.js"; | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  | require("./wds.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-app", |   "develop-app", | ||||||
| @@ -22,16 +24,16 @@ gulp.task( | |||||||
|     gulp.parallel( |     gulp.parallel( | ||||||
|       "gen-service-worker-app-dev", |       "gen-service-worker-app-dev", | ||||||
|       "gen-icons-json", |       "gen-icons-json", | ||||||
|       "gen-pages-app-dev", |       "gen-pages-dev", | ||||||
|       "build-translations", |       "gen-index-app-dev", | ||||||
|       "build-locale-data" |       "build-translations" | ||||||
|     ), |     ), | ||||||
|     "copy-static-app", |     "copy-static-app", | ||||||
|     env.useWDS() |     env.useWDS() | ||||||
|       ? "wds-watch-app" |       ? "wds-watch-app" | ||||||
|       : env.useRollup() |       : env.useRollup() | ||||||
|         ? "rollup-watch-app" |       ? "rollup-watch-app" | ||||||
|         : "webpack-watch-app" |       : "webpack-watch-app" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -42,11 +44,15 @@ gulp.task( | |||||||
|       process.env.NODE_ENV = "production"; |       process.env.NODE_ENV = "production"; | ||||||
|     }, |     }, | ||||||
|     "clean", |     "clean", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-app", |     "copy-static-app", | ||||||
|     env.useRollup() ? "rollup-prod-app" : "webpack-prod-app", |     env.useRollup() ? "rollup-prod-app" : "webpack-prod-app", | ||||||
|     gulp.parallel("gen-pages-app-prod", "gen-service-worker-app-prod"), |     ...// Don't compress running tests | ||||||
|     // Don't compress running tests |     (env.isTest() ? [] : ["compress-app"]), | ||||||
|     ...(env.isTestBuild() ? [] : ["compress-app"]) |     gulp.parallel( | ||||||
|  |       "gen-pages-prod", | ||||||
|  |       "gen-index-app-prod", | ||||||
|  |       "gen-service-worker-app-prod" | ||||||
|  |     ) | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,12 +1,14 @@ | |||||||
| import gulp from "gulp"; | const gulp = require("gulp"); | ||||||
| import env from "../env.cjs"; |  | ||||||
| import "./clean.js"; | const env = require("../env"); | ||||||
| import "./entry-html.js"; |  | ||||||
| import "./gather-static.js"; | require("./clean.js"); | ||||||
| import "./rollup.js"; | require("./translations.js"); | ||||||
| import "./service-worker.js"; | require("./gather-static.js"); | ||||||
| import "./translations.js"; | require("./webpack.js"); | ||||||
| import "./webpack.js"; | require("./service-worker.js"); | ||||||
|  | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-cast", |   "develop-cast", | ||||||
| @@ -16,9 +18,9 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-cast", |     "clean-cast", | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-cast", |     "copy-static-cast", | ||||||
|     "gen-pages-cast-dev", |     "gen-index-cast-dev", | ||||||
|     env.useRollup() ? "rollup-dev-server-cast" : "webpack-dev-server-cast" |     env.useRollup() ? "rollup-dev-server-cast" : "webpack-dev-server-cast" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
| @@ -31,9 +33,9 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-cast", |     "clean-cast", | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-cast", |     "copy-static-cast", | ||||||
|     env.useRollup() ? "rollup-prod-cast" : "webpack-prod-cast", |     env.useRollup() ? "rollup-prod-cast" : "webpack-prod-cast", | ||||||
|     "gen-pages-cast-prod" |     "gen-index-cast-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,40 +1,36 @@ | |||||||
| import { deleteSync } from "del"; | const del = require("del"); | ||||||
| import gulp from "gulp"; | const gulp = require("gulp"); | ||||||
| import paths from "../paths.cjs"; | const paths = require("../paths"); | ||||||
| import "./translations.js"; | require("./translations"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean", |   "clean", | ||||||
|   gulp.parallel("clean-translations", async () => |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     deleteSync([paths.app_output_root, paths.build_dir]) |     return del([paths.app_output_root, paths.build_dir]); | ||||||
|   ) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-demo", |   "clean-demo", | ||||||
|   gulp.parallel("clean-translations", async () => |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     deleteSync([paths.demo_output_root, paths.build_dir]) |     return del([paths.demo_output_root, paths.build_dir]); | ||||||
|   ) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-cast", |   "clean-cast", | ||||||
|   gulp.parallel("clean-translations", async () => |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     deleteSync([paths.cast_output_root, paths.build_dir]) |     return del([paths.cast_output_root, paths.build_dir]); | ||||||
|   ) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task("clean-hassio", async () => | gulp.task("clean-hassio", function cleanOutputAndBuildDir() { | ||||||
|   deleteSync([paths.hassio_output_root, paths.build_dir]) |   return del([paths.hassio_output_root, paths.build_dir]); | ||||||
| ); | }); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-gallery", |   "clean-gallery", | ||||||
|   gulp.parallel("clean-translations", async () => |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     deleteSync([ |     return del([paths.gallery_output_root, paths.build_dir]); | ||||||
|       paths.gallery_output_root, |   }) | ||||||
|       paths.gallery_build, |  | ||||||
|       paths.build_dir, |  | ||||||
|     ]) |  | ||||||
|   ) |  | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,68 +1,45 @@ | |||||||
| // Tasks to compress | // Tasks to compress | ||||||
|  |  | ||||||
| import { constants } from "node:zlib"; | const gulp = require("gulp"); | ||||||
| import gulp from "gulp"; | const zopfli = require("gulp-zopfli-green"); | ||||||
| import brotli from "gulp-brotli"; | const merge = require("merge-stream"); | ||||||
| import zopfli from "gulp-zopfli-green"; | const path = require("path"); | ||||||
| import paths from "../paths.cjs"; | const paths = require("../paths"); | ||||||
|  |  | ||||||
| const filesGlob = "*.{js,json,css,svg,xml}"; |  | ||||||
| const brotliOptions = { |  | ||||||
|   skipLarger: true, |  | ||||||
|   params: { |  | ||||||
|     [constants.BROTLI_PARAM_QUALITY]: constants.BROTLI_MAX_QUALITY, |  | ||||||
|   }, |  | ||||||
| }; |  | ||||||
| const zopfliOptions = { threshold: 150 }; | const zopfliOptions = { threshold: 150 }; | ||||||
|  |  | ||||||
| const compressDistBrotli = (rootDir, modernDir, compressServiceWorker = true) => | gulp.task("compress-app", function compressApp() { | ||||||
|   gulp |   const jsLatest = gulp | ||||||
|     .src( |     .src(path.resolve(paths.app_output_latest, "**/*.js")) | ||||||
|       [ |  | ||||||
|         `${modernDir}/**/${filesGlob}`, |  | ||||||
|         compressServiceWorker ? `${rootDir}/sw-modern.js` : undefined, |  | ||||||
|       ].filter(Boolean), |  | ||||||
|       { |  | ||||||
|         base: rootDir, |  | ||||||
|       } |  | ||||||
|     ) |  | ||||||
|     .pipe(brotli(brotliOptions)) |  | ||||||
|     .pipe(gulp.dest(rootDir)); |  | ||||||
|  |  | ||||||
| const compressDistZopfli = (rootDir, modernDir, compressModern = false) => |  | ||||||
|   gulp |  | ||||||
|     .src( |  | ||||||
|       [ |  | ||||||
|         `${rootDir}/**/${filesGlob}`, |  | ||||||
|         compressModern ? undefined : `!${modernDir}/**/${filesGlob}`, |  | ||||||
|         `!${rootDir}/{sw-modern,service_worker}.js`, |  | ||||||
|         `${rootDir}/{authorize,onboarding}.html`, |  | ||||||
|       ].filter(Boolean), |  | ||||||
|       { base: rootDir } |  | ||||||
|     ) |  | ||||||
|     .pipe(zopfli(zopfliOptions)) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(rootDir)); |     .pipe(gulp.dest(paths.app_output_latest)); | ||||||
|  |  | ||||||
| const compressAppBrotli = () => |   const jsEs5 = gulp | ||||||
|   compressDistBrotli(paths.app_output_root, paths.app_output_latest); |     .src(path.resolve(paths.app_output_es5, "**/*.js")) | ||||||
| const compressHassioBrotli = () => |     .pipe(zopfli(zopfliOptions)) | ||||||
|   compressDistBrotli( |     .pipe(gulp.dest(paths.app_output_es5)); | ||||||
|     paths.hassio_output_root, |  | ||||||
|     paths.hassio_output_latest, |  | ||||||
|     false |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
| const compressAppZopfli = () => |   const polyfills = gulp | ||||||
|   compressDistZopfli(paths.app_output_root, paths.app_output_latest); |     .src(path.resolve(paths.app_output_static, "polyfills/*.js")) | ||||||
| const compressHassioZopfli = () => |     .pipe(zopfli(zopfliOptions)) | ||||||
|   compressDistZopfli( |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "polyfills"))); | ||||||
|     paths.hassio_output_root, |  | ||||||
|     paths.hassio_output_latest, |  | ||||||
|     true |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
| gulp.task("compress-app", gulp.parallel(compressAppBrotli, compressAppZopfli)); |   const translations = gulp | ||||||
| gulp.task( |     .src(path.resolve(paths.app_output_static, "translations/**/*.json")) | ||||||
|   "compress-hassio", |     .pipe(zopfli(zopfliOptions)) | ||||||
|   gulp.parallel(compressHassioBrotli, compressHassioZopfli) |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "translations"))); | ||||||
| ); |  | ||||||
|  |   const icons = gulp | ||||||
|  |     .src(path.resolve(paths.app_output_static, "mdi/*.json")) | ||||||
|  |     .pipe(zopfli(zopfliOptions)) | ||||||
|  |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "mdi"))); | ||||||
|  |  | ||||||
|  |   return merge(jsLatest, jsEs5, polyfills, translations, icons); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("compress-hassio", function compressApp() { | ||||||
|  |   return gulp | ||||||
|  |     .src(path.resolve(paths.hassio_output_root, "**/*.js")) | ||||||
|  |     .pipe(zopfli(zopfliOptions)) | ||||||
|  |     .pipe(gulp.dest(paths.hassio_output_root)); | ||||||
|  | }); | ||||||
|   | |||||||
| @@ -1,13 +1,16 @@ | |||||||
| import gulp from "gulp"; | // Run demo develop mode | ||||||
| import env from "../env.cjs"; | const gulp = require("gulp"); | ||||||
| import "./clean.js"; |  | ||||||
| import "./entry-html.js"; | const env = require("../env"); | ||||||
| import "./gather-static.js"; |  | ||||||
| import "./gen-icons-json.js"; | require("./clean.js"); | ||||||
| import "./rollup.js"; | require("./translations.js"); | ||||||
| import "./service-worker.js"; | require("./gen-icons-json.js"); | ||||||
| import "./translations.js"; | require("./gather-static.js"); | ||||||
| import "./webpack.js"; | require("./webpack.js"); | ||||||
|  | require("./service-worker.js"); | ||||||
|  | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-demo", |   "develop-demo", | ||||||
| @@ -17,12 +20,7 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-demo", |     "clean-demo", | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel( |     gulp.parallel("gen-icons-json", "gen-index-demo-dev", "build-translations"), | ||||||
|       "gen-icons-json", |  | ||||||
|       "gen-pages-demo-dev", |  | ||||||
|       "build-translations", |  | ||||||
|       "build-locale-data" |  | ||||||
|     ), |  | ||||||
|     "copy-static-demo", |     "copy-static-demo", | ||||||
|     env.useRollup() ? "rollup-dev-server-demo" : "webpack-dev-server-demo" |     env.useRollup() ? "rollup-dev-server-demo" : "webpack-dev-server-demo" | ||||||
|   ) |   ) | ||||||
| @@ -37,9 +35,9 @@ gulp.task( | |||||||
|     "clean-demo", |     "clean-demo", | ||||||
|     // Cast needs to be backwards compatible and older HA has no translations |     // Cast needs to be backwards compatible and older HA has no translations | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-demo", |     "copy-static-demo", | ||||||
|     env.useRollup() ? "rollup-prod-demo" : "webpack-prod-demo", |     env.useRollup() ? "rollup-prod-demo" : "webpack-prod-demo", | ||||||
|     "gen-pages-demo-prod" |     "gen-index-demo-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,180 +0,0 @@ | |||||||
| import fs from "fs/promises"; |  | ||||||
| import gulp from "gulp"; |  | ||||||
| import path from "path"; |  | ||||||
| import mapStream from "map-stream"; |  | ||||||
| import transform from "gulp-json-transform"; |  | ||||||
| import { LokaliseApi } from "@lokalise/node-api"; |  | ||||||
| import JSZip from "jszip"; |  | ||||||
|  |  | ||||||
| const inDir = "translations"; |  | ||||||
| const inDirFrontend = `${inDir}/frontend`; |  | ||||||
| const inDirBackend = `${inDir}/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); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function convertBackendTranslations(data, _file) { |  | ||||||
|   const output = { component: {} }; |  | ||||||
|   if (!data.component) { |  | ||||||
|     return output; |  | ||||||
|   } |  | ||||||
|   Object.keys(data.component).forEach((domain) => { |  | ||||||
|     if (!("entity_component" in data.component[domain])) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     output.component[domain] = { entity_component: {} }; |  | ||||||
|     Object.keys(data.component[domain].entity_component).forEach((key) => { |  | ||||||
|       output.component[domain].entity_component[key] = |  | ||||||
|         data.component[domain].entity_component[key]; |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|   return output; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| gulp.task("convert-backend-translations", function () { |  | ||||||
|   return gulp |  | ||||||
|     .src([`${inDirBackend}/*.json`]) |  | ||||||
|     .pipe(transform((data, file) => convertBackendTranslations(data, file))) |  | ||||||
|     .pipe(gulp.dest(inDirBackend)); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task("check-translations-html", function () { |  | ||||||
|   return gulp |  | ||||||
|     .src([`${inDirFrontend}/*.json`, `${inDirBackend}/*.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); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const lokaliseProjects = { |  | ||||||
|   backend: "130246255a974bd3b5e8a1.51616605", |  | ||||||
|   frontend: "3420425759f6d6d241f598.13594006", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| gulp.task("fetch-lokalise", async function () { |  | ||||||
|   let apiKey; |  | ||||||
|   try { |  | ||||||
|     apiKey = |  | ||||||
|       process.env.LOKALISE_TOKEN || |  | ||||||
|       (await fs.readFile(".lokalise_token", { encoding })); |  | ||||||
|   } catch { |  | ||||||
|     throw new Error( |  | ||||||
|       "An Administrator Lokalise API token is required to download the latest set of translations. Place your token in a new file `.lokalise_token` in the repo root directory." |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|   const lokaliseApi = new LokaliseApi({ apiKey }); |  | ||||||
|  |  | ||||||
|   const mkdirPromise = Promise.all([ |  | ||||||
|     fs.mkdir(inDirFrontend, { recursive: true }), |  | ||||||
|     fs.mkdir(inDirBackend, { recursive: true }), |  | ||||||
|   ]); |  | ||||||
|  |  | ||||||
|   await Promise.all( |  | ||||||
|     Object.entries(lokaliseProjects).map(([project, projectId]) => |  | ||||||
|       lokaliseApi |  | ||||||
|         .files() |  | ||||||
|         .download(projectId, { |  | ||||||
|           format: "json", |  | ||||||
|           original_filenames: false, |  | ||||||
|           replace_breaks: false, |  | ||||||
|           json_unescaped_slashes: true, |  | ||||||
|           export_empty_as: "skip", |  | ||||||
|         }) |  | ||||||
|         .then((download) => fetch(download.bundle_url)) |  | ||||||
|         .then((response) => { |  | ||||||
|           if (response.status === 200 || response.status === 0) { |  | ||||||
|             return response.arrayBuffer(); |  | ||||||
|           } |  | ||||||
|           throw new Error(response.statusText); |  | ||||||
|         }) |  | ||||||
|         .then(JSZip.loadAsync) |  | ||||||
|         .then(async (contents) => { |  | ||||||
|           await mkdirPromise; |  | ||||||
|           return Promise.all( |  | ||||||
|             Object.keys(contents.files).map(async (filename) => { |  | ||||||
|               const file = contents.file(filename); |  | ||||||
|               if (!file) { |  | ||||||
|                 // no file, probably a directory |  | ||||||
|                 return Promise.resolve(); |  | ||||||
|               } |  | ||||||
|               return file |  | ||||||
|                 .async("nodebuffer") |  | ||||||
|                 .then((content) => |  | ||||||
|                   fs.writeFile( |  | ||||||
|                     path.join( |  | ||||||
|                       inDir, |  | ||||||
|                       project, |  | ||||||
|                       filename.split("/").splice(-1)[0] |  | ||||||
|                     ), |  | ||||||
|                     content, |  | ||||||
|                     { flag: "w", encoding } |  | ||||||
|                   ) |  | ||||||
|                 ); |  | ||||||
|             }) |  | ||||||
|           ); |  | ||||||
|         }) |  | ||||||
|         .catch((err) => { |  | ||||||
|           console.error(err); |  | ||||||
|           throw err; |  | ||||||
|         }) |  | ||||||
|     ) |  | ||||||
|   ); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task( |  | ||||||
|   "download-translations", |  | ||||||
|   gulp.series( |  | ||||||
|     "fetch-lokalise", |  | ||||||
|     "convert-backend-translations", |  | ||||||
|     "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,285 +1,318 @@ | |||||||
| // Tasks to generate entry HTML | // 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").minify; | ||||||
|  | const paths = require("../paths.js"); | ||||||
|  | const env = require("../env.js"); | ||||||
|  |  | ||||||
| import { | const templatePath = (tpl) => | ||||||
|   applyVersionsToRegexes, |   path.resolve(paths.polymer_dir, "src/html/", `${tpl}.html.template`); | ||||||
|   compileRegex, |  | ||||||
|   getPreUserAgentRegexes, |  | ||||||
| } from "browserslist-useragent-regexp"; |  | ||||||
| import fs from "fs-extra"; |  | ||||||
| import gulp from "gulp"; |  | ||||||
| import { minify } from "html-minifier-terser"; |  | ||||||
| import template from "lodash.template"; |  | ||||||
| import { dirname, extname, resolve } from "node:path"; |  | ||||||
| import { htmlMinifierOptions, terserOptions } from "../bundle.cjs"; |  | ||||||
| import env from "../env.cjs"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
|  |  | ||||||
| // macOS companion app has no way to obtain the Safari version used by WKWebView, | const readFile = (pth) => fs.readFileSync(pth).toString(); | ||||||
| // and it is not in the default user agent string. So we add an additional regex |  | ||||||
| // to serve modern based on a minimum macOS version. We take the minimum Safari |  | ||||||
| // major version from browserslist and manually map that to a supported macOS |  | ||||||
| // version. Note this assumes the user has kept Safari updated. |  | ||||||
| const HA_MACOS_REGEX = |  | ||||||
|   /Home Assistant\/[\d.]+ \(.+; macOS (\d+)\.(\d+)(?:\.(\d+))?\)/; |  | ||||||
| const SAFARI_TO_MACOS = { |  | ||||||
|   15: [10, 15, 0], |  | ||||||
|   16: [11, 0, 0], |  | ||||||
|   17: [12, 0, 0], |  | ||||||
|   18: [13, 0, 0], |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const getCommonTemplateVars = () => { | const renderTemplate = (pth, data = {}, pathFunc = templatePath) => { | ||||||
|   const browserRegexes = getPreUserAgentRegexes({ |   const compiled = template(readFile(pathFunc(pth))); | ||||||
|     env: "modern", |  | ||||||
|     allowHigherVersions: true, |  | ||||||
|     mobileToDesktop: true, |  | ||||||
|     throwOnMissing: true, |  | ||||||
|   }); |  | ||||||
|   const minSafariVersion = browserRegexes.find( |  | ||||||
|     (regex) => regex.family === "safari" |  | ||||||
|   )?.matchedVersions[0][0]; |  | ||||||
|   const minMacOSVersion = SAFARI_TO_MACOS[minSafariVersion]; |  | ||||||
|   if (!minMacOSVersion) { |  | ||||||
|     throw Error( |  | ||||||
|       `Could not find minimum MacOS version for Safari ${minSafariVersion}.` |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|   const haMacOSRegex = applyVersionsToRegexes( |  | ||||||
|     [ |  | ||||||
|       { |  | ||||||
|         family: "ha_macos", |  | ||||||
|         regex: HA_MACOS_REGEX, |  | ||||||
|         matchedVersions: [minMacOSVersion], |  | ||||||
|         requestVersions: [minMacOSVersion], |  | ||||||
|       }, |  | ||||||
|     ], |  | ||||||
|     { ignorePatch: true, allowHigherVersions: true } |  | ||||||
|   ); |  | ||||||
|   return { |  | ||||||
|     useRollup: env.useRollup(), |  | ||||||
|     useWDS: env.useWDS(), |  | ||||||
|     modernRegex: compileRegex(browserRegexes.concat(haMacOSRegex)).toString(), |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const renderTemplate = (templateFile, data = {}) => { |  | ||||||
|   const compiled = template( |  | ||||||
|     fs.readFileSync(templateFile, { encoding: "utf-8" }) |  | ||||||
|   ); |  | ||||||
|   return compiled({ |   return compiled({ | ||||||
|     ...data, |     ...data, | ||||||
|     // Resolve any child/nested templates relative to the parent and pass the same data |     useRollup: env.useRollup(), | ||||||
|     renderTemplate: (childTemplate) => |     useWDS: env.useWDS(), | ||||||
|       renderTemplate(resolve(dirname(templateFile), childTemplate), data), |     renderTemplate, | ||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const WRAP_TAGS = { ".js": "script", ".css": "style" }; | const renderDemoTemplate = (pth, data = {}) => | ||||||
|  |   renderTemplate(pth, data, (tpl) => | ||||||
| const minifyHtml = (content, ext) => { |     path.resolve(paths.demo_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   const wrapTag = WRAP_TAGS[ext] || ""; |  | ||||||
|   const begTag = wrapTag && `<${wrapTag}>`; |  | ||||||
|   const endTag = wrapTag && `</${wrapTag}>`; |  | ||||||
|   return minify(begTag + content + endTag, { |  | ||||||
|     ...htmlMinifierOptions, |  | ||||||
|     conservativeCollapse: false, |  | ||||||
|     minifyJS: terserOptions({ |  | ||||||
|       latestBuild: false, // Shared scripts should be ES5 |  | ||||||
|       isTestBuild: true, // Don't need source maps |  | ||||||
|     }), |  | ||||||
|   }).then((wrapped) => |  | ||||||
|     wrapTag ? wrapped.slice(begTag.length, -endTag.length) : wrapped |  | ||||||
|   ); |   ); | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Function to generate a dev task for each project's configuration | const renderCastTemplate = (pth, data = {}) => | ||||||
| // Note Currently WDS paths are hard-coded to only work for app |   renderTemplate(pth, data, (tpl) => | ||||||
| const genPagesDevTask = |     path.resolve(paths.cast_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   ( |   ); | ||||||
|     pageEntries, |  | ||||||
|     inputRoot, |  | ||||||
|     outputRoot, |  | ||||||
|     useWDS = false, |  | ||||||
|     inputSub = "src/html", |  | ||||||
|     publicRoot = "" |  | ||||||
|   ) => |  | ||||||
|   async () => { |  | ||||||
|     const commonVars = getCommonTemplateVars(); |  | ||||||
|     for (const [page, entries] of Object.entries(pageEntries)) { |  | ||||||
|       const content = renderTemplate( |  | ||||||
|         resolve(inputRoot, inputSub, `${page}.template`), |  | ||||||
|         { |  | ||||||
|           ...commonVars, |  | ||||||
|           latestEntryJS: entries.map((entry) => |  | ||||||
|             useWDS |  | ||||||
|               ? `http://localhost:8000/src/entrypoints/${entry}.ts` |  | ||||||
|               : `${publicRoot}/frontend_latest/${entry}.js` |  | ||||||
|           ), |  | ||||||
|           es5EntryJS: entries.map( |  | ||||||
|             (entry) => `${publicRoot}/frontend_es5/${entry}.js` |  | ||||||
|           ), |  | ||||||
|           latestCustomPanelJS: useWDS |  | ||||||
|             ? "http://localhost:8000/src/entrypoints/custom-panel.ts" |  | ||||||
|             : `${publicRoot}/frontend_latest/custom-panel.js`, |  | ||||||
|           es5CustomPanelJS: `${publicRoot}/frontend_es5/custom-panel.js`, |  | ||||||
|         } |  | ||||||
|       ); |  | ||||||
|       fs.outputFileSync(resolve(outputRoot, page), content); |  | ||||||
|     } |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
| // Same as previous but for production builds | const renderGalleryTemplate = (pth, data = {}) => | ||||||
| // (includes minification and hashed file names from manifest) |   renderTemplate(pth, data, (tpl) => | ||||||
| const genPagesProdTask = |     path.resolve(paths.gallery_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   ( |   ); | ||||||
|     pageEntries, |  | ||||||
|     inputRoot, | const minifyHtml = (content) => | ||||||
|     outputRoot, |   minify(content, { | ||||||
|     outputLatest, |     collapseWhitespace: true, | ||||||
|     outputES5, |     minifyJS: true, | ||||||
|     inputSub = "src/html" |     minifyCSS: true, | ||||||
|   ) => |     removeComments: true, | ||||||
|   async () => { |   }); | ||||||
|     const latestManifest = fs.readJsonSync( |  | ||||||
|       resolve(outputLatest, "manifest.json") | const PAGES = ["onboarding", "authorize"]; | ||||||
|  |  | ||||||
|  | gulp.task("gen-pages-dev", (done) => { | ||||||
|  |   for (const page of PAGES) { | ||||||
|  |     const content = renderTemplate(page, { | ||||||
|  |       latestPageJS: `/frontend_latest/${page}.js`, | ||||||
|  |  | ||||||
|  |       es5PageJS: `/frontend_es5/${page}.js`, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     fs.outputFileSync( | ||||||
|  |       path.resolve(paths.app_output_root, `${page}.html`), | ||||||
|  |       content | ||||||
|     ); |     ); | ||||||
|     const es5Manifest = outputES5 |   } | ||||||
|       ? fs.readJsonSync(resolve(outputES5, "manifest.json")) |   done(); | ||||||
|       : {}; | }); | ||||||
|     const commonVars = getCommonTemplateVars(); |  | ||||||
|     const minifiedHTML = []; |  | ||||||
|     for (const [page, entries] of Object.entries(pageEntries)) { |  | ||||||
|       const content = renderTemplate( |  | ||||||
|         resolve(inputRoot, inputSub, `${page}.template`), |  | ||||||
|         { |  | ||||||
|           ...commonVars, |  | ||||||
|           latestEntryJS: entries.map((entry) => latestManifest[`${entry}.js`]), |  | ||||||
|           es5EntryJS: entries.map((entry) => es5Manifest[`${entry}.js`]), |  | ||||||
|           latestCustomPanelJS: latestManifest["custom-panel.js"], |  | ||||||
|           es5CustomPanelJS: es5Manifest["custom-panel.js"], |  | ||||||
|         } |  | ||||||
|       ); |  | ||||||
|       minifiedHTML.push( |  | ||||||
|         minifyHtml(content, extname(page)).then((minified) => |  | ||||||
|           fs.outputFileSync(resolve(outputRoot, page), minified) |  | ||||||
|         ) |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|     await Promise.all(minifiedHTML); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
| // Map HTML pages to their required entrypoints | gulp.task("gen-pages-prod", (done) => { | ||||||
| const APP_PAGE_ENTRIES = { |   const latestManifest = require(path.resolve( | ||||||
|   "authorize.html": ["authorize"], |  | ||||||
|   "onboarding.html": ["onboarding"], |  | ||||||
|   "index.html": ["core", "app"], |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| gulp.task( |  | ||||||
|   "gen-pages-app-dev", |  | ||||||
|   genPagesDevTask( |  | ||||||
|     APP_PAGE_ENTRIES, |  | ||||||
|     paths.polymer_dir, |  | ||||||
|     paths.app_output_root, |  | ||||||
|     env.useWDS() |  | ||||||
|   ) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task( |  | ||||||
|   "gen-pages-app-prod", |  | ||||||
|   genPagesProdTask( |  | ||||||
|     APP_PAGE_ENTRIES, |  | ||||||
|     paths.polymer_dir, |  | ||||||
|     paths.app_output_root, |  | ||||||
|     paths.app_output_latest, |     paths.app_output_latest, | ||||||
|     paths.app_output_es5 |     "manifest.json" | ||||||
|   ) |   )); | ||||||
| ); |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.app_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |  | ||||||
| const CAST_PAGE_ENTRIES = { |   for (const page of PAGES) { | ||||||
|   "faq.html": ["launcher"], |     const content = renderTemplate(page, { | ||||||
|   "index.html": ["launcher"], |       latestPageJS: latestManifest[`${page}.js`], | ||||||
|   "media.html": ["media"], |  | ||||||
|   "receiver.html": ["receiver"], |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| gulp.task( |       es5PageJS: es5Manifest[`${page}.js`], | ||||||
|   "gen-pages-cast-dev", |     }); | ||||||
|   genPagesDevTask(CAST_PAGE_ENTRIES, paths.cast_dir, paths.cast_output_root) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task( |     fs.outputFileSync( | ||||||
|   "gen-pages-cast-prod", |       path.resolve(paths.app_output_root, `${page}.html`), | ||||||
|   genPagesProdTask( |       minifyHtml(content) | ||||||
|     CAST_PAGE_ENTRIES, |     ); | ||||||
|     paths.cast_dir, |   } | ||||||
|     paths.cast_output_root, |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-app-dev", (done) => { | ||||||
|  |   let latestAppJS, latestCoreJS, latestCustomPanelJS; | ||||||
|  |  | ||||||
|  |   if (env.useWDS()) { | ||||||
|  |     latestAppJS = "http://localhost:8000/src/entrypoints/app.ts"; | ||||||
|  |     latestCoreJS = "http://localhost:8000/src/entrypoints/core.ts"; | ||||||
|  |     latestCustomPanelJS = | ||||||
|  |       "http://localhost:8000/src/entrypoints/custom-panel.ts"; | ||||||
|  |   } else { | ||||||
|  |     latestAppJS = "/frontend_latest/app.js"; | ||||||
|  |     latestCoreJS = "/frontend_latest/core.js"; | ||||||
|  |     latestCustomPanelJS = "/frontend_latest/custom-panel.js"; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const content = renderTemplate("index", { | ||||||
|  |     latestAppJS, | ||||||
|  |     latestCoreJS, | ||||||
|  |     latestCustomPanelJS, | ||||||
|  |  | ||||||
|  |     es5AppJS: "/frontend_es5/app.js", | ||||||
|  |     es5CoreJS: "/frontend_es5/core.js", | ||||||
|  |     es5CustomPanelJS: "/frontend_es5/custom-panel.js", | ||||||
|  |   }).replace(/#THEMEC/g, "{{ theme_color }}"); | ||||||
|  |  | ||||||
|  |   fs.outputFileSync(path.resolve(paths.app_output_root, "index.html"), content); | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-app-prod", (done) => { | ||||||
|  |   const latestManifest = require(path.resolve( | ||||||
|  |     paths.app_output_latest, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.app_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const content = renderTemplate("index", { | ||||||
|  |     latestAppJS: latestManifest["app.js"], | ||||||
|  |     latestCoreJS: latestManifest["core.js"], | ||||||
|  |     latestCustomPanelJS: latestManifest["custom-panel.js"], | ||||||
|  |  | ||||||
|  |     es5AppJS: es5Manifest["app.js"], | ||||||
|  |     es5CoreJS: es5Manifest["core.js"], | ||||||
|  |     es5CustomPanelJS: es5Manifest["custom-panel.js"], | ||||||
|  |   }); | ||||||
|  |   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) => { | ||||||
|  |   const contentReceiver = renderCastTemplate("receiver", { | ||||||
|  |     latestReceiverJS: "/frontend_latest/receiver.js", | ||||||
|  |   }); | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "receiver.html"), | ||||||
|  |     contentReceiver | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   const contentFAQ = renderCastTemplate("launcher-faq", { | ||||||
|  |     latestLauncherJS: "/frontend_latest/launcher.js", | ||||||
|  |     es5LauncherJS: "/frontend_es5/launcher.js", | ||||||
|  |   }); | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "faq.html"), | ||||||
|  |     contentFAQ | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   const contentLauncher = renderCastTemplate("launcher", { | ||||||
|  |     latestLauncherJS: "/frontend_latest/launcher.js", | ||||||
|  |     es5LauncherJS: "/frontend_es5/launcher.js", | ||||||
|  |   }); | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "index.html"), | ||||||
|  |     contentLauncher | ||||||
|  |   ); | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-cast-prod", (done) => { | ||||||
|  |   const latestManifest = require(path.resolve( | ||||||
|     paths.cast_output_latest, |     paths.cast_output_latest, | ||||||
|     paths.cast_output_es5 |     "manifest.json" | ||||||
|   ) |   )); | ||||||
| ); |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.cast_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |  | ||||||
| const DEMO_PAGE_ENTRIES = { "index.html": ["main"] }; |   const contentReceiver = renderCastTemplate("receiver", { | ||||||
|  |     latestReceiverJS: latestManifest["receiver.js"], | ||||||
|  |   }); | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "receiver.html"), | ||||||
|  |     contentReceiver | ||||||
|  |   ); | ||||||
|  |  | ||||||
| gulp.task( |   const contentFAQ = renderCastTemplate("launcher-faq", { | ||||||
|   "gen-pages-demo-dev", |     latestLauncherJS: latestManifest["launcher.js"], | ||||||
|   genPagesDevTask(DEMO_PAGE_ENTRIES, paths.demo_dir, paths.demo_output_root) |     es5LauncherJS: es5Manifest["launcher.js"], | ||||||
| ); |   }); | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "faq.html"), | ||||||
|  |     contentFAQ | ||||||
|  |   ); | ||||||
|  |  | ||||||
| gulp.task( |   const contentLauncher = renderCastTemplate("launcher", { | ||||||
|   "gen-pages-demo-prod", |     latestLauncherJS: latestManifest["launcher.js"], | ||||||
|   genPagesProdTask( |     es5LauncherJS: es5Manifest["launcher.js"], | ||||||
|     DEMO_PAGE_ENTRIES, |   }); | ||||||
|     paths.demo_dir, |   fs.outputFileSync( | ||||||
|     paths.demo_output_root, |     path.resolve(paths.cast_output_root, "index.html"), | ||||||
|  |     contentLauncher | ||||||
|  |   ); | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-demo-dev", (done) => { | ||||||
|  |   const content = renderDemoTemplate("index", { | ||||||
|  |     latestDemoJS: "/frontend_latest/main.js", | ||||||
|  |  | ||||||
|  |     es5DemoJS: "/frontend_es5/main.js", | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.demo_output_root, "index.html"), | ||||||
|  |     content | ||||||
|  |   ); | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-demo-prod", (done) => { | ||||||
|  |   const latestManifest = require(path.resolve( | ||||||
|     paths.demo_output_latest, |     paths.demo_output_latest, | ||||||
|     paths.demo_output_es5 |     "manifest.json" | ||||||
|   ) |   )); | ||||||
| ); |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.demo_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const content = renderDemoTemplate("index", { | ||||||
|  |     latestDemoJS: latestManifest["main.js"], | ||||||
|  |  | ||||||
| const GALLERY_PAGE_ENTRIES = { "index.html": ["entrypoint"] }; |     es5DemoJS: es5Manifest["main.js"], | ||||||
|  |   }); | ||||||
|  |   const minified = minifyHtml(content); | ||||||
|  |  | ||||||
| gulp.task( |   fs.outputFileSync( | ||||||
|   "gen-pages-gallery-dev", |     path.resolve(paths.demo_output_root, "index.html"), | ||||||
|   genPagesDevTask( |     minified | ||||||
|     GALLERY_PAGE_ENTRIES, |   ); | ||||||
|     paths.gallery_dir, |   done(); | ||||||
|     paths.gallery_output_root | }); | ||||||
|   ) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task( | gulp.task("gen-index-gallery-dev", (done) => { | ||||||
|   "gen-pages-gallery-prod", |   const content = renderGalleryTemplate("index", { | ||||||
|   genPagesProdTask( |     latestGalleryJS: "./frontend_latest/entrypoint.js", | ||||||
|     GALLERY_PAGE_ENTRIES, |   }); | ||||||
|     paths.gallery_dir, |  | ||||||
|     paths.gallery_output_root, |  | ||||||
|     paths.gallery_output_latest |  | ||||||
|   ) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| const HASSIO_PAGE_ENTRIES = { "entrypoint.js": ["entrypoint"] }; |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.gallery_output_root, "index.html"), | ||||||
|  |     content | ||||||
|  |   ); | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task("gen-index-gallery-prod", (done) => { | ||||||
|   "gen-pages-hassio-dev", |   const latestManifest = require(path.resolve( | ||||||
|   genPagesDevTask( |     paths.gallery_output_latest, | ||||||
|     HASSIO_PAGE_ENTRIES, |     "manifest.json" | ||||||
|     paths.hassio_dir, |   )); | ||||||
|     paths.hassio_output_root, |   const content = renderGalleryTemplate("index", { | ||||||
|     undefined, |     latestGalleryJS: latestManifest["entrypoint.js"], | ||||||
|     "src", |   }); | ||||||
|     paths.hassio_publicPath |   const minified = minifyHtml(content); | ||||||
|   ) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task( |   fs.outputFileSync( | ||||||
|   "gen-pages-hassio-prod", |     path.resolve(paths.gallery_output_root, "index.html"), | ||||||
|   genPagesProdTask( |     minified | ||||||
|     HASSIO_PAGE_ENTRIES, |   ); | ||||||
|     paths.hassio_dir, |   done(); | ||||||
|     paths.hassio_output_root, | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-hassio-dev", async () => { | ||||||
|  |   writeHassioEntrypoint( | ||||||
|  |     `${paths.hassio_publicPath}/frontend_latest/entrypoint.js`, | ||||||
|  |     `${paths.hassio_publicPath}/frontend_es5/entrypoint.js` | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-hassio-prod", async () => { | ||||||
|  |   const latestManifest = require(path.resolve( | ||||||
|     paths.hassio_output_latest, |     paths.hassio_output_latest, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const es5Manifest = require(path.resolve( | ||||||
|     paths.hassio_output_es5, |     paths.hassio_output_es5, | ||||||
|     "src" |     "manifest.json" | ||||||
|   ) |   )); | ||||||
| ); |   writeHassioEntrypoint( | ||||||
|  |     latestManifest["entrypoint.js"], | ||||||
|  |     es5Manifest["entrypoint.js"] | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function writeHassioEntrypoint(latestEntrypoint, es5Entrypoint) { | ||||||
|  |   fs.mkdirSync(paths.hassio_output_root, { recursive: true }); | ||||||
|  |   fs.writeFileSync( | ||||||
|  |     path.resolve(paths.hassio_output_root, "entrypoint.js"), | ||||||
|  |     ` | ||||||
|  | try { | ||||||
|  |   new Function("import('${latestEntrypoint}')")(); | ||||||
|  | } catch (err) { | ||||||
|  |   var el = document.createElement('script'); | ||||||
|  |   el.src = '${es5Entrypoint}'; | ||||||
|  |   document.body.appendChild(el); | ||||||
|  | } | ||||||
|  |   `, | ||||||
|  |     { encoding: "utf-8" } | ||||||
|  |   ); | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,171 +0,0 @@ | |||||||
| // Task to download the latest Lokalise translations from the nightly workflow artifacts |  | ||||||
|  |  | ||||||
| import { createOAuthDeviceAuth } from "@octokit/auth-oauth-device"; |  | ||||||
| import { retry } from "@octokit/plugin-retry"; |  | ||||||
| import { Octokit } from "@octokit/rest"; |  | ||||||
| import { deleteAsync } from "del"; |  | ||||||
| import { mkdir, readFile, writeFile } from "fs/promises"; |  | ||||||
| import gulp from "gulp"; |  | ||||||
| import jszip from "jszip"; |  | ||||||
| import path from "path"; |  | ||||||
| import process from "process"; |  | ||||||
| import { extract } from "tar"; |  | ||||||
|  |  | ||||||
| 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 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 = 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 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( |  | ||||||
|           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( |  | ||||||
|       writeFile(ARTIFACT_FILE, JSON.stringify(latestArtifact, null, 2)) |  | ||||||
|     ) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   // Remove the current translations |  | ||||||
|   const deleteCurrent = Promise.all(writings).then( |  | ||||||
|     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(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,141 +1,40 @@ | |||||||
| import fs from "fs"; | // Run demo develop mode | ||||||
| import { glob } from "glob"; | const gulp = require("gulp"); | ||||||
| import gulp from "gulp"; | const fs = require("fs"); | ||||||
| import yaml from "js-yaml"; | const path = require("path"); | ||||||
| import { marked } from "marked"; |  | ||||||
| import path from "path"; |  | ||||||
| import env from "../env.cjs"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
| import "./clean.js"; |  | ||||||
| import "./entry-html.js"; |  | ||||||
| import "./gather-static.js"; |  | ||||||
| import "./gen-icons-json.js"; |  | ||||||
| import "./rollup.js"; |  | ||||||
| import "./service-worker.js"; |  | ||||||
| import "./translations.js"; |  | ||||||
| import "./webpack.js"; |  | ||||||
|  |  | ||||||
| gulp.task("gather-gallery-pages", async function gatherPages() { | const env = require("../env"); | ||||||
|   const pageDir = path.resolve(paths.gallery_dir, "src/pages"); | const paths = require("../paths"); | ||||||
|   const files = await glob(path.resolve(pageDir, "**/*")); |  | ||||||
|  |  | ||||||
|   const galleryBuild = path.resolve(paths.gallery_dir, "build"); | require("./clean.js"); | ||||||
|   fs.mkdirSync(galleryBuild, { recursive: true }); | 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"); | ||||||
|  |  | ||||||
|   let content = "export const PAGES = {\n"; | gulp.task("gather-gallery-demos", async function gatherDemos() { | ||||||
|  |   const files = await fs.promises.readdir( | ||||||
|  |     path.resolve(paths.gallery_dir, "src/demos") | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const processed = new Set(); |   let content = "export const DEMOS = {\n"; | ||||||
|  |  | ||||||
|   for (const file of files) { |   for (const file of files) { | ||||||
|     if (fs.lstatSync(file).isDirectory()) { |     const demoId = path.basename(file, ".ts"); | ||||||
|       continue; |     const demoPath = "../src/demos/" + demoId; | ||||||
|     } |     content += `  "${demoId}": () => import("${demoPath}"),\n`; | ||||||
|     const pageId = file.substring(pageDir.length + 1, file.lastIndexOf(".")); |  | ||||||
|  |  | ||||||
|     if (processed.has(pageId)) { |  | ||||||
|       continue; |  | ||||||
|     } |  | ||||||
|     processed.add(pageId); |  | ||||||
|  |  | ||||||
|     const [category] = pageId.split("/", 2); |  | ||||||
|  |  | ||||||
|     const demoFile = path.resolve(pageDir, `${pageId}.ts`); |  | ||||||
|     const descriptionFile = path.resolve(pageDir, `${pageId}.markdown`); |  | ||||||
|     const hasDemo = fs.existsSync(demoFile); |  | ||||||
|     let hasDescription = fs.existsSync(descriptionFile); |  | ||||||
|     let metadata = {}; |  | ||||||
|     if (hasDescription) { |  | ||||||
|       let descriptionContent = fs.readFileSync(descriptionFile, "utf-8"); |  | ||||||
|  |  | ||||||
|       if (descriptionContent.startsWith("---")) { |  | ||||||
|         const metadataEnd = descriptionContent.indexOf("---", 3); |  | ||||||
|         metadata = yaml.load(descriptionContent.substring(3, metadataEnd)); |  | ||||||
|         descriptionContent = descriptionContent |  | ||||||
|           .substring(metadataEnd + 3) |  | ||||||
|           .trim(); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       // If description is just metadata |  | ||||||
|       if (descriptionContent === "") { |  | ||||||
|         hasDescription = false; |  | ||||||
|       } else { |  | ||||||
|         descriptionContent = marked(descriptionContent).replace(/`/g, "\\`"); |  | ||||||
|         fs.mkdirSync(path.resolve(galleryBuild, category), { recursive: true }); |  | ||||||
|         fs.writeFileSync( |  | ||||||
|           path.resolve(galleryBuild, `${pageId}-description.ts`), |  | ||||||
|           ` |  | ||||||
|           import {html} from "lit"; |  | ||||||
|           export default html\`${descriptionContent}\` |  | ||||||
|           ` |  | ||||||
|         ); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     content += `  "${pageId}": { |  | ||||||
|       metadata: ${JSON.stringify(metadata)}, |  | ||||||
|       ${ |  | ||||||
|         hasDescription |  | ||||||
|           ? `description: () => import("./${pageId}-description").then(m => m.default),` |  | ||||||
|           : "" |  | ||||||
|       } |  | ||||||
|       ${hasDemo ? `demo: () => import("../src/pages/${pageId}")` : ""} |  | ||||||
|  |  | ||||||
|     },\n`; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   content += "};\n"; |   content += "};"; | ||||||
|  |  | ||||||
|   // Generate sidebar |   const galleryBuild = path.resolve(paths.gallery_dir, "build"); | ||||||
|   const sidebarPath = path.resolve(paths.gallery_dir, "sidebar.js"); |  | ||||||
|   const sidebar = (await import(sidebarPath)).default; |  | ||||||
|  |  | ||||||
|   const pagesToProcess = {}; |  | ||||||
|   for (const key of processed) { |  | ||||||
|     const [category, page] = key.split("/", 2); |  | ||||||
|     if (!(category in pagesToProcess)) { |  | ||||||
|       pagesToProcess[category] = new Set(); |  | ||||||
|     } |  | ||||||
|     pagesToProcess[category].add(page); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   for (const group of Object.values(sidebar)) { |  | ||||||
|     const toProcess = pagesToProcess[group.category]; |  | ||||||
|     delete pagesToProcess[group.category]; |  | ||||||
|  |  | ||||||
|     if (!toProcess) { |  | ||||||
|       console.error("Unknown category", group.category); |  | ||||||
|       if (!group.pages) { |  | ||||||
|         group.pages = []; |  | ||||||
|       } |  | ||||||
|       continue; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Any pre-defined groups will not be sorted. |  | ||||||
|     if (group.pages) { |  | ||||||
|       for (const page of group.pages) { |  | ||||||
|         if (!toProcess.delete(page)) { |  | ||||||
|           console.error("Found unreferenced demo", page); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       group.pages = []; |  | ||||||
|     } |  | ||||||
|     for (const page of Array.from(toProcess).sort()) { |  | ||||||
|       group.pages.push(page); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   for (const [category, pages] of Object.entries(pagesToProcess)) { |  | ||||||
|     sidebar.push({ |  | ||||||
|       category, |  | ||||||
|       header: category, |  | ||||||
|       pages: Array.from(pages).sort(), |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   content += `export const SIDEBAR = ${JSON.stringify(sidebar, null, 2)};\n`; |  | ||||||
|  |  | ||||||
|  |   fs.mkdirSync(galleryBuild, { recursive: true }); | ||||||
|   fs.writeFileSync( |   fs.writeFileSync( | ||||||
|     path.resolve(galleryBuild, "import-pages.ts"), |     path.resolve(galleryBuild, "import-demos.ts"), | ||||||
|     content, |     content, | ||||||
|     "utf-8" |     "utf-8" | ||||||
|   ); |   ); | ||||||
| @@ -152,25 +51,11 @@ gulp.task( | |||||||
|     gulp.parallel( |     gulp.parallel( | ||||||
|       "gen-icons-json", |       "gen-icons-json", | ||||||
|       "build-translations", |       "build-translations", | ||||||
|       "build-locale-data", |       "gather-gallery-demos" | ||||||
|       "gather-gallery-pages" |  | ||||||
|     ), |     ), | ||||||
|     "copy-static-gallery", |     "copy-static-gallery", | ||||||
|     "gen-pages-gallery-dev", |     "gen-index-gallery-dev", | ||||||
|     gulp.parallel( |     env.useRollup() ? "rollup-dev-server-gallery" : "webpack-dev-server-gallery" | ||||||
|       env.useRollup() |  | ||||||
|         ? "rollup-dev-server-gallery" |  | ||||||
|         : "webpack-dev-server-gallery", |  | ||||||
|       async function watchMarkdownFiles() { |  | ||||||
|         gulp.watch( |  | ||||||
|           [ |  | ||||||
|             path.resolve(paths.gallery_dir, "src/pages/**/*.markdown"), |  | ||||||
|             path.resolve(paths.gallery_dir, "sidebar.js"), |  | ||||||
|           ], |  | ||||||
|           gulp.series("gather-gallery-pages") |  | ||||||
|         ); |  | ||||||
|       } |  | ||||||
|     ) |  | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -185,11 +70,10 @@ gulp.task( | |||||||
|     gulp.parallel( |     gulp.parallel( | ||||||
|       "gen-icons-json", |       "gen-icons-json", | ||||||
|       "build-translations", |       "build-translations", | ||||||
|       "build-locale-data", |       "gather-gallery-demos" | ||||||
|       "gather-gallery-pages" |  | ||||||
|     ), |     ), | ||||||
|     "copy-static-gallery", |     "copy-static-gallery", | ||||||
|     env.useRollup() ? "rollup-prod-gallery" : "webpack-prod-gallery", |     env.useRollup() ? "rollup-prod-gallery" : "webpack-prod-gallery", | ||||||
|     "gen-pages-gallery-prod" |     "gen-index-gallery-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| // Gulp task to gather all static files. | // Gulp task to gather all static files. | ||||||
|  |  | ||||||
| import fs from "fs-extra"; | const gulp = require("gulp"); | ||||||
| import gulp from "gulp"; | const path = require("path"); | ||||||
| import path from "path"; | const cpx = require("cpx"); | ||||||
| import paths from "../paths.cjs"; | const fs = require("fs-extra"); | ||||||
| import env from "../env.cjs"; | const paths = require("../paths"); | ||||||
|  |  | ||||||
| const npmPath = (...parts) => | const npmPath = (...parts) => | ||||||
|   path.resolve(paths.polymer_dir, "node_modules", ...parts); |   path.resolve(paths.polymer_dir, "node_modules", ...parts); | ||||||
| @@ -13,28 +13,19 @@ const polyPath = (...parts) => path.resolve(paths.polymer_dir, ...parts); | |||||||
| const copyFileDir = (fromFile, toDir) => | const copyFileDir = (fromFile, toDir) => | ||||||
|   fs.copySync(fromFile, path.join(toDir, path.basename(fromFile))); |   fs.copySync(fromFile, path.join(toDir, path.basename(fromFile))); | ||||||
|  |  | ||||||
| const genStaticPath = | const genStaticPath = (staticDir) => (...parts) => | ||||||
|   (staticDir) => |   path.resolve(staticDir, ...parts); | ||||||
|   (...parts) => |  | ||||||
|     path.resolve(staticDir, ...parts); |  | ||||||
|  |  | ||||||
| function copyTranslations(staticDir) { | function copyTranslations(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|  |  | ||||||
|   // Translation output |   // Translation output | ||||||
|   fs.copySync( |   fs.copySync( | ||||||
|     polyPath("build/translations/output"), |     polyPath("build-translations/output"), | ||||||
|     staticPath("translations") |     staticPath("translations") | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| function copyLocaleData(staticDir) { |  | ||||||
|   const staticPath = genStaticPath(staticDir); |  | ||||||
|  |  | ||||||
|   // Locale data output |  | ||||||
|   fs.copySync(polyPath("build/locale-data"), staticPath("locale-data")); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function copyMdiIcons(staticDir) { | function copyMdiIcons(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|  |  | ||||||
| @@ -60,18 +51,9 @@ function copyPolyfills(staticDir) { | |||||||
|     npmPath("@webcomponents/webcomponentsjs/webcomponents-bundle.js.map"), |     npmPath("@webcomponents/webcomponentsjs/webcomponents-bundle.js.map"), | ||||||
|     staticPath("polyfills/") |     staticPath("polyfills/") | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   // dialog-polyfill css |  | ||||||
|   copyFileDir( |  | ||||||
|     npmPath("dialog-polyfill/dialog-polyfill.css"), |  | ||||||
|     staticPath("polyfills/") |  | ||||||
|   ); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function copyLoaderJS(staticDir) { | function copyLoaderJS(staticDir) { | ||||||
|   if (!env.useRollup()) { |  | ||||||
|     return; |  | ||||||
|   } |  | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|   copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js")); |   copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js")); | ||||||
|   copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js")); |   copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js")); | ||||||
| @@ -80,20 +62,12 @@ function copyLoaderJS(staticDir) { | |||||||
| function copyFonts(staticDir) { | function copyFonts(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|   // Local fonts |   // Local fonts | ||||||
|   fs.copySync( |   cpx.copySync( | ||||||
|     npmPath("roboto-fontface/fonts/roboto/"), |     npmPath("roboto-fontface/fonts/roboto/*.woff2"), | ||||||
|     staticPath("fonts/roboto/"), |     staticPath("fonts/roboto") | ||||||
|     { |  | ||||||
|       filter: (src) => !src.includes(".") || src.endsWith(".woff2"), |  | ||||||
|     } |  | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| function copyQrScannerWorker(staticDir) { |  | ||||||
|   const staticPath = genStaticPath(staticDir); |  | ||||||
|   copyFileDir(npmPath("qr-scanner/qr-scanner-worker.min.js"), staticPath("js")); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function copyMapPanel(staticDir) { | function copyMapPanel(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|   copyFileDir( |   copyFileDir( | ||||||
| @@ -106,27 +80,11 @@ function copyMapPanel(staticDir) { | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| gulp.task("copy-locale-data", async () => { |  | ||||||
|   const staticDir = paths.app_output_static; |  | ||||||
|   copyLocaleData(staticDir); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task("copy-translations-app", async () => { | gulp.task("copy-translations-app", async () => { | ||||||
|   const staticDir = paths.app_output_static; |   const staticDir = paths.app_output_static; | ||||||
|   copyTranslations(staticDir); |   copyTranslations(staticDir); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-translations-supervisor", async () => { |  | ||||||
|   const staticDir = paths.hassio_output_static; |  | ||||||
|   copyTranslations(staticDir); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task("copy-static-supervisor", async () => { |  | ||||||
|   const staticDir = paths.hassio_output_static; |  | ||||||
|   copyLocaleData(staticDir); |  | ||||||
|   copyFonts(staticDir); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task("copy-static-app", async () => { | gulp.task("copy-static-app", async () => { | ||||||
|   const staticDir = paths.app_output_static; |   const staticDir = paths.app_output_static; | ||||||
|   // Basic static files |   // Basic static files | ||||||
| @@ -136,14 +94,10 @@ gulp.task("copy-static-app", async () => { | |||||||
|   copyPolyfills(staticDir); |   copyPolyfills(staticDir); | ||||||
|   copyFonts(staticDir); |   copyFonts(staticDir); | ||||||
|   copyTranslations(staticDir); |   copyTranslations(staticDir); | ||||||
|   copyLocaleData(staticDir); |  | ||||||
|   copyMdiIcons(staticDir); |   copyMdiIcons(staticDir); | ||||||
|  |  | ||||||
|   // Panel assets |   // Panel assets | ||||||
|   copyMapPanel(staticDir); |   copyMapPanel(staticDir); | ||||||
|  |  | ||||||
|   // Qr Scanner assets |  | ||||||
|   copyQrScannerWorker(staticDir); |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-static-demo", async () => { | gulp.task("copy-static-demo", async () => { | ||||||
| @@ -160,7 +114,6 @@ gulp.task("copy-static-demo", async () => { | |||||||
|   copyMapPanel(paths.demo_output_static); |   copyMapPanel(paths.demo_output_static); | ||||||
|   copyFonts(paths.demo_output_static); |   copyFonts(paths.demo_output_static); | ||||||
|   copyTranslations(paths.demo_output_static); |   copyTranslations(paths.demo_output_static); | ||||||
|   copyLocaleData(paths.demo_output_static); |  | ||||||
|   copyMdiIcons(paths.demo_output_static); |   copyMdiIcons(paths.demo_output_static); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -175,7 +128,6 @@ gulp.task("copy-static-cast", async () => { | |||||||
|   copyMapPanel(paths.cast_output_static); |   copyMapPanel(paths.cast_output_static); | ||||||
|   copyFonts(paths.cast_output_static); |   copyFonts(paths.cast_output_static); | ||||||
|   copyTranslations(paths.cast_output_static); |   copyTranslations(paths.cast_output_static); | ||||||
|   copyLocaleData(paths.cast_output_static); |  | ||||||
|   copyMdiIcons(paths.cast_output_static); |   copyMdiIcons(paths.cast_output_static); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -191,6 +143,5 @@ gulp.task("copy-static-gallery", async () => { | |||||||
|   copyMapPanel(paths.gallery_output_static); |   copyMapPanel(paths.gallery_output_static); | ||||||
|   copyFonts(paths.gallery_output_static); |   copyFonts(paths.gallery_output_static); | ||||||
|   copyTranslations(paths.gallery_output_static); |   copyTranslations(paths.gallery_output_static); | ||||||
|   copyLocaleData(paths.gallery_output_static); |  | ||||||
|   copyMdiIcons(paths.gallery_output_static); |   copyMdiIcons(paths.gallery_output_static); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,15 +1,17 @@ | |||||||
| import fs from "fs"; | const gulp = require("gulp"); | ||||||
| import gulp from "gulp"; | const path = require("path"); | ||||||
| import hash from "object-hash"; | const fs = require("fs"); | ||||||
| import path from "path"; | const hash = require("object-hash"); | ||||||
| import paths from "../paths.cjs"; |  | ||||||
|  |  | ||||||
| const ICON_PACKAGE_PATH = path.resolve("node_modules/@mdi/svg/"); | const ICON_PACKAGE_PATH = path.resolve( | ||||||
|  |   __dirname, | ||||||
|  |   "../../node_modules/@mdi/svg/" | ||||||
|  | ); | ||||||
| const META_PATH = path.resolve(ICON_PACKAGE_PATH, "meta.json"); | const META_PATH = path.resolve(ICON_PACKAGE_PATH, "meta.json"); | ||||||
| const PACKAGE_PATH = path.resolve(ICON_PACKAGE_PATH, "package.json"); | const PACKAGE_PATH = path.resolve(ICON_PACKAGE_PATH, "package.json"); | ||||||
| const ICON_PATH = path.resolve(ICON_PACKAGE_PATH, "svg"); | const ICON_PATH = path.resolve(ICON_PACKAGE_PATH, "svg"); | ||||||
| const OUTPUT_DIR = path.resolve(paths.build_dir, "mdi"); | const OUTPUT_DIR = path.resolve(__dirname, "../../build/mdi"); | ||||||
| const REMOVED_ICONS_PATH = new URL("../removedIcons.json", import.meta.url); | const REMOVED_ICONS_PATH = path.resolve(__dirname, "../removedIcons.json"); | ||||||
|  |  | ||||||
| const encoding = "utf8"; | const encoding = "utf8"; | ||||||
|  |  | ||||||
| @@ -20,40 +22,17 @@ const getMeta = () => { | |||||||
|     const svg = fs.readFileSync(`${ICON_PATH}/${icon.name}.svg`, { |     const svg = fs.readFileSync(`${ICON_PATH}/${icon.name}.svg`, { | ||||||
|       encoding, |       encoding, | ||||||
|     }); |     }); | ||||||
|     return { |     return { path: svg.match(/ d="([^"]+)"/)[1], name: icon.name }; | ||||||
|       path: svg.match(/ d="([^"]+)"/)[1], |  | ||||||
|       name: icon.name, |  | ||||||
|       tags: icon.tags, |  | ||||||
|       aliases: icon.aliases, |  | ||||||
|     }; |  | ||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const addRemovedMeta = (meta) => { | const addRemovedMeta = (meta) => { | ||||||
|   const file = fs.readFileSync(REMOVED_ICONS_PATH, { encoding }); |   const file = fs.readFileSync(REMOVED_ICONS_PATH, { encoding }); | ||||||
|   const removed = JSON.parse(file); |   const removed = JSON.parse(file); | ||||||
|   const removedMeta = removed.map((removeIcon) => ({ |   const combinedMeta = [...meta, ...removed]; | ||||||
|     path: removeIcon.path, |  | ||||||
|     name: removeIcon.name, |  | ||||||
|     tags: [], |  | ||||||
|     aliases: [], |  | ||||||
|   })); |  | ||||||
|   const combinedMeta = [...meta, ...removedMeta]; |  | ||||||
|   return combinedMeta.sort((a, b) => a.name.localeCompare(b.name)); |   return combinedMeta.sort((a, b) => a.name.localeCompare(b.name)); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const homeAutomationTag = "Home Automation"; |  | ||||||
|  |  | ||||||
| const orderMeta = (meta) => { |  | ||||||
|   const homeAutomationMeta = meta.filter((icon) => |  | ||||||
|     icon.tags.includes(homeAutomationTag) |  | ||||||
|   ); |  | ||||||
|   const otherMeta = meta.filter( |  | ||||||
|     (icon) => !icon.tags.includes(homeAutomationTag) |  | ||||||
|   ); |  | ||||||
|   return [...homeAutomationMeta, ...otherMeta]; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const splitBySize = (meta) => { | const splitBySize = (meta) => { | ||||||
|   const chunks = []; |   const chunks = []; | ||||||
|   const CHUNK_SIZE = 50000; |   const CHUNK_SIZE = 50000; | ||||||
| @@ -98,10 +77,8 @@ const findDifferentiator = (curString, prevString) => { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| gulp.task("gen-icons-json", (done) => { | gulp.task("gen-icons-json", (done) => { | ||||||
|   const meta = getMeta(); |   const meta = addRemovedMeta(getMeta()); | ||||||
|  |   const split = splitBySize(meta); | ||||||
|   const metaAndRemoved = addRemovedMeta(meta); |  | ||||||
|   const split = splitBySize(metaAndRemoved); |  | ||||||
|  |  | ||||||
|   if (!fs.existsSync(OUTPUT_DIR)) { |   if (!fs.existsSync(OUTPUT_DIR)) { | ||||||
|     fs.mkdirSync(OUTPUT_DIR, { recursive: true }); |     fs.mkdirSync(OUTPUT_DIR, { recursive: true }); | ||||||
| @@ -132,34 +109,12 @@ gulp.task("gen-icons-json", (done) => { | |||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   const file = fs.readFileSync(PACKAGE_PATH, { encoding }); |   const file = fs.readFileSync(PACKAGE_PATH, { encoding }); | ||||||
|   const packageMeta = JSON.parse(file); |   const package = JSON.parse(file); | ||||||
|  |  | ||||||
|   fs.writeFileSync( |   fs.writeFileSync( | ||||||
|     path.resolve(OUTPUT_DIR, "iconMetadata.json"), |     path.resolve(OUTPUT_DIR, "iconMetadata.json"), | ||||||
|     JSON.stringify({ version: packageMeta.version, parts }) |     JSON.stringify({ version: package.version, parts }) | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   fs.writeFileSync( |  | ||||||
|     path.resolve(OUTPUT_DIR, "iconList.json"), |  | ||||||
|     JSON.stringify( |  | ||||||
|       orderMeta(meta).map((icon) => ({ |  | ||||||
|         name: icon.name, |  | ||||||
|         keywords: [ |  | ||||||
|           ...icon.tags.map((t) => t.toLowerCase().replace(/\s\/\s/g, " ")), |  | ||||||
|           ...icon.aliases, |  | ||||||
|         ], |  | ||||||
|       })) |  | ||||||
|     ) |  | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-dummy-icons-json", (done) => { |  | ||||||
|   if (!fs.existsSync(OUTPUT_DIR)) { |  | ||||||
|     fs.mkdirSync(OUTPUT_DIR, { recursive: true }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   fs.writeFileSync(path.resolve(OUTPUT_DIR, "iconList.json"), "[]"); |  | ||||||
|   done(); |  | ||||||
| }); |  | ||||||
|   | |||||||
| @@ -1,13 +1,15 @@ | |||||||
| import gulp from "gulp"; | const gulp = require("gulp"); | ||||||
| import env from "../env.cjs"; | const fs = require("fs"); | ||||||
| import "./clean.js"; | const path = require("path"); | ||||||
| import "./compress.js"; |  | ||||||
| import "./entry-html.js"; | const env = require("../env"); | ||||||
| import "./gather-static.js"; | const paths = require("../paths"); | ||||||
| import "./gen-icons-json.js"; |  | ||||||
| import "./rollup.js"; | require("./clean.js"); | ||||||
| import "./translations.js"; | require("./gen-icons-json.js"); | ||||||
| import "./webpack.js"; | require("./webpack.js"); | ||||||
|  | require("./compress.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-hassio", |   "develop-hassio", | ||||||
| @@ -16,12 +18,8 @@ gulp.task( | |||||||
|       process.env.NODE_ENV = "development"; |       process.env.NODE_ENV = "development"; | ||||||
|     }, |     }, | ||||||
|     "clean-hassio", |     "clean-hassio", | ||||||
|     "gen-dummy-icons-json", |     "gen-icons-json", | ||||||
|     "gen-pages-hassio-dev", |     "gen-index-hassio-dev", | ||||||
|     "build-supervisor-translations", |  | ||||||
|     "copy-translations-supervisor", |  | ||||||
|     "build-locale-data", |  | ||||||
|     "copy-static-supervisor", |  | ||||||
|     env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio" |     env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
| @@ -33,14 +31,10 @@ gulp.task( | |||||||
|       process.env.NODE_ENV = "production"; |       process.env.NODE_ENV = "production"; | ||||||
|     }, |     }, | ||||||
|     "clean-hassio", |     "clean-hassio", | ||||||
|     "gen-dummy-icons-json", |     "gen-icons-json", | ||||||
|     "build-supervisor-translations", |  | ||||||
|     "copy-translations-supervisor", |  | ||||||
|     "build-locale-data", |  | ||||||
|     "copy-static-supervisor", |  | ||||||
|     env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio", |     env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio", | ||||||
|     "gen-pages-hassio-prod", |     "gen-index-hassio-prod", | ||||||
|     ...// Don't compress running tests |     ...// Don't compress running tests | ||||||
|     (env.isTestBuild() ? [] : ["compress-hassio"]) |     (env.isTest() ? [] : ["compress-hassio"]) | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,89 +0,0 @@ | |||||||
| import { deleteSync } from "del"; |  | ||||||
| import { mkdir, readFile, writeFile } from "fs/promises"; |  | ||||||
| import gulp from "gulp"; |  | ||||||
| import { join, resolve } from "node:path"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
|  |  | ||||||
| const formatjsDir = join(paths.polymer_dir, "node_modules", "@formatjs"); |  | ||||||
| const outDir = join(paths.build_dir, "locale-data"); |  | ||||||
|  |  | ||||||
| const INTL_POLYFILLS = { |  | ||||||
|   "intl-datetimeformat": "DateTimeFormat", |  | ||||||
|   "intl-displaynames": "DisplayNames", |  | ||||||
|   "intl-listformat": "ListFormat", |  | ||||||
|   "intl-numberformat": "NumberFormat", |  | ||||||
|   "intl-relativetimeformat": "RelativeTimeFormat", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const convertToJSON = async ( |  | ||||||
|   pkg, |  | ||||||
|   lang, |  | ||||||
|   subDir = "locale-data", |  | ||||||
|   addFunc = "__addLocaleData", |  | ||||||
|   skipMissing = true |  | ||||||
| ) => { |  | ||||||
|   let localeData; |  | ||||||
|   try { |  | ||||||
|     // use "pt" for "pt-BR", because "pt-BR" is unsupported by @formatjs |  | ||||||
|     const language = lang === "pt-BR" ? "pt" : lang; |  | ||||||
|  |  | ||||||
|     localeData = await readFile( |  | ||||||
|       join(formatjsDir, pkg, subDir, `${language}.js`), |  | ||||||
|       "utf-8" |  | ||||||
|     ); |  | ||||||
|   } catch (e) { |  | ||||||
|     // Ignore if language is missing (i.e. not supported by @formatjs) |  | ||||||
|     if (e.code === "ENOENT" && skipMissing) { |  | ||||||
|       console.warn(`Skipped missing data for language ${lang} from ${pkg}`); |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     throw e; |  | ||||||
|   } |  | ||||||
|   // Convert to JSON |  | ||||||
|   const obj = INTL_POLYFILLS[pkg]; |  | ||||||
|   const dataRegex = new RegExp( |  | ||||||
|     `Intl\\.${obj}\\.${addFunc}\\((?<data>.*)\\)`, |  | ||||||
|     "s" |  | ||||||
|   ); |  | ||||||
|   localeData = localeData.match(dataRegex)?.groups?.data; |  | ||||||
|   if (!localeData) { |  | ||||||
|     throw Error(`Failed to extract data for language ${lang} from ${pkg}`); |  | ||||||
|   } |  | ||||||
|   // Parse to validate JSON, then stringify to minify |  | ||||||
|   localeData = JSON.stringify(JSON.parse(localeData)); |  | ||||||
|   await writeFile(join(outDir, `${pkg}/${lang}.json`), localeData); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| gulp.task("clean-locale-data", async () => deleteSync([outDir])); |  | ||||||
|  |  | ||||||
| gulp.task("create-locale-data", async () => { |  | ||||||
|   const translationMeta = JSON.parse( |  | ||||||
|     await readFile( |  | ||||||
|       resolve(paths.translations_src, "translationMetadata.json"), |  | ||||||
|       "utf-8" |  | ||||||
|     ) |  | ||||||
|   ); |  | ||||||
|   const conversions = []; |  | ||||||
|   for (const pkg of Object.keys(INTL_POLYFILLS)) { |  | ||||||
|     // eslint-disable-next-line no-await-in-loop |  | ||||||
|     await mkdir(join(outDir, pkg), { recursive: true }); |  | ||||||
|     for (const lang of Object.keys(translationMeta)) { |  | ||||||
|       conversions.push(convertToJSON(pkg, lang)); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   conversions.push( |  | ||||||
|     convertToJSON( |  | ||||||
|       "intl-datetimeformat", |  | ||||||
|       "add-all-tz", |  | ||||||
|       ".", |  | ||||||
|       "__addTZData", |  | ||||||
|       false |  | ||||||
|     ) |  | ||||||
|   ); |  | ||||||
|   await Promise.all(conversions); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| gulp.task( |  | ||||||
|   "build-locale-data", |  | ||||||
|   gulp.series("clean-locale-data", "create-locale-data") |  | ||||||
| ); |  | ||||||
| @@ -1,14 +1,13 @@ | |||||||
| // Tasks to run Rollup | // Tasks to run Rollup | ||||||
|  | const path = require("path"); | ||||||
| import log from "fancy-log"; | const gulp = require("gulp"); | ||||||
| import gulp from "gulp"; | const rollup = require("rollup"); | ||||||
| import http from "http"; | const handler = require("serve-handler"); | ||||||
| import open from "open"; | const http = require("http"); | ||||||
| import path from "path"; | const log = require("fancy-log"); | ||||||
| import { rollup } from "rollup"; | const rollupConfig = require("../rollup"); | ||||||
| import handler from "serve-handler"; | const paths = require("../paths"); | ||||||
| import paths from "../paths.cjs"; | const open = require("open"); | ||||||
| import rollupConfig from "../rollup.cjs"; |  | ||||||
|  |  | ||||||
| const bothBuilds = (createConfigFunc, params) => | const bothBuilds = (createConfigFunc, params) => | ||||||
|   gulp.series( |   gulp.series( | ||||||
| @@ -31,11 +30,11 @@ const bothBuilds = (createConfigFunc, params) => | |||||||
|   ); |   ); | ||||||
|  |  | ||||||
| function createServer(serveOptions) { | function createServer(serveOptions) { | ||||||
|   const server = http.createServer((request, response) => |   const server = http.createServer((request, response) => { | ||||||
|     handler(request, response, { |     return handler(request, response, { | ||||||
|       public: serveOptions.root, |       public: serveOptions.root, | ||||||
|     }) |     }); | ||||||
|   ); |   }); | ||||||
|  |  | ||||||
|   server.listen( |   server.listen( | ||||||
|     serveOptions.port, |     serveOptions.port, | ||||||
| @@ -47,7 +46,7 @@ function createServer(serveOptions) { | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| function watchRollup(createConfig, extraWatchSrc = [], serveOptions = null) { | function watchRollup(createConfig, extraWatchSrc = [], serveOptions) { | ||||||
|   const { inputOptions, outputOptions } = createConfig({ |   const { inputOptions, outputOptions } = createConfig({ | ||||||
|     isProdBuild: false, |     isProdBuild: false, | ||||||
|     latestBuild: true, |     latestBuild: true, | ||||||
|   | |||||||
| @@ -1,19 +1,21 @@ | |||||||
| // Generate service workers | // 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.js"); | ||||||
|  |  | ||||||
| import { deleteAsync } from "del"; | const swDest = path.resolve(paths.app_output_root, "service_worker.js"); | ||||||
| import gulp from "gulp"; |  | ||||||
| import { mkdir, readFile, symlink, writeFile } from "node:fs/promises"; |  | ||||||
| import { basename, join, relative } from "node:path"; |  | ||||||
| import { injectManifest } from "workbox-build"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
|  |  | ||||||
| const SW_MAP = { | const writeSW = (content) => fs.outputFileSync(swDest, content.trim() + "\n"); | ||||||
|   [paths.app_output_latest]: "modern", |  | ||||||
|   [paths.app_output_es5]: "legacy", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const SW_DEV = | gulp.task("gen-service-worker-app-dev", (done) => { | ||||||
|   ` |   writeSW( | ||||||
|  |     ` | ||||||
| console.debug('Service worker disabled in development'); | console.debug('Service worker disabled in development'); | ||||||
|  |  | ||||||
| self.addEventListener('install', (event) => { | self.addEventListener('install', (event) => { | ||||||
| @@ -21,67 +23,74 @@ self.addEventListener('install', (event) => { | |||||||
|   // removing any prod service worker the dev might have running |   // removing any prod service worker the dev might have running | ||||||
|   self.skipWaiting(); |   self.skipWaiting(); | ||||||
| }); | }); | ||||||
|   `.trim() + "\n"; |   ` | ||||||
|  |  | ||||||
| gulp.task("gen-service-worker-app-dev", async () => { |  | ||||||
|   await mkdir(paths.app_output_root, { recursive: true }); |  | ||||||
|   await Promise.all( |  | ||||||
|     Object.values(SW_MAP).map((build) => |  | ||||||
|       writeFile(join(paths.app_output_root, `sw-${build}.js`), SW_DEV, { |  | ||||||
|         encoding: "utf-8", |  | ||||||
|       }) |  | ||||||
|     ) |  | ||||||
|   ); |   ); | ||||||
|  |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-service-worker-app-prod", () => | gulp.task("gen-service-worker-app-prod", async () => { | ||||||
|   Promise.all( |   // Read bundled source file | ||||||
|     Object.entries(SW_MAP).map(async ([outPath, build]) => { |   const bundleManifestLatest = require(path.resolve( | ||||||
|       const manifest = JSON.parse( |     paths.app_output_latest, | ||||||
|         await readFile(join(outPath, "manifest.json"), "utf-8") |     "manifest.json" | ||||||
|       ); |   )); | ||||||
|       const swSrc = join(paths.app_output_root, manifest["service-worker.js"]); |   let serviceWorkerContent = fs.readFileSync( | ||||||
|       const swDest = join(paths.app_output_root, `sw-${build}.js`); |     paths.app_output_root + bundleManifestLatest["service_worker.js"], | ||||||
|       const buildDir = relative(paths.app_output_root, outPath); |     "utf-8" | ||||||
|       const { warnings } = await injectManifest({ |   ); | ||||||
|         swSrc, |  | ||||||
|         swDest, |   // Delete old file from frontend_latest so manifest won't pick it up | ||||||
|         injectionPoint: "__WB_MANIFEST__", |   fs.removeSync( | ||||||
|         // Files that mach this pattern will be considered unique and skip revision check |     paths.app_output_root + bundleManifestLatest["service_worker.js"] | ||||||
|         // ignore JS files + translation files |   ); | ||||||
|         dontCacheBustURLsMatching: new RegExp( |   fs.removeSync( | ||||||
|           `(?:${buildDir}/.+|static/translations/.+)` |     paths.app_output_root + bundleManifestLatest["service_worker.js.map"] | ||||||
|         ), |   ); | ||||||
|         globDirectory: paths.app_output_root, |  | ||||||
|         globPatterns: [ |   // Remove ES5 | ||||||
|           `${buildDir}/*.js`, |   const bundleManifestES5 = require(path.resolve( | ||||||
|           // Cache all English translations because we catch them as fallback |     paths.app_output_es5, | ||||||
|           // Using pattern to match hash instead of * to avoid caching en-GB |     "manifest.json" | ||||||
|           // 'v' added as valid hash letter because in dev we hash with 'dev' |   )); | ||||||
|           "static/translations/**/en-+([a-fv0-9]).json", |   fs.removeSync(paths.app_output_root + bundleManifestES5["service_worker.js"]); | ||||||
|           // Icon shown on splash screen |   fs.removeSync( | ||||||
|           "static/icons/favicon-192x192.png", |     paths.app_output_root + bundleManifestES5["service_worker.js.map"] | ||||||
|           "static/icons/favicon.ico", |   ); | ||||||
|           // Common fonts |  | ||||||
|           "static/fonts/roboto/Roboto-Light.woff2", |   const workboxManifest = await workboxBuild.getManifest({ | ||||||
|           "static/fonts/roboto/Roboto-Medium.woff2", |     // Files that mach this pattern will be considered unique and skip revision check | ||||||
|           "static/fonts/roboto/Roboto-Regular.woff2", |     // ignore JS files + translation files | ||||||
|           "static/fonts/roboto/Roboto-Bold.woff2", |     dontCacheBustURLsMatching: /(frontend_latest\/.+|static\/translations\/.+)/, | ||||||
|         ], |  | ||||||
|         globIgnores: [`${buildDir}/service-worker*`], |     globDirectory: paths.app_output_root, | ||||||
|       }); |     globPatterns: [ | ||||||
|       if (warnings.length > 0) { |       "frontend_latest/*.js", | ||||||
|         console.warn( |       // Cache all English translations because we catch them as fallback | ||||||
|           `Problems while injecting ${build} service worker:\n`, |       // Using pattern to match hash instead of * to avoid caching en-GB | ||||||
|           warnings.join("\n") |       // 'v' added as valid hash letter because in dev we hash with 'dev' | ||||||
|         ); |       "static/translations/**/en-+([a-fv0-9]).json", | ||||||
|       } |       // Icon shown on splash screen | ||||||
|       await deleteAsync(`${swSrc}?(.map)`); |       "static/icons/favicon-192x192.png", | ||||||
|       // Needed to install new SW from a cached HTML |       "static/icons/favicon.ico", | ||||||
|       if (build === "modern") { |       // Common fonts | ||||||
|         const swOld = join(paths.app_output_root, "service_worker.js"); |       "static/fonts/roboto/Roboto-Light.woff2", | ||||||
|         await symlink(basename(swDest), swOld); |       "static/fonts/roboto/Roboto-Medium.woff2", | ||||||
|       } |       "static/fonts/roboto/Roboto-Regular.woff2", | ||||||
|     }) |       "static/fonts/roboto/Roboto-Bold.woff2", | ||||||
|   ) |     ], | ||||||
| ); |   }); | ||||||
|  |  | ||||||
|  |   for (const warning of workboxManifest.warnings) { | ||||||
|  |     console.warn(warning); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // remove source map and add WB manifest | ||||||
|  |   serviceWorkerContent = sourceMapUrl.removeFrom(serviceWorkerContent); | ||||||
|  |   serviceWorkerContent = serviceWorkerContent.replace( | ||||||
|  |     "WB_MANIFEST", | ||||||
|  |     JSON.stringify(workboxManifest.manifestEntries) | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   // Write new file to root | ||||||
|  |   fs.writeFileSync(swDest, serviceWorkerContent); | ||||||
|  | }); | ||||||
|   | |||||||
| @@ -1,112 +1,89 @@ | |||||||
| /* eslint-disable max-classes-per-file */ | const crypto = require("crypto"); | ||||||
|  | const del = require("del"); | ||||||
| import { deleteAsync } from "del"; | const path = require("path"); | ||||||
| import { glob } from "glob"; | const source = require("vinyl-source-stream"); | ||||||
| import gulp from "gulp"; | const vinylBuffer = require("vinyl-buffer"); | ||||||
| import rename from "gulp-rename"; | const gulp = require("gulp"); | ||||||
| import merge from "lodash.merge"; | const fs = require("fs"); | ||||||
| import { createHash } from "node:crypto"; | const foreach = require("gulp-foreach"); | ||||||
| import { mkdir, readFile } from "node:fs/promises"; | const merge = require("gulp-merge-json"); | ||||||
| import { basename, join } from "node:path"; | const rename = require("gulp-rename"); | ||||||
| import { PassThrough, Transform } from "node:stream"; | const transform = require("gulp-json-transform"); | ||||||
| import { finished } from "node:stream/promises"; | const { mapFiles } = require("../util"); | ||||||
| import env from "../env.cjs"; | const env = require("../env"); | ||||||
| import paths from "../paths.cjs"; | const paths = require("../paths"); | ||||||
| import "./fetch-nightly-translations.js"; |  | ||||||
|  |  | ||||||
| const inFrontendDir = "translations/frontend"; | const inFrontendDir = "translations/frontend"; | ||||||
| const inBackendDir = "translations/backend"; | const inBackendDir = "translations/backend"; | ||||||
| const workDir = "build/translations"; | const workDir = "build-translations"; | ||||||
| const outDir = join(workDir, "output"); | const fullDir = workDir + "/full"; | ||||||
| const EN_SRC = join(paths.translations_src, "en.json"); | const coreDir = workDir + "/core"; | ||||||
| const TEST_LOCALE = "en-x-test"; | const outDir = workDir + "/output"; | ||||||
|  |  | ||||||
| let mergeBackend = false; | let mergeBackend = false; | ||||||
|  |  | ||||||
| gulp.task( | gulp.task("translations-enable-merge-backend", (done) => { | ||||||
|   "translations-enable-merge-backend", |   mergeBackend = true; | ||||||
|   gulp.parallel(async () => { |   done(); | ||||||
|     mergeBackend = true; | }); | ||||||
|   }, "allow-setup-fetch-nightly-translations") |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| // Transform stream to apply a function on Vinyl JSON files (buffer mode only). | String.prototype.rsplit = function (sep, maxsplit) { | ||||||
| // The provided function can either return a new object, or an array of |   var split = this.split(sep); | ||||||
| // [object, subdirectory] pairs for fragmentizing the JSON. |   return maxsplit | ||||||
| class CustomJSON extends Transform { |     ? [split.slice(0, -maxsplit).join(sep)].concat(split.slice(-maxsplit)) | ||||||
|   constructor(func, reviver = null) { |     : split; | ||||||
|     super({ objectMode: true }); |  | ||||||
|     this._func = func; |  | ||||||
|     this._reviver = reviver; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   async _transform(file, _, callback) { |  | ||||||
|     try { |  | ||||||
|       let obj = JSON.parse(file.contents.toString(), this._reviver); |  | ||||||
|       if (this._func) obj = this._func(obj, file.path); |  | ||||||
|       for (const [outObj, dir] of Array.isArray(obj) ? obj : [[obj, ""]]) { |  | ||||||
|         const outFile = file.clone({ contents: false }); |  | ||||||
|         outFile.contents = Buffer.from(JSON.stringify(outObj)); |  | ||||||
|         outFile.dirname += `/${dir}`; |  | ||||||
|         this.push(outFile); |  | ||||||
|       } |  | ||||||
|       callback(null); |  | ||||||
|     } catch (err) { |  | ||||||
|       callback(err); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Transform stream to merge Vinyl JSON files (buffer mode only). |  | ||||||
| class MergeJSON extends Transform { |  | ||||||
|   _objects = []; |  | ||||||
|  |  | ||||||
|   constructor(stem, startObj = {}, reviver = null) { |  | ||||||
|     super({ objectMode: true, allowHalfOpen: false }); |  | ||||||
|     this._stem = stem; |  | ||||||
|     this._startObj = structuredClone(startObj); |  | ||||||
|     this._reviver = reviver; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   async _transform(file, _, callback) { |  | ||||||
|     try { |  | ||||||
|       this._objects.push(JSON.parse(file.contents.toString(), this._reviver)); |  | ||||||
|       if (!this._outFile) this._outFile = file.clone({ contents: false }); |  | ||||||
|       callback(null); |  | ||||||
|     } catch (err) { |  | ||||||
|       callback(err); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   async _flush(callback) { |  | ||||||
|     try { |  | ||||||
|       const mergedObj = merge(this._startObj, ...this._objects); |  | ||||||
|       this._outFile.contents = Buffer.from(JSON.stringify(mergedObj)); |  | ||||||
|       this._outFile.stem = this._stem; |  | ||||||
|       callback(null, this._outFile); |  | ||||||
|     } catch (err) { |  | ||||||
|       callback(err); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Utility to flatten object keys to single level using separator |  | ||||||
| const flatten = (data, prefix = "", sep = ".") => { |  | ||||||
|   const output = {}; |  | ||||||
|   for (const [key, value] of Object.entries(data)) { |  | ||||||
|     if (typeof value === "object") { |  | ||||||
|       Object.assign(output, flatten(value, prefix + key + sep, sep)); |  | ||||||
|     } else { |  | ||||||
|       output[prefix + key] = value; |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   return output; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // Filter functions that can be passed directly to JSON.parse() | // Panel translations which should be split from the core translations. | ||||||
| const emptyReviver = (_key, value) => value || undefined; | const TRANSLATION_FRAGMENTS = Object.keys( | ||||||
| const testReviver = (_key, value) => |   require("../../src/translations/en.json").ui.panel | ||||||
|   value && typeof value === "string" ? "TRANSLATED" : value; | ); | ||||||
|  |  | ||||||
|  | function recursiveFlatten(prefix, data) { | ||||||
|  |   let output = {}; | ||||||
|  |   Object.keys(data).forEach(function (key) { | ||||||
|  |     if (typeof data[key] === "object") { | ||||||
|  |       output = { | ||||||
|  |         ...output, | ||||||
|  |         ...recursiveFlatten(prefix + key + ".", data[key]), | ||||||
|  |       }; | ||||||
|  |     } else { | ||||||
|  |       output[prefix + key] = data[key]; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   return output; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function flatten(data) { | ||||||
|  |   return recursiveFlatten("", data); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function emptyFilter(data) { | ||||||
|  |   const newData = {}; | ||||||
|  |   Object.keys(data).forEach((key) => { | ||||||
|  |     if (data[key]) { | ||||||
|  |       if (typeof data[key] === "object") { | ||||||
|  |         newData[key] = emptyFilter(data[key]); | ||||||
|  |       } else { | ||||||
|  |         newData[key] = data[key]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   return newData; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function recursiveEmpty(data) { | ||||||
|  |   const newData = {}; | ||||||
|  |   Object.keys(data).forEach((key) => { | ||||||
|  |     if (data[key]) { | ||||||
|  |       if (typeof data[key] === "object") { | ||||||
|  |         newData[key] = recursiveEmpty(data[key]); | ||||||
|  |       } else { | ||||||
|  |         newData[key] = "TRANSLATED"; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |   return newData; | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Replace Lokalise key placeholders with their actual values. |  * Replace Lokalise key placeholders with their actual values. | ||||||
| @@ -115,44 +92,69 @@ const testReviver = (_key, value) => | |||||||
|  * be included in src/translations/en.json, but still be usable while |  * be included in src/translations/en.json, but still be usable while | ||||||
|  * developing locally. |  * developing locally. | ||||||
|  * |  * | ||||||
|  * @link https://docs.lokalise.com/en/articles/1400528-key-referencing |  * @link https://docs.lokalise.co/article/KO5SZWLLsy-key-referencing | ||||||
|  */ |  */ | ||||||
| const KEY_REFERENCE = /\[%key:([^%]+)%\]/; | const re_key_reference = /\[%key:([^%]+)%\]/; | ||||||
| const lokaliseTransform = (data, path, original = data) => { | function lokaliseTransform(data, original, file) { | ||||||
|   const output = {}; |   const output = {}; | ||||||
|   for (const [key, value] of Object.entries(data)) { |   Object.entries(data).forEach(([key, value]) => { | ||||||
|     if (typeof value === "object") { |     if (value instanceof Object) { | ||||||
|       output[key] = lokaliseTransform(value, path, original); |       output[key] = lokaliseTransform(value, original, file); | ||||||
|     } else { |     } else { | ||||||
|       output[key] = value.replace(KEY_REFERENCE, (_match, lokalise_key) => { |       output[key] = value.replace(re_key_reference, (match, key) => { | ||||||
|         const replace = lokalise_key.split("::").reduce((tr, k) => { |         const replace = key.split("::").reduce((tr, k) => { | ||||||
|           if (!tr) { |           if (!tr) { | ||||||
|             throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`); |             throw Error(`Invalid key placeholder ${key} in ${file.path}`); | ||||||
|           } |           } | ||||||
|           return tr[k]; |           return tr[k]; | ||||||
|         }, original); |         }, original); | ||||||
|         if (typeof replace !== "string") { |         if (typeof replace !== "string") { | ||||||
|           throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`); |           throw Error(`Invalid key placeholder ${key} in ${file.path}`); | ||||||
|         } |         } | ||||||
|         return replace; |         return replace; | ||||||
|       }); |       }); | ||||||
|     } |     } | ||||||
|   } |   }); | ||||||
|   return output; |   return output; | ||||||
| }; | } | ||||||
|  |  | ||||||
| gulp.task("clean-translations", () => deleteAsync([workDir])); | gulp.task("clean-translations", function () { | ||||||
|  |   return del([workDir]); | ||||||
|  | }); | ||||||
|  |  | ||||||
| const makeWorkDir = () => mkdir(workDir, { recursive: true }); | gulp.task("ensure-translations-build-dir", (done) => { | ||||||
|  |   if (!fs.existsSync(workDir)) { | ||||||
|  |     fs.mkdirSync(workDir); | ||||||
|  |   } | ||||||
|  |   done(); | ||||||
|  | }); | ||||||
|  |  | ||||||
| const createTestTranslation = () => | gulp.task("create-test-metadata", function (cb) { | ||||||
|   env.isProdBuild() |   fs.writeFile( | ||||||
|     ? Promise.resolve() |     workDir + "/testMetadata.json", | ||||||
|     : gulp |     JSON.stringify({ | ||||||
|         .src(EN_SRC) |       test: { | ||||||
|         .pipe(new CustomJSON(null, testReviver)) |         nativeName: "Test", | ||||||
|         .pipe(rename(`${TEST_LOCALE}.json`)) |       }, | ||||||
|         .pipe(gulp.dest(workDir)); |     }), | ||||||
|  |     cb | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task( | ||||||
|  |   "create-test-translation", | ||||||
|  |   gulp.series("create-test-metadata", function createTestTranslation() { | ||||||
|  |     return gulp | ||||||
|  |       .src(path.join(paths.translations_src, "en.json")) | ||||||
|  |       .pipe( | ||||||
|  |         transform(function (data, file) { | ||||||
|  |           return recursiveEmpty(data); | ||||||
|  |         }) | ||||||
|  |       ) | ||||||
|  |       .pipe(rename("test.json")) | ||||||
|  |       .pipe(gulp.dest(workDir)); | ||||||
|  |   }) | ||||||
|  | ); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * This task will build a master translation file, to be used as the base for |  * This task will build a master translation file, to be used as the base for | ||||||
| @@ -163,164 +165,230 @@ const createTestTranslation = () => | |||||||
|  * project is buildable immediately after merging new translation keys, since |  * project is buildable immediately after merging new translation keys, since | ||||||
|  * the Lokalise update to translations/en.json will not happen immediately. |  * the Lokalise update to translations/en.json will not happen immediately. | ||||||
|  */ |  */ | ||||||
| const createMasterTranslation = () => | gulp.task("build-master-translation", function () { | ||||||
|   gulp |   const src = [path.join(paths.translations_src, "en.json")]; | ||||||
|     .src([EN_SRC, ...(mergeBackend ? [`${inBackendDir}/en.json`] : [])]) |  | ||||||
|     .pipe(new CustomJSON(lokaliseTransform)) |  | ||||||
|     .pipe(new MergeJSON("en")) |  | ||||||
|     .pipe(gulp.dest(workDir)); |  | ||||||
|  |  | ||||||
| const FRAGMENTS = ["base"]; |   if (mergeBackend) { | ||||||
|  |     src.push(path.join(inBackendDir, "en.json")); | ||||||
| const toggleSupervisorFragment = async () => { |  | ||||||
|   FRAGMENTS[0] = "supervisor"; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const panelFragment = (fragment) => |  | ||||||
|   fragment !== "base" && fragment !== "supervisor"; |  | ||||||
|  |  | ||||||
| const HASHES = new Map(); |  | ||||||
|  |  | ||||||
| const createTranslations = async () => { |  | ||||||
|   // Parse and store the master to avoid repeating this for each locale, then |  | ||||||
|   // add the panel fragments when processing the app. |  | ||||||
|   const enMaster = JSON.parse(await readFile(`${workDir}/en.json`, "utf-8")); |  | ||||||
|   if (FRAGMENTS[0] === "base") { |  | ||||||
|     FRAGMENTS.push(...Object.keys(enMaster.ui.panel)); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // The downstream pipeline is setup first.  It hashes the merged data for |   return gulp | ||||||
|   // each locale, then fragmentizes and flattens the data for final output. |     .src(src) | ||||||
|   const translationFiles = await glob([ |  | ||||||
|     `${inFrontendDir}/!(en).json`, |  | ||||||
|     ...(env.isProdBuild() ? [] : [`${workDir}/${TEST_LOCALE}.json`]), |  | ||||||
|   ]); |  | ||||||
|   const hashStream = new Transform({ |  | ||||||
|     objectMode: true, |  | ||||||
|     transform: async (file, _, callback) => { |  | ||||||
|       const hash = env.isProdBuild() |  | ||||||
|         ? createHash("md5").update(file.contents).digest("hex") |  | ||||||
|         : "dev"; |  | ||||||
|       HASHES.set(file.stem, hash); |  | ||||||
|       file.stem += `-${hash}`; |  | ||||||
|       callback(null, file); |  | ||||||
|     }, |  | ||||||
|   }).setMaxListeners(translationFiles.length + 1); |  | ||||||
|   const fragmentsStream = hashStream |  | ||||||
|     .pipe( |     .pipe( | ||||||
|       new CustomJSON((data) => |       transform(function (data, file) { | ||||||
|         FRAGMENTS.map((fragment) => { |         return lokaliseTransform(data, data, file); | ||||||
|           switch (fragment) { |       }) | ||||||
|             case "base": |  | ||||||
|               // Remove the panels and supervisor to create the base translations |  | ||||||
|               return [ |  | ||||||
|                 flatten({ |  | ||||||
|                   ...data, |  | ||||||
|                   ui: { ...data.ui, panel: undefined }, |  | ||||||
|                   supervisor: undefined, |  | ||||||
|                 }), |  | ||||||
|                 "", |  | ||||||
|               ]; |  | ||||||
|             case "supervisor": |  | ||||||
|               // Supervisor key is at the top level |  | ||||||
|               return [flatten(data.supervisor), ""]; |  | ||||||
|             default: |  | ||||||
|               // Create a fragment with only the given panel |  | ||||||
|               return [ |  | ||||||
|                 flatten(data.ui.panel[fragment], `ui.panel.${fragment}.`), |  | ||||||
|                 fragment, |  | ||||||
|               ]; |  | ||||||
|           } |  | ||||||
|         }) |  | ||||||
|       ) |  | ||||||
|     ) |     ) | ||||||
|     .pipe(gulp.dest(outDir)); |  | ||||||
|  |  | ||||||
|   // Send the English master downstream first, then for each other locale |  | ||||||
|   // generate merged JSON data to continue piping. It begins with the master |  | ||||||
|   // translation as a failsafe for untranslated strings, and merges all parent |  | ||||||
|   // tags into one file for each specific subtag |  | ||||||
|   // |  | ||||||
|   // TODO: This is a naive interpretation of BCP47 that should be improved. |  | ||||||
|   //       Will be OK for now as long as we don't have anything more complicated |  | ||||||
|   // than a base translation + region. |  | ||||||
|   const masterStream = gulp |  | ||||||
|     .src(`${workDir}/en.json`) |  | ||||||
|     .pipe(new PassThrough({ objectMode: true })); |  | ||||||
|   masterStream.pipe(hashStream, { end: false }); |  | ||||||
|   const mergesFinished = [finished(masterStream)]; |  | ||||||
|   for (const translationFile of translationFiles) { |  | ||||||
|     const locale = basename(translationFile, ".json"); |  | ||||||
|     const subtags = locale.split("-"); |  | ||||||
|     const mergeFiles = []; |  | ||||||
|     for (let i = 1; i <= subtags.length; i++) { |  | ||||||
|       const lang = subtags.slice(0, i).join("-"); |  | ||||||
|       if (lang === TEST_LOCALE) { |  | ||||||
|         mergeFiles.push(`${workDir}/${TEST_LOCALE}.json`); |  | ||||||
|       } else if (lang !== "en") { |  | ||||||
|         mergeFiles.push(`${inFrontendDir}/${lang}.json`); |  | ||||||
|         if (mergeBackend) { |  | ||||||
|           mergeFiles.push(`${inBackendDir}/${lang}.json`); |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     const mergeStream = gulp |  | ||||||
|       .src(mergeFiles, { allowEmpty: true }) |  | ||||||
|       .pipe(new MergeJSON(locale, enMaster, emptyReviver)); |  | ||||||
|     mergesFinished.push(finished(mergeStream)); |  | ||||||
|     mergeStream.pipe(hashStream, { end: false }); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // Wait for all merges to finish, then it's safe to end writing to the |  | ||||||
|   // downstream pipeline and wait for all fragments to finish writing. |  | ||||||
|   await Promise.all(mergesFinished); |  | ||||||
|   hashStream.end(); |  | ||||||
|   await finished(fragmentsStream); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const writeTranslationMetaData = () => |  | ||||||
|   gulp |  | ||||||
|     .src([`${paths.translations_src}/translationMetadata.json`]) |  | ||||||
|     .pipe( |     .pipe( | ||||||
|       new CustomJSON((meta) => { |       merge({ | ||||||
|         // Add the test translation in development. |         fileName: "translationMaster.json", | ||||||
|         if (!env.isProdBuild()) { |  | ||||||
|           meta[TEST_LOCALE] = { nativeName: "Translation Test" }; |  | ||||||
|         } |  | ||||||
|         // Filter out locales without a native name, and add the hashes. |  | ||||||
|         for (const locale of Object.keys(meta)) { |  | ||||||
|           if (!meta[locale].nativeName) { |  | ||||||
|             meta[locale] = undefined; |  | ||||||
|             console.warn( |  | ||||||
|               `Skipping locale ${locale} because native name is not translated.` |  | ||||||
|             ); |  | ||||||
|           } else { |  | ||||||
|             meta[locale].hash = HASHES.get(locale); |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|         return { |  | ||||||
|           fragments: FRAGMENTS.filter(panelFragment), |  | ||||||
|           translations: meta, |  | ||||||
|         }; |  | ||||||
|       }) |       }) | ||||||
|     ) |     ) | ||||||
|     .pipe(gulp.dest(workDir)); |     .pipe(gulp.dest(workDir)); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("build-merged-translations", function () { | ||||||
|  |   return gulp | ||||||
|  |     .src([inFrontendDir + "/*.json", workDir + "/test.json"], { | ||||||
|  |       allowEmpty: true, | ||||||
|  |     }) | ||||||
|  |     .pipe( | ||||||
|  |       transform(function (data, file) { | ||||||
|  |         return lokaliseTransform(data, data, file); | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |     .pipe( | ||||||
|  |       foreach(function (stream, file) { | ||||||
|  |         // For each language generate a merged json file. It begins with the master | ||||||
|  |         // translation as a failsafe for untranslated strings, and merges all parent | ||||||
|  |         // tags into one file for each specific subtag | ||||||
|  |         // | ||||||
|  |         // TODO: This is a naive interpretation of BCP47 that should be improved. | ||||||
|  |         //       Will be OK for now as long as we don't have anything more complicated | ||||||
|  |         //       than a base translation + region. | ||||||
|  |         const tr = path.basename(file.history[0], ".json"); | ||||||
|  |         const subtags = tr.split("-"); | ||||||
|  |         const src = [workDir + "/translationMaster.json"]; | ||||||
|  |         for (let i = 1; i <= subtags.length; i++) { | ||||||
|  |           const lang = subtags.slice(0, i).join("-"); | ||||||
|  |           if (lang === "test") { | ||||||
|  |             src.push(workDir + "/test.json"); | ||||||
|  |           } else if (lang !== "en") { | ||||||
|  |             src.push(inFrontendDir + "/" + lang + ".json"); | ||||||
|  |             if (mergeBackend) { | ||||||
|  |               src.push(inBackendDir + "/" + lang + ".json"); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         return gulp | ||||||
|  |           .src(src, { allowEmpty: true }) | ||||||
|  |           .pipe(transform((data) => emptyFilter(data))) | ||||||
|  |           .pipe( | ||||||
|  |             merge({ | ||||||
|  |               fileName: tr + ".json", | ||||||
|  |             }) | ||||||
|  |           ) | ||||||
|  |           .pipe(gulp.dest(fullDir)); | ||||||
|  |       }) | ||||||
|  |     ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | var taskName; | ||||||
|  |  | ||||||
|  | const splitTasks = []; | ||||||
|  | TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||||
|  |   taskName = "build-translation-fragment-" + fragment; | ||||||
|  |   gulp.task(taskName, function () { | ||||||
|  |     // Return only the translations for this fragment. | ||||||
|  |     return gulp | ||||||
|  |       .src(fullDir + "/*.json") | ||||||
|  |       .pipe( | ||||||
|  |         transform((data) => ({ | ||||||
|  |           ui: { | ||||||
|  |             panel: { | ||||||
|  |               [fragment]: data.ui.panel[fragment], | ||||||
|  |             }, | ||||||
|  |           }, | ||||||
|  |         })) | ||||||
|  |       ) | ||||||
|  |       .pipe(gulp.dest(workDir + "/" + fragment)); | ||||||
|  |   }); | ||||||
|  |   splitTasks.push(taskName); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | taskName = "build-translation-core"; | ||||||
|  | gulp.task(taskName, function () { | ||||||
|  |   // Remove the fragment translations from the core translation. | ||||||
|  |   return gulp | ||||||
|  |     .src(fullDir + "/*.json") | ||||||
|  |     .pipe( | ||||||
|  |       transform((data, file) => { | ||||||
|  |         TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||||
|  |           delete data.ui.panel[fragment]; | ||||||
|  |         }); | ||||||
|  |         return data; | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |     .pipe(gulp.dest(coreDir)); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | splitTasks.push(taskName); | ||||||
|  |  | ||||||
|  | gulp.task("build-flattened-translations", function () { | ||||||
|  |   // Flatten the split versions of our translations, and move them into outDir | ||||||
|  |   return gulp | ||||||
|  |     .src( | ||||||
|  |       TRANSLATION_FRAGMENTS.map( | ||||||
|  |         (fragment) => workDir + "/" + fragment + "/*.json" | ||||||
|  |       ).concat(coreDir + "/*.json"), | ||||||
|  |       { base: workDir } | ||||||
|  |     ) | ||||||
|  |     .pipe( | ||||||
|  |       transform(function (data) { | ||||||
|  |         // Polymer.AppLocalizeBehavior requires flattened json | ||||||
|  |         return flatten(data); | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |     .pipe( | ||||||
|  |       rename((filePath) => { | ||||||
|  |         if (filePath.dirname === "core") { | ||||||
|  |           filePath.dirname = ""; | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |     ) | ||||||
|  |     .pipe(gulp.dest(outDir)); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | const fingerprints = {}; | ||||||
|  |  | ||||||
|  | gulp.task( | ||||||
|  |   "build-translation-fingerprints", | ||||||
|  |   function fingerprintTranslationFiles() { | ||||||
|  |     // Fingerprint full file of each language | ||||||
|  |     const files = fs.readdirSync(fullDir); | ||||||
|  |  | ||||||
|  |     for (let i = 0; i < files.length; i++) { | ||||||
|  |       fingerprints[files[i].split(".")[0]] = { | ||||||
|  |         // In dev we create fake hashes | ||||||
|  |         hash: env.isProdBuild() | ||||||
|  |           ? crypto | ||||||
|  |               .createHash("md5") | ||||||
|  |               .update(fs.readFileSync(path.join(fullDir, files[i]), "utf-8")) | ||||||
|  |               .digest("hex") | ||||||
|  |           : "dev", | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     mapFiles(outDir, ".json", (filename) => { | ||||||
|  |       const parsed = path.parse(filename); | ||||||
|  |  | ||||||
|  |       // nl.json -> nl-<hash>.json | ||||||
|  |       if (!(parsed.name in fingerprints)) { | ||||||
|  |         throw new Error(`Unable to find hash for ${filename}`); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       fs.renameSync( | ||||||
|  |         filename, | ||||||
|  |         `${parsed.dir}/${parsed.name}-${fingerprints[parsed.name].hash}${ | ||||||
|  |           parsed.ext | ||||||
|  |         }` | ||||||
|  |       ); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const stream = source("translationFingerprints.json"); | ||||||
|  |     stream.write(JSON.stringify(fingerprints)); | ||||||
|  |     process.nextTick(() => stream.end()); | ||||||
|  |     return stream.pipe(vinylBuffer()).pipe(gulp.dest(workDir)); | ||||||
|  |   } | ||||||
|  | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "build-translations", |   "build-translations", | ||||||
|   gulp.series( |   gulp.series( | ||||||
|     gulp.parallel( |     "clean-translations", | ||||||
|       "fetch-nightly-translations", |     "ensure-translations-build-dir", | ||||||
|       gulp.series("clean-translations", makeWorkDir) |     env.isProdBuild() ? (done) => done() : "create-test-translation", | ||||||
|     ), |     "build-master-translation", | ||||||
|     createTestTranslation, |     "build-merged-translations", | ||||||
|     createMasterTranslation, |     gulp.parallel(...splitTasks), | ||||||
|     createTranslations, |     "build-flattened-translations", | ||||||
|     writeTranslationMetaData |     "build-translation-fingerprints", | ||||||
|  |     function writeMetadata() { | ||||||
|  |       return gulp | ||||||
|  |         .src( | ||||||
|  |           [ | ||||||
|  |             path.join(paths.translations_src, "translationMetadata.json"), | ||||||
|  |             workDir + "/testMetadata.json", | ||||||
|  |             workDir + "/translationFingerprints.json", | ||||||
|  |           ], | ||||||
|  |           { allowEmpty: true } | ||||||
|  |         ) | ||||||
|  |         .pipe(merge({})) | ||||||
|  |         .pipe( | ||||||
|  |           transform(function (data) { | ||||||
|  |             const newData = {}; | ||||||
|  |             Object.entries(data).forEach(([key, value]) => { | ||||||
|  |               // Filter out translations without native name. | ||||||
|  |               if (data[key].nativeName) { | ||||||
|  |                 newData[key] = data[key]; | ||||||
|  |               } else { | ||||||
|  |                 console.warn( | ||||||
|  |                   `Skipping language ${key}. Native name was not translated.` | ||||||
|  |                 ); | ||||||
|  |               } | ||||||
|  |               if (data[key]) newData[key] = value; | ||||||
|  |             }); | ||||||
|  |             return newData; | ||||||
|  |           }) | ||||||
|  |         ) | ||||||
|  |         .pipe( | ||||||
|  |           transform((data) => ({ | ||||||
|  |             fragments: TRANSLATION_FRAGMENTS, | ||||||
|  |             translations: data, | ||||||
|  |           })) | ||||||
|  |         ) | ||||||
|  |         .pipe(rename("translationMetadata.json")) | ||||||
|  |         .pipe(gulp.dest(workDir)); | ||||||
|  |     } | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( |  | ||||||
|   "build-supervisor-translations", |  | ||||||
|   gulp.series(toggleSupervisorFragment, "build-translations") |  | ||||||
| ); |  | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| import gulp from "gulp"; | // Tasks to run Rollup | ||||||
| import { startDevServer } from "@web/dev-server"; | const gulp = require("gulp"); | ||||||
|  | const { startDevServer } = require("@web/dev-server"); | ||||||
|  |  | ||||||
| gulp.task("wds-watch-app", async () => { | gulp.task("wds-watch-app", () => { | ||||||
|   startDevServer({ |   startDevServer({ | ||||||
|     config: { |     config: { | ||||||
|       watch: true, |       watch: true, | ||||||
|   | |||||||
| @@ -1,33 +1,23 @@ | |||||||
| // Tasks to run webpack. | // Tasks to run webpack. | ||||||
|  | const gulp = require("gulp"); | ||||||
| import fs from "fs"; | const webpack = require("webpack"); | ||||||
| import path from "path"; | const WebpackDevServer = require("webpack-dev-server"); | ||||||
| import log from "fancy-log"; | const log = require("fancy-log"); | ||||||
| import gulp from "gulp"; | const path = require("path"); | ||||||
| import webpack from "webpack"; | const paths = require("../paths"); | ||||||
| import WebpackDevServer from "webpack-dev-server"; | const { | ||||||
| import env from "../env.cjs"; |  | ||||||
| import paths from "../paths.cjs"; |  | ||||||
| import { |  | ||||||
|   createAppConfig, |   createAppConfig, | ||||||
|   createCastConfig, |  | ||||||
|   createDemoConfig, |   createDemoConfig, | ||||||
|   createGalleryConfig, |   createCastConfig, | ||||||
|   createHassioConfig, |   createHassioConfig, | ||||||
| } from "../webpack.cjs"; |   createGalleryConfig, | ||||||
|  | } = require("../webpack"); | ||||||
|  |  | ||||||
| const bothBuilds = (createConfigFunc, params) => [ | const bothBuilds = (createConfigFunc, params) => [ | ||||||
|   createConfigFunc({ ...params, latestBuild: true }), |   createConfigFunc({ ...params, latestBuild: true }), | ||||||
|   createConfigFunc({ ...params, latestBuild: false }), |   createConfigFunc({ ...params, latestBuild: false }), | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const isWsl = |  | ||||||
|   fs.existsSync("/proc/version") && |  | ||||||
|   fs |  | ||||||
|     .readFileSync("/proc/version", "utf-8") |  | ||||||
|     .toLocaleLowerCase() |  | ||||||
|     .includes("microsoft"); |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @param {{ |  * @param {{ | ||||||
|  *   compiler: import("webpack").Compiler, |  *   compiler: import("webpack").Compiler, | ||||||
| @@ -36,34 +26,26 @@ const isWsl = | |||||||
|  *   listenHost?: string |  *   listenHost?: string | ||||||
|  * }} |  * }} | ||||||
|  */ |  */ | ||||||
| const runDevServer = async ({ | const runDevServer = ({ | ||||||
|   compiler, |   compiler, | ||||||
|   contentBase, |   contentBase, | ||||||
|   port, |   port, | ||||||
|   listenHost = undefined, |   listenHost = "localhost", | ||||||
| }) => { | }) => | ||||||
|   if (listenHost === undefined) { |   new WebpackDevServer(compiler, { | ||||||
|     // For dev container, we need to listen on all hosts |     open: true, | ||||||
|     listenHost = env.isDevContainer() ? "0.0.0.0" : "localhost"; |     watchContentBase: true, | ||||||
|   } |     contentBase, | ||||||
|   const server = new WebpackDevServer( |   }).listen(port, listenHost, function (err) { | ||||||
|     { |     if (err) { | ||||||
|       hot: false, |       throw err; | ||||||
|       open: true, |     } | ||||||
|       host: listenHost, |     // Server listening | ||||||
|       port, |     log( | ||||||
|       static: { |       "[webpack-dev-server]", | ||||||
|         directory: contentBase, |       `Project is running at http://localhost:${port}` | ||||||
|         watch: true, |     ); | ||||||
|       }, |   }); | ||||||
|     }, |  | ||||||
|     compiler |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   await server.start(); |  | ||||||
|   // Server listening |  | ||||||
|   log("[webpack-dev-server]", `Project is running at http://localhost:${port}`); |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const doneHandler = (done) => (err, stats) => { | const doneHandler = (done) => (err, stats) => { | ||||||
|   if (err) { |   if (err) { | ||||||
| @@ -96,11 +78,10 @@ const prodBuild = (conf) => | |||||||
|  |  | ||||||
| gulp.task("webpack-watch-app", () => { | gulp.task("webpack-watch-app", () => { | ||||||
|   // This command will run forever because we don't close compiler |   // This command will run forever because we don't close compiler | ||||||
|   webpack( |   webpack(createAppConfig({ isProdBuild: false, latestBuild: true })).watch( | ||||||
|     process.env.ES5 |     { ignored: /build-translations/ }, | ||||||
|       ? bothBuilds(createAppConfig, { isProdBuild: false }) |     doneHandler() | ||||||
|       : createAppConfig({ isProdBuild: false, latestBuild: true }) |   ); | ||||||
|   ).watch({ poll: isWsl }, doneHandler()); |  | ||||||
|   gulp.watch( |   gulp.watch( | ||||||
|     path.join(paths.translations_src, "en.json"), |     path.join(paths.translations_src, "en.json"), | ||||||
|     gulp.series("build-translations", "copy-translations-app") |     gulp.series("build-translations", "copy-translations-app") | ||||||
| @@ -111,21 +92,17 @@ gulp.task("webpack-prod-app", () => | |||||||
|   prodBuild( |   prodBuild( | ||||||
|     bothBuilds(createAppConfig, { |     bothBuilds(createAppConfig, { | ||||||
|       isProdBuild: true, |       isProdBuild: true, | ||||||
|       isStatsBuild: env.isStatsBuild(), |  | ||||||
|       isTestBuild: env.isTestBuild(), |  | ||||||
|     }) |     }) | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task("webpack-dev-server-demo", () => | gulp.task("webpack-dev-server-demo", () => { | ||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     compiler: webpack( |     compiler: webpack(bothBuilds(createDemoConfig, { isProdBuild: false })), | ||||||
|       createDemoConfig({ isProdBuild: false, latestBuild: true }) |  | ||||||
|     ), |  | ||||||
|     contentBase: paths.demo_output_root, |     contentBase: paths.demo_output_root, | ||||||
|     port: 8090, |     port: 8090, | ||||||
|   }) |   }); | ||||||
| ); | }); | ||||||
|  |  | ||||||
| gulp.task("webpack-prod-demo", () => | gulp.task("webpack-prod-demo", () => | ||||||
|   prodBuild( |   prodBuild( | ||||||
| @@ -135,17 +112,15 @@ gulp.task("webpack-prod-demo", () => | |||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task("webpack-dev-server-cast", () => | gulp.task("webpack-dev-server-cast", () => { | ||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     compiler: webpack( |     compiler: webpack(bothBuilds(createCastConfig, { isProdBuild: false })), | ||||||
|       createCastConfig({ isProdBuild: false, latestBuild: true }) |  | ||||||
|     ), |  | ||||||
|     contentBase: paths.cast_output_root, |     contentBase: paths.cast_output_root, | ||||||
|     port: 8080, |     port: 8080, | ||||||
|     // Accessible from the network, because that's how Cast hits it. |     // Accessible from the network, because that's how Cast hits it. | ||||||
|     listenHost: "0.0.0.0", |     listenHost: "0.0.0.0", | ||||||
|   }) |   }); | ||||||
| ); | }); | ||||||
|  |  | ||||||
| gulp.task("webpack-prod-cast", () => | gulp.task("webpack-prod-cast", () => | ||||||
|   prodBuild( |   prodBuild( | ||||||
| @@ -162,34 +137,25 @@ gulp.task("webpack-watch-hassio", () => { | |||||||
|       isProdBuild: false, |       isProdBuild: false, | ||||||
|       latestBuild: true, |       latestBuild: true, | ||||||
|     }) |     }) | ||||||
|   ).watch({ ignored: /build/, poll: isWsl }, doneHandler()); |   ).watch({}, doneHandler()); | ||||||
|  |  | ||||||
|   gulp.watch( |  | ||||||
|     path.join(paths.translations_src, "en.json"), |  | ||||||
|     gulp.series("build-supervisor-translations", "copy-translations-supervisor") |  | ||||||
|   ); |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("webpack-prod-hassio", () => | gulp.task("webpack-prod-hassio", () => | ||||||
|   prodBuild( |   prodBuild( | ||||||
|     bothBuilds(createHassioConfig, { |     bothBuilds(createHassioConfig, { | ||||||
|       isProdBuild: true, |       isProdBuild: true, | ||||||
|       isStatsBuild: env.isStatsBuild(), |  | ||||||
|       isTestBuild: env.isTestBuild(), |  | ||||||
|     }) |     }) | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task("webpack-dev-server-gallery", () => | gulp.task("webpack-dev-server-gallery", () => { | ||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     compiler: webpack( |     // We don't use the es5 build, but the dev server will fuck up the publicPath if we don't | ||||||
|       createGalleryConfig({ isProdBuild: false, latestBuild: true }) |     compiler: webpack(bothBuilds(createGalleryConfig, { isProdBuild: false })), | ||||||
|     ), |  | ||||||
|     contentBase: paths.gallery_output_root, |     contentBase: paths.gallery_output_root, | ||||||
|     port: 8100, |     port: 8100, | ||||||
|     listenHost: "0.0.0.0", |   }); | ||||||
|   }) | }); | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task("webpack-prod-gallery", () => | gulp.task("webpack-prod-gallery", () => | ||||||
|   prodBuild( |   prodBuild( | ||||||
|   | |||||||
| @@ -1,83 +0,0 @@ | |||||||
| #!/usr/bin/env node |  | ||||||
| // Script to print Babel plugins and Core JS polyfills that will be used by browserslist environments |  | ||||||
|  |  | ||||||
| import { version as babelVersion } from "@babel/core"; |  | ||||||
| import presetEnv from "@babel/preset-env"; |  | ||||||
| import compilationTargets from "@babel/helper-compilation-targets"; |  | ||||||
| import coreJSCompat from "core-js-compat"; |  | ||||||
| import { logPlugin } from "@babel/preset-env/lib/debug.js"; |  | ||||||
| // eslint-disable-next-line import/no-relative-packages |  | ||||||
| import shippedPolyfills from "../node_modules/babel-plugin-polyfill-corejs3/lib/shipped-proposals.js"; |  | ||||||
| import { babelOptions } from "./bundle.cjs"; |  | ||||||
|  |  | ||||||
| const detailsOpen = (heading) => |  | ||||||
|   `<details>\n<summary><h4>${heading}</h4></summary>\n`; |  | ||||||
| const detailsClose = "</details>\n"; |  | ||||||
|  |  | ||||||
| const dummyAPI = { |  | ||||||
|   version: babelVersion, |  | ||||||
|   assertVersion: () => {}, |  | ||||||
|   caller: (callback) => |  | ||||||
|     callback({ |  | ||||||
|       name: "Dummy Bundler", |  | ||||||
|       supportsStaticESM: true, |  | ||||||
|       supportsDynamicImport: true, |  | ||||||
|       supportsTopLevelAwait: true, |  | ||||||
|       supportsExportNamespaceFrom: true, |  | ||||||
|     }), |  | ||||||
|   targets: () => ({}), |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Generate filter function based on proposal/method inputs |  | ||||||
| // Copied and adapted from babel-plugin-polyfill-corejs3/esm/index.mjs |  | ||||||
| const polyfillFilter = (method, proposals, shippedProposals) => (name) => { |  | ||||||
|   if (proposals || method === "entry-global") return true; |  | ||||||
|   if (shippedProposals && shippedPolyfills.default.has(name)) { |  | ||||||
|     return true; |  | ||||||
|   } |  | ||||||
|   if (name.startsWith("esnext.")) { |  | ||||||
|     const esName = `es.${name.slice(7)}`; |  | ||||||
|     // If its imaginative esName is not in latest compat data, it means the proposal is not stage 4 |  | ||||||
|     return esName in coreJSCompat.data; |  | ||||||
|   } |  | ||||||
|   return true; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| // Log the plugins and polyfills for each build environment |  | ||||||
| for (const buildType of ["Modern", "Legacy"]) { |  | ||||||
|   const browserslistEnv = buildType.toLowerCase(); |  | ||||||
|   const babelOpts = babelOptions({ latestBuild: browserslistEnv === "modern" }); |  | ||||||
|   const presetEnvOpts = babelOpts.presets[0][1]; |  | ||||||
|  |  | ||||||
|   // Invoking preset-env in debug mode will log the included plugins |  | ||||||
|   console.log(detailsOpen(`${buildType} Build Babel Plugins`)); |  | ||||||
|   presetEnv.default(dummyAPI, { |  | ||||||
|     ...presetEnvOpts, |  | ||||||
|     browserslistEnv, |  | ||||||
|     debug: true, |  | ||||||
|   }); |  | ||||||
|   console.log(detailsClose); |  | ||||||
|  |  | ||||||
|   // Manually log the Core-JS polyfills using the same technique |  | ||||||
|   if (presetEnvOpts.useBuiltIns) { |  | ||||||
|     console.log(detailsOpen(`${buildType} Build Core-JS Polyfills`)); |  | ||||||
|     const targets = compilationTargets.default(babelOpts?.targets, { |  | ||||||
|       browserslistEnv, |  | ||||||
|     }); |  | ||||||
|     const polyfillList = coreJSCompat({ targets }).list.filter( |  | ||||||
|       polyfillFilter( |  | ||||||
|         `${presetEnvOpts.useBuiltIns}-global`, |  | ||||||
|         presetEnvOpts?.corejs?.proposals, |  | ||||||
|         presetEnvOpts?.shippedProposals |  | ||||||
|       ) |  | ||||||
|     ); |  | ||||||
|     console.log( |  | ||||||
|       "The following %i polyfills may be injected by Babel:\n", |  | ||||||
|       polyfillList.length |  | ||||||
|     ); |  | ||||||
|     for (const polyfill of polyfillList) { |  | ||||||
|       logPlugin(polyfill, targets, coreJSCompat.data); |  | ||||||
|     } |  | ||||||
|     console.log(detailsClose); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -25,7 +25,6 @@ module.exports = { | |||||||
|   cast_output_es5: path.resolve(__dirname, "../cast/dist/frontend_es5"), |   cast_output_es5: path.resolve(__dirname, "../cast/dist/frontend_es5"), | ||||||
| 
 | 
 | ||||||
|   gallery_dir: path.resolve(__dirname, "../gallery"), |   gallery_dir: path.resolve(__dirname, "../gallery"), | ||||||
|   gallery_build: path.resolve(__dirname, "../gallery/build"), |  | ||||||
|   gallery_output_root: path.resolve(__dirname, "../gallery/dist"), |   gallery_output_root: path.resolve(__dirname, "../gallery/dist"), | ||||||
|   gallery_output_latest: path.resolve( |   gallery_output_latest: path.resolve( | ||||||
|     __dirname, |     __dirname, | ||||||
| @@ -35,7 +34,6 @@ module.exports = { | |||||||
| 
 | 
 | ||||||
|   hassio_dir: path.resolve(__dirname, "../hassio"), |   hassio_dir: path.resolve(__dirname, "../hassio"), | ||||||
|   hassio_output_root: path.resolve(__dirname, "../hassio/build"), |   hassio_output_root: path.resolve(__dirname, "../hassio/build"), | ||||||
|   hassio_output_static: path.resolve(__dirname, "../hassio/build/static"), |  | ||||||
|   hassio_output_latest: path.resolve( |   hassio_output_latest: path.resolve( | ||||||
|     __dirname, |     __dirname, | ||||||
|     "../hassio/build/frontend_latest" |     "../hassio/build/frontend_latest" | ||||||
| @@ -81,13 +81,13 @@ module.exports = function (opts = {}) { | |||||||
|         opts.workerRegexp.flags |         opts.workerRegexp.flags | ||||||
|       ); |       ); | ||||||
|       if (!workerRegexp.test(code)) { |       if (!workerRegexp.test(code)) { | ||||||
|         return undefined; |         return; | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       const ms = new MagicString(code); |       const ms = new MagicString(code); | ||||||
|       // Reset the regexp
 |       // Reset the regexp
 | ||||||
|       workerRegexp.lastIndex = 0; |       workerRegexp.lastIndex = 0; | ||||||
|       for (;;) { |       while (true) { | ||||||
|         const match = workerRegexp.exec(code); |         const match = workerRegexp.exec(code); | ||||||
|         if (!match) { |         if (!match) { | ||||||
|           break; |           break; | ||||||
| @@ -98,12 +98,11 @@ module.exports = function (opts = {}) { | |||||||
|         // Parse the optional options object
 |         // Parse the optional options object
 | ||||||
|         if (match[3] && match[3].length > 0) { |         if (match[3] && match[3].length > 0) { | ||||||
|           // FIXME: ooooof!
 |           // FIXME: ooooof!
 | ||||||
|           // eslint-disable-next-line @typescript-eslint/no-implied-eval
 |  | ||||||
|           optionsObject = new Function(`return ${match[3].slice(1)};`)(); |           optionsObject = new Function(`return ${match[3].slice(1)};`)(); | ||||||
|         } |         } | ||||||
|         delete optionsObject.type; |         delete optionsObject.type; | ||||||
| 
 | 
 | ||||||
|         if (!/^.*\//.test(workerFile)) { |         if (!new RegExp("^.*/").test(workerFile)) { | ||||||
|           this.warn( |           this.warn( | ||||||
|             `Paths passed to the Worker constructor must be relative or absolute, i.e. start with /, ./ or ../ (just like dynamic import!). Ignoring "${workerFile}".` |             `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
 |         // 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; |         const resolvedWorkerFile = (await this.resolve(workerFile, id)).id; | ||||||
|         let chunkRefId; |         let chunkRefId; | ||||||
|         if (resolvedWorkerFile in refIds) { |         if (resolvedWorkerFile in refIds) { | ||||||
|           chunkRefId = refIds[resolvedWorkerFile]; |           chunkRefId = refIds[resolvedWorkerFile]; | ||||||
|         } else { |         } else { | ||||||
|           this.addWatchFile(resolvedWorkerFile); |           this.addWatchFile(resolvedWorkerFile); | ||||||
|           // eslint-disable-next-line no-await-in-loop
 |  | ||||||
|           const source = await getBundledWorker( |           const source = await getBundledWorker( | ||||||
|             resolvedWorkerFile, |             resolvedWorkerFile, | ||||||
|             rollupOptions |             rollupOptions | ||||||
| @@ -1,146 +0,0 @@ | |||||||
| 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 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 bundle = require("./bundle.cjs"); |  | ||||||
| const paths = require("./paths.cjs"); |  | ||||||
|  |  | ||||||
| const extensions = [".js", ".ts"]; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * @param {Object} arg |  | ||||||
|  * @param { import("rollup").InputOption } arg.input |  | ||||||
|  */ |  | ||||||
| const createRollupConfig = ({ |  | ||||||
|   entry, |  | ||||||
|   outputPath, |  | ||||||
|   defineOverlay, |  | ||||||
|   isProdBuild, |  | ||||||
|   latestBuild, |  | ||||||
|   isStatsBuild, |  | ||||||
|   publicPath, |  | ||||||
|   dontHash, |  | ||||||
|   isWDS, |  | ||||||
| }) => ({ |  | ||||||
|   /** |  | ||||||
|    * @type { import("rollup").InputOptions } |  | ||||||
|    */ |  | ||||||
|   inputOptions: { |  | ||||||
|     input: entry, |  | ||||||
|     // Some entry points contain no JavaScript. This setting silences a warning about that. |  | ||||||
|     // https://rollupjs.org/configuration-options/#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" |  | ||||||
|             ) |  | ||||||
|           ), |  | ||||||
|       }), |  | ||||||
|       resolve({ |  | ||||||
|         extensions, |  | ||||||
|         preferBuiltins: false, |  | ||||||
|         browser: true, |  | ||||||
|         rootDir: paths.polymer_dir, |  | ||||||
|       }), |  | ||||||
|       commonjs(), |  | ||||||
|       json(), |  | ||||||
|       babel({ |  | ||||||
|         ...bundle.babelOptions({ latestBuild, isProdBuild }), |  | ||||||
|         extensions, |  | ||||||
|         babelHelpers: isWDS ? "inline" : "bundled", |  | ||||||
|       }), |  | ||||||
|       string({ |  | ||||||
|         // Import certain extensions as strings |  | ||||||
|         include: [path.join(paths.polymer_dir, "node_modules/**/*.css")], |  | ||||||
|       }), |  | ||||||
|       replace(bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })), |  | ||||||
|       !isWDS && |  | ||||||
|         manifest({ |  | ||||||
|           publicPath, |  | ||||||
|         }), |  | ||||||
|       !isWDS && worker(), |  | ||||||
|       !isWDS && dontHashPlugin({ dontHash }), |  | ||||||
|       !isWDS && isProdBuild && terser(bundle.terserOptions({ latestBuild })), |  | ||||||
|       !isWDS && |  | ||||||
|         isStatsBuild && |  | ||||||
|         visualizer({ |  | ||||||
|           // https://github.com/btd/rollup-plugin-visualizer#options |  | ||||||
|           open: true, |  | ||||||
|           sourcemap: true, |  | ||||||
|         }), |  | ||||||
|     ].filter(Boolean), |  | ||||||
|   }, |  | ||||||
|   /** |  | ||||||
|    * @type { import("rollup").OutputOptions } |  | ||||||
|    */ |  | ||||||
|   outputOptions: { |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-dir |  | ||||||
|     dir: outputPath, |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-format |  | ||||||
|     format: latestBuild ? "es" : "systemjs", |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-externallivebindings |  | ||||||
|     externalLiveBindings: false, |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-entryfilenames |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-chunkfilenames |  | ||||||
|     // https://rollupjs.org/configuration-options/#output-assetfilenames |  | ||||||
|     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 |  | ||||||
|     sourcemap: isProdBuild ? true : "inline", |  | ||||||
|   }, |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) => |  | ||||||
|   createRollupConfig( |  | ||||||
|     bundle.config.app({ |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       isStatsBuild, |  | ||||||
|       isWDS, |  | ||||||
|     }) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
| const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => |  | ||||||
|   createRollupConfig( |  | ||||||
|     bundle.config.demo({ |  | ||||||
|       isProdBuild, |  | ||||||
|       latestBuild, |  | ||||||
|       isStatsBuild, |  | ||||||
|     }) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
| const createCastConfig = ({ isProdBuild, latestBuild }) => |  | ||||||
|   createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild })); |  | ||||||
|  |  | ||||||
| const createHassioConfig = ({ isProdBuild, latestBuild }) => |  | ||||||
|   createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild })); |  | ||||||
|  |  | ||||||
| const createGalleryConfig = ({ isProdBuild, latestBuild }) => |  | ||||||
|   createRollupConfig(bundle.config.gallery({ isProdBuild, latestBuild })); |  | ||||||
|  |  | ||||||
| module.exports = { |  | ||||||
|   createAppConfig, |  | ||||||
|   createDemoConfig, |  | ||||||
|   createCastConfig, |  | ||||||
|   createHassioConfig, |  | ||||||
|   createGalleryConfig, |  | ||||||
|   createRollupConfig, |  | ||||||
| }; |  | ||||||
							
								
								
									
										156
									
								
								build-scripts/rollup.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								build-scripts/rollup.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | |||||||
|  | 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").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"); | ||||||
|  | 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"); | ||||||
|  | const paths = require("./paths"); | ||||||
|  |  | ||||||
|  | const extensions = [".js", ".ts"]; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @param {Object} arg | ||||||
|  |  * @param { import("rollup").InputOption } arg.input | ||||||
|  |  */ | ||||||
|  | const createRollupConfig = ({ | ||||||
|  |   entry, | ||||||
|  |   outputPath, | ||||||
|  |   defineOverlay, | ||||||
|  |   isProdBuild, | ||||||
|  |   latestBuild, | ||||||
|  |   isStatsBuild, | ||||||
|  |   publicPath, | ||||||
|  |   dontHash, | ||||||
|  |   isWDS, | ||||||
|  | }) => { | ||||||
|  |   return { | ||||||
|  |     /** | ||||||
|  |      * @type { import("rollup").InputOptions } | ||||||
|  |      */ | ||||||
|  |     inputOptions: { | ||||||
|  |       input: entry, | ||||||
|  |       // Some entry points contain no JavaScript. This setting silences a warning about that. | ||||||
|  |       // https://rollupjs.org/guide/en/#preserveentrysignatures | ||||||
|  |       preserveEntrySignatures: false, | ||||||
|  |       plugins: [ | ||||||
|  |         ignore({ | ||||||
|  |           files: bundle.emptyPackages({ latestBuild }), | ||||||
|  |         }), | ||||||
|  |         resolve({ | ||||||
|  |           extensions, | ||||||
|  |           preferBuiltins: false, | ||||||
|  |           browser: true, | ||||||
|  |           rootDir: paths.polymer_dir, | ||||||
|  |         }), | ||||||
|  |         commonjs({ | ||||||
|  |           namedExports: { | ||||||
|  |             "js-yaml": ["safeDump", "safeLoad"], | ||||||
|  |           }, | ||||||
|  |         }), | ||||||
|  |         json(), | ||||||
|  |         babel({ | ||||||
|  |           ...bundle.babelOptions({ latestBuild }), | ||||||
|  |           extensions, | ||||||
|  |           exclude: bundle.babelExclude(), | ||||||
|  |           babelHelpers: isWDS ? "inline" : "bundled", | ||||||
|  |         }), | ||||||
|  |         string({ | ||||||
|  |           // Import certain extensions as strings | ||||||
|  |           include: [path.join(paths.polymer_dir, "node_modules/**/*.css")], | ||||||
|  |         }), | ||||||
|  |         replace( | ||||||
|  |           bundle.definedVars({ isProdBuild, latestBuild, defineOverlay }) | ||||||
|  |         ), | ||||||
|  |         !isWDS && | ||||||
|  |           manifest({ | ||||||
|  |             publicPath, | ||||||
|  |           }), | ||||||
|  |         !isWDS && worker(), | ||||||
|  |         !isWDS && dontHashPlugin({ dontHash }), | ||||||
|  |         !isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)), | ||||||
|  |         !isWDS && | ||||||
|  |           isStatsBuild && | ||||||
|  |           visualizer({ | ||||||
|  |             // https://github.com/btd/rollup-plugin-visualizer#options | ||||||
|  |             open: true, | ||||||
|  |             sourcemap: true, | ||||||
|  |           }), | ||||||
|  |       ].filter(Boolean), | ||||||
|  |     }, | ||||||
|  |     /** | ||||||
|  |      * @type { import("rollup").OutputOptions } | ||||||
|  |      */ | ||||||
|  |     outputOptions: { | ||||||
|  |       // https://rollupjs.org/guide/en/#outputdir | ||||||
|  |       dir: outputPath, | ||||||
|  |       // https://rollupjs.org/guide/en/#outputformat | ||||||
|  |       format: latestBuild ? "es" : "systemjs", | ||||||
|  |       // https://rollupjs.org/guide/en/#outputexternallivebindings | ||||||
|  |       externalLiveBindings: false, | ||||||
|  |       // 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/guide/en/#outputsourcemap | ||||||
|  |       sourcemap: isProdBuild ? true : "inline", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.app({ | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |       isWDS, | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.demo({ | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createCastConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild })); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createHassioConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild })); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createGalleryConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.gallery({ isProdBuild, latestBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  |   createAppConfig, | ||||||
|  |   createDemoConfig, | ||||||
|  |   createCastConfig, | ||||||
|  |   createHassioConfig, | ||||||
|  |   createGalleryConfig, | ||||||
|  | }; | ||||||
							
								
								
									
										16
									
								
								build-scripts/util.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								build-scripts/util.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | const path = require("path"); | ||||||
|  | const fs = require("fs"); | ||||||
|  |  | ||||||
|  | // Helper function to map recursively over files in a folder and it's subfolders | ||||||
|  | module.exports.mapFiles = function mapFiles(startPath, filter, mapFunc) { | ||||||
|  |   const files = fs.readdirSync(startPath); | ||||||
|  |   for (let i = 0; i < files.length; i++) { | ||||||
|  |     const filename = path.join(startPath, files[i]); | ||||||
|  |     const stat = fs.lstatSync(filename); | ||||||
|  |     if (stat.isDirectory()) { | ||||||
|  |       mapFiles(filename, filter, mapFunc); | ||||||
|  |     } else if (filename.indexOf(filter) >= 0) { | ||||||
|  |       mapFunc(filename); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | }; | ||||||
| @@ -1,292 +0,0 @@ | |||||||
| const { existsSync } = require("fs"); |  | ||||||
| const path = require("path"); |  | ||||||
| const webpack = require("webpack"); |  | ||||||
| const { StatsWriterPlugin } = require("webpack-stats-plugin"); |  | ||||||
| const filterStats = require("@bundle-stats/plugin-webpack-filter").default; |  | ||||||
| const TerserPlugin = require("terser-webpack-plugin"); |  | ||||||
| const { WebpackManifestPlugin } = require("webpack-manifest-plugin"); |  | ||||||
| const log = require("fancy-log"); |  | ||||||
| const WebpackBar = require("webpackbar"); |  | ||||||
| const { |  | ||||||
|   TransformAsyncModulesPlugin, |  | ||||||
| } = require("transform-async-modules-webpack-plugin"); |  | ||||||
| const { dependencies } = require("../package.json"); |  | ||||||
| const paths = require("./paths.cjs"); |  | ||||||
| const bundle = require("./bundle.cjs"); |  | ||||||
|  |  | ||||||
| class LogStartCompilePlugin { |  | ||||||
|   ignoredFirst = false; |  | ||||||
|  |  | ||||||
|   apply(compiler) { |  | ||||||
|     compiler.hooks.beforeCompile.tap("LogStartCompilePlugin", () => { |  | ||||||
|       if (!this.ignoredFirst) { |  | ||||||
|         this.ignoredFirst = true; |  | ||||||
|         return; |  | ||||||
|       } |  | ||||||
|       log("Changes detected. Starting compilation"); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const createWebpackConfig = ({ |  | ||||||
|   name, |  | ||||||
|   entry, |  | ||||||
|   outputPath, |  | ||||||
|   publicPath, |  | ||||||
|   defineOverlay, |  | ||||||
|   isProdBuild, |  | ||||||
|   latestBuild, |  | ||||||
|   isStatsBuild, |  | ||||||
|   isTestBuild, |  | ||||||
|   isHassioBuild, |  | ||||||
|   dontHash, |  | ||||||
| }) => { |  | ||||||
|   if (!dontHash) { |  | ||||||
|     dontHash = new Set(); |  | ||||||
|   } |  | ||||||
|   const ignorePackages = bundle.ignorePackages({ latestBuild }); |  | ||||||
|   return { |  | ||||||
|     name, |  | ||||||
|     mode: isProdBuild ? "production" : "development", |  | ||||||
|     target: `browserslist:${latestBuild ? "modern" : "legacy"}`, |  | ||||||
|     // 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" |  | ||||||
|         : "eval-cheap-module-source-map", |  | ||||||
|     entry, |  | ||||||
|     node: false, |  | ||||||
|     module: { |  | ||||||
|       rules: [ |  | ||||||
|         { |  | ||||||
|           test: /\.m?js$|\.ts$/, |  | ||||||
|           use: (info) => ({ |  | ||||||
|             loader: "babel-loader", |  | ||||||
|             options: { |  | ||||||
|               ...bundle.babelOptions({ |  | ||||||
|                 latestBuild, |  | ||||||
|                 isProdBuild, |  | ||||||
|                 isTestBuild, |  | ||||||
|                 sw: info.issuerLayer === "sw", |  | ||||||
|               }), |  | ||||||
|               cacheDirectory: !isProdBuild, |  | ||||||
|               cacheCompression: false, |  | ||||||
|             }, |  | ||||||
|           }), |  | ||||||
|           resolve: { |  | ||||||
|             fullySpecified: false, |  | ||||||
|           }, |  | ||||||
|           parser: { |  | ||||||
|             worker: ["*context.audioWorklet.addModule()", "..."], |  | ||||||
|           }, |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|           test: /\.css$/, |  | ||||||
|           type: "asset/source", |  | ||||||
|         }, |  | ||||||
|       ], |  | ||||||
|     }, |  | ||||||
|     optimization: { |  | ||||||
|       minimizer: [ |  | ||||||
|         new TerserPlugin({ |  | ||||||
|           parallel: true, |  | ||||||
|           extractComments: true, |  | ||||||
|           terserOptions: bundle.terserOptions({ latestBuild, isTestBuild }), |  | ||||||
|         }), |  | ||||||
|       ], |  | ||||||
|       moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", |  | ||||||
|       chunkIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", |  | ||||||
|       splitChunks: { |  | ||||||
|         // Disable splitting for web workers and worklets because imports of |  | ||||||
|         // external chunks are broken for: |  | ||||||
|         // - ESM output: https://github.com/webpack/webpack/issues/17014 |  | ||||||
|         // - Worklets use `importScripts`: https://github.com/webpack/webpack/issues/11543 |  | ||||||
|         chunks: (chunk) => |  | ||||||
|           !chunk.canBeInitial() && |  | ||||||
|           !new RegExp(`^.+-work${latestBuild ? "(?:let|er)" : "let"}$`).test( |  | ||||||
|             chunk.name |  | ||||||
|           ), |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|     plugins: [ |  | ||||||
|       !isStatsBuild && new WebpackBar({ fancy: !isProdBuild }), |  | ||||||
|       new WebpackManifestPlugin({ |  | ||||||
|         // Only include the JS of entrypoints |  | ||||||
|         filter: (file) => file.isInitial && !file.name.endsWith(".map"), |  | ||||||
|       }), |  | ||||||
|       new webpack.DefinePlugin( |  | ||||||
|         bundle.definedVars({ isProdBuild, latestBuild, defineOverlay }) |  | ||||||
|       ), |  | ||||||
|       new webpack.IgnorePlugin({ |  | ||||||
|         checkResource(resource, context) { |  | ||||||
|           // Only use ignore to intercept imports that we don't control |  | ||||||
|           // inside node_module dependencies. |  | ||||||
|           if ( |  | ||||||
|             !context.includes("/node_modules/") || |  | ||||||
|             // calling define.amd will call require("!!webpack amd options") |  | ||||||
|             resource.startsWith("!!webpack") || |  | ||||||
|             // loaded by webpack dev server but doesn't exist. |  | ||||||
|             resource === "webpack/hot" |  | ||||||
|           ) { |  | ||||||
|             return false; |  | ||||||
|           } |  | ||||||
|           let fullPath; |  | ||||||
|           try { |  | ||||||
|             fullPath = resource.startsWith(".") |  | ||||||
|               ? path.resolve(context, resource) |  | ||||||
|               : require.resolve(resource); |  | ||||||
|           } catch (err) { |  | ||||||
|             console.error( |  | ||||||
|               "Error in Home Assistant ignore plugin", |  | ||||||
|               resource, |  | ||||||
|               context |  | ||||||
|             ); |  | ||||||
|             throw err; |  | ||||||
|           } |  | ||||||
|  |  | ||||||
|           return ignorePackages.some((toIgnorePath) => |  | ||||||
|             fullPath.startsWith(toIgnorePath) |  | ||||||
|           ); |  | ||||||
|         }, |  | ||||||
|       }), |  | ||||||
|       new webpack.NormalModuleReplacementPlugin( |  | ||||||
|         new RegExp( |  | ||||||
|           bundle.emptyPackages({ latestBuild, isHassioBuild }).join("|") |  | ||||||
|         ), |  | ||||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") |  | ||||||
|       ), |  | ||||||
|       !isProdBuild && new LogStartCompilePlugin(), |  | ||||||
|       isProdBuild && |  | ||||||
|         new StatsWriterPlugin({ |  | ||||||
|           filename: path.relative( |  | ||||||
|             outputPath, |  | ||||||
|             path.join(paths.build_dir, "stats", `${name}.json`) |  | ||||||
|           ), |  | ||||||
|           stats: { assets: true, chunks: true, modules: true }, |  | ||||||
|           transform: (stats) => JSON.stringify(filterStats(stats)), |  | ||||||
|         }), |  | ||||||
|       !latestBuild && |  | ||||||
|         new TransformAsyncModulesPlugin({ |  | ||||||
|           browserslistEnv: "legacy", |  | ||||||
|           runtime: { version: dependencies["@babel/runtime"] }, |  | ||||||
|         }), |  | ||||||
|     ].filter(Boolean), |  | ||||||
|     resolve: { |  | ||||||
|       extensions: [".ts", ".js", ".json"], |  | ||||||
|       alias: { |  | ||||||
|         "lit/static-html$": "lit/static-html.js", |  | ||||||
|         "lit/decorators$": "lit/decorators.js", |  | ||||||
|         "lit/directive$": "lit/directive.js", |  | ||||||
|         "lit/directives/until$": "lit/directives/until.js", |  | ||||||
|         "lit/directives/class-map$": "lit/directives/class-map.js", |  | ||||||
|         "lit/directives/style-map$": "lit/directives/style-map.js", |  | ||||||
|         "lit/directives/if-defined$": "lit/directives/if-defined.js", |  | ||||||
|         "lit/directives/guard$": "lit/directives/guard.js", |  | ||||||
|         "lit/directives/cache$": "lit/directives/cache.js", |  | ||||||
|         "lit/directives/repeat$": "lit/directives/repeat.js", |  | ||||||
|         "lit/directives/live$": "lit/directives/live.js", |  | ||||||
|         "lit/polyfill-support$": "lit/polyfill-support.js", |  | ||||||
|         "@lit-labs/virtualizer/layouts/grid": |  | ||||||
|           "@lit-labs/virtualizer/layouts/grid.js", |  | ||||||
|         "@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver": |  | ||||||
|           "@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver.js", |  | ||||||
|         "@lit-labs/observers/resize-controller": |  | ||||||
|           "@lit-labs/observers/resize-controller.js", |  | ||||||
|       }, |  | ||||||
|     }, |  | ||||||
|     output: { |  | ||||||
|       module: latestBuild, |  | ||||||
|       filename: ({ chunk }) => |  | ||||||
|         !isProdBuild || isStatsBuild || dontHash.has(chunk.name) |  | ||||||
|           ? "[name].js" |  | ||||||
|           : "[name].[contenthash].js", |  | ||||||
|       chunkFilename: |  | ||||||
|         isProdBuild && !isStatsBuild ? "[name].[contenthash].js" : "[name].js", |  | ||||||
|       assetModuleFilename: |  | ||||||
|         isProdBuild && !isStatsBuild ? "[id].[contenthash][ext]" : "[id][ext]", |  | ||||||
|       crossOriginLoading: "use-credentials", |  | ||||||
|       hashFunction: "xxhash64", |  | ||||||
|       hashDigest: "base64url", |  | ||||||
|       hashDigestLength: 11, // full length of 64 bit base64url |  | ||||||
|       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 |  | ||||||
|       ...Object.fromEntries( |  | ||||||
|         ["", "Fallback"].map((v) => [ |  | ||||||
|           `devtool${v}ModuleFilenameTemplate`, |  | ||||||
|           !isTestBuild && isProdBuild |  | ||||||
|             ? (info) => { |  | ||||||
|                 if ( |  | ||||||
|                   !path.isAbsolute(info.absoluteResourcePath) || |  | ||||||
|                   !existsSync(info.resourcePath) || |  | ||||||
|                   info.resourcePath.startsWith("./node_modules") |  | ||||||
|                 ) { |  | ||||||
|                   // Source URLs are unknown for dependencies, so we use a relative URL with a |  | ||||||
|                   // non - existent top directory.  This results in a clean source tree in browser |  | ||||||
|                   // dev tools, and they stay happy getting 404s with valid requests. |  | ||||||
|                   return `/unknown${path.resolve("/", info.resourcePath)}`; |  | ||||||
|                 } |  | ||||||
|                 return new URL(info.resourcePath, bundle.sourceMapURL()).href; |  | ||||||
|               } |  | ||||||
|             : undefined, |  | ||||||
|         ]) |  | ||||||
|       ), |  | ||||||
|     }, |  | ||||||
|     experiments: { |  | ||||||
|       layers: true, |  | ||||||
|       outputModule: true, |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| const createAppConfig = ({ |  | ||||||
|   isProdBuild, |  | ||||||
|   latestBuild, |  | ||||||
|   isStatsBuild, |  | ||||||
|   isTestBuild, |  | ||||||
| }) => |  | ||||||
|   createWebpackConfig( |  | ||||||
|     bundle.config.app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild }) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
| const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => |  | ||||||
|   createWebpackConfig( |  | ||||||
|     bundle.config.demo({ 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 createGalleryConfig = ({ isProdBuild, latestBuild }) => |  | ||||||
|   createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild })); |  | ||||||
|  |  | ||||||
| module.exports = { |  | ||||||
|   createAppConfig, |  | ||||||
|   createDemoConfig, |  | ||||||
|   createCastConfig, |  | ||||||
|   createHassioConfig, |  | ||||||
|   createGalleryConfig, |  | ||||||
|   createWebpackConfig, |  | ||||||
| }; |  | ||||||
							
								
								
									
										181
									
								
								build-scripts/webpack.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								build-scripts/webpack.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,181 @@ | |||||||
|  | const webpack = require("webpack"); | ||||||
|  | const path = require("path"); | ||||||
|  | const TerserPlugin = require("terser-webpack-plugin"); | ||||||
|  | const ManifestPlugin = require("webpack-manifest-plugin"); | ||||||
|  | const paths = require("./paths.js"); | ||||||
|  | const bundle = require("./bundle"); | ||||||
|  | const log = require("fancy-log"); | ||||||
|  |  | ||||||
|  | class LogStartCompilePlugin { | ||||||
|  |   ignoredFirst = false; | ||||||
|  |  | ||||||
|  |   apply(compiler) { | ||||||
|  |     compiler.hooks.beforeCompile.tap("LogStartCompilePlugin", () => { | ||||||
|  |       if (!this.ignoredFirst) { | ||||||
|  |         this.ignoredFirst = true; | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       log("Changes detected. Starting compilation"); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const createWebpackConfig = ({ | ||||||
|  |   entry, | ||||||
|  |   outputPath, | ||||||
|  |   publicPath, | ||||||
|  |   defineOverlay, | ||||||
|  |   isProdBuild, | ||||||
|  |   latestBuild, | ||||||
|  |   isStatsBuild, | ||||||
|  |   dontHash, | ||||||
|  | }) => { | ||||||
|  |   if (!dontHash) { | ||||||
|  |     dontHash = new Set(); | ||||||
|  |   } | ||||||
|  |   const ignorePackages = bundle.ignorePackages({ latestBuild }); | ||||||
|  |   return { | ||||||
|  |     mode: isProdBuild ? "production" : "development", | ||||||
|  |     target: ["web", latestBuild ? "es2017" : "es5"], | ||||||
|  |     devtool: isProdBuild | ||||||
|  |       ? "cheap-module-source-map" | ||||||
|  |       : "eval-cheap-module-source-map", | ||||||
|  |     entry, | ||||||
|  |     node: false, | ||||||
|  |     module: { | ||||||
|  |       rules: [ | ||||||
|  |         { | ||||||
|  |           test: /\.m?js$|\.ts$/, | ||||||
|  |           exclude: bundle.babelExclude(), | ||||||
|  |           use: { | ||||||
|  |             loader: "babel-loader", | ||||||
|  |             options: bundle.babelOptions({ latestBuild }), | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           test: /\.css$/, | ||||||
|  |           use: "raw-loader", | ||||||
|  |         }, | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     optimization: { | ||||||
|  |       minimizer: [ | ||||||
|  |         new TerserPlugin({ | ||||||
|  |           parallel: true, | ||||||
|  |           extractComments: true, | ||||||
|  |           terserOptions: bundle.terserOptions(latestBuild), | ||||||
|  |         }), | ||||||
|  |       ], | ||||||
|  |     }, | ||||||
|  |     plugins: [ | ||||||
|  |       new ManifestPlugin({ | ||||||
|  |         // Only include the JS of entrypoints | ||||||
|  |         filter: (file) => file.isInitial && !file.name.endsWith(".map"), | ||||||
|  |       }), | ||||||
|  |       new webpack.DefinePlugin( | ||||||
|  |         bundle.definedVars({ isProdBuild, latestBuild, defineOverlay }) | ||||||
|  |       ), | ||||||
|  |       new webpack.IgnorePlugin({ | ||||||
|  |         checkResource(resource, context) { | ||||||
|  |           // Only use ignore to intercept imports that we don't control | ||||||
|  |           // inside node_module dependencies. | ||||||
|  |           if ( | ||||||
|  |             !context.includes("/node_modules/") || | ||||||
|  |             // calling define.amd will call require("!!webpack amd options") | ||||||
|  |             resource.startsWith("!!webpack") || | ||||||
|  |             // loaded by webpack dev server but doesn't exist. | ||||||
|  |             resource === "webpack/hot" | ||||||
|  |           ) { | ||||||
|  |             return false; | ||||||
|  |           } | ||||||
|  |           let fullPath; | ||||||
|  |           try { | ||||||
|  |             fullPath = resource.startsWith(".") | ||||||
|  |               ? path.resolve(context, resource) | ||||||
|  |               : require.resolve(resource); | ||||||
|  |           } catch (err) { | ||||||
|  |             console.error( | ||||||
|  |               "Error in Home Assistant ignore plugin", | ||||||
|  |               resource, | ||||||
|  |               context | ||||||
|  |             ); | ||||||
|  |             throw err; | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           return ignorePackages.some((toIgnorePath) => | ||||||
|  |             fullPath.startsWith(toIgnorePath) | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|  |       }), | ||||||
|  |       new webpack.NormalModuleReplacementPlugin( | ||||||
|  |         new RegExp(bundle.emptyPackages({ latestBuild }).join("|")), | ||||||
|  |         path.resolve(paths.polymer_dir, "src/util/empty.js") | ||||||
|  |       ), | ||||||
|  |       // We need to change the import of the polyfill for EventTarget, so we replace the polyfill file with our customized one | ||||||
|  |       new webpack.NormalModuleReplacementPlugin( | ||||||
|  |         new RegExp( | ||||||
|  |           require.resolve( | ||||||
|  |             "lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js" | ||||||
|  |           ) | ||||||
|  |         ), | ||||||
|  |         path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js") | ||||||
|  |       ), | ||||||
|  |       !isProdBuild && new LogStartCompilePlugin(), | ||||||
|  |     ].filter(Boolean), | ||||||
|  |     resolve: { | ||||||
|  |       extensions: [".ts", ".js", ".json"], | ||||||
|  |     }, | ||||||
|  |     output: { | ||||||
|  |       filename: ({ chunk }) => { | ||||||
|  |         if (!isProdBuild || dontHash.has(chunk.name)) { | ||||||
|  |           return `${chunk.name}.js`; | ||||||
|  |         } | ||||||
|  |         return `${chunk.name}.${chunk.hash.substr(0, 8)}.js`; | ||||||
|  |       }, | ||||||
|  |       chunkFilename: | ||||||
|  |         isProdBuild && !isStatsBuild | ||||||
|  |           ? "chunk.[chunkhash].js" | ||||||
|  |           : "[name].chunk.js", | ||||||
|  |       path: outputPath, | ||||||
|  |       publicPath, | ||||||
|  |       // To silence warning in worker plugin | ||||||
|  |       globalObject: "self", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|  |   return createWebpackConfig( | ||||||
|  |     bundle.config.app({ isProdBuild, latestBuild, isStatsBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|  |   return createWebpackConfig( | ||||||
|  |     bundle.config.demo({ isProdBuild, latestBuild, isStatsBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createCastConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild })); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createHassioConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createWebpackConfig( | ||||||
|  |     bundle.config.hassio({ isProdBuild, latestBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createGalleryConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createWebpackConfig( | ||||||
|  |     bundle.config.gallery({ isProdBuild, latestBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  |   createAppConfig, | ||||||
|  |   createDemoConfig, | ||||||
|  |   createCastConfig, | ||||||
|  |   createHassioConfig, | ||||||
|  |   createGalleryConfig, | ||||||
|  | }; | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| # These redirects are handled by Netlify |  | ||||||
| # |  | ||||||
|  |  | ||||||
| # Some custom cards are not prefixing the instance URL when fetching data |  | ||||||
| # and can end up fetching the data from the Cast domain instead of HA. |  | ||||||
| # This will make sure that some common ones are replaced with a placeholder. |  | ||||||
| /api/camera_proxy/* /images/google-nest-hub.png |  | ||||||
| /api/camera_proxy_stream/* /images/google-nest-hub.png |  | ||||||
| /api/media_player_proxy/* /images/google-nest-hub.png |  | ||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 15 KiB After Width: | Height: | Size: 16 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 9.8 KiB | 
							
								
								
									
										3
									
								
								cast/public/service_worker.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								cast/public/service_worker.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | self.addEventListener("fetch", function(event) { | ||||||
|  |   event.respondWith(fetch(event.request)); | ||||||
|  | }); | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| "use strict"; |  | ||||||
|  |  | ||||||
| self.addEventListener("fetch", (event) => { |  | ||||||
|   event.respondWith(fetch(event.request)); |  | ||||||
| }); |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| self.addEventListener("fetch", (event) => { |  | ||||||
|   event.respondWith(fetch(event.request)); |  | ||||||
| }); |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| import rollup from "../build-scripts/rollup.cjs"; | const rollup = require("../build-scripts/rollup.js"); | ||||||
| import env from "../build-scripts/env.cjs"; | const env = require("../build-scripts/env.js"); | ||||||
|  |  | ||||||
| const config = rollup.createCastConfig({ | const config = rollup.createCastConfig({ | ||||||
|   isProdBuild: env.isProdBuild(), |   isProdBuild: env.isProdBuild(), | ||||||
| @@ -7,4 +7,4 @@ const config = rollup.createCastConfig({ | |||||||
|   isStatsBuild: env.isStatsBuild(), |   isStatsBuild: env.isStatsBuild(), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export default { ...config.inputOptions, output: config.outputOptions }; | module.exports = { ...config.inputOptions, output: config.outputOptions }; | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| <meta property="fb:app_id" content="338291289691179" /> |  | ||||||
| <meta property="og:title" content="Home Assistant Cast" /> |  | ||||||
| <meta property="og:site_name" content="Home Assistant Cast" /> |  | ||||||
| <meta property="og:url" content="https://cast.home-assistant.io/" /> |  | ||||||
| <meta property="og:type" content="website" /> |  | ||||||
| <meta |  | ||||||
|   property="og:description" |  | ||||||
|   content="Show Home Assistant on your Chromecast or Google Assistant devices with a screen." |  | ||||||
| /> |  | ||||||
| <meta |  | ||||||
|   property="og:image" |  | ||||||
|   content="https://cast.home-assistant.io/images/google-nest-hub.png" |  | ||||||
| /> |  | ||||||
| <meta name="twitter:card" content="summary_large_image" /> |  | ||||||
| <meta name="twitter:site" content="@home_assistant" /> |  | ||||||
| <meta name="twitter:title" content="Home Assistant Cast" /> |  | ||||||
| <meta |  | ||||||
|   name="twitter:description" |  | ||||||
|   content="Show Home Assistant on your Chromecast or Google Assistant devices with a screen." |  | ||||||
| /> |  | ||||||
| <meta |  | ||||||
|   name="twitter:image" |  | ||||||
|   content="https://cast.home-assistant.io/images/google-nest-hub.png" |  | ||||||
| /> |  | ||||||
| @@ -1,29 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html> |  | ||||||
|   <head> |  | ||||||
|     <title>Home Assistant Cast</title> |  | ||||||
|     <link rel="manifest" href="/manifest.json" /> |  | ||||||
|     <link rel="icon" href="/images/ha-cast-icon.png" type="image/png" /> |  | ||||||
|     <%= renderTemplate("../../../src/html/_style_base.html.template") %> |  | ||||||
|     <style> |  | ||||||
|       body { |  | ||||||
|         background-color: #e5e5e5; |  | ||||||
|       } |  | ||||||
|     </style> |  | ||||||
|     <%= renderTemplate("_social_meta.html.template") %> |  | ||||||
|   </head> |  | ||||||
|   <body> |  | ||||||
|     <hc-connect></hc-connect> |  | ||||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> |  | ||||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> |  | ||||||
|     <script> |  | ||||||
|     (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){ |  | ||||||
|     (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o), |  | ||||||
|     m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m) |  | ||||||
|     })(window,document,'script','https://www.google-analytics.com/analytics.js','ga'); |  | ||||||
|  |  | ||||||
|     ga('create', 'UA-57927901-9', 'auto'); |  | ||||||
|     ga('send', 'pageview', location.pathname.includes("auth_callback") === -1 ? location.pathname : "/"); |  | ||||||
|     </script> |  | ||||||
|   </body> |  | ||||||
| </html> |  | ||||||
| @@ -3,7 +3,7 @@ | |||||||
|   <head> |   <head> | ||||||
|     <title>Home Assistant Cast - FAQ</title> |     <title>Home Assistant Cast - FAQ</title> | ||||||
|     <link rel="icon" href="/images/ha-cast-icon.png" type="image/png" /> |     <link rel="icon" href="/images/ha-cast-icon.png" type="image/png" /> | ||||||
|     <%= renderTemplate("../../../src/html/_style_base.html.template") %> |     <%= renderTemplate('_style_base') %> | ||||||
|     <style> |     <style> | ||||||
|       body { |       body { | ||||||
|         background-color: #e5e5e5; |         background-color: #e5e5e5; | ||||||
| @@ -35,8 +35,25 @@ | |||||||
|     /> |     /> | ||||||
|   </head> |   </head> | ||||||
|   <body> |   <body> | ||||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> |     <%= renderTemplate('_js_base') %> | ||||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> | 
 | ||||||
|  |     <script> | ||||||
|  |       import("<%= latestLauncherJS %>"); | ||||||
|  |       window.latestJS = true; | ||||||
|  |     </script> | ||||||
|  | 
 | ||||||
|  |     <script> | ||||||
|  |       if (!window.latestJS) { | ||||||
|  |         <% if (useRollup) { %> | ||||||
|  |           _ls("/static/js/s.min.js").onload = function() { | ||||||
|  |             System.import("<%= es5LauncherJS %>"); | ||||||
|  |           }; | ||||||
|  |         <% } else { %> | ||||||
|  |           _ls("<%= es5LauncherJS %>"); | ||||||
|  |         <% } %> | ||||||
|  |       } | ||||||
|  |     </script> | ||||||
|  | 
 | ||||||
|     <hc-layout subtitle="FAQ"> |     <hc-layout subtitle="FAQ"> | ||||||
|       <style> |       <style> | ||||||
|         a { |         a { | ||||||
| @@ -122,7 +139,7 @@ | |||||||
|           Your authentication credentials or Home Assistant url are never sent |           Your authentication credentials or Home Assistant url are never sent | ||||||
|           to the Cloud. You can validate this behavior in |           to the Cloud. You can validate this behavior in | ||||||
|           <a |           <a | ||||||
|             href="https://github.com/home-assistant/frontend/tree/dev/cast" |             href="https://github.com/home-assistant/home-assistant-polymer/tree/dev/cast" | ||||||
|             target="_blank" |             target="_blank" | ||||||
|             >the source code</a |             >the source code</a | ||||||
|           >. |           >. | ||||||
| @@ -139,7 +156,7 @@ | |||||||
|         </p> |         </p> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div class="section-header">What does Home Assistant Cast do?</div> |       <div class="section-header">Wat does Home Assistant Cast do?</div> | ||||||
|       <div class="card-content"> |       <div class="card-content"> | ||||||
|         <p> |         <p> | ||||||
|           Home Assistant Cast is a receiver application for the Chromecast. When |           Home Assistant Cast is a receiver application for the Chromecast. When | ||||||
| @@ -196,7 +213,7 @@ | |||||||
|         </p> |         </p> | ||||||
|         <ul> |         <ul> | ||||||
|           <li>Google Chrome (all platforms except iOS)</li> |           <li>Google Chrome (all platforms except iOS)</li> | ||||||
|           <li>Microsoft Edge (all platforms except iOS)</li> |           <li>Microsoft Edge (all platforms)</li> | ||||||
|         </ul> |         </ul> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
| @@ -226,5 +243,17 @@ http: | |||||||
|         </p> |         </p> | ||||||
|       </div> |       </div> | ||||||
|     </hc-layout> |     </hc-layout> | ||||||
|  | 
 | ||||||
|  |     <script> | ||||||
|  |       var _gaq = [["_setAccount", "UA-57927901-9"], ["_trackPageview"]]; | ||||||
|  |       (function (d, t) { | ||||||
|  |         var g = d.createElement(t), | ||||||
|  |           s = d.getElementsByTagName(t)[0]; | ||||||
|  |         g.src = | ||||||
|  |           ("https:" == location.protocol ? "//ssl" : "//www") + | ||||||
|  |           ".google-analytics.com/ga.js"; | ||||||
|  |         s.parentNode.insertBefore(g, s); | ||||||
|  |       })(document, "script"); | ||||||
|  |     </script> | ||||||
|   </body> |   </body> | ||||||
| </html> | </html> | ||||||
							
								
								
									
										57
									
								
								cast/src/html/launcher.html.template
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								cast/src/html/launcher.html.template
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | <!DOCTYPE html> | ||||||
|  | <html> | ||||||
|  |   <head> | ||||||
|  |     <title>Home Assistant Cast</title> | ||||||
|  |     <link rel="manifest" href="/manifest.json" /> | ||||||
|  |     <link rel="icon" href="/images/ha-cast-icon.png" type="image/png" /> | ||||||
|  |     <%= renderTemplate('_style_base') %> | ||||||
|  |     <style> | ||||||
|  |       body { | ||||||
|  |         background-color: #e5e5e5; | ||||||
|  |       } | ||||||
|  |     </style> | ||||||
|  |     <meta property="fb:app_id" content="338291289691179"> | ||||||
|  |     <meta property="og:title" content="Home Assistant Cast"> | ||||||
|  |     <meta property="og:site_name" content="Home Assistant Cast"> | ||||||
|  |     <meta property="og:url" content="https://cast.home-assistant.io/"> | ||||||
|  |     <meta property="og:type" content="website"> | ||||||
|  |     <meta property="og:description" content="Show Home Assistant on your Chromecast or Google Assistant devices with a screen."> | ||||||
|  |     <meta property="og:image" content="https://cast.home-assistant.io/images/google-nest-hub.png"> | ||||||
|  |     <meta name="twitter:card" content="summary_large_image"> | ||||||
|  |     <meta name="twitter:site" content="@home_assistant"> | ||||||
|  |     <meta name="twitter:title" content="Home Assistant Cast"> | ||||||
|  |     <meta name="twitter:description" content="Show Home Assistant on your Chromecast or Google Assistant devices with a screen."> | ||||||
|  |     <meta name="twitter:image" content="https://cast.home-assistant.io/images/google-nest-hub.png"> | ||||||
|  |   </head> | ||||||
|  |   <body> | ||||||
|  |     <%= renderTemplate('_js_base') %> | ||||||
|  |  | ||||||
|  |     <hc-connect></hc-connect> | ||||||
|  |  | ||||||
|  |     <script> | ||||||
|  |       import("<%= latestLauncherJS %>"); | ||||||
|  |       window.latestJS = true; | ||||||
|  |     </script> | ||||||
|  |  | ||||||
|  |     <script> | ||||||
|  |       if (!window.latestJS) { | ||||||
|  |         <% if (useRollup) { %> | ||||||
|  |           _ls("/static/js/s.min.js").onload = function() { | ||||||
|  |             System.import("<%= es5LauncherJS %>"); | ||||||
|  |           }; | ||||||
|  |         <% } else { %> | ||||||
|  |           _ls("<%= es5LauncherJS %>"); | ||||||
|  |         <% } %> | ||||||
|  |       } | ||||||
|  |     </script> | ||||||
|  |     <script> | ||||||
|  |     (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){ | ||||||
|  |     (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o), | ||||||
|  |     m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m) | ||||||
|  |     })(window,document,'script','https://www.google-analytics.com/analytics.js','ga'); | ||||||
|  |  | ||||||
|  |     ga('create', 'UA-57927901-9', 'auto'); | ||||||
|  |     ga('send', 'pageview', location.pathname.includes("auth_callback") === -1 ? location.pathname : "/"); | ||||||
|  |     </script> | ||||||
|  |   </body> | ||||||
|  | </html> | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html> |  | ||||||
|   <head> |  | ||||||
|     <script src="//www.gstatic.com/cast/sdk/libs/caf_receiver/v3/cast_receiver_framework.js"></script> |  | ||||||
|     <style> |  | ||||||
|       body { |  | ||||||
|         --logo-image: url('https://www.home-assistant.io/images/home-assistant-logo.svg'); |  | ||||||
|         --logo-repeat: no-repeat; |  | ||||||
|         --playback-logo-image: url('https://www.home-assistant.io/images/home-assistant-logo.svg'); |  | ||||||
|         --theme-hue: 200; |  | ||||||
|         --progress-color: #03a9f4; |  | ||||||
|         --splash-image: url('https://home-assistant.io/images/cast/splash.png'); |  | ||||||
|         --splash-size: cover; |  | ||||||
|         --background-color: #41bdf5; |  | ||||||
|       } |  | ||||||
|     </style> |  | ||||||
|   </head> |  | ||||||
|   <body> |  | ||||||
|     <cast-media-player></cast-media-player> |  | ||||||
|     <%= renderTemplate("../../../src/html/_js_base.html.template") %> |  | ||||||
|     <%= renderTemplate("../../../src/html/_script_loader.html.template") %> |  | ||||||
|   </body> |  | ||||||
| </html> |  | ||||||
| @@ -1,14 +1,18 @@ | |||||||
| <!DOCTYPE html> | <!DOCTYPE html> | ||||||
| <html> | <html> | ||||||
|   <script src="//www.gstatic.com/cast/sdk/libs/caf_receiver/v3/cast_receiver_framework.js"></script> |   <script src="//www.gstatic.com/cast/sdk/libs/caf_receiver/v3/cast_receiver_framework.js"></script> | ||||||
|   <% for (const entry of latestEntryJS) { %> |   <script type="module" src="<%= latestReceiverJS %>"></script> | ||||||
|     <script type="module" src="<%= entry %>"></script> |   <%= renderTemplate('_style_base') %> | ||||||
|   <% } %> |  | ||||||
|   <%= renderTemplate("../../../src/html/_style_base.html.template") %> |  | ||||||
|   <style> |   <style> | ||||||
|     body { |     body { | ||||||
|       background-color: white; |       background-color: white; | ||||||
|       font-size: initial; |       font-size: initial; | ||||||
|     } |     } | ||||||
|   </style> |   </style> | ||||||
|  |   <script> | ||||||
|  |   var _gaq=[['_setAccount','UA-57927901-10'],['_trackPageview']]; | ||||||
|  |   (function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0]; | ||||||
|  |   g.src=('https:'==location.protocol?'//ssl':'//www')+'.google-analytics.com/ga.js'; | ||||||
|  |   s.parentNode.insertBefore(g,s)}(document,'script')); | ||||||
|  |   </script> | ||||||
| </html> | </html> | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import "../../../src/resources/safari-14-attachshadow-patch"; | ||||||
|  | import "../../../src/resources/ha-style"; | ||||||
|  | import "../../../src/resources/roboto"; | ||||||
| import "./layout/hc-connect"; | import "./layout/hc-connect"; | ||||||
|  |  | ||||||
| import("../../../src/resources/ha-style"); |  | ||||||
|   | |||||||
| @@ -1,10 +1,16 @@ | |||||||
| import "@material/mwc-button/mwc-button"; | import "@polymer/paper-item/paper-icon-item"; | ||||||
| import "@material/mwc-list/mwc-list"; | import "@polymer/paper-listbox/paper-listbox"; | ||||||
| import type { ActionDetail } from "@material/mwc-list/mwc-list"; |  | ||||||
| import { mdiCast, mdiCastConnected, mdiViewDashboard } from "@mdi/js"; |  | ||||||
| import { Auth, Connection } from "home-assistant-js-websocket"; | import { Auth, Connection } from "home-assistant-js-websocket"; | ||||||
| import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit"; | import { | ||||||
| import { customElement, property, state } from "lit/decorators"; |   css, | ||||||
|  |   CSSResult, | ||||||
|  |   customElement, | ||||||
|  |   html, | ||||||
|  |   LitElement, | ||||||
|  |   property, | ||||||
|  |   internalProperty, | ||||||
|  |   TemplateResult, | ||||||
|  | } from "lit-element"; | ||||||
| import { CastManager } from "../../../../src/cast/cast_manager"; | import { CastManager } from "../../../../src/cast/cast_manager"; | ||||||
| import { | import { | ||||||
|   castSendShowLovelaceView, |   castSendShowLovelaceView, | ||||||
| @@ -18,39 +24,39 @@ import { | |||||||
| import { atLeastVersion } from "../../../../src/common/config/version"; | import { atLeastVersion } from "../../../../src/common/config/version"; | ||||||
| import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | ||||||
| import "../../../../src/components/ha-icon"; | import "../../../../src/components/ha-icon"; | ||||||
| import "../../../../src/components/ha-svg-icon"; |  | ||||||
| import { | import { | ||||||
|   getLegacyLovelaceCollection, |   getLegacyLovelaceCollection, | ||||||
|   getLovelaceCollection, |   getLovelaceCollection, | ||||||
|  |   LovelaceConfig, | ||||||
| } from "../../../../src/data/lovelace"; | } from "../../../../src/data/lovelace"; | ||||||
| import { isStrategyDashboard } from "../../../../src/data/lovelace/config/types"; |  | ||||||
| import { LovelaceViewConfig } from "../../../../src/data/lovelace/config/view"; |  | ||||||
| import "../../../../src/layouts/hass-loading-screen"; | import "../../../../src/layouts/hass-loading-screen"; | ||||||
| import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | ||||||
| import "./hc-layout"; | import "./hc-layout"; | ||||||
| import "../../../../src/components/ha-list-item"; | import "@material/mwc-button/mwc-button"; | ||||||
|  |  | ||||||
| @customElement("hc-cast") | @customElement("hc-cast") | ||||||
| class HcCast extends LitElement { | class HcCast extends LitElement { | ||||||
|   @property({ attribute: false }) public auth!: Auth; |   @property() public auth!: Auth; | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public connection!: Connection; |   @property() public connection!: Connection; | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public castManager!: CastManager; |   @property() public castManager!: CastManager; | ||||||
|  |  | ||||||
|   @state() private askWrite = false; |   @internalProperty() private askWrite = false; | ||||||
|  |  | ||||||
|   @state() private lovelaceViews?: LovelaceViewConfig[] | null; |   @internalProperty() private lovelaceConfig?: LovelaceConfig | null; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     if (this.lovelaceViews === undefined) { |     if (this.lovelaceConfig === undefined) { | ||||||
|       return html`<hass-loading-screen no-toolbar></hass-loading-screen>`; |       return html` <hass-loading-screen no-toolbar></hass-loading-screen>> `; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const error = |     const error = | ||||||
|       this.castManager.castState === "NO_DEVICES_AVAILABLE" |       this.castManager.castState === "NO_DEVICES_AVAILABLE" | ||||||
|         ? html` |         ? html` | ||||||
|             <p>There were no suitable Chromecast devices to cast to found.</p> |             <p> | ||||||
|  |               There were no suitable Chromecast devices to cast to found. | ||||||
|  |             </p> | ||||||
|           ` |           ` | ||||||
|         : undefined; |         : undefined; | ||||||
|  |  | ||||||
| @@ -74,53 +80,48 @@ class HcCast extends LitElement { | |||||||
|         ${error |         ${error | ||||||
|           ? html` <div class="card-content">${error}</div> ` |           ? html` <div class="card-content">${error}</div> ` | ||||||
|           : !this.castManager.status |           : !this.castManager.status | ||||||
|             ? html` |           ? html` | ||||||
|                 <p class="center-item"> |               <p class="center-item"> | ||||||
|                   <mwc-button raised @click=${this._handleLaunch}> |                 <mwc-button raised @click=${this._handleLaunch}> | ||||||
|                     <ha-svg-icon .path=${mdiCast}></ha-svg-icon> |                   <ha-icon icon="hass:cast"></ha-icon> | ||||||
|                     Start Casting |                   Start Casting | ||||||
|                   </mwc-button> |                 </mwc-button> | ||||||
|                 </p> |               </p> | ||||||
|               ` |             ` | ||||||
|             : html` |           : html` | ||||||
|                 <div class="section-header">PICK A VIEW</div> |               <div class="section-header">PICK A VIEW</div> | ||||||
|                 <mwc-list @action=${this._handlePickView} activatable> |               <paper-listbox | ||||||
|                   ${( |                 attr-for-selected="data-path" | ||||||
|                     this.lovelaceViews ?? [ |                 .selected=${this.castManager.status.lovelacePath || ""} | ||||||
|                       generateDefaultViewConfig({}, {}, {}, {}, () => ""), |               > | ||||||
|                     ] |                 ${(this.lovelaceConfig | ||||||
|                   ).map( |                   ? this.lovelaceConfig.views | ||||||
|                     (view, idx) => html` |                   : [generateDefaultViewConfig([], [], [], {}, () => "")] | ||||||
|                       <ha-list-item |                 ).map( | ||||||
|                         graphic="avatar" |                   (view, idx) => html` | ||||||
|                         .activated=${this.castManager.status?.lovelacePath === |                     <paper-icon-item | ||||||
|                         (view.path ?? idx)} |                       @click=${this._handlePickView} | ||||||
|                         .selected=${this.castManager.status?.lovelacePath === |                       data-path=${view.path || idx} | ||||||
|                         (view.path ?? idx)} |                     > | ||||||
|                       > |                       ${view.icon | ||||||
|                         ${view.title || view.path || "Unnamed view"} |                         ? html` | ||||||
|                         ${view.icon |                             <ha-icon | ||||||
|                           ? html` |                               .icon=${view.icon} | ||||||
|                               <ha-icon |  | ||||||
|                                 .icon=${view.icon} |  | ||||||
|                                 slot="graphic" |  | ||||||
|                               ></ha-icon> |  | ||||||
|                             ` |  | ||||||
|                           : html`<ha-svg-icon |  | ||||||
|                               slot="item-icon" |                               slot="item-icon" | ||||||
|                               .path=${mdiViewDashboard} |                             ></ha-icon> | ||||||
|                             ></ha-svg-icon>`} |                           ` | ||||||
|                       </ha-list-item> |                         : ""} | ||||||
|                     ` |                       ${view.title || view.path} | ||||||
|                   )}</mwc-list |                     </paper-icon-item> | ||||||
|                 > |                   ` | ||||||
|               `} |                 )} | ||||||
|  |               </paper-listbox> | ||||||
|  |             `} | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           ${this.castManager.status |           ${this.castManager.status | ||||||
|             ? html` |             ? html` | ||||||
|                 <mwc-button @click=${this._handleLaunch}> |                 <mwc-button @click=${this._handleLaunch}> | ||||||
|                   <ha-svg-icon .path=${mdiCastConnected}></ha-svg-icon> |                   <ha-icon icon="hass:cast-connected"></ha-icon> | ||||||
|                   Manage |                   Manage | ||||||
|                 </mwc-button> |                 </mwc-button> | ||||||
|               ` |               ` | ||||||
| @@ -143,15 +144,11 @@ class HcCast extends LitElement { | |||||||
|     llColl.refresh().then( |     llColl.refresh().then( | ||||||
|       () => { |       () => { | ||||||
|         llColl.subscribe((config) => { |         llColl.subscribe((config) => { | ||||||
|           if (isStrategyDashboard(config)) { |           this.lovelaceConfig = config; | ||||||
|             this.lovelaceViews = null; |  | ||||||
|           } else { |  | ||||||
|             this.lovelaceViews = config.views; |  | ||||||
|           } |  | ||||||
|         }); |         }); | ||||||
|       }, |       }, | ||||||
|       async () => { |       async () => { | ||||||
|         this.lovelaceViews = null; |         this.lovelaceConfig = null; | ||||||
|       } |       } | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
| @@ -170,7 +167,9 @@ class HcCast extends LitElement { | |||||||
|     toggleAttribute( |     toggleAttribute( | ||||||
|       this, |       this, | ||||||
|       "hide-icons", |       "hide-icons", | ||||||
|       this.lovelaceViews ? !this.lovelaceViews.some((view) => view.icon) : true |       this.lovelaceConfig | ||||||
|  |         ? !this.lovelaceConfig.views.some((view) => view.icon) | ||||||
|  |         : true | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -187,10 +186,10 @@ class HcCast extends LitElement { | |||||||
|     this.castManager.requestSession(); |     this.castManager.requestSession(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _handlePickView(ev: CustomEvent<ActionDetail>) { |   private async _handlePickView(ev: Event) { | ||||||
|     const path = this.lovelaceViews![ev.detail.index].path ?? ev.detail.index; |     const path = (ev.currentTarget as any).getAttribute("data-path"); | ||||||
|     await ensureConnectedCastSession(this.castManager!, this.auth!); |     await ensureConnectedCastSession(this.castManager!, this.auth!); | ||||||
|     castSendShowLovelaceView(this.castManager, this.auth.data.hassUrl, path); |     castSendShowLovelaceView(this.castManager, path); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _handleLogout() { |   private async _handleLogout() { | ||||||
| @@ -202,12 +201,12 @@ class HcCast extends LitElement { | |||||||
|       } |       } | ||||||
|       this.connection.close(); |       this.connection.close(); | ||||||
|       location.reload(); |       location.reload(); | ||||||
|     } catch (err: any) { |     } catch (err) { | ||||||
|       alert("Unable to log out!"); |       alert("Unable to log out!"); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static get styles(): CSSResultGroup { |   static get styles(): CSSResult { | ||||||
|     return css` |     return css` | ||||||
|       .center-item { |       .center-item { | ||||||
|         display: flex; |         display: flex; | ||||||
| @@ -244,21 +243,30 @@ class HcCast extends LitElement { | |||||||
|         color: var(--secondary-text-color); |         color: var(--secondary-text-color); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       mwc-button ha-svg-icon { |       mwc-button ha-icon { | ||||||
|         margin-right: 8px; |         margin-right: 8px; | ||||||
|         margin-inline-end: 8px; |  | ||||||
|         margin-inline-start: initial; |  | ||||||
|         height: 18px; |         height: 18px; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       ha-list-item ha-icon, |       paper-listbox { | ||||||
|       ha-list-item ha-svg-icon { |         padding-top: 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       paper-listbox ha-icon { | ||||||
|         padding: 12px; |         padding: 12px; | ||||||
|         color: var(--secondary-text-color); |         color: var(--secondary-text-color); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       :host([hide-icons]) ha-icon { |       paper-icon-item { | ||||||
|         display: none; |         cursor: pointer; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       paper-icon-item[disabled] { | ||||||
|  |         cursor: initial; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       :host([hide-icons]) paper-icon-item { | ||||||
|  |         --paper-item-icon-width: 0px; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .spacer { |       .spacer { | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import { mdiCastConnected, mdiCast } from "@mdi/js"; | import "@polymer/paper-input/paper-input"; | ||||||
| import { | import { | ||||||
|   Auth, |   Auth, | ||||||
|   Connection, |   Connection, | ||||||
| @@ -11,19 +11,25 @@ import { | |||||||
|   getAuth, |   getAuth, | ||||||
|   getAuthOptions, |   getAuthOptions, | ||||||
| } from "home-assistant-js-websocket"; | } from "home-assistant-js-websocket"; | ||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { | ||||||
| import { customElement, state } from "lit/decorators"; |   css, | ||||||
|  |   CSSResult, | ||||||
|  |   customElement, | ||||||
|  |   html, | ||||||
|  |   LitElement, | ||||||
|  |   TemplateResult, | ||||||
|  |   internalProperty, | ||||||
|  | } from "lit-element"; | ||||||
| import { CastManager, getCastManager } from "../../../../src/cast/cast_manager"; | import { CastManager, getCastManager } from "../../../../src/cast/cast_manager"; | ||||||
| import { castSendShowDemo } from "../../../../src/cast/receiver_messages"; | import { castSendShowDemo } from "../../../../src/cast/receiver_messages"; | ||||||
| import { | import { | ||||||
|   loadTokens, |   loadTokens, | ||||||
|   saveTokens, |   saveTokens, | ||||||
| } from "../../../../src/common/auth/token_storage"; | } from "../../../../src/common/auth/token_storage"; | ||||||
| import "../../../../src/components/ha-svg-icon"; | import "../../../../src/components/ha-icon"; | ||||||
| import "../../../../src/layouts/hass-loading-screen"; | import "../../../../src/layouts/hass-loading-screen"; | ||||||
| import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | ||||||
| import "./hc-layout"; | import "./hc-layout"; | ||||||
| import "../../../../src/components/ha-textfield"; |  | ||||||
|  |  | ||||||
| const seeFAQ = (qid) => html` | const seeFAQ = (qid) => html` | ||||||
|   See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more |   See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more | ||||||
| @@ -33,13 +39,13 @@ const translateErr = (err) => | |||||||
|   err === ERR_CANNOT_CONNECT |   err === ERR_CANNOT_CONNECT | ||||||
|     ? "Unable to connect" |     ? "Unable to connect" | ||||||
|     : err === ERR_HASS_HOST_REQUIRED |     : err === ERR_HASS_HOST_REQUIRED | ||||||
|       ? "Please enter a Home Assistant URL." |     ? "Please enter a Home Assistant URL." | ||||||
|       : err === ERR_INVALID_HTTPS_TO_HTTP |     : err === ERR_INVALID_HTTPS_TO_HTTP | ||||||
|         ? html` |     ? html` | ||||||
|             Cannot connect to Home Assistant instances over "http://". |         Cannot connect to Home Assistant instances over "http://". | ||||||
|             ${seeFAQ("https")} |         ${seeFAQ("https")} | ||||||
|           ` |       ` | ||||||
|         : `Unknown error (${err}).`; |     : `Unknown error (${err}).`; | ||||||
|  |  | ||||||
| const INTRO = html` | const INTRO = html` | ||||||
|   <p> |   <p> | ||||||
| @@ -54,19 +60,19 @@ const INTRO = html` | |||||||
|  |  | ||||||
| @customElement("hc-connect") | @customElement("hc-connect") | ||||||
| export class HcConnect extends LitElement { | export class HcConnect extends LitElement { | ||||||
|   @state() private loading = false; |   @internalProperty() private loading = false; | ||||||
|  |  | ||||||
|   // If we had stored credentials but we cannot connect, |   // If we had stored credentials but we cannot connect, | ||||||
|   // show a screen asking retry or logout. |   // show a screen asking retry or logout. | ||||||
|   @state() private cannotConnect = false; |   @internalProperty() private cannotConnect = false; | ||||||
|  |  | ||||||
|   @state() private error?: string | TemplateResult; |   @internalProperty() private error?: string | TemplateResult; | ||||||
|  |  | ||||||
|   @state() private auth?: Auth; |   @internalProperty() private auth?: Auth; | ||||||
|  |  | ||||||
|   @state() private connection?: Connection; |   @internalProperty() private connection?: Connection; | ||||||
|  |  | ||||||
|   @state() private castManager?: CastManager | null; |   @internalProperty() private castManager?: CastManager | null; | ||||||
|  |  | ||||||
|   private openDemo = false; |   private openDemo = false; | ||||||
|  |  | ||||||
| @@ -80,7 +86,9 @@ export class HcConnect extends LitElement { | |||||||
|           </div> |           </div> | ||||||
|           <div class="card-actions"> |           <div class="card-actions"> | ||||||
|             <a href="/"> |             <a href="/"> | ||||||
|               <mwc-button> Retry </mwc-button> |               <mwc-button> | ||||||
|  |                 Retry | ||||||
|  |               </mwc-button> | ||||||
|             </a> |             </a> | ||||||
|             <div class="spacer"></div> |             <div class="spacer"></div> | ||||||
|             <mwc-button @click=${this._handleLogout}>Log out</mwc-button> |             <mwc-button @click=${this._handleLogout}>Log out</mwc-button> | ||||||
| @@ -116,21 +124,23 @@ export class HcConnect extends LitElement { | |||||||
|               To get started, enter your Home Assistant URL and click authorize. |               To get started, enter your Home Assistant URL and click authorize. | ||||||
|               If you want a preview instead, click the show demo button. |               If you want a preview instead, click the show demo button. | ||||||
|             </p> |             </p> | ||||||
|             <ha-textfield |             <p> | ||||||
|               label="Home Assistant URL" |               <paper-input | ||||||
|               placeholder="https://abcdefghijklmnop.ui.nabu.casa" |                 label="Home Assistant URL" | ||||||
|               @keydown=${this._handleInputKeyDown} |                 placeholder="https://abcdefghijklmnop.ui.nabu.casa" | ||||||
|             ></ha-textfield> |                 @keydown=${this._handleInputKeyDown} | ||||||
|  |               ></paper-input> | ||||||
|  |             </p> | ||||||
|             ${this.error ? html` <p class="error">${this.error}</p> ` : ""} |             ${this.error ? html` <p class="error">${this.error}</p> ` : ""} | ||||||
|           </div> |           </div> | ||||||
|           <div class="card-actions"> |           <div class="card-actions"> | ||||||
|             <mwc-button @click=${this._handleDemo}> |             <mwc-button @click=${this._handleDemo}> | ||||||
|               Show Demo |               Show Demo | ||||||
|               <ha-svg-icon |               <ha-icon | ||||||
|                 .path=${this.castManager.castState === "CONNECTED" |                 .icon=${this.castManager.castState === "CONNECTED" | ||||||
|                   ? mdiCastConnected |                   ? "hass:cast-connected" | ||||||
|                   : mdiCast} |                   : "hass:cast"} | ||||||
|               ></ha-svg-icon> |               ></ha-icon> | ||||||
|             </mwc-button> |             </mwc-button> | ||||||
|             <div class="spacer"></div> |             <div class="spacer"></div> | ||||||
|             <mwc-button @click=${this._handleConnect}>Authorize</mwc-button> |             <mwc-button @click=${this._handleConnect}>Authorize</mwc-button> | ||||||
| @@ -188,13 +198,13 @@ export class HcConnect extends LitElement { | |||||||
|  |  | ||||||
|   private _handleInputKeyDown(ev: KeyboardEvent) { |   private _handleInputKeyDown(ev: KeyboardEvent) { | ||||||
|     // Handle pressing enter. |     // Handle pressing enter. | ||||||
|     if (ev.key === "Enter") { |     if (ev.keyCode === 13) { | ||||||
|       this._handleConnect(); |       this._handleConnect(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _handleConnect() { |   private async _handleConnect() { | ||||||
|     const inputEl = this.shadowRoot!.querySelector("ha-textfield")!; |     const inputEl = this.shadowRoot!.querySelector("paper-input")!; | ||||||
|     const value = inputEl.value || ""; |     const value = inputEl.value || ""; | ||||||
|     this.error = undefined; |     this.error = undefined; | ||||||
|  |  | ||||||
| @@ -211,7 +221,7 @@ export class HcConnect extends LitElement { | |||||||
|     let url: URL; |     let url: URL; | ||||||
|     try { |     try { | ||||||
|       url = new URL(value); |       url = new URL(value); | ||||||
|     } catch (err: any) { |     } catch (err) { | ||||||
|       this.error = "Invalid URL"; |       this.error = "Invalid URL"; | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| @@ -239,7 +249,7 @@ export class HcConnect extends LitElement { | |||||||
|     try { |     try { | ||||||
|       this.loading = true; |       this.loading = true; | ||||||
|       auth = await getAuth(options); |       auth = await getAuth(options); | ||||||
|     } catch (err: any) { |     } catch (err) { | ||||||
|       if (init === "saved-tokens" && err === ERR_CANNOT_CONNECT) { |       if (init === "saved-tokens" && err === ERR_CANNOT_CONNECT) { | ||||||
|         this.cannotConnect = true; |         this.cannotConnect = true; | ||||||
|         return; |         return; | ||||||
| @@ -258,7 +268,7 @@ export class HcConnect extends LitElement { | |||||||
|  |  | ||||||
|     try { |     try { | ||||||
|       conn = await createConnection({ auth }); |       conn = await createConnection({ auth }); | ||||||
|     } catch (err: any) { |     } catch (err) { | ||||||
|       // In case of saved tokens, silently solve problems. |       // In case of saved tokens, silently solve problems. | ||||||
|       if (init === "saved-tokens") { |       if (init === "saved-tokens") { | ||||||
|         if (err === ERR_CANNOT_CONNECT) { |         if (err === ERR_CANNOT_CONNECT) { | ||||||
| @@ -284,12 +294,12 @@ export class HcConnect extends LitElement { | |||||||
|     try { |     try { | ||||||
|       saveTokens(null); |       saveTokens(null); | ||||||
|       location.reload(); |       location.reload(); | ||||||
|     } catch (err: any) { |     } catch (err) { | ||||||
|       alert("Unable to log out!"); |       alert("Unable to log out!"); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static get styles(): CSSResultGroup { |   static get styles(): CSSResult { | ||||||
|     return css` |     return css` | ||||||
|       .card-content a { |       .card-content a { | ||||||
|         color: var(--primary-color); |         color: var(--primary-color); | ||||||
| @@ -306,17 +316,13 @@ export class HcConnect extends LitElement { | |||||||
|         color: darkred; |         color: darkred; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       mwc-button ha-svg-icon { |       mwc-button ha-icon { | ||||||
|         margin-left: 8px; |         margin-left: 8px; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .spacer { |       .spacer { | ||||||
|         flex: 1; |         flex: 1; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       ha-textfield { |  | ||||||
|         width: 100%; |  | ||||||
|       } |  | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,29 +4,32 @@ import { | |||||||
|   getUser, |   getUser, | ||||||
|   HassUser, |   HassUser, | ||||||
| } from "home-assistant-js-websocket"; | } from "home-assistant-js-websocket"; | ||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { | ||||||
| import { customElement, property } from "lit/decorators"; |   css, | ||||||
|  |   CSSResult, | ||||||
|  |   customElement, | ||||||
|  |   html, | ||||||
|  |   LitElement, | ||||||
|  |   property, | ||||||
|  |   TemplateResult, | ||||||
|  | } from "lit-element"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  |  | ||||||
| @customElement("hc-layout") | @customElement("hc-layout") | ||||||
| class HcLayout extends LitElement { | class HcLayout extends LitElement { | ||||||
|   @property() public subtitle?: string; |   @property() public subtitle?: string | undefined; | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public auth?: Auth; |   @property() public auth?: Auth; | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public connection?: Connection; |   @property() public connection?: Connection; | ||||||
|  |  | ||||||
|   @property({ attribute: false }) public user?: HassUser; |   @property() public user?: HassUser; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|       <ha-card> |       <ha-card> | ||||||
|         <div class="layout"> |         <div class="layout"> | ||||||
|           <img |           <img class="hero" src="/images/google-nest-hub.png" /> | ||||||
|             class="hero" |  | ||||||
|             alt="A Google Nest Hub with a Home Assistant dashboard on its screen" |  | ||||||
|             src="/images/google-nest-hub.png" |  | ||||||
|           /> |  | ||||||
|           <h1 class="card-header"> |           <h1 class="card-header"> | ||||||
|             Home Assistant Cast${this.subtitle ? ` – ${this.subtitle}` : ""} |             Home Assistant Cast${this.subtitle ? ` – ${this.subtitle}` : ""} | ||||||
|             ${this.auth |             ${this.auth | ||||||
| @@ -48,7 +51,7 @@ class HcLayout extends LitElement { | |||||||
|       <div class="footer"> |       <div class="footer"> | ||||||
|         <a href="./faq.html">Frequently Asked Questions</a> – Found a bug? |         <a href="./faq.html">Frequently Asked Questions</a> – Found a bug? | ||||||
|         <a |         <a | ||||||
|           href="https://github.com/home-assistant/frontend/issues" |           href="https://github.com/home-assistant/home-assistant-polymer/issues" | ||||||
|           target="_blank" |           target="_blank" | ||||||
|           >Let us know!</a |           >Let us know!</a | ||||||
|         > |         > | ||||||
| @@ -66,7 +69,7 @@ class HcLayout extends LitElement { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static get styles(): CSSResultGroup { |   static get styles(): CSSResult { | ||||||
|     return css` |     return css` | ||||||
|       :host { |       :host { | ||||||
|         display: flex; |         display: flex; | ||||||
| @@ -88,19 +91,15 @@ class HcLayout extends LitElement { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       .card-header { |       .card-header { | ||||||
|         color: var(--ha-card-header-color, var(--primary-text-color)); |         color: var(--ha-card-header-color, --primary-text-color); | ||||||
|         font-family: var(--ha-card-header-font-family, inherit); |         font-family: var(--ha-card-header-font-family, inherit); | ||||||
|         font-size: var(--ha-card-header-font-size, 24px); |         font-size: var(--ha-card-header-font-size, 24px); | ||||||
|         letter-spacing: -0.012em; |         letter-spacing: -0.012em; | ||||||
|         line-height: 32px; |         line-height: 32px; | ||||||
|         padding: 24px 16px 16px; |         padding: 24px 16px 16px; | ||||||
|         display: block; |         display: block; | ||||||
|         margin: 0; |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .hero { |  | ||||||
|         border-radius: 4px 4px 0 0; |  | ||||||
|       } |  | ||||||
|       .subtitle { |       .subtitle { | ||||||
|         font-size: 14px; |         font-size: 14px; | ||||||
|         color: var(--secondary-text-color); |         color: var(--secondary-text-color); | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| import { framework } from "../receiver/cast_framework"; |  | ||||||
|  |  | ||||||
| const castContext = framework.CastReceiverContext.getInstance(); |  | ||||||
|  |  | ||||||
| const playerManager = castContext.getPlayerManager(); |  | ||||||
|  |  | ||||||
| playerManager.setMessageInterceptor( |  | ||||||
|   framework.messages.MessageType.LOAD, |  | ||||||
|   (loadRequestData) => { |  | ||||||
|     const media = loadRequestData.media; |  | ||||||
|     // Special handling if it came from Google Assistant |  | ||||||
|     if (media.entity) { |  | ||||||
|       media.contentId = media.entity; |  | ||||||
|       media.streamType = framework.messages.StreamType.LIVE; |  | ||||||
|       media.contentType = "application/vnd.apple.mpegurl"; |  | ||||||
|       // @ts-ignore |  | ||||||
|       media.hlsVideoSegmentFormat = |  | ||||||
|         framework.messages.HlsVideoSegmentFormat.FMP4; |  | ||||||
|     } |  | ||||||
|     return loadRequestData; |  | ||||||
|   } |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| castContext.start(); |  | ||||||
| @@ -1,3 +1,2 @@ | |||||||
| import { framework } from "./cast_framework"; | /* eslint-disable no-undef */ | ||||||
|  | export const castContext = cast.framework.CastReceiverContext.getInstance(); | ||||||
| export const castContext = framework.CastReceiverContext.getInstance(); |  | ||||||
|   | |||||||
| @@ -1,3 +0,0 @@ | |||||||
| import type { cast as ReceiverCast } from "chromecast-caf-receiver"; |  | ||||||
|  |  | ||||||
| export const framework = (cast as unknown as typeof ReceiverCast).framework; |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user