mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 11:39:41 +00:00 
			
		
		
		
	Compare commits
	
		
			149 Commits
		
	
	
		
			20220226.0
			...
			add-Use-UU
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1f6243145f | ||
|   | 27ca61ec85 | ||
|   | 859f49f3eb | ||
|   | 40d878689f | ||
|   | 420e8fe1ff | ||
|   | df96199433 | ||
|   | f493280f0a | ||
|   | cbd030a379 | ||
|   | 95b80accc9 | ||
|   | c522670815 | ||
|   | 7b6d3c0e36 | ||
|   | 504b043159 | ||
|   | dffc66ccc3 | ||
|   | c7e9ee785d | ||
|   | 079cc39a6e | ||
|   | d6a1d5af79 | ||
|   | c0dce08e19 | ||
|   | a7a347ed05 | ||
|   | 2d9b50defc | ||
|   | 840858b18c | ||
|   | afd2e71f6c | ||
|   | 88af0aa788 | ||
|   | 49124f6f09 | ||
|   | 73f5580555 | ||
|   | bdde5268c6 | ||
|   | 15e972c158 | ||
|   | 0fc4c24f5a | ||
|   | 9eba50df0c | ||
|   | 0e0e07437f | ||
|   | 6ac51ede52 | ||
|   | ccf1fb573a | ||
|   | fa537968c4 | ||
|   | 6bf2111a3c | ||
|   | f5f8be8276 | ||
|   | ddf1cc0733 | ||
|   | 9c1d1cb6f6 | ||
|   | 470225abde | ||
|   | ee230b86c1 | ||
|   | f927fc64a9 | ||
|   | 03677c33f7 | ||
|   | bc36a206da | ||
|   | af06ab1e2d | ||
|   | 3e2135a485 | ||
|   | 2e7f8fb46f | ||
|   | 102568c4bd | ||
|   | 4fcdae842e | ||
|   | ea19740f5a | ||
|   | 3e0942b631 | ||
|   | 0261cea796 | ||
|   | 5247b2813f | ||
|   | 8a5090684e | ||
|   | 1784ba5e68 | ||
|   | 4fbe9a7b10 | ||
|   | 1ca9c7838a | ||
|   | 4fc2c3ef05 | ||
|   | 73ff8e28a8 | ||
|   | dde1c5e03c | ||
|   | 01eed22592 | ||
|   | 94ebb63589 | ||
|   | 29119db5ce | ||
|   | 9908162ac2 | ||
|   | 1e929ae78a | ||
|   | ab5df0fe6e | ||
|   | d5010dda9e | ||
|   | 4ac097f32b | ||
|   | 5d3d15072f | ||
|   | 5c53bc4225 | ||
|   | d5a307f8f4 | ||
|   | a27dd1e7f1 | ||
|   | c86ed1fb3e | ||
|   | 7fa7a48072 | ||
|   | 4e0fc8ee08 | ||
|   | 5f6490e54e | ||
|   | db78b046a2 | ||
|   | c37fe1e7ff | ||
|   | f1ec479d41 | ||
|   | e01cb3ca82 | ||
|   | b8d3c68a7a | ||
|   | 641003bb2a | ||
|   | 3358fc2b18 | ||
|   | dcf50e055b | ||
|   | 1fa04baa16 | ||
|   | 84ffa2369a | ||
|   | cc27ddb362 | ||
|   | c4dc6bfb0d | ||
|   | 4fbcc30a37 | ||
|   | 4916527e5f | ||
|   | fad8a27232 | ||
|   | a993d3a753 | ||
|   | 5dfe17a43a | ||
|   | 9b6c935ffb | ||
|   | f4e28da0a3 | ||
|   | 294a69d7e4 | ||
|   | f89b8cffcf | ||
|   | 99fd3a1b6f | ||
|   | 246e426182 | ||
|   | 9f1e9b43fe | ||
|   | 8301ae262c | ||
|   | d968fe41ee | ||
|   | db830e9014 | ||
|   | fc6b594a27 | ||
|   | 86dbf99ebe | ||
|   | 68e7ce1883 | ||
|   | e9003ac35e | ||
|   | 1dd5214b42 | ||
|   | 96738350bb | ||
|   | 5bdecf57cf | ||
|   | ec12282f8c | ||
|   | 552dbca201 | ||
|   | 0bbc0ebb3c | ||
|   | ac7acc5802 | ||
|   | 64e1d160d1 | ||
|   | 8e51878b6d | ||
|   | 7c94ced303 | ||
|   | a040e1d5e0 | ||
|   | 87c7407857 | ||
|   | d0d0c44ec7 | ||
|   | 4cdff3faea | ||
|   | 0dac10aa23 | ||
|   | 4b8b14a69d | ||
|   | 9d28df31bd | ||
|   | 8258641443 | ||
|   | dfcb0f6ba0 | ||
|   | 2e10eb04b6 | ||
|   | b4b52d3872 | ||
|   | 3873203721 | ||
|   | ccb91e0b49 | ||
|   | bd20c15a55 | ||
|   | 0936fd9ae4 | ||
|   | adefc7a4e2 | ||
|   | 8f8017ecff | ||
|   | 604b79696e | ||
|   | 8c445f6409 | ||
|   | 797c871137 | ||
|   | 24829bd903 | ||
|   | add92a559d | ||
|   | 7f086c0900 | ||
|   | 17018c0f26 | ||
|   | cd6a478130 | ||
|   | 4f6d7ca5c9 | ||
|   | c2994343b4 | ||
|   | e5f77c35d4 | ||
|   | a9e5a5dd44 | ||
|   | 1159798b8d | ||
|   | 437de42c55 | ||
|   | 89e0bb3f16 | ||
|   | 28c9631b6c | ||
|   | 35a41b3490 | ||
|   | f59cb661cd | 
							
								
								
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,10 +10,18 @@ env: | ||||
|   NODE_VERSION: 14 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| # Set default workflow permissions | ||||
| # All scopes not mentioned here are set to no access | ||||
| # https://docs.github.com/en/actions/security-guides/automatic-token-authentication#permissions-for-the-github_token | ||||
| permissions: | ||||
|   actions: none | ||||
|  | ||||
| jobs: | ||||
|   release: | ||||
|     name: Release | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write  # Required to upload release assets | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v2 | ||||
| @@ -47,6 +55,13 @@ jobs: | ||||
|  | ||||
|           script/release | ||||
|  | ||||
|       - name: Upload release assets | ||||
|         uses: softprops/action-gh-release@v0.1.14 | ||||
|         with: | ||||
|           files: | | ||||
|             dist/*.whl | ||||
|             dist/*.tar.gz | ||||
|  | ||||
|   wheels-init: | ||||
|     name: Init wheels build | ||||
|     needs: release | ||||
|   | ||||
| @@ -1,4 +1,3 @@ | ||||
| import "web-animations-js/web-animations-next-lite.min"; | ||||
| import "../../../src/resources/ha-style"; | ||||
| import "../../../src/resources/roboto"; | ||||
| import "./layout/hc-lovelace"; | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 44 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 35 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 67 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 27 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 32 KiB | 
| @@ -23,7 +23,7 @@ if [[ "${PULL_REQUEST}" == "true" ]]; then | ||||
|     createStatus "pending" "Building design preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     gulp build-gallery | ||||
|     if [ $? -eq 0 ]; then | ||||
|       createStatus "success" "Build complete" "$DEPLOY_URL" | ||||
|       createStatus "success" "Build complete" "$DEPLOY_PRIME_URL" | ||||
|     else | ||||
|       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     fi | ||||
|   | ||||
| @@ -36,12 +36,17 @@ module.exports = [ | ||||
|     category: "misc", | ||||
|     header: "Miscelaneous", | ||||
|   }, | ||||
|   { | ||||
|     category: "brand", | ||||
|     header: "Brand", | ||||
|   }, | ||||
|   { | ||||
|     category: "user-test", | ||||
|     header: "User Tests", | ||||
|     header: "Users", | ||||
|     pages: ["user-types", "configuration-menu"], | ||||
|   }, | ||||
|   { | ||||
|     category: "design.home-assistant.io", | ||||
|     header: "Design Documentation", | ||||
|     header: "About", | ||||
|   }, | ||||
| ]; | ||||
|   | ||||
| @@ -78,6 +78,9 @@ class DemoCards extends LitElement { | ||||
|     ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|     #container { | ||||
|       background-color: var(--primary-background-color); | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,14 @@ class PageDescription extends HaMarkdown { | ||||
|     if (!PAGES[this.page].description) { | ||||
|       return html``; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <div class="heading"> | ||||
|         <div class="title"> | ||||
|           ${PAGES[this.page].metadata.title || this.page.split("/")[1]} | ||||
|         </div> | ||||
|         <div class="subtitle">${PAGES[this.page].metadata.subtitle}</div> | ||||
|       </div> | ||||
|       ${until( | ||||
|         PAGES[this.page] | ||||
|           .description() | ||||
| @@ -25,9 +32,22 @@ class PageDescription extends HaMarkdown { | ||||
|   static styles = [ | ||||
|     HaMarkdown.styles, | ||||
|     css` | ||||
|       .heading { | ||||
|         padding: 16px; | ||||
|         border-bottom: 1px solid var(--secondary-background-color); | ||||
|       } | ||||
|       .title { | ||||
|         font-size: 42px; | ||||
|         line-height: 56px; | ||||
|         padding-bottom: 8px; | ||||
|       } | ||||
|       .subtitle { | ||||
|         font-size: 18px; | ||||
|         line-height: 24px; | ||||
|       } | ||||
|       .root { | ||||
|         max-width: 800px; | ||||
|         margin: 0 auto; | ||||
|         margin: 16px auto; | ||||
|       } | ||||
|       .root > *:first-child { | ||||
|         margin-top: 0; | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import { html, css, LitElement, PropertyValues } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../src/components/ha-icon-button"; | ||||
| import "../../src/managers/notification-manager"; | ||||
| import "../../src/components/ha-expansion-panel"; | ||||
| import { haStyle } from "../../src/resources/styles"; | ||||
| import { PAGES, SIDEBAR } from "../build/import-pages"; | ||||
| import { dynamicElement } from "../../src/common/dom/dynamic-element-directive"; | ||||
| @@ -44,6 +45,10 @@ class HaGallery extends LitElement { | ||||
|       for (const page of group.pages!) { | ||||
|         const key = `${group.category}/${page}`; | ||||
|         const active = this._page === key; | ||||
|         if (!(key in PAGES)) { | ||||
|           console.error("Undefined page referenced in sidebar.js:", key); | ||||
|           continue; | ||||
|         } | ||||
|         const title = PAGES[key].metadata.title || page; | ||||
|         links.push(html` | ||||
|           <a ?active=${active} href=${`#${group.category}/${page}`}>${title}</a> | ||||
| @@ -53,10 +58,9 @@ class HaGallery extends LitElement { | ||||
|       sidebar.push( | ||||
|         group.header | ||||
|           ? html` | ||||
|               <details> | ||||
|                 <summary class="section">${group.header}</summary> | ||||
|               <ha-expansion-panel .header=${group.header}> | ||||
|                 ${links} | ||||
|               </details> | ||||
|               </ha-expansion-panel> | ||||
|             ` | ||||
|           : links | ||||
|       ); | ||||
| @@ -92,27 +96,34 @@ class HaGallery extends LitElement { | ||||
|             ${dynamicElement(`demo-${this._page.replace("/", "-")}`)} | ||||
|           </div> | ||||
|           <div class="page-footer"> | ||||
|             ${PAGES[this._page].description || | ||||
|             Object.keys(PAGES[this._page].metadata).length > 0 | ||||
|               ? html` | ||||
|                   <a | ||||
|                     href=${`${GITHUB_DEMO_URL}${this._page}.markdown`} | ||||
|                     target="_blank" | ||||
|                   > | ||||
|                     Edit text | ||||
|                   </a> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${PAGES[this._page].demo | ||||
|               ? html` | ||||
|                   <a | ||||
|                     href=${`${GITHUB_DEMO_URL}${this._page}.ts`} | ||||
|                     target="_blank" | ||||
|                   > | ||||
|                     Edit demo | ||||
|                   </a> | ||||
|                 ` | ||||
|               : ""} | ||||
|             <div class="header">Help us to improve our documentation</div> | ||||
|             <div class="secondary"> | ||||
|               Suggest an edit to this page, or provide/view feedback for this | ||||
|               page. | ||||
|             </div> | ||||
|             <div> | ||||
|               ${PAGES[this._page].description || | ||||
|               Object.keys(PAGES[this._page].metadata).length > 0 | ||||
|                 ? html` | ||||
|                     <a | ||||
|                       href=${`${GITHUB_DEMO_URL}${this._page}.markdown`} | ||||
|                       target="_blank" | ||||
|                     > | ||||
|                       Edit text | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|               ${PAGES[this._page].demo | ||||
|                 ? html` | ||||
|                     <a | ||||
|                       href=${`${GITHUB_DEMO_URL}${this._page}.ts`} | ||||
|                       target="_blank" | ||||
|                     > | ||||
|                       Edit demo | ||||
|                     </a> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </mwc-drawer> | ||||
| @@ -186,27 +197,16 @@ class HaGallery extends LitElement { | ||||
|         padding: 4px; | ||||
|       } | ||||
|  | ||||
|       .sidebar details { | ||||
|         margin-top: 1em; | ||||
|         margin-left: 1em; | ||||
|       } | ||||
|  | ||||
|       .sidebar summary { | ||||
|         cursor: pointer; | ||||
|         font-weight: bold; | ||||
|         margin-bottom: 8px; | ||||
|       } | ||||
|  | ||||
|       .sidebar a { | ||||
|         color: var(--primary-text-color); | ||||
|         display: block; | ||||
|         padding: 4px 12px; | ||||
|         padding: 12px; | ||||
|         text-decoration: none; | ||||
|         position: relative; | ||||
|       } | ||||
|  | ||||
|       .sidebar a[active]::before { | ||||
|         border-radius: 4px; | ||||
|         border-radius: 12px; | ||||
|         position: absolute; | ||||
|         top: 0; | ||||
|         right: 2px; | ||||
| @@ -237,14 +237,32 @@ class HaGallery extends LitElement { | ||||
|  | ||||
|       .page-footer { | ||||
|         text-align: center; | ||||
|         margin: 16px 0; | ||||
|         padding-top: 16px; | ||||
|         border-top: 1px solid rgba(0, 0, 0, 0.12); | ||||
|         margin: 16px; | ||||
|         padding: 16px; | ||||
|         border-radius: 12px; | ||||
|         background-color: var(--primary-background-color); | ||||
|       } | ||||
|  | ||||
|       .page-footer div { | ||||
|         margin-top: 4px; | ||||
|       } | ||||
|  | ||||
|       .page-footer .header { | ||||
|         font-size: 16px; | ||||
|         font-weight: 500; | ||||
|         line-height: 28px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer .secondary { | ||||
|         line-height: 23px; | ||||
|         text-align: center; | ||||
|       } | ||||
|  | ||||
|       .page-footer a { | ||||
|         display: inline-block; | ||||
|         margin: 0 8px; | ||||
|         text-decoration: none; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
|   | ||||
							
								
								
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| --- | ||||
| title: "Logo" | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| # Using our logo | ||||
|  | ||||
| As a community, we are proud of our logo. Follow these guidelines to ensure it always looks its best. Our logo follows Google's material design spec and uses the blue interface color. | ||||
|  | ||||
| [Download Logo](https://github.com/home-assistant/assets/tree/master/logo) | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Using the icon | ||||
|  | ||||
| Our icon is a shorter and most used version of our logo. The icon can exist without the wordmark, the wordmark should never exist without the icon. | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Using the right variant | ||||
|  | ||||
| The pretty blue logo with a background shadow, pictured top left, is our primary logo. It should only be used with black, white, and non-duotone photography. | ||||
|  | ||||
| When needed you can use our logo without a shadow, as seen as the second variant.  | ||||
|  | ||||
| The outlined logo should only be used on packaging. | ||||
|  | ||||
| ## Exclusion zone | ||||
|  | ||||
| The logo needs some personal space. It's exclusion zone is equal to a quarter the height of the icon. | ||||
|  | ||||
|  | ||||
							
								
								
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| --- | ||||
| title: "Our story" | ||||
| --- | ||||
|  | ||||
| ## Open source home automation that puts local control and privacy first | ||||
|  | ||||
| Home Assistant is a free and open-source software for home automation that is designed to be the central control system for smart home devices with a focus on local control and privacy. It can be accessed via a web-based user interface, via apps for Android and iOS, or using voice commands via a supported virtual assistant like Google Assistant and Amazon Alexa. | ||||
|  | ||||
| IoT devices and services are supported by modular support for controlling proprietary ecosystems if they provide public access via an Open API for third-party integrations and protocols like Bluetooth, MQTT, Zigbee, and Z-Wave, After the Home Assistant software application is installed as a computer appliance it will act as a central control system for home automation. Information from all entities it sees can be used and controlled from within scripts trigger automations using scheduling and "blueprint" subroutines, e.g. for controlling lighting, climate, entertainment systems, and appliances. | ||||
|  | ||||
| # Open Home | ||||
|  | ||||
| The Open Home is our vision for the smart home. It defines the values that we put at the heart of every decision we make at Home Assistant. It’s woven into our architecture, licensing, community, and everything else. | ||||
|  | ||||
| The Open Home is about privacy, choice, and durability. | ||||
|  | ||||
| ## Privacy | ||||
|  | ||||
| Your home should be your safe space. A place where you can be your true self without having to bother about what the world thinks of you. A place where you don’t need to act differently to avoid an algorithm categorizing your behavior. Privacy for the Open Home means that devices need to work locally. No one else needs to know if you turn on a light bulb or change the thermostat. | ||||
|  | ||||
| It is okay for a product to offer a cloud connection, but it should be extra and opt-in. | ||||
|  | ||||
| ## Choice | ||||
|  | ||||
| Devices in your home gather data about themselves and their surroundings. Your data. Vendors shouldn’t be able to limit your access to your data or limit the interoperability of your devices with the rest of your smart home. | ||||
|  | ||||
| Choice for the Open Home means that devices need to make the gathered data available through local APIs. This avoids vendor lock-in and allows users to create their own smart home with devices from different manufacturers. | ||||
|  | ||||
| ## Durability | ||||
|  | ||||
| If there is one thing that technology firms are very good at, it is launching new products. However, maintaining the products and making sure they keep working is an afterthought for most. The result is that vendors can decide to no longer support your device, crippling its features or even preventing it from working at all. As we install more and more devices in our home, durability is becoming more and more important. We shouldn’t have to buy everything new every couple of years because the manufacturer decided to move on. | ||||
|  | ||||
| Durability for the Open Home means that devices are designed and built to keep working. Not just this year, but for the next decade. | ||||
|  | ||||
| # Our history | ||||
|  | ||||
| The project was started as a Python application by Paulus Schoutsen in September 2013 and first published publicly on GitHub in November 2013. In July 2017, a managed operating system called Hass.io was initially introduced to make it easier use to use Home Assistant on single-board computers like the Raspberry Pi series. Its bundled "supervisor" management system allowed users to manage, backup, and update the local installation and introduced the option to extend the functionality of the software with add-ons. | ||||
|  | ||||
| An optional subscription service was introduced in December 2017 for $5/month to solve the complexities associated with secured remote access, as well as linking to Amazon Alexa and Google Assistant. Nabu Casa, Inc. was formed in September 2018 to take over the subscription service. The company's funding is based solely on revenue from the subscription service. It is used to finance the project's infrastructure and to pay for full-time employees contributing to the project. | ||||
|  | ||||
| In January 2020, branding was adjusted to make it easier to refer to different parts of the project. The main piece of software was renamed to Home Assistant Core, while the full suite of software with the embedded operating system and bundled "supervisor" management system was renamed to Home Assistant. | ||||
| @@ -1,5 +1,6 @@ | ||||
| --- | ||||
| title: Alerts | ||||
| subtitle: An alert displays a short, important message in a way that attracts the user's attention without interrupting the user's task. | ||||
| --- | ||||
|  | ||||
| # Alert `<ha-alert>` | ||||
|   | ||||
| @@ -12,6 +12,98 @@ import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
|     friendly_name: "Alarm", | ||||
|   }), | ||||
|   getEntity("media_player", "livingroom", "playing", { | ||||
|     friendly_name: "Livingroom", | ||||
|   }), | ||||
|   getEntity("media_player", "lounge", "idle", { | ||||
|     friendly_name: "Lounge", | ||||
|     supported_features: 444983, | ||||
|   }), | ||||
|   getEntity("light", "bedroom", "on", { | ||||
|     friendly_name: "Bedroom", | ||||
|   }), | ||||
|   getEntity("switch", "coffee", "off", { | ||||
|     friendly_name: "Coffee", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| const DEVICES = [ | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_1"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_1", | ||||
|     identifiers: [["demo", "volume1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Dishwasher", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_2"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_2", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: null, | ||||
|     name: "Lamp", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: null, | ||||
|     configuration_url: null, | ||||
|     config_entries: ["config_entry_3"], | ||||
|     connections: [], | ||||
|     disabled_by: null, | ||||
|     entry_type: null, | ||||
|     id: "device_3", | ||||
|     identifiers: [["demo", "pwm1"] as [string, string]], | ||||
|     manufacturer: null, | ||||
|     model: null, | ||||
|     name_by_user: "User name", | ||||
|     name: "Technical name", | ||||
|     sw_version: null, | ||||
|     hw_version: null, | ||||
|     via_device_id: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const AREAS = [ | ||||
|   { | ||||
|     area_id: "backyard", | ||||
|     name: "Backyard", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "bedroom", | ||||
|     name: "Bedroom", | ||||
|     picture: null, | ||||
|   }, | ||||
|   { | ||||
|     area_id: "livingroom", | ||||
|     name: "Livingroom", | ||||
|     picture: null, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| const SCHEMAS: { | ||||
|   title: string; | ||||
| @@ -38,6 +130,8 @@ const SCHEMAS: { | ||||
|       select: "Select", | ||||
|       icon: "Icon", | ||||
|       media: "Media", | ||||
|       location: "Location", | ||||
|       entities: "Entities", | ||||
|     }, | ||||
|     schema: [ | ||||
|       { name: "addon", selector: { addon: {} } }, | ||||
| @@ -45,6 +139,7 @@ const SCHEMAS: { | ||||
|       { | ||||
|         name: "Attribute", | ||||
|         selector: { attribute: { entity_id: "" } }, | ||||
|         context: { filter_entity: "entity" }, | ||||
|       }, | ||||
|       { name: "Device", selector: { device: {} } }, | ||||
|       { name: "Duration", selector: { duration: {} } }, | ||||
| @@ -75,6 +170,14 @@ const SCHEMAS: { | ||||
|           media: {}, | ||||
|         }, | ||||
|       }, | ||||
|       { | ||||
|         name: "location", | ||||
|         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||
|       }, | ||||
|       { | ||||
|         name: "entities", | ||||
|         selector: { entity: { multiple: true } }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
| @@ -315,9 +418,10 @@ class DemoHaForm extends LitElement { | ||||
|     const hass = provideHass(this); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("config", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     mockEntityRegistry(hass); | ||||
|     mockDeviceRegistry(hass); | ||||
|     mockAreaRegistry(hass); | ||||
|     mockDeviceRegistry(hass, DEVICES); | ||||
|     mockAreaRegistry(hass, AREAS); | ||||
|     mockHassioSupervisor(hass); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -146,6 +146,8 @@ const SCHEMAS: { | ||||
|       }, | ||||
|       boolean: { name: "Boolean", selector: { boolean: {} } }, | ||||
|       time: { name: "Time", selector: { time: {} } }, | ||||
|       date: { name: "Date", selector: { date: {} } }, | ||||
|       datetime: { name: "Date Time", selector: { datetime: {} } }, | ||||
|       action: { name: "Action", selector: { action: {} } }, | ||||
|       text: { | ||||
|         name: "Text", | ||||
| @@ -162,12 +164,44 @@ const SCHEMAS: { | ||||
|         }, | ||||
|       }, | ||||
|       object: { name: "Object", selector: { object: {} } }, | ||||
|       select_radio: { | ||||
|         name: "Select (Radio)", | ||||
|         selector: { select: { options: ["Option 1", "Option 2"] } }, | ||||
|       }, | ||||
|       select: { | ||||
|         name: "Select", | ||||
|         selector: { select: { options: ["Option 1", "Option 2"] } }, | ||||
|         selector: { | ||||
|           select: { | ||||
|             options: [ | ||||
|               "Option 1", | ||||
|               "Option 2", | ||||
|               "Option 3", | ||||
|               "Option 4", | ||||
|               "Option 5", | ||||
|               "Option 6", | ||||
|             ], | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       icon: { name: "Icon", selector: { icon: {} } }, | ||||
|       media: { name: "Media", selector: { media: {} } }, | ||||
|       location: { name: "Location", selector: { location: {} } }, | ||||
|       location_radius: { | ||||
|         name: "Location with radius", | ||||
|         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||
|       }, | ||||
|       color_temp: { | ||||
|         name: "Color Temperature", | ||||
|         selector: { color_temp: {} }, | ||||
|       }, | ||||
|       color_rgb: { name: "Color", selector: { color_rgb: {} } }, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     name: "Multiples", | ||||
|     input: { | ||||
|       entity: { name: "Entity", selector: { entity: { multiple: true } } }, | ||||
|       device: { name: "Device", selector: { device: { multiple: true } } }, | ||||
|     }, | ||||
|   }, | ||||
| ]; | ||||
| @@ -279,7 +313,7 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | ||||
|           can_play: true, | ||||
|           can_expand: false, | ||||
|           children_media_class: null, | ||||
|           thumbnail: null, | ||||
|           thumbnail: "https://brands.home-assistant.io/_/image/logo.png", | ||||
|         }, | ||||
|         { | ||||
|           title: "movie.mp4", | ||||
|   | ||||
| @@ -2,6 +2,8 @@ | ||||
| title: Editing design.home-assistant.io | ||||
| --- | ||||
|  | ||||
|  | ||||
|  | ||||
| # How to edit design.home-assistant.io | ||||
|  | ||||
| All pages are stored in [the pages folder][pages-folder] on GitHub. Pages are grouped in a folder per sidebar section. Each page can contain a `<page name>.markdown` description file, a `<page name>.ts` demo file or both. If both are defined the description is rendered first. The description can contain metadata to specify the title of the page. | ||||
| @@ -41,15 +43,12 @@ import { html, css, LitElement } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
|  | ||||
|  | ||||
| @customElement("demo-user-experience-usability") | ||||
| export class DemoUserExperienceUsability extends LitElement { | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-card> | ||||
|         <div class="card-content"> | ||||
|           Hello world! | ||||
|         </div> | ||||
|         <div class="card-content">Hello world!</div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|   | ||||
| @@ -188,6 +188,7 @@ const createEntityRegistryEntries = ( | ||||
|     device_id: "mock-device-id", | ||||
|     area_id: null, | ||||
|     disabled_by: null, | ||||
|     hidden_by: null, | ||||
|     entity_category: null, | ||||
|     entity_id: "binary_sensor.updater", | ||||
|     name: null, | ||||
|   | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Update | ||||
| --- | ||||
							
								
								
									
										140
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,140 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { | ||||
|   UPDATE_SUPPORT_BACKUP, | ||||
|   UPDATE_SUPPORT_PROGRESS, | ||||
|   UPDATE_SUPPORT_INSTALL, | ||||
| } from "../../../../src/data/update"; | ||||
| import "../../../../src/dialogs/more-info/more-info-content"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { | ||||
|   MockHomeAssistant, | ||||
|   provideHass, | ||||
| } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-more-infos"; | ||||
|  | ||||
| const base_attributes = { | ||||
|   title: "Awesome", | ||||
|   current_version: "1.2.2", | ||||
|   latest_version: "1.2.3", | ||||
|   release_url: "https://home-assistant.io", | ||||
|   supported_features: UPDATE_SUPPORT_INSTALL, | ||||
|   skipped_version: null, | ||||
|   in_progress: false, | ||||
|   release_summary: | ||||
|     "Lorem ipsum dolor sit amet, consectetur adipiscing elit. In nec metus aliquet, porta mi ut, ultrices odio. Etiam egestas orci tellus, non semper metus blandit tincidunt. Praesent elementum turpis vel tempor pharetra. Sed quis cursus diam. Proin sem justo.", | ||||
| }; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("update", "update1", "on", { | ||||
|     ...base_attributes, | ||||
|     friendly_name: "Update", | ||||
|   }), | ||||
|   getEntity("update", "update2", "on", { | ||||
|     ...base_attributes, | ||||
|     title: null, | ||||
|     friendly_name: "Update without title", | ||||
|   }), | ||||
|   getEntity("update", "update3", "on", { | ||||
|     ...base_attributes, | ||||
|     release_url: null, | ||||
|     friendly_name: "Update without release_url", | ||||
|   }), | ||||
|   getEntity("update", "update4", "on", { | ||||
|     ...base_attributes, | ||||
|     release_summary: null, | ||||
|     friendly_name: "Update without release_summary", | ||||
|   }), | ||||
|   getEntity("update", "update5", "off", { | ||||
|     ...base_attributes, | ||||
|     current_version: "1.2.3", | ||||
|     friendly_name: "No update", | ||||
|   }), | ||||
|   getEntity("update", "update6", "off", { | ||||
|     ...base_attributes, | ||||
|     skipped_version: "1.2.3", | ||||
|     friendly_name: "Skipped version", | ||||
|   }), | ||||
|   getEntity("update", "update7", "on", { | ||||
|     ...base_attributes, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_BACKUP, | ||||
|     friendly_name: "With backup support", | ||||
|   }), | ||||
|   getEntity("update", "update8", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: true, | ||||
|     friendly_name: "With true in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update9", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 25, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 25 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update10", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 50, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 50 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update11", "on", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 75, | ||||
|     supported_features: | ||||
|       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||
|     friendly_name: "With 75 in_progress", | ||||
|   }), | ||||
|   getEntity("update", "update12", "unavailable", { | ||||
|     ...base_attributes, | ||||
|     in_progress: 50, | ||||
|     friendly_name: "Unavailable", | ||||
|   }), | ||||
|   getEntity("update", "update13", "on", { | ||||
|     ...base_attributes, | ||||
|     supported_features: 0, | ||||
|     friendly_name: "No install support", | ||||
|   }), | ||||
|   getEntity("update", "update14", "off", { | ||||
|     ...base_attributes, | ||||
|     current_version: null, | ||||
|     friendly_name: "Update without current_version", | ||||
|   }), | ||||
|   getEntity("update", "update15", "off", { | ||||
|     ...base_attributes, | ||||
|     latest_version: null, | ||||
|     friendly_name: "Update without latest_version", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-more-info-update") | ||||
| class DemoMoreInfoUpdate extends LitElement { | ||||
|   @property() public hass!: MockHomeAssistant; | ||||
|  | ||||
|   @query("demo-more-infos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <demo-more-infos | ||||
|         .hass=${this.hass} | ||||
|         .entities=${ENTITIES.map((ent) => ent.entityId)} | ||||
|       ></demo-more-infos> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-more-info-update": DemoMoreInfoUpdate; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| --- | ||||
| title: "User types" | ||||
| --- | ||||
|  | ||||
| We have defined three user types for Home Assistant. They are a lean segmentation of users that helps us make decisions throughout the product. User types differ from traditional personas in that the segmentation criteria aren’t demographic and don’t personify a group into a single character with a fictitious background story.  | ||||
|  | ||||
| # Outgrowers | ||||
|  | ||||
| Users that outgrow big tech smart home solutions. It just needs to work with easy setup via an app. | ||||
|  | ||||
| # Tinkerers | ||||
|  | ||||
| Technoid users in home networking and development that know how to code. | ||||
|  | ||||
| # Questioner | ||||
|  | ||||
| Users who want more advanced home automation, but need support to make it work. | ||||
| @@ -110,8 +110,6 @@ class HassioAddonStore extends LitElement { | ||||
|               <div class="search"> | ||||
|                 <search-input | ||||
|                   .hass=${this.hass} | ||||
|                   no-label-float | ||||
|                   no-underline | ||||
|                   .filter=${this._filter} | ||||
|                   @value-changed=${this._filterChanged} | ||||
|                 ></search-input> | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| import { mdiFolderUpload } from "@mdi/js"; | ||||
| import "@polymer/paper-input/paper-input-container"; | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
|   | ||||
| @@ -80,8 +80,6 @@ class HassioHardwareDialog extends LitElement { | ||||
|           ></ha-icon-button> | ||||
|           <search-input | ||||
|             .hass=${this.hass} | ||||
|             dialogInitialFocus | ||||
|             no-label-float | ||||
|             .filter=${this._filter} | ||||
|             @value-changed=${this._handleSearchChange} | ||||
|             .label=${this._dialogParams.supervisor.localize( | ||||
|   | ||||
| @@ -106,6 +106,9 @@ class HassioRepositoriesDialog extends LitElement { | ||||
|                     </paper-item-body> | ||||
|                     <div class="delete"> | ||||
|                       <ha-icon-button | ||||
|                         .label=${this._dialogParams!.supervisor.localize( | ||||
|                           "dialog.repositories.remove" | ||||
|                         )} | ||||
|                         .disabled=${usedRepositories.includes(repo.slug)} | ||||
|                         .slug=${repo.slug} | ||||
|                         .path=${usedRepositories.includes(repo.slug) | ||||
|   | ||||
| @@ -1,9 +1,12 @@ | ||||
| // Compat needs to be first import | ||||
| import "../../src/resources/compatibility"; | ||||
| import { setCancelSyntheticClickEvents } from "@polymer/polymer/lib/utils/settings"; | ||||
| import "../../src/resources/roboto"; | ||||
| import "../../src/resources/safari-14-attachshadow-patch"; | ||||
| import "./hassio-main"; | ||||
|  | ||||
| setCancelSyntheticClickEvents(false); | ||||
|  | ||||
| const styleEl = document.createElement("style"); | ||||
| styleEl.innerHTML = ` | ||||
| body { | ||||
|   | ||||
| @@ -121,7 +121,8 @@ export class HassioMain extends SupervisorBaseElement { | ||||
|       this.parentElement, | ||||
|       this.hass.themes, | ||||
|       themeName, | ||||
|       themeSettings | ||||
|       themeSettings, | ||||
|       true | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -45,7 +45,6 @@ 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 { addonArchIsSupported, extractChangelog } from "../util/addon"; | ||||
|  | ||||
| @@ -55,6 +54,12 @@ declare global { | ||||
|   } | ||||
| } | ||||
|  | ||||
| const SUPERVISOR_UPDATE_NAMES = { | ||||
|   core: "Home Assistant Core", | ||||
|   os: "Home Assistant Operating System", | ||||
|   supervisor: "Home Assistant Supervisor", | ||||
| }; | ||||
|  | ||||
| type updateType = "os" | "supervisor" | "core" | "addon"; | ||||
|  | ||||
| const changelogUrl = ( | ||||
|   | ||||
| @@ -72,14 +72,13 @@ | ||||
|     "@material/mwc-textfield": "0.25.3", | ||||
|     "@material/mwc-top-app-bar-fixed": "^0.25.3", | ||||
|     "@material/top-app-bar": "14.0.0-canary.261f2db59.0", | ||||
|     "@mdi/js": "6.5.95", | ||||
|     "@mdi/svg": "6.5.95", | ||||
|     "@mdi/js": "6.6.95", | ||||
|     "@mdi/svg": "6.6.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-resizable-behavior": "^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", | ||||
| @@ -109,7 +108,7 @@ | ||||
|     "fuse.js": "^6.0.0", | ||||
|     "google-timezones-json": "^1.0.2", | ||||
|     "hls.js": "^1.1.5", | ||||
|     "home-assistant-js-websocket": "^6.0.1", | ||||
|     "home-assistant-js-websocket": "^7.0.1", | ||||
|     "idb-keyval": "^5.1.3", | ||||
|     "intl-messageformat": "^9.9.1", | ||||
|     "js-yaml": "^4.1.0", | ||||
| @@ -136,7 +135,6 @@ | ||||
|     "vis-network": "^8.5.4", | ||||
|     "vue": "^2.6.12", | ||||
|     "vue2-daterange-picker": "^0.5.1", | ||||
|     "web-animations-js": "^2.3.2", | ||||
|     "workbox-cacheable-response": "^6.4.2", | ||||
|     "workbox-core": "^6.4.2", | ||||
|     "workbox-expiration": "^6.4.2", | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| [metadata] | ||||
| name         = home-assistant-frontend | ||||
| version      = 20220226.0 | ||||
| version      = 20220322.0 | ||||
| author       = The Home Assistant Authors | ||||
| author_email = hello@home-assistant.io | ||||
| license      = Apache-2.0 | ||||
|   | ||||
							
								
								
									
										7
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,7 +0,0 @@ | ||||
| """ | ||||
| Entry point for setuptools. Required for editable installs. | ||||
| TODO: Remove file after updating to pip 21.3 | ||||
| """ | ||||
| from setuptools import setup | ||||
|  | ||||
| setup() | ||||
| @@ -101,13 +101,19 @@ class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | ||||
|     this._fetchAuthProviders(); | ||||
|  | ||||
|     if (matchMedia("(prefers-color-scheme: dark)").matches) { | ||||
|       applyThemesOnElement(document.documentElement, { | ||||
|         default_theme: "default", | ||||
|         default_dark_theme: null, | ||||
|         themes: {}, | ||||
|         darkMode: true, | ||||
|         theme: "default", | ||||
|       }); | ||||
|       applyThemesOnElement( | ||||
|         document.documentElement, | ||||
|         { | ||||
|           default_theme: "default", | ||||
|           default_dark_theme: null, | ||||
|           themes: {}, | ||||
|           darkMode: true, | ||||
|           theme: "default", | ||||
|         }, | ||||
|         undefined, | ||||
|         undefined, | ||||
|         true | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (!this.redirectUri) { | ||||
|   | ||||
| @@ -187,6 +187,7 @@ export const DOMAINS_WITH_MORE_INFO = [ | ||||
|   "scene", | ||||
|   "sun", | ||||
|   "timer", | ||||
|   "update", | ||||
|   "vacuum", | ||||
|   "water_heater", | ||||
|   "weather", | ||||
| @@ -200,6 +201,7 @@ export const DOMAINS_HIDE_DEFAULT_MORE_INFO = [ | ||||
|   "input_text", | ||||
|   "number", | ||||
|   "scene", | ||||
|   "update", | ||||
|   "select", | ||||
| ]; | ||||
|  | ||||
|   | ||||
| @@ -31,11 +31,12 @@ export const applyThemesOnElement = ( | ||||
|   element, | ||||
|   themes: HomeAssistant["themes"], | ||||
|   selectedTheme?: string, | ||||
|   themeSettings?: Partial<HomeAssistant["selectedTheme"]> | ||||
|   themeSettings?: Partial<HomeAssistant["selectedTheme"]>, | ||||
|   main?: boolean | ||||
| ) => { | ||||
|   // If there is no explicitly desired theme provided, we automatically | ||||
|   // If there is no explicitly desired theme provided, and the element is the main element we automatically | ||||
|   // use the active one from `themes`. | ||||
|   const themeToApply = selectedTheme || themes.theme; | ||||
|   const themeToApply = selectedTheme || (main ? themes.theme : undefined); | ||||
|  | ||||
|   // If there is no explicitly desired dark mode provided, we automatically | ||||
|   // use the active one from `themes`. | ||||
| @@ -47,7 +48,7 @@ export const applyThemesOnElement = ( | ||||
|   let cacheKey = themeToApply; | ||||
|   let themeRules: Partial<ThemeVars> = {}; | ||||
|  | ||||
|   if (darkMode) { | ||||
|   if (themeToApply && darkMode) { | ||||
|     cacheKey = `${cacheKey}__dark`; | ||||
|     themeRules = { ...darkStyles }; | ||||
|   } | ||||
|   | ||||
| @@ -1,12 +1,18 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | ||||
| import { FrontendLocaleData } from "../../data/translation"; | ||||
| import { | ||||
|   updateIsInstalling, | ||||
|   UpdateEntity, | ||||
|   UPDATE_SUPPORT_PROGRESS, | ||||
| } from "../../data/update"; | ||||
| import { formatDate } from "../datetime/format_date"; | ||||
| import { formatDateTime } from "../datetime/format_date_time"; | ||||
| import { formatTime } from "../datetime/format_time"; | ||||
| import { formatNumber, isNumericState } from "../number/format_number"; | ||||
| import { LocalizeFunc } from "../translations/localize"; | ||||
| import { computeStateDomain } from "./compute_state_domain"; | ||||
| import { supportsFeature } from "./supports-feature"; | ||||
|  | ||||
| export const computeStateDisplay = ( | ||||
|   localize: LocalizeFunc, | ||||
| @@ -130,6 +136,28 @@ export const computeStateDisplay = ( | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (domain === "update") { | ||||
|     // When updating, and entity does not support % show "Installing" | ||||
|     // When updating, and entity does support % show "Installing (xx%)" | ||||
|     // When update available, show the version | ||||
|     // When the latest version is skipped, show the latest version | ||||
|     // When update is not available, show "Up-to-date" | ||||
|     // When update is not available and there is no latest_version show "Unavailable" | ||||
|     return compareState === "on" | ||||
|       ? updateIsInstalling(stateObj as UpdateEntity) | ||||
|         ? supportsFeature(stateObj, UPDATE_SUPPORT_PROGRESS) | ||||
|           ? localize("ui.card.update.installing_with_progress", { | ||||
|               progress: stateObj.attributes.in_progress, | ||||
|             }) | ||||
|           : localize("ui.card.update.installing") | ||||
|         : stateObj.attributes.latest_version | ||||
|       : stateObj.attributes.skipped_version === | ||||
|         stateObj.attributes.latest_version | ||||
|       ? stateObj.attributes.latest_version ?? | ||||
|         localize("state.default.unavailable") | ||||
|       : localize("ui.card.update.up_to_date"); | ||||
|   } | ||||
|  | ||||
|   return ( | ||||
|     // Return device class translation | ||||
|     (stateObj.attributes.device_class && | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import type { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { computeDomain } from "./compute_domain"; | ||||
|  | ||||
| export const computeStateDomain = (stateObj: HassEntity) => | ||||
|   | ||||
| @@ -9,11 +9,10 @@ import { | ||||
|   mdiCast, | ||||
|   mdiCastConnected, | ||||
|   mdiClock, | ||||
|   mdiEmoticonDead, | ||||
|   mdiFlash, | ||||
|   mdiGestureTapButton, | ||||
|   mdiLanConnect, | ||||
|   mdiLanDisconnect, | ||||
|   mdiLightSwitch, | ||||
|   mdiLock, | ||||
|   mdiLockAlert, | ||||
|   mdiLockClock, | ||||
| @@ -22,16 +21,16 @@ import { | ||||
|   mdiPowerPlug, | ||||
|   mdiPowerPlugOff, | ||||
|   mdiRestart, | ||||
|   mdiSleep, | ||||
|   mdiTimerSand, | ||||
|   mdiToggleSwitch, | ||||
|   mdiToggleSwitchOff, | ||||
|   mdiCheckCircleOutline, | ||||
|   mdiCloseCircleOutline, | ||||
|   mdiWeatherNight, | ||||
|   mdiZWave, | ||||
|   mdiPackage, | ||||
|   mdiPackageDown, | ||||
| } from "@mdi/js"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { updateIsInstalling, UpdateEntity } from "../../data/update"; | ||||
| /** | ||||
|  * Return the icon to be used for a domain. | ||||
|  * | ||||
| @@ -112,19 +111,7 @@ export const domainIcon = ( | ||||
|         case "switch": | ||||
|           return compareState === "on" ? mdiToggleSwitch : mdiToggleSwitchOff; | ||||
|         default: | ||||
|           return mdiFlash; | ||||
|       } | ||||
|  | ||||
|     case "zwave": | ||||
|       switch (compareState) { | ||||
|         case "dead": | ||||
|           return mdiEmoticonDead; | ||||
|         case "sleeping": | ||||
|           return mdiSleep; | ||||
|         case "initializing": | ||||
|           return mdiTimerSand; | ||||
|         default: | ||||
|           return mdiZWave; | ||||
|           return mdiLightSwitch; | ||||
|       } | ||||
|  | ||||
|     case "sensor": { | ||||
| @@ -149,6 +136,13 @@ export const domainIcon = ( | ||||
|       return stateObj?.state === "above_horizon" | ||||
|         ? FIXED_DOMAIN_ICONS[domain] | ||||
|         : mdiWeatherNight; | ||||
|  | ||||
|     case "update": | ||||
|       return compareState === "on" | ||||
|         ? updateIsInstalling(stateObj as UpdateEntity) | ||||
|           ? mdiPackageDown | ||||
|           : mdiPackageUp | ||||
|         : mdiPackage; | ||||
|   } | ||||
|  | ||||
|   if (domain in FIXED_DOMAIN_ICONS) { | ||||
|   | ||||
							
								
								
									
										4
									
								
								src/common/string/is_ip_address.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/common/string/is_ip_address.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| const regexp = | ||||
|   /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/; | ||||
|  | ||||
| export const isIPAddress = (input: string): boolean => regexp.test(input); | ||||
| @@ -70,9 +70,6 @@ export const iconColorCSS = css` | ||||
|   } | ||||
|  | ||||
|   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-state-icon[data-state="unavailable"] { | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| import "@material/mwc-button"; | ||||
| import type { Button } from "@material/mwc-button"; | ||||
| import { mdiAlertOctagram, mdiCheckBold } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import "../ha-circular-progress"; | ||||
| import "../ha-svg-icon"; | ||||
|  | ||||
| @customElement("ha-progress-button") | ||||
| export class HaProgressButton extends LitElement { | ||||
| @@ -12,38 +13,53 @@ export class HaProgressButton extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public raised = false; | ||||
|  | ||||
|   @query("mwc-button", true) private _button?: Button; | ||||
|   @state() private _result?: "success" | "error"; | ||||
|  | ||||
|   public render(): TemplateResult { | ||||
|     const overlay = this._result || this.progress; | ||||
|     return html` | ||||
|       <mwc-button | ||||
|         ?raised=${this.raised} | ||||
|         .disabled=${this.disabled || this.progress} | ||||
|         @click=${this._buttonTapped} | ||||
|         class=${this._result || ""} | ||||
|       > | ||||
|         <slot></slot> | ||||
|       </mwc-button> | ||||
|       ${this.progress | ||||
|         ? html`<div class="progress"> | ||||
|             <ha-circular-progress size="small" active></ha-circular-progress> | ||||
|           </div>` | ||||
|         : ""} | ||||
|       ${!overlay | ||||
|         ? "" | ||||
|         : html` | ||||
|             <div class="progress"> | ||||
|               ${this._result === "success" | ||||
|                 ? html`<ha-svg-icon .path=${mdiCheckBold}></ha-svg-icon>` | ||||
|                 : this._result === "error" | ||||
|                 ? html`<ha-svg-icon .path=${mdiAlertOctagram}></ha-svg-icon>` | ||||
|                 : this.progress | ||||
|                 ? html` | ||||
|                     <ha-circular-progress | ||||
|                       size="small" | ||||
|                       active | ||||
|                     ></ha-circular-progress> | ||||
|                   ` | ||||
|                 : ""} | ||||
|             </div> | ||||
|           `} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   public actionSuccess(): void { | ||||
|     this._tempClass("success"); | ||||
|     this._setResult("success"); | ||||
|   } | ||||
|  | ||||
|   public actionError(): void { | ||||
|     this._tempClass("error"); | ||||
|     this._setResult("error"); | ||||
|   } | ||||
|  | ||||
|   private _tempClass(className: string): void { | ||||
|     this._button!.classList.add(className); | ||||
|   private _setResult(result: "success" | "error"): void { | ||||
|     this._result = result; | ||||
|     setTimeout(() => { | ||||
|       this._button!.classList.remove(className); | ||||
|     }, 1000); | ||||
|       this._result = undefined; | ||||
|     }, 2000); | ||||
|   } | ||||
|  | ||||
|   private _buttonTapped(ev: Event): void { | ||||
| @@ -69,6 +85,7 @@ export class HaProgressButton extends LitElement { | ||||
|         background-color: var(--success-color); | ||||
|         transition: none; | ||||
|         border-radius: 4px; | ||||
|         pointer-events: none; | ||||
|       } | ||||
|  | ||||
|       mwc-button[raised].success { | ||||
| @@ -81,6 +98,7 @@ export class HaProgressButton extends LitElement { | ||||
|         background-color: var(--error-color); | ||||
|         transition: none; | ||||
|         border-radius: 4px; | ||||
|         pointer-events: none; | ||||
|       } | ||||
|  | ||||
|       mwc-button[raised].error { | ||||
| @@ -89,13 +107,21 @@ export class HaProgressButton extends LitElement { | ||||
|       } | ||||
|  | ||||
|       .progress { | ||||
|         bottom: 0; | ||||
|         margin-top: 4px; | ||||
|         bottom: 4px; | ||||
|         position: absolute; | ||||
|         text-align: center; | ||||
|         top: 0; | ||||
|         top: 4px; | ||||
|         width: 100%; | ||||
|       } | ||||
|  | ||||
|       ha-svg-icon { | ||||
|         color: white; | ||||
|       } | ||||
|  | ||||
|       mwc-button.success slot, | ||||
|       mwc-button.error slot { | ||||
|         visibility: hidden; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -115,6 +115,9 @@ class DateRangePickerElement extends WrappedElement { | ||||
|             color: var(--primary-text-color); | ||||
|             min-width: initial !important; | ||||
|           } | ||||
|           .daterangepicker:before { | ||||
|             display: none; | ||||
|           } | ||||
|           .daterangepicker:after { | ||||
|             border-bottom: 6px solid var(--card-background-color); | ||||
|           } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { PolymerChangedEvent } from "../../polymer-types"; | ||||
| @@ -116,6 +116,12 @@ class HaDevicesPicker extends LitElement { | ||||
|  | ||||
|     this._updateDevices([...currentDevices, toAdd]); | ||||
|   } | ||||
|  | ||||
|   static override styles = css` | ||||
|     div { | ||||
|       margin-top: 8px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -46,11 +46,29 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|   @property({ type: Array, attribute: "include-unit-of-measurement" }) | ||||
|   public includeUnitOfMeasurement?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of allowed entities to show. Will ignore all other filters. | ||||
|    * @type {Array} | ||||
|    * @attr include-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "include-entities" }) | ||||
|   public includeEntities?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of entities to be excluded. | ||||
|    * @type {Array} | ||||
|    * @attr exclude-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "exclude-entities" }) | ||||
|   public excludeEntities?: string[]; | ||||
|  | ||||
|   @property({ attribute: "picked-entity-label" }) | ||||
|   public pickedEntityLabel?: string; | ||||
|  | ||||
|   @property({ attribute: "pick-entity-label" }) public pickEntityLabel?: string; | ||||
|  | ||||
|   @property() public entityFilter?: HaEntityPickerEntityFilterFunc; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (!this.hass) { | ||||
|       return html``; | ||||
| @@ -67,6 +85,8 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|               .hass=${this.hass} | ||||
|               .includeDomains=${this.includeDomains} | ||||
|               .excludeDomains=${this.excludeDomains} | ||||
|               .includeEntities=${this.includeEntities} | ||||
|               .excludeEntities=${this.excludeEntities} | ||||
|               .includeDeviceClasses=${this.includeDeviceClasses} | ||||
|               .includeUnitOfMeasurement=${this.includeUnitOfMeasurement} | ||||
|               .entityFilter=${this._entityFilter} | ||||
| @@ -82,6 +102,8 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|           .hass=${this.hass} | ||||
|           .includeDomains=${this.includeDomains} | ||||
|           .excludeDomains=${this.excludeDomains} | ||||
|           .includeEntities=${this.includeEntities} | ||||
|           .excludeEntities=${this.excludeEntities} | ||||
|           .includeDeviceClasses=${this.includeDeviceClasses} | ||||
|           .includeUnitOfMeasurement=${this.includeUnitOfMeasurement} | ||||
|           .entityFilter=${this._entityFilter} | ||||
| @@ -94,7 +116,9 @@ class HaEntitiesPickerLight extends LitElement { | ||||
|  | ||||
|   private _entityFilter: HaEntityPickerEntityFilterFunc = ( | ||||
|     stateObj: HassEntity | ||||
|   ) => !this.value || !this.value.includes(stateObj.entity_id); | ||||
|   ) => | ||||
|     (!this.value || !this.value.includes(stateObj.entity_id)) && | ||||
|     (!this.entityFilter || this.entityFilter(stateObj)); | ||||
|  | ||||
|   private get _currentEntities() { | ||||
|     return this.value || []; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { computeDomain } from "../../common/entity/compute_domain"; | ||||
| import { computeStateName } from "../../common/entity/compute_state_name"; | ||||
| import { caseInsensitiveStringCompare } from "../../common/string/compare"; | ||||
| import { PolymerChangedEvent } from "../../polymer-types"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import "../ha-combo-box"; | ||||
| @@ -15,18 +16,22 @@ import "../ha-icon-button"; | ||||
| import "../ha-svg-icon"; | ||||
| import "./state-badge"; | ||||
|  | ||||
| interface HassEntityWithCachedName extends HassEntity { | ||||
|   friendly_name: string; | ||||
|   id: string; | ||||
| } | ||||
|  | ||||
| export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean; | ||||
|  | ||||
| // eslint-disable-next-line lit/prefer-static-styles | ||||
| const rowRenderer: ComboBoxLitRenderer<HassEntity & { friendly_name: string }> = | ||||
|   (item) => | ||||
|     html`<mwc-list-item graphic="avatar" .twoline=${!!item.entity_id}> | ||||
|       ${item.state | ||||
|         ? html`<state-badge slot="graphic" .stateObj=${item}></state-badge>` | ||||
|         : ""} | ||||
|       <span>${item.friendly_name}</span> | ||||
|       <span slot="secondary">${item.entity_id}</span> | ||||
|     </mwc-list-item>`; | ||||
| const rowRenderer: ComboBoxLitRenderer<HassEntityWithCachedName> = (item) => | ||||
|   html`<mwc-list-item graphic="avatar" .twoline=${!!item.entity_id}> | ||||
|     ${item.state | ||||
|       ? html`<state-badge slot="graphic" .stateObj=${item}></state-badge>` | ||||
|       : ""} | ||||
|     <span>${item.friendly_name}</span> | ||||
|     <span slot="secondary">${item.entity_id}</span> | ||||
|   </mwc-list-item>`; | ||||
| @customElement("ha-entity-picker") | ||||
| export class HaEntityPicker extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
| @@ -74,8 +79,27 @@ export class HaEntityPicker extends LitElement { | ||||
|   @property({ type: Array, attribute: "include-unit-of-measurement" }) | ||||
|   public includeUnitOfMeasurement?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of allowed entities to show. Will ignore all other filters. | ||||
|    * @type {Array} | ||||
|    * @attr include-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "include-entities" }) | ||||
|   public includeEntities?: string[]; | ||||
|  | ||||
|   /** | ||||
|    * List of entities to be excluded. | ||||
|    * @type {Array} | ||||
|    * @attr exclude-entities | ||||
|    */ | ||||
|   @property({ type: Array, attribute: "exclude-entities" }) | ||||
|   public excludeEntities?: string[]; | ||||
|  | ||||
|   @property() public entityFilter?: HaEntityPickerEntityFilterFunc; | ||||
|  | ||||
|   @property({ attribute: "item-value-path" }) public itemValuePath = | ||||
|     "entity_id"; | ||||
|  | ||||
|   @property({ type: Boolean }) public hideClearIcon = false; | ||||
|  | ||||
|   @state() private _opened = false; | ||||
| @@ -96,7 +120,7 @@ export class HaEntityPicker extends LitElement { | ||||
|  | ||||
|   private _initedStates = false; | ||||
|  | ||||
|   private _states: HassEntity[] = []; | ||||
|   private _states: HassEntityWithCachedName[] = []; | ||||
|  | ||||
|   private _getStates = memoizeOne( | ||||
|     ( | ||||
| @@ -106,9 +130,11 @@ export class HaEntityPicker extends LitElement { | ||||
|       excludeDomains: this["excludeDomains"], | ||||
|       entityFilter: this["entityFilter"], | ||||
|       includeDeviceClasses: this["includeDeviceClasses"], | ||||
|       includeUnitOfMeasurement: this["includeUnitOfMeasurement"] | ||||
|     ) => { | ||||
|       let states: HassEntity[] = []; | ||||
|       includeUnitOfMeasurement: this["includeUnitOfMeasurement"], | ||||
|       includeEntities: this["includeEntities"], | ||||
|       excludeEntities: this["excludeEntities"] | ||||
|     ): HassEntityWithCachedName[] => { | ||||
|       let states: HassEntityWithCachedName[] = []; | ||||
|  | ||||
|       if (!hass) { | ||||
|         return []; | ||||
| @@ -122,7 +148,8 @@ export class HaEntityPicker extends LitElement { | ||||
|             state: "", | ||||
|             last_changed: "", | ||||
|             last_updated: "", | ||||
|             context: { id: "", user_id: null }, | ||||
|             id: "", | ||||
|             context: { id: "", user_id: null, parent_id: null }, | ||||
|             friendly_name: this.hass!.localize( | ||||
|               "ui.components.entity.entity-picker.no_entities" | ||||
|             ), | ||||
| @@ -136,6 +163,35 @@ export class HaEntityPicker extends LitElement { | ||||
|         ]; | ||||
|       } | ||||
|  | ||||
|       if (includeEntities) { | ||||
|         entityIds = entityIds.filter((entityId) => | ||||
|           this.includeEntities!.includes(entityId) | ||||
|         ); | ||||
|  | ||||
|         return entityIds | ||||
|           .map((key) => { | ||||
|             const stateObj = hass!.states[key]; | ||||
|  | ||||
|             return { | ||||
|               ...stateObj, | ||||
|               friendly_name: computeStateName(stateObj) || key, | ||||
|               id: stateObj.context.id, | ||||
|             }; | ||||
|           }) | ||||
|           .sort((entityA, entityB) => | ||||
|             caseInsensitiveStringCompare( | ||||
|               entityA.friendly_name, | ||||
|               entityB.friendly_name | ||||
|             ) | ||||
|           ); | ||||
|       } | ||||
|  | ||||
|       if (excludeEntities) { | ||||
|         entityIds = entityIds.filter( | ||||
|           (entityId) => !excludeEntities!.includes(entityId) | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       if (includeDomains) { | ||||
|         entityIds = entityIds.filter((eid) => | ||||
|           includeDomains.includes(computeDomain(eid)) | ||||
| @@ -148,10 +204,22 @@ export class HaEntityPicker extends LitElement { | ||||
|         ); | ||||
|       } | ||||
|  | ||||
|       states = entityIds.sort().map((key) => ({ | ||||
|         ...hass!.states[key], | ||||
|         friendly_name: computeStateName(hass!.states[key]) || key, | ||||
|       })); | ||||
|       states = entityIds | ||||
|         .map((key) => { | ||||
|           const stateObj = hass!.states[key]; | ||||
|  | ||||
|           return { | ||||
|             ...stateObj, | ||||
|             friendly_name: computeStateName(stateObj) || key, | ||||
|             id: stateObj.context?.id, | ||||
|           }; | ||||
|         }) | ||||
|         .sort((entityA, entityB) => | ||||
|           caseInsensitiveStringCompare( | ||||
|             entityA.friendly_name, | ||||
|             entityB.friendly_name | ||||
|           ) | ||||
|         ); | ||||
|  | ||||
|       if (includeDeviceClasses) { | ||||
|         states = states.filter( | ||||
| @@ -190,7 +258,8 @@ export class HaEntityPicker extends LitElement { | ||||
|             state: "", | ||||
|             last_changed: "", | ||||
|             last_updated: "", | ||||
|             context: { id: "", user_id: null }, | ||||
|             id: "", | ||||
|             context: { id: "", user_id: null, parent_id: null }, | ||||
|             friendly_name: this.hass!.localize( | ||||
|               "ui.components.entity.entity-picker.no_match" | ||||
|             ), | ||||
| @@ -228,7 +297,9 @@ export class HaEntityPicker extends LitElement { | ||||
|         this.excludeDomains, | ||||
|         this.entityFilter, | ||||
|         this.includeDeviceClasses, | ||||
|         this.includeUnitOfMeasurement | ||||
|         this.includeUnitOfMeasurement, | ||||
|         this.includeEntities, | ||||
|         this.excludeEntities | ||||
|       ); | ||||
|       if (this._initedStates) { | ||||
|         (this.comboBox as any).filteredItems = this._states; | ||||
| @@ -240,8 +311,8 @@ export class HaEntityPicker extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-combo-box | ||||
|         item-value-path="entity_id" | ||||
|         item-label-path="friendly_name" | ||||
|         .itemValuePath=${this.itemValuePath} | ||||
|         .hass=${this.hass} | ||||
|         .value=${this._value} | ||||
|         .label=${this.label === undefined | ||||
|   | ||||
| @@ -41,7 +41,7 @@ export class HaDateInput extends LitElement { | ||||
|     return html`<ha-textfield | ||||
|       .label=${this.label} | ||||
|       .disabled=${this.disabled} | ||||
|       iconTrailing="calendar" | ||||
|       iconTrailing | ||||
|       @click=${this._openDialog} | ||||
|       .value=${this.value | ||||
|         ? formatDateNumeric(new Date(this.value), this.locale) | ||||
|   | ||||
| @@ -1,6 +1,13 @@ | ||||
| import { mdiChevronDown } from "@mdi/js"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
|   html, | ||||
|   LitElement, | ||||
|   PropertyValues, | ||||
|   TemplateResult, | ||||
| } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import { fireEvent } from "../common/dom/fire_event"; | ||||
| import { nextRender } from "../common/util/render-status"; | ||||
| @@ -16,11 +23,21 @@ class HaExpansionPanel extends LitElement { | ||||
|  | ||||
|   @property() secondary?: string; | ||||
|  | ||||
|   @state() _showContent = this.expanded; | ||||
|  | ||||
|   @query(".container") private _container!: HTMLDivElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <div class="summary" @click=${this._toggleContainer}> | ||||
|       <div | ||||
|         id="summary" | ||||
|         @click=${this._toggleContainer} | ||||
|         @keydown=${this._toggleContainer} | ||||
|         role="button" | ||||
|         tabindex="0" | ||||
|         aria-expanded=${this.expanded} | ||||
|         aria-controls="sect1" | ||||
|       > | ||||
|         <slot class="header" name="header"> | ||||
|           ${this.header} | ||||
|           <slot class="secondary" name="secondary">${this.secondary}</slot> | ||||
| @@ -33,21 +50,37 @@ class HaExpansionPanel extends LitElement { | ||||
|       <div | ||||
|         class="container ${classMap({ expanded: this.expanded })}" | ||||
|         @transitionend=${this._handleTransitionEnd} | ||||
|         role="region" | ||||
|         aria-labelledby="summary" | ||||
|         aria-hidden=${!this.expanded} | ||||
|         tabindex="-1" | ||||
|       > | ||||
|         <slot></slot> | ||||
|         ${this._showContent ? html`<slot></slot>` : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleTransitionEnd() { | ||||
|     this._container.style.removeProperty("height"); | ||||
|   protected willUpdate(changedProps: PropertyValues) { | ||||
|     if (changedProps.has("expanded") && this.expanded) { | ||||
|       this._showContent = this.expanded; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private async _toggleContainer(): Promise<void> { | ||||
|   private _handleTransitionEnd() { | ||||
|     this._container.style.removeProperty("height"); | ||||
|     this._showContent = this.expanded; | ||||
|   } | ||||
|  | ||||
|   private async _toggleContainer(ev): Promise<void> { | ||||
|     if (ev.type === "keydown" && ev.key !== "Enter" && ev.key !== " ") { | ||||
|       return; | ||||
|     } | ||||
|     ev.preventDefault(); | ||||
|     const newExpanded = !this.expanded; | ||||
|     fireEvent(this, "expanded-will-change", { expanded: newExpanded }); | ||||
|  | ||||
|     if (newExpanded) { | ||||
|       this._showContent = true; | ||||
|       // allow for dynamic content to be rendered | ||||
|       await nextRender(); | ||||
|     } | ||||
| @@ -80,17 +113,21 @@ class HaExpansionPanel extends LitElement { | ||||
|           var(--divider-color, #e0e0e0) | ||||
|         ); | ||||
|         border-radius: var(--ha-card-border-radius, 4px); | ||||
|         padding: 0 8px; | ||||
|       } | ||||
|  | ||||
|       .summary { | ||||
|       #summary { | ||||
|         display: flex; | ||||
|         padding: var(--expansion-panel-summary-padding, 0); | ||||
|         padding: var(--expansion-panel-summary-padding, 0 8px); | ||||
|         min-height: 48px; | ||||
|         align-items: center; | ||||
|         cursor: pointer; | ||||
|         overflow: hidden; | ||||
|         font-weight: 500; | ||||
|         outline: none; | ||||
|       } | ||||
|  | ||||
|       #summary:focus { | ||||
|         background: var(--input-fill-color); | ||||
|       } | ||||
|  | ||||
|       .summary-icon { | ||||
| @@ -103,6 +140,7 @@ class HaExpansionPanel extends LitElement { | ||||
|       } | ||||
|  | ||||
|       .container { | ||||
|         padding: var(--expansion-panel-content-padding, 0 8px); | ||||
|         overflow: hidden; | ||||
|         transition: height 300ms cubic-bezier(0.4, 0, 0.2, 1); | ||||
|         height: 0px; | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import { styles } from "@material/mwc-textfield/mwc-textfield.css"; | ||||
| import { mdiClose } from "@mdi/js"; | ||||
| import "@polymer/iron-input/iron-input"; | ||||
| import "@polymer/paper-input/paper-input-container"; | ||||
| import { css, html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| @@ -21,7 +20,7 @@ export class HaFileUpload extends LitElement { | ||||
|  | ||||
|   @property() public accept!: string; | ||||
|  | ||||
|   @property() public icon!: string; | ||||
|   @property() public icon?: string; | ||||
|  | ||||
|   @property() public label!: string; | ||||
|  | ||||
| @@ -39,15 +38,7 @@ export class HaFileUpload extends LitElement { | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     if (this.autoOpenFileDialog) { | ||||
|       this._input?.click(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected updated(changedProperties: PropertyValues) { | ||||
|     if (changedProperties.has("_drag") && !this.uploading) { | ||||
|       ( | ||||
|         this.shadowRoot!.querySelector("paper-input-container") as any | ||||
|       )._setFocused(this._drag); | ||||
|       this._openFilePicker(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -60,51 +51,75 @@ export class HaFileUpload extends LitElement { | ||||
|             active | ||||
|           ></ha-circular-progress>` | ||||
|         : html` | ||||
|             <label for="input"> | ||||
|               <paper-input-container | ||||
|                 .alwaysFloatLabel=${Boolean(this.value)} | ||||
|                 @drop=${this._handleDrop} | ||||
|                 @dragenter=${this._handleDragStart} | ||||
|                 @dragover=${this._handleDragStart} | ||||
|                 @dragleave=${this._handleDragEnd} | ||||
|                 @dragend=${this._handleDragEnd} | ||||
|                 class=${classMap({ | ||||
|                   dragged: this._drag, | ||||
|                 })} | ||||
|             <label | ||||
|               for="input" | ||||
|               class="mdc-text-field mdc-text-field--filled ${classMap({ | ||||
|                 "mdc-text-field--focused": this._drag, | ||||
|                 "mdc-text-field--with-leading-icon": Boolean(this.icon), | ||||
|                 "mdc-text-field--with-trailing-icon": Boolean(this.value), | ||||
|               })}" | ||||
|               @drop=${this._handleDrop} | ||||
|               @dragenter=${this._handleDragStart} | ||||
|               @dragover=${this._handleDragStart} | ||||
|               @dragleave=${this._handleDragEnd} | ||||
|               @dragend=${this._handleDragEnd} | ||||
|             > | ||||
|               <span class="mdc-text-field__ripple"></span> | ||||
|               <span | ||||
|                 class="mdc-floating-label ${this.value || this._drag | ||||
|                   ? "mdc-floating-label--float-above" | ||||
|                   : ""}" | ||||
|                 id="label" | ||||
|                 >${this.label}</span | ||||
|               > | ||||
|                 <label for="input" slot="label"> ${this.label} </label> | ||||
|                 <iron-input slot="input"> | ||||
|                   <input | ||||
|                     id="input" | ||||
|                     type="file" | ||||
|                     class="file" | ||||
|                     accept=${this.accept} | ||||
|                     @change=${this._handleFilePicked} | ||||
|                   /> | ||||
|                   ${this.value} | ||||
|                 </iron-input> | ||||
|                 ${this.value | ||||
|                   ? html` | ||||
|                       <ha-icon-button | ||||
|                         slot="suffix" | ||||
|                         @click=${this._clearValue} | ||||
|                         .label=${this.hass?.localize("ui.common.close") || | ||||
|                         "close"} | ||||
|                         .path=${mdiClose} | ||||
|                       ></ha-icon-button> | ||||
|                     ` | ||||
|                   : html` | ||||
|                       <ha-icon-button | ||||
|                         slot="suffix" | ||||
|                         .path=${this.icon} | ||||
|                       ></ha-icon-button> | ||||
|                     `} | ||||
|               </paper-input-container> | ||||
|               ${this.icon | ||||
|                 ? html`<span | ||||
|                     class="mdc-text-field__icon mdc-text-field__icon--leading" | ||||
|                     tabindex="-1" | ||||
|                   > | ||||
|                     <ha-icon-button | ||||
|                       @click=${this._openFilePicker} | ||||
|                       .path=${this.icon} | ||||
|                     ></ha-icon-button> | ||||
|                   </span>` | ||||
|                 : ""} | ||||
|               <div class="value">${this.value}</div> | ||||
|               <input | ||||
|                 id="input" | ||||
|                 type="file" | ||||
|                 class="mdc-text-field__input file" | ||||
|                 accept=${this.accept} | ||||
|                 @change=${this._handleFilePicked} | ||||
|                 aria-labelledby="label" | ||||
|               /> | ||||
|               ${this.value | ||||
|                 ? html`<span | ||||
|                     class="mdc-text-field__icon mdc-text-field__icon--trailing" | ||||
|                     tabindex="1" | ||||
|                   > | ||||
|                     <ha-icon-button | ||||
|                       slot="suffix" | ||||
|                       @click=${this._clearValue} | ||||
|                       .label=${this.hass?.localize("ui.common.close") || | ||||
|                       "close"} | ||||
|                       .path=${mdiClose} | ||||
|                     ></ha-icon-button> | ||||
|                   </span>` | ||||
|                 : ""} | ||||
|               <span | ||||
|                 class="mdc-line-ripple ${this._drag | ||||
|                   ? "mdc-line-ripple--active" | ||||
|                   : ""}" | ||||
|               ></span> | ||||
|             </label> | ||||
|           `} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _openFilePicker() { | ||||
|     this._input?.click(); | ||||
|   } | ||||
|  | ||||
|   private _handleDrop(ev: DragEvent) { | ||||
|     ev.preventDefault(); | ||||
|     ev.stopPropagation(); | ||||
| @@ -137,40 +152,66 @@ export class HaFileUpload extends LitElement { | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       paper-input-container { | ||||
|         position: relative; | ||||
|         padding: 8px; | ||||
|         margin: 0 -8px; | ||||
|       } | ||||
|       paper-input-container.dragged:before { | ||||
|         position: var(--layout-fit_-_position); | ||||
|         top: var(--layout-fit_-_top); | ||||
|         right: var(--layout-fit_-_right); | ||||
|         bottom: var(--layout-fit_-_bottom); | ||||
|         left: var(--layout-fit_-_left); | ||||
|         background: currentColor; | ||||
|         content: ""; | ||||
|         opacity: var(--dark-divider-opacity); | ||||
|         pointer-events: none; | ||||
|         border-radius: 4px; | ||||
|       } | ||||
|       input.file { | ||||
|         display: none; | ||||
|       } | ||||
|       img { | ||||
|         max-width: 125px; | ||||
|         max-height: 125px; | ||||
|       } | ||||
|       ha-icon-button { | ||||
|         --mdc-icon-button-size: 24px; | ||||
|         --mdc-icon-size: 20px; | ||||
|       } | ||||
|       ha-circular-progress { | ||||
|         display: block; | ||||
|         text-align-last: center; | ||||
|       } | ||||
|     `; | ||||
|     return [ | ||||
|       styles, | ||||
|       css` | ||||
|         :host { | ||||
|           display: block; | ||||
|         } | ||||
|         .mdc-text-field--filled { | ||||
|           height: auto; | ||||
|           padding-top: 16px; | ||||
|           cursor: pointer; | ||||
|         } | ||||
|         .mdc-text-field--filled.mdc-text-field--with-trailing-icon { | ||||
|           padding-top: 28px; | ||||
|         } | ||||
|         .mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field__icon { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|         .mdc-text-field--filled.mdc-text-field--with-trailing-icon | ||||
|           .mdc-text-field__icon { | ||||
|           align-self: flex-end; | ||||
|         } | ||||
|         .mdc-text-field__icon--leading { | ||||
|           margin-bottom: 12px; | ||||
|         } | ||||
|         .mdc-text-field--filled .mdc-floating-label--float-above { | ||||
|           transform: scale(0.75); | ||||
|           top: 8px; | ||||
|         } | ||||
|         .dragged:before { | ||||
|           position: var(--layout-fit_-_position); | ||||
|           top: var(--layout-fit_-_top); | ||||
|           right: var(--layout-fit_-_right); | ||||
|           bottom: var(--layout-fit_-_bottom); | ||||
|           left: var(--layout-fit_-_left); | ||||
|           background: currentColor; | ||||
|           content: ""; | ||||
|           opacity: var(--dark-divider-opacity); | ||||
|           pointer-events: none; | ||||
|           border-radius: 4px; | ||||
|         } | ||||
|         .value { | ||||
|           width: 100%; | ||||
|         } | ||||
|         input.file { | ||||
|           display: none; | ||||
|         } | ||||
|         img { | ||||
|           max-width: 100%; | ||||
|           max-height: 125px; | ||||
|         } | ||||
|         ha-icon-button { | ||||
|           --mdc-icon-button-size: 24px; | ||||
|           --mdc-icon-size: 20px; | ||||
|         } | ||||
|         ha-circular-progress { | ||||
|           display: block; | ||||
|           text-align-last: center; | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { HaFormSchema } from "./types"; | ||||
| import type { Selector } from "../../data/selector"; | ||||
| import type { HaFormSchema } from "./types"; | ||||
|  | ||||
| export const computeInitialHaFormData = ( | ||||
|   schema: HaFormSchema[] | ||||
| @@ -31,6 +32,25 @@ export const computeInitialHaFormData = ( | ||||
|         minutes: 0, | ||||
|         seconds: 0, | ||||
|       }; | ||||
|     } else if ("selector" in field) { | ||||
|       const selector: Selector = field.selector; | ||||
|       if ("boolean" in selector) { | ||||
|         data[field.name] = false; | ||||
|       } else if ("text" in selector) { | ||||
|         data[field.name] = ""; | ||||
|       } else if ("number" in selector) { | ||||
|         data[field.name] = "min" in selector.number ? selector.number.min : 0; | ||||
|       } else if ("select" in selector) { | ||||
|         if (selector.select.options.length) { | ||||
|           data[field.name] = selector.select.options[0][0]; | ||||
|         } | ||||
|       } else if ("duration" in selector) { | ||||
|         data[field.name] = { | ||||
|           hours: 0, | ||||
|           minutes: 0, | ||||
|           seconds: 0, | ||||
|         }; | ||||
|       } | ||||
|     } | ||||
|   }); | ||||
|   return data; | ||||
|   | ||||
| @@ -9,7 +9,9 @@ export class HaFormConstant extends LitElement implements HaFormElement { | ||||
|   @property() public label!: string; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<span class="label">${this.label}</span>: ${this.schema.value}`; | ||||
|     return html`<span class="label">${this.label}</span>${this.schema.value | ||||
|         ? `: ${this.schema.value}` | ||||
|         : ""}`; | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|   | ||||
| @@ -1,16 +1,20 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { stopPropagation } from "../../common/dom/stop_propagation"; | ||||
| import "../ha-radio"; | ||||
| import type { HaRadio } from "../ha-radio"; | ||||
| import "../ha-select"; | ||||
| import type { HaSelect } from "../ha-select"; | ||||
| import { HaFormElement, HaFormSelectData, HaFormSelectSchema } from "./types"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { | ||||
|   HaFormElement, | ||||
|   HaFormSelectData, | ||||
|   HaFormSelectSchema, | ||||
| } from "./types"; | ||||
| import type { SelectSelector } from "../../data/selector"; | ||||
| import "../ha-selector/ha-selector-select"; | ||||
|  | ||||
| @customElement("ha-form-select") | ||||
| export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public schema!: HaFormSelectSchema; | ||||
|  | ||||
|   @property() public data!: HaFormSelectData; | ||||
| @@ -19,60 +23,35 @@ export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   @query("ha-select", true) private _input?: HTMLElement; | ||||
|  | ||||
|   public focus() { | ||||
|     if (this._input) { | ||||
|       this._input.focus(); | ||||
|     } | ||||
|   } | ||||
|   private _selectSchema = memoizeOne( | ||||
|     (options): SelectSelector => ({ | ||||
|       select: { | ||||
|         options: options.map((option) => ({ | ||||
|           value: option[0], | ||||
|           label: option[1], | ||||
|         })), | ||||
|       }, | ||||
|     }) | ||||
|   ); | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (this.schema.required && this.schema.options!.length < 6) { | ||||
|       return html` | ||||
|         <div> | ||||
|           ${this.label} | ||||
|           ${this.schema.options.map( | ||||
|             ([value, label]) => html` | ||||
|               <mwc-formfield .label=${label}> | ||||
|                 <ha-radio | ||||
|                   .checked=${value === this.data} | ||||
|                   .value=${value} | ||||
|                   .disabled=${this.disabled} | ||||
|                   @change=${this._valueChanged} | ||||
|                 ></ha-radio> | ||||
|               </mwc-formfield> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       `; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <ha-select | ||||
|         fixedMenuPosition | ||||
|         naturalMenuWidth | ||||
|         .label=${this.label} | ||||
|       <ha-selector-select | ||||
|         .hass=${this.hass} | ||||
|         .schema=${this.schema} | ||||
|         .value=${this.data} | ||||
|         .label=${this.label} | ||||
|         .disabled=${this.disabled} | ||||
|         @closed=${stopPropagation} | ||||
|         @selected=${this._valueChanged} | ||||
|       > | ||||
|         ${!this.schema.required | ||||
|           ? html`<mwc-list-item value=""></mwc-list-item>` | ||||
|           : ""} | ||||
|         ${this.schema.options!.map( | ||||
|           ([value, label]) => html` | ||||
|             <mwc-list-item .value=${value}>${label}</mwc-list-item> | ||||
|           ` | ||||
|         )} | ||||
|       </ha-select> | ||||
|         .required=${this.schema.required} | ||||
|         .selector=${this._selectSchema(this.schema.options)} | ||||
|         @value-changed=${this._valueChanged} | ||||
|       ></ha-selector-select> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent) { | ||||
|     ev.stopPropagation(); | ||||
|     let value: string | undefined = (ev.target as HaSelect | HaRadio).value; | ||||
|     let value: string | undefined = ev.detail.value; | ||||
|  | ||||
|     if (value === this.data) { | ||||
|       return; | ||||
| @@ -86,15 +65,6 @@ export class HaFormSelect extends LitElement implements HaFormElement { | ||||
|       value, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return css` | ||||
|       ha-select, | ||||
|       mwc-formfield { | ||||
|         display: block; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -25,6 +25,8 @@ import { HomeAssistant } from "../../types"; | ||||
| const getValue = (obj, item) => | ||||
|   obj ? (!item.name ? obj : obj[item.name]) : null; | ||||
|  | ||||
| const getError = (obj, item) => (obj && item.name ? obj[item.name] : null); | ||||
|  | ||||
| let selectorImported = false; | ||||
|  | ||||
| @customElement("ha-form") | ||||
| @@ -84,7 +86,7 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|             ` | ||||
|           : ""} | ||||
|         ${this.schema.map((item) => { | ||||
|           const error = getValue(this.error, item); | ||||
|           const error = getError(this.error, item); | ||||
|  | ||||
|           return html` | ||||
|             ${error | ||||
| @@ -104,6 +106,7 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|                   .disabled=${this.disabled} | ||||
|                   .helper=${this._computeHelper(item)} | ||||
|                   .required=${item.required || false} | ||||
|                   .context=${this._generateContext(item)} | ||||
|                 ></ha-selector>` | ||||
|               : dynamicElement(`ha-form-${item.type}`, { | ||||
|                   schema: item, | ||||
| @@ -113,6 +116,7 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|                   hass: this.hass, | ||||
|                   computeLabel: this.computeLabel, | ||||
|                   computeHelper: this.computeHelper, | ||||
|                   context: this._generateContext(item), | ||||
|                 })} | ||||
|           `; | ||||
|         })} | ||||
| @@ -120,6 +124,20 @@ export class HaForm extends LitElement implements HaFormElement { | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _generateContext( | ||||
|     schema: HaFormSchema | ||||
|   ): Record<string, any> | undefined { | ||||
|     if (!schema.context) { | ||||
|       return undefined; | ||||
|     } | ||||
|  | ||||
|     const context = {}; | ||||
|     for (const [context_key, data_key] of Object.entries(schema.context)) { | ||||
|       context[context_key] = this.data[data_key]; | ||||
|     } | ||||
|     return context; | ||||
|   } | ||||
|  | ||||
|   protected createRenderRoot() { | ||||
|     const root = super.createRenderRoot(); | ||||
|     // attach it as soon as possible to make sure we fetch all events. | ||||
|   | ||||
| @@ -24,6 +24,7 @@ export interface HaFormBaseSchema { | ||||
|     // This value will be set initially when form is loaded | ||||
|     suggested_value?: HaFormData; | ||||
|   }; | ||||
|   context?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export interface HaFormGridSchema extends HaFormBaseSchema { | ||||
| @@ -40,7 +41,7 @@ export interface HaFormSelector extends HaFormBaseSchema { | ||||
|  | ||||
| export interface HaFormConstantSchema extends HaFormBaseSchema { | ||||
|   type: "constant"; | ||||
|   value: string; | ||||
|   value?: string; | ||||
| } | ||||
|  | ||||
| export interface HaFormIntegerSchema extends HaFormBaseSchema { | ||||
|   | ||||
| @@ -44,6 +44,9 @@ export class HaSelect extends SelectBase { | ||||
|       .mdc-select:not(.mdc-select--disabled) .mdc-select__icon { | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|       .mdc-select__anchor { | ||||
|         width: var(--ha-select-min-width, 200px); | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
| } | ||||
|   | ||||
| @@ -28,7 +28,11 @@ export class HaAreaSelector extends LitElement { | ||||
|         oldSelector !== this.selector && | ||||
|         this.selector.area.device?.integration | ||||
|       ) { | ||||
|         this._loadConfigEntries(); | ||||
|         getConfigEntries(this.hass, { | ||||
|           domain: this.selector.area.device.integration, | ||||
|         }).then((entries) => { | ||||
|           this._configEntries = entries; | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| @@ -85,12 +89,6 @@ export class HaAreaSelector extends LitElement { | ||||
|     } | ||||
|     return true; | ||||
|   }; | ||||
|  | ||||
|   private async _loadConfigEntries() { | ||||
|     this._configEntries = (await getConfigEntries(this.hass)).filter( | ||||
|       (entry) => entry.domain === this.selector.area.device?.integration | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import "../entity/ha-entity-attribute-picker"; | ||||
| import { html, LitElement } from "lit"; | ||||
| import { html, LitElement, PropertyValues } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { AttributeSelector } from "../../data/selector"; | ||||
| import { SubscribeMixin } from "../../mixins/subscribe-mixin"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
|  | ||||
| @customElement("ha-selector-attribute") | ||||
| export class HaSelectorAttribute extends SubscribeMixin(LitElement) { | ||||
| @@ -17,11 +18,16 @@ export class HaSelectorAttribute extends SubscribeMixin(LitElement) { | ||||
|  | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   @property() public context?: { | ||||
|     filter_entity?: string; | ||||
|   }; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-entity-attribute-picker | ||||
|         .hass=${this.hass} | ||||
|         .entityId=${this.selector.attribute.entity_id} | ||||
|         .entityId=${this.selector.attribute.entity_id || | ||||
|         this.context?.filter_entity} | ||||
|         .value=${this.value} | ||||
|         .label=${this.label} | ||||
|         .disabled=${this.disabled} | ||||
| @@ -29,6 +35,47 @@ export class HaSelectorAttribute extends SubscribeMixin(LitElement) { | ||||
|       ></ha-entity-attribute-picker> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   protected updated(changedProps: PropertyValues): void { | ||||
|     super.updated(changedProps); | ||||
|     if ( | ||||
|       // No need to filter value if no value | ||||
|       !this.value || | ||||
|       // Only adjust value if we used the context | ||||
|       this.selector.attribute.entity_id || | ||||
|       // Only check if context has changed | ||||
|       !changedProps.has("context") | ||||
|     ) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const oldContext = changedProps.get("context") as this["context"]; | ||||
|  | ||||
|     if ( | ||||
|       !this.context || | ||||
|       oldContext?.filter_entity === this.context.filter_entity | ||||
|     ) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     // Validate that that the attribute is still valid for this entity, else unselect. | ||||
|     let invalid = false; | ||||
|     if (this.context.filter_entity) { | ||||
|       const stateObj = this.hass.states[this.context.filter_entity]; | ||||
|  | ||||
|       if (!(stateObj && this.value in stateObj.attributes)) { | ||||
|         invalid = true; | ||||
|       } | ||||
|     } else { | ||||
|       invalid = this.value !== undefined; | ||||
|     } | ||||
|  | ||||
|     if (invalid) { | ||||
|       fireEvent(this, "value-changed", { | ||||
|         value: undefined, | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
							
								
								
									
										58
									
								
								src/components/ha-selector/ha-selector-color-rgb.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/components/ha-selector/ha-selector-color-rgb.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { ColorRGBSelector } from "../../data/selector"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { hex2rgb, rgb2hex } from "../../common/color/convert-color"; | ||||
| import "../ha-textfield"; | ||||
|  | ||||
| @customElement("ha-selector-color_rgb") | ||||
| export class HaColorRGBSelector extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public selector!: ColorRGBSelector; | ||||
|  | ||||
|   @property() public value?: string; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-textfield | ||||
|         type="color" | ||||
|         .value=${this.value ? rgb2hex(this.value as any) : ""} | ||||
|         .label=${this.label || ""} | ||||
|         @change=${this._valueChanged} | ||||
|       ></ha-textfield> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent) { | ||||
|     const value = (ev.target as any).value; | ||||
|     fireEvent(this, "value-changed", { | ||||
|       value: hex2rgb(value), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host { | ||||
|       display: flex; | ||||
|       justify-content: flex-end; | ||||
|       align-items: center; | ||||
|     } | ||||
|     ha-textfield { | ||||
|       --text-field-padding: 8px; | ||||
|       min-width: 75px; | ||||
|       flex-grow: 1; | ||||
|       margin: 0 4px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-selector-color_rgb": HaColorRGBSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										58
									
								
								src/components/ha-selector/ha-selector-color-temp.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/components/ha-selector/ha-selector-color-temp.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { ColorTempSelector } from "../../data/selector"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../ha-labeled-slider"; | ||||
|  | ||||
| @customElement("ha-selector-color_temp") | ||||
| export class HaColorTempSelector extends LitElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public selector!: ColorTempSelector; | ||||
|  | ||||
|   @property() public value?: string; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-labeled-slider | ||||
|         pin | ||||
|         icon="hass:thermometer" | ||||
|         .caption=${this.label} | ||||
|         .min=${this.selector.color_temp.min_mireds ?? 153} | ||||
|         .max=${this.selector.color_temp.max_mireds ?? 500} | ||||
|         .value=${this.value} | ||||
|         @change=${this._valueChanged} | ||||
|       ></ha-labeled-slider> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent) { | ||||
|     fireEvent(this, "value-changed", { | ||||
|       value: Number((ev.target as any).value), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     ha-labeled-slider { | ||||
|       --ha-slider-background: -webkit-linear-gradient( | ||||
|         right, | ||||
|         rgb(255, 160, 0) 0%, | ||||
|         white 50%, | ||||
|         rgb(166, 209, 255) 100% | ||||
|       ); | ||||
|       /* The color temp minimum value shouldn't be rendered differently. It's not "off". */ | ||||
|       --paper-slider-knob-start-border-color: var(--primary-color); | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-selector-color_temp": HaColorTempSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										36
									
								
								src/components/ha-selector/ha-selector-date.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								src/components/ha-selector/ha-selector-date.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { DateSelector } from "../../data/selector"; | ||||
| import "../ha-date-input"; | ||||
|  | ||||
| @customElement("ha-selector-date") | ||||
| export class HaDateSelector extends LitElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public selector!: DateSelector; | ||||
|  | ||||
|   @property() public value?: string; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-date-input | ||||
|         .label=${this.label} | ||||
|         .locale=${this.hass.locale} | ||||
|         .disabled=${this.disabled} | ||||
|         .value=${this.value} | ||||
|       > | ||||
|       </ha-date-input> | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-selector-date": HaDateSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										74
									
								
								src/components/ha-selector/ha-selector-datetime.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								src/components/ha-selector/ha-selector-datetime.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| import { css, html, LitElement } from "lit"; | ||||
| import { customElement, property, query } from "lit/decorators"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { DateTimeSelector } from "../../data/selector"; | ||||
| import type { HaDateInput } from "../ha-date-input"; | ||||
| import type { HaTimeInput } from "../ha-time-input"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import "../ha-date-input"; | ||||
| import "../ha-time-input"; | ||||
|  | ||||
| @customElement("ha-selector-datetime") | ||||
| export class HaDateTimeSelector extends LitElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public selector!: DateTimeSelector; | ||||
|  | ||||
|   @property() public value?: string; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public disabled = false; | ||||
|  | ||||
|   @query("ha-date-input") private _dateInput!: HaDateInput; | ||||
|  | ||||
|   @query("ha-time-input") private _timeInput!: HaTimeInput; | ||||
|  | ||||
|   protected render() { | ||||
|     const values = this.value?.split(" "); | ||||
|  | ||||
|     return html` | ||||
|       <ha-date-input | ||||
|         .label=${this.label} | ||||
|         .locale=${this.hass.locale} | ||||
|         .disabled=${this.disabled} | ||||
|         .value=${values?.[0]} | ||||
|         @value-changed=${this._valueChanged} | ||||
|       > | ||||
|       </ha-date-input> | ||||
|       <ha-time-input | ||||
|         enable-second | ||||
|         .value=${values?.[1] || "0:00:00"} | ||||
|         .locale=${this.hass.locale} | ||||
|         .disabled=${this.disabled} | ||||
|         @value-changed=${this._valueChanged} | ||||
|       ></ha-time-input> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev: CustomEvent): void { | ||||
|     ev.stopPropagation(); | ||||
|     fireEvent(this, "value-changed", { | ||||
|       value: `${this._dateInput.value} ${this._timeInput.value}`, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     :host { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       flex-direction: row; | ||||
|     } | ||||
|  | ||||
|     ha-date-input { | ||||
|       min-width: 150px; | ||||
|       margin-right: 4px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-selector-datetime": HaDateTimeSelector; | ||||
|   } | ||||
| } | ||||
| @@ -1,10 +1,11 @@ | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { ConfigEntry, getConfigEntries } from "../../data/config_entries"; | ||||
| import { DeviceRegistryEntry } from "../../data/device_registry"; | ||||
| import { DeviceSelector } from "../../data/selector"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import type { DeviceRegistryEntry } from "../../data/device_registry"; | ||||
| import type { DeviceSelector } from "../../data/selector"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import "../device/ha-device-picker"; | ||||
| import "../device/ha-devices-picker"; | ||||
|  | ||||
| @customElement("ha-selector-device") | ||||
| export class HaDeviceSelector extends LitElement { | ||||
| @@ -24,26 +25,46 @@ export class HaDeviceSelector extends LitElement { | ||||
|     if (changedProperties.has("selector")) { | ||||
|       const oldSelector = changedProperties.get("selector"); | ||||
|       if (oldSelector !== this.selector && this.selector.device?.integration) { | ||||
|         this._loadConfigEntries(); | ||||
|         getConfigEntries(this.hass, { | ||||
|           domain: this.selector.device.integration, | ||||
|         }).then((entries) => { | ||||
|           this._configEntries = entries; | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   protected render() { | ||||
|     return html`<ha-device-picker | ||||
|       .hass=${this.hass} | ||||
|       .value=${this.value} | ||||
|       .label=${this.label} | ||||
|       .deviceFilter=${this._filterDevices} | ||||
|       .includeDeviceClasses=${this.selector.device.entity?.device_class | ||||
|         ? [this.selector.device.entity.device_class] | ||||
|         : undefined} | ||||
|       .includeDomains=${this.selector.device.entity?.domain | ||||
|         ? [this.selector.device.entity.domain] | ||||
|         : undefined} | ||||
|       .disabled=${this.disabled} | ||||
|       allow-custom-entity | ||||
|     ></ha-device-picker>`; | ||||
|     if (!this.selector.device.multiple) { | ||||
|       return html`<ha-device-picker | ||||
|         .hass=${this.hass} | ||||
|         .value=${this.value} | ||||
|         .label=${this.label} | ||||
|         .deviceFilter=${this._filterDevices} | ||||
|         .includeDeviceClasses=${this.selector.device.entity?.device_class | ||||
|           ? [this.selector.device.entity.device_class] | ||||
|           : undefined} | ||||
|         .includeDomains=${this.selector.device.entity?.domain | ||||
|           ? [this.selector.device.entity.domain] | ||||
|           : undefined} | ||||
|         .disabled=${this.disabled} | ||||
|         allow-custom-entity | ||||
|       ></ha-device-picker> `; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       ${this.label ? html`<label>${this.label}</label>` : ""} | ||||
|       <ha-devices-picker | ||||
|         .hass=${this.hass} | ||||
|         .value=${this.value} | ||||
|         .includeDeviceClasses=${this.selector.device.entity?.device_class | ||||
|           ? [this.selector.device.entity.device_class] | ||||
|           : undefined} | ||||
|         .includeDomains=${this.selector.device.entity?.domain | ||||
|           ? [this.selector.device.entity.domain] | ||||
|           : undefined} | ||||
|       ></ha-devices-picker> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _filterDevices = (device: DeviceRegistryEntry): boolean => { | ||||
| @@ -71,12 +92,6 @@ export class HaDeviceSelector extends LitElement { | ||||
|     } | ||||
|     return true; | ||||
|   }; | ||||
|  | ||||
|   private async _loadConfigEntries() { | ||||
|     this._configEntries = (await getConfigEntries(this.hass)).filter( | ||||
|       (entry) => entry.domain === this.selector.device.integration | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import "../ha-duration-input"; | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { DurationSelector } from "../../data/selector"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import type { DurationSelector } from "../../data/selector"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import "../ha-duration-input"; | ||||
|  | ||||
| @customElement("ha-selector-duration") | ||||
| export class HaTimeDuration extends LitElement { | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import { subscribeEntityRegistry } from "../../data/entity_registry"; | ||||
| import { EntitySelector } from "../../data/selector"; | ||||
| import { SubscribeMixin } from "../../mixins/subscribe-mixin"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import "../entity/ha-entities-picker"; | ||||
| import "../entity/ha-entity-picker"; | ||||
|  | ||||
| @customElement("ha-selector-entity") | ||||
| @@ -23,14 +24,30 @@ export class HaEntitySelector extends SubscribeMixin(LitElement) { | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html`<ha-entity-picker | ||||
|       .hass=${this.hass} | ||||
|       .value=${this.value} | ||||
|       .label=${this.label} | ||||
|       .entityFilter=${this._filterEntities} | ||||
|       .disabled=${this.disabled} | ||||
|       allow-custom-entity | ||||
|     ></ha-entity-picker>`; | ||||
|     if (!this.selector.entity.multiple) { | ||||
|       return html`<ha-entity-picker | ||||
|         .hass=${this.hass} | ||||
|         .value=${this.value} | ||||
|         .label=${this.label} | ||||
|         .includeEntities=${this.selector.entity.include_entities} | ||||
|         .excludeEntities=${this.selector.entity.exclude_entities} | ||||
|         .entityFilter=${this._filterEntities} | ||||
|         .disabled=${this.disabled} | ||||
|         .itemValuePath=${!this.selector.entity.use_uuid ? "entity_id" : "id"} | ||||
|         allow-custom-entity | ||||
|       ></ha-entity-picker>`; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       ${this.label ? html`<label>${this.label}</label>` : ""} | ||||
|       <ha-entities-picker | ||||
|         .hass=${this.hass} | ||||
|         .value=${this.value} | ||||
|         .entityFilter=${this._filterEntities} | ||||
|         .includeEntities=${this.selector.entity.include_entities} | ||||
|         .excludeEntities=${this.selector.entity.exclude_entities} | ||||
|       ></ha-entities-picker> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   public hassSubscribe(): UnsubscribeFunc[] { | ||||
| @@ -51,10 +68,11 @@ export class HaEntitySelector extends SubscribeMixin(LitElement) { | ||||
|   private _filterEntities = (entity: HassEntity): boolean => { | ||||
|     if (this.selector.entity?.domain) { | ||||
|       const filterDomain = this.selector.entity.domain; | ||||
|       const filterDomainIsArray = Array.isArray(filterDomain); | ||||
|       const entityDomain = computeStateDomain(entity); | ||||
|       if ( | ||||
|         (Array.isArray(filterDomain) && !filterDomain.includes(entityDomain)) || | ||||
|         entityDomain !== filterDomain | ||||
|         (filterDomainIsArray && !filterDomain.includes(entityDomain)) || | ||||
|         (!filterDomainIsArray && entityDomain !== filterDomain) | ||||
|       ) { | ||||
|         return false; | ||||
|       } | ||||
|   | ||||
							
								
								
									
										80
									
								
								src/components/ha-selector/ha-selector-location.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								src/components/ha-selector/ha-selector-location.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| import { html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import memoizeOne from "memoize-one"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import type { | ||||
|   LocationSelector, | ||||
|   LocationSelectorValue, | ||||
| } from "../../data/selector"; | ||||
| import "../../panels/lovelace/components/hui-theme-select-editor"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { MarkerLocation } from "../map/ha-locations-editor"; | ||||
| import "../map/ha-locations-editor"; | ||||
|  | ||||
| @customElement("ha-selector-location") | ||||
| export class HaLocationSelector extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public selector!: LocationSelector; | ||||
|  | ||||
|   @property() public value?: LocationSelectorValue; | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean, reflect: true }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html` | ||||
|       <ha-locations-editor | ||||
|         class="flex" | ||||
|         .hass=${this.hass} | ||||
|         .locations=${this._location(this.selector, this.value)} | ||||
|         @location-updated=${this._locationChanged} | ||||
|         @radius-updated=${this._radiusChanged} | ||||
|       ></ha-locations-editor> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _location = memoizeOne( | ||||
|     ( | ||||
|       selector: LocationSelector, | ||||
|       value?: LocationSelectorValue | ||||
|     ): MarkerLocation[] => { | ||||
|       const computedStyles = getComputedStyle(this); | ||||
|       const zoneRadiusColor = selector.location.radius | ||||
|         ? computedStyles.getPropertyValue("--zone-radius-color") || | ||||
|           computedStyles.getPropertyValue("--accent-color") | ||||
|         : undefined; | ||||
|       return [ | ||||
|         { | ||||
|           id: "location", | ||||
|           latitude: value?.latitude || this.hass.config.latitude, | ||||
|           longitude: value?.longitude || this.hass.config.longitude, | ||||
|           radius: selector.location.radius ? value?.radius || 1000 : undefined, | ||||
|           radius_color: zoneRadiusColor, | ||||
|           icon: selector.location.icon, | ||||
|           location_editable: true, | ||||
|           radius_editable: true, | ||||
|         }, | ||||
|       ]; | ||||
|     } | ||||
|   ); | ||||
|  | ||||
|   private _locationChanged(ev: CustomEvent) { | ||||
|     const [latitude, longitude] = ev.detail.location; | ||||
|     fireEvent(this, "value-changed", { | ||||
|       value: { ...this.value, latitude, longitude }, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   private _radiusChanged(ev: CustomEvent) { | ||||
|     const radius = ev.detail.radius; | ||||
|     fireEvent(this, "value-changed", { value: { ...this.value, radius } }); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-selector-location": HaLocationSelector; | ||||
|   } | ||||
| } | ||||
| @@ -12,6 +12,7 @@ import { | ||||
| } from "../../data/media-player"; | ||||
| import type { MediaSelector, MediaSelectorValue } from "../../data/selector"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import { brandsUrl, extractDomainFromBrandUrl } from "../../util/brands-url"; | ||||
| import "../ha-alert"; | ||||
| import "../ha-form/ha-form"; | ||||
| import type { HaFormSchema } from "../ha-form/types"; | ||||
| @@ -50,6 +51,18 @@ export class HaMediaSelector extends LitElement { | ||||
|         getSignedPath(this.hass, thumbnail).then((signedPath) => { | ||||
|           this._thumbnailUrl = signedPath.path; | ||||
|         }); | ||||
|       } else if ( | ||||
|         thumbnail && | ||||
|         thumbnail.startsWith("https://brands.home-assistant.io") | ||||
|       ) { | ||||
|         // The backend is not aware of the theme used by the users, | ||||
|         // so we rewrite the URL to show a proper icon | ||||
|         this._thumbnailUrl = brandsUrl({ | ||||
|           domain: extractDomainFromBrandUrl(thumbnail), | ||||
|           type: "icon", | ||||
|           useFallback: true, | ||||
|           darkOptimized: this.hass.themes?.darkMode, | ||||
|         }); | ||||
|       } else { | ||||
|         this._thumbnailUrl = thumbnail; | ||||
|       } | ||||
|   | ||||
| @@ -46,7 +46,7 @@ export class HaNumberSelector extends LitElement { | ||||
|         class=${classMap({ single: this.selector.number.mode === "box" })} | ||||
|         .min=${this.selector.number.min} | ||||
|         .max=${this.selector.number.max} | ||||
|         .value=${this.value || ""} | ||||
|         .value=${this.value ?? ""} | ||||
|         .step=${this.selector.number.step ?? 1} | ||||
|         .disabled=${this.disabled} | ||||
|         .required=${this.required} | ||||
|   | ||||
| @@ -1,17 +1,19 @@ | ||||
| import "@material/mwc-formfield/mwc-formfield"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, CSSResultGroup, html, LitElement } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
| import { stopPropagation } from "../../common/dom/stop_propagation"; | ||||
| import { SelectOption, SelectSelector } from "../../data/selector"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| import type { SelectOption, SelectSelector } from "../../data/selector"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import "../ha-select"; | ||||
| import "../ha-radio"; | ||||
|  | ||||
| @customElement("ha-selector-select") | ||||
| export class HaSelectSelector extends LitElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public selector!: SelectSelector; | ||||
|   @property({ attribute: false }) public selector!: SelectSelector; | ||||
|  | ||||
|   @property() public value?: string; | ||||
|  | ||||
| @@ -21,24 +23,51 @@ export class HaSelectSelector extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public disabled = false; | ||||
|  | ||||
|   protected render() { | ||||
|     return html`<ha-select | ||||
|       fixedMenuPosition | ||||
|       naturalMenuWidth | ||||
|       .label=${this.label} | ||||
|       .value=${this.value} | ||||
|       .helper=${this.helper} | ||||
|       .disabled=${this.disabled} | ||||
|       @closed=${stopPropagation} | ||||
|       @selected=${this._valueChanged} | ||||
|     > | ||||
|       ${this.selector.select.options.map((item: string | SelectOption) => { | ||||
|         const value = typeof item === "object" ? item.value : item; | ||||
|         const label = typeof item === "object" ? item.label : item; | ||||
|   @property({ type: Boolean }) public required = true; | ||||
|  | ||||
|         return html`<mwc-list-item .value=${value}>${label}</mwc-list-item>`; | ||||
|       })} | ||||
|     </ha-select>`; | ||||
|   protected render() { | ||||
|     if (this.required && this.selector.select.options!.length < 6) { | ||||
|       return html` | ||||
|         <div> | ||||
|           ${this.label} | ||||
|           ${this.selector.select.options.map((item: string | SelectOption) => { | ||||
|             const value = typeof item === "object" ? item.value : item; | ||||
|             const label = typeof item === "object" ? item.label : item; | ||||
|  | ||||
|             return html` | ||||
|               <mwc-formfield .label=${label}> | ||||
|                 <ha-radio | ||||
|                   .checked=${value === this.value} | ||||
|                   .value=${value} | ||||
|                   .disabled=${this.disabled} | ||||
|                   @change=${this._valueChanged} | ||||
|                 ></ha-radio> | ||||
|               </mwc-formfield> | ||||
|             `; | ||||
|           })} | ||||
|         </div> | ||||
|       `; | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <ha-select | ||||
|         fixedMenuPosition | ||||
|         naturalMenuWidth | ||||
|         .label=${this.label} | ||||
|         .value=${this.value} | ||||
|         .helper=${this.helper} | ||||
|         .disabled=${this.disabled} | ||||
|         @closed=${stopPropagation} | ||||
|         @selected=${this._valueChanged} | ||||
|       > | ||||
|         ${this.selector.select.options.map((item: string | SelectOption) => { | ||||
|           const value = typeof item === "object" ? item.value : item; | ||||
|           const label = typeof item === "object" ? item.label : item; | ||||
|  | ||||
|           return html`<mwc-list-item .value=${value}>${label}</mwc-list-item>`; | ||||
|         })} | ||||
|       </ha-select> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _valueChanged(ev) { | ||||
| @@ -56,6 +85,9 @@ export class HaSelectSelector extends LitElement { | ||||
|       ha-select { | ||||
|         width: 100%; | ||||
|       } | ||||
|       mwc-formfield { | ||||
|         display: block; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -134,9 +134,8 @@ export class HaTargetSelector extends SubscribeMixin(LitElement) { | ||||
|   private async _loadConfigEntries() { | ||||
|     this._configEntries = (await getConfigEntries(this.hass)).filter( | ||||
|       (entry) => | ||||
|         entry.domain === | ||||
|         (this.selector.target.device?.integration || | ||||
|           this.selector.target.entity?.integration) | ||||
|         entry.domain === this.selector.target.device?.integration || | ||||
|         entry.domain === this.selector.target.entity?.integration | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   | ||||
| @@ -8,6 +8,9 @@ import "./ha-selector-addon"; | ||||
| import "./ha-selector-area"; | ||||
| import "./ha-selector-attribute"; | ||||
| import "./ha-selector-boolean"; | ||||
| import "./ha-selector-color-rgb"; | ||||
| import "./ha-selector-date"; | ||||
| import "./ha-selector-datetime"; | ||||
| import "./ha-selector-device"; | ||||
| import "./ha-selector-duration"; | ||||
| import "./ha-selector-entity"; | ||||
| @@ -20,6 +23,8 @@ import "./ha-selector-time"; | ||||
| import "./ha-selector-icon"; | ||||
| import "./ha-selector-media"; | ||||
| import "./ha-selector-theme"; | ||||
| import "./ha-selector-location"; | ||||
| import "./ha-selector-color-temp"; | ||||
|  | ||||
| @customElement("ha-selector") | ||||
| export class HaSelector extends LitElement { | ||||
| @@ -39,6 +44,8 @@ export class HaSelector extends LitElement { | ||||
|  | ||||
|   @property({ type: Boolean }) public required = true; | ||||
|  | ||||
|   @property() public context?: Record<string, any>; | ||||
|  | ||||
|   public focus() { | ||||
|     this.shadowRoot?.getElementById("selector")?.focus(); | ||||
|   } | ||||
| @@ -58,6 +65,7 @@ export class HaSelector extends LitElement { | ||||
|         disabled: this.disabled, | ||||
|         required: this.required, | ||||
|         helper: this.helper, | ||||
|         context: this.context, | ||||
|         id: "selector", | ||||
|       })} | ||||
|     `; | ||||
|   | ||||
| @@ -42,9 +42,7 @@ export class HaTab extends LitElement { | ||||
|         @keydown=${this._handleKeyDown} | ||||
|       > | ||||
|         ${this.narrow ? html`<slot name="icon"></slot>` : ""} | ||||
|         ${!this.narrow || this.active | ||||
|           ? html`<span class="name">${this.name}</span>` | ||||
|           : ""} | ||||
|         <span class="name">${this.name}</span> | ||||
|         ${this._shouldRenderRipple ? html`<mwc-ripple></mwc-ripple>` : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   | ||||
| @@ -9,6 +9,12 @@ export class HaTextField extends TextFieldBase { | ||||
|  | ||||
|   @property({ attribute: "error-message" }) public errorMessage?: string; | ||||
|  | ||||
|   // @ts-ignore | ||||
|   @property({ type: Boolean }) public icon?: boolean; | ||||
|  | ||||
|   // @ts-ignore | ||||
|   @property({ type: Boolean }) public iconTrailing?: boolean; | ||||
|  | ||||
|   override updated(changedProperties: PropertyValues) { | ||||
|     super.updated(changedProperties); | ||||
|     if ( | ||||
| @@ -53,6 +59,11 @@ export class HaTextField extends TextFieldBase { | ||||
|         padding-right: var(--text-field-suffix-padding-right, 0px); | ||||
|       } | ||||
|  | ||||
|       .mdc-text-field:not(.mdc-text-field--disabled) | ||||
|         .mdc-text-field__affix--suffix { | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|  | ||||
|       .mdc-text-field__icon { | ||||
|         color: var(--secondary-text-color); | ||||
|       } | ||||
|   | ||||
| @@ -31,6 +31,8 @@ export class HaYamlEditor extends LitElement { | ||||
|  | ||||
|   @property() public label?: string; | ||||
|  | ||||
|   @property({ type: Boolean }) public readOnly = false; | ||||
|  | ||||
|   @state() private _yaml = ""; | ||||
|  | ||||
|   public setValue(value): void { | ||||
| @@ -61,6 +63,7 @@ export class HaYamlEditor extends LitElement { | ||||
|       <ha-code-editor | ||||
|         .hass=${this.hass} | ||||
|         .value=${this._yaml} | ||||
|         .readOnly=${this.readOnly} | ||||
|         mode="yaml" | ||||
|         autocomplete-entities | ||||
|         .error=${this.isValid === false} | ||||
|   | ||||
| @@ -34,23 +34,24 @@ import { | ||||
|   MediaPickedEvent, | ||||
|   MediaPlayerBrowseAction, | ||||
| } from "../../data/media-player"; | ||||
| import { browseLocalMediaPlayer } from "../../data/media_source"; | ||||
| import { isTTSMediaSource } from "../../data/tts"; | ||||
| import { showAlertDialog } from "../../dialogs/generic/show-dialog-box"; | ||||
| import { installResizeObserver } from "../../panels/lovelace/common/install-resize-observer"; | ||||
| import { haStyle } from "../../resources/styles"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import { brandsUrl, extractDomainFromBrandUrl } from "../../util/brands-url"; | ||||
| import { documentationUrl } from "../../util/documentation-url"; | ||||
| import "../entity/ha-entity-picker"; | ||||
| import "../ha-button-menu"; | ||||
| import "../ha-card"; | ||||
| import type { HaCard } from "../ha-card"; | ||||
| import "../ha-circular-progress"; | ||||
| import "../ha-fab"; | ||||
| import "../ha-icon-button"; | ||||
| import "../ha-svg-icon"; | ||||
| import "../ha-fab"; | ||||
| import { browseLocalMediaPlayer } from "../../data/media_source"; | ||||
| import { isTTSMediaSource } from "../../data/tts"; | ||||
| import type { TtsMediaPickedEvent } from "./ha-browse-media-tts"; | ||||
| import "./ha-browse-media-tts"; | ||||
| import type { TtsMediaPickedEvent } from "./ha-browse-media-tts"; | ||||
|  | ||||
| declare global { | ||||
|   interface HASSDomEvents { | ||||
| @@ -681,6 +682,17 @@ export class HaMediaPlayerBrowse extends LitElement { | ||||
|                 // Thumbnails served by local API require authentication | ||||
|                 const signedPath = await getSignedPath(this.hass, thumbnailUrl); | ||||
|                 thumbnailUrl = signedPath.path; | ||||
|               } else if ( | ||||
|                 thumbnailUrl.startsWith("https://brands.home-assistant.io") | ||||
|               ) { | ||||
|                 // The backend is not aware of the theme used by the users, | ||||
|                 // so we rewrite the URL to show a proper icon | ||||
|                 thumbnailUrl = brandsUrl({ | ||||
|                   domain: extractDomainFromBrandUrl(thumbnailUrl), | ||||
|                   type: "icon", | ||||
|                   useFallback: true, | ||||
|                   darkOptimized: this.hass.themes?.darkMode, | ||||
|                 }); | ||||
|               } | ||||
|               thumbnailCard.style.backgroundImage = `url(${thumbnailUrl})`; | ||||
|               observer.unobserve(thumbnailCard); // loaded, so no need to observe anymore | ||||
|   | ||||
| @@ -35,7 +35,7 @@ class SearchInput extends LitElement { | ||||
|         .autofocus=${this.autofocus} | ||||
|         .label=${this.label || "Search"} | ||||
|         .value=${this.filter || ""} | ||||
|         .icon=${true} | ||||
|         icon | ||||
|         .iconTrailing=${this.filter || this.suffix} | ||||
|         @input=${this._filterInputChanged} | ||||
|       > | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import { | ||||
|   HistoryResult, | ||||
|   LineChartUnit, | ||||
|   TimelineEntity, | ||||
|   entityIdHistoryNeedsAttributes, | ||||
| } from "./history"; | ||||
|  | ||||
| export interface CacheConfig { | ||||
| @@ -53,7 +54,17 @@ export const getRecent = ( | ||||
|     return cache.data; | ||||
|   } | ||||
|  | ||||
|   const prom = fetchRecent(hass, entityId, startTime, endTime).then( | ||||
|   const noAttributes = !entityIdHistoryNeedsAttributes(hass, entityId); | ||||
|   const prom = fetchRecent( | ||||
|     hass, | ||||
|     entityId, | ||||
|     startTime, | ||||
|     endTime, | ||||
|     false, | ||||
|     undefined, | ||||
|     true, | ||||
|     noAttributes | ||||
|   ).then( | ||||
|     (stateHistory) => computeHistory(hass, stateHistory, localize), | ||||
|     (err) => { | ||||
|       delete RECENT_CACHE[entityId]; | ||||
| @@ -120,6 +131,7 @@ export const getRecentWithCache = ( | ||||
|   } | ||||
|  | ||||
|   const curCacheProm = cache.prom; | ||||
|   const noAttributes = !entityIdHistoryNeedsAttributes(hass, entityId); | ||||
|  | ||||
|   const genProm = async () => { | ||||
|     let fetchedHistory: HassEntity[][]; | ||||
| @@ -132,7 +144,10 @@ export const getRecentWithCache = ( | ||||
|           entityId, | ||||
|           toFetchStartTime, | ||||
|           endTime, | ||||
|           appendingToCache | ||||
|           appendingToCache, | ||||
|           undefined, | ||||
|           true, | ||||
|           noAttributes | ||||
|         ), | ||||
|       ]); | ||||
|       fetchedHistory = results[1]; | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import { AutomationConfig } from "./automation"; | ||||
| interface CloudStatusNotLoggedIn { | ||||
|   logged_in: false; | ||||
|   cloud: "disconnected" | "connecting" | "connected"; | ||||
|   http_use_ssl: boolean; | ||||
| } | ||||
|  | ||||
| export interface GoogleEntityConfig { | ||||
| @@ -59,6 +60,7 @@ export interface CloudStatusLoggedIn { | ||||
|   remote_connected: boolean; | ||||
|   remote_certificate: undefined | CertificateInformation; | ||||
|   http_use_ssl: boolean; | ||||
|   active_subscription: boolean; | ||||
| } | ||||
|  | ||||
| export type CloudStatus = CloudStatusNotLoggedIn | CloudStatusLoggedIn; | ||||
|   | ||||
| @@ -34,8 +34,24 @@ export const ERROR_STATES: ConfigEntry["state"][] = [ | ||||
|   "setup_retry", | ||||
| ]; | ||||
|  | ||||
| export const getConfigEntries = (hass: HomeAssistant) => | ||||
|   hass.callApi<ConfigEntry[]>("GET", "config/config_entries/entry"); | ||||
| export const getConfigEntries = ( | ||||
|   hass: HomeAssistant, | ||||
|   filters?: { type?: "helper" | "integration"; domain?: string } | ||||
| ): Promise<ConfigEntry[]> => { | ||||
|   const params = new URLSearchParams(); | ||||
|   if (filters) { | ||||
|     if (filters.type) { | ||||
|       params.append("type", filters.type); | ||||
|     } | ||||
|     if (filters.domain) { | ||||
|       params.append("domain", filters.domain); | ||||
|     } | ||||
|   } | ||||
|   return hass.callApi<ConfigEntry[]>( | ||||
|     "GET", | ||||
|     `config/config_entries/entry?${params.toString()}` | ||||
|   ); | ||||
| }; | ||||
|  | ||||
| export const updateConfigEntry = ( | ||||
|   hass: HomeAssistant, | ||||
|   | ||||
| @@ -65,8 +65,14 @@ export const ignoreConfigFlow = ( | ||||
| export const deleteConfigFlow = (hass: HomeAssistant, flowId: string) => | ||||
|   hass.callApi("DELETE", `config/config_entries/flow/${flowId}`); | ||||
|  | ||||
| export const getConfigFlowHandlers = (hass: HomeAssistant) => | ||||
|   hass.callApi<string[]>("GET", "config/config_entries/flow_handlers"); | ||||
| export const getConfigFlowHandlers = ( | ||||
|   hass: HomeAssistant, | ||||
|   type?: "helper" | "integration" | ||||
| ) => | ||||
|   hass.callApi<string[]>( | ||||
|     "GET", | ||||
|     `config/config_entries/flow_handlers${type ? `?type=${type}` : ""}` | ||||
|   ); | ||||
|  | ||||
| export const fetchConfigFlowInProgress = ( | ||||
|   conn: Connection | ||||
|   | ||||
| @@ -28,7 +28,7 @@ export interface DataEntryFlowStepForm { | ||||
|   step_id: string; | ||||
|   data_schema: HaFormSchema[]; | ||||
|   errors: Record<string, string>; | ||||
|   description_placeholders: Record<string, string>; | ||||
|   description_placeholders?: Record<string, string>; | ||||
|   last_step: boolean | null; | ||||
| } | ||||
|  | ||||
| @@ -49,7 +49,7 @@ export interface DataEntryFlowStepCreateEntry { | ||||
|   title: string; | ||||
|   result?: ConfigEntry; | ||||
|   description: string; | ||||
|   description_placeholders: Record<string, string>; | ||||
|   description_placeholders?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export interface DataEntryFlowStepAbort { | ||||
| @@ -57,7 +57,7 @@ export interface DataEntryFlowStepAbort { | ||||
|   flow_id: string; | ||||
|   handler: string; | ||||
|   reason: string; | ||||
|   description_placeholders: Record<string, string>; | ||||
|   description_placeholders?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export interface DataEntryFlowStepProgress { | ||||
| @@ -66,7 +66,17 @@ export interface DataEntryFlowStepProgress { | ||||
|   handler: string; | ||||
|   step_id: string; | ||||
|   progress_action: string; | ||||
|   description_placeholders: Record<string, string>; | ||||
|   description_placeholders?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export interface DataEntryFlowStepMenu { | ||||
|   type: "menu"; | ||||
|   flow_id: string; | ||||
|   handler: string; | ||||
|   step_id: string; | ||||
|   /** If array, use value to lookup translations in strings.json */ | ||||
|   menu_options: string[] | Record<string, string>; | ||||
|   description_placeholders?: Record<string, string>; | ||||
| } | ||||
|  | ||||
| export type DataEntryFlowStep = | ||||
| @@ -74,7 +84,8 @@ export type DataEntryFlowStep = | ||||
|   | DataEntryFlowStepExternal | ||||
|   | DataEntryFlowStepCreateEntry | ||||
|   | DataEntryFlowStepAbort | ||||
|   | DataEntryFlowStepProgress; | ||||
|   | DataEntryFlowStepProgress | ||||
|   | DataEntryFlowStepMenu; | ||||
|  | ||||
| export const subscribeDataEntryFlowProgressed = ( | ||||
|   conn: Connection, | ||||
|   | ||||
| @@ -12,7 +12,12 @@ import { subscribeOne } from "../common/util/subscribe-one"; | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { ConfigEntry, getConfigEntries } from "./config_entries"; | ||||
| import { subscribeEntityRegistry } from "./entity_registry"; | ||||
| import { fetchStatistics, Statistics } from "./history"; | ||||
| import { | ||||
|   fetchStatistics, | ||||
|   Statistics, | ||||
|   StatisticsMetaData, | ||||
|   getStatisticMetadata, | ||||
| } from "./history"; | ||||
|  | ||||
| const energyCollectionKeys: (string | undefined)[] = []; | ||||
|  | ||||
| @@ -136,6 +141,7 @@ export interface GasSourceTypeEnergyPreference { | ||||
|   entity_energy_from: string | null; | ||||
|   entity_energy_price: string | null; | ||||
|   number_energy_price: number | null; | ||||
|   unit_of_measurement?: string | null; | ||||
| } | ||||
|  | ||||
| type EnergySource = | ||||
| @@ -241,14 +247,14 @@ const getEnergyData = async ( | ||||
|   end?: Date | ||||
| ): Promise<EnergyData> => { | ||||
|   const [configEntries, entityRegistryEntries, info] = await Promise.all([ | ||||
|     getConfigEntries(hass), | ||||
|     getConfigEntries(hass, { domain: "co2signal" }), | ||||
|     subscribeOne(hass.connection, subscribeEntityRegistry), | ||||
|     getEnergyInfo(hass), | ||||
|   ]); | ||||
|  | ||||
|   const co2SignalConfigEntry = configEntries.find( | ||||
|     (entry) => entry.domain === "co2signal" | ||||
|   ); | ||||
|   const co2SignalConfigEntry = configEntries.length | ||||
|     ? configEntries[0] | ||||
|     : undefined; | ||||
|  | ||||
|   let co2SignalEntity: string | undefined; | ||||
|  | ||||
| @@ -271,6 +277,15 @@ const getEnergyData = async ( | ||||
|  | ||||
|   const consumptionStatIDs: string[] = []; | ||||
|   const statIDs: string[] = []; | ||||
|   const gasSources: GasSourceTypeEnergyPreference[] = | ||||
|     prefs.energy_sources.filter( | ||||
|       (source) => source.type === "gas" | ||||
|     ) as GasSourceTypeEnergyPreference[]; | ||||
|   const gasStatisticIdsWithMeta: StatisticsMetaData[] = | ||||
|     await getStatisticMetadata( | ||||
|       hass, | ||||
|       gasSources.map((source) => source.stat_energy_from) | ||||
|     ); | ||||
|  | ||||
|   for (const source of prefs.energy_sources) { | ||||
|     if (source.type === "solar") { | ||||
| @@ -280,6 +295,20 @@ const getEnergyData = async ( | ||||
|  | ||||
|     if (source.type === "gas") { | ||||
|       statIDs.push(source.stat_energy_from); | ||||
|       const entity = hass.states[source.stat_energy_from]; | ||||
|       if (!entity) { | ||||
|         for (const statisticIdWithMeta of gasStatisticIdsWithMeta) { | ||||
|           if ( | ||||
|             statisticIdWithMeta?.statistic_id === source.stat_energy_from && | ||||
|             statisticIdWithMeta?.unit_of_measurement | ||||
|           ) { | ||||
|             source.unit_of_measurement = | ||||
|               statisticIdWithMeta?.unit_of_measurement === "Wh" | ||||
|                 ? "kWh" | ||||
|                 : statisticIdWithMeta?.unit_of_measurement; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       if (source.stat_cost) { | ||||
|         statIDs.push(source.stat_cost); | ||||
|       } | ||||
| @@ -559,6 +588,9 @@ export const getEnergyGasUnit = ( | ||||
|         ? "kWh" | ||||
|         : entity.attributes.unit_of_measurement; | ||||
|     } | ||||
|     if (source.unit_of_measurement) { | ||||
|       return source.unit_of_measurement; | ||||
|     } | ||||
|   } | ||||
|   return undefined; | ||||
| }; | ||||
|   | ||||
| @@ -14,6 +14,7 @@ export interface EntityRegistryEntry { | ||||
|   device_id: string | null; | ||||
|   area_id: string | null; | ||||
|   disabled_by: string | null; | ||||
|   hidden_by: string | null; | ||||
|   entity_category: "config" | "diagnostic" | null; | ||||
| } | ||||
|  | ||||
| @@ -38,6 +39,7 @@ export interface EntityRegistryEntryUpdateParams { | ||||
|   device_class?: string | null; | ||||
|   area_id?: string | null; | ||||
|   disabled_by?: string | null; | ||||
|   hidden_by: string | null; | ||||
|   new_entity_id?: string; | ||||
| } | ||||
|  | ||||
|   | ||||
							
								
								
									
										71
									
								
								src/data/helpers_crud.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/data/helpers_crud.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| import { fetchCounter, updateCounter, deleteCounter } from "./counter"; | ||||
| import { | ||||
|   fetchInputBoolean, | ||||
|   updateInputBoolean, | ||||
|   deleteInputBoolean, | ||||
| } from "./input_boolean"; | ||||
| import { | ||||
|   fetchInputButton, | ||||
|   updateInputButton, | ||||
|   deleteInputButton, | ||||
| } from "./input_button"; | ||||
| import { | ||||
|   fetchInputDateTime, | ||||
|   updateInputDateTime, | ||||
|   deleteInputDateTime, | ||||
| } from "./input_datetime"; | ||||
| import { | ||||
|   fetchInputNumber, | ||||
|   updateInputNumber, | ||||
|   deleteInputNumber, | ||||
| } from "./input_number"; | ||||
| import { | ||||
|   fetchInputSelect, | ||||
|   updateInputSelect, | ||||
|   deleteInputSelect, | ||||
| } from "./input_select"; | ||||
| import { fetchInputText, updateInputText, deleteInputText } from "./input_text"; | ||||
| import { fetchTimer, updateTimer, deleteTimer } from "./timer"; | ||||
|  | ||||
| export const HELPERS_CRUD = { | ||||
|   input_boolean: { | ||||
|     fetch: fetchInputBoolean, | ||||
|     update: updateInputBoolean, | ||||
|     delete: deleteInputBoolean, | ||||
|   }, | ||||
|   input_button: { | ||||
|     fetch: fetchInputButton, | ||||
|     update: updateInputButton, | ||||
|     delete: deleteInputButton, | ||||
|   }, | ||||
|   input_text: { | ||||
|     fetch: fetchInputText, | ||||
|     update: updateInputText, | ||||
|     delete: deleteInputText, | ||||
|   }, | ||||
|   input_number: { | ||||
|     fetch: fetchInputNumber, | ||||
|     update: updateInputNumber, | ||||
|     delete: deleteInputNumber, | ||||
|   }, | ||||
|   input_datetime: { | ||||
|     fetch: fetchInputDateTime, | ||||
|     update: updateInputDateTime, | ||||
|     delete: deleteInputDateTime, | ||||
|   }, | ||||
|   input_select: { | ||||
|     fetch: fetchInputSelect, | ||||
|     update: updateInputSelect, | ||||
|     delete: deleteInputSelect, | ||||
|   }, | ||||
|   counter: { | ||||
|     fetch: fetchCounter, | ||||
|     update: updateCounter, | ||||
|     delete: deleteCounter, | ||||
|   }, | ||||
|   timer: { | ||||
|     fetch: fetchTimer, | ||||
|     update: updateTimer, | ||||
|     delete: deleteTimer, | ||||
|   }, | ||||
| }; | ||||
| @@ -1,4 +1,5 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { computeDomain } from "../common/entity/compute_domain"; | ||||
| import { computeStateDisplay } from "../common/entity/compute_state_display"; | ||||
| import { computeStateDomain } from "../common/entity/compute_state_domain"; | ||||
| import { computeStateName } from "../common/entity/compute_state_name"; | ||||
| @@ -7,6 +8,13 @@ import { HomeAssistant } from "../types"; | ||||
| import { FrontendLocaleData } from "./translation"; | ||||
|  | ||||
| const DOMAINS_USE_LAST_UPDATED = ["climate", "humidifier", "water_heater"]; | ||||
| const NEED_ATTRIBUTE_DOMAINS = [ | ||||
|   "climate", | ||||
|   "humidifier", | ||||
|   "input_datetime", | ||||
|   "thermostat", | ||||
|   "water_heater", | ||||
| ]; | ||||
| const LINE_ATTRIBUTES_TO_KEEP = [ | ||||
|   "temperature", | ||||
|   "current_temperature", | ||||
| @@ -131,6 +139,13 @@ export interface StatisticsValidationResults { | ||||
|   [statisticId: string]: StatisticsValidationResult[]; | ||||
| } | ||||
|  | ||||
| export const entityIdHistoryNeedsAttributes = ( | ||||
|   hass: HomeAssistant, | ||||
|   entityId: string | ||||
| ) => | ||||
|   !hass.states[entityId] || | ||||
|   NEED_ATTRIBUTE_DOMAINS.includes(computeDomain(entityId)); | ||||
|  | ||||
| export const fetchRecent = ( | ||||
|   hass: HomeAssistant, | ||||
|   entityId: string, | ||||
| @@ -138,7 +153,8 @@ export const fetchRecent = ( | ||||
|   endTime: Date, | ||||
|   skipInitialState = false, | ||||
|   significantChangesOnly?: boolean, | ||||
|   minimalResponse = true | ||||
|   minimalResponse = true, | ||||
|   noAttributes?: boolean | ||||
| ): Promise<HassEntity[][]> => { | ||||
|   let url = "history/period"; | ||||
|   if (startTime) { | ||||
| @@ -157,7 +173,9 @@ export const fetchRecent = ( | ||||
|   if (minimalResponse) { | ||||
|     url += "&minimal_response"; | ||||
|   } | ||||
|  | ||||
|   if (noAttributes) { | ||||
|     url += "&no_attributes"; | ||||
|   } | ||||
|   return hass.callApi("GET", url); | ||||
| }; | ||||
|  | ||||
| @@ -171,6 +189,10 @@ export const fetchDate = ( | ||||
|     "GET", | ||||
|     `history/period/${startTime.toISOString()}?end_time=${endTime.toISOString()}&minimal_response${ | ||||
|       entityId ? `&filter_entity_id=${entityId}` : `` | ||||
|     }${ | ||||
|       entityId && !entityIdHistoryNeedsAttributes(hass, entityId) | ||||
|         ? `&no_attributes` | ||||
|         : `` | ||||
|     }` | ||||
|   ); | ||||
|  | ||||
| @@ -278,6 +300,10 @@ const processLineChartEntities = ( | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| const stateUsesUnits = (state: HassEntity) => | ||||
|   "unit_of_measurement" in state.attributes || | ||||
|   "state_class" in state.attributes; | ||||
|  | ||||
| export const computeHistory = ( | ||||
|   hass: HomeAssistant, | ||||
|   stateHistory: HassEntity[][], | ||||
| @@ -294,16 +320,18 @@ export const computeHistory = ( | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const stateWithUnitorStateClass = stateInfo.find( | ||||
|       (state) => | ||||
|         state.attributes && | ||||
|         ("unit_of_measurement" in state.attributes || | ||||
|           "state_class" in state.attributes) | ||||
|     ); | ||||
|     const entityId = stateInfo[0].entity_id; | ||||
|     const currentState = | ||||
|       entityId in hass.states ? hass.states[entityId] : undefined; | ||||
|     const stateWithUnitorStateClass = | ||||
|       !currentState && | ||||
|       stateInfo.find((state) => state.attributes && stateUsesUnits(state)); | ||||
|  | ||||
|     let unit: string | undefined; | ||||
|  | ||||
|     if (stateWithUnitorStateClass) { | ||||
|     if (currentState && stateUsesUnits(currentState)) { | ||||
|       unit = currentState.attributes.unit_of_measurement || " "; | ||||
|     } else if (stateWithUnitorStateClass) { | ||||
|       unit = stateWithUnitorStateClass.attributes.unit_of_measurement || " "; | ||||
|     } else { | ||||
|       unit = { | ||||
| @@ -313,7 +341,7 @@ export const computeHistory = ( | ||||
|         input_number: "#", | ||||
|         number: "#", | ||||
|         water_heater: hass.config.unit_system.temperature, | ||||
|       }[computeStateDomain(stateInfo[0])]; | ||||
|       }[computeDomain(entityId)]; | ||||
|     } | ||||
|  | ||||
|     if (!unit) { | ||||
| @@ -345,6 +373,15 @@ export const getStatisticIds = ( | ||||
|     statistic_type, | ||||
|   }); | ||||
|  | ||||
| export const getStatisticMetadata = ( | ||||
|   hass: HomeAssistant, | ||||
|   statistic_ids?: string[] | ||||
| ) => | ||||
|   hass.callWS<StatisticsMetaData[]>({ | ||||
|     type: "recorder/get_statistics_metadata", | ||||
|     statistic_ids, | ||||
|   }); | ||||
|  | ||||
| export const fetchStatistics = ( | ||||
|   hass: HomeAssistant, | ||||
|   startTime: Date, | ||||
| @@ -428,3 +465,16 @@ export const statisticsHaveType = ( | ||||
|   stats: StatisticValue[], | ||||
|   type: StatisticType | ||||
| ) => stats.some((stat) => stat[type] !== null); | ||||
|  | ||||
| export const adjustStatisticsSum = ( | ||||
|   hass: HomeAssistant, | ||||
|   statistic_id: string, | ||||
|   start_time: string, | ||||
|   adjustment: number | ||||
| ): Promise<void> => | ||||
|   hass.callWS({ | ||||
|     type: "recorder/adjust_sum_statistics", | ||||
|     statistic_id, | ||||
|     start_time, | ||||
|     adjustment, | ||||
|   }); | ||||
|   | ||||
| @@ -29,7 +29,7 @@ export const createImage = async ( | ||||
|     body: fd, | ||||
|   }); | ||||
|   if (resp.status === 413) { | ||||
|     throw new Error("Uploaded image is too large"); | ||||
|     throw new Error(`Uploaded image is too large (${file.name})`); | ||||
|   } else if (resp.status !== 200) { | ||||
|     throw new Error("Unknown error"); | ||||
|   } | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { HomeAssistant } from "../types"; | ||||
|  | ||||
| export interface InputDateTime { | ||||
| @@ -17,6 +18,19 @@ export interface InputDateTimeMutableParams { | ||||
|   has_date: boolean; | ||||
| } | ||||
|  | ||||
| export const stateToIsoDateString = (entityState: HassEntity) => | ||||
|   `${entityState.attributes.year || "1970"}-${String( | ||||
|     entityState.attributes.month || "01" | ||||
|   ).padStart(2, "0")}-${String(entityState.attributes.day || "01").padStart( | ||||
|     2, | ||||
|     "0" | ||||
|   )}T${String(entityState.attributes.hour || "00").padStart(2, "0")}:${String( | ||||
|     entityState.attributes.minute || "00" | ||||
|   ).padStart(2, "0")}:${String(entityState.attributes.second || "00").padStart( | ||||
|     2, | ||||
|     "0" | ||||
|   )}`; | ||||
|  | ||||
| export const setInputDateTimeValue = ( | ||||
|   hass: HomeAssistant, | ||||
|   entityId: string, | ||||
|   | ||||
| @@ -43,7 +43,7 @@ export const uploadLocalMedia = async ( | ||||
|     } | ||||
|   ); | ||||
|   if (resp.status === 413) { | ||||
|     throw new Error("Uploaded image is too large"); | ||||
|     throw new Error(`Uploaded file is too large (${file.name})`); | ||||
|   } else if (resp.status !== 200) { | ||||
|     throw new Error("Unknown error"); | ||||
|   } | ||||
|   | ||||
							
								
								
									
										213
									
								
								src/data/ozw.ts
									
									
									
									
									
								
							
							
						
						
									
										213
									
								
								src/data/ozw.ts
									
									
									
									
									
								
							| @@ -1,213 +0,0 @@ | ||||
| import { HomeAssistant } from "../types"; | ||||
| import { DeviceRegistryEntry } from "./device_registry"; | ||||
|  | ||||
| export interface OZWNodeIdentifiers { | ||||
|   ozw_instance: number; | ||||
|   node_id: number; | ||||
| } | ||||
|  | ||||
| export interface OZWDevice { | ||||
|   node_id: number; | ||||
|   node_query_stage: string; | ||||
|   is_awake: boolean; | ||||
|   is_failed: boolean; | ||||
|   is_zwave_plus: boolean; | ||||
|   ozw_instance: number; | ||||
|   event: string; | ||||
|   node_manufacturer_name: string; | ||||
|   node_product_name: string; | ||||
| } | ||||
|  | ||||
| export interface OZWDeviceMetaDataResponse { | ||||
|   node_id: number; | ||||
|   ozw_instance: number; | ||||
|   metadata: OZWDeviceMetaData; | ||||
| } | ||||
|  | ||||
| export interface OZWDeviceMetaData { | ||||
|   OZWInfoURL: string; | ||||
|   ZWAProductURL: string; | ||||
|   ProductPic: string; | ||||
|   Description: string; | ||||
|   ProductManualURL: string; | ||||
|   ProductPageURL: string; | ||||
|   InclusionHelp: string; | ||||
|   ExclusionHelp: string; | ||||
|   ResetHelp: string; | ||||
|   WakeupHelp: string; | ||||
|   ProductSupportURL: string; | ||||
|   Frequency: string; | ||||
|   Name: string; | ||||
|   ProductPicBase64: string; | ||||
| } | ||||
|  | ||||
| export interface OZWInstance { | ||||
|   ozw_instance: number; | ||||
|   OZWDaemon_Version: string; | ||||
|   OpenZWave_Version: string; | ||||
|   QTOpenZWave_Version: string; | ||||
|   Status: string; | ||||
|   getControllerPath: string; | ||||
|   homeID: string; | ||||
| } | ||||
|  | ||||
| export interface OZWNetworkStatistics { | ||||
|   ozw_instance: number; | ||||
|   node_count: number; | ||||
|   readCnt: number; | ||||
|   writeCnt: number; | ||||
|   ACKCnt: number; | ||||
|   CANCnt: number; | ||||
|   NAKCnt: number; | ||||
|   dropped: number; | ||||
|   retries: number; | ||||
| } | ||||
|  | ||||
| export interface OZWDeviceConfig { | ||||
|   label: string; | ||||
|   type: string; | ||||
|   value: string | number; | ||||
|   parameter: number; | ||||
|   min: number; | ||||
|   max: number; | ||||
|   help: string; | ||||
| } | ||||
|  | ||||
| export const nodeQueryStages = [ | ||||
|   "ProtocolInfo", | ||||
|   "Probe", | ||||
|   "WakeUp", | ||||
|   "ManufacturerSpecific1", | ||||
|   "NodeInfo", | ||||
|   "NodePlusInfo", | ||||
|   "ManufacturerSpecific2", | ||||
|   "Versions", | ||||
|   "Instances", | ||||
|   "Static", | ||||
|   "CacheLoad", | ||||
|   "Associations", | ||||
|   "Neighbors", | ||||
|   "Session", | ||||
|   "Dynamic", | ||||
|   "Configuration", | ||||
|   "Complete", | ||||
| ]; | ||||
|  | ||||
| export const networkOnlineStatuses = [ | ||||
|   "driverAllNodesQueried", | ||||
|   "driverAllNodesQueriedSomeDead", | ||||
|   "driverAwakeNodesQueried", | ||||
| ]; | ||||
| export const networkStartingStatuses = [ | ||||
|   "starting", | ||||
|   "started", | ||||
|   "Ready", | ||||
|   "driverReady", | ||||
| ]; | ||||
| export const networkOfflineStatuses = [ | ||||
|   "Offline", | ||||
|   "stopped", | ||||
|   "driverFailed", | ||||
|   "driverReset", | ||||
|   "driverRemoved", | ||||
|   "driverAllNodesOnFire", | ||||
| ]; | ||||
|  | ||||
| export const getIdentifiersFromDevice = function ( | ||||
|   device: DeviceRegistryEntry | ||||
| ): OZWNodeIdentifiers | undefined { | ||||
|   if (!device) { | ||||
|     return undefined; | ||||
|   } | ||||
|  | ||||
|   const ozwIdentifier = device.identifiers.find( | ||||
|     (identifier) => identifier[0] === "ozw" | ||||
|   ); | ||||
|   if (!ozwIdentifier) { | ||||
|     return undefined; | ||||
|   } | ||||
|  | ||||
|   const identifiers = ozwIdentifier[1].split("."); | ||||
|   return { | ||||
|     node_id: parseInt(identifiers[1]), | ||||
|     ozw_instance: parseInt(identifiers[0]), | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| export const fetchOZWInstances = ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<OZWInstance[]> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/get_instances", | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNetworkStatus = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number | ||||
| ): Promise<OZWInstance> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/network_status", | ||||
|     ozw_instance, | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNetworkStatistics = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number | ||||
| ): Promise<OZWNetworkStatistics> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/network_statistics", | ||||
|     ozw_instance, | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNodes = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number | ||||
| ): Promise<OZWDevice[]> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/get_nodes", | ||||
|     ozw_instance, | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNodeStatus = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number, | ||||
|   node_id: number | ||||
| ): Promise<OZWDevice> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/node_status", | ||||
|     ozw_instance, | ||||
|     node_id, | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNodeMetadata = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number, | ||||
|   node_id: number | ||||
| ): Promise<OZWDeviceMetaDataResponse> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/node_metadata", | ||||
|     ozw_instance, | ||||
|     node_id, | ||||
|   }); | ||||
|  | ||||
| export const fetchOZWNodeConfig = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number, | ||||
|   node_id: number | ||||
| ): Promise<OZWDeviceConfig[]> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/get_config_parameters", | ||||
|     ozw_instance, | ||||
|     node_id, | ||||
|   }); | ||||
|  | ||||
| export const refreshNodeInfo = ( | ||||
|   hass: HomeAssistant, | ||||
|   ozw_instance: number, | ||||
|   node_id: number | ||||
| ): Promise<OZWDevice> => | ||||
|   hass.callWS({ | ||||
|     type: "ozw/refresh_node_info", | ||||
|     ozw_instance, | ||||
|     node_id, | ||||
|   }); | ||||
| @@ -1,51 +1,30 @@ | ||||
| export type Selector = | ||||
|   | ActionSelector | ||||
|   | AddonSelector | ||||
|   | AreaSelector | ||||
|   | AttributeSelector | ||||
|   | EntitySelector | ||||
|   | BooleanSelector | ||||
|   | ColorRGBSelector | ||||
|   | ColorTempSelector | ||||
|   | DateSelector | ||||
|   | DateTimeSelector | ||||
|   | DeviceSelector | ||||
|   | DurationSelector | ||||
|   | AreaSelector | ||||
|   | TargetSelector | ||||
|   | EntitySelector | ||||
|   | IconSelector | ||||
|   | LocationSelector | ||||
|   | MediaSelector | ||||
|   | NumberSelector | ||||
|   | BooleanSelector | ||||
|   | TimeSelector | ||||
|   | ActionSelector | ||||
|   | StringSelector | ||||
|   | ObjectSelector | ||||
|   | SelectSelector | ||||
|   | IconSelector | ||||
|   | MediaSelector | ||||
|   | ThemeSelector; | ||||
|   | StringSelector | ||||
|   | TargetSelector | ||||
|   | ThemeSelector | ||||
|   | TimeSelector; | ||||
|  | ||||
| export interface EntitySelector { | ||||
|   entity: { | ||||
|     integration?: string; | ||||
|     domain?: string | string[]; | ||||
|     device_class?: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface AttributeSelector { | ||||
|   attribute: { | ||||
|     entity_id: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface DeviceSelector { | ||||
|   device: { | ||||
|     integration?: string; | ||||
|     manufacturer?: string; | ||||
|     model?: string; | ||||
|     entity?: { | ||||
|       domain?: EntitySelector["entity"]["domain"]; | ||||
|       device_class?: EntitySelector["entity"]["device_class"]; | ||||
|     }; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface DurationSelector { | ||||
| export interface ActionSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   duration: {}; | ||||
|   action: {}; | ||||
| } | ||||
|  | ||||
| export interface AddonSelector { | ||||
| @@ -70,18 +49,101 @@ export interface AreaSelector { | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface TargetSelector { | ||||
|   target: { | ||||
| export interface AttributeSelector { | ||||
|   attribute: { | ||||
|     entity_id?: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface BooleanSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   boolean: {}; | ||||
| } | ||||
|  | ||||
| export interface ColorRGBSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   color_rgb: {}; | ||||
| } | ||||
|  | ||||
| export interface ColorTempSelector { | ||||
|   color_temp: { | ||||
|     min_mireds?: number; | ||||
|     max_mireds?: number; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface DateSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   date: {}; | ||||
| } | ||||
|  | ||||
| export interface DateTimeSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   datetime: {}; | ||||
| } | ||||
|  | ||||
| export interface DeviceSelector { | ||||
|   device: { | ||||
|     integration?: string; | ||||
|     manufacturer?: string; | ||||
|     model?: string; | ||||
|     entity?: { | ||||
|       integration?: EntitySelector["entity"]["integration"]; | ||||
|       domain?: EntitySelector["entity"]["domain"]; | ||||
|       device_class?: EntitySelector["entity"]["device_class"]; | ||||
|     }; | ||||
|     device?: { | ||||
|       integration?: DeviceSelector["device"]["integration"]; | ||||
|       manufacturer?: DeviceSelector["device"]["manufacturer"]; | ||||
|       model?: DeviceSelector["device"]["model"]; | ||||
|     }; | ||||
|     multiple?: boolean; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface DurationSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   duration: {}; | ||||
| } | ||||
|  | ||||
| export interface EntitySelector { | ||||
|   entity: { | ||||
|     integration?: string; | ||||
|     domain?: string | string[]; | ||||
|     device_class?: string; | ||||
|     multiple?: boolean; | ||||
|     use_uuid?: boolean; | ||||
|     include_entities?: string[]; | ||||
|     exclude_entities?: string[]; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface IconSelector { | ||||
|   icon: { | ||||
|     placeholder?: string; | ||||
|     fallbackPath?: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface LocationSelector { | ||||
|   location: { radius?: boolean; icon?: string }; | ||||
| } | ||||
|  | ||||
| export interface LocationSelectorValue { | ||||
|   latitude: number; | ||||
|   longitude: number; | ||||
|   radius?: number; | ||||
| } | ||||
|  | ||||
| export interface MediaSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   media: {}; | ||||
| } | ||||
|  | ||||
| export interface MediaSelectorValue { | ||||
|   entity_id?: string; | ||||
|   media_content_id?: string; | ||||
|   media_content_type?: string; | ||||
|   metadata?: { | ||||
|     title?: string; | ||||
|     thumbnail?: string | null; | ||||
|     media_class?: string; | ||||
|     children_media_class?: string | null; | ||||
|     navigateIds?: { media_content_type: string; media_content_id: string }[]; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| @@ -95,19 +157,20 @@ export interface NumberSelector { | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface BooleanSelector { | ||||
| export interface ObjectSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   boolean: {}; | ||||
|   object: {}; | ||||
| } | ||||
|  | ||||
| export interface TimeSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   time: {}; | ||||
| export interface SelectOption { | ||||
|   value: string; | ||||
|   label: string; | ||||
| } | ||||
|  | ||||
| export interface ActionSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   action: {}; | ||||
| export interface SelectSelector { | ||||
|   select: { | ||||
|     options: string[] | SelectOption[]; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface StringSelector { | ||||
| @@ -131,48 +194,25 @@ export interface StringSelector { | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface ObjectSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   object: {}; | ||||
| } | ||||
|  | ||||
| export interface SelectOption { | ||||
|   value: string; | ||||
|   label: string; | ||||
| } | ||||
|  | ||||
| export interface SelectSelector { | ||||
|   select: { | ||||
|     options: string[] | SelectOption[]; | ||||
| export interface TargetSelector { | ||||
|   target: { | ||||
|     entity?: { | ||||
|       integration?: EntitySelector["entity"]["integration"]; | ||||
|       domain?: EntitySelector["entity"]["domain"]; | ||||
|       device_class?: EntitySelector["entity"]["device_class"]; | ||||
|     }; | ||||
|     device?: { | ||||
|       integration?: DeviceSelector["device"]["integration"]; | ||||
|       manufacturer?: DeviceSelector["device"]["manufacturer"]; | ||||
|       model?: DeviceSelector["device"]["model"]; | ||||
|     }; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface IconSelector { | ||||
|   icon: { | ||||
|     placeholder?: string; | ||||
|     fallbackPath?: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface ThemeSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   theme: {}; | ||||
| } | ||||
|  | ||||
| export interface MediaSelector { | ||||
| export interface TimeSelector { | ||||
|   // eslint-disable-next-line @typescript-eslint/ban-types | ||||
|   media: {}; | ||||
| } | ||||
|  | ||||
| export interface MediaSelectorValue { | ||||
|   entity_id?: string; | ||||
|   media_content_id?: string; | ||||
|   media_content_type?: string; | ||||
|   metadata?: { | ||||
|     title?: string; | ||||
|     thumbnail?: string | null; | ||||
|     media_class?: string; | ||||
|     children_media_class?: string | null; | ||||
|     navigateIds?: { media_content_type: string; media_content_id: string }[]; | ||||
|   }; | ||||
|   time: {}; | ||||
| } | ||||
|   | ||||
| @@ -1,58 +0,0 @@ | ||||
| import { HomeAssistant } from "../../types"; | ||||
|  | ||||
| interface SupervisorBaseAvailableUpdates { | ||||
|   panel_path?: string; | ||||
|   update_type?: string; | ||||
|   version_latest?: string; | ||||
| } | ||||
|  | ||||
| interface SupervisorAddonAvailableUpdates | ||||
|   extends SupervisorBaseAvailableUpdates { | ||||
|   update_type?: "addon"; | ||||
|   icon?: string; | ||||
|   name?: string; | ||||
| } | ||||
|  | ||||
| interface SupervisorCoreAvailableUpdates | ||||
|   extends SupervisorBaseAvailableUpdates { | ||||
|   update_type?: "core"; | ||||
| } | ||||
|  | ||||
| interface SupervisorOsAvailableUpdates extends SupervisorBaseAvailableUpdates { | ||||
|   update_type?: "os"; | ||||
| } | ||||
|  | ||||
| interface SupervisorSupervisorAvailableUpdates | ||||
|   extends SupervisorBaseAvailableUpdates { | ||||
|   update_type?: "supervisor"; | ||||
| } | ||||
|  | ||||
| export type SupervisorAvailableUpdates = | ||||
|   | SupervisorAddonAvailableUpdates | ||||
|   | SupervisorCoreAvailableUpdates | ||||
|   | SupervisorOsAvailableUpdates | ||||
|   | SupervisorSupervisorAvailableUpdates; | ||||
|  | ||||
| export interface SupervisorAvailableUpdatesResponse { | ||||
|   available_updates: SupervisorAvailableUpdates[]; | ||||
| } | ||||
|  | ||||
| export const fetchSupervisorAvailableUpdates = async ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<SupervisorAvailableUpdates[]> => | ||||
|   ( | ||||
|     await hass.callWS<SupervisorAvailableUpdatesResponse>({ | ||||
|       type: "supervisor/api", | ||||
|       endpoint: "/available_updates", | ||||
|       method: "get", | ||||
|     }) | ||||
|   ).available_updates; | ||||
|  | ||||
| export const refreshSupervisorAvailableUpdates = async ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<void> => | ||||
|   hass.callWS<void>({ | ||||
|     type: "supervisor/api", | ||||
|     endpoint: "/refresh_updates", | ||||
|     method: "post", | ||||
|   }); | ||||
							
								
								
									
										36
									
								
								src/data/update.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								src/data/update.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| import type { | ||||
|   HassEntityAttributeBase, | ||||
|   HassEntityBase, | ||||
| } from "home-assistant-js-websocket"; | ||||
| import { supportsFeature } from "../common/entity/supports-feature"; | ||||
|  | ||||
| export const UPDATE_SUPPORT_INSTALL = 1; | ||||
| export const UPDATE_SUPPORT_SPECIFIC_VERSION = 2; | ||||
| export const UPDATE_SUPPORT_PROGRESS = 4; | ||||
| export const UPDATE_SUPPORT_BACKUP = 8; | ||||
|  | ||||
| interface UpdateEntityAttributes extends HassEntityAttributeBase { | ||||
|   current_version: string | null; | ||||
|   in_progress: boolean | number; | ||||
|   latest_version: string | null; | ||||
|   release_summary: string | null; | ||||
|   release_url: string | null; | ||||
|   skipped_version: string | null; | ||||
|   title: string | null; | ||||
| } | ||||
|  | ||||
| export interface UpdateEntity extends HassEntityBase { | ||||
|   attributes: UpdateEntityAttributes; | ||||
| } | ||||
|  | ||||
| export const updateUsesProgress = (entity: UpdateEntity): boolean => | ||||
|   supportsFeature(entity, UPDATE_SUPPORT_PROGRESS) && | ||||
|   typeof entity.attributes.in_progress === "number"; | ||||
|  | ||||
| export const updateCanInstall = (entity: UpdateEntity): boolean => | ||||
|   supportsFeature(entity, UPDATE_SUPPORT_INSTALL) && | ||||
|   entity.attributes.latest_version !== entity.attributes.current_version && | ||||
|   entity.attributes.latest_version !== entity.attributes.skipped_version; | ||||
|  | ||||
| export const updateIsInstalling = (entity: UpdateEntity): boolean => | ||||
|   updateUsesProgress(entity) || !!entity.attributes.in_progress; | ||||
| @@ -12,12 +12,12 @@ export interface Zone { | ||||
| } | ||||
|  | ||||
| export interface ZoneMutableParams { | ||||
|   name: string; | ||||
|   icon?: string; | ||||
|   latitude: number; | ||||
|   longitude: number; | ||||
|   name: string; | ||||
|   passive: boolean; | ||||
|   radius: number; | ||||
|   passive?: boolean; | ||||
|   radius?: number; | ||||
| } | ||||
|  | ||||
| export const fetchZones = (hass: HomeAssistant) => | ||||
|   | ||||
| @@ -1,81 +0,0 @@ | ||||
| import { HomeAssistant } from "../types"; | ||||
|  | ||||
| export interface ZWaveNetworkStatus { | ||||
|   state: number; | ||||
| } | ||||
|  | ||||
| export interface ZWaveValue { | ||||
|   key: number; | ||||
|   value: { | ||||
|     index: number; | ||||
|     instance: number; | ||||
|     label: string; | ||||
|     poll_intensity: number; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface ZWaveConfigItem { | ||||
|   key: number; | ||||
|   value: { | ||||
|     data: any; | ||||
|     data_items: any[]; | ||||
|     help: string; | ||||
|     label: string; | ||||
|     max: number; | ||||
|     min: number; | ||||
|     type: string; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export interface ZWaveConfigServiceData { | ||||
|   node_id: number; | ||||
|   parameter: number; | ||||
|   value: number | string; | ||||
| } | ||||
|  | ||||
| export interface ZWaveNode { | ||||
|   attributes: ZWaveAttributes; | ||||
| } | ||||
|  | ||||
| export interface ZWaveAttributes { | ||||
|   node_id: number; | ||||
|   wake_up_interval?: number; | ||||
| } | ||||
|  | ||||
| export interface ZWaveMigrationConfig { | ||||
|   usb_path: string; | ||||
|   network_key: string; | ||||
| } | ||||
|  | ||||
| export const ZWAVE_NETWORK_STATE_STOPPED = 0; | ||||
| export const ZWAVE_NETWORK_STATE_FAILED = 1; | ||||
| export const ZWAVE_NETWORK_STATE_STARTED = 5; | ||||
| export const ZWAVE_NETWORK_STATE_AWAKED = 7; | ||||
| export const ZWAVE_NETWORK_STATE_READY = 10; | ||||
|  | ||||
| export const fetchNetworkStatus = ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<ZWaveNetworkStatus> => | ||||
|   hass.callWS({ | ||||
|     type: "zwave/network_status", | ||||
|   }); | ||||
|  | ||||
| export const startZwaveJsConfigFlow = ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<{ flow_id: string }> => | ||||
|   hass.callWS({ | ||||
|     type: "zwave/start_zwave_js_config_flow", | ||||
|   }); | ||||
|  | ||||
| export const fetchMigrationConfig = ( | ||||
|   hass: HomeAssistant | ||||
| ): Promise<ZWaveMigrationConfig> => | ||||
|   hass.callWS({ | ||||
|     type: "zwave/get_migration_config", | ||||
|   }); | ||||
|  | ||||
| export const fetchValues = (hass: HomeAssistant, nodeId: number) => | ||||
|   hass.callApi<ZWaveValue[]>("GET", `zwave/values/${nodeId}`); | ||||
|  | ||||
| export const fetchNodeConfig = (hass: HomeAssistant, nodeId: number) => | ||||
|   hass.callApi<ZWaveConfigItem[]>("GET", `zwave/config/${nodeId}`); | ||||
| @@ -1,5 +1,5 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { mdiClose } from "@mdi/js"; | ||||
| import { mdiClose, mdiHelpCircle } from "@mdi/js"; | ||||
| import type { UnsubscribeFunc } from "home-assistant-js-websocket"; | ||||
| import { | ||||
|   css, | ||||
| @@ -33,6 +33,7 @@ import { | ||||
| } from "../../data/device_registry"; | ||||
| import { haStyleDialog } from "../../resources/styles"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import { documentationUrl } from "../../util/documentation-url"; | ||||
| import { showAlertDialog } from "../generic/show-dialog-box"; | ||||
| import { | ||||
|   DataEntryFlowDialogParams, | ||||
| @@ -46,6 +47,7 @@ import "./step-flow-loading"; | ||||
| import "./step-flow-pick-flow"; | ||||
| import "./step-flow-pick-handler"; | ||||
| import "./step-flow-progress"; | ||||
| import "./step-flow-menu"; | ||||
|  | ||||
| let instance = 0; | ||||
|  | ||||
| @@ -117,13 +119,17 @@ class DataEntryFlowDialog extends LitElement { | ||||
|         ); | ||||
|       } catch (err: any) { | ||||
|         this.closeDialog(); | ||||
|         let message = err.message || err.body || "Unknown error"; | ||||
|         if (typeof message !== "string") { | ||||
|           message = JSON.stringify(message); | ||||
|         } | ||||
|         showAlertDialog(this, { | ||||
|           title: this.hass.localize( | ||||
|             "ui.panel.config.integrations.config_flow.error" | ||||
|           ), | ||||
|           text: `${this.hass.localize( | ||||
|             "ui.panel.config.integrations.config_flow.could_not_load" | ||||
|           )}: ${err.message || err.body}`, | ||||
|           )}: ${message}`, | ||||
|         }); | ||||
|         return; | ||||
|       } | ||||
| @@ -231,14 +237,35 @@ class DataEntryFlowDialog extends LitElement { | ||||
|               // to reset the element. | ||||
|               "" | ||||
|             : html` | ||||
|                 <ha-icon-button | ||||
|                   .label=${this.hass.localize( | ||||
|                     "ui.panel.config.integrations.config_flow.dismiss" | ||||
|                   )} | ||||
|                   .path=${mdiClose} | ||||
|                   dialogAction="close" | ||||
|                   ?rtl=${computeRTL(this.hass)} | ||||
|                 ></ha-icon-button> | ||||
|                 <div class="dialog-actions"> | ||||
|                   ${["form", "menu", "external"].includes( | ||||
|                     this._step?.type as any | ||||
|                   ) | ||||
|                     ? html` | ||||
|                         <a | ||||
|                           href=${documentationUrl( | ||||
|                             this.hass, | ||||
|                             `/integrations/${this._step!.handler}` | ||||
|                           )} | ||||
|                           target="_blank" | ||||
|                           rel="noreferrer noopener" | ||||
|                           ><ha-icon-button | ||||
|                             .label=${this.hass.localize("ui.common.help")} | ||||
|                             .path=${mdiHelpCircle} | ||||
|                             ?rtl=${computeRTL(this.hass)} | ||||
|                           ></ha-icon-button | ||||
|                         ></a> | ||||
|                       ` | ||||
|                     : ""} | ||||
|                   <ha-icon-button | ||||
|                     .label=${this.hass.localize( | ||||
|                       "ui.panel.config.integrations.config_flow.dismiss" | ||||
|                     )} | ||||
|                     .path=${mdiClose} | ||||
|                     dialogAction="close" | ||||
|                     ?rtl=${computeRTL(this.hass)} | ||||
|                   ></ha-icon-button> | ||||
|                 </div> | ||||
|                 ${this._step === null | ||||
|                   ? this._handler | ||||
|                     ? html`<step-flow-pick-flow | ||||
| @@ -288,6 +315,14 @@ class DataEntryFlowDialog extends LitElement { | ||||
|                         .hass=${this.hass} | ||||
|                       ></step-flow-progress> | ||||
|                     ` | ||||
|                   : this._step.type === "menu" | ||||
|                   ? html` | ||||
|                       <step-flow-menu | ||||
|                         .flowConfig=${this._params.flowConfig} | ||||
|                         .step=${this._step} | ||||
|                         .hass=${this.hass} | ||||
|                       ></step-flow-menu> | ||||
|                     ` | ||||
|                   : this._devices === undefined || this._areas === undefined | ||||
|                   ? // When it's a create entry result, we will fetch device & area registry | ||||
|                     html` | ||||
| @@ -373,13 +408,20 @@ class DataEntryFlowDialog extends LitElement { | ||||
|         step = await this._params!.flowConfig.createFlow(this.hass, handler); | ||||
|       } catch (err: any) { | ||||
|         this.closeDialog(); | ||||
|         const message = | ||||
|           err?.status_code === 404 | ||||
|             ? this.hass.localize( | ||||
|                 "ui.panel.config.integrations.config_flow.no_config_flow" | ||||
|               ) | ||||
|             : `${this.hass.localize( | ||||
|                 "ui.panel.config.integrations.config_flow.could_not_load" | ||||
|               )}: ${err?.body?.message || err?.message}`; | ||||
|  | ||||
|         showAlertDialog(this, { | ||||
|           title: this.hass.localize( | ||||
|             "ui.panel.config.integrations.config_flow.error" | ||||
|           ), | ||||
|           text: `${this.hass.localize( | ||||
|             "ui.panel.config.integrations.config_flow.could_not_load" | ||||
|           )}: ${err.message || err.body}`, | ||||
|           text: message, | ||||
|         }); | ||||
|         return; | ||||
|       } finally { | ||||
| @@ -410,7 +452,7 @@ class DataEntryFlowDialog extends LitElement { | ||||
|           title: this.hass.localize( | ||||
|             "ui.panel.config.integrations.config_flow.error" | ||||
|           ), | ||||
|           text: err.message || err.body, | ||||
|           text: err?.body?.message, | ||||
|         }); | ||||
|         return; | ||||
|       } finally { | ||||
| @@ -452,16 +494,19 @@ class DataEntryFlowDialog extends LitElement { | ||||
|         ha-dialog { | ||||
|           --dialog-content-padding: 0; | ||||
|         } | ||||
|         ha-icon-button { | ||||
|         .dialog-actions { | ||||
|           padding: 16px; | ||||
|           position: absolute; | ||||
|           top: 0; | ||||
|           right: 0; | ||||
|         } | ||||
|         ha-icon-button[rtl] { | ||||
|         .dialog-actions[rtl] { | ||||
|           right: auto; | ||||
|           left: 0; | ||||
|         } | ||||
|         .dialog-actions > * { | ||||
|           color: var(--secondary-text-color); | ||||
|         } | ||||
|       `, | ||||
|     ]; | ||||
|   } | ||||
|   | ||||
| @@ -24,7 +24,7 @@ export const showConfigFlowDialog = ( | ||||
|     loadDevicesAndAreas: true, | ||||
|     getFlowHandlers: async (hass) => { | ||||
|       const [handlers] = await Promise.all([ | ||||
|         getConfigFlowHandlers(hass), | ||||
|         getConfigFlowHandlers(hass, "integration"), | ||||
|         hass.loadBackendTranslation("title", undefined, true), | ||||
|       ]); | ||||
|  | ||||
| @@ -181,6 +181,33 @@ export const showConfigFlowDialog = ( | ||||
|         : ""; | ||||
|     }, | ||||
|  | ||||
|     renderMenuHeader(hass, step) { | ||||
|       return ( | ||||
|         hass.localize( | ||||
|           `component.${step.handler}.config.step.${step.step_id}.title` | ||||
|         ) || hass.localize(`component.${step.handler}.title`) | ||||
|       ); | ||||
|     }, | ||||
|  | ||||
|     renderMenuDescription(hass, step) { | ||||
|       const description = hass.localize( | ||||
|         `component.${step.handler}.config.step.${step.step_id}.description`, | ||||
|         step.description_placeholders | ||||
|       ); | ||||
|       return description | ||||
|         ? html` | ||||
|             <ha-markdown allowsvg breaks .content=${description}></ha-markdown> | ||||
|           ` | ||||
|         : ""; | ||||
|     }, | ||||
|  | ||||
|     renderMenuOption(hass, step, option) { | ||||
|       return hass.localize( | ||||
|         `component.${step.handler}.config.step.${step.step_id}.menu_options.${option}`, | ||||
|         step.description_placeholders | ||||
|       ); | ||||
|     }, | ||||
|  | ||||
|     renderLoadingDescription(hass, reason, handler, step) { | ||||
|       if (!["loading_flow", "loading_step"].includes(reason)) { | ||||
|         return ""; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import { | ||||
|   DataEntryFlowStepCreateEntry, | ||||
|   DataEntryFlowStepExternal, | ||||
|   DataEntryFlowStepForm, | ||||
|   DataEntryFlowStepMenu, | ||||
|   DataEntryFlowStepProgress, | ||||
| } from "../../data/data_entry_flow"; | ||||
| import { HomeAssistant } from "../../types"; | ||||
| @@ -80,6 +81,19 @@ export interface FlowConfig { | ||||
|     step: DataEntryFlowStepProgress | ||||
|   ): TemplateResult | ""; | ||||
|  | ||||
|   renderMenuHeader(hass: HomeAssistant, step: DataEntryFlowStepMenu): string; | ||||
|  | ||||
|   renderMenuDescription( | ||||
|     hass: HomeAssistant, | ||||
|     step: DataEntryFlowStepMenu | ||||
|   ): TemplateResult | ""; | ||||
|  | ||||
|   renderMenuOption( | ||||
|     hass: HomeAssistant, | ||||
|     step: DataEntryFlowStepMenu, | ||||
|     option: string | ||||
|   ): string; | ||||
|  | ||||
|   renderLoadingDescription( | ||||
|     hass: HomeAssistant, | ||||
|     loadingReason: LoadingReason, | ||||
|   | ||||
| @@ -134,6 +134,37 @@ export const showOptionsFlowDialog = ( | ||||
|           : ""; | ||||
|       }, | ||||
|  | ||||
|       renderMenuHeader(hass, step) { | ||||
|         return ( | ||||
|           hass.localize( | ||||
|             `component.${step.handler}.option.step.${step.step_id}.title` | ||||
|           ) || hass.localize(`component.${step.handler}.title`) | ||||
|         ); | ||||
|       }, | ||||
|  | ||||
|       renderMenuDescription(hass, step) { | ||||
|         const description = hass.localize( | ||||
|           `component.${step.handler}.option.step.${step.step_id}.description`, | ||||
|           step.description_placeholders | ||||
|         ); | ||||
|         return description | ||||
|           ? html` | ||||
|               <ha-markdown | ||||
|                 allowsvg | ||||
|                 breaks | ||||
|                 .content=${description} | ||||
|               ></ha-markdown> | ||||
|             ` | ||||
|           : ""; | ||||
|       }, | ||||
|  | ||||
|       renderMenuOption(hass, step, option) { | ||||
|         return hass.localize( | ||||
|           `component.${step.handler}.options.step.${step.step_id}.menu_options.${option}`, | ||||
|           step.description_placeholders | ||||
|         ); | ||||
|       }, | ||||
|  | ||||
|       renderLoadingDescription(hass, reason) { | ||||
|         return ( | ||||
|           hass.localize(`component.${configEntry.domain}.options.loading`) || | ||||
|   | ||||
							
								
								
									
										96
									
								
								src/dialogs/config-flow/step-flow-menu.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								src/dialogs/config-flow/step-flow-menu.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import type { DataEntryFlowStepMenu } from "../../data/data_entry_flow"; | ||||
| import type { HomeAssistant } from "../../types"; | ||||
| import type { FlowConfig } from "./show-dialog-data-entry-flow"; | ||||
| import "../../components/ha-icon-next"; | ||||
| import { configFlowContentStyles } from "./styles"; | ||||
| import { fireEvent } from "../../common/dom/fire_event"; | ||||
|  | ||||
| @customElement("step-flow-menu") | ||||
| class StepFlowMenu extends LitElement { | ||||
|   @property({ attribute: false }) public flowConfig!: FlowConfig; | ||||
|  | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
|  | ||||
|   @property({ attribute: false }) public step!: DataEntryFlowStepMenu; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     let options: string[]; | ||||
|     let translations: Record<string, string>; | ||||
|  | ||||
|     if (Array.isArray(this.step.menu_options)) { | ||||
|       options = this.step.menu_options; | ||||
|       translations = {}; | ||||
|       for (const option of options) { | ||||
|         translations[option] = this.flowConfig.renderMenuOption( | ||||
|           this.hass, | ||||
|           this.step, | ||||
|           option | ||||
|         ); | ||||
|       } | ||||
|     } else { | ||||
|       options = Object.keys(this.step.menu_options); | ||||
|       translations = this.step.menu_options; | ||||
|     } | ||||
|  | ||||
|     const description = this.flowConfig.renderMenuDescription( | ||||
|       this.hass, | ||||
|       this.step | ||||
|     ); | ||||
|  | ||||
|     return html` | ||||
|       <h2>${this.flowConfig.renderMenuHeader(this.hass, this.step)}</h2> | ||||
|       ${description ? html`<div class="content">${description}</div>` : ""} | ||||
|       <div class="options"> | ||||
|         ${options.map( | ||||
|           (option) => html` | ||||
|             <mwc-list-item hasMeta .step=${option} @click=${this._handleStep}> | ||||
|               <span>${translations[option]}</span> | ||||
|               <ha-icon-next slot="meta"></ha-icon-next> | ||||
|             </mwc-list-item> | ||||
|           ` | ||||
|         )} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private _handleStep(ev) { | ||||
|     fireEvent(this, "flow-update", { | ||||
|       stepPromise: this.flowConfig.handleFlowStep( | ||||
|         this.hass, | ||||
|         this.step.flow_id, | ||||
|         { | ||||
|           next_step_id: ev.currentTarget.step, | ||||
|         } | ||||
|       ), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = [ | ||||
|     configFlowContentStyles, | ||||
|     css` | ||||
|       .options { | ||||
|         margin-top: 20px; | ||||
|         margin-bottom: 8px; | ||||
|       } | ||||
|       .content { | ||||
|         padding-bottom: 16px; | ||||
|         border-bottom: 1px solid var(--divider-color); | ||||
|       } | ||||
|       .content + .options { | ||||
|         margin-top: 8px; | ||||
|       } | ||||
|       mwc-list-item { | ||||
|         --mdc-list-side-padding: 24px; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "step-flow-menu": StepFlowMenu; | ||||
|   } | ||||
| } | ||||
| @@ -216,15 +216,16 @@ class StepFlowPickHandler extends LitElement { | ||||
|  | ||||
|     if (handler.is_add) { | ||||
|       if (handler.slug === "zwave_js") { | ||||
|         const entries = await getConfigEntries(this.hass); | ||||
|         const entry = entries.find((ent) => ent.domain === "zwave_js"); | ||||
|         const entries = await getConfigEntries(this.hass, { | ||||
|           domain: "zwave_js", | ||||
|         }); | ||||
|  | ||||
|         if (!entry) { | ||||
|         if (!entries.length) { | ||||
|           return; | ||||
|         } | ||||
|  | ||||
|         showZWaveJSAddNodeDialog(this, { | ||||
|           entry_id: entry.entry_id, | ||||
|           entry_id: entries[0].entry_id, | ||||
|         }); | ||||
|       } else if (handler.slug === "zha") { | ||||
|         navigate("/config/zha/add"); | ||||
|   | ||||
| @@ -12,7 +12,7 @@ import { | ||||
| import type { HomeAssistant } from "../../../types"; | ||||
|  | ||||
| const BUTTONS = ["1", "2", "3", "4", "5", "6", "7", "8", "9", "", "0", "clear"]; | ||||
| const ARM_ACTIONS = ["arm_away", "arm_home"]; | ||||
| const ARM_ACTIONS = ["arm_home", "arm_away"]; | ||||
| const DISARM_ACTIONS = ["disarm"]; | ||||
|  | ||||
| @customElement("more-info-alarm_control_panel") | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
| import { | ||||
|   css, | ||||
|   CSSResultGroup, | ||||
| @@ -12,6 +13,7 @@ import { fireEvent } from "../../../common/dom/fire_event"; | ||||
| import { stopPropagation } from "../../../common/dom/stop_propagation"; | ||||
| import { supportsFeature } from "../../../common/entity/supports-feature"; | ||||
| import { computeRTLDirection } from "../../../common/util/compute_rtl"; | ||||
| import "../../../components/ha-select"; | ||||
| import "../../../components/ha-slider"; | ||||
| import "../../../components/ha-switch"; | ||||
| import { | ||||
| @@ -19,8 +21,6 @@ import { | ||||
|   HUMIDIFIER_SUPPORT_MODES, | ||||
| } from "../../../data/humidifier"; | ||||
| import { HomeAssistant } from "../../../types"; | ||||
| import "@material/mwc-list/mwc-list"; | ||||
| import "@material/mwc-list/mwc-list-item"; | ||||
|  | ||||
| class MoreInfoHumidifier extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
| @@ -67,26 +67,24 @@ class MoreInfoHumidifier extends LitElement { | ||||
|  | ||||
|         ${supportModes | ||||
|           ? html` | ||||
|               <div class="container-modes"> | ||||
|                 <mwc-list | ||||
|                   .label=${hass.localize("ui.card.humidifier.mode")} | ||||
|                   .value=${stateObj.attributes.mode} | ||||
|                   fixedMenuPosition | ||||
|                   naturalMenuWidth | ||||
|                   @selected=${this._handleModeChanged} | ||||
|                   @closed=${stopPropagation} | ||||
|                 > | ||||
|                   ${stateObj.attributes.available_modes!.map( | ||||
|                     (mode) => html` | ||||
|                       <mwc-list-item .value=${mode}> | ||||
|                         ${hass.localize( | ||||
|                           `state_attributes.humidifier.mode.${mode}` | ||||
|                         ) || mode} | ||||
|                       </mwc-list-item> | ||||
|                     ` | ||||
|                   )} | ||||
|                 </mwc-list> | ||||
|               </div> | ||||
|               <ha-select | ||||
|                 .label=${hass.localize("ui.card.humidifier.mode")} | ||||
|                 .value=${stateObj.attributes.mode} | ||||
|                 fixedMenuPosition | ||||
|                 naturalMenuWidth | ||||
|                 @selected=${this._handleModeChanged} | ||||
|                 @closed=${stopPropagation} | ||||
|               > | ||||
|                 ${stateObj.attributes.available_modes!.map( | ||||
|                   (mode) => html` | ||||
|                     <mwc-list-item .value=${mode}> | ||||
|                       ${hass.localize( | ||||
|                         `state_attributes.humidifier.mode.${mode}` | ||||
|                       ) || mode} | ||||
|                     </mwc-list-item> | ||||
|                   ` | ||||
|                 )} | ||||
|               </ha-select> | ||||
|             ` | ||||
|           : ""} | ||||
|       </div> | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user