mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 03:29:41 +00:00 
			
		
		
		
	Compare commits
	
		
			254 Commits
		
	
	
		
			20211004.0
			...
			ha-formfie
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 294967014d | ||
|   | 366aa8aed1 | ||
|   | 43011179eb | ||
|   | 6177d2b416 | ||
|   | f70485bc49 | ||
|   | 921763b5f1 | ||
|   | 5fd4315789 | ||
|   | ed291b57d0 | ||
|   | f833701e7c | ||
|   | 8533b90957 | ||
|   | c95a54c6f3 | ||
|   | a991640f52 | ||
|   | 3d99b92c07 | ||
|   | d28ad17135 | ||
|   | 3c67fc96b1 | ||
|   | 4719636176 | ||
|   | 45efee28b8 | ||
|   | 3bcf225380 | ||
|   | 2e81f843ce | ||
|   | a430142296 | ||
|   | 6335b13c5e | ||
|   | 6c4e987a24 | ||
|   | 1a5c43d72a | ||
|   | 91dbfca899 | ||
|   | 96f103644a | ||
|   | 5304e5a670 | ||
|   | 390e5b3881 | ||
|   | 9f5756c9fa | ||
|   | 0ca35d7012 | ||
|   | 0d19f4792f | ||
|   | 91b009af79 | ||
|   | 1ebd2fb9f1 | ||
|   | 4684979ae7 | ||
|   | a567312bdb | ||
|   | 1e851e0e8c | ||
|   | 7d94615f47 | ||
|   | 582fab7ea1 | ||
|   | 822590ec8a | ||
|   | e9f0967578 | ||
|   | 481da19c74 | ||
|   | b969db0c0f | ||
|   | a6b98fc3c3 | ||
|   | 87c2046ab5 | ||
|   | 4b992fb0c4 | ||
|   | 3154011c65 | ||
|   | 4e68383cf7 | ||
|   | db6ef22ebb | ||
|   | c238c7dbbc | ||
|   | d04823b4c5 | ||
|   | 4cb45d6313 | ||
|   | 6623e5f017 | ||
|   | 6518aefb7f | ||
|   | d5600b7c08 | ||
|   | 4789295d32 | ||
|   | 70d54aa855 | ||
|   | 77549efc47 | ||
|   | 00299bc74d | ||
|   | b74fc5578d | ||
|   | 9018d4cc18 | ||
|   | fcdceba09d | ||
|   | 06d4ccf344 | ||
|   | a268040ae7 | ||
|   | 67d79d618a | ||
|   | 0e8a06e24d | ||
|   | d7732ee850 | ||
|   | 729a928cfe | ||
|   | fe5a582a74 | ||
|   | c26a59d805 | ||
|   | ea331dbe0b | ||
|   | b97d6d7059 | ||
|   | 9425b943dd | ||
|   | 3fd0becfd4 | ||
|   | 12ef191a0f | ||
|   | 2bbb4acf3d | ||
|   | 77d54df007 | ||
|   | 1c35571ef0 | ||
|   | c8804160bf | ||
|   | 0a6ffb6bc8 | ||
|   | 6984f19aa0 | ||
|   | cb8de53d74 | ||
|   | 93680b9764 | ||
|   | 3cf9b745b5 | ||
|   | 5851fe26ff | ||
|   | b188c4ec81 | ||
|   | 4624c3d75b | ||
|   | 7d196b4b95 | ||
|   | 6347e44d94 | ||
|   | 719d9386c5 | ||
|   | bb734be4bc | ||
|   | 7cadaf1dc3 | ||
|   | c30453a86f | ||
|   | c2e3d0188e | ||
|   | aabb8ea16f | ||
|   | df572d59c5 | ||
|   | 5ef7a37c20 | ||
|   | 4b44e197ae | ||
|   | 8b5b21ae69 | ||
|   | f5417fad6f | ||
|   | 7fa6317f5c | ||
|   | 74533cebc6 | ||
|   | 10986db7c6 | ||
|   | 67648baca7 | ||
|   | dc9182e9ab | ||
|   | 4a7a81ffdb | ||
|   | 09ef72647e | ||
|   | da38e6f986 | ||
|   | bd1a9f2cb0 | ||
|   | 171eddd779 | ||
|   | 7acc2f9e08 | ||
|   | 27a6341137 | ||
|   | 6c5e15e707 | ||
|   | 06b1718ade | ||
|   | e50d2e16a7 | ||
|   | 0b2404a0f2 | ||
|   | 371804591d | ||
|   | 54c64c15f3 | ||
|   | 0e1124cd4f | ||
|   | 70fd759e18 | ||
|   | 8e383b2bec | ||
|   | 63cd576d56 | ||
|   | 32ac04ea78 | ||
|   | 5d6bacb0bd | ||
|   | 398d777681 | ||
|   | 549a360d98 | ||
|   | 1140e6026c | ||
|   | 29a1167782 | ||
|   | d61a77f2d9 | ||
|   | b9bde1960b | ||
|   | a12c2eea5d | ||
|   | b5c717a559 | ||
|   | 3adbc4cfaf | ||
|   | dd11fb1b99 | ||
|   | bf0d102c86 | ||
|   | dad2b92d2e | ||
|   | d027ec0018 | ||
|   | 0c038398aa | ||
|   | 5c3e0cc016 | ||
|   | 9bcd26ce57 | ||
|   | 3e8a6c418c | ||
|   | 279f3e1183 | ||
|   | f77339ad85 | ||
|   | da73b316ff | ||
|   | 82a49d2cbf | ||
|   | 05711b4636 | ||
|   | 2c2809573f | ||
|   | bbbeafcc92 | ||
|   | 95c6adc739 | ||
|   | 7c2e0aea92 | ||
|   | d05c76356f | ||
|   | f1a0623447 | ||
|   | 41d02fdb72 | ||
|   | 52d45d482c | ||
|   | a0fea94db2 | ||
|   | c3975e48d9 | ||
|   | f062e13921 | ||
|   | 08ca9c9064 | ||
|   | 667fd39147 | ||
|   | b760e543b0 | ||
|   | 760ead4860 | ||
|   | 9a4cce74f0 | ||
|   | 7488eb782d | ||
|   | b1e6935df9 | ||
|   | df53364d16 | ||
|   | 777e6c4c72 | ||
|   | e47a5effe6 | ||
|   | 62d3f74513 | ||
|   | 21e1fef0fb | ||
|   | b3f8daa758 | ||
|   | 04f586721f | ||
|   | 8e22e41605 | ||
|   | 2770d1f36b | ||
|   | 403c042235 | ||
|   | bdb3c04037 | ||
|   | f1cb21e7fc | ||
|   | a8486eda9f | ||
|   | d5b98d306d | ||
|   | bb2fe650ac | ||
|   | b576c3de40 | ||
|   | 84533b8843 | ||
|   | a8ff98b808 | ||
|   | f0062b1e67 | ||
|   | 93f64de875 | ||
|   | ec47e320d2 | ||
|   | 816d5ee594 | ||
|   | 588f5bd6b7 | ||
|   | 825ea93dba | ||
|   | a690a1d7bf | ||
|   | 9fe4c79782 | ||
|   | 42613d6519 | ||
|   | 4b77910e4f | ||
|   | 3f2cce936c | ||
|   | 6e8e9824f9 | ||
|   | 33e1d34cb1 | ||
|   | 48948d5854 | ||
|   | 7fc00ce1cb | ||
|   | 0c940be5fb | ||
|   | bddb505b7f | ||
|   | a91d25b27d | ||
|   | 4ad005f0bf | ||
|   | 7472545204 | ||
|   | 164c9c8e73 | ||
|   | e52118db93 | ||
|   | 9e7acacb06 | ||
|   | 56deb15bca | ||
|   | a3d4969d7b | ||
|   | 4a00957b71 | ||
|   | 56bd731361 | ||
|   | b6c470edf1 | ||
|   | 5bc0feacf0 | ||
|   | dc8d837e88 | ||
|   | cddf6ce1f4 | ||
|   | 8abb212ae7 | ||
|   | 0056d75127 | ||
|   | 5be475ea17 | ||
|   | b157cf5294 | ||
|   | 48c9c89e3d | ||
|   | 83f405b695 | ||
|   | 9bf41a37b4 | ||
|   | 774f22b7e7 | ||
|   | aaa3964bb3 | ||
|   | 6f6fc759cc | ||
|   | 4358b7f924 | ||
|   | 2841369d3d | ||
|   | ad031d4bda | ||
|   | 588ee2c3b1 | ||
|   | 038033cf27 | ||
|   | 84c4bbd380 | ||
|   | 807ce468d6 | ||
|   | a839494a1e | ||
|   | 80bbc9990a | ||
|   | fa52442c1c | ||
|   | 919ce2afb1 | ||
|   | db55be6d33 | ||
|   | 2dc7c1afed | ||
|   | 85956dc7fd | ||
|   | 910cd98a38 | ||
|   | 8022bd2868 | ||
|   | d5ca7e1719 | ||
|   | 066a0771b3 | ||
|   | 9e35c1ab68 | ||
|   | fb1deb838c | ||
|   | 8e010618bb | ||
|   | 365cf1f7ef | ||
|   | b226b20e3d | ||
|   | ec21f4c2c6 | ||
|   | a696d849b2 | ||
|   | ea3fae2ce4 | ||
|   | 736e117eca | ||
|   | 2fb3ac74eb | ||
|   | 2d5c8ec3e9 | ||
|   | 25c1156c88 | ||
|   | c44624282c | ||
|   | 370f2eb9e4 | ||
|   | 1793c68aae | 
| @@ -29,6 +29,7 @@ | ||||
|     "__BUILD__": false, | ||||
|     "__VERSION__": false, | ||||
|     "__STATIC_PATH__": false, | ||||
|     "__SUPERVISOR__": false, | ||||
|     "Polymer": true | ||||
|   }, | ||||
|   "env": { | ||||
| @@ -111,8 +112,7 @@ | ||||
|     ], | ||||
|     "unused-imports/no-unused-imports": "error", | ||||
|     "lit/attribute-value-entities": "off", | ||||
|     "lit/no-template-map": "off", | ||||
|     "lit/no-template-arrow": "warn" | ||||
|     "lit/no-template-map": "off" | ||||
|   }, | ||||
|   "plugins": ["disable", "unused-imports"], | ||||
|   "processor": "disable/disable" | ||||
|   | ||||
							
								
								
									
										12
									
								
								.yarn/patches/@material/mwc-icon-button/remove-icon.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								.yarn/patches/@material/mwc-icon-button/remove-icon.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| diff --git a/mwc-icon-button-base.js b/mwc-icon-button-base.js | ||||
| index 45cdaab93ccc0a6daaaaabc01266dcdc32e46bfd..b3ea5b541597308d85f86ce6c23fd00785fda835 100644 | ||||
| --- a/mwc-icon-button-base.js | ||||
| +++ b/mwc-icon-button-base.js | ||||
| @@ -63,7 +63,6 @@ export class IconButtonBase extends LitElement { | ||||
|          @touchend="${this.handleRippleDeactivate}" | ||||
|          @touchcancel="${this.handleRippleDeactivate}" | ||||
|      >${this.renderRipple()} | ||||
| -    <i class="material-icons">${this.icon}</i> | ||||
|      <span | ||||
|        ><slot></slot | ||||
|      ></span> | ||||
| @@ -35,6 +35,7 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
|   __BUILD__: JSON.stringify(latestBuild ? "latest" : "es5"), | ||||
|   __VERSION__: JSON.stringify(env.version()), | ||||
|   __DEMO__: false, | ||||
|   __SUPERVISOR__: false, | ||||
|   __BACKWARDS_COMPAT__: false, | ||||
|   __STATIC_PATH__: "/static/", | ||||
|   "process.env.NODE_ENV": JSON.stringify( | ||||
| @@ -164,6 +165,7 @@ module.exports.config = { | ||||
|   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) { | ||||
| @@ -194,6 +196,9 @@ module.exports.config = { | ||||
|       publicPath: publicPath(latestBuild, paths.hassio_publicPath), | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       defineOverlay: { | ||||
|         __SUPERVISOR__: true, | ||||
|       }, | ||||
|     }; | ||||
|   }, | ||||
|  | ||||
| @@ -206,6 +211,9 @@ module.exports.config = { | ||||
|       publicPath: publicPath(latestBuild), | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       defineOverlay: { | ||||
|         __DEMO__: true, | ||||
|       }, | ||||
|     }; | ||||
|   }, | ||||
| }; | ||||
|   | ||||
| @@ -154,6 +154,15 @@ gulp.task("gen-index-cast-dev", (done) => { | ||||
|     contentReceiver | ||||
|   ); | ||||
|  | ||||
|   const contentMedia = renderCastTemplate("media", { | ||||
|     latestMediaJS: "/frontend_latest/media.js", | ||||
|     es5MediaJS: "/frontend_es5/media.js", | ||||
|   }); | ||||
|   fs.outputFileSync( | ||||
|     path.resolve(paths.cast_output_root, "media.html"), | ||||
|     contentMedia | ||||
|   ); | ||||
|  | ||||
|   const contentFAQ = renderCastTemplate("launcher-faq", { | ||||
|     latestLauncherJS: "/frontend_latest/launcher.js", | ||||
|     es5LauncherJS: "/frontend_es5/launcher.js", | ||||
| @@ -192,6 +201,15 @@ gulp.task("gen-index-cast-prod", (done) => { | ||||
|     contentReceiver | ||||
|   ); | ||||
|  | ||||
|   const contentMedia = renderCastTemplate("media", { | ||||
|     latestMediaJS: latestManifest["media.js"], | ||||
|     es5MediaJS: es5Manifest["media.js"], | ||||
|   }); | ||||
|   fs.outputFileSync( | ||||
|     path.resolve(paths.cast_output_root, "media.html"), | ||||
|     contentMedia | ||||
|   ); | ||||
|  | ||||
|   const contentFAQ = renderCastTemplate("launcher-faq", { | ||||
|     latestLauncherJS: latestManifest["launcher.js"], | ||||
|     es5LauncherJS: es5Manifest["launcher.js"], | ||||
|   | ||||
| @@ -22,17 +22,40 @@ const getMeta = () => { | ||||
|     const svg = fs.readFileSync(`${ICON_PATH}/${icon.name}.svg`, { | ||||
|       encoding, | ||||
|     }); | ||||
|     return { path: svg.match(/ d="([^"]+)"/)[1], name: icon.name }; | ||||
|     return { | ||||
|       path: svg.match(/ d="([^"]+)"/)[1], | ||||
|       name: icon.name, | ||||
|       tags: icon.tags, | ||||
|       aliases: icon.aliases, | ||||
|     }; | ||||
|   }); | ||||
| }; | ||||
|  | ||||
| const addRemovedMeta = (meta) => { | ||||
|   const file = fs.readFileSync(REMOVED_ICONS_PATH, { encoding }); | ||||
|   const removed = JSON.parse(file); | ||||
|   const combinedMeta = [...meta, ...removed]; | ||||
|   const removedMeta = removed.map((removeIcon) => ({ | ||||
|     path: removeIcon.path, | ||||
|     name: removeIcon.name, | ||||
|     tags: [], | ||||
|     aliases: [], | ||||
|   })); | ||||
|   const combinedMeta = [...meta, ...removedMeta]; | ||||
|   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 chunks = []; | ||||
|   const CHUNK_SIZE = 50000; | ||||
| @@ -77,8 +100,10 @@ const findDifferentiator = (curString, prevString) => { | ||||
| }; | ||||
|  | ||||
| gulp.task("gen-icons-json", (done) => { | ||||
|   const meta = addRemovedMeta(getMeta()); | ||||
|   const split = splitBySize(meta); | ||||
|   const meta = getMeta(); | ||||
|  | ||||
|   const metaAndRemoved = addRemovedMeta(meta); | ||||
|   const split = splitBySize(metaAndRemoved); | ||||
|  | ||||
|   if (!fs.existsSync(OUTPUT_DIR)) { | ||||
|     fs.mkdirSync(OUTPUT_DIR, { recursive: true }); | ||||
| @@ -116,5 +141,18 @@ gulp.task("gen-icons-json", (done) => { | ||||
|     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(); | ||||
| }); | ||||
|   | ||||
| @@ -1,9 +1,6 @@ | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const path = require("path"); | ||||
|  | ||||
| const env = require("../env"); | ||||
| const paths = require("../paths"); | ||||
|  | ||||
| require("./clean.js"); | ||||
| require("./gen-icons-json.js"); | ||||
| @@ -20,7 +17,6 @@ gulp.task( | ||||
|       process.env.NODE_ENV = "development"; | ||||
|     }, | ||||
|     "clean-hassio", | ||||
|     "gen-icons-json", | ||||
|     "gen-index-hassio-dev", | ||||
|     "build-supervisor-translations", | ||||
|     "copy-translations-supervisor", | ||||
| @@ -37,7 +33,6 @@ gulp.task( | ||||
|       process.env.NODE_ENV = "production"; | ||||
|     }, | ||||
|     "clean-hassio", | ||||
|     "gen-icons-json", | ||||
|     "build-supervisor-translations", | ||||
|     "copy-translations-supervisor", | ||||
|     "build-locale-data", | ||||
|   | ||||
| @@ -4,9 +4,6 @@ const del = require("del"); | ||||
| const path = require("path"); | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const merge = require("gulp-merge-json"); | ||||
| const rename = require("gulp-rename"); | ||||
| const transform = require("gulp-json-transform"); | ||||
| const paths = require("../paths"); | ||||
|  | ||||
| const outDir = "build/locale-data"; | ||||
|   | ||||
| @@ -173,6 +173,7 @@ gulp.task("webpack-dev-server-gallery", () => | ||||
|     compiler: webpack(bothBuilds(createGalleryConfig, { isProdBuild: false })), | ||||
|     contentBase: paths.gallery_output_root, | ||||
|     port: 8100, | ||||
|     listenHost: "0.0.0.0", | ||||
|   }) | ||||
| ); | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										46
									
								
								cast/src/html/media.html.template
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								cast/src/html/media.html.template
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| <!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> | ||||
|     <script> | ||||
|       var _gaq=[['_setAccount','UA-57927901-10'],['_trackPageview']]; | ||||
|       (function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0]; | ||||
|       g.src=('https:'==location.protocol?'//ssl':'//www')+'.google-analytics.com/ga.js'; | ||||
|       s.parentNode.insertBefore(g,s)}(document,'script')); | ||||
|     </script> | ||||
|   </head> | ||||
|   <body> | ||||
|     <%= renderTemplate('_js_base') %> | ||||
|  | ||||
|     <cast-media-player></cast-media-player> | ||||
|  | ||||
|     <script> | ||||
|       import("<%= latestMediaJS %>"); | ||||
|       window.latestJS = true; | ||||
|     </script> | ||||
|  | ||||
|     <script> | ||||
|       if (!window.latestJS) { | ||||
|         <% if (useRollup) { %> | ||||
|           _ls("/static/js/s.min.js").onload = function() { | ||||
|             System.import("<%= es5MediaJS %>"); | ||||
|           }; | ||||
|         <% } else { %> | ||||
|           _ls("<%= es5MediaJS %>"); | ||||
|         <% } %> | ||||
|       } | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
| @@ -1,4 +1,5 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { mdiCast, mdiCastConnected } from "@mdi/js"; | ||||
| import "@polymer/paper-item/paper-icon-item"; | ||||
| import "@polymer/paper-listbox/paper-listbox"; | ||||
| import { Auth, Connection } from "home-assistant-js-websocket"; | ||||
| @@ -17,6 +18,7 @@ import { | ||||
| import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | ||||
| import "../../../../src/components/ha-icon"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { | ||||
|   getLegacyLovelaceCollection, | ||||
|   getLovelaceCollection, | ||||
| @@ -73,7 +75,7 @@ class HcCast extends LitElement { | ||||
|           ? html` | ||||
|               <p class="center-item"> | ||||
|                 <mwc-button raised @click=${this._handleLaunch}> | ||||
|                   <ha-icon icon="hass:cast"></ha-icon> | ||||
|                   <ha-svg-icon .path=${mdiCast}></ha-svg-icon> | ||||
|                   Start Casting | ||||
|                 </mwc-button> | ||||
|               </p> | ||||
| @@ -111,7 +113,7 @@ class HcCast extends LitElement { | ||||
|           ${this.castManager.status | ||||
|             ? html` | ||||
|                 <mwc-button @click=${this._handleLaunch}> | ||||
|                   <ha-icon icon="hass:cast-connected"></ha-icon> | ||||
|                   <ha-svg-icon .path=${mdiCastConnected}></ha-svg-icon> | ||||
|                   Manage | ||||
|                 </mwc-button> | ||||
|               ` | ||||
| @@ -233,7 +235,7 @@ class HcCast extends LitElement { | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|  | ||||
|       mwc-button ha-icon { | ||||
|       mwc-button ha-svg-icon { | ||||
|         margin-right: 8px; | ||||
|         height: 18px; | ||||
|       } | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { mdiCastConnected, mdiCast } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import { | ||||
|   Auth, | ||||
| @@ -19,7 +20,7 @@ import { | ||||
|   loadTokens, | ||||
|   saveTokens, | ||||
| } from "../../../../src/common/auth/token_storage"; | ||||
| import "../../../../src/components/ha-icon"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/layouts/hass-loading-screen"; | ||||
| import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | ||||
| import "./hc-layout"; | ||||
| @@ -127,11 +128,11 @@ export class HcConnect extends LitElement { | ||||
|           <div class="card-actions"> | ||||
|             <mwc-button @click=${this._handleDemo}> | ||||
|               Show Demo | ||||
|               <ha-icon | ||||
|                 .icon=${this.castManager.castState === "CONNECTED" | ||||
|                   ? "hass:cast-connected" | ||||
|                   : "hass:cast"} | ||||
|               ></ha-icon> | ||||
|               <ha-svg-icon | ||||
|                 .path=${this.castManager.castState === "CONNECTED" | ||||
|                   ? mdiCastConnected | ||||
|                   : mdiCast} | ||||
|               ></ha-svg-icon> | ||||
|             </mwc-button> | ||||
|             <div class="spacer"></div> | ||||
|             <mwc-button @click=${this._handleConnect}>Authorize</mwc-button> | ||||
| @@ -307,7 +308,7 @@ export class HcConnect extends LitElement { | ||||
|         color: darkred; | ||||
|       } | ||||
|  | ||||
|       mwc-button ha-icon { | ||||
|       mwc-button ha-svg-icon { | ||||
|         margin-left: 8px; | ||||
|       } | ||||
|  | ||||
|   | ||||
							
								
								
									
										22
									
								
								cast/src/media/entrypoint.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								cast/src/media/entrypoint.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| const castContext = cast.framework.CastReceiverContext.getInstance(); | ||||
|  | ||||
| const playerManager = castContext.getPlayerManager(); | ||||
|  | ||||
| playerManager.setMessageInterceptor( | ||||
|   cast.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 = cast.framework.messages.StreamType.LIVE; | ||||
|       media.contentType = "application/vnd.apple.mpegurl"; | ||||
|       // @ts-ignore | ||||
|       media.hlsVideoSegmentFormat = | ||||
|         cast.framework.messages.HlsVideoSegmentFormat.FMP4; | ||||
|     } | ||||
|     return loadRequestData; | ||||
|   } | ||||
| ); | ||||
|  | ||||
| castContext.start(); | ||||
| @@ -8,6 +8,9 @@ import { ReceivedMessage } from "./types"; | ||||
|  | ||||
| const lovelaceController = new HcMain(); | ||||
| document.body.append(lovelaceController); | ||||
| lovelaceController.addEventListener("cast-view-changed", (ev) => { | ||||
|   playDummyMedia(ev.detail.title); | ||||
| }); | ||||
|  | ||||
| const mediaPlayer = document.createElement("cast-media-player"); | ||||
| mediaPlayer.style.display = "none"; | ||||
| @@ -28,6 +31,31 @@ const setTouchControlsVisibility = (visible: boolean) => { | ||||
|   } | ||||
| }; | ||||
|  | ||||
| let timeOut: number | undefined; | ||||
|  | ||||
| const playDummyMedia = (viewTitle?: string) => { | ||||
|   const loadRequestData = new cast.framework.messages.LoadRequestData(); | ||||
|   loadRequestData.autoplay = true; | ||||
|   loadRequestData.media = new cast.framework.messages.MediaInformation(); | ||||
|   loadRequestData.media.contentId = | ||||
|     "https://cast.home-assistant.io/images/google-nest-hub.png"; | ||||
|   loadRequestData.media.contentType = "image/jpeg"; | ||||
|   loadRequestData.media.streamType = cast.framework.messages.StreamType.NONE; | ||||
|   const metadata = new cast.framework.messages.GenericMediaMetadata(); | ||||
|   metadata.title = viewTitle; | ||||
|   loadRequestData.media.metadata = metadata; | ||||
|  | ||||
|   loadRequestData.requestId = 0; | ||||
|   playerManager.load(loadRequestData); | ||||
|   if (timeOut) { | ||||
|     clearTimeout(timeOut); | ||||
|     timeOut = undefined; | ||||
|   } | ||||
|   if (castContext.getDeviceCapabilities().touch_input_supported) { | ||||
|     timeOut = window.setTimeout(() => playDummyMedia(viewTitle), 540000); // repeat every 9 minutes to keep it active (gets deactivated after 10 minutes) | ||||
|   } | ||||
| }; | ||||
|  | ||||
| const showLovelaceController = () => { | ||||
|   mediaPlayer.style.display = "none"; | ||||
|   lovelaceController.style.display = "initial"; | ||||
| @@ -51,6 +79,7 @@ const showMediaPlayer = () => { | ||||
|       --progress-color: #03a9f4; | ||||
|       --splash-image: url('https://home-assistant.io/images/cast/splash.png'); | ||||
|       --splash-size: cover; | ||||
|       --background-color: #41bdf5; | ||||
|     } | ||||
|     `; | ||||
|     document.head.appendChild(style); | ||||
| @@ -63,22 +92,6 @@ options.customNamespaces = { | ||||
|   [CAST_NS]: cast.framework.system.MessageType.JSON, | ||||
| }; | ||||
|  | ||||
| // The docs say we need to set options.touchScreenOptimizeApp = true | ||||
| // https://developers.google.com/cast/docs/caf_receiver/customize_ui#accessing_ui_controls | ||||
| // This doesn't work. | ||||
| // @ts-ignore | ||||
| options.touchScreenOptimizedApp = true; | ||||
|  | ||||
| // The class reference say we can set a uiConfig in options to set it | ||||
| // https://developers.google.com/cast/docs/reference/caf_receiver/cast.framework.CastReceiverOptions#uiConfig | ||||
| // This doesn't work either. | ||||
| // @ts-ignore | ||||
| options.uiConfig = new cast.framework.ui.UiConfig(); | ||||
| // @ts-ignore | ||||
| options.uiConfig.touchScreenOptimizedApp = true; | ||||
|  | ||||
| castContext.setInactivityTimeout(86400); // 1 day | ||||
|  | ||||
| castContext.addCustomMessageListener( | ||||
|   CAST_NS, | ||||
|   // @ts-ignore | ||||
| @@ -103,6 +116,12 @@ const playerManager = castContext.getPlayerManager(); | ||||
| playerManager.setMessageInterceptor( | ||||
|   cast.framework.messages.MessageType.LOAD, | ||||
|   (loadRequestData) => { | ||||
|     if ( | ||||
|       loadRequestData.media.contentId === | ||||
|       "https://cast.home-assistant.io/images/google-nest-hub.png" | ||||
|     ) { | ||||
|       return loadRequestData; | ||||
|     } | ||||
|     // We received a play media command, hide Lovelace and show media player | ||||
|     showMediaPlayer(); | ||||
|     const media = loadRequestData.media; | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { LovelaceConfig } from "../../../../src/data/lovelace"; | ||||
| import { Lovelace } from "../../../../src/panels/lovelace/types"; | ||||
| import "../../../../src/panels/lovelace/views/hui-view"; | ||||
| @@ -14,7 +15,7 @@ class HcLovelace extends LitElement { | ||||
|  | ||||
|   @property() public viewPath?: string | number; | ||||
|  | ||||
|   public urlPath?: string | null; | ||||
|   @property() public urlPath: string | null = null; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const index = this._viewIndex; | ||||
| @@ -30,7 +31,7 @@ class HcLovelace extends LitElement { | ||||
|       config: this.lovelaceConfig, | ||||
|       rawConfig: this.lovelaceConfig, | ||||
|       editMode: false, | ||||
|       urlPath: this.urlPath!, | ||||
|       urlPath: this.urlPath, | ||||
|       enableFullEditMode: () => undefined, | ||||
|       mode: "storage", | ||||
|       locale: this.hass.locale, | ||||
| @@ -54,6 +55,21 @@ class HcLovelace extends LitElement { | ||||
|       const index = this._viewIndex; | ||||
|  | ||||
|       if (index !== undefined) { | ||||
|         const dashboardTitle = this.lovelaceConfig.title || this.urlPath; | ||||
|  | ||||
|         const viewTitle = | ||||
|           this.lovelaceConfig.views[index].title || | ||||
|           this.lovelaceConfig.views[index].path; | ||||
|  | ||||
|         fireEvent(this, "cast-view-changed", { | ||||
|           title: | ||||
|             dashboardTitle || viewTitle | ||||
|               ? `${dashboardTitle || ""}${ | ||||
|                   dashboardTitle && viewTitle ? ": " : "" | ||||
|                 }${viewTitle || ""}` | ||||
|               : undefined, | ||||
|         }); | ||||
|  | ||||
|         const configBackground = | ||||
|           this.lovelaceConfig.views[index].background || | ||||
|           this.lovelaceConfig.background; | ||||
| @@ -101,8 +117,15 @@ class HcLovelace extends LitElement { | ||||
|   } | ||||
| } | ||||
|  | ||||
| export interface CastViewChanged { | ||||
|   title: string | undefined; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "hc-lovelace": HcLovelace; | ||||
|   } | ||||
|   interface HASSDomEvents { | ||||
|     "cast-view-changed": CastViewChanged; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -13,7 +13,11 @@ import { | ||||
|   ShowDemoMessage, | ||||
|   ShowLovelaceViewMessage, | ||||
| } from "../../../../src/cast/receiver_messages"; | ||||
| import { ReceiverStatusMessage } from "../../../../src/cast/sender_messages"; | ||||
| import { | ||||
|   ReceiverErrorCode, | ||||
|   ReceiverErrorMessage, | ||||
|   ReceiverStatusMessage, | ||||
| } from "../../../../src/cast/sender_messages"; | ||||
| import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { isNavigationClick } from "../../../../src/common/dom/is-navigation-click"; | ||||
| import { | ||||
| @@ -40,9 +44,9 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
|   private _unsubLovelace?: UnsubscribeFunc; | ||||
|   @state() private _urlPath?: string | null; | ||||
|  | ||||
|   private _urlPath?: string | null; | ||||
|   private _unsubLovelace?: UnsubscribeFunc; | ||||
|  | ||||
|   public processIncomingMessage(msg: HassMessage) { | ||||
|     if (msg.type === "connect") { | ||||
| @@ -68,8 +72,10 @@ export class HcMain extends HassElement { | ||||
|       !this._lovelaceConfig || | ||||
|       this._lovelacePath === null || | ||||
|       // Guard against part of HA not being loaded yet. | ||||
|       (this.hass && | ||||
|         (!this.hass.states || !this.hass.config || !this.hass.services)) | ||||
|       !this.hass || | ||||
|       !this.hass.states || | ||||
|       !this.hass.config || | ||||
|       !this.hass.services | ||||
|     ) { | ||||
|       return html` | ||||
|         <hc-launch-screen | ||||
| @@ -107,6 +113,7 @@ export class HcMain extends HassElement { | ||||
|         this._sendStatus(); | ||||
|       } | ||||
|     }); | ||||
|     this.addEventListener("dialog-closed", this._dialogClosed); | ||||
|   } | ||||
|  | ||||
|   private _sendStatus(senderId?: string) { | ||||
| @@ -118,7 +125,7 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|     if (this.hass) { | ||||
|       status.hassUrl = this.hass.auth.data.hassUrl; | ||||
|       status.lovelacePath = this._lovelacePath!; | ||||
|       status.lovelacePath = this._lovelacePath; | ||||
|       status.urlPath = this._urlPath; | ||||
|     } | ||||
|  | ||||
| @@ -131,6 +138,30 @@ export class HcMain extends HassElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _sendError( | ||||
|     error_code: number, | ||||
|     error_message: string, | ||||
|     senderId?: string | ||||
|   ) { | ||||
|     const error: ReceiverErrorMessage = { | ||||
|       type: "receiver_error", | ||||
|       error_code, | ||||
|       error_message, | ||||
|     }; | ||||
|  | ||||
|     if (senderId) { | ||||
|       this.sendMessage(senderId, error); | ||||
|     } else { | ||||
|       for (const sender of castContext.getSenders()) { | ||||
|         this.sendMessage(sender.id, error); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _dialogClosed = () => { | ||||
|     document.body.setAttribute("style", "overflow-y: auto !important"); | ||||
|   }; | ||||
|  | ||||
|   private async _handleGetStatusMessage(msg: GetStatusMessage) { | ||||
|     this._sendStatus(msg.senderId!); | ||||
|   } | ||||
| @@ -149,14 +180,18 @@ export class HcMain extends HassElement { | ||||
|         }), | ||||
|       }); | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       const errorMessage = this._getErrorMessage(err); | ||||
|       this._error = errorMessage; | ||||
|       this._sendError(err, errorMessage); | ||||
|       return; | ||||
|     } | ||||
|     let connection; | ||||
|     try { | ||||
|       connection = await createConnection({ auth }); | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       const errorMessage = this._getErrorMessage(err); | ||||
|       this._error = errorMessage; | ||||
|       this._sendError(err, errorMessage); | ||||
|       return; | ||||
|     } | ||||
|     if (this.hass) { | ||||
| @@ -168,24 +203,29 @@ export class HcMain extends HassElement { | ||||
|   } | ||||
|  | ||||
|   private async _handleShowLovelaceMessage(msg: ShowLovelaceViewMessage) { | ||||
|     this._showDemo = false; | ||||
|     // We should not get this command before we are connected. | ||||
|     // Means a client got out of sync. Let's send status to them. | ||||
|     if (!this.hass) { | ||||
|       this._sendStatus(msg.senderId!); | ||||
|       this._error = "Cannot show Lovelace because we're not connected."; | ||||
|       this._sendError(ReceiverErrorCode.NOT_CONNECTED, this._error); | ||||
|       return; | ||||
|     } | ||||
|     this._error = undefined; | ||||
|     if (msg.urlPath === "lovelace") { | ||||
|       msg.urlPath = null; | ||||
|     } | ||||
|     this._lovelacePath = msg.viewPath; | ||||
|     if (!this._unsubLovelace || this._urlPath !== msg.urlPath) { | ||||
|       this._urlPath = msg.urlPath; | ||||
|       this._lovelaceConfig = undefined; | ||||
|       if (this._unsubLovelace) { | ||||
|         this._unsubLovelace(); | ||||
|       } | ||||
|       const llColl = atLeastVersion(this.hass.connection.haVersion, 0, 107) | ||||
|         ? getLovelaceCollection(this.hass!.connection, msg.urlPath) | ||||
|         : getLegacyLovelaceCollection(this.hass!.connection); | ||||
|         ? getLovelaceCollection(this.hass.connection, msg.urlPath) | ||||
|         : getLegacyLovelaceCollection(this.hass.connection); | ||||
|       // We first do a single refresh because we need to check if there is LL | ||||
|       // configuration. | ||||
|       try { | ||||
| @@ -194,8 +234,16 @@ export class HcMain extends HassElement { | ||||
|           this._handleNewLovelaceConfig(lovelaceConfig) | ||||
|         ); | ||||
|       } catch (err: any) { | ||||
|         // eslint-disable-next-line | ||||
|         console.log("Error fetching Lovelace configuration", err, msg); | ||||
|         if ( | ||||
|           atLeastVersion(this.hass.connection.haVersion, 0, 107) && | ||||
|           err.code !== "config_not_found" | ||||
|         ) { | ||||
|           // eslint-disable-next-line | ||||
|           console.log("Error fetching Lovelace configuration", err, msg); | ||||
|           this._error = `Error fetching Lovelace configuration: ${err.message}`; | ||||
|           this._sendError(ReceiverErrorCode.FETCH_CONFIG_FAILED, this._error); | ||||
|           return; | ||||
|         } | ||||
|         // Generate a Lovelace config. | ||||
|         this._unsubLovelace = () => undefined; | ||||
|         await this._generateLovelaceConfig(); | ||||
| @@ -210,8 +258,6 @@ export class HcMain extends HassElement { | ||||
|         loadLovelaceResources(resources, this.hass!.auth.data.hassUrl); | ||||
|       } | ||||
|     } | ||||
|     this._showDemo = false; | ||||
|     this._lovelacePath = msg.viewPath; | ||||
|  | ||||
|     this._sendStatus(); | ||||
|   } | ||||
| @@ -232,7 +278,7 @@ export class HcMain extends HassElement { | ||||
|   } | ||||
|  | ||||
|   private _handleNewLovelaceConfig(lovelaceConfig: LovelaceConfig) { | ||||
|     castContext.setApplicationState(lovelaceConfig.title!); | ||||
|     castContext.setApplicationState(lovelaceConfig.title || ""); | ||||
|     this._lovelaceConfig = lovelaceConfig; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import { mdiTelevision } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { CastManager } from "../../../src/cast/cast_manager"; | ||||
| @@ -27,7 +28,7 @@ class CastDemoRow extends LitElement implements LovelaceRow { | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <ha-icon icon="hademo:television"></ha-icon> | ||||
|       <ha-svg-icon .path=${mdiTelevision}></ha-svg-icon> | ||||
|       <div class="flex"> | ||||
|         <div class="name">Show Chromecast interface</div> | ||||
|         <google-cast-launcher></google-cast-launcher> | ||||
| @@ -72,7 +73,7 @@ class CastDemoRow extends LitElement implements LovelaceRow { | ||||
|         display: flex; | ||||
|         align-items: center; | ||||
|       } | ||||
|       ha-icon { | ||||
|       ha-svg-icon { | ||||
|         padding: 8px; | ||||
|         color: var(--paper-item-icon-color); | ||||
|       } | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										7
									
								
								demo/src/stubs/area_registry.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								demo/src/stubs/area_registry.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| import { AreaRegistryEntry } from "../../../src/data/area_registry"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockAreaRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: AreaRegistryEntry[] = [] | ||||
| ) => hass.mockWS("config/area_registry/list", () => data); | ||||
							
								
								
									
										7
									
								
								demo/src/stubs/device_registry.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								demo/src/stubs/device_registry.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| import { DeviceRegistryEntry } from "../../../src/data/device_registry"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockDeviceRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: DeviceRegistryEntry[] = [] | ||||
| ) => hass.mockWS("config/device_registry/list", () => data); | ||||
							
								
								
									
										7
									
								
								demo/src/stubs/entity_registry.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								demo/src/stubs/entity_registry.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| import { EntityRegistryEntry } from "../../../src/data/entity_registry"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockEntityRegistry = ( | ||||
|   hass: MockHomeAssistant, | ||||
|   data: EntityRegistryEntry[] = [] | ||||
| ) => hass.mockWS("config/entity_registry/list", () => data); | ||||
							
								
								
									
										59
									
								
								demo/src/stubs/hassio_supervisor.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								demo/src/stubs/hassio_supervisor.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| import { HassioSupervisorInfo } from "../../../src/data/hassio/supervisor"; | ||||
| import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockHassioSupervisor = (hass: MockHomeAssistant) => { | ||||
|   hass.config.components.push("hassio"); | ||||
|   hass.mockWS("supervisor/api", (msg) => { | ||||
|     if (msg.endpoint === "/supervisor/info") { | ||||
|       const data: HassioSupervisorInfo = { | ||||
|         version: "2021.10.dev0805", | ||||
|         version_latest: "2021.10.dev0806", | ||||
|         update_available: true, | ||||
|         channel: "dev", | ||||
|         arch: "aarch64", | ||||
|         supported: true, | ||||
|         healthy: true, | ||||
|         ip_address: "172.30.32.2", | ||||
|         wait_boot: 5, | ||||
|         timezone: "America/Los_Angeles", | ||||
|         logging: "info", | ||||
|         debug: false, | ||||
|         debug_block: false, | ||||
|         diagnostics: true, | ||||
|         addons: [ | ||||
|           { | ||||
|             name: "Visual Studio Code", | ||||
|             slug: "a0d7b954_vscode", | ||||
|             description: | ||||
|               "Fully featured VSCode experience, to edit your HA config in the browser, including auto-completion!", | ||||
|             state: "started", | ||||
|             version: "3.6.2", | ||||
|             version_latest: "3.6.2", | ||||
|             update_available: false, | ||||
|             repository: "a0d7b954", | ||||
|             icon: true, | ||||
|             logo: true, | ||||
|           }, | ||||
|           { | ||||
|             name: "Z-Wave JS", | ||||
|             slug: "core_zwave_js", | ||||
|             description: | ||||
|               "Control a ZWave network with Home Assistant Z-Wave JS", | ||||
|             state: "started", | ||||
|             version: "0.1.45", | ||||
|             version_latest: "0.1.45", | ||||
|             update_available: false, | ||||
|             repository: "core", | ||||
|             icon: true, | ||||
|             logo: true, | ||||
|           }, | ||||
|         ] as any, | ||||
|         addons_repositories: [ | ||||
|           "https://github.com/hassio-addons/repository", | ||||
|         ] as any, | ||||
|       }; | ||||
|       return data; | ||||
|     } | ||||
|     return Promise.reject(`${msg.method} ${msg.endpoint} is not implemented`); | ||||
|   }); | ||||
| }; | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/office.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/office.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 147 KiB | 
							
								
								
									
										35
									
								
								gallery/script/netlify_build_gallery
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										35
									
								
								gallery/script/netlify_build_gallery
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,35 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| TARGET_LABEL="Needs gallery preview" | ||||
|  | ||||
| if [[ "$NETLIFY" != "true" ]]; then | ||||
|   echo "This script can only be run on Netlify" | ||||
|   exit 1 | ||||
| fi | ||||
|  | ||||
| function createStatus() { | ||||
|   state="$1" | ||||
|   description="$2" | ||||
|   target_url="$3" | ||||
|   curl -X POST -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/statuses/$COMMIT_REF" \ | ||||
|     -d '{"state": "'"${state}"'", "context": "Netlify/Gallery Preview Build", "description": "'"$description"'", "target_url":  "'"$target_url"'"}' | ||||
| } | ||||
|  | ||||
|  | ||||
| if [[ "${PULL_REQUEST}" == "false" ]]; then | ||||
|   gulp build-gallery | ||||
| else | ||||
|   if [[ "$(curl -sSLf -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/pulls/${REVIEW_ID}" | jq '.labels[].name' -r)" =~ "$TARGET_LABEL" ]]; then | ||||
|     createStatus "pending" "Building gallery preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     gulp build-gallery | ||||
|     if [ $? -eq 0 ]; then | ||||
|       createStatus "success" "Build complete" "$DEPLOY_URL" | ||||
|     else | ||||
|       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     fi | ||||
|   else | ||||
|     createStatus "success" "Build was not requested by PR label" | ||||
|   fi | ||||
| fi | ||||
							
								
								
									
										143
									
								
								gallery/src/components/demo-black-white-row.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										143
									
								
								gallery/src/components/demo-black-white-row.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,143 @@ | ||||
| import { Button } from "@material/mwc-button"; | ||||
| import { html, LitElement, css, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
|  | ||||
| @customElement("demo-black-white-row") | ||||
| class DemoBlackWhiteRow extends LitElement { | ||||
|   @property() title!: string; | ||||
|  | ||||
|   @property() value!: any; | ||||
|  | ||||
|   @property() disabled = false; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <div class="row"> | ||||
|         <div class="content light"> | ||||
|           <ha-card .header=${this.title}> | ||||
|             <div class="card-content"> | ||||
|               <slot name="light"></slot> | ||||
|             </div> | ||||
|             <div class="card-actions"> | ||||
|               <mwc-button | ||||
|                 .disabled=${this.disabled} | ||||
|                 @click=${this.handleSubmit} | ||||
|               > | ||||
|                 Submit | ||||
|               </mwc-button> | ||||
|             </div> | ||||
|           </ha-card> | ||||
|         </div> | ||||
|         <div class="content dark"> | ||||
|           <ha-card .header=${this.title}> | ||||
|             <div class="card-content"> | ||||
|               <slot name="dark"></slot> | ||||
|             </div> | ||||
|             <div class="card-actions"> | ||||
|               <mwc-button | ||||
|                 .disabled=${this.disabled} | ||||
|                 @click=${this.handleSubmit} | ||||
|               > | ||||
|                 Submit | ||||
|               </mwc-button> | ||||
|             </div> | ||||
|           </ha-card> | ||||
|           <pre>${JSON.stringify(this.value, undefined, 2)}</pre> | ||||
|         </div> | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: "default", | ||||
|         themes: {}, | ||||
|         darkMode: false, | ||||
|       }, | ||||
|       "default", | ||||
|       { dark: true } | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   handleSubmit(ev) { | ||||
|     const content = (ev.target as Button).closest(".content")!; | ||||
|     fireEvent(this, "submitted" as any, { | ||||
|       slot: content.classList.contains("light") ? "light" : "dark", | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .row { | ||||
|       display: flex; | ||||
|     } | ||||
|     .content { | ||||
|       padding: 50px 0; | ||||
|       background-color: var(--primary-background-color); | ||||
|     } | ||||
|     .light { | ||||
|       flex: 1; | ||||
|       padding-left: 50px; | ||||
|       padding-right: 50px; | ||||
|       box-sizing: border-box; | ||||
|     } | ||||
|     .light ha-card { | ||||
|       margin-left: auto; | ||||
|     } | ||||
|     .dark { | ||||
|       display: flex; | ||||
|       flex: 1; | ||||
|       padding-left: 50px; | ||||
|       box-sizing: border-box; | ||||
|       flex-wrap: wrap; | ||||
|     } | ||||
|     ha-card { | ||||
|       width: 400px; | ||||
|     } | ||||
|     pre { | ||||
|       width: 300px; | ||||
|       margin: 0 16px 0; | ||||
|       overflow: auto; | ||||
|       color: var(--primary-text-color); | ||||
|     } | ||||
|     .card-actions { | ||||
|       display: flex; | ||||
|       flex-direction: row-reverse; | ||||
|       border-top: none; | ||||
|     } | ||||
|     @media only screen and (max-width: 1500px) { | ||||
|       .light { | ||||
|         flex: initial; | ||||
|       } | ||||
|     } | ||||
|     @media only screen and (max-width: 1000px) { | ||||
|       .light, | ||||
|       .dark { | ||||
|         padding: 16px; | ||||
|       } | ||||
|       .row, | ||||
|       .dark { | ||||
|         flex-direction: column; | ||||
|       } | ||||
|       ha-card { | ||||
|         margin: 0 auto; | ||||
|         width: 100%; | ||||
|         max-width: 400px; | ||||
|       } | ||||
|       pre { | ||||
|         margin: 16px auto; | ||||
|       } | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-black-white-row": DemoBlackWhiteRow; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										91
									
								
								gallery/src/demos/demo-automation-editor-action.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								gallery/src/demos/demo-automation-editor-action.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../demo/src/stubs/hassio_supervisor"; | ||||
| import "../../../src/panels/config/automation/action/ha-automation-action"; | ||||
| import { HaChooseAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-choose"; | ||||
| import { HaDelayAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-delay"; | ||||
| import { HaDeviceAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-device_id"; | ||||
| import { HaEventAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-event"; | ||||
| import { HaRepeatAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-repeat"; | ||||
| import { HaSceneAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-scene"; | ||||
| import { HaServiceAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-service"; | ||||
| import { HaWaitForTriggerAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-wait_for_trigger"; | ||||
| import { HaWaitAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | ||||
| import { Action } from "../../../src/data/script"; | ||||
| import { HaConditionAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
|  | ||||
| const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
|   { name: "Event", actions: [HaEventAction.defaultConfig] }, | ||||
|   { name: "Device", actions: [HaDeviceAction.defaultConfig] }, | ||||
|   { name: "Service", actions: [HaServiceAction.defaultConfig] }, | ||||
|   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, | ||||
|   { name: "Delay", actions: [HaDelayAction.defaultConfig] }, | ||||
|   { name: "Scene", actions: [HaSceneAction.defaultConfig] }, | ||||
|   { name: "Wait", actions: [HaWaitAction.defaultConfig] }, | ||||
|   { name: "WaitForTrigger", actions: [HaWaitForTriggerAction.defaultConfig] }, | ||||
|   { name: "Repeat", actions: [HaRepeatAction.defaultConfig] }, | ||||
|   { name: "Choose", actions: [HaChooseAction.defaultConfig] }, | ||||
|   { name: "Variables", actions: [{ variables: { hello: "1" } }] }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-automation-editor-action") | ||||
| class DemoHaAutomationEditorAction extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.actions); | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const valueChanged = (ev) => { | ||||
|       const sampleIdx = ev.target.sampleIdx; | ||||
|       this.data[sampleIdx] = ev.detail.value; | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
|             .title=${info.name} | ||||
|             .value=${this.data[sampleIdx]} | ||||
|           > | ||||
|             ${["light", "dark"].map( | ||||
|               (slot) => | ||||
|                 html` | ||||
|                   <ha-automation-action | ||||
|                     slot=${slot} | ||||
|                     .hass=${this.hass} | ||||
|                     .actions=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-action> | ||||
|                 ` | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-automation-editor-action": DemoHaAutomationEditorAction; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										127
									
								
								gallery/src/demos/demo-automation-editor-condition.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								gallery/src/demos/demo-automation-editor-condition.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../demo/src/stubs/hassio_supervisor"; | ||||
| import type { Condition } from "../../../src/data/automation"; | ||||
| import "../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||
| import { HaDeviceCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||
| import { HaLogicalCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | ||||
| import HaNumericStateCondition from "../../../src/panels/config/automation/condition/types/ha-automation-condition-numeric_state"; | ||||
| import { HaStateCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-state"; | ||||
| import { HaSunCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-sun"; | ||||
| import { HaTemplateCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-template"; | ||||
| import { HaTimeCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-time"; | ||||
| import { HaTriggerCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||
| import { HaZoneCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||
|  | ||||
| const SCHEMAS: { name: string; conditions: Condition[] }[] = [ | ||||
|   { | ||||
|     name: "State", | ||||
|     conditions: [{ condition: "state", ...HaStateCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Numeric State", | ||||
|     conditions: [ | ||||
|       { condition: "numeric_state", ...HaNumericStateCondition.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     name: "Sun", | ||||
|     conditions: [{ condition: "sun", ...HaSunCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Zone", | ||||
|     conditions: [{ condition: "zone", ...HaZoneCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Time", | ||||
|     conditions: [{ condition: "time", ...HaTimeCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Template", | ||||
|     conditions: [ | ||||
|       { condition: "template", ...HaTemplateCondition.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     name: "Device", | ||||
|     conditions: [{ condition: "device", ...HaDeviceCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "And", | ||||
|     conditions: [{ condition: "and", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Or", | ||||
|     conditions: [{ condition: "or", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Not", | ||||
|     conditions: [{ condition: "not", ...HaLogicalCondition.defaultConfig }], | ||||
|   }, | ||||
|   { | ||||
|     name: "Trigger", | ||||
|     conditions: [{ condition: "trigger", ...HaTriggerCondition.defaultConfig }], | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-automation-editor-condition") | ||||
| class DemoHaAutomationEditorCondition extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.conditions); | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const valueChanged = (ev) => { | ||||
|       const sampleIdx = ev.target.sampleIdx; | ||||
|       this.data[sampleIdx] = ev.detail.value; | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
|             .title=${info.name} | ||||
|             .value=${this.data[sampleIdx]} | ||||
|           > | ||||
|             ${["light", "dark"].map( | ||||
|               (slot) => | ||||
|                 html` | ||||
|                   <ha-automation-condition | ||||
|                     slot=${slot} | ||||
|                     .hass=${this.hass} | ||||
|                     .conditions=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-condition> | ||||
|                 ` | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-automation-editor-condition": DemoHaAutomationEditorCondition; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										159
									
								
								gallery/src/demos/demo-automation-editor-trigger.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								gallery/src/demos/demo-automation-editor-trigger.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { mockEntityRegistry } from "../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../demo/src/stubs/hassio_supervisor"; | ||||
| import type { Trigger } from "../../../src/data/automation"; | ||||
| import { HaGeolocationTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-geo_location"; | ||||
| import { HaEventTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-event"; | ||||
| import { HaHassTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-homeassistant"; | ||||
| import { HaNumericStateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-numeric_state"; | ||||
| import { HaSunTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-sun"; | ||||
| import { HaTagTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-tag"; | ||||
| import { HaTemplateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-template"; | ||||
| import { HaTimeTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time"; | ||||
| import { HaTimePatternTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time_pattern"; | ||||
| import { HaWebhookTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-webhook"; | ||||
| import { HaZoneTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-zone"; | ||||
| import { HaDeviceTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-device"; | ||||
| import { HaStateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-state"; | ||||
| import { HaMQTTTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-mqtt"; | ||||
| import "../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
|  | ||||
| const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
|   { | ||||
|     name: "State", | ||||
|     triggers: [{ platform: "state", ...HaStateTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "MQTT", | ||||
|     triggers: [{ platform: "mqtt", ...HaMQTTTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "GeoLocation", | ||||
|     triggers: [ | ||||
|       { platform: "geo_location", ...HaGeolocationTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Home Assistant", | ||||
|     triggers: [{ platform: "homeassistant", ...HaHassTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Numeric State", | ||||
|     triggers: [ | ||||
|       { platform: "numeric_state", ...HaNumericStateTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Sun", | ||||
|     triggers: [{ platform: "sun", ...HaSunTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Time Pattern", | ||||
|     triggers: [ | ||||
|       { platform: "time_pattern", ...HaTimePatternTrigger.defaultConfig }, | ||||
|     ], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Webhook", | ||||
|     triggers: [{ platform: "webhook", ...HaWebhookTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Zone", | ||||
|     triggers: [{ platform: "zone", ...HaZoneTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Tag", | ||||
|     triggers: [{ platform: "tag", ...HaTagTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Time", | ||||
|     triggers: [{ platform: "time", ...HaTimeTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Template", | ||||
|     triggers: [{ platform: "template", ...HaTemplateTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Event", | ||||
|     triggers: [{ platform: "event", ...HaEventTrigger.defaultConfig }], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     name: "Device Trigger", | ||||
|     triggers: [{ platform: "device", ...HaDeviceTrigger.defaultConfig }], | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-automation-editor-trigger") | ||||
| class DemoHaAutomationEditorTrigger extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   private data: any = SCHEMAS.map((info) => info.triggers); | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const valueChanged = (ev) => { | ||||
|       const sampleIdx = ev.target.sampleIdx; | ||||
|       this.data[sampleIdx] = ev.detail.value; | ||||
|       this.requestUpdate(); | ||||
|     }; | ||||
|     return html` | ||||
|       ${SCHEMAS.map( | ||||
|         (info, sampleIdx) => html` | ||||
|           <demo-black-white-row | ||||
|             .title=${info.name} | ||||
|             .value=${this.data[sampleIdx]} | ||||
|           > | ||||
|             ${["light", "dark"].map( | ||||
|               (slot) => | ||||
|                 html` | ||||
|                   <ha-automation-trigger | ||||
|                     slot=${slot} | ||||
|                     .hass=${this.hass} | ||||
|                     .triggers=${this.data[sampleIdx]} | ||||
|                     .sampleIdx=${sampleIdx} | ||||
|                     @value-changed=${valueChanged} | ||||
|                   ></ha-automation-trigger> | ||||
|                 ` | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-automation-editor-trigger": DemoHaAutomationEditorTrigger; | ||||
|   } | ||||
| } | ||||
| @@ -1,15 +1,19 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-logo-svg"; | ||||
|  | ||||
| const alerts: { | ||||
|   title?: string; | ||||
|   description: string | TemplateResult; | ||||
|   type: "info" | "warning" | "error" | "success"; | ||||
|   dismissable?: boolean; | ||||
|   action?: string; | ||||
|   rtl?: boolean; | ||||
|   iconSlot?: TemplateResult; | ||||
|   actionSlot?: TemplateResult; | ||||
| }[] = [ | ||||
|   { | ||||
|     title: "Test info alert", | ||||
| @@ -73,13 +77,35 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Unsaved data", | ||||
|     description: "You have unsaved data", | ||||
|     type: "warning", | ||||
|     action: "save", | ||||
|     actionSlot: html`<mwc-button slot="action" label="save"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted icon", | ||||
|     description: "Alert with slotted icon", | ||||
|     type: "warning", | ||||
|     iconSlot: html`<span slot="icon" class="image"> | ||||
|       <ha-logo-svg></ha-logo-svg> | ||||
|     </span>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted image", | ||||
|     description: "Alert with slotted image", | ||||
|     type: "warning", | ||||
|     iconSlot: html`<span slot="icon" class="image" | ||||
|       ><img src="https://www.home-assistant.io/images/home-assistant-logo.svg" | ||||
|     /></span>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted action", | ||||
|     description: "Alert with slotted action", | ||||
|     type: "info", | ||||
|     actionSlot: html`<mwc-button slot="action" label="action"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     description: "Dismissable information (RTL)", | ||||
| @@ -91,7 +117,7 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action (RTL)", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|     rtl: true, | ||||
|   }, | ||||
|   { | ||||
| @@ -106,30 +132,60 @@ const alerts: { | ||||
| export class DemoHaAlert extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-alert demo"> | ||||
|         <div class="card-content"> | ||||
|           ${alerts.map( | ||||
|             (alert) => html` | ||||
|               <ha-alert | ||||
|                 .title=${alert.title || ""} | ||||
|                 .alertType=${alert.type} | ||||
|                 .dismissable=${alert.dismissable || false} | ||||
|                 .actionText=${alert.action || ""} | ||||
|                 .rtl=${alert.rtl || false} | ||||
|               > | ||||
|                 ${alert.description} | ||||
|               </ha-alert> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       </ha-card> | ||||
|       ${["light", "dark"].map( | ||||
|         (mode) => html` | ||||
|           <div class=${mode}> | ||||
|             <ha-card header="ha-alert ${mode} demo"> | ||||
|               <div class="card-content"> | ||||
|                 ${alerts.map( | ||||
|                   (alert) => html` | ||||
|                     <ha-alert | ||||
|                       .title=${alert.title || ""} | ||||
|                       .alertType=${alert.type} | ||||
|                       .dismissable=${alert.dismissable || false} | ||||
|                       .rtl=${alert.rtl || false} | ||||
|                     > | ||||
|                       ${alert.iconSlot} ${alert.description} ${alert.actionSlot} | ||||
|                     </ha-alert> | ||||
|                   ` | ||||
|                 )} | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           </div> | ||||
|         ` | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: "default", | ||||
|         themes: {}, | ||||
|         darkMode: false, | ||||
|       }, | ||||
|       "default", | ||||
|       { dark: true } | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-content: space-between; | ||||
|       } | ||||
|       .dark, | ||||
|       .light { | ||||
|         display: block; | ||||
|         background-color: var(--primary-background-color); | ||||
|         padding: 0 50px; | ||||
|       } | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       ha-alert { | ||||
| @@ -142,8 +198,17 @@ export class DemoHaAlert extends LitElement { | ||||
|         align-items: center; | ||||
|         justify-content: space-between; | ||||
|       } | ||||
|       span { | ||||
|         margin-right: 16px; | ||||
|       .image { | ||||
|         display: inline-flex; | ||||
|         height: 100%; | ||||
|         align-items: center; | ||||
|       } | ||||
|       img { | ||||
|         max-height: 24px; | ||||
|         width: 24px; | ||||
|       } | ||||
|       mwc-button { | ||||
|         --mdc-theme-primary: var(--primary-text-color); | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
|   | ||||
							
								
								
									
										85
									
								
								gallery/src/demos/demo-ha-bar.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								gallery/src/demos/demo-ha-bar.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import "../../../src/components/ha-bar"; | ||||
| import "../../../src/components/ha-card"; | ||||
|  | ||||
| const bars: { | ||||
|   min?: number; | ||||
|   max?: number; | ||||
|   value: number; | ||||
|   warning?: number; | ||||
|   error?: number; | ||||
| }[] = [ | ||||
|   { | ||||
|     value: 33, | ||||
|   }, | ||||
|   { | ||||
|     value: 150, | ||||
|   }, | ||||
|   { | ||||
|     min: -10, | ||||
|     value: 0, | ||||
|   }, | ||||
|   { | ||||
|     value: 80, | ||||
|   }, | ||||
|   { | ||||
|     value: 200, | ||||
|     max: 13, | ||||
|   }, | ||||
|   { | ||||
|     value: 4, | ||||
|     min: 13, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-bar") | ||||
| export class DemoHaBar extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${bars | ||||
|         .map((bar) => ({ min: 0, max: 100, warning: 70, error: 90, ...bar })) | ||||
|         .map( | ||||
|           (bar) => html` | ||||
|             <ha-card> | ||||
|               <div class="card-content"> | ||||
|                 <pre>Config: ${JSON.stringify(bar)}</pre> | ||||
|                 <ha-bar | ||||
|                   class=${classMap({ | ||||
|                     warning: bar.value > bar.warning, | ||||
|                     error: bar.value > bar.error, | ||||
|                   })} | ||||
|                   .min=${bar.min} | ||||
|                   .max=${bar.max} | ||||
|                   .value=${bar.value} | ||||
|                 > | ||||
|                 </ha-bar> | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           ` | ||||
|         )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       .warning { | ||||
|         --ha-bar-primary-color: var(--warning-color); | ||||
|       } | ||||
|       .error { | ||||
|         --ha-bar-primary-color: var(--error-color); | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-bar": DemoHaBar; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										86
									
								
								gallery/src/demos/demo-ha-chips.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										86
									
								
								gallery/src/demos/demo-ha-chips.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,86 @@ | ||||
| import { mdiHomeAssistant } from "@mdi/js"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-chip"; | ||||
| import "../../../src/components/ha-chip-set"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
|  | ||||
| const chips: { | ||||
|   icon?: string; | ||||
|   content?: string; | ||||
| }[] = [ | ||||
|   {}, | ||||
|   { | ||||
|     icon: mdiHomeAssistant, | ||||
|   }, | ||||
|   { | ||||
|     content: "Content", | ||||
|   }, | ||||
|   { | ||||
|     icon: mdiHomeAssistant, | ||||
|     content: "Content", | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-chips") | ||||
| export class DemoHaChips extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-chip demo"> | ||||
|         <div class="card-content"> | ||||
|           ${chips.map( | ||||
|             (chip) => html` | ||||
|               <ha-chip .hasIcon=${chip.icon !== undefined}> | ||||
|                 ${chip.icon | ||||
|                   ? html`<ha-svg-icon slot="icon" .path=${chip.icon}> | ||||
|                     </ha-svg-icon>` | ||||
|                   : ""} | ||||
|                 ${chip.content} | ||||
|               </ha-chip> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       </ha-card> | ||||
|       <ha-card header="ha-chip-set demo"> | ||||
|         <div class="card-content"> | ||||
|           <ha-chip-set> | ||||
|             ${chips.map( | ||||
|               (chip) => html` | ||||
|                 <ha-chip .hasIcon=${chip.icon !== undefined}> | ||||
|                   ${chip.icon | ||||
|                     ? html`<ha-svg-icon slot="icon" .path=${chip.icon}> | ||||
|                       </ha-svg-icon>` | ||||
|                     : ""} | ||||
|                   ${chip.content} | ||||
|                 </ha-chip> | ||||
|               ` | ||||
|             )} | ||||
|           </ha-chip-set> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       ha-chip { | ||||
|         margin-bottom: 4px; | ||||
|       } | ||||
|       .card-content { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-chips": DemoHaChips; | ||||
|   } | ||||
| } | ||||
| @@ -1,23 +1,25 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { LitElement, TemplateResult, css, html } from "lit"; | ||||
| import "@material/mwc-button"; | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { computeInitialHaFormData } from "../../../src/components/ha-form/compute-initial-ha-form-data"; | ||||
| import type { HaFormSchema } from "../../../src/components/ha-form/types"; | ||||
| import "../../../src/components/ha-form/ha-form"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import type { HaFormSchema } from "../../../src/components/ha-form/ha-form"; | ||||
| import "../components/demo-black-white-row"; | ||||
|  | ||||
| const SCHEMAS: { | ||||
|   title: string; | ||||
|   translations?: Record<string, string>; | ||||
|   error?: Record<string, string>; | ||||
|   schema: HaFormSchema[]; | ||||
|   data?: Record<string, any>; | ||||
| }[] = [ | ||||
|   { | ||||
|     title: "Authentication", | ||||
|     translations: { | ||||
|       username: "Username", | ||||
|       password: "Password", | ||||
|       invalid_login: "Invalid login", | ||||
|       invalid_login: "Invalid username or password", | ||||
|     }, | ||||
|     error: { | ||||
|       base: "invalid_login", | ||||
| @@ -57,6 +59,11 @@ const SCHEMAS: { | ||||
|         optional: true, | ||||
|         default: 10, | ||||
|       }, | ||||
|       { | ||||
|         type: "float", | ||||
|         name: "float", | ||||
|         required: true, | ||||
|       }, | ||||
|       { | ||||
|         type: "string", | ||||
|         name: "string", | ||||
| @@ -83,6 +90,80 @@ const SCHEMAS: { | ||||
|         optional: true, | ||||
|         default: ["default"], | ||||
|       }, | ||||
|       { | ||||
|         type: "positive_time_period_dict", | ||||
|         name: "time", | ||||
|         required: true, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     title: "Numbers", | ||||
|     schema: [ | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int", | ||||
|         required: true, | ||||
|       }, | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int with default", | ||||
|         optional: true, | ||||
|         default: 10, | ||||
|       }, | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int range required", | ||||
|         required: true, | ||||
|         default: 5, | ||||
|         valueMin: 0, | ||||
|         valueMax: 10, | ||||
|       }, | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int range optional", | ||||
|         optional: true, | ||||
|         valueMin: 0, | ||||
|         valueMax: 10, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     title: "select", | ||||
|     schema: [ | ||||
|       { | ||||
|         type: "select", | ||||
|         options: [ | ||||
|           ["default", "Default"], | ||||
|           ["other", "Other"], | ||||
|         ], | ||||
|         name: "select", | ||||
|         required: true, | ||||
|         default: "default", | ||||
|       }, | ||||
|       { | ||||
|         type: "select", | ||||
|         options: [ | ||||
|           ["default", "Default"], | ||||
|           ["other", "Other"], | ||||
|         ], | ||||
|         name: "select optional", | ||||
|         optional: true, | ||||
|       }, | ||||
|       { | ||||
|         type: "select", | ||||
|         options: [ | ||||
|           ["default", "Default"], | ||||
|           ["other", "Other"], | ||||
|           ["uno", "mas"], | ||||
|           ["one", "more"], | ||||
|           ["and", "another_one"], | ||||
|           ["option", "1000"], | ||||
|         ], | ||||
|         name: "select many otions", | ||||
|         optional: true, | ||||
|         default: "default", | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
| @@ -95,7 +176,7 @@ const SCHEMAS: { | ||||
|           other: "Other", | ||||
|         }, | ||||
|         name: "multi", | ||||
|         optional: true, | ||||
|         required: true, | ||||
|         default: ["default"], | ||||
|       }, | ||||
|       { | ||||
| @@ -108,101 +189,114 @@ const SCHEMAS: { | ||||
|           and: "another_one", | ||||
|           option: "1000", | ||||
|         }, | ||||
|         name: "multi", | ||||
|         name: "multi many otions", | ||||
|         optional: true, | ||||
|         default: ["default"], | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     title: "Field specific error", | ||||
|     data: { | ||||
|       new_password: "hello", | ||||
|       new_password_2: "bye", | ||||
|     }, | ||||
|     translations: { | ||||
|       new_password: "New Password", | ||||
|       new_password_2: "Re-type Password", | ||||
|       not_match: "The passwords do not match", | ||||
|     }, | ||||
|     error: { | ||||
|       new_password_2: "not_match", | ||||
|     }, | ||||
|     schema: [ | ||||
|       { | ||||
|         type: "string", | ||||
|         name: "new_password", | ||||
|         required: true, | ||||
|       }, | ||||
|       { | ||||
|         type: "string", | ||||
|         name: "new_password_2", | ||||
|         required: true, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     title: "OctoPrint", | ||||
|     translations: { | ||||
|       username: "Username", | ||||
|       host: "Host", | ||||
|       port: "Port Number", | ||||
|       path: "Application Path", | ||||
|       ssl: "Use SSL", | ||||
|     }, | ||||
|     schema: [ | ||||
|       { type: "string", name: "username", required: true, default: "" }, | ||||
|       { type: "string", name: "host", required: true, default: "" }, | ||||
|       { | ||||
|         type: "integer", | ||||
|         valueMin: 1, | ||||
|         valueMax: 65535, | ||||
|         name: "port", | ||||
|         optional: true, | ||||
|         default: 80, | ||||
|       }, | ||||
|       { type: "string", name: "path", optional: true, default: "/" }, | ||||
|       { type: "boolean", name: "ssl", optional: true, default: false }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-form") | ||||
| class DemoHaForm extends LitElement { | ||||
|   private lightModeData: any = []; | ||||
|   private data = SCHEMAS.map( | ||||
|     ({ schema, data }) => data || computeInitialHaFormData(schema) | ||||
|   ); | ||||
|  | ||||
|   private darkModeData: any = []; | ||||
|   private disabled = SCHEMAS.map(() => false); | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${SCHEMAS.map((info, idx) => { | ||||
|         const translations = info.translations || {}; | ||||
|         const computeLabel = (schema) => | ||||
|           translations[schema.name] || schema.name; | ||||
|         const computeError = (error) => translations[error] || error; | ||||
|  | ||||
|         return [ | ||||
|           [this.lightModeData, "light"], | ||||
|           [this.darkModeData, "dark"], | ||||
|         ].map( | ||||
|           ([data, type]) => html` | ||||
|             <div class="row" data-type=${type}> | ||||
|               <ha-card .header=${info.title}> | ||||
|                 <div class="card-content"> | ||||
|                   <ha-form | ||||
|                     .data=${data[idx]} | ||||
|                     .schema=${info.schema} | ||||
|                     .error=${info.error} | ||||
|                     .computeError=${computeError} | ||||
|                     .computeLabel=${computeLabel} | ||||
|                     @value-changed=${(e) => { | ||||
|                       data[idx] = e.detail.value; | ||||
|                       this.requestUpdate(); | ||||
|                     }} | ||||
|                   ></ha-form> | ||||
|                 </div> | ||||
|               </ha-card> | ||||
|               <pre>${JSON.stringify(data[idx], undefined, 2)}</pre> | ||||
|             </div> | ||||
|           ` | ||||
|         ); | ||||
|         return html` | ||||
|           <demo-black-white-row | ||||
|             .title=${info.title} | ||||
|             .value=${this.data[idx]} | ||||
|             .disabled=${this.disabled[idx]} | ||||
|             @submitted=${() => { | ||||
|               this.disabled[idx] = true; | ||||
|               this.requestUpdate(); | ||||
|               setTimeout(() => { | ||||
|                 this.disabled[idx] = false; | ||||
|                 this.requestUpdate(); | ||||
|               }, 2000); | ||||
|             }} | ||||
|           > | ||||
|             ${["light", "dark"].map( | ||||
|               (slot) => html` | ||||
|                 <ha-form | ||||
|                   slot=${slot} | ||||
|                   .data=${this.data[idx]} | ||||
|                   .schema=${info.schema} | ||||
|                   .error=${info.error} | ||||
|                   .disabled=${this.disabled[idx]} | ||||
|                   .computeError=${(error) => translations[error] || error} | ||||
|                   .computeLabel=${(schema) => | ||||
|                     translations[schema.name] || schema.name} | ||||
|                   @value-changed=${(e) => { | ||||
|                     this.data[idx] = e.detail.value; | ||||
|                     this.requestUpdate(); | ||||
|                   }} | ||||
|                 ></ha-form> | ||||
|               ` | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         `; | ||||
|       })} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     this.shadowRoot!.querySelectorAll("[data-type=dark]").forEach((el) => { | ||||
|       applyThemesOnElement( | ||||
|         el, | ||||
|         { | ||||
|           default_theme: "default", | ||||
|           default_dark_theme: "default", | ||||
|           themes: {}, | ||||
|           darkMode: false, | ||||
|         }, | ||||
|         "default", | ||||
|         { dark: true } | ||||
|       ); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .row { | ||||
|       margin: 0 auto; | ||||
|       max-width: 800px; | ||||
|       display: flex; | ||||
|       padding: 50px; | ||||
|       background-color: var(--primary-background-color); | ||||
|     } | ||||
|     ha-card { | ||||
|       width: 100%; | ||||
|       max-width: 384px; | ||||
|     } | ||||
|     pre { | ||||
|       width: 400px; | ||||
|       margin: 0 16px; | ||||
|       overflow: auto; | ||||
|       color: var(--primary-text-color); | ||||
|     } | ||||
|     @media only screen and (max-width: 800px) { | ||||
|       .row { | ||||
|         flex-direction: column; | ||||
|       } | ||||
|       pre { | ||||
|         margin: 16px 0; | ||||
|       } | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
							
								
								
									
										122
									
								
								gallery/src/demos/demo-ha-label-badge.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								gallery/src/demos/demo-ha-label-badge.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,122 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-label-badge"; | ||||
| import "../../../src/components/ha-card"; | ||||
|  | ||||
| const colors = ["#03a9f4", "#ffa600", "#43a047"]; | ||||
|  | ||||
| const badges: { | ||||
|   label?: string; | ||||
|   description?: string; | ||||
|   image?: string; | ||||
| }[] = [ | ||||
|   { | ||||
|     label: "label", | ||||
|   }, | ||||
|   { | ||||
|     label: "label", | ||||
|     description: "Description", | ||||
|   }, | ||||
|   { | ||||
|     description: "Description", | ||||
|   }, | ||||
|   { | ||||
|     label: "label", | ||||
|     description: "Description", | ||||
|     image: "/images/living_room.png", | ||||
|   }, | ||||
|   { | ||||
|     description: "Description", | ||||
|     image: "/images/living_room.png", | ||||
|   }, | ||||
|   { | ||||
|     label: "label", | ||||
|     image: "/images/living_room.png", | ||||
|   }, | ||||
|   { | ||||
|     image: "/images/living_room.png", | ||||
|   }, | ||||
|   { | ||||
|     label: "big label", | ||||
|   }, | ||||
|   { | ||||
|     label: "big label", | ||||
|     description: "Description", | ||||
|   }, | ||||
|   { | ||||
|     label: "big label", | ||||
|     description: "Description", | ||||
|     image: "/images/living_room.png", | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-label-badge") | ||||
| export class DemoHaLabelBadge extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card> | ||||
|         <div class="card-content"> | ||||
|           ${badges.map( | ||||
|             (badge) => html` | ||||
|               <ha-label-badge | ||||
|                 style="--ha-label-badge-color: ${colors[ | ||||
|                   Math.floor(Math.random() * colors.length) | ||||
|                 ]}" | ||||
|                 .label=${badge.label} | ||||
|                 .description=${badge.description} | ||||
|                 .image=${badge.image} | ||||
|               > | ||||
|               </ha-label-badge> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       </ha-card> | ||||
|       <ha-card> | ||||
|         <div class="card-content"> | ||||
|           ${badges.map( | ||||
|             (badge) => html` | ||||
|               <div class="badge"> | ||||
|                 <ha-label-badge | ||||
|                   style="--ha-label-badge-color: ${colors[ | ||||
|                     Math.floor(Math.random() * colors.length) | ||||
|                   ]}" | ||||
|                   .label=${badge.label} | ||||
|                   .description=${badge.description} | ||||
|                   .image=${badge.image} | ||||
|                 > | ||||
|                 </ha-label-badge> | ||||
|                 <pre>${JSON.stringify(badge, null, 2)}</pre> | ||||
|               </div> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       pre { | ||||
|         margin-left: 16px; | ||||
|         background-color: var(--markdown-code-background-color); | ||||
|         padding: 8px; | ||||
|       } | ||||
|       .badge { | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         margin-bottom: 16px; | ||||
|         align-items: center; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-label-badge": DemoHaLabelBadge; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										131
									
								
								gallery/src/demos/demo-ha-selector.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										131
									
								
								gallery/src/demos/demo-ha-selector.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,131 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import "@material/mwc-button"; | ||||
| import { LitElement, TemplateResult, css, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import "../../../src/components/ha-selector/ha-selector"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { BlueprintInput } from "../../../src/data/blueprint"; | ||||
| import { mockEntityRegistry } from "../../../demo/src/stubs/entity_registry"; | ||||
| import { mockDeviceRegistry } from "../../../demo/src/stubs/device_registry"; | ||||
| import { mockAreaRegistry } from "../../../demo/src/stubs/area_registry"; | ||||
| import { mockHassioSupervisor } from "../../../demo/src/stubs/hassio_supervisor"; | ||||
|  | ||||
| const SCHEMAS: { | ||||
|   name: string; | ||||
|   input: Record<string, BlueprintInput | null>; | ||||
| }[] = [ | ||||
|   { | ||||
|     name: "One of each", | ||||
|     input: { | ||||
|       entity: { name: "Entity", selector: { entity: {} } }, | ||||
|       device: { name: "Device", selector: { device: {} } }, | ||||
|       addon: { name: "Addon", selector: { addon: {} } }, | ||||
|       area: { name: "Area", selector: { area: {} } }, | ||||
|       target: { name: "Target", selector: { target: {} } }, | ||||
|       number_box: { | ||||
|         name: "Number Box", | ||||
|         selector: { | ||||
|           number: { | ||||
|             min: 0, | ||||
|             max: 10, | ||||
|             mode: "box", | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       number_slider: { | ||||
|         name: "Number Slider", | ||||
|         selector: { | ||||
|           number: { | ||||
|             min: 0, | ||||
|             max: 10, | ||||
|             mode: "slider", | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       boolean: { name: "Boolean", selector: { boolean: {} } }, | ||||
|       time: { name: "Time", selector: { time: {} } }, | ||||
|       action: { name: "Action", selector: { action: {} } }, | ||||
|       text: { name: "Text", selector: { text: { multiline: false } } }, | ||||
|       text_multiline: { | ||||
|         name: "Text multiline", | ||||
|         selector: { text: { multiline: true } }, | ||||
|       }, | ||||
|       object: { name: "Object", selector: { object: {} } }, | ||||
|       select: { | ||||
|         name: "Select", | ||||
|         selector: { select: { options: ["Option 1", "Option 2"] } }, | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-selector") | ||||
| class DemoHaSelector extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
|  | ||||
|   private data = SCHEMAS.map(() => ({})); | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       ${SCHEMAS.map((info, idx) => { | ||||
|         const data = this.data[idx]; | ||||
|         const valueChanged = (ev) => { | ||||
|           this.data[idx] = { | ||||
|             ...data, | ||||
|             [ev.target.key]: ev.detail.value, | ||||
|           }; | ||||
|           this.requestUpdate(); | ||||
|         }; | ||||
|         return html` | ||||
|           <demo-black-white-row .title=${info.name} .value=${this.data[idx]}> | ||||
|             ${["light", "dark"].map((slot) => | ||||
|               Object.entries(info.input).map( | ||||
|                 ([key, value]) => | ||||
|                   html` | ||||
|                     <ha-settings-row narrow slot=${slot}> | ||||
|                       <span slot="heading">${value?.name || key}</span> | ||||
|                       <span slot="description">${value?.description}</span> | ||||
|                       <ha-selector | ||||
|                         .hass=${this.hass} | ||||
|                         .selector=${value!.selector} | ||||
|                         .key=${key} | ||||
|                         .value=${data[key] ?? value!.default} | ||||
|                         @value-changed=${valueChanged} | ||||
|                       ></ha-selector> | ||||
|                     </ha-settings-row> | ||||
|                   ` | ||||
|               ) | ||||
|             )} | ||||
|           </demo-black-white-row> | ||||
|         `; | ||||
|       })} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     paper-input, | ||||
|     ha-selector { | ||||
|       width: 60; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-selector": DemoHaSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										156
									
								
								gallery/src/demos/demo-hui-area-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								gallery/src/demos/demo-hui-area-card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
|     friendly_name: "Bed Light", | ||||
|   }), | ||||
|   getEntity("switch", "bed_ac", "on", { | ||||
|     friendly_name: "Ecobee", | ||||
|   }), | ||||
|   getEntity("sensor", "bed_temp", "72", { | ||||
|     friendly_name: "Bedroom Temp", | ||||
|     device_class: "temperature", | ||||
|     unit_of_measurement: "°F", | ||||
|   }), | ||||
|   getEntity("light", "living_room_light", "off", { | ||||
|     friendly_name: "Living Room Light", | ||||
|   }), | ||||
|   getEntity("fan", "living_room", "on", { | ||||
|     friendly_name: "Living Room Fan", | ||||
|   }), | ||||
|   getEntity("sensor", "office_humidity", "73", { | ||||
|     friendly_name: "Office Humidity", | ||||
|     device_class: "humidity", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("light", "office", "on", { | ||||
|     friendly_name: "Office Light", | ||||
|   }), | ||||
|   getEntity("fan", "kitchen", "on", { | ||||
|     friendly_name: "Second Office Fan", | ||||
|   }), | ||||
|   getEntity("binary_sensor", "kitchen_door", "on", { | ||||
|     friendly_name: "Office Door", | ||||
|     device_class: "door", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| // TODO: Update image here | ||||
| const CONFIGS = [ | ||||
|   { | ||||
|     heading: "Bedroom", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: bedroom | ||||
|   image: "/images/bed.png" | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Living Room", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: living_room | ||||
|   image: "/images/living_room.png" | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Office", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: office | ||||
|   image: "/images/office.jpg" | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Kitchen", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: kitchen | ||||
|   image: "/images/kitchen.png" | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-hui-area-card") | ||||
| class DemoArea extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("lovelace", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     hass.mockWS("config/area_registry/list", () => [ | ||||
|       { | ||||
|         name: "Bedroom", | ||||
|         area_id: "bedroom", | ||||
|       }, | ||||
|       { | ||||
|         name: "Living Room", | ||||
|         area_id: "living_room", | ||||
|       }, | ||||
|       { | ||||
|         name: "Office", | ||||
|         area_id: "office", | ||||
|       }, | ||||
|       { | ||||
|         name: "Second Office", | ||||
|         area_id: "kitchen", | ||||
|       }, | ||||
|     ]); | ||||
|     hass.mockWS("config/device_registry/list", () => []); | ||||
|     hass.mockWS("config/entity_registry/list", () => [ | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "light.bed_light", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "switch.bed_ac", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "sensor.bed_temp", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "living_room", | ||||
|         entity_id: "light.living_room_light", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "living_room", | ||||
|         entity_id: "fan.living_room", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "office", | ||||
|         entity_id: "light.office", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "office", | ||||
|         entity_id: "sensor.office_humidity", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "kitchen", | ||||
|         entity_id: "fan.kitchen", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "kitchen", | ||||
|         entity_id: "binary_sensor.kitchen_door", | ||||
|       }, | ||||
|     ]); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-area-card": DemoArea; | ||||
|   } | ||||
| } | ||||
| @@ -187,6 +187,7 @@ const createEntityRegistryEntries = ( | ||||
|     device_id: "mock-device-id", | ||||
|     area_id: null, | ||||
|     disabled_by: null, | ||||
|     entity_category: null, | ||||
|     entity_id: "binary_sensor.updater", | ||||
|     name: null, | ||||
|     icon: null, | ||||
| @@ -211,6 +212,7 @@ const createDeviceRegistryEntries = ( | ||||
|     area_id: null, | ||||
|     name_by_user: null, | ||||
|     disabled_by: null, | ||||
|     configuration_url: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -65,10 +65,11 @@ class HaGallery extends PolymerElement { | ||||
|         <app-header slot="header" fixed> | ||||
|           <app-toolbar> | ||||
|             <ha-icon-button | ||||
|               icon="hass:arrow-left" | ||||
|               on-click="_backTapped" | ||||
|               class$="[[_computeHeaderButtonClass(_demo)]]" | ||||
|             ></ha-icon-button> | ||||
|             > | ||||
|               <ha-icon icon="hass:arrow-left"></ha-icon> | ||||
|             </ha-icon-button> | ||||
|             <div main-title> | ||||
|               [[_withDefault(_demo, "Home Assistant Gallery")]] | ||||
|             </div> | ||||
| @@ -175,11 +176,6 @@ class HaGallery extends PolymerElement { | ||||
|     this.addEventListener("alert-dismissed-clicked", () => | ||||
|       this.$.notifications.showDialog({ message: "Alert dismissed clicked" }) | ||||
|     ); | ||||
|  | ||||
|     this.addEventListener("alert-action-clicked", () => | ||||
|       this.$.notifications.showDialog({ message: "Alert action clicked" }) | ||||
|     ); | ||||
|  | ||||
|     this.addEventListener("hass-more-info", (ev) => { | ||||
|       if (ev.detail.entityId) { | ||||
|         this.$.notifications.showDialog({ | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import { property } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { navigate } from "../../../src/common/navigate"; | ||||
| import { caseInsensitiveStringCompare } from "../../../src/common/string/compare"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { | ||||
|   HassioAddonInfo, | ||||
| @@ -32,7 +33,7 @@ class HassioAddonRepositoryEl extends LitElement { | ||||
|         return filterAndSort(addons, filter); | ||||
|       } | ||||
|       return addons.sort((a, b) => | ||||
|         a.name.toUpperCase() < b.name.toUpperCase() ? -1 : 1 | ||||
|         caseInsensitiveStringCompare(a.name, b.name) | ||||
|       ); | ||||
|     } | ||||
|   ); | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "@material/mwc-icon-button/mwc-icon-button"; | ||||
| import { ActionDetail } from "@material/mwc-list/mwc-list-foundation"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { mdiDotsVertical } from "@mdi/js"; | ||||
| @@ -18,7 +17,7 @@ import { navigate } from "../../../src/common/navigate"; | ||||
| import "../../../src/common/search/search-input"; | ||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import { | ||||
|   HassioAddonInfo, | ||||
|   HassioAddonRepository, | ||||
| @@ -26,11 +25,10 @@ import { | ||||
| } from "../../../src/data/hassio/addon"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import "../../../src/layouts/hass-loading-screen"; | ||||
| import "../../../src/layouts/hass-tabs-subpage"; | ||||
| import "../../../src/layouts/hass-subpage"; | ||||
| import { HomeAssistant, Route } from "../../../src/types"; | ||||
| import { showRegistriesDialog } from "../dialogs/registries/show-dialog-registries"; | ||||
| import { showRepositoriesDialog } from "../dialogs/repositories/show-dialog-repositories"; | ||||
| import { supervisorTabs } from "../hassio-tabs"; | ||||
| import "./hassio-addon-repository"; | ||||
|  | ||||
| const sortRepos = (a: HassioAddonRepository, b: HassioAddonRepository) => { | ||||
| @@ -77,24 +75,22 @@ class HassioAddonStore extends LitElement { | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <hass-tabs-subpage | ||||
|       <hass-subpage | ||||
|         .hass=${this.hass} | ||||
|         .localizeFunc=${this.supervisor.localize} | ||||
|         .narrow=${this.narrow} | ||||
|         .route=${this.route} | ||||
|         .tabs=${supervisorTabs} | ||||
|         main-page | ||||
|         supervisor | ||||
|         .header=${this.supervisor.localize("panel.store")} | ||||
|       > | ||||
|         <span slot="header"> ${this.supervisor.localize("panel.store")} </span> | ||||
|         <ha-button-menu | ||||
|           corner="BOTTOM_START" | ||||
|           slot="toolbar-icon" | ||||
|           @action=${this._handleAction} | ||||
|         > | ||||
|           <mwc-icon-button slot="trigger" alt="menu"> | ||||
|             <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon> | ||||
|           </mwc-icon-button> | ||||
|           <ha-icon-button | ||||
|             .label=${this.supervisor.localize("common.menu")} | ||||
|             .path=${mdiDotsVertical} | ||||
|             slot="trigger" | ||||
|           ></ha-icon-button> | ||||
|           <mwc-list-item> | ||||
|             ${this.supervisor.localize("store.repositories")} | ||||
|           </mwc-list-item> | ||||
| @@ -113,6 +109,7 @@ class HassioAddonStore extends LitElement { | ||||
|           : html` | ||||
|               <div class="search"> | ||||
|                 <search-input | ||||
|                   .hass=${this.hass} | ||||
|                   no-label-float | ||||
|                   no-underline | ||||
|                   .filter=${this._filter} | ||||
| @@ -131,7 +128,7 @@ class HassioAddonStore extends LitElement { | ||||
|               </div> | ||||
|             ` | ||||
|           : ""} | ||||
|       </hass-tabs-subpage> | ||||
|       </hass-subpage> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -15,12 +15,13 @@ import { customElement, property, query, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button-menu"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/ha-form"; | ||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-switch"; | ||||
| import "../../../../src/components/ha-yaml-editor"; | ||||
| import type { HaYamlEditor } from "../../../../src/components/ha-yaml-editor"; | ||||
| @@ -77,6 +78,18 @@ class HassioAddonConfig extends LitElement { | ||||
|     this.addon.translations.en?.configuration?.[entry.name].name || | ||||
|     entry.name; | ||||
|  | ||||
|   private _schema = memoizeOne((schema: HaFormSchema[]): HaFormSchema[] => | ||||
|     // @ts-expect-error supervisor does not implement [string, string] for select.options[] | ||||
|     schema.map((entry) => | ||||
|       entry.type === "select" | ||||
|         ? { | ||||
|             ...entry, | ||||
|             options: entry.options.map((option) => [option, option]), | ||||
|           } | ||||
|         : entry | ||||
|     ) | ||||
|   ); | ||||
|  | ||||
|   private _filteredShchema = memoizeOne( | ||||
|     (options: Record<string, unknown>, schema: HaFormSchema[]) => | ||||
|       schema.filter((entry) => entry.name in options || entry.required) | ||||
| @@ -100,9 +113,11 @@ class HassioAddonConfig extends LitElement { | ||||
|           </h2> | ||||
|           <div class="card-menu"> | ||||
|             <ha-button-menu corner="BOTTOM_START" @action=${this._handleAction}> | ||||
|               <mwc-icon-button slot="trigger"> | ||||
|                 <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon> | ||||
|               </mwc-icon-button> | ||||
|               <ha-icon-button | ||||
|                 .label=${this.hass.localize("common.menu")} | ||||
|                 .path=${mdiDotsVertical} | ||||
|                 slot="trigger" | ||||
|               ></ha-icon-button> | ||||
|               <mwc-list-item .disabled=${!this._canShowSchema}> | ||||
|                 ${this._yamlMode | ||||
|                   ? this.supervisor.localize( | ||||
| @@ -125,12 +140,14 @@ class HassioAddonConfig extends LitElement { | ||||
|                 .data=${this._options!} | ||||
|                 @value-changed=${this._configChanged} | ||||
|                 .computeLabel=${this.computeLabel} | ||||
|                 .schema=${this._showOptional | ||||
|                   ? this.addon.schema! | ||||
|                   : this._filteredShchema( | ||||
|                       this.addon.options, | ||||
|                       this.addon.schema! | ||||
|                     )} | ||||
|                 .schema=${this._schema( | ||||
|                   this._showOptional | ||||
|                     ? this.addon.schema! | ||||
|                     : this._filteredShchema( | ||||
|                         this.addon.options, | ||||
|                         this.addon.schema! | ||||
|                       ) | ||||
|                 )} | ||||
|               ></ha-form>` | ||||
|             : html` <ha-yaml-editor | ||||
|                 @value-changed=${this._configChanged} | ||||
|   | ||||
| @@ -108,7 +108,6 @@ class HassioAddonDashboard extends LitElement { | ||||
|         .hass=${this.hass} | ||||
|         .localizeFunc=${this.supervisor.localize} | ||||
|         .narrow=${this.narrow} | ||||
|         .backPath=${this.addon.version ? "/hassio/dashboard" : "/hassio/store"} | ||||
|         .route=${route} | ||||
|         .tabs=${addonTabs} | ||||
|         supervisor | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import "../../../../src/components/ha-circular-progress"; | ||||
| import { HassioAddonDetails } from "../../../../src/data/hassio/addon"; | ||||
| import { Supervisor } from "../../../../src/data/supervisor/supervisor"; | ||||
| import { haStyle } from "../../../../src/resources/styles"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { HomeAssistant, Route } from "../../../../src/types"; | ||||
| import { hassioStyle } from "../../resources/hassio-style"; | ||||
| import "./hassio-addon-info"; | ||||
|  | ||||
| @@ -12,6 +12,8 @@ import "./hassio-addon-info"; | ||||
| class HassioAddonInfoDashboard extends LitElement { | ||||
|   @property({ type: Boolean }) public narrow!: boolean; | ||||
|  | ||||
|   @property({ attribute: false }) public route!: Route; | ||||
|  | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public supervisor!: Supervisor; | ||||
| @@ -27,6 +29,7 @@ class HassioAddonInfoDashboard extends LitElement { | ||||
|       <div class="content"> | ||||
|         <hassio-addon-info | ||||
|           .narrow=${this.narrow} | ||||
|           .route=${this.route} | ||||
|           .hass=${this.hass} | ||||
|           .supervisor=${this.supervisor} | ||||
|           .addon=${this.addon} | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { | ||||
|   mdiArrowUpBoldCircle, | ||||
|   mdiCheckCircle, | ||||
|   mdiChip, | ||||
|   mdiCircle, | ||||
| @@ -11,6 +10,12 @@ import { | ||||
|   mdiHomeAssistant, | ||||
|   mdiKey, | ||||
|   mdiNetwork, | ||||
|   mdiNumeric1, | ||||
|   mdiNumeric2, | ||||
|   mdiNumeric3, | ||||
|   mdiNumeric4, | ||||
|   mdiNumeric5, | ||||
|   mdiNumeric6, | ||||
|   mdiPound, | ||||
|   mdiShield, | ||||
| } from "@mdi/js"; | ||||
| @@ -25,7 +30,7 @@ import "../../../../src/components/buttons/ha-call-api-button"; | ||||
| import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-label-badge"; | ||||
| import "../../../../src/components/ha-chip"; | ||||
| import "../../../../src/components/ha-markdown"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| @@ -43,7 +48,6 @@ import { | ||||
|   startHassioAddon, | ||||
|   stopHassioAddon, | ||||
|   uninstallHassioAddon, | ||||
|   updateHassioAddon, | ||||
|   validateHassioAddonOption, | ||||
| } from "../../../../src/data/hassio/addon"; | ||||
| import { | ||||
| @@ -58,14 +62,14 @@ import { | ||||
|   showConfirmationDialog, | ||||
| } from "../../../../src/dialogs/generic/show-dialog-box"; | ||||
| import { haStyle } from "../../../../src/resources/styles"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { HomeAssistant, Route } from "../../../../src/types"; | ||||
| import { bytesToString } from "../../../../src/util/bytes-to-string"; | ||||
| import "../../components/hassio-card-content"; | ||||
| import "../../components/supervisor-metric"; | ||||
| import { showHassioMarkdownDialog } from "../../dialogs/markdown/show-dialog-hassio-markdown"; | ||||
| import { showDialogSupervisorUpdate } from "../../dialogs/update/show-dialog-update"; | ||||
| import { hassioStyle } from "../../resources/hassio-style"; | ||||
| import { addonArchIsSupported } from "../../util/addon"; | ||||
| import "../../update-available/update-available-card"; | ||||
| import { addonArchIsSupported, extractChangelog } from "../../util/addon"; | ||||
|  | ||||
| const STAGE_ICON = { | ||||
|   stable: mdiCheckCircle, | ||||
| @@ -73,10 +77,21 @@ const STAGE_ICON = { | ||||
|   deprecated: mdiExclamationThick, | ||||
| }; | ||||
|  | ||||
| const RATING_ICON = { | ||||
|   1: mdiNumeric1, | ||||
|   2: mdiNumeric2, | ||||
|   3: mdiNumeric3, | ||||
|   4: mdiNumeric4, | ||||
|   5: mdiNumeric5, | ||||
|   6: mdiNumeric6, | ||||
| }; | ||||
|  | ||||
| @customElement("hassio-addon-info") | ||||
| class HassioAddonInfo extends LitElement { | ||||
|   @property({ type: Boolean }) public narrow!: boolean; | ||||
|  | ||||
|   @property({ attribute: false }) public route!: Route; | ||||
|  | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public addon!: HassioAddonDetails; | ||||
| @@ -113,91 +128,35 @@ class HassioAddonInfo extends LitElement { | ||||
|     return html` | ||||
|       ${this.addon.update_available | ||||
|         ? html` | ||||
|             <ha-card | ||||
|               .header="${this.supervisor.localize( | ||||
|                 "common.update_available", | ||||
|                 "count", | ||||
|                 1 | ||||
|               )}🎉" | ||||
|             > | ||||
|               <div class="card-content"> | ||||
|                 <hassio-card-content | ||||
|                   .hass=${this.hass} | ||||
|                   .title=${this.supervisor.localize( | ||||
|                     "addon.dashboard.new_update_available", | ||||
|                     "name", | ||||
|                     this.addon.name, | ||||
|                     "version", | ||||
|                     this.addon.version_latest | ||||
|                   )} | ||||
|                   .description=${this.supervisor.localize( | ||||
|                     "common.running_version", | ||||
|                     "version", | ||||
|                     this.addon.version | ||||
|                   )} | ||||
|                   icon=${mdiArrowUpBoldCircle} | ||||
|                   iconClass="update" | ||||
|                 ></hassio-card-content> | ||||
|                 ${!this.addon.available && addonStoreInfo | ||||
|                   ? !addonArchIsSupported( | ||||
|                       this.supervisor.info.supported_arch, | ||||
|                       this.addon.arch | ||||
|                     ) | ||||
|                     ? html` | ||||
|                         <ha-alert alert-type="warning"> | ||||
|                           ${this.supervisor.localize( | ||||
|                             "addon.dashboard.not_available_arch" | ||||
|                           )} | ||||
|                         </ha-alert> | ||||
|                       ` | ||||
|                     : html` | ||||
|                         <ha-alert alert-type="warning"> | ||||
|                           ${this.supervisor.localize( | ||||
|                             "addon.dashboard.not_available_arch", | ||||
|                             "core_version_installed", | ||||
|                             this.supervisor.core.version, | ||||
|                             "core_version_needed", | ||||
|                             addonStoreInfo.homeassistant | ||||
|                           )} | ||||
|                         </ha-alert> | ||||
|                       ` | ||||
|                   : ""} | ||||
|               </div> | ||||
|               <div class="card-actions"> | ||||
|                 ${this.addon.changelog | ||||
|                   ? html` | ||||
|                       <mwc-button @click=${this._openChangelog}> | ||||
|                         ${this.supervisor.localize("addon.dashboard.changelog")} | ||||
|                       </mwc-button> | ||||
|                     ` | ||||
|                   : html`<span></span>`} | ||||
|                 <mwc-button @click=${this._updateClicked}> | ||||
|                   ${this.supervisor.localize("common.update")} | ||||
|                 </mwc-button> | ||||
|               </div> | ||||
|             </ha-card> | ||||
|             <update-available-card | ||||
|               .hass=${this.hass} | ||||
|               .narrow=${this.narrow} | ||||
|               .supervisor=${this.supervisor} | ||||
|               .addonSlug=${this.addon.slug} | ||||
|             ></update-available-card> | ||||
|           ` | ||||
|         : ""} | ||||
|       ${!this.addon.protected | ||||
|         ? html` | ||||
|         <ha-card class="warning"> | ||||
|           <h1 class="card-header">${this.supervisor.localize( | ||||
|             "addon.dashboard.protection_mode.title" | ||||
|           )} | ||||
|           </h1> | ||||
|           <div class="card-content"> | ||||
|           ${this.supervisor.localize("addon.dashboard.protection_mode.content")} | ||||
|           </div> | ||||
|           <div class="card-actions protection-enable"> | ||||
|               <mwc-button @click=${this._protectionToggled}> | ||||
|               ${this.supervisor.localize( | ||||
|                 "addon.dashboard.protection_mode.enable" | ||||
|             <ha-alert | ||||
|               alert-type="error" | ||||
|               .title=${this.supervisor.localize( | ||||
|                 "addon.dashboard.protection_mode.title" | ||||
|               )} | ||||
|             > | ||||
|               ${this.supervisor.localize( | ||||
|                 "addon.dashboard.protection_mode.content" | ||||
|               )} | ||||
|               <mwc-button | ||||
|                 slot="action" | ||||
|                 .label=${this.supervisor.localize( | ||||
|                   "addon.dashboard.protection_mode.enable" | ||||
|                 )} | ||||
|                 @click=${this._protectionToggled} | ||||
|               > | ||||
|               </mwc-button> | ||||
|             </div> | ||||
|           </div> | ||||
|         </ha-card> | ||||
|       ` | ||||
|             </ha-alert> | ||||
|           ` | ||||
|         : ""} | ||||
|  | ||||
|       <ha-card> | ||||
| @@ -249,6 +208,163 @@ class HassioAddonInfo extends LitElement { | ||||
|                 >`} | ||||
|           </div> | ||||
|  | ||||
|           <div class="capabilities"> | ||||
|             ${this.addon.stage !== "stable" | ||||
|               ? html` <ha-chip | ||||
|                   hasIcon | ||||
|                   class=${classMap({ | ||||
|                     yellow: this.addon.stage === "experimental", | ||||
|                     red: this.addon.stage === "deprecated", | ||||
|                   })} | ||||
|                   @click=${this._showMoreInfo} | ||||
|                   id="stage" | ||||
|                 > | ||||
|                   <ha-svg-icon | ||||
|                     slot="icon" | ||||
|                     .path=${STAGE_ICON[this.addon.stage]} | ||||
|                   > | ||||
|                   </ha-svg-icon> | ||||
|                   ${this.supervisor.localize( | ||||
|                     `addon.dashboard.capability.stages.${this.addon.stage}` | ||||
|                   )} | ||||
|                 </ha-chip>` | ||||
|               : ""} | ||||
|  | ||||
|             <ha-chip | ||||
|               hasIcon | ||||
|               class=${classMap({ | ||||
|                 green: [5, 6].includes(Number(this.addon.rating)), | ||||
|                 yellow: [3, 4].includes(Number(this.addon.rating)), | ||||
|                 red: [1, 2].includes(Number(this.addon.rating)), | ||||
|               })} | ||||
|               @click=${this._showMoreInfo} | ||||
|               id="rating" | ||||
|             > | ||||
|               <ha-svg-icon slot="icon" .path=${RATING_ICON[this.addon.rating]}> | ||||
|               </ha-svg-icon> | ||||
|  | ||||
|               ${this.supervisor.localize( | ||||
|                 "addon.dashboard.capability.label.rating" | ||||
|               )} | ||||
|             </ha-chip> | ||||
|             ${this.addon.host_network | ||||
|               ? html` | ||||
|                   <ha-chip | ||||
|                     hasIcon | ||||
|                     @click=${this._showMoreInfo} | ||||
|                     id="host_network" | ||||
|                   > | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiNetwork}> </ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.host" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.full_access | ||||
|               ? html` | ||||
|                   <ha-chip | ||||
|                     hasIcon | ||||
|                     @click=${this._showMoreInfo} | ||||
|                     id="full_access" | ||||
|                   > | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiChip}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.hardware" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.homeassistant_api | ||||
|               ? html` | ||||
|                   <ha-chip | ||||
|                     hasIcon | ||||
|                     @click=${this._showMoreInfo} | ||||
|                     id="homeassistant_api" | ||||
|                   > | ||||
|                     <ha-svg-icon | ||||
|                       slot="icon" | ||||
|                       .path=${mdiHomeAssistant} | ||||
|                     ></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.core" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this._computeHassioApi | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="hassio_api"> | ||||
|                     <ha-svg-icon | ||||
|                       slot="icon" | ||||
|                       .path=${mdiHomeAssistant} | ||||
|                     ></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       `addon.dashboard.capability.role.${this.addon.hassio_role}` | ||||
|                     ) || this.addon.hassio_role} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.docker_api | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="docker_api"> | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiDocker}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.docker" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.host_pid | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="host_pid"> | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiPound}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.host_pid" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.apparmor !== "default" | ||||
|               ? html` | ||||
|                   <ha-chip | ||||
|                     hasIcon | ||||
|                     @click=${this._showMoreInfo} | ||||
|                     class=${this._computeApparmorClassName} | ||||
|                     id="apparmor" | ||||
|                   > | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiShield}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.apparmor" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.auth_api | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="auth_api"> | ||||
|                     <ha-svg-icon slot="icon" .path=${mdiKey}></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.auth" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${this.addon.ingress | ||||
|               ? html` | ||||
|                   <ha-chip hasIcon @click=${this._showMoreInfo} id="ingress"> | ||||
|                     <ha-svg-icon | ||||
|                       slot="icon" | ||||
|                       .path=${mdiCursorDefaultClickOutline} | ||||
|                     ></ha-svg-icon> | ||||
|                     ${this.supervisor.localize( | ||||
|                       "addon.dashboard.capability.label.ingress" | ||||
|                     )} | ||||
|                   </ha-chip> | ||||
|                 ` | ||||
|               : ""} | ||||
|           </div> | ||||
|  | ||||
|           <div class="description light-color"> | ||||
|             ${this.addon.description}.<br /> | ||||
|             ${this.supervisor.localize( | ||||
| @@ -269,171 +385,6 @@ class HassioAddonInfo extends LitElement { | ||||
|                     /> | ||||
|                   ` | ||||
|                 : ""} | ||||
|               <div class="security"> | ||||
|                 ${this.addon.stage !== "stable" | ||||
|                   ? html` <ha-label-badge | ||||
|                       class=${classMap({ | ||||
|                         yellow: this.addon.stage === "experimental", | ||||
|                         red: this.addon.stage === "deprecated", | ||||
|                       })} | ||||
|                       @click=${this._showMoreInfo} | ||||
|                       id="stage" | ||||
|                       .label=${this.supervisor.localize( | ||||
|                         "addon.dashboard.capability.label.stage" | ||||
|                       )} | ||||
|                       description="" | ||||
|                     > | ||||
|                       <ha-svg-icon | ||||
|                         .path=${STAGE_ICON[this.addon.stage]} | ||||
|                       ></ha-svg-icon> | ||||
|                     </ha-label-badge>` | ||||
|                   : ""} | ||||
|  | ||||
|                 <ha-label-badge | ||||
|                   class=${classMap({ | ||||
|                     green: [5, 6].includes(Number(this.addon.rating)), | ||||
|                     yellow: [3, 4].includes(Number(this.addon.rating)), | ||||
|                     red: [1, 2].includes(Number(this.addon.rating)), | ||||
|                   })} | ||||
|                   @click=${this._showMoreInfo} | ||||
|                   id="rating" | ||||
|                   .value=${this.addon.rating} | ||||
|                   label="rating" | ||||
|                   description="" | ||||
|                 ></ha-label-badge> | ||||
|                 ${this.addon.host_network | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="host_network" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.host" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiNetwork}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.full_access | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="full_access" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.hardware" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiChip}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.homeassistant_api | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="homeassistant_api" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.hass" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this._computeHassioApi | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="hassio_api" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.hassio" | ||||
|                         )} | ||||
|                         .description=${this.supervisor.localize( | ||||
|                           `addon.dashboard.capability.role.${this.addon.hassio_role}` | ||||
|                         ) || this.addon.hassio_role} | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.docker_api | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="docker_api" | ||||
|                         .label=".${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.docker" | ||||
|                         )}" | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiDocker}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.host_pid | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="host_pid" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.host_pid" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiPound}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.apparmor | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         class=${this._computeApparmorClassName} | ||||
|                         id="apparmor" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.apparmor" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiShield}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.auth_api | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="auth_api" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.auth" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiKey}></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 ${this.addon.ingress | ||||
|                   ? html` | ||||
|                       <ha-label-badge | ||||
|                         @click=${this._showMoreInfo} | ||||
|                         id="ingress" | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "addon.dashboard.capability.label.ingress" | ||||
|                         )} | ||||
|                         description="" | ||||
|                       > | ||||
|                         <ha-svg-icon | ||||
|                           .path=${mdiCursorDefaultClickOutline} | ||||
|                         ></ha-svg-icon> | ||||
|                       </ha-label-badge> | ||||
|                     ` | ||||
|                   : ""} | ||||
|               </div> | ||||
|  | ||||
|               ${this.addon.version | ||||
|                 ? html` | ||||
|                     <div | ||||
| @@ -895,22 +846,14 @@ class HassioAddonInfo extends LitElement { | ||||
|  | ||||
|   private async _openChangelog(): Promise<void> { | ||||
|     try { | ||||
|       let content = await fetchHassioAddonChangelog(this.hass, this.addon.slug); | ||||
|       if ( | ||||
|         content.includes(`# ${this.addon.version}`) && | ||||
|         content.includes(`# ${this.addon.version_latest}`) | ||||
|       ) { | ||||
|         const newcontent = content.split(`# ${this.addon.version}`)[0]; | ||||
|         if (newcontent.includes(`# ${this.addon.version_latest}`)) { | ||||
|           // Only change the content if the new version still exist | ||||
|           // if the changelog does not have the newests version on top | ||||
|           // this will not be true, and we don't modify the content | ||||
|           content = newcontent; | ||||
|         } | ||||
|       } | ||||
|       const content = await fetchHassioAddonChangelog( | ||||
|         this.hass, | ||||
|         this.addon.slug | ||||
|       ); | ||||
|  | ||||
|       showHassioMarkdownDialog(this, { | ||||
|         title: this.supervisor.localize("addon.dashboard.changelog"), | ||||
|         content, | ||||
|         content: extractChangelog(this.addon, content), | ||||
|       }); | ||||
|     } catch (err: any) { | ||||
|       showAlertDialog(this, { | ||||
| @@ -985,33 +928,6 @@ class HassioAddonInfo extends LitElement { | ||||
|     button.progress = false; | ||||
|   } | ||||
|  | ||||
|   private async _updateClicked(): Promise<void> { | ||||
|     showDialogSupervisorUpdate(this, { | ||||
|       supervisor: this.supervisor, | ||||
|       name: this.addon.name, | ||||
|       version: this.addon.version_latest, | ||||
|       backupParams: { | ||||
|         name: `addon_${this.addon.slug}_${this.addon.version}`, | ||||
|         addons: [this.addon.slug], | ||||
|         homeassistant: false, | ||||
|       }, | ||||
|       updateHandler: async () => this._updateAddon(), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   private async _updateAddon(): Promise<void> { | ||||
|     await updateHassioAddon(this.hass, this.addon.slug); | ||||
|     fireEvent(this, "supervisor-collection-refresh", { | ||||
|       collection: "addon", | ||||
|     }); | ||||
|     const eventdata = { | ||||
|       success: true, | ||||
|       response: undefined, | ||||
|       path: "update", | ||||
|     }; | ||||
|     fireEvent(this, "hass-api-called", eventdata); | ||||
|   } | ||||
|  | ||||
|   private async _startClicked(ev: CustomEvent): Promise<void> { | ||||
|     const button = ev.currentTarget as any; | ||||
|     button.progress = true; | ||||
| @@ -1177,34 +1093,31 @@ class HassioAddonInfo extends LitElement { | ||||
|         .description a { | ||||
|           color: var(--primary-color); | ||||
|         } | ||||
|         ha-chip { | ||||
|           text-transform: capitalize; | ||||
|           --ha-chip-text-color: var(--text-primary-color); | ||||
|           --ha-chip-background-color: var(--primary-color); | ||||
|         } | ||||
|  | ||||
|         .red { | ||||
|           --ha-label-badge-color: var(--label-badge-red, #df4c1e); | ||||
|           --ha-chip-background-color: var(--label-badge-red, #df4c1e); | ||||
|         } | ||||
|         .blue { | ||||
|           --ha-label-badge-color: var(--label-badge-blue, #039be5); | ||||
|           --ha-chip-background-color: var(--label-badge-blue, #039be5); | ||||
|         } | ||||
|         .green { | ||||
|           --ha-label-badge-color: var(--label-badge-green, #0da035); | ||||
|           --ha-chip-background-color: var(--label-badge-green, #0da035); | ||||
|         } | ||||
|         .yellow { | ||||
|           --ha-label-badge-color: var(--label-badge-yellow, #f4b400); | ||||
|           --ha-chip-background-color: var(--label-badge-yellow, #f4b400); | ||||
|         } | ||||
|         .security { | ||||
|         .capabilities { | ||||
|           margin-bottom: 16px; | ||||
|         } | ||||
|         .card-actions { | ||||
|           justify-content: space-between; | ||||
|           display: flex; | ||||
|         } | ||||
|         .security h3 { | ||||
|           margin-bottom: 8px; | ||||
|           font-weight: normal; | ||||
|         } | ||||
|         .security ha-label-badge { | ||||
|           cursor: pointer; | ||||
|           margin-right: 4px; | ||||
|           --ha-label-badge-padding: 8px 0 0 0; | ||||
|         } | ||||
|         .changelog { | ||||
|           display: contents; | ||||
|         } | ||||
| @@ -1243,7 +1156,21 @@ class HassioAddonInfo extends LitElement { | ||||
|           align-self: end; | ||||
|         } | ||||
|  | ||||
|         ha-alert mwc-button { | ||||
|           --mdc-theme-primary: var(--primary-text-color); | ||||
|         } | ||||
|         a { | ||||
|           text-decoration: none; | ||||
|         } | ||||
|  | ||||
|         update-available-card { | ||||
|           padding-bottom: 16px; | ||||
|         } | ||||
|  | ||||
|         @media (max-width: 720px) { | ||||
|           ha-chip { | ||||
|             line-height: 36px; | ||||
|           } | ||||
|           .addon-options { | ||||
|             max-width: 100%; | ||||
|           } | ||||
|   | ||||
| @@ -23,7 +23,8 @@ import { | ||||
| } from "../../../src/components/data-table/ha-data-table"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-fab"; | ||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import { | ||||
|   fetchHassioBackups, | ||||
|   friendlyFolderName, | ||||
| @@ -31,6 +32,7 @@ import { | ||||
|   reloadHassioBackups, | ||||
|   removeBackup, | ||||
| } from "../../../src/data/hassio/backup"; | ||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { | ||||
|   showAlertDialog, | ||||
| @@ -40,9 +42,9 @@ import "../../../src/layouts/hass-tabs-subpage-data-table"; | ||||
| import type { HaTabsSubpageDataTable } from "../../../src/layouts/hass-tabs-subpage-data-table"; | ||||
| import { haStyle } from "../../../src/resources/styles"; | ||||
| import { HomeAssistant, Route } from "../../../src/types"; | ||||
| import { showHassioCreateBackupDialog } from "../dialogs/backup/show-dialog-hassio-create-backup"; | ||||
| import { showHassioBackupDialog } from "../dialogs/backup/show-dialog-hassio-backup"; | ||||
| import { showBackupUploadDialog } from "../dialogs/backup/show-dialog-backup-upload"; | ||||
| import { showHassioBackupDialog } from "../dialogs/backup/show-dialog-hassio-backup"; | ||||
| import { showHassioCreateBackupDialog } from "../dialogs/backup/show-dialog-hassio-create-backup"; | ||||
| import { supervisorTabs } from "../hassio-tabs"; | ||||
| import { hassioStyle } from "../resources/hassio-style"; | ||||
|  | ||||
| @@ -156,7 +158,7 @@ export class HassioBackups extends LitElement { | ||||
|     } | ||||
|     return html` | ||||
|       <hass-tabs-subpage-data-table | ||||
|         .tabs=${supervisorTabs} | ||||
|         .tabs=${supervisorTabs(this.hass)} | ||||
|         .hass=${this.hass} | ||||
|         .localizeFunc=${this.supervisor.localize} | ||||
|         .searchLabel=${this.supervisor.localize("search")} | ||||
| @@ -179,9 +181,11 @@ export class HassioBackups extends LitElement { | ||||
|           slot="toolbar-icon" | ||||
|           @action=${this._handleAction} | ||||
|         > | ||||
|           <mwc-icon-button slot="trigger" alt="menu"> | ||||
|             <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon> | ||||
|           </mwc-icon-button> | ||||
|           <ha-icon-button | ||||
|             .label=${this.hass.localize("common.menu")} | ||||
|             .path=${mdiDotsVertical} | ||||
|             slot="trigger" | ||||
|           ></ha-icon-button> | ||||
|           <mwc-list-item> | ||||
|             ${this.supervisor?.localize("common.reload")} | ||||
|           </mwc-list-item> | ||||
| @@ -216,13 +220,15 @@ export class HassioBackups extends LitElement { | ||||
|                       </mwc-button> | ||||
|                     ` | ||||
|                   : html` | ||||
|                       <mwc-icon-button | ||||
|                       <ha-icon-button | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "snapshot.delete_selected" | ||||
|                         )} | ||||
|                         .path=${mdiDelete} | ||||
|                         id="delete-btn" | ||||
|                         class="warning" | ||||
|                         @click=${this._deleteSelected} | ||||
|                       > | ||||
|                         <ha-svg-icon .path=${mdiDelete}></ha-svg-icon> | ||||
|                       </mwc-icon-button> | ||||
|                       ></ha-icon-button> | ||||
|                       <paper-tooltip animation-delay="0" for="delete-btn"> | ||||
|                         ${this.supervisor.localize("backup.delete_selected")} | ||||
|                       </paper-tooltip> | ||||
| @@ -368,7 +374,7 @@ export class HassioBackups extends LitElement { | ||||
|           margin-right: -12px; | ||||
|         } | ||||
|         .header-btns > mwc-button, | ||||
|         .header-btns > mwc-icon-button { | ||||
|         .header-btns > ha-icon-button { | ||||
|           margin: 8px; | ||||
|         } | ||||
|       `, | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| import { mdiHelpCircle } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../src/components/ha-relative-time"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
|  | ||||
| @@ -19,8 +18,6 @@ class HassioCardContent extends LitElement { | ||||
|  | ||||
|   @property() public topbarClass?: string; | ||||
|  | ||||
|   @property() public datetime?: string; | ||||
|  | ||||
|   @property() public iconTitle?: string; | ||||
|  | ||||
|   @property() public iconClass?: string; | ||||
| @@ -56,15 +53,6 @@ class HassioCardContent extends LitElement { | ||||
|             /* treat as available when undefined */ | ||||
|             this.available === false ? " (Not available)" : "" | ||||
|           } | ||||
|           ${this.datetime | ||||
|             ? html` | ||||
|                 <ha-relative-time | ||||
|                   .hass=${this.hass} | ||||
|                   class="addition" | ||||
|                   .datetime=${this.datetime} | ||||
|                 ></ha-relative-time> | ||||
|               ` | ||||
|             : undefined} | ||||
|         </div> | ||||
|       </div> | ||||
|     `; | ||||
| @@ -106,9 +94,6 @@ class HassioCardContent extends LitElement { | ||||
|         height: 2.4em; | ||||
|         line-height: 1.2em; | ||||
|       } | ||||
|       ha-relative-time { | ||||
|         display: block; | ||||
|       } | ||||
|       .icon_image img { | ||||
|         max-height: 40px; | ||||
|         max-width: 40px; | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "@material/mwc-icon-button/mwc-icon-button"; | ||||
| import { mdiFolderUpload } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input-container"; | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| @@ -6,9 +5,8 @@ import { customElement, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/ha-circular-progress"; | ||||
| import "../../../src/components/ha-file-upload"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||
| import { HassioBackup, uploadBackup } from "../../../src/data/hassio/backup"; | ||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||
| import { showAlertDialog } from "../../../src/dialogs/generic/show-dialog-box"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
|  | ||||
| @@ -18,11 +16,9 @@ declare global { | ||||
|   } | ||||
| } | ||||
|  | ||||
| const MAX_FILE_SIZE = 1 * 1024 * 1024 * 1024; // 1GB | ||||
|  | ||||
| @customElement("hassio-upload-backup") | ||||
| export class HassioUploadBackup extends LitElement { | ||||
|   public hass!: HomeAssistant; | ||||
|   public hass?: HomeAssistant; | ||||
|  | ||||
|   @state() public value: string | null = null; | ||||
|  | ||||
| @@ -31,6 +27,7 @@ export class HassioUploadBackup extends LitElement { | ||||
|   public render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-file-upload | ||||
|         .hass=${this.hass} | ||||
|         .uploading=${this._uploading} | ||||
|         .icon=${mdiFolderUpload} | ||||
|         accept="application/x-tar" | ||||
| @@ -44,20 +41,6 @@ export class HassioUploadBackup extends LitElement { | ||||
|   private async _uploadFile(ev) { | ||||
|     const file = ev.detail.files[0]; | ||||
|  | ||||
|     if (file.size > MAX_FILE_SIZE) { | ||||
|       showAlertDialog(this, { | ||||
|         title: "Backup file is too big", | ||||
|         text: html`The maximum allowed filesize is 1GB.<br /> | ||||
|           <a | ||||
|             href="https://www.home-assistant.io/hassio/haos_common_tasks/#restoring-a-backup-on-a-new-install" | ||||
|             target="_blank" | ||||
|             >Have a look here on how to restore it.</a | ||||
|           >`, | ||||
|         confirmText: "ok", | ||||
|       }); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if (!["application/x-tar"].includes(file.type)) { | ||||
|       showAlertDialog(this, { | ||||
|         title: "Unsupported file format", | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { navigate } from "../../../src/common/navigate"; | ||||
| import { stringCompare } from "../../../src/common/string/compare"; | ||||
| import { caseInsensitiveStringCompare } from "../../../src/common/string/compare"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { haStyle } from "../../../src/resources/styles"; | ||||
| @@ -20,7 +20,9 @@ class HassioAddons extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <div class="content"> | ||||
|         <h1>${this.supervisor.localize("dashboard.addons")}</h1> | ||||
|         ${!atLeastVersion(this.hass.config.version, 2021, 12) | ||||
|           ? html` <h1>${this.supervisor.localize("dashboard.addons")}</h1> ` | ||||
|           : ""} | ||||
|         <div class="card-group"> | ||||
|           ${!this.supervisor.supervisor.addons?.length | ||||
|             ? html` | ||||
| @@ -33,7 +35,7 @@ class HassioAddons extends LitElement { | ||||
|                 </ha-card> | ||||
|               ` | ||||
|             : this.supervisor.supervisor.addons | ||||
|                 .sort((a, b) => stringCompare(a.name, b.name)) | ||||
|                 .sort((a, b) => caseInsensitiveStringCompare(a.name, b.name)) | ||||
|                 .map( | ||||
|                   (addon) => html` | ||||
|                     <ha-card .addon=${addon} @click=${this._addonTapped}> | ||||
|   | ||||
| @@ -1,5 +1,8 @@ | ||||
| import { mdiStorePlus } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import "../../../src/components/ha-fab"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import "../../../src/layouts/hass-tabs-subpage"; | ||||
| import { haStyle } from "../../../src/resources/styles"; | ||||
| @@ -25,23 +28,37 @@ class HassioDashboard extends LitElement { | ||||
|         .localizeFunc=${this.supervisor.localize} | ||||
|         .narrow=${this.narrow} | ||||
|         .route=${this.route} | ||||
|         .tabs=${supervisorTabs} | ||||
|         .tabs=${supervisorTabs(this.hass)} | ||||
|         main-page | ||||
|         supervisor | ||||
|         hasFab | ||||
|       > | ||||
|         <span slot="header"> | ||||
|           ${this.supervisor.localize("panel.dashboard")} | ||||
|         </span> | ||||
|         <div class="content"> | ||||
|           <hassio-update | ||||
|             .hass=${this.hass} | ||||
|             .supervisor=${this.supervisor} | ||||
|           ></hassio-update> | ||||
|           ${this.hass.config.version.includes("dev") || | ||||
|           !atLeastVersion(this.hass.config.version, 2021, 12) | ||||
|             ? html` | ||||
|                 <hassio-update | ||||
|                   .hass=${this.hass} | ||||
|                   .supervisor=${this.supervisor} | ||||
|                 ></hassio-update> | ||||
|               ` | ||||
|             : ""} | ||||
|           <hassio-addons | ||||
|             .hass=${this.hass} | ||||
|             .supervisor=${this.supervisor} | ||||
|           ></hassio-addons> | ||||
|         </div> | ||||
|  | ||||
|         <a href="/hassio/store" slot="fab"> | ||||
|           <ha-fab .label=${this.supervisor.localize("panel.store")} extended> | ||||
|             <ha-svg-icon | ||||
|               slot="icon" | ||||
|               .path=${mdiStorePlus} | ||||
|             ></ha-svg-icon> </ha-fab | ||||
|         ></a> | ||||
|       </hass-tabs-subpage> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -3,34 +3,18 @@ import { mdiHomeAssistant } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
|   HassioResponse, | ||||
|   ignoreSupervisorError, | ||||
| } from "../../../src/data/hassio/common"; | ||||
| import { HassioHassOSInfo } from "../../../src/data/hassio/host"; | ||||
| import { | ||||
|   HassioHomeAssistantInfo, | ||||
|   HassioSupervisorInfo, | ||||
| } from "../../../src/data/hassio/supervisor"; | ||||
| import { updateCore } from "../../../src/data/supervisor/core"; | ||||
| import { | ||||
|   Supervisor, | ||||
|   supervisorApiWsRequest, | ||||
| } from "../../../src/data/supervisor/supervisor"; | ||||
| import { | ||||
|   showAlertDialog, | ||||
|   showConfirmationDialog, | ||||
| } from "../../../src/dialogs/generic/show-dialog-box"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { haStyle } from "../../../src/resources/styles"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { showDialogSupervisorUpdate } from "../dialogs/update/show-dialog-update"; | ||||
| import { hassioStyle } from "../resources/hassio-style"; | ||||
|  | ||||
| const computeVersion = (key: string, version: string): string => | ||||
| @@ -73,26 +57,18 @@ export class HassioUpdate extends LitElement { | ||||
|           ${this._renderUpdateCard( | ||||
|             "Home Assistant Core", | ||||
|             "core", | ||||
|             this.supervisor.core, | ||||
|             "hassio/homeassistant/update", | ||||
|             `https://${ | ||||
|               this.supervisor.core.version_latest.includes("b") ? "rc" : "www" | ||||
|             }.home-assistant.io/latest-release-notes/` | ||||
|             this.supervisor.core | ||||
|           )} | ||||
|           ${this._renderUpdateCard( | ||||
|             "Supervisor", | ||||
|             "supervisor", | ||||
|             this.supervisor.supervisor, | ||||
|             "hassio/supervisor/update", | ||||
|             `https://github.com//home-assistant/hassio/releases/tag/${this.supervisor.supervisor.version_latest}` | ||||
|             this.supervisor.supervisor | ||||
|           )} | ||||
|           ${this.supervisor.host.features.includes("haos") | ||||
|             ? this._renderUpdateCard( | ||||
|                 "Operating System", | ||||
|                 "os", | ||||
|                 this.supervisor.os, | ||||
|                 "hassio/os/update", | ||||
|                 `https://github.com//home-assistant/hassos/releases/tag/${this.supervisor.os.version_latest}` | ||||
|                 this.supervisor.os | ||||
|               ) | ||||
|             : ""} | ||||
|         </div> | ||||
| @@ -103,9 +79,7 @@ export class HassioUpdate extends LitElement { | ||||
|   private _renderUpdateCard( | ||||
|     name: string, | ||||
|     key: string, | ||||
|     object: HassioHomeAssistantInfo | HassioSupervisorInfo | HassioHassOSInfo, | ||||
|     apiPath: string, | ||||
|     releaseNotesUrl: string | ||||
|     object: HassioHomeAssistantInfo | HassioSupervisorInfo | HassioHassOSInfo | ||||
|   ): TemplateResult { | ||||
|     if (!object.update_available) { | ||||
|       return html``; | ||||
| @@ -136,96 +110,15 @@ export class HassioUpdate extends LitElement { | ||||
|           </ha-settings-row> | ||||
|         </div> | ||||
|         <div class="card-actions"> | ||||
|           <a href=${releaseNotesUrl} target="_blank" rel="noreferrer"> | ||||
|             <mwc-button> | ||||
|               ${this.supervisor.localize("common.release_notes")} | ||||
|           <a href="/hassio/update-available/${key}"> | ||||
|             <mwc-button .label=${this.supervisor.localize("common.show")}> | ||||
|             </mwc-button> | ||||
|           </a> | ||||
|           <ha-progress-button | ||||
|             .apiPath=${apiPath} | ||||
|             .name=${name} | ||||
|             .key=${key} | ||||
|             .version=${object.version_latest} | ||||
|             @click=${this._confirmUpdate} | ||||
|           > | ||||
|             ${this.supervisor.localize("common.update")} | ||||
|           </ha-progress-button> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private async _confirmUpdate(ev): Promise<void> { | ||||
|     const item = ev.currentTarget; | ||||
|     if (item.key === "core") { | ||||
|       showDialogSupervisorUpdate(this, { | ||||
|         supervisor: this.supervisor, | ||||
|         name: "Home Assistant Core", | ||||
|         version: this.supervisor.core.version_latest, | ||||
|         backupParams: { | ||||
|           name: `core_${this.supervisor.core.version}`, | ||||
|           folders: ["homeassistant"], | ||||
|           homeassistant: true, | ||||
|         }, | ||||
|         updateHandler: async () => this._updateCore(), | ||||
|       }); | ||||
|       return; | ||||
|     } | ||||
|     item.progress = true; | ||||
|     const confirmed = await showConfirmationDialog(this, { | ||||
|       title: this.supervisor.localize( | ||||
|         "confirm.update.title", | ||||
|         "name", | ||||
|         item.name | ||||
|       ), | ||||
|       text: this.supervisor.localize( | ||||
|         "confirm.update.text", | ||||
|         "name", | ||||
|         item.name, | ||||
|         "version", | ||||
|         computeVersion(item.key, item.version) | ||||
|       ), | ||||
|       confirmText: this.supervisor.localize("common.update"), | ||||
|       dismissText: this.supervisor.localize("common.cancel"), | ||||
|     }); | ||||
|  | ||||
|     if (!confirmed) { | ||||
|       item.progress = false; | ||||
|       return; | ||||
|     } | ||||
|     try { | ||||
|       if (atLeastVersion(this.hass.config.version, 2021, 2, 4)) { | ||||
|         await supervisorApiWsRequest(this.hass.connection, { | ||||
|           method: "post", | ||||
|           endpoint: item.apiPath.replace("hassio", ""), | ||||
|           timeout: null, | ||||
|         }); | ||||
|       } else { | ||||
|         await this.hass.callApi<HassioResponse<void>>("POST", item.apiPath); | ||||
|       } | ||||
|       fireEvent(this, "supervisor-collection-refresh", { | ||||
|         collection: item.key, | ||||
|       }); | ||||
|     } catch (err: any) { | ||||
|       // Only show an error if the status code was not expected (user behind proxy) | ||||
|       // or no status at all(connection terminated) | ||||
|       if (this.hass.connection.connected && !ignoreSupervisorError(err)) { | ||||
|         showAlertDialog(this, { | ||||
|           title: this.supervisor.localize("common.error.update_failed"), | ||||
|           text: extractApiErrorMessage(err), | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|     item.progress = false; | ||||
|   } | ||||
|  | ||||
|   private async _updateCore(): Promise<void> { | ||||
|     await updateCore(this.hass); | ||||
|     fireEvent(this, "supervisor-collection-refresh", { | ||||
|       collection: "core", | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return [ | ||||
|       haStyle, | ||||
|   | ||||
| @@ -3,6 +3,7 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-header-bar"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import { HassDialog } from "../../../../src/dialogs/make-dialog-manager"; | ||||
| import { haStyleDialog } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| @@ -14,7 +15,7 @@ export class DialogHassioBackupUpload | ||||
|   extends LitElement | ||||
|   implements HassDialog<HassioBackupUploadDialogParams> | ||||
| { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|   @property({ attribute: false }) public hass?: HomeAssistant; | ||||
|  | ||||
|   @state() private _params?: HassioBackupUploadDialogParams; | ||||
|  | ||||
| @@ -52,9 +53,12 @@ export class DialogHassioBackupUpload | ||||
|         <div slot="heading"> | ||||
|           <ha-header-bar> | ||||
|             <span slot="title"> Upload backup </span> | ||||
|             <mwc-icon-button slot="actionItems" dialogAction="cancel"> | ||||
|               <ha-svg-icon .path=${mdiClose}></ha-svg-icon> | ||||
|             </mwc-icon-button> | ||||
|             <ha-icon-button | ||||
|               .label=${this.hass?.localize("common.close") || "close"} | ||||
|               .path=${mdiClose} | ||||
|               slot="actionItems" | ||||
|               dialogAction="cancel" | ||||
|             ></ha-icon-button> | ||||
|           </ha-header-bar> | ||||
|         </div> | ||||
|         <hassio-upload-backup | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import "../../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-button-menu"; | ||||
| import "../../../../src/components/ha-header-bar"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import { getSignedPath } from "../../../../src/data/auth"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| import { | ||||
| @@ -35,7 +35,7 @@ class HassioBackupDialog | ||||
|   extends LitElement | ||||
|   implements HassDialog<HassioBackupDialogParams> | ||||
| { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|   @property({ attribute: false }) public hass?: HomeAssistant; | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
| @@ -76,9 +76,12 @@ class HassioBackupDialog | ||||
|         <div slot="heading"> | ||||
|           <ha-header-bar> | ||||
|             <span slot="title">${this._backup.name}</span> | ||||
|             <mwc-icon-button slot="actionItems" dialogAction="cancel"> | ||||
|               <ha-svg-icon .path=${mdiClose}></ha-svg-icon> | ||||
|             </mwc-icon-button> | ||||
|             <ha-icon-button | ||||
|               .label=${this.hass?.localize("common.close") || "close"} | ||||
|               .path=${mdiClose} | ||||
|               slot="actionItems" | ||||
|               dialogAction="cancel" | ||||
|             ></ha-icon-button> | ||||
|           </ha-header-bar> | ||||
|         </div> | ||||
|         ${this._restoringBackup | ||||
| @@ -110,9 +113,11 @@ class HassioBackupDialog | ||||
|               @action=${this._handleMenuAction} | ||||
|               @closed=${stopPropagation} | ||||
|             > | ||||
|               <mwc-icon-button slot="trigger" alt="menu"> | ||||
|                 <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon> | ||||
|               </mwc-icon-button> | ||||
|               <ha-icon-button | ||||
|                 .label=${this.hass!.localize("common.menu")} | ||||
|                 .path=${mdiDotsVertical} | ||||
|                 slot="trigger" | ||||
|               ></ha-icon-button> | ||||
|               <mwc-list-item>Download Backup</mwc-list-item> | ||||
|               <mwc-list-item class="error">Delete Backup</mwc-list-item> | ||||
|             </ha-button-menu>` | ||||
| @@ -126,9 +131,6 @@ class HassioBackupDialog | ||||
|       haStyle, | ||||
|       haStyleDialog, | ||||
|       css` | ||||
|         ha-svg-icon { | ||||
|           color: var(--primary-text-color); | ||||
|         } | ||||
|         ha-circular-progress { | ||||
|           display: block; | ||||
|           text-align: center; | ||||
| @@ -139,6 +141,9 @@ class HassioBackupDialog | ||||
|           flex-shrink: 0; | ||||
|           display: block; | ||||
|         } | ||||
|         ha-icon-button { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
| @@ -187,25 +192,23 @@ class HassioBackupDialog | ||||
|     } | ||||
|  | ||||
|     if (!this._dialogParams?.onboarding) { | ||||
|       this.hass | ||||
|         .callApi( | ||||
|           "POST", | ||||
|       this.hass!.callApi( | ||||
|         "POST", | ||||
|  | ||||
|           `hassio/${ | ||||
|             atLeastVersion(this.hass.config.version, 2021, 9) | ||||
|               ? "backups" | ||||
|               : "snapshots" | ||||
|           }/${this._backup!.slug}/restore/partial`, | ||||
|           backupDetails | ||||
|         ) | ||||
|         .then( | ||||
|           () => { | ||||
|             this.closeDialog(); | ||||
|           }, | ||||
|           (error) => { | ||||
|             this._error = error.body.message; | ||||
|           } | ||||
|         ); | ||||
|         `hassio/${ | ||||
|           atLeastVersion(this.hass!.config.version, 2021, 9) | ||||
|             ? "backups" | ||||
|             : "snapshots" | ||||
|         }/${this._backup!.slug}/restore/partial`, | ||||
|         backupDetails | ||||
|       ).then( | ||||
|         () => { | ||||
|           this.closeDialog(); | ||||
|         }, | ||||
|         (error) => { | ||||
|           this._error = error.body.message; | ||||
|         } | ||||
|       ); | ||||
|     } else { | ||||
|       fireEvent(this, "restoring"); | ||||
|       fetch(`/api/hassio/backups/${this._backup!.slug}/restore/partial`, { | ||||
| @@ -239,24 +242,22 @@ class HassioBackupDialog | ||||
|     } | ||||
|  | ||||
|     if (!this._dialogParams?.onboarding) { | ||||
|       this.hass | ||||
|         .callApi( | ||||
|           "POST", | ||||
|           `hassio/${ | ||||
|             atLeastVersion(this.hass.config.version, 2021, 9) | ||||
|               ? "backups" | ||||
|               : "snapshots" | ||||
|           }/${this._backup!.slug}/restore/full`, | ||||
|           backupDetails | ||||
|         ) | ||||
|         .then( | ||||
|           () => { | ||||
|             this.closeDialog(); | ||||
|           }, | ||||
|           (error) => { | ||||
|             this._error = error.body.message; | ||||
|           } | ||||
|         ); | ||||
|       this.hass!.callApi( | ||||
|         "POST", | ||||
|         `hassio/${ | ||||
|           atLeastVersion(this.hass!.config.version, 2021, 9) | ||||
|             ? "backups" | ||||
|             : "snapshots" | ||||
|         }/${this._backup!.slug}/restore/full`, | ||||
|         backupDetails | ||||
|       ).then( | ||||
|         () => { | ||||
|           this.closeDialog(); | ||||
|         }, | ||||
|         (error) => { | ||||
|           this._error = error.body.message; | ||||
|         } | ||||
|       ); | ||||
|     } else { | ||||
|       fireEvent(this, "restoring"); | ||||
|       fetch(`/api/hassio/backups/${this._backup!.slug}/restore/full`, { | ||||
| @@ -278,36 +279,33 @@ class HassioBackupDialog | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     this.hass | ||||
|  | ||||
|       .callApi( | ||||
|         atLeastVersion(this.hass.config.version, 2021, 9) ? "DELETE" : "POST", | ||||
|         `hassio/${ | ||||
|           atLeastVersion(this.hass.config.version, 2021, 9) | ||||
|             ? `backups/${this._backup!.slug}` | ||||
|             : `snapshots/${this._backup!.slug}/remove` | ||||
|         }` | ||||
|       ) | ||||
|       .then( | ||||
|         () => { | ||||
|           if (this._dialogParams!.onDelete) { | ||||
|             this._dialogParams!.onDelete(); | ||||
|           } | ||||
|           this.closeDialog(); | ||||
|         }, | ||||
|         (error) => { | ||||
|           this._error = error.body.message; | ||||
|     this.hass!.callApi( | ||||
|       atLeastVersion(this.hass!.config.version, 2021, 9) ? "DELETE" : "POST", | ||||
|       `hassio/${ | ||||
|         atLeastVersion(this.hass!.config.version, 2021, 9) | ||||
|           ? `backups/${this._backup!.slug}` | ||||
|           : `snapshots/${this._backup!.slug}/remove` | ||||
|       }` | ||||
|     ).then( | ||||
|       () => { | ||||
|         if (this._dialogParams!.onDelete) { | ||||
|           this._dialogParams!.onDelete(); | ||||
|         } | ||||
|       ); | ||||
|         this.closeDialog(); | ||||
|       }, | ||||
|       (error) => { | ||||
|         this._error = error.body.message; | ||||
|       } | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   private async _downloadClicked() { | ||||
|     let signedPath: { path: string }; | ||||
|     try { | ||||
|       signedPath = await getSignedPath( | ||||
|         this.hass, | ||||
|         this.hass!, | ||||
|         `/api/hassio/${ | ||||
|           atLeastVersion(this.hass.config.version, 2021, 9) | ||||
|           atLeastVersion(this.hass!.config.version, 2021, 9) | ||||
|             ? "backups" | ||||
|             : "snapshots" | ||||
|         }/${this._backup!.slug}/download` | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import "../../../../src/common/search/search-input"; | ||||
| import { stringCompare } from "../../../../src/common/string/compare"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-expansion-panel"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import { HassioHardwareInfo } from "../../../../src/data/hassio/hardware"; | ||||
| import { dump } from "../../../../src/resources/js-yaml-dump"; | ||||
| import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | ||||
| @@ -70,10 +71,13 @@ class HassioHardwareDialog extends LitElement { | ||||
|           <h2> | ||||
|             ${this._dialogParams.supervisor.localize("dialog.hardware.title")} | ||||
|           </h2> | ||||
|           <mwc-icon-button dialogAction="close"> | ||||
|             <ha-svg-icon .path=${mdiClose}></ha-svg-icon> | ||||
|           </mwc-icon-button> | ||||
|           <ha-icon-button | ||||
|             .label=${this.hass.localize("common.close")} | ||||
|             .path=${mdiClose} | ||||
|             dialogAction="close" | ||||
|           ></ha-icon-button> | ||||
|           <search-input | ||||
|             .hass=${this.hass} | ||||
|             autofocus | ||||
|             no-label-float | ||||
|             .filter=${this._filter} | ||||
| @@ -141,7 +145,7 @@ class HassioHardwareDialog extends LitElement { | ||||
|       haStyle, | ||||
|       haStyleDialog, | ||||
|       css` | ||||
|         mwc-icon-button { | ||||
|         ha-icon-button { | ||||
|           position: absolute; | ||||
|           right: 16px; | ||||
|           top: 10px; | ||||
|   | ||||
| @@ -47,11 +47,6 @@ class HassioMarkdownDialog extends LitElement { | ||||
|       haStyleDialog, | ||||
|       hassioStyle, | ||||
|       css` | ||||
|         ha-paper-dialog { | ||||
|           min-width: 350px; | ||||
|           font-size: 14px; | ||||
|           border-radius: 2px; | ||||
|         } | ||||
|         app-toolbar { | ||||
|           margin: 0; | ||||
|           padding: 0 16px; | ||||
| @@ -62,19 +57,6 @@ class HassioMarkdownDialog extends LitElement { | ||||
|           margin-left: 16px; | ||||
|         } | ||||
|         @media all and (max-width: 450px), all and (max-height: 500px) { | ||||
|           ha-paper-dialog { | ||||
|             max-height: 100%; | ||||
|           } | ||||
|           ha-paper-dialog::before { | ||||
|             content: ""; | ||||
|             position: fixed; | ||||
|             z-index: -1; | ||||
|             top: 0px; | ||||
|             left: 0px; | ||||
|             right: 0px; | ||||
|             bottom: 0px; | ||||
|             background-color: inherit; | ||||
|           } | ||||
|           app-toolbar { | ||||
|             color: var(--text-primary-color); | ||||
|             background-color: var(--primary-color); | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import "@material/mwc-icon-button"; | ||||
| import "@material/mwc-list/mwc-list"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import "@material/mwc-tab"; | ||||
| @@ -16,9 +15,9 @@ import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-expansion-panel"; | ||||
| import "../../../../src/components/ha-formfield"; | ||||
| import "../../../../src/components/ha-header-bar"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-radio"; | ||||
| import "../../../../src/components/ha-related-items"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| import { | ||||
|   AccessPoints, | ||||
| @@ -104,9 +103,12 @@ export class DialogHassioNetwork | ||||
|             <span slot="title"> | ||||
|               ${this.supervisor.localize("dialog.network.title")} | ||||
|             </span> | ||||
|             <mwc-icon-button slot="actionItems" dialogAction="cancel"> | ||||
|               <ha-svg-icon .path=${mdiClose}></ha-svg-icon> | ||||
|             </mwc-icon-button> | ||||
|             <ha-icon-button | ||||
|               .label=${this.hass.localize("common.close")} | ||||
|               .path=${mdiClose} | ||||
|               slot="actionItems" | ||||
|               dialogAction="cancel" | ||||
|             ></ha-icon-button> | ||||
|           </ha-header-bar> | ||||
|           ${this._interfaces.length > 1 | ||||
|             ? html`<mwc-tab-bar | ||||
|   | ||||
| @@ -1,13 +1,12 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import "@material/mwc-icon-button/mwc-icon-button"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { mdiDelete } from "@mdi/js"; | ||||
| import { PaperInputElement } from "@polymer/paper-input/paper-input"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-circular-progress"; | ||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||
| import { | ||||
|   addHassioDockerRegistry, | ||||
| @@ -20,22 +19,41 @@ import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import { RegistriesDialogParams } from "./show-dialog-registries"; | ||||
|  | ||||
| const SCHEMA = [ | ||||
|   { | ||||
|     type: "string", | ||||
|     name: "registry", | ||||
|     required: true, | ||||
|   }, | ||||
|   { | ||||
|     type: "string", | ||||
|     name: "username", | ||||
|     required: true, | ||||
|   }, | ||||
|   { | ||||
|     type: "string", | ||||
|     name: "password", | ||||
|     required: true, | ||||
|     format: "password", | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("dialog-hassio-registries") | ||||
| class HassioRegistriesDialog extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public supervisor!: Supervisor; | ||||
|  | ||||
|   @property({ attribute: false }) private _registries?: { | ||||
|   @state() private _registries?: { | ||||
|     registry: string; | ||||
|     username: string; | ||||
|   }[]; | ||||
|  | ||||
|   @state() private _registry?: string; | ||||
|  | ||||
|   @state() private _username?: string; | ||||
|  | ||||
|   @state() private _password?: string; | ||||
|   @state() private _input: { | ||||
|     registry?: string; | ||||
|     username?: string; | ||||
|     password?: string; | ||||
|   } = {}; | ||||
|  | ||||
|   @state() private _opened = false; | ||||
|  | ||||
| @@ -48,6 +66,7 @@ class HassioRegistriesDialog extends LitElement { | ||||
|         @closed=${this.closeDialog} | ||||
|         scrimClickAction | ||||
|         escapeKeyAction | ||||
|         hideActions | ||||
|         .heading=${createCloseHeading( | ||||
|           this.hass, | ||||
|           this._addingRegistry | ||||
| @@ -55,100 +74,77 @@ class HassioRegistriesDialog extends LitElement { | ||||
|             : this.supervisor.localize("dialog.registries.title_manage") | ||||
|         )} | ||||
|       > | ||||
|         <div class="form"> | ||||
|           ${this._addingRegistry | ||||
|             ? html` | ||||
|                 <paper-input | ||||
|                   @value-changed=${this._inputChanged} | ||||
|                   class="flex-auto" | ||||
|                   name="registry" | ||||
|                   .label=${this.supervisor.localize( | ||||
|                     "dialog.registries.registry" | ||||
|                   )} | ||||
|                   required | ||||
|                   auto-validate | ||||
|                 ></paper-input> | ||||
|                 <paper-input | ||||
|                   @value-changed=${this._inputChanged} | ||||
|                   class="flex-auto" | ||||
|                   name="username" | ||||
|                   .label=${this.supervisor.localize( | ||||
|                     "dialog.registries.username" | ||||
|                   )} | ||||
|                   required | ||||
|                   auto-validate | ||||
|                 ></paper-input> | ||||
|                 <paper-input | ||||
|                   @value-changed=${this._inputChanged} | ||||
|                   class="flex-auto" | ||||
|                   name="password" | ||||
|                   .label=${this.supervisor.localize( | ||||
|                     "dialog.registries.password" | ||||
|                   )} | ||||
|                   type="password" | ||||
|                   required | ||||
|                   auto-validate | ||||
|                 ></paper-input> | ||||
|  | ||||
|         ${this._addingRegistry | ||||
|           ? html` | ||||
|               <ha-form | ||||
|                 .data=${this._input} | ||||
|                 .schema=${SCHEMA} | ||||
|                 @value-changed=${this._valueChanged} | ||||
|                 .computeLabel=${this._computeLabel} | ||||
|               ></ha-form> | ||||
|               <div class="action"> | ||||
|                 <mwc-button | ||||
|                   ?disabled=${Boolean( | ||||
|                     !this._registry || !this._username || !this._password | ||||
|                     !this._input.registry || | ||||
|                       !this._input.username || | ||||
|                       !this._input.password | ||||
|                   )} | ||||
|                   @click=${this._addNewRegistry} | ||||
|                 > | ||||
|                   ${this.supervisor.localize("dialog.registries.add_registry")} | ||||
|                 </mwc-button> | ||||
|               ` | ||||
|             : html`${this._registries?.length | ||||
|                   ? this._registries.map( | ||||
|                       (entry) => html` | ||||
|                         <mwc-list-item class="option" hasMeta twoline> | ||||
|                           <span>${entry.registry}</span> | ||||
|                           <span slot="secondary" | ||||
|                             >${this.supervisor.localize( | ||||
|                               "dialog.registries.username" | ||||
|                             )}: | ||||
|                             ${entry.username}</span | ||||
|                           > | ||||
|                           <mwc-icon-button | ||||
|                             .entry=${entry} | ||||
|                             .title=${this.supervisor.localize( | ||||
|                               "dialog.registries.remove" | ||||
|                             )} | ||||
|                             slot="meta" | ||||
|                             @click=${this._removeRegistry} | ||||
|                           > | ||||
|                             <ha-svg-icon .path=${mdiDelete}></ha-svg-icon> | ||||
|                           </mwc-icon-button> | ||||
|                         </mwc-list-item> | ||||
|                       ` | ||||
|                     ) | ||||
|                   : html` | ||||
|                       <mwc-list-item> | ||||
|                         <span | ||||
|                           >${this.supervisor.localize( | ||||
|                             "dialog.registries.no_registries" | ||||
|                           )}</span | ||||
|                         > | ||||
|                       </mwc-list-item> | ||||
|                     `} | ||||
|               </div> | ||||
|             ` | ||||
|           : html`${this._registries?.length | ||||
|                 ? this._registries.map( | ||||
|                     (entry) => html` | ||||
|                       <ha-settings-row class="registry"> | ||||
|                         <span slot="heading"> ${entry.registry} </span> | ||||
|                         <span slot="description"> | ||||
|                           ${this.supervisor.localize( | ||||
|                             "dialog.registries.username" | ||||
|                           )}: | ||||
|                           ${entry.username} | ||||
|                         </span> | ||||
|                         <ha-icon-button | ||||
|                           .entry=${entry} | ||||
|                           .label=${this.supervisor.localize( | ||||
|                             "dialog.registries.remove" | ||||
|                           )} | ||||
|                           .path=${mdiDelete} | ||||
|                           @click=${this._removeRegistry} | ||||
|                         ></ha-icon-button> | ||||
|                       </ha-settings-row> | ||||
|                     ` | ||||
|                   ) | ||||
|                 : html` | ||||
|                     <ha-alert> | ||||
|                       ${this.supervisor.localize( | ||||
|                         "dialog.registries.no_registries" | ||||
|                       )} | ||||
|                     </ha-alert> | ||||
|                   `} | ||||
|               <div class="action"> | ||||
|                 <mwc-button @click=${this._addRegistry}> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "dialog.registries.add_new_registry" | ||||
|                   )} | ||||
|                 </mwc-button> `} | ||||
|         </div> | ||||
|                 </mwc-button> | ||||
|               </div> `} | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _inputChanged(ev: Event) { | ||||
|     const target = ev.currentTarget as PaperInputElement; | ||||
|     this[`_${target.name}`] = target.value; | ||||
|   private _computeLabel = (schema: HaFormSchema) => | ||||
|     this.supervisor.localize(`dialog.registries.${schema.name}`) || schema.name; | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent) { | ||||
|     this._input = ev.detail.value; | ||||
|   } | ||||
|  | ||||
|   public async showDialog(dialogParams: RegistriesDialogParams): Promise<void> { | ||||
|     this._opened = true; | ||||
|     this._input = {}; | ||||
|     this.supervisor = dialogParams.supervisor; | ||||
|     await this._loadRegistries(); | ||||
|     await this.updateComplete; | ||||
| @@ -157,6 +153,7 @@ class HassioRegistriesDialog extends LitElement { | ||||
|   public closeDialog(): void { | ||||
|     this._addingRegistry = false; | ||||
|     this._opened = false; | ||||
|     this._input = {}; | ||||
|   } | ||||
|  | ||||
|   public focus(): void { | ||||
| @@ -181,15 +178,16 @@ class HassioRegistriesDialog extends LitElement { | ||||
|  | ||||
|   private async _addNewRegistry(): Promise<void> { | ||||
|     const data = {}; | ||||
|     data[this._registry!] = { | ||||
|       username: this._username, | ||||
|       password: this._password, | ||||
|     data[this._input.registry!] = { | ||||
|       username: this._input.username, | ||||
|       password: this._input.password, | ||||
|     }; | ||||
|  | ||||
|     try { | ||||
|       await addHassioDockerRegistry(this.hass, data); | ||||
|       await this._loadRegistries(); | ||||
|       this._addingRegistry = false; | ||||
|       this._input = {}; | ||||
|     } catch (err: any) { | ||||
|       showAlertDialog(this, { | ||||
|         title: this.supervisor.localize("dialog.registries.failed_to_add"), | ||||
| @@ -217,32 +215,20 @@ class HassioRegistriesDialog extends LitElement { | ||||
|       haStyle, | ||||
|       haStyleDialog, | ||||
|       css` | ||||
|         ha-dialog.button-left { | ||||
|           --justify-action-buttons: flex-start; | ||||
|         } | ||||
|         paper-icon-item { | ||||
|           cursor: pointer; | ||||
|         } | ||||
|         .form { | ||||
|           color: var(--primary-text-color); | ||||
|         } | ||||
|         .option { | ||||
|         .registry { | ||||
|           border: 1px solid var(--divider-color); | ||||
|           border-radius: 4px; | ||||
|           margin-top: 4px; | ||||
|         } | ||||
|         mwc-button { | ||||
|           margin-left: 8px; | ||||
|         .action { | ||||
|           margin-top: 24px; | ||||
|           width: 100%; | ||||
|           display: flex; | ||||
|           justify-content: flex-end; | ||||
|         } | ||||
|         mwc-icon-button { | ||||
|         ha-icon-button { | ||||
|           color: var(--error-color); | ||||
|           margin: -10px; | ||||
|         } | ||||
|         mwc-list-item { | ||||
|           cursor: default; | ||||
|         } | ||||
|         mwc-list-item span[slot="secondary"] { | ||||
|           color: var(--secondary-text-color); | ||||
|           margin-right: -10px; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import "@material/mwc-icon-button/mwc-icon-button"; | ||||
| import { mdiDelete } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import type { PaperInputElement } from "@polymer/paper-input/paper-input"; | ||||
| @@ -9,10 +8,11 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { caseInsensitiveStringCompare } from "../../../../src/common/string/compare"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-circular-progress"; | ||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-icon-button"; | ||||
| import { | ||||
|   fetchHassioAddonsInfo, | ||||
|   HassioAddonRepository, | ||||
| @@ -57,7 +57,7 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|   private _filteredRepositories = memoizeOne((repos: HassioAddonRepository[]) => | ||||
|     repos | ||||
|       .filter((repo) => repo.slug !== "core" && repo.slug !== "local") | ||||
|       .sort((a, b) => (a.name < b.name ? -1 : 1)) | ||||
|       .sort((a, b) => caseInsensitiveStringCompare(a.name, b.name)) | ||||
|   ); | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
| @@ -89,15 +89,14 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|                       <div secondary>${repo.maintainer}</div> | ||||
|                       <div secondary>${repo.url}</div> | ||||
|                     </paper-item-body> | ||||
|                     <mwc-icon-button | ||||
|                     <ha-icon-button | ||||
|                       .slug=${repo.slug} | ||||
|                       .title=${this._dialogParams!.supervisor.localize( | ||||
|                       .label=${this._dialogParams!.supervisor.localize( | ||||
|                         "dialog.repositories.remove" | ||||
|                       )} | ||||
|                       .path=${mdiDelete} | ||||
|                       @click=${this._removeRepository} | ||||
|                     > | ||||
|                       <ha-svg-icon .path=${mdiDelete}></ha-svg-icon> | ||||
|                     </mwc-icon-button> | ||||
|                     ></ha-icon-button> | ||||
|                   </paper-item> | ||||
|                 ` | ||||
|               ) | ||||
|   | ||||
| @@ -1,204 +0,0 @@ | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-circular-progress"; | ||||
| import "../../../../src/components/ha-dialog"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-switch"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
|   ignoreSupervisorError, | ||||
| } from "../../../../src/data/hassio/common"; | ||||
| import { createHassioPartialBackup } from "../../../../src/data/hassio/backup"; | ||||
| import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import { SupervisorDialogSupervisorUpdateParams } from "./show-dialog-update"; | ||||
|  | ||||
| @customElement("dialog-supervisor-update") | ||||
| class DialogSupervisorUpdate extends LitElement { | ||||
|   public hass!: HomeAssistant; | ||||
|  | ||||
|   @state() private _opened = false; | ||||
|  | ||||
|   @state() private _createBackup = true; | ||||
|  | ||||
|   @state() private _action: "backup" | "update" | null = null; | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
|   @state() | ||||
|   private _dialogParams?: SupervisorDialogSupervisorUpdateParams; | ||||
|  | ||||
|   public async showDialog( | ||||
|     params: SupervisorDialogSupervisorUpdateParams | ||||
|   ): Promise<void> { | ||||
|     this._opened = true; | ||||
|     this._dialogParams = params; | ||||
|     await this.updateComplete; | ||||
|   } | ||||
|  | ||||
|   public closeDialog(): void { | ||||
|     this._action = null; | ||||
|     this._createBackup = true; | ||||
|     this._error = undefined; | ||||
|     this._dialogParams = undefined; | ||||
|     fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||
|   } | ||||
|  | ||||
|   public focus(): void { | ||||
|     this.updateComplete.then(() => | ||||
|       ( | ||||
|         this.shadowRoot?.querySelector("[dialogInitialFocus]") as HTMLElement | ||||
|       )?.focus() | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this._dialogParams) { | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       <ha-dialog .open=${this._opened} scrimClickAction escapeKeyAction> | ||||
|         ${this._action === null | ||||
|           ? html`<slot name="heading"> | ||||
|                 <h2 id="title" class="header_title"> | ||||
|                   ${this._dialogParams.supervisor.localize( | ||||
|                     "confirm.update.title", | ||||
|                     "name", | ||||
|                     this._dialogParams.name | ||||
|                   )} | ||||
|                 </h2> | ||||
|               </slot> | ||||
|               <div> | ||||
|                 ${this._dialogParams.supervisor.localize( | ||||
|                   "confirm.update.text", | ||||
|                   "name", | ||||
|                   this._dialogParams.name, | ||||
|                   "version", | ||||
|                   this._dialogParams.version | ||||
|                 )} | ||||
|               </div> | ||||
|  | ||||
|               <ha-settings-row> | ||||
|                 <span slot="heading"> | ||||
|                   ${this._dialogParams.supervisor.localize( | ||||
|                     "dialog.update.backup" | ||||
|                   )} | ||||
|                 </span> | ||||
|                 <span slot="description"> | ||||
|                   ${this._dialogParams.supervisor.localize( | ||||
|                     "dialog.update.create_backup", | ||||
|                     "name", | ||||
|                     this._dialogParams.name | ||||
|                   )} | ||||
|                 </span> | ||||
|                 <ha-switch | ||||
|                   .checked=${this._createBackup} | ||||
|                   haptic | ||||
|                   @click=${this._toggleBackup} | ||||
|                 > | ||||
|                 </ha-switch> | ||||
|               </ha-settings-row> | ||||
|               <mwc-button @click=${this.closeDialog} slot="secondaryAction"> | ||||
|                 ${this._dialogParams.supervisor.localize("common.cancel")} | ||||
|               </mwc-button> | ||||
|               <mwc-button | ||||
|                 .disabled=${this._error !== undefined} | ||||
|                 @click=${this._update} | ||||
|                 slot="primaryAction" | ||||
|               > | ||||
|                 ${this._dialogParams.supervisor.localize("common.update")} | ||||
|               </mwc-button>` | ||||
|           : html`<ha-circular-progress alt="Updating" size="large" active> | ||||
|               </ha-circular-progress> | ||||
|               <p class="progress-text"> | ||||
|                 ${this._action === "update" | ||||
|                   ? this._dialogParams.supervisor.localize( | ||||
|                       "dialog.update.updating", | ||||
|                       "name", | ||||
|                       this._dialogParams.name, | ||||
|                       "version", | ||||
|                       this._dialogParams.version | ||||
|                     ) | ||||
|                   : this._dialogParams.supervisor.localize( | ||||
|                       "dialog.update.creating_backup", | ||||
|                       "name", | ||||
|                       this._dialogParams.name | ||||
|                     )} | ||||
|               </p>`} | ||||
|         ${this._error | ||||
|           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||
|           : ""} | ||||
|       </ha-dialog> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _toggleBackup() { | ||||
|     this._createBackup = !this._createBackup; | ||||
|   } | ||||
|  | ||||
|   private async _update() { | ||||
|     if (this._createBackup) { | ||||
|       this._action = "backup"; | ||||
|       try { | ||||
|         await createHassioPartialBackup( | ||||
|           this.hass, | ||||
|           this._dialogParams!.backupParams | ||||
|         ); | ||||
|       } catch (err: any) { | ||||
|         this._error = extractApiErrorMessage(err); | ||||
|         this._action = null; | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     this._action = "update"; | ||||
|     try { | ||||
|       await this._dialogParams!.updateHandler!(); | ||||
|     } catch (err: any) { | ||||
|       if (this.hass.connection.connected && !ignoreSupervisorError(err)) { | ||||
|         this._error = extractApiErrorMessage(err); | ||||
|         this._action = null; | ||||
|       } | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     this.closeDialog(); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return [ | ||||
|       haStyle, | ||||
|       haStyleDialog, | ||||
|       css` | ||||
|         .form { | ||||
|           color: var(--primary-text-color); | ||||
|         } | ||||
|  | ||||
|         ha-settings-row { | ||||
|           margin-top: 32px; | ||||
|           padding: 0; | ||||
|         } | ||||
|  | ||||
|         ha-circular-progress { | ||||
|           display: block; | ||||
|           margin: 32px; | ||||
|           text-align: center; | ||||
|         } | ||||
|  | ||||
|         .progress-text { | ||||
|           text-align: center; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "dialog-supervisor-update": DialogSupervisorUpdate; | ||||
|   } | ||||
| } | ||||
| @@ -1,21 +0,0 @@ | ||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||
| import { Supervisor } from "../../../../src/data/supervisor/supervisor"; | ||||
|  | ||||
| export interface SupervisorDialogSupervisorUpdateParams { | ||||
|   supervisor: Supervisor; | ||||
|   name: string; | ||||
|   version: string; | ||||
|   backupParams: any; | ||||
|   updateHandler: () => Promise<void>; | ||||
| } | ||||
|  | ||||
| export const showDialogSupervisorUpdate = ( | ||||
|   element: HTMLElement, | ||||
|   dialogParams: SupervisorDialogSupervisorUpdateParams | ||||
| ): void => { | ||||
|   fireEvent(element, "show-dialog", { | ||||
|     dialogTag: "dialog-supervisor-update", | ||||
|     dialogImport: () => import("./dialog-supervisor-update"), | ||||
|     dialogParams, | ||||
|   }); | ||||
| }; | ||||
| @@ -10,7 +10,7 @@ import { HassioPanelInfo } from "../../src/data/hassio/supervisor"; | ||||
| import { Supervisor } from "../../src/data/supervisor/supervisor"; | ||||
| import { makeDialogManager } from "../../src/dialogs/make-dialog-manager"; | ||||
| import "../../src/layouts/hass-loading-screen"; | ||||
| import { HomeAssistant, Route } from "../../src/types"; | ||||
| import { HomeAssistant } from "../../src/types"; | ||||
| import "./hassio-router"; | ||||
| import { SupervisorBaseElement } from "./supervisor-base-element"; | ||||
|  | ||||
| @@ -24,8 +24,6 @@ export class HassioMain extends SupervisorBaseElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public narrow!: boolean; | ||||
|  | ||||
|   @property({ attribute: false }) public route?: Route; | ||||
|  | ||||
|   protected firstUpdated(changedProps: PropertyValues) { | ||||
|     super.firstUpdated(changedProps); | ||||
|  | ||||
| @@ -113,12 +111,6 @@ export class HassioMain extends SupervisorBaseElement { | ||||
|           : this.hass.themes.default_theme); | ||||
|  | ||||
|       themeSettings = this.hass.selectedTheme; | ||||
|       if (themeSettings?.dark === undefined) { | ||||
|         themeSettings = { | ||||
|           ...this.hass.selectedTheme, | ||||
|           dark: this.hass.themes.darkMode, | ||||
|         }; | ||||
|       } | ||||
|     } else { | ||||
|       themeName = | ||||
|         (this.hass.selectedTheme as unknown as string) || | ||||
|   | ||||
| @@ -34,6 +34,9 @@ const REDIRECTS: Redirects = { | ||||
|   supervisor_store: { | ||||
|     redirect: "/hassio/store", | ||||
|   }, | ||||
|   supervisor_addons: { | ||||
|     redirect: "/hassio/dashboard", | ||||
|   }, | ||||
|   supervisor_addon: { | ||||
|     redirect: "/hassio/addon", | ||||
|     params: { | ||||
|   | ||||
| @@ -35,6 +35,10 @@ class HassioRouter extends HassRouterPage { | ||||
|       backups: "dashboard", | ||||
|       store: "dashboard", | ||||
|       system: "dashboard", | ||||
|       "update-available": { | ||||
|         tag: "update-available-dashboard", | ||||
|         load: () => import("./update-available/update-available-dashboard"), | ||||
|       }, | ||||
|       addon: { | ||||
|         tag: "hassio-addon-dashboard", | ||||
|         load: () => import("./addon-view/hassio-addon-dashboard"), | ||||
|   | ||||
| @@ -1,16 +1,22 @@ | ||||
| import { mdiBackupRestore, mdiCogs, mdiStore, mdiViewDashboard } from "@mdi/js"; | ||||
| import { | ||||
|   mdiBackupRestore, | ||||
|   mdiCogs, | ||||
|   mdiPuzzle, | ||||
|   mdiViewDashboard, | ||||
| } from "@mdi/js"; | ||||
| import { atLeastVersion } from "../../src/common/config/version"; | ||||
| import type { PageNavigation } from "../../src/layouts/hass-tabs-subpage"; | ||||
| import { HomeAssistant } from "../../src/types"; | ||||
|  | ||||
| export const supervisorTabs: PageNavigation[] = [ | ||||
| export const supervisorTabs = (hass: HomeAssistant): PageNavigation[] => [ | ||||
|   { | ||||
|     translationKey: "panel.dashboard", | ||||
|     translationKey: atLeastVersion(hass.config.version, 2021, 12) | ||||
|       ? "panel.addons" | ||||
|       : "panel.dashboard", | ||||
|     path: `/hassio/dashboard`, | ||||
|     iconPath: mdiViewDashboard, | ||||
|   }, | ||||
|   { | ||||
|     translationKey: "panel.store", | ||||
|     path: `/hassio/store`, | ||||
|     iconPath: mdiStore, | ||||
|     iconPath: atLeastVersion(hass.config.version, 2021, 12) | ||||
|       ? mdiPuzzle | ||||
|       : mdiViewDashboard, | ||||
|   }, | ||||
|   { | ||||
|     translationKey: "panel.backups", | ||||
|   | ||||
| @@ -12,6 +12,7 @@ import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { navigate } from "../../../src/common/navigate"; | ||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||
| import { nextRender } from "../../../src/common/util/render-status"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import { | ||||
|   fetchHassioAddonInfo, | ||||
|   HassioAddonDetails, | ||||
| @@ -72,12 +73,11 @@ class HassioIngressView extends LitElement { | ||||
|  | ||||
|     return html`${this.narrow || this.hass.dockedSidebar === "always_hidden" | ||||
|       ? html`<div class="header"> | ||||
|             <mwc-icon-button | ||||
|               aria-label=${this.hass.localize("ui.sidebar.sidebar_toggle")} | ||||
|             <ha-icon-button | ||||
|               .label=${this.hass.localize("ui.sidebar.sidebar_toggle")} | ||||
|               .path=${mdiMenu} | ||||
|               @click=${this._toggleMenu} | ||||
|             > | ||||
|               <ha-svg-icon .path=${mdiMenu}></ha-svg-icon> | ||||
|             </mwc-icon-button> | ||||
|             ></ha-icon-button> | ||||
|             <div class="main-title">${this._addon.name}</div> | ||||
|           </div> | ||||
|           ${iframe}` | ||||
| @@ -241,7 +241,7 @@ class HassioIngressView extends LitElement { | ||||
|         flex-grow: 1; | ||||
|       } | ||||
|  | ||||
|       mwc-icon-button { | ||||
|       ha-icon-button { | ||||
|         pointer-events: auto; | ||||
|       } | ||||
|  | ||||
|   | ||||
| @@ -25,7 +25,7 @@ import { | ||||
| } from "../../src/data/supervisor/supervisor"; | ||||
| import { ProvideHassLitMixin } from "../../src/mixins/provide-hass-lit-mixin"; | ||||
| import { urlSyncMixin } from "../../src/state/url-sync-mixin"; | ||||
| import { HomeAssistant } from "../../src/types"; | ||||
| import { HomeAssistant, Route } from "../../src/types"; | ||||
| import { getTranslation } from "../../src/util/common-translation"; | ||||
|  | ||||
| declare global { | ||||
| @@ -38,6 +38,8 @@ declare global { | ||||
| export class SupervisorBaseElement extends urlSyncMixin( | ||||
|   ProvideHassLitMixin(LitElement) | ||||
| ) { | ||||
|   @property({ attribute: false }) public route?: Route; | ||||
|  | ||||
|   @property({ attribute: false }) public supervisor: Partial<Supervisor> = { | ||||
|     localize: () => "", | ||||
|   }; | ||||
| @@ -108,7 +110,9 @@ export class SupervisorBaseElement extends urlSyncMixin( | ||||
|       this._language = this.hass.language; | ||||
|     } | ||||
|     this._initializeLocalize(); | ||||
|     this._initSupervisor(); | ||||
|     if (this.route?.prefix === "/hassio") { | ||||
|       this._initSupervisor(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _initializeLocalize() { | ||||
|   | ||||
| @@ -2,7 +2,7 @@ import "@material/mwc-button"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import { atLeastVersion } from "../../../src/common/config/version"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-card"; | ||||
| @@ -12,7 +12,7 @@ import { | ||||
|   fetchHassioStats, | ||||
|   HassioStats, | ||||
| } from "../../../src/data/hassio/common"; | ||||
| import { restartCore, updateCore } from "../../../src/data/supervisor/core"; | ||||
| import { restartCore } from "../../../src/data/supervisor/core"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { | ||||
|   showAlertDialog, | ||||
| @@ -22,7 +22,6 @@ import { haStyle } from "../../../src/resources/styles"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { bytesToString } from "../../../src/util/bytes-to-string"; | ||||
| import "../components/supervisor-metric"; | ||||
| import { showDialogSupervisorUpdate } from "../dialogs/update/show-dialog-update"; | ||||
| import { hassioStyle } from "../resources/hassio-style"; | ||||
|  | ||||
| @customElement("hassio-core-info") | ||||
| @@ -67,14 +66,15 @@ class HassioCoreInfo extends LitElement { | ||||
|               <span slot="description"> | ||||
|                 core-${this.supervisor.core.version_latest} | ||||
|               </span> | ||||
|               ${this.supervisor.core.update_available | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.core.update_available | ||||
|                 ? html` | ||||
|                     <ha-progress-button | ||||
|                       .title=${this.supervisor.localize("common.update")} | ||||
|                       @click=${this._coreUpdate} | ||||
|                     > | ||||
|                       ${this.supervisor.localize("common.update")} | ||||
|                     </ha-progress-button> | ||||
|                     <a href="/hassio/update-available/core"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -160,27 +160,6 @@ class HassioCoreInfo extends LitElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _coreUpdate(): Promise<void> { | ||||
|     showDialogSupervisorUpdate(this, { | ||||
|       supervisor: this.supervisor, | ||||
|       name: "Home Assistant Core", | ||||
|       version: this.supervisor.core.version_latest, | ||||
|       backupParams: { | ||||
|         name: `core_${this.supervisor.core.version}`, | ||||
|         folders: ["homeassistant"], | ||||
|         homeassistant: true, | ||||
|       }, | ||||
|       updateHandler: async () => this._updateCore(), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   private async _updateCore(): Promise<void> { | ||||
|     await updateCore(this.hass); | ||||
|     fireEvent(this, "supervisor-collection-refresh", { | ||||
|       collection: "core", | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return [ | ||||
|       haStyle, | ||||
| @@ -239,6 +218,9 @@ class HassioCoreInfo extends LitElement { | ||||
|         mwc-list-item ha-svg-icon { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         a { | ||||
|           text-decoration: none; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
| @@ -20,7 +21,6 @@ import { | ||||
|   configSyncOS, | ||||
|   rebootHost, | ||||
|   shutdownHost, | ||||
|   updateOS, | ||||
| } from "../../../src/data/hassio/host"; | ||||
| import { | ||||
|   fetchNetworkInfo, | ||||
| @@ -105,11 +105,15 @@ class HassioHostInfo extends LitElement { | ||||
|               <span slot="description"> | ||||
|                 ${this.supervisor.host.operating_system} | ||||
|               </span> | ||||
|               ${this.supervisor.os.update_available | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.os.update_available | ||||
|                 ? html` | ||||
|                     <ha-progress-button @click=${this._osUpdate}> | ||||
|                       ${this.supervisor.localize("commmon.update")} | ||||
|                     </ha-progress-button> | ||||
|                     <a href="/hassio/update-available/os"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -181,9 +185,11 @@ class HassioHostInfo extends LitElement { | ||||
|             : ""} | ||||
|  | ||||
|           <ha-button-menu corner="BOTTOM_START"> | ||||
|             <mwc-icon-button slot="trigger"> | ||||
|               <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon> | ||||
|             </mwc-icon-button> | ||||
|             <ha-icon-button | ||||
|               .label=${this.hass.localize("common.menu")} | ||||
|               .path=${mdiDotsVertical} | ||||
|               slot="trigger" | ||||
|             ></ha-icon-button> | ||||
|             <mwc-list-item | ||||
|               .action=${"hardware"} | ||||
|               @click=${this._handleMenuAction} | ||||
| @@ -330,50 +336,6 @@ class HassioHostInfo extends LitElement { | ||||
|     button.progress = false; | ||||
|   } | ||||
|  | ||||
|   private async _osUpdate(ev: CustomEvent): Promise<void> { | ||||
|     const button = ev.currentTarget as any; | ||||
|     button.progress = true; | ||||
|  | ||||
|     const confirmed = await showConfirmationDialog(this, { | ||||
|       title: this.supervisor.localize( | ||||
|         "confirm.update.title", | ||||
|         "name", | ||||
|         "Home Assistant Operating System" | ||||
|       ), | ||||
|       text: this.supervisor.localize( | ||||
|         "confirm.update.text", | ||||
|         "name", | ||||
|         "Home Assistant Operating System", | ||||
|         "version", | ||||
|         this.supervisor.os.version_latest | ||||
|       ), | ||||
|       confirmText: this.supervisor.localize("common.update"), | ||||
|       dismissText: "no", | ||||
|     }); | ||||
|  | ||||
|     if (!confirmed) { | ||||
|       button.progress = false; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       await updateOS(this.hass); | ||||
|       fireEvent(this, "supervisor-collection-refresh", { collection: "os" }); | ||||
|     } catch (err: any) { | ||||
|       if (this.hass.connection.connected) { | ||||
|         showAlertDialog(this, { | ||||
|           title: this.supervisor.localize( | ||||
|             "common.failed_to_update_name", | ||||
|             "name", | ||||
|             "Home Assistant Operating System" | ||||
|           ), | ||||
|           text: extractApiErrorMessage(err), | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|     button.progress = false; | ||||
|   } | ||||
|  | ||||
|   private async _changeNetworkClicked(): Promise<void> { | ||||
|     showNetworkDialog(this, { | ||||
|       supervisor: this.supervisor, | ||||
| @@ -491,6 +453,9 @@ class HassioHostInfo extends LitElement { | ||||
|         mwc-list-item ha-svg-icon { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         a { | ||||
|           text-decoration: none; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
|   | ||||
| @@ -17,7 +17,6 @@ import { | ||||
|   restartSupervisor, | ||||
|   setSupervisorOption, | ||||
|   SupervisorOptions, | ||||
|   updateSupervisor, | ||||
| } from "../../../src/data/hassio/supervisor"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { | ||||
| @@ -31,29 +30,9 @@ import { documentationUrl } from "../../../src/util/documentation-url"; | ||||
| import "../components/supervisor-metric"; | ||||
| import { hassioStyle } from "../resources/hassio-style"; | ||||
|  | ||||
| const UNSUPPORTED_REASON_URL = { | ||||
|   apparmor: "/more-info/unsupported/apparmor", | ||||
|   container: "/more-info/unsupported/container", | ||||
|   content_trust: "/more-info/unsupported/content_trust", | ||||
|   dbus: "/more-info/unsupported/dbus", | ||||
|   docker_configuration: "/more-info/unsupported/docker_configuration", | ||||
|   docker_version: "/more-info/unsupported/docker_version", | ||||
|   job_conditions: "/more-info/unsupported/job_conditions", | ||||
|   lxc: "/more-info/unsupported/lxc", | ||||
|   network_manager: "/more-info/unsupported/network_manager", | ||||
|   os_agent: "/more-info/unsupported/os_agent", | ||||
|   os: "/more-info/unsupported/os", | ||||
|   privileged: "/more-info/unsupported/privileged", | ||||
|   source_mods: "/more-info/unsupported/source_mods", | ||||
|   systemd: "/more-info/unsupported/systemd", | ||||
| }; | ||||
|  | ||||
| const UNSUPPORTED_REASON_URL = {}; | ||||
| const UNHEALTHY_REASON_URL = { | ||||
|   privileged: "/more-info/unsupported/privileged", | ||||
|   supervisor: "/more-info/unhealthy/supervisor", | ||||
|   setup: "/more-info/unhealthy/setup", | ||||
|   docker: "/more-info/unhealthy/docker", | ||||
|   untrusted: "/more-info/unhealthy/untrusted", | ||||
| }; | ||||
|  | ||||
| @customElement("hassio-supervisor-info") | ||||
| @@ -97,16 +76,15 @@ class HassioSupervisorInfo extends LitElement { | ||||
|               <span slot="description"> | ||||
|                 supervisor-${this.supervisor.supervisor.version_latest} | ||||
|               </span> | ||||
|               ${this.supervisor.supervisor.update_available | ||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||
|               this.supervisor.supervisor.update_available | ||||
|                 ? html` | ||||
|                     <ha-progress-button | ||||
|                       .title=${this.supervisor.localize( | ||||
|                         "system.supervisor.update_supervisor" | ||||
|                       )} | ||||
|                       @click=${this._supervisorUpdate} | ||||
|                     > | ||||
|                       ${this.supervisor.localize("common.update")} | ||||
|                     </ha-progress-button> | ||||
|                     <a href="/hassio/update-available/supervisor"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize("common.show")} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </ha-settings-row> | ||||
| @@ -173,24 +151,28 @@ class HassioSupervisorInfo extends LitElement { | ||||
|                     ></ha-switch> | ||||
|                   </ha-settings-row>` | ||||
|                 : "" | ||||
|               : html`<ha-alert | ||||
|                   alert-type="warning" | ||||
|                   .actionText=${this.supervisor.localize("common.learn_more")} | ||||
|                   @alert-action-clicked=${this._unsupportedDialog} | ||||
|                 > | ||||
|               : html`<ha-alert alert-type="warning"> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "system.supervisor.unsupported_title" | ||||
|                   )} | ||||
|                   <mwc-button | ||||
|                     slot="action" | ||||
|                     .label=${this.supervisor.localize("common.learn_more")} | ||||
|                     @click=${this._unsupportedDialog} | ||||
|                   > | ||||
|                   </mwc-button> | ||||
|                 </ha-alert>`} | ||||
|             ${!this.supervisor.supervisor.healthy | ||||
|               ? html`<ha-alert | ||||
|                   alert-type="error" | ||||
|                   .actionText=${this.supervisor.localize("common.learn_more")} | ||||
|                   @alert-action-clicked=${this._unhealthyDialog} | ||||
|                 > | ||||
|               ? html`<ha-alert alert-type="error"> | ||||
|                   ${this.supervisor.localize( | ||||
|                     "system.supervisor.unhealthy_title" | ||||
|                   )} | ||||
|                   <mwc-button | ||||
|                     slot="action" | ||||
|                     .label=${this.supervisor.localize("common.learn_more")} | ||||
|                     @click=${this._unhealthyDialog} | ||||
|                   > | ||||
|                   </mwc-button> | ||||
|                 </ha-alert>` | ||||
|               : ""} | ||||
|           </div> | ||||
| @@ -357,51 +339,6 @@ class HassioSupervisorInfo extends LitElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _supervisorUpdate(ev: CustomEvent): Promise<void> { | ||||
|     const button = ev.currentTarget as any; | ||||
|     button.progress = true; | ||||
|  | ||||
|     const confirmed = await showConfirmationDialog(this, { | ||||
|       title: this.supervisor.localize( | ||||
|         "confirm.update.title", | ||||
|         "name", | ||||
|         "Supervisor" | ||||
|       ), | ||||
|       text: this.supervisor.localize( | ||||
|         "confirm.update.text", | ||||
|         "name", | ||||
|         "Supervisor", | ||||
|         "version", | ||||
|         this.supervisor.supervisor.version_latest | ||||
|       ), | ||||
|       confirmText: this.supervisor.localize("common.update"), | ||||
|       dismissText: this.supervisor.localize("common.cancel"), | ||||
|     }); | ||||
|  | ||||
|     if (!confirmed) { | ||||
|       button.progress = false; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       await updateSupervisor(this.hass); | ||||
|       fireEvent(this, "supervisor-collection-refresh", { | ||||
|         collection: "supervisor", | ||||
|       }); | ||||
|     } catch (err: any) { | ||||
|       showAlertDialog(this, { | ||||
|         title: this.supervisor.localize( | ||||
|           "common.failed_to_update_name", | ||||
|           "name", | ||||
|           "Supervisor" | ||||
|         ), | ||||
|         text: extractApiErrorMessage(err), | ||||
|       }); | ||||
|     } finally { | ||||
|       button.progress = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _diagnosticsInformationDialog(): Promise<void> { | ||||
|     await showAlertDialog(this, { | ||||
|       title: this.supervisor.localize( | ||||
| @@ -425,20 +362,19 @@ class HassioSupervisorInfo extends LitElement { | ||||
|           ${this.supervisor.resolution.unsupported.map( | ||||
|             (reason) => html` | ||||
|               <li> | ||||
|                 ${UNSUPPORTED_REASON_URL[reason] | ||||
|                   ? html`<a | ||||
|                       href=${documentationUrl( | ||||
|                         this.hass, | ||||
|                         UNSUPPORTED_REASON_URL[reason] | ||||
|                       )} | ||||
|                       target="_blank" | ||||
|                       rel="noreferrer" | ||||
|                     > | ||||
|                       ${this.supervisor.localize( | ||||
|                         `system.supervisor.unsupported_reason.${reason}` | ||||
|                       ) || reason} | ||||
|                     </a>` | ||||
|                   : reason} | ||||
|                 <a | ||||
|                   href=${documentationUrl( | ||||
|                     this.hass, | ||||
|                     UNSUPPORTED_REASON_URL[reason] || | ||||
|                       `/more-info/unsupported/${reason}` | ||||
|                   )} | ||||
|                   target="_blank" | ||||
|                   rel="noreferrer" | ||||
|                 > | ||||
|                   ${this.supervisor.localize( | ||||
|                     `system.supervisor.unsupported_reason.${reason}` | ||||
|                   ) || reason} | ||||
|                 </a> | ||||
|               </li> | ||||
|             ` | ||||
|           )} | ||||
| @@ -456,20 +392,19 @@ class HassioSupervisorInfo extends LitElement { | ||||
|           ${this.supervisor.resolution.unhealthy.map( | ||||
|             (reason) => html` | ||||
|               <li> | ||||
|                 ${UNHEALTHY_REASON_URL[reason] | ||||
|                   ? html`<a | ||||
|                       href=${documentationUrl( | ||||
|                         this.hass, | ||||
|                         UNHEALTHY_REASON_URL[reason] | ||||
|                       )} | ||||
|                       target="_blank" | ||||
|                       rel="noreferrer" | ||||
|                     > | ||||
|                       ${this.supervisor.localize( | ||||
|                         `system.supervisor.unhealthy_reason.${reason}` | ||||
|                       ) || reason} | ||||
|                     </a>` | ||||
|                   : reason} | ||||
|                 <a | ||||
|                   href=${documentationUrl( | ||||
|                     this.hass, | ||||
|                     UNHEALTHY_REASON_URL[reason] || | ||||
|                       `/more-info/unhealthy/${reason}` | ||||
|                   )} | ||||
|                   target="_blank" | ||||
|                   rel="noreferrer" | ||||
|                 > | ||||
|                   ${this.supervisor.localize( | ||||
|                     `system.supervisor.unhealthy_reason.${reason}` | ||||
|                   ) || reason} | ||||
|                 </a> | ||||
|               </li> | ||||
|             ` | ||||
|           )} | ||||
| @@ -535,6 +470,12 @@ class HassioSupervisorInfo extends LitElement { | ||||
|           white-space: normal; | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         ha-alert mwc-button { | ||||
|           --mdc-theme-primary: var(--primary-text-color); | ||||
|         } | ||||
|         a { | ||||
|           text-decoration: none; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
|   | ||||
| @@ -28,7 +28,7 @@ class HassioSystem extends LitElement { | ||||
|         .localizeFunc=${this.supervisor.localize} | ||||
|         .narrow=${this.narrow} | ||||
|         .route=${this.route} | ||||
|         .tabs=${supervisorTabs} | ||||
|         .tabs=${supervisorTabs(this.hass)} | ||||
|         main-page | ||||
|         supervisor | ||||
|       > | ||||
|   | ||||
							
								
								
									
										401
									
								
								hassio/src/update-available/update-available-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										401
									
								
								hassio/src/update-available/update-available-card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,401 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/common/search/search-input"; | ||||
| import "../../../src/components/buttons/ha-progress-button"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-button-menu"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-checkbox"; | ||||
| import "../../../src/components/ha-expansion-panel"; | ||||
| import "../../../src/components/ha-formfield"; | ||||
| import "../../../src/components/ha-icon-button"; | ||||
| import "../../../src/components/ha-markdown"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import "../../../src/components/ha-switch"; | ||||
| import { | ||||
|   fetchHassioAddonChangelog, | ||||
|   fetchHassioAddonInfo, | ||||
|   HassioAddonDetails, | ||||
|   updateHassioAddon, | ||||
| } from "../../../src/data/hassio/addon"; | ||||
| import { | ||||
|   createHassioPartialBackup, | ||||
|   HassioPartialBackupCreateParams, | ||||
| } from "../../../src/data/hassio/backup"; | ||||
| import { | ||||
|   extractApiErrorMessage, | ||||
|   ignoreSupervisorError, | ||||
| } from "../../../src/data/hassio/common"; | ||||
| import { updateOS } from "../../../src/data/hassio/host"; | ||||
| import { updateSupervisor } from "../../../src/data/hassio/supervisor"; | ||||
| import { updateCore } from "../../../src/data/supervisor/core"; | ||||
| import { StoreAddon } from "../../../src/data/supervisor/store"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import { showAlertDialog } from "../../../src/dialogs/generic/show-dialog-box"; | ||||
| import "../../../src/layouts/hass-loading-screen"; | ||||
| import "../../../src/layouts/hass-subpage"; | ||||
| import "../../../src/layouts/hass-tabs-subpage"; | ||||
| import { SUPERVISOR_UPDATE_NAMES } from "../../../src/panels/config/dashboard/ha-config-updates"; | ||||
| import { HomeAssistant, Route } from "../../../src/types"; | ||||
| import { documentationUrl } from "../../../src/util/documentation-url"; | ||||
| import { addonArchIsSupported, extractChangelog } from "../util/addon"; | ||||
|  | ||||
| declare global { | ||||
|   interface HASSDomEvents { | ||||
|     "update-complete": undefined; | ||||
|   } | ||||
| } | ||||
|  | ||||
| type updateType = "os" | "supervisor" | "core" | "addon"; | ||||
|  | ||||
| const changelogUrl = ( | ||||
|   hass: HomeAssistant, | ||||
|   entry: updateType, | ||||
|   version: string | ||||
| ): string | undefined => { | ||||
|   if (entry === "addon") { | ||||
|     return undefined; | ||||
|   } | ||||
|   if (entry === "core") { | ||||
|     return version?.includes("dev") | ||||
|       ? "https://github.com/home-assistant/core/commits/dev" | ||||
|       : documentationUrl(hass, "/latest-release-notes/"); | ||||
|   } | ||||
|   if (entry === "os") { | ||||
|     return version?.includes("dev") | ||||
|       ? "https://github.com/home-assistant/operating-system/commits/dev" | ||||
|       : `https://github.com/home-assistant/operating-system/releases/tag/${version}`; | ||||
|   } | ||||
|   if (entry === "supervisor") { | ||||
|     return version?.includes("dev") | ||||
|       ? "https://github.com/home-assistant/supervisor/commits/main" | ||||
|       : `https://github.com/home-assistant/supervisor/releases/tag/${version}`; | ||||
|   } | ||||
|   return undefined; | ||||
| }; | ||||
|  | ||||
| @customElement("update-available-card") | ||||
| class UpdateAvailableCard extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public supervisor!: Supervisor; | ||||
|  | ||||
|   @property({ attribute: false }) public route!: Route; | ||||
|  | ||||
|   @property({ type: Boolean }) public narrow!: boolean; | ||||
|  | ||||
|   @property({ attribute: false }) public addonSlug?: string; | ||||
|  | ||||
|   @state() private _updateType?: updateType; | ||||
|  | ||||
|   @state() private _changelogContent?: string; | ||||
|  | ||||
|   @state() private _addonInfo?: HassioAddonDetails; | ||||
|  | ||||
|   @state() private _action: "backup" | "update" | null = null; | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
|   private _addonStoreInfo = memoizeOne( | ||||
|     (slug: string, storeAddons: StoreAddon[]) => | ||||
|       storeAddons.find((addon) => addon.slug === slug) | ||||
|   ); | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if ( | ||||
|       !this._updateType || | ||||
|       (this._updateType === "addon" && !this._addonInfo) | ||||
|     ) { | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     const changelog = changelogUrl(this.hass, this._updateType, this._version); | ||||
|  | ||||
|     return html` | ||||
|       <ha-card | ||||
|         .header=${this.supervisor.localize("update_available.update_name", { | ||||
|           name: this._name, | ||||
|         })} | ||||
|       > | ||||
|         <div class="card-content"> | ||||
|           ${this._error | ||||
|             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||
|             : ""} | ||||
|           ${this._action === null | ||||
|             ? html` | ||||
|                 ${this._changelogContent | ||||
|                   ? html` | ||||
|                       <ha-expansion-panel header="Changelog" outlined> | ||||
|                         <ha-markdown .content=${this._changelogContent}> | ||||
|                         </ha-markdown> | ||||
|                       </ha-expansion-panel> | ||||
|                     ` | ||||
|                   : ""} | ||||
|                 <div class="versions"> | ||||
|                   <p> | ||||
|                     ${this.supervisor.localize("update_available.description", { | ||||
|                       name: this._name, | ||||
|                       version: this._version, | ||||
|                       newest_version: this._version_latest, | ||||
|                     })} | ||||
|                   </p> | ||||
|                 </div> | ||||
|                 ${["core", "addon"].includes(this._updateType) | ||||
|                   ? html` | ||||
|                       <ha-formfield | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "update_available.create_backup" | ||||
|                         )} | ||||
|                       > | ||||
|                         <ha-checkbox checked></ha-checkbox> | ||||
|                       </ha-formfield> | ||||
|                     ` | ||||
|                   : ""} | ||||
|               ` | ||||
|             : html`<ha-circular-progress alt="Updating" size="large" active> | ||||
|                 </ha-circular-progress> | ||||
|                 <p class="progress-text"> | ||||
|                   ${this._action === "update" | ||||
|                     ? this.supervisor.localize("update_available.updating", { | ||||
|                         name: this._name, | ||||
|                         version: this._version_latest, | ||||
|                       }) | ||||
|                     : this.supervisor.localize( | ||||
|                         "update_available.creating_backup", | ||||
|                         { name: this._name } | ||||
|                       )} | ||||
|                 </p>`} | ||||
|         </div> | ||||
|         ${this._action === null | ||||
|           ? html` | ||||
|               <div class="card-actions"> | ||||
|                 ${changelog | ||||
|                   ? html`<a .href=${changelog} target="_blank" rel="noreferrer"> | ||||
|                       <mwc-button | ||||
|                         .label=${this.supervisor.localize( | ||||
|                           "update_available.open_release_notes" | ||||
|                         )} | ||||
|                       > | ||||
|                       </mwc-button> | ||||
|                     </a>` | ||||
|                   : ""} | ||||
|                 <span></span> | ||||
|                 <ha-progress-button | ||||
|                   .disabled=${!this._version || | ||||
|                   (this._shouldCreateBackup && | ||||
|                     this.supervisor.info.state !== "running")} | ||||
|                   @click=${this._update} | ||||
|                   raised | ||||
|                 > | ||||
|                   ${this.supervisor.localize("common.update")} | ||||
|                 </ha-progress-button> | ||||
|               </div> | ||||
|             ` | ||||
|           : ""} | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProps: PropertyValues) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     const pathPart = this.route?.path.substring(1, this.route.path.length); | ||||
|     const updateType = ["core", "os", "supervisor"].includes(pathPart) | ||||
|       ? pathPart | ||||
|       : "addon"; | ||||
|     this._updateType = updateType as updateType; | ||||
|  | ||||
|     if (updateType === "addon") { | ||||
|       if (!this.addonSlug) { | ||||
|         this.addonSlug = pathPart; | ||||
|       } | ||||
|       this._loadAddonData(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   get _shouldCreateBackup(): boolean { | ||||
|     return this.shadowRoot?.querySelector("ha-checkbox")?.checked || true; | ||||
|   } | ||||
|  | ||||
|   get _version(): string { | ||||
|     return this._updateType | ||||
|       ? this._updateType === "addon" | ||||
|         ? this._addonInfo!.version | ||||
|         : this.supervisor[this._updateType]?.version || "" | ||||
|       : ""; | ||||
|   } | ||||
|  | ||||
|   get _version_latest(): string { | ||||
|     return this._updateType | ||||
|       ? this._updateType === "addon" | ||||
|         ? this._addonInfo!.version_latest | ||||
|         : this.supervisor[this._updateType]?.version_latest || "" | ||||
|       : ""; | ||||
|   } | ||||
|  | ||||
|   get _name(): string { | ||||
|     return this._updateType | ||||
|       ? this._updateType === "addon" | ||||
|         ? this._addonInfo!.name | ||||
|         : SUPERVISOR_UPDATE_NAMES[this._updateType] | ||||
|       : ""; | ||||
|   } | ||||
|  | ||||
|   private async _loadAddonData() { | ||||
|     try { | ||||
|       this._addonInfo = await fetchHassioAddonInfo(this.hass, this.addonSlug!); | ||||
|     } catch (err) { | ||||
|       showAlertDialog(this, { | ||||
|         title: this._updateType, | ||||
|         text: extractApiErrorMessage(err), | ||||
|       }); | ||||
|       return; | ||||
|     } | ||||
|     const addonStoreInfo = | ||||
|       !this._addonInfo.detached && !this._addonInfo.available | ||||
|         ? this._addonStoreInfo( | ||||
|             this._addonInfo.slug, | ||||
|             this.supervisor.store.addons | ||||
|           ) | ||||
|         : undefined; | ||||
|  | ||||
|     if (this._addonInfo.changelog) { | ||||
|       try { | ||||
|         const content = await fetchHassioAddonChangelog( | ||||
|           this.hass, | ||||
|           this.addonSlug! | ||||
|         ); | ||||
|         this._changelogContent = extractChangelog(this._addonInfo, content); | ||||
|       } catch (err) { | ||||
|         this._error = extractApiErrorMessage(err); | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (!this._addonInfo.available && addonStoreInfo) { | ||||
|       if ( | ||||
|         !addonArchIsSupported( | ||||
|           this.supervisor.info.supported_arch, | ||||
|           this._addonInfo.arch | ||||
|         ) | ||||
|       ) { | ||||
|         this._error = this.supervisor.localize( | ||||
|           "addon.dashboard.not_available_arch" | ||||
|         ); | ||||
|       } else { | ||||
|         this._error = this.supervisor.localize( | ||||
|           "addon.dashboard.not_available_version", | ||||
|           { | ||||
|             core_version_installed: this.supervisor.core.version, | ||||
|             core_version_needed: addonStoreInfo.homeassistant, | ||||
|           } | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _update() { | ||||
|     this._error = undefined; | ||||
|     if (this._shouldCreateBackup) { | ||||
|       let backupArgs: HassioPartialBackupCreateParams; | ||||
|       if (this._updateType === "addon") { | ||||
|         backupArgs = { | ||||
|           name: `addon_${this.addonSlug}_${this._version}`, | ||||
|           addons: [this.addonSlug!], | ||||
|           homeassistant: false, | ||||
|         }; | ||||
|       } else { | ||||
|         backupArgs = { | ||||
|           name: `${this._updateType}_${this._version}`, | ||||
|           folders: ["homeassistant"], | ||||
|           homeassistant: true, | ||||
|         }; | ||||
|       } | ||||
|       this._action = "backup"; | ||||
|       try { | ||||
|         await createHassioPartialBackup(this.hass, backupArgs); | ||||
|       } catch (err: any) { | ||||
|         this._error = extractApiErrorMessage(err); | ||||
|         this._action = null; | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     this._action = "update"; | ||||
|     try { | ||||
|       if (this._updateType === "addon") { | ||||
|         await updateHassioAddon(this.hass, this.addonSlug!); | ||||
|       } else if (this._updateType === "core") { | ||||
|         await updateCore(this.hass); | ||||
|       } else if (this._updateType === "os") { | ||||
|         await updateOS(this.hass); | ||||
|       } else if (this._updateType === "supervisor") { | ||||
|         await updateSupervisor(this.hass); | ||||
|       } | ||||
|     } catch (err: any) { | ||||
|       if (this.hass.connection.connected && !ignoreSupervisorError(err)) { | ||||
|         this._error = extractApiErrorMessage(err); | ||||
|         this._action = null; | ||||
|         return; | ||||
|       } | ||||
|     } | ||||
|     fireEvent(this, "update-complete"); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: block; | ||||
|       } | ||||
|       ha-card { | ||||
|         margin: auto; | ||||
|       } | ||||
|       a { | ||||
|         text-decoration: none; | ||||
|         color: var(--primary-text-color); | ||||
|       } | ||||
|       ha-settings-row { | ||||
|         padding: 0; | ||||
|       } | ||||
|       .card-actions { | ||||
|         display: flex; | ||||
|         justify-content: space-between; | ||||
|         border-top: none; | ||||
|         padding: 0 8px 8px; | ||||
|       } | ||||
|  | ||||
|       ha-circular-progress { | ||||
|         display: block; | ||||
|         margin: 32px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .progress-text { | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       ha-markdown { | ||||
|         padding-bottom: 8px; | ||||
|       } | ||||
|       ha-formfield { | ||||
|         cursor: pointer; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "update-available-card": UpdateAvailableCard; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										59
									
								
								hassio/src/update-available/update-available-dashboard.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								hassio/src/update-available/update-available-dashboard.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||
| import "../../../src/layouts/hass-subpage"; | ||||
| import { HomeAssistant, Route } from "../../../src/types"; | ||||
| import "./update-available-card"; | ||||
|  | ||||
| @customElement("update-available-dashboard") | ||||
| class UpdateAvailableDashboard extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public supervisor!: Supervisor; | ||||
|  | ||||
|   @property({ type: Boolean }) public narrow!: boolean; | ||||
|  | ||||
|   @property({ attribute: false }) public route!: Route; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <hass-subpage | ||||
|         .hass=${this.hass} | ||||
|         .narrow=${this.narrow} | ||||
|         .route=${this.route} | ||||
|       > | ||||
|         <update-available-card | ||||
|           .hass=${this.hass} | ||||
|           .supervisor=${this.supervisor} | ||||
|           .route=${this.route} | ||||
|           .narrow=${this.narrow} | ||||
|           @update-complete=${this._updateComplete} | ||||
|         ></update-available-card> | ||||
|       </hass-subpage> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _updateComplete() { | ||||
|     history.back(); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       hass-subpage { | ||||
|         --app-header-background-color: var(--primary-background-color); | ||||
|         --app-header-text-color: var(--sidebar-text-color); | ||||
|       } | ||||
|       update-available-card { | ||||
|         margin: auto; | ||||
|         margin-top: 16px; | ||||
|         max-width: 600px; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "update-available-dashboard": UpdateAvailableDashboard; | ||||
|   } | ||||
| } | ||||
| @@ -1,7 +1,30 @@ | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { HassioAddonDetails } from "../../../src/data/hassio/addon"; | ||||
| import { SupervisorArch } from "../../../src/data/supervisor/supervisor"; | ||||
|  | ||||
| export const addonArchIsSupported = memoizeOne( | ||||
|   (supported_archs: SupervisorArch[], addon_archs: SupervisorArch[]) => | ||||
|     addon_archs.some((arch) => supported_archs.includes(arch)) | ||||
| ); | ||||
|  | ||||
| export const extractChangelog = ( | ||||
|   addon: HassioAddonDetails, | ||||
|   content: string | ||||
| ): string => { | ||||
|   if (content.startsWith("# Changelog")) { | ||||
|     content = content.substr(12, content.length); | ||||
|   } | ||||
|   if ( | ||||
|     content.includes(`# ${addon.version}`) && | ||||
|     content.includes(`# ${addon.version_latest}`) | ||||
|   ) { | ||||
|     const newcontent = content.split(`# ${addon.version}`)[0]; | ||||
|     if (newcontent.includes(`# ${addon.version_latest}`)) { | ||||
|       // Only change the content if the new version still exist | ||||
|       // if the changelog does not have the newests version on top | ||||
|       // this will not be true, and we don't modify the content | ||||
|       content = newcontent; | ||||
|     } | ||||
|   } | ||||
|   return content; | ||||
| }; | ||||
|   | ||||
							
								
								
									
										95
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										95
									
								
								package.json
									
									
									
									
									
								
							| @@ -22,23 +22,23 @@ | ||||
|   "license": "Apache-2.0", | ||||
|   "dependencies": { | ||||
|     "@braintree/sanitize-url": "^5.0.2", | ||||
|     "@codemirror/commands": "^0.19.2", | ||||
|     "@codemirror/gutter": "^0.19.1", | ||||
|     "@codemirror/highlight": "^0.19.2", | ||||
|     "@codemirror/commands": "^0.19.5", | ||||
|     "@codemirror/gutter": "^0.19.4", | ||||
|     "@codemirror/highlight": "^0.19.6", | ||||
|     "@codemirror/history": "^0.19.0", | ||||
|     "@codemirror/legacy-modes": "^0.19.0", | ||||
|     "@codemirror/rectangular-selection": "^0.19.0", | ||||
|     "@codemirror/search": "^0.19.0", | ||||
|     "@codemirror/state": "^0.19.1", | ||||
|     "@codemirror/stream-parser": "^0.19.1", | ||||
|     "@codemirror/text": "^0.19.2", | ||||
|     "@codemirror/view": "^0.19.4", | ||||
|     "@formatjs/intl-datetimeformat": "^4.2.4", | ||||
|     "@formatjs/intl-getcanonicallocales": "^1.7.3", | ||||
|     "@formatjs/intl-locale": "^2.4.38", | ||||
|     "@formatjs/intl-numberformat": "^7.2.4", | ||||
|     "@formatjs/intl-pluralrules": "^4.1.4", | ||||
|     "@formatjs/intl-relativetimeformat": "^9.3.1", | ||||
|     "@codemirror/rectangular-selection": "^0.19.1", | ||||
|     "@codemirror/search": "^0.19.2", | ||||
|     "@codemirror/state": "^0.19.4", | ||||
|     "@codemirror/stream-parser": "^0.19.2", | ||||
|     "@codemirror/text": "^0.19.5", | ||||
|     "@codemirror/view": "^0.19.15", | ||||
|     "@formatjs/intl-datetimeformat": "^4.2.5", | ||||
|     "@formatjs/intl-getcanonicallocales": "^1.8.0", | ||||
|     "@formatjs/intl-locale": "^2.4.40", | ||||
|     "@formatjs/intl-numberformat": "^7.2.5", | ||||
|     "@formatjs/intl-pluralrules": "^4.1.5", | ||||
|     "@formatjs/intl-relativetimeformat": "^9.3.2", | ||||
|     "@formatjs/intl-utils": "^3.8.4", | ||||
|     "@fullcalendar/common": "5.9.0", | ||||
|     "@fullcalendar/core": "5.9.0", | ||||
| @@ -46,45 +46,38 @@ | ||||
|     "@fullcalendar/interaction": "5.9.0", | ||||
|     "@fullcalendar/list": "5.9.0", | ||||
|     "@lit-labs/virtualizer": "patch:@lit-labs/virtualizer@0.6.0#./.yarn/patches/@lit-labs/virtualizer/0.7.0.patch", | ||||
|     "@material/chips": "13.0.0-canary.65125b3a6.0", | ||||
|     "@material/data-table": "13.0.0-canary.65125b3a6.0", | ||||
|     "@material/mwc-button": "0.25.1", | ||||
|     "@material/mwc-checkbox": "0.25.1", | ||||
|     "@material/mwc-circular-progress": "0.25.1", | ||||
|     "@material/mwc-dialog": "0.25.1", | ||||
|     "@material/mwc-fab": "0.25.1", | ||||
|     "@material/mwc-formfield": "0.25.1", | ||||
|     "@material/mwc-icon-button": "0.25.1", | ||||
|     "@material/mwc-linear-progress": "0.25.1", | ||||
|     "@material/mwc-list": "0.25.1", | ||||
|     "@material/mwc-menu": "0.25.1", | ||||
|     "@material/mwc-radio": "0.25.1", | ||||
|     "@material/mwc-ripple": "0.25.1", | ||||
|     "@material/mwc-switch": "0.25.1", | ||||
|     "@material/mwc-tab": "0.25.1", | ||||
|     "@material/mwc-tab-bar": "0.25.1", | ||||
|     "@material/top-app-bar": "13.0.0-canary.65125b3a6.0", | ||||
|     "@mdi/js": "6.2.95", | ||||
|     "@mdi/svg": "6.2.95", | ||||
|     "@material/chips": "14.0.0-canary.261f2db59.0", | ||||
|     "@material/data-table": "14.0.0-canary.261f2db59.0", | ||||
|     "@material/mwc-button": "0.25.3", | ||||
|     "@material/mwc-checkbox": "0.25.3", | ||||
|     "@material/mwc-circular-progress": "0.25.3", | ||||
|     "@material/mwc-dialog": "0.25.3", | ||||
|     "@material/mwc-fab": "0.25.3", | ||||
|     "@material/mwc-formfield": "0.25.3", | ||||
|     "@material/mwc-icon-button": "patch:@material/mwc-icon-button@0.25.3#./.yarn/patches/@material/mwc-icon-button/remove-icon.patch", | ||||
|     "@material/mwc-linear-progress": "0.25.3", | ||||
|     "@material/mwc-list": "0.25.3", | ||||
|     "@material/mwc-menu": "0.25.3", | ||||
|     "@material/mwc-radio": "0.25.3", | ||||
|     "@material/mwc-ripple": "0.25.3", | ||||
|     "@material/mwc-select": "0.25.3", | ||||
|     "@material/mwc-slider": "0.25.3", | ||||
|     "@material/mwc-switch": "0.25.3", | ||||
|     "@material/mwc-tab": "0.25.3", | ||||
|     "@material/mwc-tab-bar": "0.25.3", | ||||
|     "@material/mwc-textfield": "0.25.3", | ||||
|     "@material/top-app-bar": "14.0.0-canary.261f2db59.0", | ||||
|     "@mdi/js": "6.5.95", | ||||
|     "@mdi/svg": "6.5.95", | ||||
|     "@polymer/app-layout": "^3.1.0", | ||||
|     "@polymer/iron-flex-layout": "^3.0.1", | ||||
|     "@polymer/iron-icon": "^3.0.1", | ||||
|     "@polymer/iron-input": "^3.0.1", | ||||
|     "@polymer/iron-overlay-behavior": "^3.0.3", | ||||
|     "@polymer/iron-resizable-behavior": "^3.0.1", | ||||
|     "@polymer/paper-checkbox": "^3.1.0", | ||||
|     "@polymer/paper-dialog": "^3.0.1", | ||||
|     "@polymer/paper-dialog-behavior": "^3.0.1", | ||||
|     "@polymer/paper-dialog-scrollable": "^3.0.1", | ||||
|     "@polymer/paper-dropdown-menu": "^3.2.0", | ||||
|     "@polymer/paper-input": "^3.2.1", | ||||
|     "@polymer/paper-item": "^3.0.1", | ||||
|     "@polymer/paper-listbox": "^3.0.1", | ||||
|     "@polymer/paper-menu-button": "^3.1.0", | ||||
|     "@polymer/paper-progress": "^3.0.1", | ||||
|     "@polymer/paper-radio-button": "^3.0.1", | ||||
|     "@polymer/paper-radio-group": "^3.0.1", | ||||
|     "@polymer/paper-ripple": "^3.0.2", | ||||
|     "@polymer/paper-slider": "^3.0.1", | ||||
|     "@polymer/paper-styles": "^3.0.1", | ||||
|     "@polymer/paper-tabs": "^3.1.0", | ||||
| @@ -115,7 +108,7 @@ | ||||
|     "js-yaml": "^4.1.0", | ||||
|     "leaflet": "^1.7.1", | ||||
|     "leaflet-draw": "^1.0.4", | ||||
|     "lit": "^2.0.0", | ||||
|     "lit": "^2.0.2", | ||||
|     "lit-vaadin-helpers": "^0.2.1", | ||||
|     "marked": "^3.0.2", | ||||
|     "memoize-one": "^5.2.1", | ||||
| @@ -187,7 +180,7 @@ | ||||
|     "eslint-import-resolver-webpack": "^0.13.1", | ||||
|     "eslint-plugin-disable": "^2.0.1", | ||||
|     "eslint-plugin-import": "^2.24.2", | ||||
|     "eslint-plugin-lit": "^1.5.1", | ||||
|     "eslint-plugin-lit": "^1.6.1", | ||||
|     "eslint-plugin-prettier": "^4.0.0", | ||||
|     "eslint-plugin-unused-imports": "^1.1.5", | ||||
|     "eslint-plugin-wc": "^1.3.2", | ||||
| @@ -237,10 +230,10 @@ | ||||
|   "resolutions": { | ||||
|     "@polymer/polymer": "patch:@polymer/polymer@3.4.1#./.yarn/patches/@polymer/polymer/pr-5569.patch", | ||||
|     "@webcomponents/webcomponentsjs": "^2.2.10", | ||||
|     "lit": "^2.0.0", | ||||
|     "lit-html": "2.0.0", | ||||
|     "lit-element": "3.0.0", | ||||
|     "@lit/reactive-element": "1.0.0" | ||||
|     "lit": "^2.0.2", | ||||
|     "lit-html": "2.0.1", | ||||
|     "lit-element": "3.0.1", | ||||
|     "@lit/reactive-element": "1.0.1" | ||||
|   }, | ||||
|   "main": "src/home-assistant.js", | ||||
|   "husky": { | ||||
|   | ||||
							
								
								
									
										2
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								setup.py
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ from setuptools import setup, find_packages | ||||
|  | ||||
| setup( | ||||
|     name="home-assistant-frontend", | ||||
|     version="20211004.0", | ||||
|     version="20211123.0", | ||||
|     description="The Home Assistant frontend", | ||||
|     url="https://github.com/home-assistant/frontend", | ||||
|     author="The Home Assistant Authors", | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { genClientId } from "home-assistant-js-websocket"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
| @@ -7,9 +8,12 @@ import { | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import "./ha-password-manager-polyfill"; | ||||
| import { property, state } from "lit/decorators"; | ||||
| import "../components/ha-alert"; | ||||
| import "../components/ha-checkbox"; | ||||
| import { computeInitialHaFormData } from "../components/ha-form/compute-initial-ha-form-data"; | ||||
| import "../components/ha-form/ha-form"; | ||||
| import "../components/ha-formfield"; | ||||
| import "../components/ha-markdown"; | ||||
| import { AuthProvider } from "../data/auth"; | ||||
| import { | ||||
| @@ -17,6 +21,7 @@ import { | ||||
|   DataEntryFlowStepForm, | ||||
| } from "../data/data_entry_flow"; | ||||
| import { litLocalizeLiteMixin } from "../mixins/lit-localize-lite-mixin"; | ||||
| import "./ha-password-manager-polyfill"; | ||||
|  | ||||
| type State = "loading" | "error" | "step"; | ||||
|  | ||||
| @@ -31,12 +36,44 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|  | ||||
|   @state() private _state: State = "loading"; | ||||
|  | ||||
|   @state() private _stepData: any = {}; | ||||
|   @state() private _stepData?: Record<string, any>; | ||||
|  | ||||
|   @state() private _step?: DataEntryFlowStep; | ||||
|  | ||||
|   @state() private _errorMessage?: string; | ||||
|  | ||||
|   @state() private _submitting = false; | ||||
|  | ||||
|   @state() private _storeToken = false; | ||||
|  | ||||
|   willUpdate(changedProps: PropertyValues) { | ||||
|     super.willUpdate(changedProps); | ||||
|  | ||||
|     if (!changedProps.has("_step")) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if (!this._step) { | ||||
|       this._stepData = undefined; | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const oldStep = changedProps.get("_step") as HaAuthFlow["_step"]; | ||||
|  | ||||
|     if ( | ||||
|       !oldStep || | ||||
|       this._step.flow_id !== oldStep.flow_id || | ||||
|       (this._step.type === "form" && | ||||
|         oldStep.type === "form" && | ||||
|         this._step.step_id !== oldStep.step_id) | ||||
|     ) { | ||||
|       this._stepData = | ||||
|         this._step.type === "form" | ||||
|           ? computeInitialHaFormData(this._step.data_schema) | ||||
|           : undefined; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <form>${this._renderForm()}</form> | ||||
| @@ -76,6 +113,24 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|     if (changedProps.has("authProvider")) { | ||||
|       this._providerChanged(this.authProvider); | ||||
|     } | ||||
|  | ||||
|     if (!changedProps.has("_step") || this._step?.type !== "form") { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     // 100ms to give all the form elements time to initialize. | ||||
|     setTimeout(() => { | ||||
|       const form = this.renderRoot.querySelector("ha-form"); | ||||
|       if (form) { | ||||
|         (form as any).focus(); | ||||
|       } | ||||
|     }, 100); | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       this.renderRoot.querySelector( | ||||
|         "ha-password-manager-polyfill" | ||||
|       )!.boundingRect = this.getBoundingClientRect(); | ||||
|     }, 500); | ||||
|   } | ||||
|  | ||||
|   private _renderForm(): TemplateResult { | ||||
| @@ -87,27 +142,33 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|         return html` | ||||
|           ${this._renderStep(this._step)} | ||||
|           <div class="action"> | ||||
|             <mwc-button raised @click=${this._handleSubmit} | ||||
|               >${this._step.type === "form" | ||||
|                 ? this.localize("ui.panel.page-authorize.form.next") | ||||
|                 : this.localize( | ||||
|                     "ui.panel.page-authorize.form.start_over" | ||||
|                   )}</mwc-button | ||||
|             <mwc-button | ||||
|               raised | ||||
|               @click=${this._handleSubmit} | ||||
|               .disabled=${this._submitting} | ||||
|             > | ||||
|               ${this._step.type === "form" | ||||
|                 ? this.localize("ui.panel.page-authorize.form.next") | ||||
|                 : this.localize("ui.panel.page-authorize.form.start_over")} | ||||
|             </mwc-button> | ||||
|           </div> | ||||
|         `; | ||||
|       case "error": | ||||
|         return html` | ||||
|           <div class="error"> | ||||
|           <ha-alert alert-type="error"> | ||||
|             ${this.localize( | ||||
|               "ui.panel.page-authorize.form.error", | ||||
|               "error", | ||||
|               this._errorMessage | ||||
|             )} | ||||
|           </div> | ||||
|           </ha-alert> | ||||
|         `; | ||||
|       case "loading": | ||||
|         return html` ${this.localize("ui.panel.page-authorize.form.working")} `; | ||||
|         return html` | ||||
|           <ha-alert alert-type="info"> | ||||
|             ${this.localize("ui.panel.page-authorize.form.working")} | ||||
|           </ha-alert> | ||||
|         `; | ||||
|       default: | ||||
|         return html``; | ||||
|     } | ||||
| @@ -140,16 +201,35 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|             .data=${this._stepData} | ||||
|             .schema=${step.data_schema} | ||||
|             .error=${step.errors} | ||||
|             .disabled=${this._submitting} | ||||
|             .computeLabel=${this._computeLabelCallback(step)} | ||||
|             .computeError=${this._computeErrorCallback(step)} | ||||
|             @value-changed=${this._stepDataChanged} | ||||
|           ></ha-form> | ||||
|           ${this.clientId === genClientId() && | ||||
|           !["select_mfa_module", "mfa"].includes(step.step_id) | ||||
|             ? html` | ||||
|                 <ha-formfield | ||||
|                   class="store-token" | ||||
|                   .label=${this.localize("ui.panel.page-authorize.store_token")} | ||||
|                 > | ||||
|                   <ha-checkbox | ||||
|                     .checked=${this._storeToken} | ||||
|                     @change=${this._storeTokenChanged} | ||||
|                   ></ha-checkbox> | ||||
|                 </ha-formfield> | ||||
|               ` | ||||
|             : ""} | ||||
|         `; | ||||
|       default: | ||||
|         return html``; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _storeTokenChanged(e: CustomEvent<HTMLInputElement>) { | ||||
|     this._storeToken = (e.currentTarget as HTMLInputElement).checked; | ||||
|   } | ||||
|  | ||||
|   private async _providerChanged(newProvider?: AuthProvider) { | ||||
|     if (this._step && this._step.type === "form") { | ||||
|       fetch(`/auth/login_flow/${this._step.flow_id}`, { | ||||
| @@ -189,7 +269,8 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|           return; | ||||
|         } | ||||
|  | ||||
|         await this._updateStep(data); | ||||
|         this._step = data; | ||||
|         this._state = "step"; | ||||
|       } else { | ||||
|         this._state = "error"; | ||||
|         this._errorMessage = data.message; | ||||
| @@ -216,43 +297,13 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|     if (this.oauth2State) { | ||||
|       url += `&state=${encodeURIComponent(this.oauth2State)}`; | ||||
|     } | ||||
|     if (this._storeToken) { | ||||
|       url += `&storeToken=true`; | ||||
|     } | ||||
|  | ||||
|     document.location.assign(url); | ||||
|   } | ||||
|  | ||||
|   private async _updateStep(step: DataEntryFlowStep) { | ||||
|     let stepData: any = null; | ||||
|     if ( | ||||
|       this._step && | ||||
|       (step.flow_id !== this._step.flow_id || | ||||
|         (step.type === "form" && | ||||
|           this._step.type === "form" && | ||||
|           step.step_id !== this._step.step_id)) | ||||
|     ) { | ||||
|       stepData = {}; | ||||
|     } | ||||
|     this._step = step; | ||||
|     this._state = "step"; | ||||
|     if (stepData != null) { | ||||
|       this._stepData = stepData; | ||||
|     } | ||||
|  | ||||
|     await this.updateComplete; | ||||
|     // 100ms to give all the form elements time to initialize. | ||||
|     setTimeout(() => { | ||||
|       const form = this.renderRoot.querySelector("ha-form"); | ||||
|       if (form) { | ||||
|         (form as any).focus(); | ||||
|       } | ||||
|     }, 100); | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       this.renderRoot.querySelector( | ||||
|         "ha-password-manager-polyfill" | ||||
|       )!.boundingRect = this.getBoundingClientRect(); | ||||
|     }, 500); | ||||
|   } | ||||
|  | ||||
|   private _stepDataChanged(ev: CustomEvent) { | ||||
|     this._stepData = ev.detail.value; | ||||
|   } | ||||
| @@ -297,9 +348,7 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|       this._providerChanged(this.authProvider); | ||||
|       return; | ||||
|     } | ||||
|     this._state = "loading"; | ||||
|     // To avoid a jumping UI. | ||||
|     this.style.setProperty("min-height", `${this.offsetHeight}px`); | ||||
|     this._submitting = true; | ||||
|  | ||||
|     const postData = { ...this._stepData, client_id: this.clientId }; | ||||
|  | ||||
| @@ -316,29 +365,28 @@ class HaAuthFlow extends litLocalizeLiteMixin(LitElement) { | ||||
|         this._redirect(newStep.result); | ||||
|         return; | ||||
|       } | ||||
|       await this._updateStep(newStep); | ||||
|       this._step = newStep; | ||||
|       this._state = "step"; | ||||
|     } catch (err: any) { | ||||
|       // eslint-disable-next-line no-console | ||||
|       console.error("Error submitting step", err); | ||||
|       this._state = "error"; | ||||
|       this._errorMessage = this._unknownError(); | ||||
|     } finally { | ||||
|       this.style.setProperty("min-height", ""); | ||||
|       this._submitting = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       :host { | ||||
|         /* So we can set min-height to avoid jumping during loading */ | ||||
|         display: block; | ||||
|       } | ||||
|       .action { | ||||
|         margin: 24px 0 8px; | ||||
|         text-align: center; | ||||
|       } | ||||
|       .error { | ||||
|         color: red; | ||||
|       /* Align with the rest of the form. */ | ||||
|       .store-token { | ||||
|         margin-top: 10px; | ||||
|         margin-left: -16px; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -174,6 +174,10 @@ class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | ||||
|         display: block; | ||||
|         margin-top: 48px; | ||||
|       } | ||||
|       ha-auth-flow { | ||||
|         display: block; | ||||
|         margin-top: 24px; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -2,8 +2,8 @@ | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { HaFormSchema } from "../components/ha-form/ha-form"; | ||||
| import { DataEntryFlowStep } from "../data/data_entry_flow"; | ||||
| import type { HaFormSchema } from "../components/ha-form/types"; | ||||
| import type { DataEntryFlowStep } from "../data/data_entry_flow"; | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|   | ||||
| @@ -21,7 +21,11 @@ class HaPickAuthProvider extends litLocalizeLiteMixin(LitElement) { | ||||
|       <p>${this.localize("ui.panel.page-authorize.pick_auth_provider")}:</p> | ||||
|       ${this.authProviders.map( | ||||
|         (provider) => html` | ||||
|           <paper-item .auth_provider=${provider} @click=${this._handlePick}> | ||||
|           <paper-item | ||||
|             role="button" | ||||
|             .auth_provider=${provider} | ||||
|             @click=${this._handlePick} | ||||
|           > | ||||
|             <paper-item-body>${provider.name}</paper-item-body> | ||||
|             <ha-icon-next></ha-icon-next> | ||||
|           </paper-item> | ||||
|   | ||||
| @@ -3,5 +3,5 @@ import { CAST_DEV_APP_ID } from "./dev_const"; | ||||
| // Guard dev mode with `__dev__` so it can only ever be enabled in dev mode. | ||||
| export const CAST_DEV = __DEV__ && true; | ||||
|  | ||||
| export const CAST_APP_ID = CAST_DEV ? CAST_DEV_APP_ID : "B12CE3CA"; | ||||
| export const CAST_APP_ID = CAST_DEV ? CAST_DEV_APP_ID : "A078F6B0"; | ||||
| export const CAST_NS = "urn:x-cast:com.nabucasa.hast"; | ||||
|   | ||||
| @@ -11,4 +11,20 @@ export interface ReceiverStatusMessage extends BaseCastMessage { | ||||
|   urlPath?: string | null; | ||||
| } | ||||
|  | ||||
| export interface ReceiverErrorMessage extends BaseCastMessage { | ||||
|   type: "receiver_error"; | ||||
|   error_code: ReceiverErrorCode; | ||||
|   error_message: string; | ||||
| } | ||||
|  | ||||
| export const enum ReceiverErrorCode { | ||||
|   CONNECTION_FAILED = 1, | ||||
|   AUTHENTICATION_FAILED = 2, | ||||
|   CONNECTION_LOST = 3, | ||||
|   HASS_URL_MISSING = 4, | ||||
|   NO_HTTPS = 5, | ||||
|   NOT_CONNECTED = 21, | ||||
|   FETCH_CONFIG_FAILED = 22, | ||||
| } | ||||
|  | ||||
| export type SenderMessage = ReceiverStatusMessage; | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { AuthData } from "home-assistant-js-websocket"; | ||||
| import { extractSearchParam } from "../url/search-params"; | ||||
|  | ||||
| const storage = window.localStorage || {}; | ||||
|  | ||||
| @@ -30,6 +31,11 @@ export function askWrite() { | ||||
|  | ||||
| export function saveTokens(tokens: AuthData | null) { | ||||
|   tokenCache.tokens = tokens; | ||||
|  | ||||
|   if (!tokenCache.writeEnabled && extractSearchParam("storeToken") === "true") { | ||||
|     tokenCache.writeEnabled = true; | ||||
|   } | ||||
|  | ||||
|   if (tokenCache.writeEnabled) { | ||||
|     try { | ||||
|       storage.hassTokens = JSON.stringify(tokens); | ||||
| @@ -45,7 +51,6 @@ export function enableWrite() { | ||||
|     saveTokens(tokenCache.tokens); | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function loadTokens() { | ||||
|   if (tokenCache.tokens === undefined) { | ||||
|     try { | ||||
|   | ||||
| @@ -7,7 +7,13 @@ export const canShowPage = (hass: HomeAssistant, page: PageNavigation) => | ||||
|   !hideAdvancedPage(hass, page); | ||||
|  | ||||
| const isLoadedIntegration = (hass: HomeAssistant, page: PageNavigation) => | ||||
|   !page.component || isComponentLoaded(hass, page.component); | ||||
|   page.component | ||||
|     ? isComponentLoaded(hass, page.component) | ||||
|     : page.components | ||||
|     ? page.components.some((integration) => | ||||
|         isComponentLoaded(hass, integration) | ||||
|       ) | ||||
|     : true; | ||||
| const isCore = (page: PageNavigation) => page.core; | ||||
| const isAdvancedPage = (page: PageNavigation) => page.advancedOnly; | ||||
| const userWantsAdvanced = (hass: HomeAssistant) => hass.userData?.showAdvanced; | ||||
|   | ||||
| @@ -4,6 +4,10 @@ export const atLeastVersion = ( | ||||
|   minor: number, | ||||
|   patch?: number | ||||
| ): boolean => { | ||||
|   if (__DEMO__) { | ||||
|     return true; | ||||
|   } | ||||
|  | ||||
|   const [haMajor, haMinor, haPatch] = version.split(".", 3); | ||||
|  | ||||
|   return ( | ||||
|   | ||||
| @@ -1,91 +1,150 @@ | ||||
| /** Constants to be used in the frontend. */ | ||||
|  | ||||
| import { | ||||
|   mdiAccount, | ||||
|   mdiAirFilter, | ||||
|   mdiAlert, | ||||
|   mdiAngleAcute, | ||||
|   mdiAppleSafari, | ||||
|   mdiBell, | ||||
|   mdiBookmark, | ||||
|   mdiBrightness5, | ||||
|   mdiBullhorn, | ||||
|   mdiCalendar, | ||||
|   mdiCalendarClock, | ||||
|   mdiCash, | ||||
|   mdiClock, | ||||
|   mdiCloudUpload, | ||||
|   mdiCog, | ||||
|   mdiCommentAlert, | ||||
|   mdiCounter, | ||||
|   mdiCurrentAc, | ||||
|   mdiEye, | ||||
|   mdiFan, | ||||
|   mdiFlash, | ||||
|   mdiFlower, | ||||
|   mdiFormatListBulleted, | ||||
|   mdiFormTextbox, | ||||
|   mdiGasCylinder, | ||||
|   mdiGauge, | ||||
|   mdiGoogleAssistant, | ||||
|   mdiGoogleCirclesCommunities, | ||||
|   mdiHomeAssistant, | ||||
|   mdiHomeAutomation, | ||||
|   mdiImageFilterFrames, | ||||
|   mdiLightbulb, | ||||
|   mdiLightningBolt, | ||||
|   mdiMailbox, | ||||
|   mdiMapMarkerRadius, | ||||
|   mdiMolecule, | ||||
|   mdiMoleculeCo, | ||||
|   mdiMoleculeCo2, | ||||
|   mdiPalette, | ||||
|   mdiRayVertex, | ||||
|   mdiRemote, | ||||
|   mdiRobot, | ||||
|   mdiRobotVacuum, | ||||
|   mdiScriptText, | ||||
|   mdiSineWave, | ||||
|   mdiTextToSpeech, | ||||
|   mdiThermometer, | ||||
|   mdiThermostat, | ||||
|   mdiTimerOutline, | ||||
|   mdiToggleSwitchOutline, | ||||
|   mdiVideo, | ||||
|   mdiWaterPercent, | ||||
|   mdiWeatherCloudy, | ||||
|   mdiWhiteBalanceSunny, | ||||
|   mdiWifi, | ||||
| } from "@mdi/js"; | ||||
|  | ||||
| // Constants should be alphabetically sorted by name. | ||||
| // Arrays with values should be alphabetically sorted if order doesn't matter. | ||||
| // Each constant should have a description what it is supposed to be used for. | ||||
|  | ||||
| /** Icon to use when no icon specified for domain. */ | ||||
| export const DEFAULT_DOMAIN_ICON = "hass:bookmark"; | ||||
| export const DEFAULT_DOMAIN_ICON = mdiBookmark; | ||||
|  | ||||
| /** Icons for each domain */ | ||||
| export const FIXED_DOMAIN_ICONS = { | ||||
|   alert: "hass:alert", | ||||
|   alexa: "hass:amazon-alexa", | ||||
|   air_quality: "hass:air-filter", | ||||
|   automation: "hass:robot", | ||||
|   calendar: "hass:calendar", | ||||
|   camera: "hass:video", | ||||
|   climate: "hass:thermostat", | ||||
|   configurator: "hass:cog", | ||||
|   conversation: "hass:text-to-speech", | ||||
|   counter: "hass:counter", | ||||
|   device_tracker: "hass:account", | ||||
|   fan: "hass:fan", | ||||
|   google_assistant: "hass:google-assistant", | ||||
|   group: "hass:google-circles-communities", | ||||
|   homeassistant: "hass:home-assistant", | ||||
|   homekit: "hass:home-automation", | ||||
|   image_processing: "hass:image-filter-frames", | ||||
|   input_boolean: "hass:toggle-switch-outline", | ||||
|   input_datetime: "hass:calendar-clock", | ||||
|   input_number: "hass:ray-vertex", | ||||
|   input_select: "hass:format-list-bulleted", | ||||
|   input_text: "hass:form-textbox", | ||||
|   light: "hass:lightbulb", | ||||
|   mailbox: "hass:mailbox", | ||||
|   notify: "hass:comment-alert", | ||||
|   number: "hass:ray-vertex", | ||||
|   persistent_notification: "hass:bell", | ||||
|   person: "hass:account", | ||||
|   plant: "hass:flower", | ||||
|   proximity: "hass:apple-safari", | ||||
|   remote: "hass:remote", | ||||
|   scene: "hass:palette", | ||||
|   script: "hass:script-text", | ||||
|   select: "hass:format-list-bulleted", | ||||
|   sensor: "hass:eye", | ||||
|   simple_alarm: "hass:bell", | ||||
|   sun: "hass:white-balance-sunny", | ||||
|   switch: "hass:flash", | ||||
|   timer: "hass:timer-outline", | ||||
|   updater: "hass:cloud-upload", | ||||
|   vacuum: "hass:robot-vacuum", | ||||
|   water_heater: "hass:thermometer", | ||||
|   weather: "hass:weather-cloudy", | ||||
|   zone: "hass:map-marker-radius", | ||||
|   alert: mdiAlert, | ||||
|   air_quality: mdiAirFilter, | ||||
|   automation: mdiRobot, | ||||
|   calendar: mdiCalendar, | ||||
|   camera: mdiVideo, | ||||
|   climate: mdiThermostat, | ||||
|   configurator: mdiCog, | ||||
|   conversation: mdiTextToSpeech, | ||||
|   counter: mdiCounter, | ||||
|   fan: mdiFan, | ||||
|   google_assistant: mdiGoogleAssistant, | ||||
|   group: mdiGoogleCirclesCommunities, | ||||
|   homeassistant: mdiHomeAssistant, | ||||
|   homekit: mdiHomeAutomation, | ||||
|   image_processing: mdiImageFilterFrames, | ||||
|   input_boolean: mdiToggleSwitchOutline, | ||||
|   input_datetime: mdiCalendarClock, | ||||
|   input_number: mdiRayVertex, | ||||
|   input_select: mdiFormatListBulleted, | ||||
|   input_text: mdiFormTextbox, | ||||
|   light: mdiLightbulb, | ||||
|   mailbox: mdiMailbox, | ||||
|   notify: mdiCommentAlert, | ||||
|   number: mdiRayVertex, | ||||
|   persistent_notification: mdiBell, | ||||
|   person: mdiAccount, | ||||
|   plant: mdiFlower, | ||||
|   proximity: mdiAppleSafari, | ||||
|   remote: mdiRemote, | ||||
|   scene: mdiPalette, | ||||
|   script: mdiScriptText, | ||||
|   select: mdiFormatListBulleted, | ||||
|   sensor: mdiEye, | ||||
|   siren: mdiBullhorn, | ||||
|   simple_alarm: mdiBell, | ||||
|   sun: mdiWhiteBalanceSunny, | ||||
|   timer: mdiTimerOutline, | ||||
|   updater: mdiCloudUpload, | ||||
|   vacuum: mdiRobotVacuum, | ||||
|   water_heater: mdiThermometer, | ||||
|   weather: mdiWeatherCloudy, | ||||
|   zone: mdiMapMarkerRadius, | ||||
| }; | ||||
|  | ||||
| export const FIXED_DEVICE_CLASS_ICONS = { | ||||
|   aqi: "hass:air-filter", | ||||
|   current: "hass:current-ac", | ||||
|   carbon_dioxide: "mdi:molecule-co2", | ||||
|   carbon_monoxide: "mdi:molecule-co", | ||||
|   date: "hass:calendar", | ||||
|   energy: "hass:lightning-bolt", | ||||
|   gas: "hass:gas-cylinder", | ||||
|   humidity: "hass:water-percent", | ||||
|   illuminance: "hass:brightness-5", | ||||
|   nitrogen_dioxide: "mdi:molecule", | ||||
|   nitrogen_monoxide: "mdi:molecule", | ||||
|   nitrous_oxide: "mdi:molecule", | ||||
|   ozone: "mdi:molecule", | ||||
|   temperature: "hass:thermometer", | ||||
|   monetary: "mdi:cash", | ||||
|   pm25: "mdi:molecule", | ||||
|   pm1: "mdi:molecule", | ||||
|   pm10: "mdi:molecule", | ||||
|   pressure: "hass:gauge", | ||||
|   power: "hass:flash", | ||||
|   power_factor: "hass:angle-acute", | ||||
|   signal_strength: "hass:wifi", | ||||
|   sulphur_dioxide: "mdi:molecule", | ||||
|   timestamp: "hass:clock", | ||||
|   volatile_organic_compounds: "mdi:molecule", | ||||
|   voltage: "hass:sine-wave", | ||||
|   aqi: mdiAirFilter, | ||||
|   // battery: mdiBattery, => not included by design since `sensorIcon()` will dynamically determine the icon | ||||
|   carbon_dioxide: mdiMoleculeCo2, | ||||
|   carbon_monoxide: mdiMoleculeCo, | ||||
|   current: mdiCurrentAc, | ||||
|   date: mdiCalendar, | ||||
|   energy: mdiLightningBolt, | ||||
|   frequency: mdiSineWave, | ||||
|   gas: mdiGasCylinder, | ||||
|   humidity: mdiWaterPercent, | ||||
|   illuminance: mdiBrightness5, | ||||
|   monetary: mdiCash, | ||||
|   nitrogen_dioxide: mdiMolecule, | ||||
|   nitrogen_monoxide: mdiMolecule, | ||||
|   nitrous_oxide: mdiMolecule, | ||||
|   ozone: mdiMolecule, | ||||
|   pm1: mdiMolecule, | ||||
|   pm10: mdiMolecule, | ||||
|   pm25: mdiMolecule, | ||||
|   power: mdiFlash, | ||||
|   power_factor: mdiAngleAcute, | ||||
|   pressure: mdiGauge, | ||||
|   signal_strength: mdiWifi, | ||||
|   sulphur_dioxide: mdiMolecule, | ||||
|   temperature: mdiThermometer, | ||||
|   timestamp: mdiClock, | ||||
|   volatile_organic_compounds: mdiMolecule, | ||||
|   voltage: mdiSineWave, | ||||
| }; | ||||
|  | ||||
| /** Domains that have a state card. */ | ||||
| export const DOMAINS_WITH_CARD = [ | ||||
|   "button", | ||||
|   "climate", | ||||
|   "cover", | ||||
|   "configurator", | ||||
|   | ||||
| @@ -36,55 +36,62 @@ export const applyThemesOnElement = ( | ||||
|   let cacheKey = selectedTheme; | ||||
|   let themeRules: Partial<ThemeVars> = {}; | ||||
|  | ||||
|   if (themeSettings) { | ||||
|     if (themeSettings.dark) { | ||||
|       cacheKey = `${cacheKey}__dark`; | ||||
|       themeRules = { ...darkStyles }; | ||||
|   // If there is no explicitly desired dark mode provided, we automatically | ||||
|   // use the active one from hass.themes. | ||||
|   if (!themeSettings || themeSettings?.dark === undefined) { | ||||
|     themeSettings = { | ||||
|       ...themeSettings, | ||||
|       dark: themes.darkMode, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   if (themeSettings.dark) { | ||||
|     cacheKey = `${cacheKey}__dark`; | ||||
|     themeRules = { ...darkStyles }; | ||||
|   } | ||||
|  | ||||
|   if (selectedTheme === "default") { | ||||
|     // Determine the primary and accent colors from the current settings. | ||||
|     // Fallbacks are implicitly the HA default blue and orange or the | ||||
|     // derived "darkStyles" values, depending on the light vs dark mode. | ||||
|     const primaryColor = themeSettings.primaryColor; | ||||
|     const accentColor = themeSettings.accentColor; | ||||
|  | ||||
|     if (themeSettings.dark && primaryColor) { | ||||
|       themeRules["app-header-background-color"] = hexBlend( | ||||
|         primaryColor, | ||||
|         "#121212", | ||||
|         8 | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (selectedTheme === "default") { | ||||
|       // Determine the primary and accent colors from the current settings. | ||||
|       // Fallbacks are implicitly the HA default blue and orange or the | ||||
|       // derived "darkStyles" values, depending on the light vs dark mode. | ||||
|       const primaryColor = themeSettings.primaryColor; | ||||
|       const accentColor = themeSettings.accentColor; | ||||
|     if (primaryColor) { | ||||
|       cacheKey = `${cacheKey}__primary_${primaryColor}`; | ||||
|       const rgbPrimaryColor = hex2rgb(primaryColor); | ||||
|       const labPrimaryColor = rgb2lab(rgbPrimaryColor); | ||||
|       themeRules["primary-color"] = primaryColor; | ||||
|       const rgbLightPrimaryColor = lab2rgb(labBrighten(labPrimaryColor)); | ||||
|       themeRules["light-primary-color"] = rgb2hex(rgbLightPrimaryColor); | ||||
|       themeRules["dark-primary-color"] = lab2hex(labDarken(labPrimaryColor)); | ||||
|       themeRules["text-primary-color"] = | ||||
|         rgbContrast(rgbPrimaryColor, [33, 33, 33]) < 6 ? "#fff" : "#212121"; | ||||
|       themeRules["text-light-primary-color"] = | ||||
|         rgbContrast(rgbLightPrimaryColor, [33, 33, 33]) < 6 | ||||
|           ? "#fff" | ||||
|           : "#212121"; | ||||
|       themeRules["state-icon-color"] = themeRules["dark-primary-color"]; | ||||
|     } | ||||
|     if (accentColor) { | ||||
|       cacheKey = `${cacheKey}__accent_${accentColor}`; | ||||
|       themeRules["accent-color"] = accentColor; | ||||
|       const rgbAccentColor = hex2rgb(accentColor); | ||||
|       themeRules["text-accent-color"] = | ||||
|         rgbContrast(rgbAccentColor, [33, 33, 33]) < 6 ? "#fff" : "#212121"; | ||||
|     } | ||||
|  | ||||
|       if (themeSettings.dark && primaryColor) { | ||||
|         themeRules["app-header-background-color"] = hexBlend( | ||||
|           primaryColor, | ||||
|           "#121212", | ||||
|           8 | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       if (primaryColor) { | ||||
|         cacheKey = `${cacheKey}__primary_${primaryColor}`; | ||||
|         const rgbPrimaryColor = hex2rgb(primaryColor); | ||||
|         const labPrimaryColor = rgb2lab(rgbPrimaryColor); | ||||
|         themeRules["primary-color"] = primaryColor; | ||||
|         const rgbLightPrimaryColor = lab2rgb(labBrighten(labPrimaryColor)); | ||||
|         themeRules["light-primary-color"] = rgb2hex(rgbLightPrimaryColor); | ||||
|         themeRules["dark-primary-color"] = lab2hex(labDarken(labPrimaryColor)); | ||||
|         themeRules["text-primary-color"] = | ||||
|           rgbContrast(rgbPrimaryColor, [33, 33, 33]) < 6 ? "#fff" : "#212121"; | ||||
|         themeRules["text-light-primary-color"] = | ||||
|           rgbContrast(rgbLightPrimaryColor, [33, 33, 33]) < 6 | ||||
|             ? "#fff" | ||||
|             : "#212121"; | ||||
|         themeRules["state-icon-color"] = themeRules["dark-primary-color"]; | ||||
|       } | ||||
|       if (accentColor) { | ||||
|         cacheKey = `${cacheKey}__accent_${accentColor}`; | ||||
|         themeRules["accent-color"] = accentColor; | ||||
|         const rgbAccentColor = hex2rgb(accentColor); | ||||
|         themeRules["text-accent-color"] = | ||||
|           rgbContrast(rgbAccentColor, [33, 33, 33]) < 6 ? "#fff" : "#212121"; | ||||
|       } | ||||
|  | ||||
|       // Nothing was changed | ||||
|       if (element._themes?.cacheKey === cacheKey) { | ||||
|         return; | ||||
|       } | ||||
|     // Nothing was changed | ||||
|     if (element._themes?.cacheKey === cacheKey) { | ||||
|       return; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -115,7 +122,7 @@ export const applyThemesOnElement = ( | ||||
|   } | ||||
|  | ||||
|   const newTheme = | ||||
|     themeRules && cacheKey | ||||
|     Object.keys(themeRules).length && cacheKey | ||||
|       ? PROCESSED_THEMES[cacheKey] || processTheme(cacheKey, themeRules) | ||||
|       : undefined; | ||||
|  | ||||
|   | ||||
| @@ -1,2 +1,3 @@ | ||||
| /** An empty image which can be set as src of an img element. */ | ||||
| export default "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7"; | ||||
| export const emptyImageBase64 = | ||||
|   "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7"; | ||||
|   | ||||
| @@ -1,24 +1,36 @@ | ||||
| /** Return an icon representing a alarm panel state. */ | ||||
|  | ||||
| import { | ||||
|   mdiShieldLock, | ||||
|   mdiShieldAirplane, | ||||
|   mdiShieldHome, | ||||
|   mdiShieldMoon, | ||||
|   mdiSecurity, | ||||
|   mdiShieldOutline, | ||||
|   mdiBellRing, | ||||
|   mdiShieldOff, | ||||
|   mdiShield, | ||||
| } from "@mdi/js"; | ||||
|  | ||||
| export const alarmPanelIcon = (state?: string) => { | ||||
|   switch (state) { | ||||
|     case "armed_away": | ||||
|       return "hass:shield-lock"; | ||||
|       return mdiShieldLock; | ||||
|     case "armed_vacation": | ||||
|       return "hass:shield-airplane"; | ||||
|       return mdiShieldAirplane; | ||||
|     case "armed_home": | ||||
|       return "hass:shield-home"; | ||||
|       return mdiShieldHome; | ||||
|     case "armed_night": | ||||
|       return "hass:shield-moon"; | ||||
|       return mdiShieldMoon; | ||||
|     case "armed_custom_bypass": | ||||
|       return "hass:security"; | ||||
|       return mdiSecurity; | ||||
|     case "pending": | ||||
|       return "hass:shield-outline"; | ||||
|       return mdiShieldOutline; | ||||
|     case "triggered": | ||||
|       return "hass:bell-ring"; | ||||
|       return mdiBellRing; | ||||
|     case "disarmed": | ||||
|       return "hass:shield-off"; | ||||
|       return mdiShieldOff; | ||||
|     default: | ||||
|       return "hass:shield"; | ||||
|       return mdiShield; | ||||
|   } | ||||
| }; | ||||
|   | ||||
| @@ -1,35 +1,92 @@ | ||||
| /** Return an icon representing a battery state. */ | ||||
| import { | ||||
|   mdiBattery, | ||||
|   mdiBattery10, | ||||
|   mdiBattery20, | ||||
|   mdiBattery30, | ||||
|   mdiBattery40, | ||||
|   mdiBattery50, | ||||
|   mdiBattery60, | ||||
|   mdiBattery70, | ||||
|   mdiBattery80, | ||||
|   mdiBattery90, | ||||
|   mdiBatteryAlert, | ||||
|   mdiBatteryAlertVariantOutline, | ||||
|   mdiBatteryCharging, | ||||
|   mdiBatteryCharging10, | ||||
|   mdiBatteryCharging20, | ||||
|   mdiBatteryCharging30, | ||||
|   mdiBatteryCharging40, | ||||
|   mdiBatteryCharging50, | ||||
|   mdiBatteryCharging60, | ||||
|   mdiBatteryCharging70, | ||||
|   mdiBatteryCharging80, | ||||
|   mdiBatteryCharging90, | ||||
|   mdiBatteryChargingOutline, | ||||
|   mdiBatteryUnknown, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
|  | ||||
| export const batteryIcon = ( | ||||
| const BATTERY_ICONS = { | ||||
|   10: mdiBattery10, | ||||
|   20: mdiBattery20, | ||||
|   30: mdiBattery30, | ||||
|   40: mdiBattery40, | ||||
|   50: mdiBattery50, | ||||
|   60: mdiBattery60, | ||||
|   70: mdiBattery70, | ||||
|   80: mdiBattery80, | ||||
|   90: mdiBattery90, | ||||
|   100: mdiBattery, | ||||
| }; | ||||
| const BATTERY_CHARGING_ICONS = { | ||||
|   10: mdiBatteryCharging10, | ||||
|   20: mdiBatteryCharging20, | ||||
|   30: mdiBatteryCharging30, | ||||
|   40: mdiBatteryCharging40, | ||||
|   50: mdiBatteryCharging50, | ||||
|   60: mdiBatteryCharging60, | ||||
|   70: mdiBatteryCharging70, | ||||
|   80: mdiBatteryCharging80, | ||||
|   90: mdiBatteryCharging90, | ||||
|   100: mdiBatteryCharging, | ||||
| }; | ||||
|  | ||||
| export const batteryStateIcon = ( | ||||
|   batteryState: HassEntity, | ||||
|   batteryChargingState?: HassEntity | ||||
| ) => { | ||||
|   const battery = Number(batteryState.state); | ||||
|   const battery_charging = | ||||
|   const battery = batteryState.state; | ||||
|   const batteryCharging = | ||||
|     batteryChargingState && batteryChargingState.state === "on"; | ||||
|   let icon = "hass:battery"; | ||||
|  | ||||
|   if (isNaN(battery)) { | ||||
|     if (batteryState.state === "off") { | ||||
|       icon += "-full"; | ||||
|     } else if (batteryState.state === "on") { | ||||
|       icon += "-alert"; | ||||
|     } else { | ||||
|       icon += "-unknown"; | ||||
|     } | ||||
|     return icon; | ||||
|   } | ||||
|  | ||||
|   const batteryRound = Math.round(battery / 10) * 10; | ||||
|   if (battery_charging && battery > 10) { | ||||
|     icon += `-charging-${batteryRound}`; | ||||
|   } else if (battery_charging) { | ||||
|     icon += "-outline"; | ||||
|   } else if (battery <= 5) { | ||||
|     icon += "-alert"; | ||||
|   } else if (battery > 5 && battery < 95) { | ||||
|     icon += `-${batteryRound}`; | ||||
|   } | ||||
|   return icon; | ||||
|   return batteryIcon(battery, batteryCharging); | ||||
| }; | ||||
|  | ||||
| export const batteryIcon = ( | ||||
|   batteryState: number | string, | ||||
|   batteryCharging?: boolean | ||||
| ) => { | ||||
|   const batteryValue = Number(batteryState); | ||||
|   if (isNaN(batteryValue)) { | ||||
|     if (batteryState === "off") { | ||||
|       return mdiBattery; | ||||
|     } | ||||
|     if (batteryState === "on") { | ||||
|       return mdiBatteryAlert; | ||||
|     } | ||||
|     return mdiBatteryUnknown; | ||||
|   } | ||||
|  | ||||
|   const batteryRound = Math.round(batteryValue / 10) * 10; | ||||
|   if (batteryCharging && batteryValue >= 10) { | ||||
|     return BATTERY_CHARGING_ICONS[batteryRound]; | ||||
|   } | ||||
|   if (batteryCharging) { | ||||
|     return mdiBatteryChargingOutline; | ||||
|   } | ||||
|   if (batteryValue <= 5) { | ||||
|     return mdiBatteryAlertVariantOutline; | ||||
|   } | ||||
|   return BATTERY_ICONS[batteryRound]; | ||||
| }; | ||||
|   | ||||
| @@ -1,3 +1,46 @@ | ||||
| import { | ||||
|   mdiAlertCircle, | ||||
|   mdiBattery, | ||||
|   mdiBatteryCharging, | ||||
|   mdiBatteryOutline, | ||||
|   mdiBrightness5, | ||||
|   mdiBrightness7, | ||||
|   mdiCheckboxMarkedCircle, | ||||
|   mdiCheckNetworkOutline, | ||||
|   mdiCloseNetworkOutline, | ||||
|   mdiCheckCircle, | ||||
|   mdiCropPortrait, | ||||
|   mdiDoorClosed, | ||||
|   mdiDoorOpen, | ||||
|   mdiFire, | ||||
|   mdiGarage, | ||||
|   mdiGarageOpen, | ||||
|   mdiHome, | ||||
|   mdiHomeOutline, | ||||
|   mdiLock, | ||||
|   mdiLockOpen, | ||||
|   mdiMusicNote, | ||||
|   mdiMusicNoteOff, | ||||
|   mdiPackage, | ||||
|   mdiPackageUp, | ||||
|   mdiPlay, | ||||
|   mdiPowerPlug, | ||||
|   mdiPowerPlugOff, | ||||
|   mdiRadioboxBlank, | ||||
|   mdiRun, | ||||
|   mdiSmoke, | ||||
|   mdiSnowflake, | ||||
|   mdiSquare, | ||||
|   mdiSquareOutline, | ||||
|   mdiStop, | ||||
|   mdiThermometer, | ||||
|   mdiVibrate, | ||||
|   mdiWalk, | ||||
|   mdiWater, | ||||
|   mdiWaterOff, | ||||
|   mdiWindowClosed, | ||||
|   mdiWindowOpen, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
|  | ||||
| /** Return an icon representing a binary sensor state. */ | ||||
| @@ -6,52 +49,55 @@ export const binarySensorIcon = (state?: string, stateObj?: HassEntity) => { | ||||
|   const is_off = state === "off"; | ||||
|   switch (stateObj?.attributes.device_class) { | ||||
|     case "battery": | ||||
|       return is_off ? "hass:battery" : "hass:battery-outline"; | ||||
|       return is_off ? mdiBattery : mdiBatteryOutline; | ||||
|     case "battery_charging": | ||||
|       return is_off ? "hass:battery" : "hass:battery-charging"; | ||||
|       return is_off ? mdiBattery : mdiBatteryCharging; | ||||
|     case "cold": | ||||
|       return is_off ? "hass:thermometer" : "hass:snowflake"; | ||||
|       return is_off ? mdiThermometer : mdiSnowflake; | ||||
|     case "connectivity": | ||||
|       return is_off ? "hass:server-network-off" : "hass:server-network"; | ||||
|       return is_off ? mdiCloseNetworkOutline : mdiCheckNetworkOutline; | ||||
|     case "door": | ||||
|       return is_off ? "hass:door-closed" : "hass:door-open"; | ||||
|       return is_off ? mdiDoorClosed : mdiDoorOpen; | ||||
|     case "garage_door": | ||||
|       return is_off ? "hass:garage" : "hass:garage-open"; | ||||
|       return is_off ? mdiGarage : mdiGarageOpen; | ||||
|     case "power": | ||||
|       return is_off ? "hass:power-plug-off" : "hass:power-plug"; | ||||
|       return is_off ? mdiPowerPlugOff : mdiPowerPlug; | ||||
|     case "gas": | ||||
|     case "problem": | ||||
|     case "safety": | ||||
|       return is_off ? "hass:check-circle" : "hass:alert-circle"; | ||||
|     case "tamper": | ||||
|       return is_off ? mdiCheckCircle : mdiAlertCircle; | ||||
|     case "smoke": | ||||
|       return is_off ? "hass:check-circle" : "hass:smoke"; | ||||
|       return is_off ? mdiCheckCircle : mdiSmoke; | ||||
|     case "heat": | ||||
|       return is_off ? "hass:thermometer" : "hass:fire"; | ||||
|       return is_off ? mdiThermometer : mdiFire; | ||||
|     case "light": | ||||
|       return is_off ? "hass:brightness-5" : "hass:brightness-7"; | ||||
|       return is_off ? mdiBrightness5 : mdiBrightness7; | ||||
|     case "lock": | ||||
|       return is_off ? "hass:lock" : "hass:lock-open"; | ||||
|       return is_off ? mdiLock : mdiLockOpen; | ||||
|     case "moisture": | ||||
|       return is_off ? "hass:water-off" : "hass:water"; | ||||
|       return is_off ? mdiWaterOff : mdiWater; | ||||
|     case "motion": | ||||
|       return is_off ? "hass:walk" : "hass:run"; | ||||
|       return is_off ? mdiWalk : mdiRun; | ||||
|     case "occupancy": | ||||
|       return is_off ? "hass:home-outline" : "hass:home"; | ||||
|       return is_off ? mdiHomeOutline : mdiHome; | ||||
|     case "opening": | ||||
|       return is_off ? "hass:square" : "hass:square-outline"; | ||||
|       return is_off ? mdiSquare : mdiSquareOutline; | ||||
|     case "plug": | ||||
|       return is_off ? "hass:power-plug-off" : "hass:power-plug"; | ||||
|       return is_off ? mdiPowerPlugOff : mdiPowerPlug; | ||||
|     case "presence": | ||||
|       return is_off ? "hass:home-outline" : "hass:home"; | ||||
|       return is_off ? mdiHomeOutline : mdiHome; | ||||
|     case "running": | ||||
|       return is_off ? mdiStop : mdiPlay; | ||||
|     case "sound": | ||||
|       return is_off ? "hass:music-note-off" : "hass:music-note"; | ||||
|       return is_off ? mdiMusicNoteOff : mdiMusicNote; | ||||
|     case "update": | ||||
|       return is_off ? "mdi:package" : "mdi:package-up"; | ||||
|       return is_off ? mdiPackage : mdiPackageUp; | ||||
|     case "vibration": | ||||
|       return is_off ? "hass:crop-portrait" : "hass:vibrate"; | ||||
|       return is_off ? mdiCropPortrait : mdiVibrate; | ||||
|     case "window": | ||||
|       return is_off ? "hass:window-closed" : "hass:window-open"; | ||||
|       return is_off ? mdiWindowClosed : mdiWindowOpen; | ||||
|     default: | ||||
|       return is_off ? "hass:radiobox-blank" : "hass:checkbox-marked-circle"; | ||||
|       return is_off ? mdiRadioboxBlank : mdiCheckboxMarkedCircle; | ||||
|   } | ||||
| }; | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import { FrontendLocaleData } from "../../data/translation"; | ||||
| import { formatDate } from "../datetime/format_date"; | ||||
| import { formatDateTime } from "../datetime/format_date_time"; | ||||
| import { formatTime } from "../datetime/format_time"; | ||||
| import { formatNumber } from "../number/format_number"; | ||||
| import { formatNumber, isNumericState } from "../number/format_number"; | ||||
| import { LocalizeFunc } from "../translations/localize"; | ||||
| import { computeStateDomain } from "./compute_state_domain"; | ||||
|  | ||||
| @@ -20,7 +20,8 @@ export const computeStateDisplay = ( | ||||
|     return localize(`state.default.${compareState}`); | ||||
|   } | ||||
|  | ||||
|   if (stateObj.attributes.unit_of_measurement) { | ||||
|   // Entities with a `unit_of_measurement` or `state_class` are numeric values and should use `formatNumber` | ||||
|   if (isNumericState(stateObj)) { | ||||
|     if (stateObj.attributes.device_class === "monetary") { | ||||
|       try { | ||||
|         return formatNumber(compareState, locale, { | ||||
| @@ -31,15 +32,17 @@ export const computeStateDisplay = ( | ||||
|         // fallback to default | ||||
|       } | ||||
|     } | ||||
|     return `${formatNumber(compareState, locale)} ${ | ||||
|     return `${formatNumber(compareState, locale)}${ | ||||
|       stateObj.attributes.unit_of_measurement | ||||
|         ? " " + stateObj.attributes.unit_of_measurement | ||||
|         : "" | ||||
|     }`; | ||||
|   } | ||||
|  | ||||
|   const domain = computeStateDomain(stateObj); | ||||
|  | ||||
|   if (domain === "input_datetime") { | ||||
|     if (state) { | ||||
|     if (state !== undefined) { | ||||
|       // If trying to display an explicit state, need to parse the explict state to `Date` then format. | ||||
|       // Attributes aren't available, we have to use `state`. | ||||
|       try { | ||||
| @@ -63,7 +66,7 @@ export const computeStateDisplay = ( | ||||
|           } | ||||
|         } | ||||
|         return state; | ||||
|       } catch { | ||||
|       } catch (_e) { | ||||
|         // Formatting methods may throw error if date parsing doesn't go well, | ||||
|         // just return the state string in that case. | ||||
|         return state; | ||||
| @@ -71,7 +74,17 @@ export const computeStateDisplay = ( | ||||
|     } else { | ||||
|       // If not trying to display an explicit state, create `Date` object from `stateObj`'s attributes then format. | ||||
|       let date: Date; | ||||
|       if (!stateObj.attributes.has_time) { | ||||
|       if (stateObj.attributes.has_date && stateObj.attributes.has_time) { | ||||
|         date = new Date( | ||||
|           stateObj.attributes.year, | ||||
|           stateObj.attributes.month - 1, | ||||
|           stateObj.attributes.day, | ||||
|           stateObj.attributes.hour, | ||||
|           stateObj.attributes.minute | ||||
|         ); | ||||
|         return formatDateTime(date, locale); | ||||
|       } | ||||
|       if (stateObj.attributes.has_date) { | ||||
|         date = new Date( | ||||
|           stateObj.attributes.year, | ||||
|           stateObj.attributes.month - 1, | ||||
| @@ -79,20 +92,12 @@ export const computeStateDisplay = ( | ||||
|         ); | ||||
|         return formatDate(date, locale); | ||||
|       } | ||||
|       if (!stateObj.attributes.has_date) { | ||||
|       if (stateObj.attributes.has_time) { | ||||
|         date = new Date(); | ||||
|         date.setHours(stateObj.attributes.hour, stateObj.attributes.minute); | ||||
|         return formatTime(date, locale); | ||||
|       } | ||||
|  | ||||
|       date = new Date( | ||||
|         stateObj.attributes.year, | ||||
|         stateObj.attributes.month - 1, | ||||
|         stateObj.attributes.day, | ||||
|         stateObj.attributes.hour, | ||||
|         stateObj.attributes.minute | ||||
|       ); | ||||
|       return formatDateTime(date, locale); | ||||
|       return stateObj.state; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -111,6 +116,14 @@ export const computeStateDisplay = ( | ||||
|     return formatNumber(compareState, locale); | ||||
|   } | ||||
|  | ||||
|   // state of button is a timestamp | ||||
|   if ( | ||||
|     domain === "button" || | ||||
|     (domain === "sensor" && stateObj.attributes.device_class === "timestamp") | ||||
|   ) { | ||||
|     return formatDateTime(new Date(compareState), locale); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     // Return device class translation | ||||
|     (stateObj.attributes.device_class && | ||||
|   | ||||
| @@ -1,4 +1,30 @@ | ||||
| /** Return an icon representing a cover state. */ | ||||
| import { | ||||
|   mdiArrowUpBox, | ||||
|   mdiArrowDownBox, | ||||
|   mdiGarage, | ||||
|   mdiGarageOpen, | ||||
|   mdiGateArrowRight, | ||||
|   mdiGate, | ||||
|   mdiGateOpen, | ||||
|   mdiDoorOpen, | ||||
|   mdiDoorClosed, | ||||
|   mdiCircle, | ||||
|   mdiWindowShutter, | ||||
|   mdiWindowShutterOpen, | ||||
|   mdiBlinds, | ||||
|   mdiBlindsOpen, | ||||
|   mdiWindowClosed, | ||||
|   mdiWindowOpen, | ||||
|   mdiArrowExpandHorizontal, | ||||
|   mdiArrowUp, | ||||
|   mdiArrowCollapseHorizontal, | ||||
|   mdiArrowDown, | ||||
|   mdiCircleSlice8, | ||||
|   mdiArrowSplitVertical, | ||||
|   mdiCurtains, | ||||
|   mdiCurtainsClosed, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
|  | ||||
| export const coverIcon = (state?: string, stateObj?: HassEntity): string => { | ||||
| @@ -8,74 +34,84 @@ export const coverIcon = (state?: string, stateObj?: HassEntity): string => { | ||||
|     case "garage": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|           return "hass:arrow-up-box"; | ||||
|           return mdiArrowUpBox; | ||||
|         case "closing": | ||||
|           return "hass:arrow-down-box"; | ||||
|           return mdiArrowDownBox; | ||||
|         case "closed": | ||||
|           return "hass:garage"; | ||||
|           return mdiGarage; | ||||
|         default: | ||||
|           return "hass:garage-open"; | ||||
|           return mdiGarageOpen; | ||||
|       } | ||||
|     case "gate": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|         case "closing": | ||||
|           return "hass:gate-arrow-right"; | ||||
|           return mdiGateArrowRight; | ||||
|         case "closed": | ||||
|           return "hass:gate"; | ||||
|           return mdiGate; | ||||
|         default: | ||||
|           return "hass:gate-open"; | ||||
|           return mdiGateOpen; | ||||
|       } | ||||
|     case "door": | ||||
|       return open ? "hass:door-open" : "hass:door-closed"; | ||||
|       return open ? mdiDoorOpen : mdiDoorClosed; | ||||
|     case "damper": | ||||
|       return open ? "hass:circle" : "hass:circle-slice-8"; | ||||
|       return open ? mdiCircle : mdiCircleSlice8; | ||||
|     case "shutter": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|           return "hass:arrow-up-box"; | ||||
|           return mdiArrowUpBox; | ||||
|         case "closing": | ||||
|           return "hass:arrow-down-box"; | ||||
|           return mdiArrowDownBox; | ||||
|         case "closed": | ||||
|           return "hass:window-shutter"; | ||||
|           return mdiWindowShutter; | ||||
|         default: | ||||
|           return "hass:window-shutter-open"; | ||||
|           return mdiWindowShutterOpen; | ||||
|       } | ||||
|     case "curtain": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|           return mdiArrowSplitVertical; | ||||
|         case "closing": | ||||
|           return mdiArrowCollapseHorizontal; | ||||
|         case "closed": | ||||
|           return mdiCurtainsClosed; | ||||
|         default: | ||||
|           return mdiCurtains; | ||||
|       } | ||||
|     case "blind": | ||||
|     case "curtain": | ||||
|     case "shade": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|           return "hass:arrow-up-box"; | ||||
|           return mdiArrowUpBox; | ||||
|         case "closing": | ||||
|           return "hass:arrow-down-box"; | ||||
|           return mdiArrowDownBox; | ||||
|         case "closed": | ||||
|           return "hass:blinds"; | ||||
|           return mdiBlinds; | ||||
|         default: | ||||
|           return "hass:blinds-open"; | ||||
|           return mdiBlindsOpen; | ||||
|       } | ||||
|     case "window": | ||||
|       switch (state) { | ||||
|         case "opening": | ||||
|           return "hass:arrow-up-box"; | ||||
|           return mdiArrowUpBox; | ||||
|         case "closing": | ||||
|           return "hass:arrow-down-box"; | ||||
|           return mdiArrowDownBox; | ||||
|         case "closed": | ||||
|           return "hass:window-closed"; | ||||
|           return mdiWindowClosed; | ||||
|         default: | ||||
|           return "hass:window-open"; | ||||
|           return mdiWindowOpen; | ||||
|       } | ||||
|   } | ||||
|  | ||||
|   switch (state) { | ||||
|     case "opening": | ||||
|       return "hass:arrow-up-box"; | ||||
|       return mdiArrowUpBox; | ||||
|     case "closing": | ||||
|       return "hass:arrow-down-box"; | ||||
|       return mdiArrowDownBox; | ||||
|     case "closed": | ||||
|       return "hass:window-closed"; | ||||
|       return mdiWindowClosed; | ||||
|     default: | ||||
|       return "hass:window-open"; | ||||
|       return mdiWindowOpen; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| @@ -84,9 +120,9 @@ export const computeOpenIcon = (stateObj: HassEntity): string => { | ||||
|     case "awning": | ||||
|     case "door": | ||||
|     case "gate": | ||||
|       return "hass:arrow-expand-horizontal"; | ||||
|       return mdiArrowExpandHorizontal; | ||||
|     default: | ||||
|       return "hass:arrow-up"; | ||||
|       return mdiArrowUp; | ||||
|   } | ||||
| }; | ||||
|  | ||||
| @@ -95,8 +131,8 @@ export const computeCloseIcon = (stateObj: HassEntity): string => { | ||||
|     case "awning": | ||||
|     case "door": | ||||
|     case "gate": | ||||
|       return "hass:arrow-collapse-horizontal"; | ||||
|       return mdiArrowCollapseHorizontal; | ||||
|     default: | ||||
|       return "hass:arrow-down"; | ||||
|       return mdiArrowDown; | ||||
|   } | ||||
| }; | ||||
|   | ||||
| @@ -1,3 +1,31 @@ | ||||
| import { | ||||
|   mdiAccount, | ||||
|   mdiAccountArrowRight, | ||||
|   mdiAirHumidifierOff, | ||||
|   mdiAirHumidifier, | ||||
|   mdiFlash, | ||||
|   mdiBluetooth, | ||||
|   mdiBluetoothConnect, | ||||
|   mdiLanConnect, | ||||
|   mdiLanDisconnect, | ||||
|   mdiLockOpen, | ||||
|   mdiLockAlert, | ||||
|   mdiLockClock, | ||||
|   mdiLock, | ||||
|   mdiCastConnected, | ||||
|   mdiCast, | ||||
|   mdiEmoticonDead, | ||||
|   mdiPowerPlug, | ||||
|   mdiPowerPlugOff, | ||||
|   mdiSleep, | ||||
|   mdiTimerSand, | ||||
|   mdiToggleSwitch, | ||||
|   mdiToggleSwitchOff, | ||||
|   mdiZWave, | ||||
|   mdiClock, | ||||
|   mdiCalendar, | ||||
|   mdiWeatherNight, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| /** | ||||
|  * Return the icon to be used for a domain. | ||||
| @@ -27,37 +55,56 @@ export const domainIcon = ( | ||||
|     case "cover": | ||||
|       return coverIcon(compareState, stateObj); | ||||
|  | ||||
|     case "device_tracker": | ||||
|       if (stateObj?.attributes.source_type === "router") { | ||||
|         return compareState === "home" ? mdiLanConnect : mdiLanDisconnect; | ||||
|       } | ||||
|       if ( | ||||
|         ["bluetooth", "bluetooth_le"].includes(stateObj?.attributes.source_type) | ||||
|       ) { | ||||
|         return compareState === "home" ? mdiBluetoothConnect : mdiBluetooth; | ||||
|       } | ||||
|       return compareState === "not_home" ? mdiAccountArrowRight : mdiAccount; | ||||
|  | ||||
|     case "humidifier": | ||||
|       return state && state === "off" | ||||
|         ? "hass:air-humidifier-off" | ||||
|         : "hass:air-humidifier"; | ||||
|       return state && state === "off" ? mdiAirHumidifierOff : mdiAirHumidifier; | ||||
|  | ||||
|     case "lock": | ||||
|       switch (compareState) { | ||||
|         case "unlocked": | ||||
|           return "hass:lock-open"; | ||||
|           return mdiLockOpen; | ||||
|         case "jammed": | ||||
|           return "hass:lock-alert"; | ||||
|           return mdiLockAlert; | ||||
|         case "locking": | ||||
|         case "unlocking": | ||||
|           return "hass:lock-clock"; | ||||
|           return mdiLockClock; | ||||
|         default: | ||||
|           return "hass:lock"; | ||||
|           return mdiLock; | ||||
|       } | ||||
|  | ||||
|     case "media_player": | ||||
|       return compareState === "playing" ? "hass:cast-connected" : "hass:cast"; | ||||
|       return compareState === "playing" ? mdiCastConnected : mdiCast; | ||||
|  | ||||
|     case "switch": | ||||
|       switch (stateObj?.attributes.device_class) { | ||||
|         case "outlet": | ||||
|           return state === "on" ? mdiPowerPlug : mdiPowerPlugOff; | ||||
|         case "switch": | ||||
|           return state === "on" ? mdiToggleSwitch : mdiToggleSwitchOff; | ||||
|         default: | ||||
|           return mdiFlash; | ||||
|       } | ||||
|  | ||||
|     case "zwave": | ||||
|       switch (compareState) { | ||||
|         case "dead": | ||||
|           return "hass:emoticon-dead"; | ||||
|           return mdiEmoticonDead; | ||||
|         case "sleeping": | ||||
|           return "hass:sleep"; | ||||
|           return mdiSleep; | ||||
|         case "initializing": | ||||
|           return "hass:timer-sand"; | ||||
|           return mdiTimerSand; | ||||
|         default: | ||||
|           return "hass:z-wave"; | ||||
|           return mdiZWave; | ||||
|       } | ||||
|  | ||||
|     case "sensor": { | ||||
| @@ -71,17 +118,17 @@ export const domainIcon = ( | ||||
|  | ||||
|     case "input_datetime": | ||||
|       if (!stateObj?.attributes.has_date) { | ||||
|         return "hass:clock"; | ||||
|         return mdiClock; | ||||
|       } | ||||
|       if (!stateObj.attributes.has_time) { | ||||
|         return "hass:calendar"; | ||||
|         return mdiCalendar; | ||||
|       } | ||||
|       break; | ||||
|  | ||||
|     case "sun": | ||||
|       return stateObj?.state === "above_horizon" | ||||
|         ? FIXED_DOMAIN_ICONS[domain] | ||||
|         : "hass:weather-night"; | ||||
|         : mdiWeatherNight; | ||||
|   } | ||||
|  | ||||
|   if (domain in FIXED_DOMAIN_ICONS) { | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| /** Return an icon representing a sensor state. */ | ||||
| import { mdiBattery, mdiThermometer } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { FIXED_DEVICE_CLASS_ICONS, UNIT_C, UNIT_F } from "../const"; | ||||
| import { batteryIcon } from "./battery_icon"; | ||||
| import { SENSOR_DEVICE_CLASS_BATTERY } from "../../data/sensor"; | ||||
| import { FIXED_DEVICE_CLASS_ICONS, UNIT_C, UNIT_F } from "../const"; | ||||
| import { batteryStateIcon } from "./battery_icon"; | ||||
|  | ||||
| export const sensorIcon = (stateObj?: HassEntity): string | undefined => { | ||||
|   const dclass = stateObj?.attributes.device_class; | ||||
| @@ -12,12 +13,12 @@ export const sensorIcon = (stateObj?: HassEntity): string | undefined => { | ||||
|   } | ||||
|  | ||||
|   if (dclass === SENSOR_DEVICE_CLASS_BATTERY) { | ||||
|     return stateObj ? batteryIcon(stateObj) : "hass:battery"; | ||||
|     return stateObj ? batteryStateIcon(stateObj) : mdiBattery; | ||||
|   } | ||||
|  | ||||
|   const unit = stateObj?.attributes.unit_of_measurement; | ||||
|   if (unit === UNIT_C || unit === UNIT_F) { | ||||
|     return "hass:thermometer"; | ||||
|     return mdiThermometer; | ||||
|   } | ||||
|  | ||||
|   return undefined; | ||||
|   | ||||
| @@ -4,13 +4,9 @@ import { DEFAULT_DOMAIN_ICON } from "../const"; | ||||
| import { computeDomain } from "./compute_domain"; | ||||
| import { domainIcon } from "./domain_icon"; | ||||
| 
 | ||||
| export const stateIcon = (state?: HassEntity) => { | ||||
| export const stateIconPath = (state?: HassEntity) => { | ||||
|   if (!state) { | ||||
|     return DEFAULT_DOMAIN_ICON; | ||||
|   } | ||||
|   if (state.attributes.icon) { | ||||
|     return state.attributes.icon; | ||||
|   } | ||||
| 
 | ||||
|   return domainIcon(computeDomain(state.entity_id), state); | ||||
| }; | ||||
							
								
								
									
										24
									
								
								src/common/entity/strip_prefix_from_entity_name.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								src/common/entity/strip_prefix_from_entity_name.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| /** | ||||
|  * Strips a device name from an entity name. | ||||
|  * @param entityName the entity name | ||||
|  * @param lowerCasedPrefixWithSpaceSuffix the prefix to strip, lower cased with a space suffix | ||||
|  * @returns | ||||
|  */ | ||||
| export const stripPrefixFromEntityName = ( | ||||
|   entityName: string, | ||||
|   lowerCasedPrefixWithSpaceSuffix: string | ||||
| ) => { | ||||
|   if (!entityName.toLowerCase().startsWith(lowerCasedPrefixWithSpaceSuffix)) { | ||||
|     return undefined; | ||||
|   } | ||||
|  | ||||
|   const newName = entityName.substring(lowerCasedPrefixWithSpaceSuffix.length); | ||||
|  | ||||
|   // If first word already has an upper case letter (e.g. from brand name) | ||||
|   // leave as-is, otherwise capitalize the first word. | ||||
|   return hasUpperCase(newName.substr(0, newName.indexOf(" "))) | ||||
|     ? newName | ||||
|     : newName[0].toUpperCase() + newName.slice(1); | ||||
| }; | ||||
|  | ||||
| const hasUpperCase = (str: string): boolean => str.toLowerCase() !== str; | ||||
| @@ -1,6 +1,15 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { FrontendLocaleData, NumberFormat } from "../../data/translation"; | ||||
| import { round } from "./round"; | ||||
|  | ||||
| /** | ||||
|  * Returns true if the entity is considered numeric based on the attributes it has | ||||
|  * @param stateObj The entity state object | ||||
|  */ | ||||
| export const isNumericState = (stateObj: HassEntity): boolean => | ||||
|   !!stateObj.attributes.unit_of_measurement || | ||||
|   !!stateObj.attributes.state_class; | ||||
|  | ||||
| export const numberFormatToLocale = ( | ||||
|   localeOptions: FrontendLocaleData | ||||
| ): string | string[] | undefined => { | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "@material/mwc-icon-button/mwc-icon-button"; | ||||
| import { mdiClose, mdiMagnify } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input"; | ||||
| import type { PaperInputElement } from "@polymer/paper-input/paper-input"; | ||||
| @@ -11,11 +10,15 @@ import { | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../components/ha-icon-button"; | ||||
| import "../../components/ha-svg-icon"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import { fireEvent } from "../dom/fire_event"; | ||||
|  | ||||
| @customElement("search-input") | ||||
| class SearchInput extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public filter?: string; | ||||
|  | ||||
|   @property({ type: Boolean, attribute: "no-label-float" }) | ||||
| @@ -50,13 +53,12 @@ class SearchInput extends LitElement { | ||||
|         </slot> | ||||
|         ${this.filter && | ||||
|         html` | ||||
|           <mwc-icon-button | ||||
|           <ha-icon-button | ||||
|             slot="suffix" | ||||
|             @click=${this._clearSearch} | ||||
|             title="Clear" | ||||
|           > | ||||
|             <ha-svg-icon .path=${mdiClose}></ha-svg-icon> | ||||
|           </mwc-icon-button> | ||||
|             .label=${this.hass.localize("ui.common.clear")} | ||||
|             .path=${mdiClose} | ||||
|           ></ha-icon-button> | ||||
|         `} | ||||
|       </paper-input> | ||||
|     `; | ||||
| @@ -90,10 +92,10 @@ class SearchInput extends LitElement { | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       ha-svg-icon, | ||||
|       mwc-icon-button { | ||||
|       ha-icon-button { | ||||
|         color: var(--primary-text-color); | ||||
|       } | ||||
|       mwc-icon-button { | ||||
|       ha-icon-button { | ||||
|         --mdc-icon-button-size: 24px; | ||||
|       } | ||||
|       ha-svg-icon.prefix { | ||||
|   | ||||
							
								
								
									
										2
									
								
								src/common/string/capitalize-first-letter.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								src/common/string/capitalize-first-letter.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| export const capitalizeFirstLetter = (str: string) => | ||||
|   str.charAt(0).toUpperCase() + str.slice(1); | ||||
| @@ -180,10 +180,10 @@ export function fuzzyScore( | ||||
|     wordLow | ||||
|   ); | ||||
|  | ||||
|   let row = 1; | ||||
|   let row: number; | ||||
|   let column = 1; | ||||
|   let patternPos = patternStart; | ||||
|   let wordPos = wordStart; | ||||
|   let patternPos: number; | ||||
|   let wordPos: number; | ||||
|  | ||||
|   const hasStrongFirstMatch = [false]; | ||||
|  | ||||
|   | ||||
| @@ -12,8 +12,8 @@ export const slugify = (value: string, delimiter = "_") => { | ||||
|     .replace(p, (c) => b.charAt(a.indexOf(c))) // Replace special characters | ||||
|     .replace(/&/g, `${delimiter}and${delimiter}`) // Replace & with 'and' | ||||
|     .replace(/[^\w-]+/g, "") // Remove all non-word characters | ||||
|     .replace(/-/, delimiter) // Replace - with delimiter | ||||
|     .replace(new RegExp(`/${delimiter}${delimiter}+/`, "g"), delimiter) // Replace multiple delimiters with single delimiter | ||||
|     .replace(new RegExp(`/^${delimiter}+/`), "") // Trim delimiter from start of text | ||||
|     .replace(new RegExp(`/-+$/`), ""); // Trim delimiter from end of text | ||||
|     .replace(/-/g, delimiter) // Replace - with delimiter | ||||
|     .replace(new RegExp(`(${delimiter})\\1+`, "g"), "$1") // Replace multiple delimiters with single delimiter | ||||
|     .replace(new RegExp(`^${delimiter}+`), "") // Trim delimiter from start of text | ||||
|     .replace(new RegExp(`${delimiter}+$`), ""); // Trim delimiter from end of text | ||||
| }; | ||||
|   | ||||
| @@ -1,57 +1,57 @@ | ||||
| import { css } from "lit"; | ||||
|  | ||||
| export const iconColorCSS = css` | ||||
|   ha-icon[data-domain="alert"][data-state="on"], | ||||
|   ha-icon[data-domain="automation"][data-state="on"], | ||||
|   ha-icon[data-domain="binary_sensor"][data-state="on"], | ||||
|   ha-icon[data-domain="calendar"][data-state="on"], | ||||
|   ha-icon[data-domain="camera"][data-state="streaming"], | ||||
|   ha-icon[data-domain="cover"][data-state="open"], | ||||
|   ha-icon[data-domain="fan"][data-state="on"], | ||||
|   ha-icon[data-domain="humidifier"][data-state="on"], | ||||
|   ha-icon[data-domain="light"][data-state="on"], | ||||
|   ha-icon[data-domain="input_boolean"][data-state="on"], | ||||
|   ha-icon[data-domain="lock"][data-state="unlocked"], | ||||
|   ha-icon[data-domain="media_player"][data-state="on"], | ||||
|   ha-icon[data-domain="media_player"][data-state="paused"], | ||||
|   ha-icon[data-domain="media_player"][data-state="playing"], | ||||
|   ha-icon[data-domain="script"][data-state="on"], | ||||
|   ha-icon[data-domain="sun"][data-state="above_horizon"], | ||||
|   ha-icon[data-domain="switch"][data-state="on"], | ||||
|   ha-icon[data-domain="timer"][data-state="active"], | ||||
|   ha-icon[data-domain="vacuum"][data-state="cleaning"], | ||||
|   ha-icon[data-domain="group"][data-state="on"], | ||||
|   ha-icon[data-domain="group"][data-state="home"], | ||||
|   ha-icon[data-domain="group"][data-state="open"], | ||||
|   ha-icon[data-domain="group"][data-state="locked"], | ||||
|   ha-icon[data-domain="group"][data-state="problem"] { | ||||
|   ha-state-icon[data-domain="alert"][data-state="on"], | ||||
|   ha-state-icon[data-domain="automation"][data-state="on"], | ||||
|   ha-state-icon[data-domain="binary_sensor"][data-state="on"], | ||||
|   ha-state-icon[data-domain="calendar"][data-state="on"], | ||||
|   ha-state-icon[data-domain="camera"][data-state="streaming"], | ||||
|   ha-state-icon[data-domain="cover"][data-state="open"], | ||||
|   ha-state-icon[data-domain="fan"][data-state="on"], | ||||
|   ha-state-icon[data-domain="humidifier"][data-state="on"], | ||||
|   ha-state-icon[data-domain="light"][data-state="on"], | ||||
|   ha-state-icon[data-domain="input_boolean"][data-state="on"], | ||||
|   ha-state-icon[data-domain="lock"][data-state="unlocked"], | ||||
|   ha-state-icon[data-domain="media_player"][data-state="on"], | ||||
|   ha-state-icon[data-domain="media_player"][data-state="paused"], | ||||
|   ha-state-icon[data-domain="media_player"][data-state="playing"], | ||||
|   ha-state-icon[data-domain="script"][data-state="on"], | ||||
|   ha-state-icon[data-domain="sun"][data-state="above_horizon"], | ||||
|   ha-state-icon[data-domain="switch"][data-state="on"], | ||||
|   ha-state-icon[data-domain="timer"][data-state="active"], | ||||
|   ha-state-icon[data-domain="vacuum"][data-state="cleaning"], | ||||
|   ha-state-icon[data-domain="group"][data-state="on"], | ||||
|   ha-state-icon[data-domain="group"][data-state="home"], | ||||
|   ha-state-icon[data-domain="group"][data-state="open"], | ||||
|   ha-state-icon[data-domain="group"][data-state="locked"], | ||||
|   ha-state-icon[data-domain="group"][data-state="problem"] { | ||||
|     color: var(--paper-item-icon-active-color, #fdd835); | ||||
|   } | ||||
|  | ||||
|   ha-icon[data-domain="climate"][data-state="cooling"] { | ||||
|   ha-state-icon[data-domain="climate"][data-state="cooling"] { | ||||
|     color: var(--cool-color, var(--state-climate-cool-color)); | ||||
|   } | ||||
|  | ||||
|   ha-icon[data-domain="climate"][data-state="heating"] { | ||||
|   ha-state-icon[data-domain="climate"][data-state="heating"] { | ||||
|     color: var(--heat-color, var(--state-climate-heat-color)); | ||||
|   } | ||||
|  | ||||
|   ha-icon[data-domain="climate"][data-state="drying"] { | ||||
|   ha-state-icon[data-domain="climate"][data-state="drying"] { | ||||
|     color: var(--dry-color, var(--state-climate-dry-color)); | ||||
|   } | ||||
|  | ||||
|   ha-icon[data-domain="alarm_control_panel"] { | ||||
|   ha-state-icon[data-domain="alarm_control_panel"] { | ||||
|     color: var(--alarm-color-armed, var(--label-badge-red)); | ||||
|   } | ||||
|   ha-icon[data-domain="alarm_control_panel"][data-state="disarmed"] { | ||||
|   ha-state-icon[data-domain="alarm_control_panel"][data-state="disarmed"] { | ||||
|     color: var(--alarm-color-disarmed, var(--label-badge-green)); | ||||
|   } | ||||
|   ha-icon[data-domain="alarm_control_panel"][data-state="pending"], | ||||
|   ha-icon[data-domain="alarm_control_panel"][data-state="arming"] { | ||||
|   ha-state-icon[data-domain="alarm_control_panel"][data-state="pending"], | ||||
|   ha-state-icon[data-domain="alarm_control_panel"][data-state="arming"] { | ||||
|     color: var(--alarm-color-pending, var(--label-badge-yellow)); | ||||
|     animation: pulse 1s infinite; | ||||
|   } | ||||
|   ha-icon[data-domain="alarm_control_panel"][data-state="triggered"] { | ||||
|   ha-state-icon[data-domain="alarm_control_panel"][data-state="triggered"] { | ||||
|     color: var(--alarm-color-triggered, var(--label-badge-red)); | ||||
|     animation: pulse 1s infinite; | ||||
|   } | ||||
| @@ -68,13 +68,13 @@ export const iconColorCSS = css` | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   ha-icon[data-domain="plant"][data-state="problem"], | ||||
|   ha-icon[data-domain="zwave"][data-state="dead"] { | ||||
|   ha-state-icon[data-domain="plant"][data-state="problem"], | ||||
|   ha-state-icon[data-domain="zwave"][data-state="dead"] { | ||||
|     color: var(--state-icon-error-color); | ||||
|   } | ||||
|  | ||||
|   /* Color the icon if unavailable */ | ||||
|   ha-icon[data-state="unavailable"] { | ||||
|   ha-state-icon[data-state="unavailable"] { | ||||
|     color: var(--state-unavailable-color); | ||||
|   } | ||||
| `; | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user