├── .browserslistrc
├── .devcontainer
├── Dockerfile
├── devcontainer.json
└── post_create.sh
├── .gitattributes
├── .github
├── ISSUE_TEMPLATE.md
├── ISSUE_TEMPLATE
│ ├── bug_report.yml
│ └── config.yml
├── PULL_REQUEST_TEMPLATE.md
├── dependabot.yml
├── labeler.yml
├── move.yml
├── release-drafter.yml
└── workflows
│ ├── cast_deployment.yaml
│ ├── ci.yaml
│ ├── codeql-analysis.yml
│ ├── demo_deployment.yaml
│ ├── design_deployment.yaml
│ ├── design_preview.yaml
│ ├── labeler.yaml
│ ├── lock.yml
│ ├── nightly.yaml
│ ├── relative-ci.yaml
│ ├── release-drafter.yaml
│ ├── release.yaml
│ ├── stale.yml
│ └── translations.yaml
├── .gitignore
├── .gitmodules
├── .husky
└── pre-commit
├── .nvmrc
├── .prettierignore
├── .vscode
├── extensions.json
├── launch.json
└── tasks.json
├── .yarn
├── patches
│ ├── @material-mwc-formfield-npm-0.27.0-9528cb60f6.patch
│ ├── @material-mwc-list-npm-0.27.0-5344fc9de4.patch
│ ├── leaflet-draw-npm-1.0.4-0ca0ebcf65.patch
│ ├── sortablejs-npm-1.15.6-3235a8f83b.patch
│ └── workbox-build-npm-7.1.1-a854f3faae.patch
└── releases
│ └── yarn-4.9.1.cjs
├── .yarnrc.yml
├── CLA.md
├── CODE_OF_CONDUCT.md
├── LICENSE.md
├── MANIFEST.in
├── README.md
├── build-scripts
├── README.md
├── babel-plugins
│ ├── custom-polyfill-plugin.js
│ └── inline-constants-plugin.cjs
├── bundle.cjs
├── env.cjs
├── eslint.config.mjs
├── gulp
│ ├── app.js
│ ├── cast.js
│ ├── clean.js
│ ├── compress.js
│ ├── demo.js
│ ├── download-translations.js
│ ├── entry-html.js
│ ├── fetch-nightly-translations.js
│ ├── gallery.js
│ ├── gather-static.js
│ ├── gen-icons-json.js
│ ├── hassio.js
│ ├── index.mjs
│ ├── landing-page.js
│ ├── locale-data.js
│ ├── rspack.js
│ ├── service-worker.js
│ └── translations.js
├── list-plugins-and-polyfills.js
├── paths.cjs
├── removedIcons.json
└── rspack.cjs
├── cast
├── README.md
├── public
│ ├── _headers
│ ├── _redirects
│ ├── images
│ │ ├── arsaboo.jpg
│ │ ├── favicon.ico
│ │ ├── google-nest-hub.png
│ │ ├── ha-cast-icon.png
│ │ ├── melody.jpg
│ │ └── nabu-loves-hass.png
│ ├── manifest.json
│ ├── sw-legacy.js
│ └── sw-modern.js
├── script
│ ├── build_cast
│ ├── develop_cast
│ └── upload
└── src
│ ├── html
│ ├── _social_meta.html.template
│ ├── faq.html.template
│ ├── index.html.template
│ ├── media.html.template
│ └── receiver.html.template
│ ├── launcher
│ ├── entrypoint.ts
│ └── layout
│ │ ├── hc-cast.ts
│ │ ├── hc-connect.ts
│ │ └── hc-layout.ts
│ ├── media
│ └── entrypoint.ts
│ └── receiver
│ ├── cast_context.ts
│ ├── cast_framework.ts
│ ├── demo
│ ├── cast-demo-entities.ts
│ └── cast-demo-lovelace.ts
│ ├── entrypoint.ts
│ ├── layout
│ ├── hc-demo.ts
│ ├── hc-launch-screen.ts
│ ├── hc-lovelace.ts
│ └── hc-main.ts
│ └── types.ts
├── demo
├── public
│ ├── _headers
│ ├── assets
│ │ ├── arsaboo
│ │ │ ├── floorplans
│ │ │ │ ├── ecobee_blank.png
│ │ │ │ ├── main.png
│ │ │ │ └── second.png
│ │ │ ├── icons
│ │ │ │ ├── Harmony.png
│ │ │ │ ├── abode_disabled.png
│ │ │ │ ├── abode_enabled.png
│ │ │ │ ├── automation_disabled.png
│ │ │ │ ├── automation_enabled.png
│ │ │ │ ├── camera_backyard_recording.png
│ │ │ │ ├── camera_backyard_streaming.png
│ │ │ │ ├── camera_driveway_recording.png
│ │ │ │ ├── camera_driveway_streaming.png
│ │ │ │ ├── camera_patio_recording.png
│ │ │ │ ├── camera_patio_streaming.png
│ │ │ │ ├── camera_porch_recording.png
│ │ │ │ ├── camera_porch_streaming.png
│ │ │ │ ├── ecobee_blank.png
│ │ │ │ ├── garage_door_closed.png
│ │ │ │ ├── garage_door_open.png
│ │ │ │ ├── light_bulb_off.png
│ │ │ │ ├── light_bulb_on.png
│ │ │ │ ├── light_off.png
│ │ │ │ ├── light_on.png
│ │ │ │ ├── security_armed_red.png
│ │ │ │ ├── security_disarmed.png
│ │ │ │ ├── tv_disabled.png
│ │ │ │ ├── tv_enabled.png
│ │ │ │ ├── tv_off2.png
│ │ │ │ └── tv_on2.png
│ │ │ └── images
│ │ │ │ ├── arsaboo.jpg
│ │ │ │ ├── camera.backyard.jpg
│ │ │ │ ├── camera.driveway.jpg
│ │ │ │ ├── camera.patio.jpg
│ │ │ │ ├── camera.porch.jpg
│ │ │ │ ├── media_player_family_room.jpg
│ │ │ │ └── melody.jpg
│ │ ├── jimpower
│ │ │ ├── background-15.jpg
│ │ │ ├── cardbackK.png
│ │ │ ├── home
│ │ │ │ ├── bus_10.jpg
│ │ │ │ ├── git.png
│ │ │ │ ├── house_4.png
│ │ │ │ ├── james_10.jpg
│ │ │ │ └── tina_4.jpg
│ │ │ └── security
│ │ │ │ ├── air_8.jpg
│ │ │ │ ├── alarm_3.jpg
│ │ │ │ ├── door_3.png
│ │ │ │ ├── leak_2.png
│ │ │ │ ├── motion_3.jpg
│ │ │ │ ├── smoke_4.jpg
│ │ │ │ └── window_2.jpg
│ │ ├── kernehed
│ │ │ ├── bella.jpg
│ │ │ ├── camera.entre.jpg
│ │ │ └── oscar.jpg
│ │ ├── sections
│ │ │ └── images
│ │ │ │ └── media_player_family_room.jpg
│ │ └── teachingbirds
│ │ │ ├── House_square.jpg
│ │ │ ├── Stefan_square.jpg
│ │ │ ├── background_square.png
│ │ │ ├── cleaning_square.jpg
│ │ │ ├── clothes_drying_square.jpg
│ │ │ ├── dryer_square.jpg
│ │ │ ├── folded_clothes_square.jpg
│ │ │ ├── guests_square.jpg
│ │ │ ├── isa_square.jpg
│ │ │ ├── laundry_clean_2_square.jpg
│ │ │ ├── laundry_running_square.jpg
│ │ │ ├── mailbox_bw_square.jpg
│ │ │ ├── mailbox_square.jpg
│ │ │ ├── meteogram.png
│ │ │ ├── plants.png
│ │ │ ├── radiator_off.jpg
│ │ │ ├── radiator_on.jpg
│ │ │ ├── roomba_bw_square.jpg
│ │ │ ├── roomba_square.jpg
│ │ │ ├── trash_bear_bw_square.jpg
│ │ │ ├── trash_square.jpg
│ │ │ └── washer_square.jpg
│ ├── manifest.json
│ ├── stub_config
│ │ ├── bedroom.png
│ │ ├── floorplan.png
│ │ ├── kitchen.png
│ │ └── t-shirt-promo.png
│ ├── sw-legacy.js
│ └── sw-modern.js
├── script
│ ├── build_demo
│ ├── develop_demo
│ └── size_stats
└── src
│ ├── configs
│ ├── arsaboo
│ │ ├── entities.ts
│ │ ├── index.ts
│ │ ├── lovelace.ts
│ │ └── theme.ts
│ ├── demo-configs.ts
│ ├── jimpower
│ │ ├── entities.ts
│ │ ├── index.ts
│ │ ├── lovelace.ts
│ │ └── theme.ts
│ ├── kernehed
│ │ ├── entities.ts
│ │ ├── index.ts
│ │ ├── lovelace.ts
│ │ └── theme.ts
│ ├── sections
│ │ ├── entities.ts
│ │ ├── index.ts
│ │ └── lovelace.ts
│ ├── teachingbirds
│ │ ├── entities.ts
│ │ ├── index.ts
│ │ ├── lovelace.ts
│ │ └── theme.ts
│ └── types.ts
│ ├── custom-cards
│ ├── card-modder.js
│ ├── card-tools.js
│ ├── cast-demo-row.ts
│ └── ha-demo-card.ts
│ ├── entrypoint.ts
│ ├── ha-demo.ts
│ ├── html
│ ├── _social_meta.html.template
│ └── index.html.template
│ ├── stubs
│ ├── area_registry.ts
│ ├── auth.ts
│ ├── config.ts
│ ├── config_entries.ts
│ ├── device_registry.ts
│ ├── energy.ts
│ ├── entities.ts
│ ├── entity_registry.ts
│ ├── events.ts
│ ├── floor_registry.ts
│ ├── frontend.ts
│ ├── hassio_supervisor.ts
│ ├── history.ts
│ ├── icons.ts
│ ├── label_registry.ts
│ ├── lovelace.ts
│ ├── media_player.ts
│ ├── persistent_notification.ts
│ ├── recorder.ts
│ ├── sensor.ts
│ ├── system_log.ts
│ ├── tags.ts
│ ├── template.ts
│ ├── todo.ts
│ └── translations.ts
│ └── util
│ └── is_frontpage.ts
├── docs
└── screenshot.png
├── eslint.config.mjs
├── gallery
├── eslint.config.mjs
├── public
│ ├── api
│ │ ├── hassio
│ │ │ └── addons
│ │ │ │ └── core_zwave_js
│ │ │ │ └── icon
│ │ └── media_player_proxy
│ │ │ ├── media_player.bedroom
│ │ │ ├── media_player.living_room
│ │ │ └── media_player.walkman
│ └── images
│ │ ├── album_cover.jpg
│ │ ├── album_cover_2.jpg
│ │ ├── bed.png
│ │ ├── brand
│ │ ├── README.md
│ │ ├── logo-exclusion-zone.png
│ │ ├── logo-layout-variants.png
│ │ └── logo.png
│ │ ├── clearspace.png
│ │ ├── divider.png
│ │ ├── floorplan.png
│ │ ├── frenck.jpg
│ │ ├── kitchen.png
│ │ ├── light_bulb_off.png
│ │ ├── light_bulb_on.png
│ │ ├── living_room.png
│ │ ├── logo-variants.png
│ │ ├── logo-with-text.png
│ │ ├── netflix.jpg
│ │ ├── office.jpg
│ │ ├── paulus.jpg
│ │ ├── select_box
│ │ ├── card.svg
│ │ └── text_only.svg
│ │ └── sunflowers.jpg
├── script
│ ├── build_gallery
│ └── develop_gallery
├── sidebar.js
└── src
│ ├── components
│ ├── demo-black-white-row.ts
│ ├── demo-card.ts
│ ├── demo-cards.ts
│ ├── demo-more-info.ts
│ ├── demo-more-infos.ts
│ └── page-description.ts
│ ├── data
│ ├── date-options.ts
│ ├── demo_states.js
│ ├── hass.js
│ ├── media_players.ts
│ ├── plants.ts
│ ├── text.ts
│ └── traces
│ │ ├── basic_trace.ts
│ │ ├── mock-demo-trace.ts
│ │ ├── motion-light-trace.ts
│ │ └── types.ts
│ ├── entrypoint.js
│ ├── ha-demo-options.ts
│ ├── ha-gallery.ts
│ ├── html
│ └── index.html.template
│ └── pages
│ ├── Text
│ └── remove-delete-add-create.markdown
│ ├── automation
│ ├── describe-action.markdown
│ ├── describe-action.ts
│ ├── describe-condition.markdown
│ ├── describe-condition.ts
│ ├── describe-trigger.markdown
│ ├── describe-trigger.ts
│ ├── editor-action.markdown
│ ├── editor-action.ts
│ ├── editor-condition.markdown
│ ├── editor-condition.ts
│ ├── editor-trigger.markdown
│ ├── editor-trigger.ts
│ ├── trace-timeline.markdown
│ ├── trace-timeline.ts
│ ├── trace.markdown
│ └── trace.ts
│ ├── brand
│ ├── logo.markdown
│ └── our-story.markdown
│ ├── components
│ ├── ha-alert.markdown
│ ├── ha-alert.ts
│ ├── ha-badge.markdown
│ ├── ha-badge.ts
│ ├── ha-bar.markdown
│ ├── ha-bar.ts
│ ├── ha-chips.markdown
│ ├── ha-chips.ts
│ ├── ha-control-button.markdown
│ ├── ha-control-button.ts
│ ├── ha-control-circular-slider.markdown
│ ├── ha-control-circular-slider.ts
│ ├── ha-control-number-buttons.markdown
│ ├── ha-control-number-buttons.ts
│ ├── ha-control-select-menu.markdown
│ ├── ha-control-select-menu.ts
│ ├── ha-control-select.markdown
│ ├── ha-control-select.ts
│ ├── ha-control-slider.markdown
│ ├── ha-control-slider.ts
│ ├── ha-control-switch.markdown
│ ├── ha-control-switch.ts
│ ├── ha-dialogs.markdown
│ ├── ha-expansion-panel.markdown
│ ├── ha-expansion-panel.ts
│ ├── ha-faded.markdown
│ ├── ha-faded.ts
│ ├── ha-form.markdown
│ ├── ha-form.ts
│ ├── ha-gauge.markdown
│ ├── ha-gauge.ts
│ ├── ha-hs-color-picker.markdown
│ ├── ha-hs-color-picker.ts
│ ├── ha-label-badge.markdown
│ ├── ha-label-badge.ts
│ ├── ha-select-box.markdown
│ ├── ha-select-box.ts
│ ├── ha-selector.markdown
│ ├── ha-selector.ts
│ ├── ha-spinner.markdown
│ ├── ha-spinner.ts
│ ├── ha-switch.markdown
│ ├── ha-switch.ts
│ ├── ha-tip.markdown
│ ├── ha-tip.ts
│ ├── ha-tooltip.markdown
│ └── ha-tooltip.ts
│ ├── concepts
│ └── home.markdown
│ ├── date-time
│ ├── date-time-numeric.markdown
│ ├── date-time-numeric.ts
│ ├── date-time-seconds.markdown
│ ├── date-time-seconds.ts
│ ├── date-time-short-year.markdown
│ ├── date-time-short-year.ts
│ ├── date-time-short.markdown
│ ├── date-time-short.ts
│ ├── date-time.markdown
│ ├── date-time.ts
│ ├── date.markdown
│ ├── date.ts
│ ├── time-seconds.markdown
│ ├── time-seconds.ts
│ ├── time-weekday.markdown
│ ├── time-weekday.ts
│ ├── time.markdown
│ └── time.ts
│ ├── design.home-assistant.io
│ └── editing.markdown
│ ├── lovelace
│ ├── alarm-panel-card.markdown
│ ├── alarm-panel-card.ts
│ ├── area-card.markdown
│ ├── area-card.ts
│ ├── conditional-card.markdown
│ ├── conditional-card.ts
│ ├── entities-card.markdown
│ ├── entities-card.ts
│ ├── entity-button-card.markdown
│ ├── entity-button-card.ts
│ ├── entity-filter-card.markdown
│ ├── entity-filter-card.ts
│ ├── gauge-card.markdown
│ ├── gauge-card.ts
│ ├── glance-card.markdown
│ ├── glance-card.ts
│ ├── grid-and-stack-card.markdown
│ ├── grid-and-stack-card.ts
│ ├── iframe-card.markdown
│ ├── iframe-card.ts
│ ├── introduction.markdown
│ ├── light-card.markdown
│ ├── light-card.ts
│ ├── map-card.markdown
│ ├── map-card.ts
│ ├── markdown-card.markdown
│ ├── markdown-card.ts
│ ├── media-control-card.markdown
│ ├── media-control-card.ts
│ ├── media-player-row.markdown
│ ├── media-player-row.ts
│ ├── picture-card.markdown
│ ├── picture-card.ts
│ ├── picture-elements-card.markdown
│ ├── picture-elements-card.ts
│ ├── picture-entity-card.markdown
│ ├── picture-entity-card.ts
│ ├── picture-glance-card.markdown
│ ├── picture-glance-card.ts
│ ├── plant-card.markdown
│ ├── plant-card.ts
│ ├── thermostat-card.markdown
│ ├── thermostat-card.ts
│ ├── tile-card.markdown
│ ├── tile-card.ts
│ ├── todo-list-card.markdown
│ └── todo-list-card.ts
│ ├── misc
│ ├── entity-state.markdown
│ ├── entity-state.ts
│ ├── ha-markdown.markdown
│ ├── ha-markdown.ts
│ ├── integration-card.markdown
│ ├── integration-card.ts
│ ├── util-long-press.markdown
│ └── util-long-press.ts
│ ├── more-info
│ ├── climate.markdown
│ ├── climate.ts
│ ├── cover.markdown
│ ├── cover.ts
│ ├── humidifier.markdown
│ ├── humidifier.ts
│ ├── input-number.markdown
│ ├── input-number.ts
│ ├── input-text.markdown
│ ├── input-text.ts
│ ├── light.markdown
│ ├── light.ts
│ ├── lock.markdown
│ ├── lock.ts
│ ├── media-player.markdown
│ ├── media-player.ts
│ ├── number.markdown
│ ├── number.ts
│ ├── scene.markdown
│ ├── scene.ts
│ ├── timer.markdown
│ ├── timer.ts
│ ├── update.markdown
│ ├── update.ts
│ ├── vacuum.markdown
│ ├── vacuum.ts
│ ├── water-heater.markdown
│ └── water-heater.ts
│ └── user-test
│ ├── configuration-menu.markdown
│ └── user-types.markdown
├── gulpfile.js
├── hassio
├── config.cjs
├── script
│ ├── build_hassio
│ └── develop
└── src
│ ├── addon-store
│ ├── hassio-addon-repository.ts
│ └── hassio-addon-store.ts
│ ├── addon-view
│ ├── config
│ │ ├── hassio-addon-audio.ts
│ │ ├── hassio-addon-config-tab.ts
│ │ ├── hassio-addon-config.ts
│ │ └── hassio-addon-network.ts
│ ├── documentation
│ │ └── hassio-addon-documentation-tab.ts
│ ├── hassio-addon-dashboard.ts
│ ├── hassio-addon-router.ts
│ ├── info
│ │ ├── hassio-addon-info-tab.ts
│ │ ├── hassio-addon-info.ts
│ │ └── hassio-addon-system-managed.ts
│ └── log
│ │ └── hassio-addon-log-tab.ts
│ ├── backups
│ └── hassio-backups.ts
│ ├── components
│ ├── hassio-card-content.ts
│ ├── hassio-filter-addons.ts
│ ├── hassio-upload-backup.ts
│ ├── supervisor-backup-content.ts
│ ├── supervisor-formfield-label.ts
│ └── supervisor-metric.ts
│ ├── dashboard
│ ├── hassio-addons.ts
│ ├── hassio-dashboard.ts
│ └── hassio-update.ts
│ ├── dialogs
│ ├── backup
│ │ ├── dialog-hassio-backup-location.ts
│ │ ├── dialog-hassio-backup-upload.ts
│ │ ├── dialog-hassio-backup.ts
│ │ ├── dialog-hassio-create-backup.ts
│ │ ├── show-dialog-backup-upload.ts
│ │ ├── show-dialog-hassio-backu-location.ts
│ │ ├── show-dialog-hassio-backup.ts
│ │ └── show-dialog-hassio-create-backup.ts
│ ├── datadisk
│ │ ├── dialog-hassio-datadisk.ts
│ │ └── show-dialog-hassio-datadisk.ts
│ ├── hardware
│ │ ├── dialog-hassio-hardware.ts
│ │ └── show-dialog-hassio-hardware.ts
│ ├── markdown
│ │ ├── dialog-hassio-markdown.ts
│ │ └── show-dialog-hassio-markdown.ts
│ ├── network
│ │ ├── dialog-hassio-network.ts
│ │ └── show-dialog-network.ts
│ ├── registries
│ │ ├── dialog-hassio-registries.ts
│ │ └── show-dialog-registries.ts
│ ├── repositories
│ │ ├── dialog-hassio-repositories.ts
│ │ └── show-dialog-repositories.ts
│ ├── suggestAddonRestart.ts
│ └── system-managed
│ │ ├── dialog-system-managed.ts
│ │ └── show-dialog-system-managed.ts
│ ├── entrypoint.js.template
│ ├── entrypoint.ts
│ ├── hassio-main.ts
│ ├── hassio-my-redirect.ts
│ ├── hassio-panel-router.ts
│ ├── hassio-panel.ts
│ ├── hassio-router.ts
│ ├── hassio-tabs.ts
│ ├── ingress-view
│ └── hassio-ingress-view.ts
│ ├── resources
│ └── hassio-style.ts
│ ├── supervisor-base-element.ts
│ ├── system
│ ├── hassio-core-info.ts
│ ├── hassio-host-info.ts
│ ├── hassio-supervisor-info.ts
│ ├── hassio-supervisor-log.ts
│ └── hassio-system.ts
│ ├── update-available
│ ├── update-available-card.ts
│ └── update-available-dashboard.ts
│ └── util
│ └── addon.ts
├── landing-page
├── README.md
├── eslintrc.config.mjs
├── public
│ └── static
│ │ ├── icons
│ │ ├── favicon-192x192.png
│ │ ├── favicon.ico
│ │ ├── logo_ohf.svg
│ │ └── ohf.svg
│ │ └── images
│ │ ├── appstore.svg
│ │ ├── logo_discord.png
│ │ ├── logo_mastodon.svg
│ │ ├── logo_x.svg
│ │ ├── playstore.svg
│ │ ├── qr-appstore.svg
│ │ └── qr-playstore.svg
├── script
│ ├── build_landing_page
│ └── develop
└── src
│ ├── components
│ ├── landing-page-logs.ts
│ └── landing-page-network.ts
│ ├── data
│ ├── observer.ts
│ └── supervisor.ts
│ ├── entrypoint.js
│ ├── ha-landing-page.ts
│ ├── html
│ └── index.html.template
│ └── landing-page-base-element.ts
├── lint-staged.config.js
├── netlify.toml
├── package.json
├── prettier.config.js
├── public
├── __init__.py
├── py.typed
├── robots.txt
└── static
│ ├── icons
│ ├── browserconfig.xml
│ ├── favicon-1024x1024.png
│ ├── favicon-16x16.png
│ ├── favicon-192x192.png
│ ├── favicon-32x32.png
│ ├── favicon-384x384.png
│ ├── favicon-512x512.png
│ ├── favicon-apple-180x180.png
│ ├── favicon.ico
│ ├── logo_ohf.svg
│ ├── mask-icon.svg
│ ├── maskable_icon-128x128.png
│ ├── maskable_icon-192x192.png
│ ├── maskable_icon-384x384.png
│ ├── maskable_icon-48x48.png
│ ├── maskable_icon-512x512.png
│ ├── maskable_icon-72x72.png
│ ├── maskable_icon-96x96.png
│ ├── ohf.svg
│ ├── tile-win-150x150.png
│ ├── tile-win-310x150.png
│ ├── tile-win-310x310.png
│ └── tile-win-70x70.png
│ └── images
│ ├── appstore.svg
│ ├── card_media_player_bg.png
│ ├── color_wheel.png
│ ├── config_ecobee_thermostat.png
│ ├── config_fitbit_app.png
│ ├── config_flows
│ └── config_homematicip_cloud.png
│ ├── config_freebox.png
│ ├── config_icloud.png
│ ├── config_insteon.png
│ ├── config_philips_hue.jpg
│ ├── config_webos.png
│ ├── config_wink.png
│ ├── darksky
│ ├── weather-cloudy.svg
│ ├── weather-fog.svg
│ ├── weather-hail.svg
│ ├── weather-night.svg
│ ├── weather-partlycloudy.svg
│ ├── weather-pouring.svg
│ ├── weather-rainy.svg
│ ├── weather-snowy.svg
│ ├── weather-sunny.svg
│ └── weather-windy.svg
│ ├── form
│ ├── markdown_card.svg
│ ├── markdown_card_dark.svg
│ ├── markdown_text_only.svg
│ ├── markdown_text_only_dark.svg
│ ├── tile_content_layout_horizontal.svg
│ ├── tile_content_layout_horizontal_dark.svg
│ ├── tile_content_layout_vertical.svg
│ ├── tile_content_layout_vertical_dark.svg
│ ├── tile_features_position_bottom.svg
│ ├── tile_features_position_bottom_dark.svg
│ ├── tile_features_position_inline.svg
│ ├── tile_features_position_inline_dark.svg
│ ├── view_header_badges_position_bottom.svg
│ ├── view_header_badges_position_bottom_dark.svg
│ ├── view_header_badges_position_top.svg
│ ├── view_header_badges_position_top_dark.svg
│ ├── view_header_badges_wrap_scroll.svg
│ ├── view_header_badges_wrap_scroll_dark.svg
│ ├── view_header_badges_wrap_wrap.svg
│ ├── view_header_badges_wrap_wrap_dark.svg
│ ├── view_header_layout_center.svg
│ ├── view_header_layout_center_dark.svg
│ ├── view_header_layout_responsive.svg
│ ├── view_header_layout_responsive_dark.svg
│ ├── view_header_layout_start.svg
│ └── view_header_layout_start_dark.svg
│ ├── image-broken.svg
│ ├── logo_apple_home.png
│ ├── logo_automatic.png
│ ├── logo_axis.png
│ ├── logo_deconz.jpeg
│ ├── logo_discord.png
│ ├── logo_google_home.png
│ ├── logo_mastodon.svg
│ ├── logo_nabu_casa.png
│ ├── logo_nabu_casa_dark.png
│ ├── logo_philips_hue.png
│ ├── logo_plex_mediaserver.png
│ ├── logo_x.svg
│ ├── notification-badge.png
│ ├── ohf-badge.svg
│ ├── playstore.svg
│ ├── qr-appstore.svg
│ ├── qr-playstore.svg
│ ├── screenshots
│ └── screenshot-1.png
│ ├── smart-tv.png
│ ├── voice-assistant
│ ├── area.png
│ ├── change-wake-word.png
│ ├── error.png
│ ├── great-job.png
│ ├── heart.png
│ ├── hi.png
│ ├── ok-nabu.png
│ ├── sleep.png
│ └── update.png
│ └── z-wave-add-node
│ ├── long-range.svg
│ ├── long-range_dark.svg
│ ├── mesh.svg
│ └── mesh_dark.svg
├── pyproject.toml
├── renovate.json
├── rspack.config.cjs
├── script
├── bootstrap
├── build_frontend
├── core
├── develop
├── develop_and_serve
├── release
├── serve-config.json
├── setup
├── setup_translations
├── size_stats
├── translations_download
├── translations_upload_base
└── version_bump.js
├── src
├── auth
│ ├── ha-auth-flow.ts
│ ├── ha-auth-form-string.ts
│ ├── ha-auth-form.ts
│ ├── ha-auth-textfield.ts
│ ├── ha-authorize.ts
│ └── ha-pick-auth-provider.ts
├── cast
│ ├── cast_framework.ts
│ ├── cast_manager.ts
│ ├── const.ts
│ ├── dev_const.ts
│ ├── receiver_messages.ts
│ ├── sender_messages.ts
│ └── types.ts
├── common
│ ├── array
│ │ ├── combinations.ts
│ │ ├── ensure-array.ts
│ │ └── literal-includes.ts
│ ├── auth
│ │ └── token_storage.ts
│ ├── color
│ │ ├── colors.ts
│ │ ├── compute-color.ts
│ │ ├── convert-color.ts
│ │ ├── convert-light-color.ts
│ │ ├── hex.ts
│ │ ├── lab.ts
│ │ └── rgb.ts
│ ├── config
│ │ ├── can_show_page.ts
│ │ ├── components_with_service.ts
│ │ ├── is_component_loaded.ts
│ │ ├── is_pwa.ts
│ │ ├── is_service_loaded.ts
│ │ ├── location_name.ts
│ │ └── version.ts
│ ├── const.ts
│ ├── controllers
│ │ └── drag-scroll-controller.ts
│ ├── datetime
│ │ ├── absolute_time.ts
│ │ ├── calc_date.ts
│ │ ├── calc_date_range.ts
│ │ ├── check_valid_date.ts
│ │ ├── create_duration_data.ts
│ │ ├── duration_to_seconds.ts
│ │ ├── first_weekday.ts
│ │ ├── format_date.ts
│ │ ├── format_date_time.ts
│ │ ├── format_duration.ts
│ │ ├── format_time.ts
│ │ ├── localize_date.ts
│ │ ├── milliseconds_to_duration.ts
│ │ ├── relative_time.ts
│ │ ├── resolve-time-zone.ts
│ │ ├── seconds_to_duration.ts
│ │ └── use_am_pm.ts
│ ├── decorators
│ │ ├── restore-scroll.ts
│ │ ├── storage.ts
│ │ └── transform.ts
│ ├── dom
│ │ ├── ancestors-with-property.ts
│ │ ├── apply_themes_on_element.ts
│ │ ├── can-override-input.ts
│ │ ├── deep-active-element.ts
│ │ ├── dynamic-element-directive.ts
│ │ ├── fire_event.ts
│ │ ├── get_main_window.ts
│ │ ├── is-navigation-click.ts
│ │ ├── load_resource.ts
│ │ ├── media_query.ts
│ │ ├── prevent_default.ts
│ │ ├── scroll-to-target.ts
│ │ ├── setup-leaflet-map.ts
│ │ ├── stop_propagation.ts
│ │ └── toggle_attribute.ts
│ ├── empty_image_base64.ts
│ ├── entity
│ │ ├── attribute_class_names.ts
│ │ ├── battery_icon.ts
│ │ ├── can_toggle_domain.ts
│ │ ├── can_toggle_state.ts
│ │ ├── color
│ │ │ └── battery_color.ts
│ │ ├── compute_area_name.ts
│ │ ├── compute_attribute_display.ts
│ │ ├── compute_device_name.ts
│ │ ├── compute_domain.ts
│ │ ├── compute_entity_name.ts
│ │ ├── compute_floor_name.ts
│ │ ├── compute_object_id.ts
│ │ ├── compute_state_display.ts
│ │ ├── compute_state_domain.ts
│ │ ├── compute_state_name.ts
│ │ ├── context
│ │ │ ├── get_area_context.ts
│ │ │ ├── get_device_context.ts
│ │ │ └── get_entity_context.ts
│ │ ├── cover_icon.ts
│ │ ├── delete_entity.ts
│ │ ├── device_tracker_icon.ts
│ │ ├── entity_domain_filter.ts
│ │ ├── entity_filter.ts
│ │ ├── extract_views.ts
│ │ ├── feature_class_names.ts
│ │ ├── get_group_entities.ts
│ │ ├── get_states.ts
│ │ ├── get_view_entities.ts
│ │ ├── has_location.ts
│ │ ├── split_by_groups.ts
│ │ ├── state_active.ts
│ │ ├── state_card_type.ts
│ │ ├── state_color.ts
│ │ ├── state_icon.ts
│ │ ├── states_sort_by_name.ts
│ │ ├── strip_prefix_from_entity_name.ts
│ │ ├── supports-feature.ts
│ │ ├── update_icon.ts
│ │ ├── valid_entity_id.ts
│ │ └── valid_service_id.ts
│ ├── feature-detect
│ │ └── support-web-components.ts
│ ├── image
│ │ └── extract_color.ts
│ ├── integrations
│ │ └── protocolIntegrationPicked.ts
│ ├── language
│ │ └── format_language.ts
│ ├── location
│ │ └── add_distance_to_coord.ts
│ ├── map
│ │ └── decorated_marker.ts
│ ├── mwc
│ │ └── handle-request-selected-event.ts
│ ├── navigate.ts
│ ├── number
│ │ ├── clamp.ts
│ │ ├── format_number.ts
│ │ └── round.ts
│ ├── string
│ │ ├── capitalize-first-letter.ts
│ │ ├── compare.ts
│ │ ├── escape_regexp.ts
│ │ ├── filter
│ │ │ ├── char-code.ts
│ │ │ ├── filter.ts
│ │ │ └── sequence-matching.ts
│ │ ├── format-list.ts
│ │ ├── get_duplicates.ts
│ │ ├── has-template.ts
│ │ ├── is_date.ts
│ │ ├── is_ip_address.ts
│ │ ├── is_timestamp.ts
│ │ ├── slugify.ts
│ │ ├── starts-with.ts
│ │ ├── strip-diacritics.ts
│ │ └── title-case.ts
│ ├── structs
│ │ ├── handle-errors.ts
│ │ ├── is-custom-type.ts
│ │ └── is-icon.ts
│ ├── style
│ │ ├── derived-css-vars.ts
│ │ └── icon_color_css.ts
│ ├── translations
│ │ ├── auto_case_noun.ts
│ │ ├── blank_before_percent.ts
│ │ ├── blank_before_unit.ts
│ │ ├── day_names.ts
│ │ ├── entity-state.ts
│ │ ├── localize.ts
│ │ ├── markdown_support.ts
│ │ └── month_names.ts
│ ├── url
│ │ ├── construct-url.ts
│ │ └── search-params.ts
│ └── util
│ │ ├── compute_rtl.ts
│ │ ├── copy-clipboard.ts
│ │ ├── debounce.ts
│ │ ├── deep-equal.ts
│ │ ├── group-by.ts
│ │ ├── parse-aspect-ratio.ts
│ │ ├── patch.ts
│ │ ├── promise-all-settled-results.ts
│ │ ├── promise-timeout.ts
│ │ ├── render-status.ts
│ │ ├── select-unit.ts
│ │ ├── shallow-equal.ts
│ │ ├── subscribe-one.ts
│ │ ├── subscribe-polling.ts
│ │ ├── throttle.ts
│ │ ├── time-cache-entity-promise-func.ts
│ │ ├── time-cache-function-promise.ts
│ │ ├── uid.ts
│ │ └── wait.ts
├── components
│ ├── buttons
│ │ ├── ha-call-service-button.ts
│ │ └── ha-progress-button.ts
│ ├── chart
│ │ ├── axis-label.ts
│ │ ├── down-sample.ts
│ │ ├── ha-chart-base.ts
│ │ ├── ha-network-graph.ts
│ │ ├── ha-sankey-chart.ts
│ │ ├── state-history-chart-line.ts
│ │ ├── state-history-chart-timeline.ts
│ │ ├── state-history-charts.ts
│ │ ├── statistics-chart.ts
│ │ └── timeline-color.ts
│ ├── chips
│ │ ├── ha-assist-chip.ts
│ │ ├── ha-chip-set.ts
│ │ ├── ha-filter-chip.ts
│ │ └── ha-input-chip.ts
│ ├── data-table
│ │ ├── dialog-data-table-settings.ts
│ │ ├── ha-data-table-icon.ts
│ │ ├── ha-data-table-labels.ts
│ │ ├── ha-data-table.ts
│ │ ├── show-dialog-data-table-settings.ts
│ │ ├── sort-filter-worker.ts
│ │ └── sort-filter.ts
│ ├── date-range-picker.ts
│ ├── device
│ │ ├── ha-device-action-picker.ts
│ │ ├── ha-device-automation-picker.ts
│ │ ├── ha-device-condition-picker.ts
│ │ ├── ha-device-picker.ts
│ │ ├── ha-device-trigger-picker.ts
│ │ └── ha-devices-picker.ts
│ ├── entity
│ │ ├── ha-battery-icon.ts
│ │ ├── ha-entities-picker.ts
│ │ ├── ha-entity-attribute-picker.ts
│ │ ├── ha-entity-picker.ts
│ │ ├── ha-entity-state-content-picker.ts
│ │ ├── ha-entity-state-picker.ts
│ │ ├── ha-entity-toggle.ts
│ │ ├── ha-state-label-badge.ts
│ │ ├── ha-statistic-picker.ts
│ │ ├── ha-statistics-picker.ts
│ │ ├── state-badge.ts
│ │ └── state-info.ts
│ ├── ha-absolute-time.ts
│ ├── ha-addon-picker.ts
│ ├── ha-alert.ts
│ ├── ha-aliases-editor.ts
│ ├── ha-analytics.ts
│ ├── ha-ansi-to-html.ts
│ ├── ha-area-floor-picker.ts
│ ├── ha-area-picker.ts
│ ├── ha-areas-display-editor.ts
│ ├── ha-areas-picker.ts
│ ├── ha-assist-chat.ts
│ ├── ha-assist-pipeline-picker.ts
│ ├── ha-attribute-icon.ts
│ ├── ha-attribute-value.ts
│ ├── ha-attributes.ts
│ ├── ha-badge.ts
│ ├── ha-bar.ts
│ ├── ha-base-time-input.ts
│ ├── ha-big-number.ts
│ ├── ha-blueprint-picker.ts
│ ├── ha-button-menu.ts
│ ├── ha-button-toggle-group.ts
│ ├── ha-button.ts
│ ├── ha-camera-stream.ts
│ ├── ha-card.ts
│ ├── ha-check-list-item.ts
│ ├── ha-checkbox.ts
│ ├── ha-climate-state.ts
│ ├── ha-code-editor.ts
│ ├── ha-color-picker.ts
│ ├── ha-combo-box-item.ts
│ ├── ha-combo-box-textfield.ts
│ ├── ha-combo-box.ts
│ ├── ha-config-entry-picker.ts
│ ├── ha-control-button-group.ts
│ ├── ha-control-button.ts
│ ├── ha-control-circular-slider.ts
│ ├── ha-control-number-buttons.ts
│ ├── ha-control-select-menu.ts
│ ├── ha-control-select.ts
│ ├── ha-control-slider.ts
│ ├── ha-control-switch.ts
│ ├── ha-conversation-agent-picker.ts
│ ├── ha-copy-textfield.ts
│ ├── ha-country-picker.ts
│ ├── ha-cover-controls.ts
│ ├── ha-cover-tilt-controls.ts
│ ├── ha-currency-picker.ts
│ ├── ha-date-input.ts
│ ├── ha-date-range-picker.ts
│ ├── ha-dialog-date-picker.ts
│ ├── ha-dialog-header.ts
│ ├── ha-dialog.ts
│ ├── ha-divider.ts
│ ├── ha-domain-icon.ts
│ ├── ha-drawer.ts
│ ├── ha-duration-input.ts
│ ├── ha-entities-display-editor.ts
│ ├── ha-expansion-panel.ts
│ ├── ha-fab.ts
│ ├── ha-fade-in.ts
│ ├── ha-faded.ts
│ ├── ha-file-upload.ts
│ ├── ha-filter-blueprints.ts
│ ├── ha-filter-categories.ts
│ ├── ha-filter-devices.ts
│ ├── ha-filter-domains.ts
│ ├── ha-filter-entities.ts
│ ├── ha-filter-floor-areas.ts
│ ├── ha-filter-integrations.ts
│ ├── ha-filter-labels.ts
│ ├── ha-filter-states.ts
│ ├── ha-floor-icon.ts
│ ├── ha-floor-picker.ts
│ ├── ha-floors-picker.ts
│ ├── ha-form
│ │ ├── compute-initial-ha-form-data.ts
│ │ ├── ha-form-boolean.ts
│ │ ├── ha-form-constant.ts
│ │ ├── ha-form-expandable.ts
│ │ ├── ha-form-float.ts
│ │ ├── ha-form-grid.ts
│ │ ├── ha-form-integer.ts
│ │ ├── ha-form-multi_select.ts
│ │ ├── ha-form-optional_actions.ts
│ │ ├── ha-form-positive_time_period_dict.ts
│ │ ├── ha-form-select.ts
│ │ ├── ha-form-string.ts
│ │ ├── ha-form.ts
│ │ └── types.ts
│ ├── ha-formfield.ts
│ ├── ha-gauge.ts
│ ├── ha-generic-picker.ts
│ ├── ha-grid-size-picker.ts
│ ├── ha-header-bar.ts
│ ├── ha-heading-badge.ts
│ ├── ha-help-tooltip.ts
│ ├── ha-hls-player.ts
│ ├── ha-hs-color-picker.ts
│ ├── ha-humidifier-state.ts
│ ├── ha-icon-button-arrow-next.ts
│ ├── ha-icon-button-arrow-prev.ts
│ ├── ha-icon-button-group.ts
│ ├── ha-icon-button-next.ts
│ ├── ha-icon-button-prev.ts
│ ├── ha-icon-button-toggle.ts
│ ├── ha-icon-button.ts
│ ├── ha-icon-next.ts
│ ├── ha-icon-overflow-menu.ts
│ ├── ha-icon-picker.ts
│ ├── ha-icon-prev.ts
│ ├── ha-icon.ts
│ ├── ha-input-helper-text.ts
│ ├── ha-items-display-editor.ts
│ ├── ha-label-badge.ts
│ ├── ha-label-picker.ts
│ ├── ha-label.ts
│ ├── ha-labeled-slider.ts
│ ├── ha-labels-picker.ts
│ ├── ha-language-picker.ts
│ ├── ha-lawn_mower-action-button.ts
│ ├── ha-list-item.ts
│ ├── ha-list.ts
│ ├── ha-logo-svg.ts
│ ├── ha-markdown-element.ts
│ ├── ha-markdown.ts
│ ├── ha-md-button-menu.ts
│ ├── ha-md-dialog.ts
│ ├── ha-md-divider.ts
│ ├── ha-md-list-item.ts
│ ├── ha-md-list.ts
│ ├── ha-md-menu-item.ts
│ ├── ha-md-menu.ts
│ ├── ha-md-select-option.ts
│ ├── ha-md-select.ts
│ ├── ha-md-textfield.ts
│ ├── ha-menu-button.ts
│ ├── ha-menu.ts
│ ├── ha-metric.ts
│ ├── ha-mount-picker.ts
│ ├── ha-multi-textfield.ts
│ ├── ha-navigation-list.ts
│ ├── ha-navigation-picker.ts
│ ├── ha-network.ts
│ ├── ha-outlined-button.ts
│ ├── ha-outlined-field.ts
│ ├── ha-outlined-icon-button.ts
│ ├── ha-outlined-text-field.ts
│ ├── ha-password-field.ts
│ ├── ha-picker-combo-box.ts
│ ├── ha-picker-field.ts
│ ├── ha-picture-upload.ts
│ ├── ha-progress-ring.ts
│ ├── ha-push-notifications-toggle.ts
│ ├── ha-qr-code.ts
│ ├── ha-qr-scanner.ts
│ ├── ha-radio-list-item.ts
│ ├── ha-radio.ts
│ ├── ha-related-items.ts
│ ├── ha-relative-time.ts
│ ├── ha-ripple.ts
│ ├── ha-select-box.ts
│ ├── ha-select.ts
│ ├── ha-selector
│ │ ├── ha-selector-action.ts
│ │ ├── ha-selector-addon.ts
│ │ ├── ha-selector-area.ts
│ │ ├── ha-selector-areas-display.ts
│ │ ├── ha-selector-assist-pipeline.ts
│ │ ├── ha-selector-attribute.ts
│ │ ├── ha-selector-background.ts
│ │ ├── ha-selector-backup-location.ts
│ │ ├── ha-selector-boolean.ts
│ │ ├── ha-selector-button-toggle.ts
│ │ ├── ha-selector-color-rgb.ts
│ │ ├── ha-selector-color-temp.ts
│ │ ├── ha-selector-condition.ts
│ │ ├── ha-selector-config-entry.ts
│ │ ├── ha-selector-constant.ts
│ │ ├── ha-selector-conversation-agent.ts
│ │ ├── ha-selector-country.ts
│ │ ├── ha-selector-date.ts
│ │ ├── ha-selector-datetime.ts
│ │ ├── ha-selector-device.ts
│ │ ├── ha-selector-duration.ts
│ │ ├── ha-selector-entity.ts
│ │ ├── ha-selector-file.ts
│ │ ├── ha-selector-floor.ts
│ │ ├── ha-selector-icon.ts
│ │ ├── ha-selector-image.ts
│ │ ├── ha-selector-label.ts
│ │ ├── ha-selector-language.ts
│ │ ├── ha-selector-location.ts
│ │ ├── ha-selector-media.ts
│ │ ├── ha-selector-navigation.ts
│ │ ├── ha-selector-number.ts
│ │ ├── ha-selector-object.ts
│ │ ├── ha-selector-qr-code.ts
│ │ ├── ha-selector-select.ts
│ │ ├── ha-selector-selector.ts
│ │ ├── ha-selector-state.ts
│ │ ├── ha-selector-statistic.ts
│ │ ├── ha-selector-stt.ts
│ │ ├── ha-selector-target.ts
│ │ ├── ha-selector-template.ts
│ │ ├── ha-selector-text.ts
│ │ ├── ha-selector-theme.ts
│ │ ├── ha-selector-time.ts
│ │ ├── ha-selector-trigger.ts
│ │ ├── ha-selector-tts-voice.ts
│ │ ├── ha-selector-tts.ts
│ │ ├── ha-selector-ui-action.ts
│ │ ├── ha-selector-ui-color.ts
│ │ ├── ha-selector-ui-state-content.ts
│ │ └── ha-selector.ts
│ ├── ha-service-control.ts
│ ├── ha-service-icon.ts
│ ├── ha-service-picker.ts
│ ├── ha-service-section-icon.ts
│ ├── ha-settings-row.ts
│ ├── ha-sidebar.ts
│ ├── ha-slider.ts
│ ├── ha-sortable.ts
│ ├── ha-spinner.ts
│ ├── ha-state-icon.ts
│ ├── ha-stt-picker.ts
│ ├── ha-sub-menu.ts
│ ├── ha-svg-icon.ts
│ ├── ha-switch.ts
│ ├── ha-tab.ts
│ ├── ha-target-picker.ts
│ ├── ha-textarea.ts
│ ├── ha-textfield.ts
│ ├── ha-theme-picker.ts
│ ├── ha-time-input.ts
│ ├── ha-timezone-picker.ts
│ ├── ha-tip.ts
│ ├── ha-toast.ts
│ ├── ha-tooltip.ts
│ ├── ha-top-app-bar-fixed.ts
│ ├── ha-top-app-bar.ts
│ ├── ha-tree-indicator.ts
│ ├── ha-tts-picker.ts
│ ├── ha-tts-voice-picker.ts
│ ├── ha-two-pane-top-app-bar-fixed.ts
│ ├── ha-vacuum-state.ts
│ ├── ha-valve-controls.ts
│ ├── ha-water_heater-state.ts
│ ├── ha-web-rtc-player.ts
│ ├── ha-yaml-editor.ts
│ ├── map
│ │ ├── ha-entity-marker.ts
│ │ ├── ha-locations-editor.ts
│ │ └── ha-map.ts
│ ├── media-player
│ │ ├── dialog-join-media-players.ts
│ │ ├── dialog-media-manage.ts
│ │ ├── dialog-media-player-browse.ts
│ │ ├── ha-browse-media-tts.ts
│ │ ├── ha-media-manage-button.ts
│ │ ├── ha-media-player-browse.ts
│ │ ├── ha-media-player-toggle.ts
│ │ ├── ha-media-upload-button.ts
│ │ ├── show-join-media-players-dialog.ts
│ │ ├── show-media-browser-dialog.ts
│ │ └── show-media-manage-dialog.ts
│ ├── search-input-outlined.ts
│ ├── search-input.ts
│ ├── sl-tab-group.ts
│ ├── tile
│ │ ├── ha-tile-badge.ts
│ │ ├── ha-tile-icon.ts
│ │ └── ha-tile-info.ts
│ ├── timezone-datalist.ts
│ ├── trace
│ │ ├── ha-timeline.ts
│ │ ├── ha-trace-blueprint-config.ts
│ │ ├── ha-trace-config.ts
│ │ ├── ha-trace-logbook.ts
│ │ ├── ha-trace-path-details.ts
│ │ ├── ha-trace-timeline.ts
│ │ ├── hat-graph-branch.ts
│ │ ├── hat-graph-const.ts
│ │ ├── hat-graph-node.ts
│ │ ├── hat-graph-spacer.ts
│ │ ├── hat-logbook-note.ts
│ │ ├── hat-script-graph.ts
│ │ ├── hat-trace-timeline.ts
│ │ └── trace-tab-styles.ts
│ └── user
│ │ ├── ha-person-badge.ts
│ │ ├── ha-user-badge.ts
│ │ ├── ha-user-picker.ts
│ │ └── ha-users-picker.ts
├── data
│ ├── action.ts
│ ├── alarm_control_panel.ts
│ ├── alexa.ts
│ ├── analytics.ts
│ ├── application_credential.ts
│ ├── area_registry.ts
│ ├── assist_pipeline.ts
│ ├── assist_satellite.ts
│ ├── auth.ts
│ ├── automation.ts
│ ├── automation_i18n.ts
│ ├── backup.ts
│ ├── backup_manager.ts
│ ├── backup_onboarding.ts
│ ├── blueprint.ts
│ ├── bluetooth.ts
│ ├── bootstrap_integrations.ts
│ ├── calendar.ts
│ ├── camera.ts
│ ├── category_registry.ts
│ ├── climate.ts
│ ├── cloud.ts
│ ├── cloud
│ │ └── tts.ts
│ ├── collection.ts
│ ├── condition.ts
│ ├── config.ts
│ ├── config_entries.ts
│ ├── config_flow.ts
│ ├── connection-status.ts
│ ├── context.ts
│ ├── conversation.ts
│ ├── core.ts
│ ├── counter.ts
│ ├── cover.ts
│ ├── currency.ts
│ ├── custom_icons.ts
│ ├── custom_iconsets.ts
│ ├── data_entry_flow.ts
│ ├── data_table_filters.ts
│ ├── date.ts
│ ├── datetime.ts
│ ├── device_automation.ts
│ ├── device_registry.ts
│ ├── dhcp.ts
│ ├── diagnostics.ts
│ ├── energy.ts
│ ├── entity.ts
│ ├── entity_attributes.ts
│ ├── entity_registry.ts
│ ├── entity_sources.ts
│ ├── error_log.ts
│ ├── expose.ts
│ ├── external.ts
│ ├── fan.ts
│ ├── file_upload.ts
│ ├── floor_registry.ts
│ ├── frontend.ts
│ ├── google_assistant.ts
│ ├── graph.ts
│ ├── group.ts
│ ├── haptics.ts
│ ├── hardware.ts
│ ├── hassio
│ │ ├── addon.ts
│ │ ├── backup.ts
│ │ ├── common.ts
│ │ ├── docker.ts
│ │ ├── hardware.ts
│ │ ├── host.ts
│ │ ├── ingress.ts
│ │ ├── network.ts
│ │ ├── resolution.ts
│ │ └── supervisor.ts
│ ├── helpers_crud.ts
│ ├── history.ts
│ ├── humidifier.ts
│ ├── icons.ts
│ ├── iconsets.ts
│ ├── image.ts
│ ├── image_upload.ts
│ ├── input_boolean.ts
│ ├── input_button.ts
│ ├── input_datetime.ts
│ ├── input_number.ts
│ ├── input_select.ts
│ ├── input_text.ts
│ ├── integration.ts
│ ├── integration_quality_scale.ts
│ ├── integrations.ts
│ ├── label_registry.ts
│ ├── lawn_mower.ts
│ ├── light.ts
│ ├── lock.ts
│ ├── logbook.ts
│ ├── lovelace.ts
│ ├── lovelace
│ │ ├── action_handler.ts
│ │ ├── config
│ │ │ ├── action.ts
│ │ │ ├── badge.ts
│ │ │ ├── card.ts
│ │ │ ├── section.ts
│ │ │ ├── strategy.ts
│ │ │ ├── types.ts
│ │ │ └── view.ts
│ │ ├── dashboard.ts
│ │ └── resource.ts
│ ├── lovelace_custom_cards.ts
│ ├── main_window.ts
│ ├── matter.ts
│ ├── media-player.ts
│ ├── media_source.ts
│ ├── mqtt.ts
│ ├── network.ts
│ ├── notify_html5.ts
│ ├── number.ts
│ ├── onboarding.ts
│ ├── openstreetmap.ts
│ ├── options_flow.ts
│ ├── otbr.ts
│ ├── panel.ts
│ ├── panel_custom.ts
│ ├── persistent_notification.ts
│ ├── person.ts
│ ├── preloads.ts
│ ├── preview.ts
│ ├── recorder.ts
│ ├── refresh_token.ts
│ ├── regenerate_entity_ids.ts
│ ├── registry.ts
│ ├── remote.ts
│ ├── repairs.ts
│ ├── scene.ts
│ ├── schedule.ts
│ ├── script.ts
│ ├── script_i18n.ts
│ ├── search.ts
│ ├── select.ts
│ ├── selector.ts
│ ├── sensor.ts
│ ├── service.ts
│ ├── siren.ts
│ ├── ssdp.ts
│ ├── stt.ts
│ ├── sub_config_flow.ts
│ ├── supervisor
│ │ ├── common.ts
│ │ ├── core.ts
│ │ ├── mounts.ts
│ │ ├── store.ts
│ │ ├── supervisor.ts
│ │ └── update.ts
│ ├── system_health.ts
│ ├── system_log.ts
│ ├── tag.ts
│ ├── text.ts
│ ├── thread.ts
│ ├── time.ts
│ ├── timer.ts
│ ├── todo.ts
│ ├── trace.ts
│ ├── translation.ts
│ ├── trigger.ts
│ ├── tts.ts
│ ├── update.ts
│ ├── usb.ts
│ ├── user.ts
│ ├── vacuum.ts
│ ├── valve.ts
│ ├── wake_word.ts
│ ├── water_heater.ts
│ ├── weather.ts
│ ├── webhook.ts
│ ├── websocket_api.ts
│ ├── ws-area_registry.ts
│ ├── ws-device_registry.ts
│ ├── ws-entity_registry_display.ts
│ ├── ws-floor_registry.ts
│ ├── ws-panels.ts
│ ├── ws-templates.ts
│ ├── ws-themes.ts
│ ├── ws-user.ts
│ ├── wyoming.ts
│ ├── zeroconf.ts
│ ├── zha.ts
│ ├── zone.ts
│ └── zwave_js.ts
├── dialogs
│ ├── config-entry-system-options
│ │ ├── dialog-config-entry-system-options.ts
│ │ └── show-dialog-config-entry-system-options.ts
│ ├── config-flow
│ │ ├── dialog-data-entry-flow.ts
│ │ ├── previews
│ │ │ ├── entity-preview-row.ts
│ │ │ ├── flow-preview-generic.ts
│ │ │ ├── flow-preview-generic_camera.ts
│ │ │ └── flow-preview-template.ts
│ │ ├── show-dialog-config-flow.ts
│ │ ├── show-dialog-data-entry-flow.ts
│ │ ├── show-dialog-options-flow.ts
│ │ ├── show-dialog-sub-config-flow.ts
│ │ ├── step-flow-abort.ts
│ │ ├── step-flow-create-entry.ts
│ │ ├── step-flow-external.ts
│ │ ├── step-flow-form.ts
│ │ ├── step-flow-loading.ts
│ │ ├── step-flow-menu.ts
│ │ ├── step-flow-progress.ts
│ │ └── styles.ts
│ ├── enter-code
│ │ ├── dialog-enter-code.ts
│ │ └── show-enter-code-dialog.ts
│ ├── generic
│ │ ├── dialog-box.ts
│ │ └── show-dialog-box.ts
│ ├── image-cropper-dialog
│ │ ├── image-cropper-dialog.ts
│ │ └── show-image-cropper-dialog.ts
│ ├── make-dialog-manager.ts
│ ├── more-info
│ │ ├── components
│ │ │ ├── ha-more-info-control-select-container.ts
│ │ │ ├── ha-more-info-state-header.ts
│ │ │ ├── lights
│ │ │ │ ├── dialog-light-color-favorite.ts
│ │ │ │ ├── ha-favorite-color-button.ts
│ │ │ │ ├── ha-more-info-light-favorite-colors.ts
│ │ │ │ ├── light-color-rgb-picker.ts
│ │ │ │ ├── light-color-temp-picker.ts
│ │ │ │ └── show-dialog-light-color-favorite.ts
│ │ │ ├── more-info-control-style.ts
│ │ │ ├── siren
│ │ │ │ ├── ha-more-info-siren-advanced-controls.ts
│ │ │ │ └── show-dialog-siren-advanced-controls.ts
│ │ │ └── voice
│ │ │ │ ├── ha-more-info-view-voice-assistants.ts
│ │ │ │ └── show-view-voice-assistants.ts
│ │ ├── const.ts
│ │ ├── controls
│ │ │ ├── more-info-alarm_control_panel.ts
│ │ │ ├── more-info-automation.ts
│ │ │ ├── more-info-camera.ts
│ │ │ ├── more-info-climate.ts
│ │ │ ├── more-info-configurator.ts
│ │ │ ├── more-info-conversation.ts
│ │ │ ├── more-info-counter.ts
│ │ │ ├── more-info-cover.ts
│ │ │ ├── more-info-date.ts
│ │ │ ├── more-info-datetime.ts
│ │ │ ├── more-info-default.ts
│ │ │ ├── more-info-fan.ts
│ │ │ ├── more-info-group.ts
│ │ │ ├── more-info-humidifier.ts
│ │ │ ├── more-info-image.ts
│ │ │ ├── more-info-input_boolean.ts
│ │ │ ├── more-info-input_datetime.ts
│ │ │ ├── more-info-lawn_mower.ts
│ │ │ ├── more-info-light.ts
│ │ │ ├── more-info-lock.ts
│ │ │ ├── more-info-media_player.ts
│ │ │ ├── more-info-person.ts
│ │ │ ├── more-info-remote.ts
│ │ │ ├── more-info-script.ts
│ │ │ ├── more-info-siren.ts
│ │ │ ├── more-info-sun.ts
│ │ │ ├── more-info-switch.ts
│ │ │ ├── more-info-time.ts
│ │ │ ├── more-info-timer.ts
│ │ │ ├── more-info-update.ts
│ │ │ ├── more-info-vacuum.ts
│ │ │ ├── more-info-valve.ts
│ │ │ ├── more-info-water_heater.ts
│ │ │ └── more-info-weather.ts
│ │ ├── ha-more-info-dialog.ts
│ │ ├── ha-more-info-history-and-logbook.ts
│ │ ├── ha-more-info-history.ts
│ │ ├── ha-more-info-info.ts
│ │ ├── ha-more-info-logbook.ts
│ │ ├── ha-more-info-settings.ts
│ │ ├── more-info-content.ts
│ │ ├── show-ha-more-info-dialog.ts
│ │ └── state_more_info_control.ts
│ ├── notifications
│ │ ├── configurator-notification-item.ts
│ │ ├── notification-drawer.ts
│ │ ├── notification-item-template.ts
│ │ ├── notification-item.ts
│ │ ├── persistent-notification-item.ts
│ │ └── show-notification-drawer.ts
│ ├── quick-bar
│ │ ├── ha-quick-bar.ts
│ │ └── show-dialog-quick-bar.ts
│ ├── restart
│ │ ├── dialog-restart-wait.ts
│ │ ├── dialog-restart.ts
│ │ └── show-dialog-restart.ts
│ ├── shortcuts
│ │ ├── dialog-shortcuts.ts
│ │ └── show-shortcuts-dialog.ts
│ ├── sidebar
│ │ ├── dialog-edit-sidebar.ts
│ │ └── show-dialog-edit-sidebar.ts
│ ├── tts-try
│ │ ├── dialog-tts-try.ts
│ │ └── show-dialog-tts-try.ts
│ ├── update_backup
│ │ ├── dialog-update-backup.ts
│ │ └── show-update-backup-dialog.ts
│ ├── voice-assistant-setup
│ │ ├── cloud
│ │ │ ├── cloud-step-intro.ts
│ │ │ ├── cloud-step-signin.ts
│ │ │ └── cloud-step-signup.ts
│ │ ├── show-voice-assistant-setup-dialog.ts
│ │ ├── styles.ts
│ │ ├── voice-assistant-setup-dialog.ts
│ │ ├── voice-assistant-setup-step-area.ts
│ │ ├── voice-assistant-setup-step-change-wake-word.ts
│ │ ├── voice-assistant-setup-step-check.ts
│ │ ├── voice-assistant-setup-step-cloud.ts
│ │ ├── voice-assistant-setup-step-local.ts
│ │ ├── voice-assistant-setup-step-pipeline.ts
│ │ ├── voice-assistant-setup-step-success.ts
│ │ ├── voice-assistant-setup-step-update.ts
│ │ └── voice-assistant-setup-step-wake-word.ts
│ └── voice-command-dialog
│ │ ├── ha-voice-command-dialog.ts
│ │ └── show-ha-voice-command-dialog.ts
├── entrypoints
│ ├── app.ts
│ ├── authorize.ts
│ ├── core.ts
│ ├── custom-panel.ts
│ ├── onboarding.ts
│ └── service-worker.ts
├── external_app
│ ├── external_app_entrypoint.ts
│ ├── external_auth.ts
│ └── external_messaging.ts
├── fake_data
│ ├── demo_config.ts
│ ├── demo_panels.ts
│ ├── demo_services.ts
│ ├── entity.ts
│ ├── entity_component_icons.ts
│ └── provide_hass.ts
├── html
│ ├── _header.html.template
│ ├── _js_base.html.template
│ ├── _preload_roboto.html.template
│ ├── _script_loader.html.template
│ ├── _style_base.html.template
│ ├── authorize.html.template
│ ├── index.html.template
│ └── onboarding.html.template
├── layouts
│ ├── ha-init-page.ts
│ ├── hass-error-screen.ts
│ ├── hass-loading-screen.ts
│ ├── hass-router-page.ts
│ ├── hass-subpage.ts
│ ├── hass-tabs-subpage-data-table.ts
│ ├── hass-tabs-subpage.ts
│ ├── home-assistant-main.ts
│ ├── home-assistant.ts
│ ├── partial-panel-resolver.ts
│ └── supervisor-error-screen.ts
├── managers
│ └── notification-manager.ts
├── mixins
│ ├── keyboard-shortcut-mixin.ts
│ ├── lit-localize-lite-mixin.ts
│ ├── prevent-unsaved-mixin.ts
│ ├── provide-hass-lit-mixin.ts
│ ├── subscribe-mixin.ts
│ └── wakelock-mixin.ts
├── onboarding
│ ├── dialogs
│ │ ├── app-dialog.ts
│ │ ├── community-dialog.ts
│ │ ├── show-app-dialog.ts
│ │ └── show-community-dialog.ts
│ ├── ha-onboarding.ts
│ ├── integration-badge.ts
│ ├── onboarding-analytics.ts
│ ├── onboarding-core-config.ts
│ ├── onboarding-create-user.ts
│ ├── onboarding-integrations.ts
│ ├── onboarding-loading.ts
│ ├── onboarding-location.ts
│ ├── onboarding-restore-backup.ts
│ ├── onboarding-welcome-link.ts
│ ├── onboarding-welcome-links.ts
│ ├── onboarding-welcome.ts
│ ├── restore-backup
│ │ ├── onboarding-restore-backup-cloud-login.ts
│ │ ├── onboarding-restore-backup-no-cloud-backup.ts
│ │ ├── onboarding-restore-backup-restore.ts
│ │ ├── onboarding-restore-backup-status.ts
│ │ └── onboarding-restore-backup-upload.ts
│ └── styles.ts
├── panels
│ ├── calendar
│ │ ├── confirm-event-dialog-box.ts
│ │ ├── dialog-calendar-event-detail.ts
│ │ ├── dialog-calendar-event-editor.ts
│ │ ├── ha-full-calendar.ts
│ │ ├── ha-panel-calendar.ts
│ │ ├── ha-recurrence-rule-editor.ts
│ │ ├── recurrence.ts
│ │ ├── show-confirm-event-dialog-box.ts
│ │ ├── show-dialog-calendar-event-detail.ts
│ │ └── show-dialog-calendar-event-editor.ts
│ ├── config
│ │ ├── application_credentials
│ │ │ ├── dialog-add-application-credential.ts
│ │ │ ├── ha-config-application-credentials.ts
│ │ │ └── show-dialog-add-application-credential.ts
│ │ ├── areas
│ │ │ ├── dialog-area-registry-detail.ts
│ │ │ ├── dialog-floor-registry-detail.ts
│ │ │ ├── ha-config-area-page.ts
│ │ │ ├── ha-config-areas-dashboard.ts
│ │ │ ├── ha-config-areas.ts
│ │ │ ├── show-dialog-area-registry-detail.ts
│ │ │ └── show-dialog-floor-registry-detail.ts
│ │ ├── automation
│ │ │ ├── action
│ │ │ │ ├── ha-automation-action-row.ts
│ │ │ │ ├── ha-automation-action.ts
│ │ │ │ └── types
│ │ │ │ │ ├── ha-automation-action-choose.ts
│ │ │ │ │ ├── ha-automation-action-condition.ts
│ │ │ │ │ ├── ha-automation-action-delay.ts
│ │ │ │ │ ├── ha-automation-action-device_id.ts
│ │ │ │ │ ├── ha-automation-action-event.ts
│ │ │ │ │ ├── ha-automation-action-if.ts
│ │ │ │ │ ├── ha-automation-action-parallel.ts
│ │ │ │ │ ├── ha-automation-action-play_media.ts
│ │ │ │ │ ├── ha-automation-action-repeat.ts
│ │ │ │ │ ├── ha-automation-action-sequence.ts
│ │ │ │ │ ├── ha-automation-action-service.ts
│ │ │ │ │ ├── ha-automation-action-set_conversation_response.ts
│ │ │ │ │ ├── ha-automation-action-stop.ts
│ │ │ │ │ ├── ha-automation-action-wait_for_trigger.ts
│ │ │ │ │ └── ha-automation-action-wait_template.ts
│ │ │ ├── add-automation-element-dialog.ts
│ │ │ ├── automation-mode-dialog
│ │ │ │ ├── dialog-automation-mode.ts
│ │ │ │ └── show-dialog-automation-mode.ts
│ │ │ ├── automation-save-dialog
│ │ │ │ ├── dialog-automation-save.ts
│ │ │ │ └── show-dialog-automation-save.ts
│ │ │ ├── blueprint-automation-editor.ts
│ │ │ ├── condition
│ │ │ │ ├── ha-automation-condition-editor.ts
│ │ │ │ ├── ha-automation-condition-row.ts
│ │ │ │ ├── ha-automation-condition.ts
│ │ │ │ └── types
│ │ │ │ │ ├── ha-automation-condition-and.ts
│ │ │ │ │ ├── ha-automation-condition-device.ts
│ │ │ │ │ ├── ha-automation-condition-logical.ts
│ │ │ │ │ ├── ha-automation-condition-not.ts
│ │ │ │ │ ├── ha-automation-condition-numeric_state.ts
│ │ │ │ │ ├── ha-automation-condition-or.ts
│ │ │ │ │ ├── ha-automation-condition-state.ts
│ │ │ │ │ ├── ha-automation-condition-sun.ts
│ │ │ │ │ ├── ha-automation-condition-template.ts
│ │ │ │ │ ├── ha-automation-condition-time.ts
│ │ │ │ │ ├── ha-automation-condition-trigger.ts
│ │ │ │ │ └── ha-automation-condition-zone.ts
│ │ │ ├── dialog-new-automation.ts
│ │ │ ├── ha-automation-editor.ts
│ │ │ ├── ha-automation-picker.ts
│ │ │ ├── ha-automation-trace.ts
│ │ │ ├── ha-config-automation.ts
│ │ │ ├── manual-automation-editor.ts
│ │ │ ├── option
│ │ │ │ ├── ha-automation-option-row.ts
│ │ │ │ └── ha-automation-option.ts
│ │ │ ├── paste-replace-dialog
│ │ │ │ ├── dialog-paste-replace.ts
│ │ │ │ └── show-dialog-paste-replace.ts
│ │ │ ├── show-add-automation-element-dialog.ts
│ │ │ ├── show-dialog-new-automation.ts
│ │ │ ├── structs.ts
│ │ │ └── trigger
│ │ │ │ ├── ha-automation-trigger-row.ts
│ │ │ │ ├── ha-automation-trigger.ts
│ │ │ │ └── types
│ │ │ │ ├── ha-automation-trigger-calendar.ts
│ │ │ │ ├── ha-automation-trigger-conversation.ts
│ │ │ │ ├── ha-automation-trigger-device.ts
│ │ │ │ ├── ha-automation-trigger-event.ts
│ │ │ │ ├── ha-automation-trigger-geo_location.ts
│ │ │ │ ├── ha-automation-trigger-homeassistant.ts
│ │ │ │ ├── ha-automation-trigger-list.ts
│ │ │ │ ├── ha-automation-trigger-mqtt.ts
│ │ │ │ ├── ha-automation-trigger-numeric_state.ts
│ │ │ │ ├── ha-automation-trigger-persistent_notification.ts
│ │ │ │ ├── ha-automation-trigger-state.ts
│ │ │ │ ├── ha-automation-trigger-sun.ts
│ │ │ │ ├── ha-automation-trigger-tag.ts
│ │ │ │ ├── ha-automation-trigger-template.ts
│ │ │ │ ├── ha-automation-trigger-time.ts
│ │ │ │ ├── ha-automation-trigger-time_pattern.ts
│ │ │ │ ├── ha-automation-trigger-webhook.ts
│ │ │ │ └── ha-automation-trigger-zone.ts
│ │ ├── backup
│ │ │ ├── components
│ │ │ │ ├── config
│ │ │ │ │ ├── ha-backup-config-addon.ts
│ │ │ │ │ ├── ha-backup-config-agents.ts
│ │ │ │ │ ├── ha-backup-config-data.ts
│ │ │ │ │ ├── ha-backup-config-encryption-key.ts
│ │ │ │ │ ├── ha-backup-config-retention.ts
│ │ │ │ │ └── ha-backup-config-schedule.ts
│ │ │ │ ├── ha-backup-addons-picker.ts
│ │ │ │ ├── ha-backup-agents-picker.ts
│ │ │ │ ├── ha-backup-data-picker.ts
│ │ │ │ ├── ha-backup-details-restore.ts
│ │ │ │ ├── ha-backup-details-summary.ts
│ │ │ │ ├── ha-backup-formfield-label.ts
│ │ │ │ ├── ha-backup-summary-card.ts
│ │ │ │ └── overview
│ │ │ │ │ ├── ha-backup-overview-backups.ts
│ │ │ │ │ ├── ha-backup-overview-onboarding.ts
│ │ │ │ │ ├── ha-backup-overview-progress.ts
│ │ │ │ │ ├── ha-backup-overview-settings.ts
│ │ │ │ │ └── ha-backup-overview-summary.ts
│ │ │ ├── dialogs
│ │ │ │ ├── dialog-backup-onboarding.ts
│ │ │ │ ├── dialog-change-backup-encryption-key.ts
│ │ │ │ ├── dialog-download-decrypted-backup.ts
│ │ │ │ ├── dialog-generate-backup.ts
│ │ │ │ ├── dialog-local-backup-location.ts
│ │ │ │ ├── dialog-new-backup.ts
│ │ │ │ ├── dialog-restore-backup.ts
│ │ │ │ ├── dialog-set-backup-encryption-key.ts
│ │ │ │ ├── dialog-show-backup-encryption-key.ts
│ │ │ │ ├── dialog-upload-backup.ts
│ │ │ │ ├── show-dialog-backup_onboarding.ts
│ │ │ │ ├── show-dialog-change-backup-encryption-key.ts
│ │ │ │ ├── show-dialog-download-decrypted-backup.ts
│ │ │ │ ├── show-dialog-generate-backup.ts
│ │ │ │ ├── show-dialog-local-backup-location.ts
│ │ │ │ ├── show-dialog-new-backup.ts
│ │ │ │ ├── show-dialog-restore-backup.ts
│ │ │ │ ├── show-dialog-set-backup-encryption-key.ts
│ │ │ │ ├── show-dialog-show-backup-encryption-key.ts
│ │ │ │ └── show-dialog-upload-backup.ts
│ │ │ ├── ha-config-backup-backups.ts
│ │ │ ├── ha-config-backup-details.ts
│ │ │ ├── ha-config-backup-location.ts
│ │ │ ├── ha-config-backup-overview.ts
│ │ │ ├── ha-config-backup-settings.ts
│ │ │ ├── ha-config-backup.ts
│ │ │ └── helper
│ │ │ │ └── download_backup.ts
│ │ ├── blueprint
│ │ │ ├── blueprint-generic-editor.ts
│ │ │ ├── dialog-import-blueprint.ts
│ │ │ ├── ha-blueprint-overview.ts
│ │ │ ├── ha-config-blueprint.ts
│ │ │ └── show-dialog-import-blueprint.ts
│ │ ├── category
│ │ │ ├── dialog-assign-category.ts
│ │ │ ├── dialog-category-registry-detail.ts
│ │ │ ├── ha-category-picker.ts
│ │ │ ├── show-dialog-assign-category.ts
│ │ │ └── show-dialog-category-registry-detail.ts
│ │ ├── cloud
│ │ │ ├── account
│ │ │ │ ├── cloud-account.ts
│ │ │ │ ├── cloud-ice-servers-pref.ts
│ │ │ │ ├── cloud-remote-pref.ts
│ │ │ │ ├── cloud-tts-pref.ts
│ │ │ │ ├── cloud-webhooks.ts
│ │ │ │ ├── dialog-cloud-support-package.ts
│ │ │ │ ├── dialog-cloud-tts-try.ts
│ │ │ │ ├── show-dialog-cloud-support-package.ts
│ │ │ │ └── show-dialog-cloud-tts-try.ts
│ │ │ ├── dialog-cloud-already-connected
│ │ │ │ ├── dialog-cloud-already-connected.ts
│ │ │ │ └── show-dialog-cloud-already-connected.ts
│ │ │ ├── dialog-cloud-certificate
│ │ │ │ ├── dialog-cloud-certificate.ts
│ │ │ │ └── show-dialog-cloud-certificate.ts
│ │ │ ├── dialog-manage-cloudhook
│ │ │ │ ├── dialog-manage-cloudhook.ts
│ │ │ │ └── show-dialog-manage-cloudhook.ts
│ │ │ ├── forgot-password
│ │ │ │ ├── cloud-forgot-password-card.ts
│ │ │ │ └── cloud-forgot-password.ts
│ │ │ ├── ha-config-cloud.ts
│ │ │ ├── login
│ │ │ │ ├── cloud-login-panel.ts
│ │ │ │ └── cloud-login.ts
│ │ │ └── register
│ │ │ │ └── cloud-register.ts
│ │ ├── core
│ │ │ ├── ha-config-analytics.ts
│ │ │ ├── ha-config-section-analytics.ts
│ │ │ ├── ha-config-section-general.ts
│ │ │ ├── ha-config-section-updates.ts
│ │ │ ├── ha-config-system-navigation.ts
│ │ │ └── updates
│ │ │ │ ├── dialog-join-beta.ts
│ │ │ │ └── show-dialog-join-beta.ts
│ │ ├── dashboard
│ │ │ ├── dialog-new-dashboard.ts
│ │ │ ├── ha-config-dashboard.ts
│ │ │ ├── ha-config-navigation.ts
│ │ │ ├── ha-config-updates.ts
│ │ │ └── show-dialog-new-dashboard.ts
│ │ ├── devices
│ │ │ ├── device-detail
│ │ │ │ ├── ha-device-automation-dialog.ts
│ │ │ │ ├── ha-device-entities-card.ts
│ │ │ │ ├── ha-device-info-card.ts
│ │ │ │ ├── ha-device-via-devices-card.ts
│ │ │ │ ├── integration-elements
│ │ │ │ │ ├── matter
│ │ │ │ │ │ ├── device-actions.ts
│ │ │ │ │ │ └── ha-device-info-matter.ts
│ │ │ │ │ ├── mqtt
│ │ │ │ │ │ ├── device-actions.ts
│ │ │ │ │ │ ├── dialog-mqtt-device-debug-info.ts
│ │ │ │ │ │ ├── mqtt-discovery-payload.ts
│ │ │ │ │ │ ├── mqtt-messages.ts
│ │ │ │ │ │ └── show-dialog-mqtt-device-debug-info.ts
│ │ │ │ │ ├── zha
│ │ │ │ │ │ ├── device-actions.ts
│ │ │ │ │ │ └── ha-device-info-zha.ts
│ │ │ │ │ └── zwave_js
│ │ │ │ │ │ ├── device-actions.ts
│ │ │ │ │ │ ├── device-alerts.ts
│ │ │ │ │ │ └── ha-device-info-zwave_js.ts
│ │ │ │ └── show-dialog-device-automation.ts
│ │ │ ├── device-registry-detail
│ │ │ │ ├── dialog-device-registry-detail.ts
│ │ │ │ └── show-dialog-device-registry-detail.ts
│ │ │ ├── ha-config-device-page.ts
│ │ │ ├── ha-config-devices-dashboard.ts
│ │ │ └── ha-config-devices.ts
│ │ ├── energy
│ │ │ ├── components
│ │ │ │ ├── ha-energy-battery-settings.ts
│ │ │ │ ├── ha-energy-device-settings.ts
│ │ │ │ ├── ha-energy-gas-settings.ts
│ │ │ │ ├── ha-energy-grid-settings.ts
│ │ │ │ ├── ha-energy-solar-settings.ts
│ │ │ │ ├── ha-energy-validation-result.ts
│ │ │ │ ├── ha-energy-water-settings.ts
│ │ │ │ └── styles.ts
│ │ │ ├── dialogs
│ │ │ │ ├── dialog-energy-battery-settings.ts
│ │ │ │ ├── dialog-energy-device-settings.ts
│ │ │ │ ├── dialog-energy-gas-settings.ts
│ │ │ │ ├── dialog-energy-grid-flow-settings.ts
│ │ │ │ ├── dialog-energy-solar-settings.ts
│ │ │ │ ├── dialog-energy-water-settings.ts
│ │ │ │ └── show-dialogs-energy.ts
│ │ │ └── ha-config-energy.ts
│ │ ├── entities
│ │ │ ├── const.ts
│ │ │ ├── editor-tabs
│ │ │ │ └── settings
│ │ │ │ │ └── entity-settings-helper-tab.ts
│ │ │ ├── entity-registry-settings-editor.ts
│ │ │ ├── entity-registry-settings.ts
│ │ │ └── ha-config-entities.ts
│ │ ├── ha-config-section.ts
│ │ ├── ha-panel-config.ts
│ │ ├── hardware
│ │ │ ├── dialog-hardware-available.ts
│ │ │ ├── ha-config-hardware.ts
│ │ │ └── show-dialog-hardware-available.ts
│ │ ├── helpers
│ │ │ ├── const.ts
│ │ │ ├── dialog-helper-detail.ts
│ │ │ ├── forms
│ │ │ │ ├── dialog-schedule-block-info.ts
│ │ │ │ ├── ha-counter-form.ts
│ │ │ │ ├── ha-input_boolean-form.ts
│ │ │ │ ├── ha-input_button-form.ts
│ │ │ │ ├── ha-input_datetime-form.ts
│ │ │ │ ├── ha-input_number-form.ts
│ │ │ │ ├── ha-input_select-form.ts
│ │ │ │ ├── ha-input_text-form.ts
│ │ │ │ ├── ha-schedule-form.ts
│ │ │ │ ├── ha-timer-form.ts
│ │ │ │ └── show-dialog-schedule-block-info.ts
│ │ │ ├── ha-config-helpers.ts
│ │ │ └── show-dialog-helper-detail.ts
│ │ ├── info
│ │ │ └── ha-config-info.ts
│ │ ├── integrations
│ │ │ ├── dialog-add-integration.ts
│ │ │ ├── dialog-yaml-integration.ts
│ │ │ ├── ha-config-flow-card.ts
│ │ │ ├── ha-config-integration-page.ts
│ │ │ ├── ha-config-integrations-dashboard.ts
│ │ │ ├── ha-config-integrations.ts
│ │ │ ├── ha-disabled-config-entry-card.ts
│ │ │ ├── ha-domain-integrations.ts
│ │ │ ├── ha-ignored-config-entry-card.ts
│ │ │ ├── ha-integration-action-card.ts
│ │ │ ├── ha-integration-card.ts
│ │ │ ├── ha-integration-header.ts
│ │ │ ├── ha-integration-list-item.ts
│ │ │ ├── ha-integration-overflow-menu.ts
│ │ │ ├── integration-panels
│ │ │ │ ├── bluetooth
│ │ │ │ │ ├── bluetooth-advertisement-monitor.ts
│ │ │ │ │ ├── bluetooth-config-dashboard-router.ts
│ │ │ │ │ ├── bluetooth-config-dashboard.ts
│ │ │ │ │ ├── bluetooth-connection-monitor.ts
│ │ │ │ │ ├── bluetooth-network-visualization.ts
│ │ │ │ │ ├── dialog-bluetooth-device-info.ts
│ │ │ │ │ └── show-dialog-bluetooth-device-info.ts
│ │ │ │ ├── dhcp
│ │ │ │ │ └── dhcp-config-panel.ts
│ │ │ │ ├── matter
│ │ │ │ │ ├── dialog-matter-add-device.ts
│ │ │ │ │ ├── dialog-matter-manage-fabrics.ts
│ │ │ │ │ ├── dialog-matter-open-commissioning-window.ts
│ │ │ │ │ ├── dialog-matter-ping-node.ts
│ │ │ │ │ ├── dialog-matter-reinterview-node.ts
│ │ │ │ │ ├── matter-add-device.ts
│ │ │ │ │ ├── matter-add-device
│ │ │ │ │ │ ├── matter-add-device-apple-home.ts
│ │ │ │ │ │ ├── matter-add-device-commissioning.ts
│ │ │ │ │ │ ├── matter-add-device-existing.ts
│ │ │ │ │ │ ├── matter-add-device-generic.ts
│ │ │ │ │ │ ├── matter-add-device-google-home-fallback.ts
│ │ │ │ │ │ ├── matter-add-device-google-home.ts
│ │ │ │ │ │ ├── matter-add-device-main.ts
│ │ │ │ │ │ ├── matter-add-device-new.ts
│ │ │ │ │ │ └── matter-add-device-shared-styles.ts
│ │ │ │ │ ├── matter-config-dashboard.ts
│ │ │ │ │ ├── matter-config-panel.ts
│ │ │ │ │ ├── show-dialog-add-matter-device.ts
│ │ │ │ │ ├── show-dialog-matter-manage-fabrics.ts
│ │ │ │ │ ├── show-dialog-matter-open-commissioning-window.ts
│ │ │ │ │ ├── show-dialog-matter-ping-node.ts
│ │ │ │ │ └── show-dialog-matter-reinterview-node.ts
│ │ │ │ ├── mqtt
│ │ │ │ │ ├── mqtt-config-panel.ts
│ │ │ │ │ └── mqtt-subscribe-card.ts
│ │ │ │ ├── ssdp
│ │ │ │ │ ├── dialog-ssdp-discovery-info.ts
│ │ │ │ │ ├── dialog-ssdp-raw-data.ts
│ │ │ │ │ ├── show-dialog-ssdp-discovery-info.ts
│ │ │ │ │ ├── show-dialog-ssdp-raw-data.ts
│ │ │ │ │ └── ssdp-config-panel.ts
│ │ │ │ ├── thread
│ │ │ │ │ ├── dialog-thread-dataset.ts
│ │ │ │ │ ├── show-dialog-thread-dataset.ts
│ │ │ │ │ └── thread-config-panel.ts
│ │ │ │ ├── zeroconf
│ │ │ │ │ ├── dialog-zeroconf-discovery-info.ts
│ │ │ │ │ ├── show-dialog-zeroconf-discovery-info.ts
│ │ │ │ │ └── zeroconf-config-panel.ts
│ │ │ │ ├── zha
│ │ │ │ │ ├── dialog-zha-change-channel.ts
│ │ │ │ │ ├── dialog-zha-manage-zigbee-device.ts
│ │ │ │ │ ├── dialog-zha-reconfigure-device.ts
│ │ │ │ │ ├── functions.ts
│ │ │ │ │ ├── show-dialog-zha-change-channel.ts
│ │ │ │ │ ├── show-dialog-zha-manage-zigbee-device.ts
│ │ │ │ │ ├── show-dialog-zha-reconfigure-device.ts
│ │ │ │ │ ├── types.ts
│ │ │ │ │ ├── zha-add-devices-page.ts
│ │ │ │ │ ├── zha-add-group-page.ts
│ │ │ │ │ ├── zha-cluster-attributes.ts
│ │ │ │ │ ├── zha-cluster-commands.ts
│ │ │ │ │ ├── zha-clusters-data-table.ts
│ │ │ │ │ ├── zha-config-dashboard-router.ts
│ │ │ │ │ ├── zha-config-dashboard.ts
│ │ │ │ │ ├── zha-device-binding.ts
│ │ │ │ │ ├── zha-device-card.ts
│ │ │ │ │ ├── zha-device-endpoint-data-table.ts
│ │ │ │ │ ├── zha-device-neighbors.ts
│ │ │ │ │ ├── zha-device-pairing-status-card.ts
│ │ │ │ │ ├── zha-device-signature.ts
│ │ │ │ │ ├── zha-group-binding.ts
│ │ │ │ │ ├── zha-group-page.ts
│ │ │ │ │ ├── zha-groups-dashboard.ts
│ │ │ │ │ ├── zha-manage-clusters.ts
│ │ │ │ │ └── zha-network-visualization-page.ts
│ │ │ │ └── zwave_js
│ │ │ │ │ ├── add-node
│ │ │ │ │ ├── data.ts
│ │ │ │ │ ├── dialog-zwave_js-add-node.ts
│ │ │ │ │ ├── show-dialog-zwave_js-add-node.ts
│ │ │ │ │ ├── zwave-js-add-node-added-insecure.ts
│ │ │ │ │ ├── zwave-js-add-node-code-input.ts
│ │ │ │ │ ├── zwave-js-add-node-configure-device.ts
│ │ │ │ │ ├── zwave-js-add-node-failed.ts
│ │ │ │ │ ├── zwave-js-add-node-grant-security-classes.ts
│ │ │ │ │ ├── zwave-js-add-node-loading.ts
│ │ │ │ │ ├── zwave-js-add-node-searching-devices.ts
│ │ │ │ │ ├── zwave-js-add-node-select-method.ts
│ │ │ │ │ ├── zwave-js-add-node-select-security-strategy.ts
│ │ │ │ │ └── zwave_js-add-node.ts
│ │ │ │ │ ├── capability-controls
│ │ │ │ │ ├── zwave_js-capability-control-color-switch.ts
│ │ │ │ │ ├── zwave_js-capability-control-door-lock.ts
│ │ │ │ │ ├── zwave_js-capability-control-multilevel-switch.ts
│ │ │ │ │ └── zwave_js-capability-control-thermostat-setback.ts
│ │ │ │ │ ├── dialog-zwave_js-hard-reset-controller.ts
│ │ │ │ │ ├── dialog-zwave_js-node-statistics.ts
│ │ │ │ │ ├── dialog-zwave_js-rebuild-network-routes.ts
│ │ │ │ │ ├── dialog-zwave_js-rebuild-node-routes.ts
│ │ │ │ │ ├── dialog-zwave_js-reinterview-node.ts
│ │ │ │ │ ├── dialog-zwave_js-remove-failed-node.ts
│ │ │ │ │ ├── dialog-zwave_js-remove-node.ts
│ │ │ │ │ ├── dialog-zwave_js-update-firmware-node.ts
│ │ │ │ │ ├── show-dialog-zwave_js-hard-reset-controller.ts
│ │ │ │ │ ├── show-dialog-zwave_js-node-statistics.ts
│ │ │ │ │ ├── show-dialog-zwave_js-rebuild-network-routes.ts
│ │ │ │ │ ├── show-dialog-zwave_js-rebuild-node-routes.ts
│ │ │ │ │ ├── show-dialog-zwave_js-reinterview-node.ts
│ │ │ │ │ ├── show-dialog-zwave_js-remove-failed-node.ts
│ │ │ │ │ ├── show-dialog-zwave_js-remove-node.ts
│ │ │ │ │ ├── show-dialog-zwave_js-update-firmware-node.ts
│ │ │ │ │ ├── zwave_js-config-dashboard.ts
│ │ │ │ │ ├── zwave_js-config-router.ts
│ │ │ │ │ ├── zwave_js-custom-param.ts
│ │ │ │ │ ├── zwave_js-logs.ts
│ │ │ │ │ ├── zwave_js-node-config.ts
│ │ │ │ │ ├── zwave_js-node-installer.ts
│ │ │ │ │ └── zwave_js-provisioned.ts
│ │ │ └── show-add-integration-dialog.ts
│ │ ├── labels
│ │ │ ├── dialog-label-detail.ts
│ │ │ ├── ha-config-labels.ts
│ │ │ └── show-dialog-label-detail.ts
│ │ ├── logs
│ │ │ ├── dialog-download-logs.ts
│ │ │ ├── dialog-system-log-detail.ts
│ │ │ ├── error-log-card.ts
│ │ │ ├── ha-config-logs.ts
│ │ │ ├── show-dialog-download-logs.ts
│ │ │ ├── show-dialog-system-log-detail.ts
│ │ │ ├── system-log-card.ts
│ │ │ └── util.ts
│ │ ├── lovelace
│ │ │ ├── dashboards
│ │ │ │ ├── dialog-lovelace-dashboard-configure-strategy.ts
│ │ │ │ ├── dialog-lovelace-dashboard-detail.ts
│ │ │ │ ├── ha-config-lovelace-dashboards.ts
│ │ │ │ ├── show-dialog-lovelace-dashboard-configure-strategy.ts
│ │ │ │ └── show-dialog-lovelace-dashboard-detail.ts
│ │ │ ├── ha-config-lovelace.ts
│ │ │ └── resources
│ │ │ │ ├── dialog-lovelace-resource-detail.ts
│ │ │ │ ├── ha-config-lovelace-resources.ts
│ │ │ │ └── show-dialog-lovelace-resource-detail.ts
│ │ ├── network
│ │ │ ├── ha-config-network.ts
│ │ │ ├── ha-config-section-network.ts
│ │ │ ├── ha-config-url-form.ts
│ │ │ ├── supervisor-hostname.ts
│ │ │ └── supervisor-network.ts
│ │ ├── person
│ │ │ ├── dialog-person-detail.ts
│ │ │ ├── ha-config-person.ts
│ │ │ └── show-dialog-person-detail.ts
│ │ ├── repairs
│ │ │ ├── dialog-integration-startup.ts
│ │ │ ├── dialog-repairs-issue-subtitle.ts
│ │ │ ├── dialog-repairs-issue.ts
│ │ │ ├── dialog-system-information.ts
│ │ │ ├── ha-config-repairs-dashboard.ts
│ │ │ ├── ha-config-repairs.ts
│ │ │ ├── integrations-startup-time.ts
│ │ │ ├── show-dialog-repair-flow.ts
│ │ │ ├── show-integration-startup-dialog.ts
│ │ │ ├── show-repair-issue-dialog.ts
│ │ │ └── show-system-information-dialog.ts
│ │ ├── scene
│ │ │ ├── ha-config-scene.ts
│ │ │ ├── ha-scene-dashboard.ts
│ │ │ └── ha-scene-editor.ts
│ │ ├── script
│ │ │ ├── blueprint-script-editor.ts
│ │ │ ├── ha-config-script.ts
│ │ │ ├── ha-script-editor.ts
│ │ │ ├── ha-script-field-row.ts
│ │ │ ├── ha-script-fields.ts
│ │ │ ├── ha-script-picker.ts
│ │ │ ├── ha-script-trace.ts
│ │ │ └── manual-script-editor.ts
│ │ ├── storage
│ │ │ ├── dialog-mount-view.ts
│ │ │ ├── dialog-move-datadisk.ts
│ │ │ ├── ha-config-section-storage.ts
│ │ │ ├── show-dialog-move-datadisk.ts
│ │ │ └── show-dialog-view-mount.ts
│ │ ├── tags
│ │ │ ├── dialog-tag-detail.ts
│ │ │ ├── ha-config-tags.ts
│ │ │ ├── show-dialog-tag-detail.ts
│ │ │ └── tag-image.ts
│ │ ├── users
│ │ │ ├── dialog-add-user.ts
│ │ │ ├── dialog-admin-change-password.ts
│ │ │ ├── dialog-user-detail.ts
│ │ │ ├── ha-config-users.ts
│ │ │ ├── show-dialog-add-user.ts
│ │ │ ├── show-dialog-admin-change-password.ts
│ │ │ └── show-dialog-user-detail.ts
│ │ ├── voice-assistants
│ │ │ ├── assist-pipeline-detail
│ │ │ │ ├── assist-pipeline-detail-config.ts
│ │ │ │ ├── assist-pipeline-detail-conversation.ts
│ │ │ │ ├── assist-pipeline-detail-stt.ts
│ │ │ │ ├── assist-pipeline-detail-tts.ts
│ │ │ │ └── assist-pipeline-detail-wakeword.ts
│ │ │ ├── assist-pref.ts
│ │ │ ├── assist
│ │ │ │ └── ha-config-voice-assistants-assist-devices.ts
│ │ │ ├── cloud-alexa-pref.ts
│ │ │ ├── cloud-discover.ts
│ │ │ ├── cloud-google-pref.ts
│ │ │ ├── debug
│ │ │ │ ├── assist-debug.ts
│ │ │ │ ├── assist-pipeline-debug.ts
│ │ │ │ ├── assist-pipeline-run-debug.ts
│ │ │ │ ├── assist-render-pipeline-events.ts
│ │ │ │ └── assist-render-pipeline-run.ts
│ │ │ ├── dialog-expose-entity.ts
│ │ │ ├── dialog-voice-assistant-pipeline-detail.ts
│ │ │ ├── dialog-voice-settings.ts
│ │ │ ├── entity-voice-settings.ts
│ │ │ ├── expose
│ │ │ │ └── expose-assistant-icon.ts
│ │ │ ├── ha-config-voice-assistants-assistants.ts
│ │ │ ├── ha-config-voice-assistants-expose.ts
│ │ │ ├── ha-config-voice-assistants.ts
│ │ │ ├── show-dialog-expose-entity.ts
│ │ │ ├── show-dialog-voice-assistant-pipeline-detail.ts
│ │ │ └── show-dialog-voice-settings.ts
│ │ └── zone
│ │ │ ├── dialog-home-zone-detail.ts
│ │ │ ├── dialog-zone-detail.ts
│ │ │ ├── ha-config-zone.ts
│ │ │ ├── show-dialog-home-zone-detail.ts
│ │ │ └── show-dialog-zone-detail.ts
│ ├── custom
│ │ └── ha-panel-custom.ts
│ ├── developer-tools
│ │ ├── action
│ │ │ └── developer-tools-action.ts
│ │ ├── assist
│ │ │ └── developer-tools-assist.ts
│ │ ├── debug
│ │ │ ├── developer-tools-debug.ts
│ │ │ └── ha-debug-connection-row.ts
│ │ ├── developer-tools-router.ts
│ │ ├── event
│ │ │ ├── developer-tools-event.ts
│ │ │ ├── event-subscribe-card.ts
│ │ │ └── events-list.ts
│ │ ├── ha-panel-developer-tools.ts
│ │ ├── state
│ │ │ └── developer-tools-state.ts
│ │ ├── statistics
│ │ │ ├── developer-tools-statistics.ts
│ │ │ ├── dialog-statistics-adjust-sum.ts
│ │ │ ├── dialog-statistics-fix-units-changed.ts
│ │ │ ├── dialog-statistics-fix.ts
│ │ │ ├── fix-statistics.ts
│ │ │ ├── show-dialog-statistics-adjust-sum.ts
│ │ │ ├── show-dialog-statistics-fix-units-changed.ts
│ │ │ └── show-dialog-statistics-fix.ts
│ │ ├── template
│ │ │ └── developer-tools-template.ts
│ │ └── yaml_configuration
│ │ │ └── developer-yaml-config.ts
│ ├── energy
│ │ ├── cards
│ │ │ └── energy-setup-wizard-card.ts
│ │ ├── ha-panel-energy.ts
│ │ └── strategies
│ │ │ └── energy-view-strategy.ts
│ ├── history
│ │ └── ha-panel-history.ts
│ ├── iframe
│ │ └── ha-panel-iframe.ts
│ ├── logbook
│ │ ├── ha-logbook-renderer.ts
│ │ ├── ha-logbook.ts
│ │ └── ha-panel-logbook.ts
│ ├── lovelace
│ │ ├── badges
│ │ │ ├── hui-badge.ts
│ │ │ ├── hui-entity-badge.ts
│ │ │ ├── hui-entity-filter-badge.ts
│ │ │ ├── hui-error-badge.ts
│ │ │ ├── hui-state-label-badge.ts
│ │ │ ├── hui-view-badges.ts
│ │ │ └── types.ts
│ │ ├── card-features
│ │ │ ├── common
│ │ │ │ ├── card-feature-styles.ts
│ │ │ │ └── filter-modes.ts
│ │ │ ├── hui-alarm-modes-card-feature.ts
│ │ │ ├── hui-card-feature.ts
│ │ │ ├── hui-card-features.ts
│ │ │ ├── hui-climate-fan-modes-card-feature.ts
│ │ │ ├── hui-climate-hvac-modes-card-feature.ts
│ │ │ ├── hui-climate-preset-modes-card-feature.ts
│ │ │ ├── hui-climate-swing-horizontal-modes-card-feature.ts
│ │ │ ├── hui-climate-swing-modes-card-feature.ts
│ │ │ ├── hui-counter-actions-card-feature.ts
│ │ │ ├── hui-cover-open-close-card-feature.ts
│ │ │ ├── hui-cover-position-card-feature.ts
│ │ │ ├── hui-cover-tilt-card-feature.ts
│ │ │ ├── hui-cover-tilt-position-card-feature.ts
│ │ │ ├── hui-fan-preset-modes-card-feature.ts
│ │ │ ├── hui-fan-speed-card-feature.ts
│ │ │ ├── hui-humidifier-modes-card-feature.ts
│ │ │ ├── hui-humidifier-toggle-card-feature.ts
│ │ │ ├── hui-lawn-mower-commands-card-feature.ts
│ │ │ ├── hui-light-brightness-card-feature.ts
│ │ │ ├── hui-light-color-temp-card-feature.ts
│ │ │ ├── hui-lock-commands-card-feature.ts
│ │ │ ├── hui-lock-open-door-card-feature.ts
│ │ │ ├── hui-media-player-volume-slider-card-feature.ts
│ │ │ ├── hui-numeric-input-card-feature.ts
│ │ │ ├── hui-select-options-card-feature.ts
│ │ │ ├── hui-target-humidity-card-feature.ts
│ │ │ ├── hui-target-temperature-card-feature.ts
│ │ │ ├── hui-toggle-card-feature.ts
│ │ │ ├── hui-update-actions-card-feature.ts
│ │ │ ├── hui-vacuum-commands-card-feature.ts
│ │ │ ├── hui-water-heater-operation-modes-card-feature.ts
│ │ │ └── types.ts
│ │ ├── cards
│ │ │ ├── energy
│ │ │ │ ├── common
│ │ │ │ │ ├── color.ts
│ │ │ │ │ └── energy-chart-options.ts
│ │ │ │ ├── hui-energy-carbon-consumed-gauge-card.ts
│ │ │ │ ├── hui-energy-compare-card.ts
│ │ │ │ ├── hui-energy-date-selection-card.ts
│ │ │ │ ├── hui-energy-devices-detail-graph-card.ts
│ │ │ │ ├── hui-energy-devices-graph-card.ts
│ │ │ │ ├── hui-energy-distribution-card.ts
│ │ │ │ ├── hui-energy-gas-graph-card.ts
│ │ │ │ ├── hui-energy-grid-neutrality-gauge-card.ts
│ │ │ │ ├── hui-energy-sankey-card.ts
│ │ │ │ ├── hui-energy-self-sufficiency-gauge-card.ts
│ │ │ │ ├── hui-energy-solar-consumed-gauge-card.ts
│ │ │ │ ├── hui-energy-solar-graph-card.ts
│ │ │ │ ├── hui-energy-sources-table-card.ts
│ │ │ │ ├── hui-energy-usage-graph-card.ts
│ │ │ │ └── hui-energy-water-graph-card.ts
│ │ │ ├── hui-alarm-panel-card.ts
│ │ │ ├── hui-area-card.ts
│ │ │ ├── hui-button-card.ts
│ │ │ ├── hui-calendar-card.ts
│ │ │ ├── hui-card.ts
│ │ │ ├── hui-clock-card.ts
│ │ │ ├── hui-conditional-card.ts
│ │ │ ├── hui-empty-state-card.ts
│ │ │ ├── hui-entities-card.ts
│ │ │ ├── hui-entity-button-card.ts
│ │ │ ├── hui-entity-card.ts
│ │ │ ├── hui-entity-filter-card.ts
│ │ │ ├── hui-error-card.ts
│ │ │ ├── hui-gauge-card.ts
│ │ │ ├── hui-glance-card.ts
│ │ │ ├── hui-grid-card.ts
│ │ │ ├── hui-heading-card.ts
│ │ │ ├── hui-history-graph-card.ts
│ │ │ ├── hui-horizontal-stack-card.ts
│ │ │ ├── hui-humidifier-card.ts
│ │ │ ├── hui-iframe-card.ts
│ │ │ ├── hui-light-card.ts
│ │ │ ├── hui-logbook-card.ts
│ │ │ ├── hui-map-card.ts
│ │ │ ├── hui-markdown-card.ts
│ │ │ ├── hui-media-control-card.ts
│ │ │ ├── hui-picture-card.ts
│ │ │ ├── hui-picture-elements-card.ts
│ │ │ ├── hui-picture-entity-card.ts
│ │ │ ├── hui-picture-glance-card.ts
│ │ │ ├── hui-plant-status-card.ts
│ │ │ ├── hui-recovery-mode-card.ts
│ │ │ ├── hui-sensor-card.ts
│ │ │ ├── hui-shopping-list-card.ts
│ │ │ ├── hui-stack-card.ts
│ │ │ ├── hui-starting-card.ts
│ │ │ ├── hui-statistic-card.ts
│ │ │ ├── hui-statistics-graph-card.ts
│ │ │ ├── hui-thermostat-card.ts
│ │ │ ├── hui-tile-card.ts
│ │ │ ├── hui-todo-list-card.ts
│ │ │ ├── hui-vertical-stack-card.ts
│ │ │ ├── hui-weather-forecast-card.ts
│ │ │ ├── picture-elements
│ │ │ │ └── create-styled-hui-element.ts
│ │ │ ├── tile
│ │ │ │ └── badges
│ │ │ │ │ ├── tile-badge-climate.ts
│ │ │ │ │ ├── tile-badge-humidifier.ts
│ │ │ │ │ ├── tile-badge-person.ts
│ │ │ │ │ └── tile-badge.ts
│ │ │ └── types.ts
│ │ ├── common
│ │ │ ├── check-lovelace-config.ts
│ │ │ ├── compute-card-grid-size.ts
│ │ │ ├── compute-card-size.ts
│ │ │ ├── compute-tooltip.ts
│ │ │ ├── compute-unused-entities.ts
│ │ │ ├── confirm-action.ts
│ │ │ ├── directives
│ │ │ │ └── action-handler-directive.ts
│ │ │ ├── entity
│ │ │ │ ├── toggle-entity.ts
│ │ │ │ ├── turn-on-off-entities.ts
│ │ │ │ └── turn-on-off-entity.ts
│ │ │ ├── evaluate-filter.ts
│ │ │ ├── find-entities.ts
│ │ │ ├── generate-lovelace-config.ts
│ │ │ ├── graph
│ │ │ │ ├── coordinates.ts
│ │ │ │ └── get-path.ts
│ │ │ ├── handle-action.ts
│ │ │ ├── has-action.ts
│ │ │ ├── has-changed.ts
│ │ │ ├── icon-condition.ts
│ │ │ ├── is-valid-object.ts
│ │ │ ├── load-resources.ts
│ │ │ ├── process-config-entities.ts
│ │ │ └── validate-condition.ts
│ │ ├── components
│ │ │ ├── hui-action-editor.ts
│ │ │ ├── hui-badge-edit-mode.ts
│ │ │ ├── hui-buttons-base.ts
│ │ │ ├── hui-card-edit-mode.ts
│ │ │ ├── hui-card-options.ts
│ │ │ ├── hui-conditional-base.ts
│ │ │ ├── hui-energy-period-selector.ts
│ │ │ ├── hui-entities-toggle.ts
│ │ │ ├── hui-entity-editor.ts
│ │ │ ├── hui-generic-entity-row.ts
│ │ │ ├── hui-graph-base.ts
│ │ │ ├── hui-image.ts
│ │ │ ├── hui-marquee.ts
│ │ │ ├── hui-timestamp-display.ts
│ │ │ ├── hui-warning-element.ts
│ │ │ ├── hui-warning.ts
│ │ │ └── types.ts
│ │ ├── create-element
│ │ │ ├── create-badge-element.ts
│ │ │ ├── create-card-element.ts
│ │ │ ├── create-card-feature-element.ts
│ │ │ ├── create-element-base.ts
│ │ │ ├── create-header-footer-element.ts
│ │ │ ├── create-heading-badge-element.ts
│ │ │ ├── create-hui-element.ts
│ │ │ ├── create-picture-element.ts
│ │ │ ├── create-row-element.ts
│ │ │ ├── create-section-element.ts
│ │ │ └── create-view-element.ts
│ │ ├── custom-card-helpers.ts
│ │ ├── editor
│ │ │ ├── add-entities-to-view.ts
│ │ │ ├── badge-editor
│ │ │ │ ├── hui-badge-element-editor.ts
│ │ │ │ ├── hui-badge-picker.ts
│ │ │ │ ├── hui-badge-visibility-editor.ts
│ │ │ │ ├── hui-dialog-create-badge.ts
│ │ │ │ ├── hui-dialog-edit-badge.ts
│ │ │ │ ├── hui-dialog-suggest-badge.ts
│ │ │ │ ├── show-create-badge-dialog.ts
│ │ │ │ ├── show-edit-badge-dialog.ts
│ │ │ │ └── show-suggest-badge-dialog.ts
│ │ │ ├── card-editor
│ │ │ │ ├── ha-grid-layout-slider.ts
│ │ │ │ ├── hui-card-element-editor.ts
│ │ │ │ ├── hui-card-layout-editor.ts
│ │ │ │ ├── hui-card-picker.ts
│ │ │ │ ├── hui-card-visibility-editor.ts
│ │ │ │ ├── hui-dialog-create-card.ts
│ │ │ │ ├── hui-dialog-delete-card.ts
│ │ │ │ ├── hui-dialog-edit-card.ts
│ │ │ │ ├── hui-dialog-suggest-card.ts
│ │ │ │ ├── hui-entity-picker-table.ts
│ │ │ │ ├── show-create-card-dialog.ts
│ │ │ │ ├── show-delete-card-dialog.ts
│ │ │ │ ├── show-edit-card-dialog.ts
│ │ │ │ └── show-suggest-card-dialog.ts
│ │ │ ├── conditions
│ │ │ │ ├── ha-card-condition-editor.ts
│ │ │ │ ├── ha-card-conditions-editor.ts
│ │ │ │ ├── types.ts
│ │ │ │ └── types
│ │ │ │ │ ├── ha-card-condition-and.ts
│ │ │ │ │ ├── ha-card-condition-numeric_state.ts
│ │ │ │ │ ├── ha-card-condition-or.ts
│ │ │ │ │ ├── ha-card-condition-screen.ts
│ │ │ │ │ ├── ha-card-condition-state.ts
│ │ │ │ │ └── ha-card-condition-user.ts
│ │ │ ├── config-elements
│ │ │ │ ├── config-elements-style.ts
│ │ │ │ ├── elements
│ │ │ │ │ ├── hui-conditional-element-editor.ts
│ │ │ │ │ ├── hui-icon-element-editor.ts
│ │ │ │ │ ├── hui-image-element-editor.ts
│ │ │ │ │ ├── hui-service-button-element-editor.ts
│ │ │ │ │ ├── hui-state-badge-element-editor.ts
│ │ │ │ │ ├── hui-state-icon-element-editor.ts
│ │ │ │ │ └── hui-state-label-element-editor.ts
│ │ │ │ ├── hui-alarm-modes-card-feature-editor.ts
│ │ │ │ ├── hui-alarm-panel-card-editor.ts
│ │ │ │ ├── hui-area-card-editor.ts
│ │ │ │ ├── hui-button-card-editor.ts
│ │ │ │ ├── hui-calendar-card-editor.ts
│ │ │ │ ├── hui-card-features-editor.ts
│ │ │ │ ├── hui-climate-fan-modes-card-feature-editor.ts
│ │ │ │ ├── hui-climate-hvac-modes-card-feature-editor.ts
│ │ │ │ ├── hui-climate-preset-modes-card-feature-editor.ts
│ │ │ │ ├── hui-climate-swing-horizontal-modes-card-feature-editor.ts
│ │ │ │ ├── hui-climate-swing-modes-card-feature-editor.ts
│ │ │ │ ├── hui-clock-card-editor.ts
│ │ │ │ ├── hui-conditional-card-editor.ts
│ │ │ │ ├── hui-counter-actions-card-feature-editor.ts
│ │ │ │ ├── hui-entities-card-editor.ts
│ │ │ │ ├── hui-entity-badge-editor.ts
│ │ │ │ ├── hui-entity-card-editor.ts
│ │ │ │ ├── hui-fan-preset-modes-card-feature-editor.ts
│ │ │ │ ├── hui-form-editor.ts
│ │ │ │ ├── hui-gauge-card-editor.ts
│ │ │ │ ├── hui-generic-entity-row-editor.ts
│ │ │ │ ├── hui-glance-card-editor.ts
│ │ │ │ ├── hui-graph-footer-editor.ts
│ │ │ │ ├── hui-grid-card-editor.ts
│ │ │ │ ├── hui-heading-badges-editor.ts
│ │ │ │ ├── hui-heading-card-editor.ts
│ │ │ │ ├── hui-history-graph-card-editor.ts
│ │ │ │ ├── hui-humidifier-card-editor.ts
│ │ │ │ ├── hui-humidifier-modes-card-feature-editor.ts
│ │ │ │ ├── hui-iframe-card-editor.ts
│ │ │ │ ├── hui-lawn-mower-commands-card-feature-editor.ts
│ │ │ │ ├── hui-light-card-editor.ts
│ │ │ │ ├── hui-logbook-card-editor.ts
│ │ │ │ ├── hui-map-card-editor.ts
│ │ │ │ ├── hui-markdown-card-editor.ts
│ │ │ │ ├── hui-media-control-card-editor.ts
│ │ │ │ ├── hui-numeric-input-card-feature-editor.ts
│ │ │ │ ├── hui-picture-card-editor.ts
│ │ │ │ ├── hui-picture-elements-card-editor.ts
│ │ │ │ ├── hui-picture-entity-card-editor.ts
│ │ │ │ ├── hui-picture-glance-card-editor.ts
│ │ │ │ ├── hui-plant-status-card-editor.ts
│ │ │ │ ├── hui-select-options-card-feature-editor.ts
│ │ │ │ ├── hui-sensor-card-editor.ts
│ │ │ │ ├── hui-stack-card-editor.ts
│ │ │ │ ├── hui-state-label-badge-editor.ts
│ │ │ │ ├── hui-statistic-card-editor.ts
│ │ │ │ ├── hui-statistics-graph-card-editor.ts
│ │ │ │ ├── hui-thermostat-card-editor.ts
│ │ │ │ ├── hui-tile-card-editor.ts
│ │ │ │ ├── hui-todo-list-editor.ts
│ │ │ │ ├── hui-update-actions-card-feature-editor.ts
│ │ │ │ ├── hui-vacuum-commands-card-feature-editor.ts
│ │ │ │ ├── hui-water-heater-operation-modes-card-feature-editor.ts
│ │ │ │ └── hui-weather-forecast-card-editor.ts
│ │ │ ├── config-util.ts
│ │ │ ├── dashboard-strategy-editor
│ │ │ │ ├── dialogs
│ │ │ │ │ ├── dialog-dashboard-strategy-editor.ts
│ │ │ │ │ └── show-dialog-dashboard-strategy-editor.ts
│ │ │ │ ├── hui-dashboard-strategy-element-editor.ts
│ │ │ │ ├── hui-iframe-dashboard-strategy-editor.ts
│ │ │ │ └── hui-original-states-dashboard-strategy-editor.ts
│ │ │ ├── delete-badge.ts
│ │ │ ├── delete-card.ts
│ │ │ ├── entity-row-editor
│ │ │ │ └── hui-row-element-editor.ts
│ │ │ ├── feature-editor
│ │ │ │ └── hui-card-feature-element-editor.ts
│ │ │ ├── get-badge-stub-config.ts
│ │ │ ├── get-card-stub-config.ts
│ │ │ ├── get-dashboard-documentation-url.ts
│ │ │ ├── get-element-stub-config.ts
│ │ │ ├── gui-support-error.ts
│ │ │ ├── header-footer-editor
│ │ │ │ ├── get-headerfooter-stub-config.ts
│ │ │ │ ├── hui-dialog-create-headerfooter.ts
│ │ │ │ ├── hui-header-footer-editor.ts
│ │ │ │ ├── hui-header-footer-element-editor.ts
│ │ │ │ └── show-create-headerfooter-dialog.ts
│ │ │ ├── heading-badge-editor
│ │ │ │ ├── hui-entity-heading-badge-editor.ts
│ │ │ │ └── hui-heading-badge-element-editor.ts
│ │ │ ├── hui-dialog-save-config.ts
│ │ │ ├── hui-element-editor.ts
│ │ │ ├── hui-entities-card-row-editor.ts
│ │ │ ├── hui-picture-elements-card-row-editor.ts
│ │ │ ├── hui-sub-element-editor.ts
│ │ │ ├── hui-typed-element-editor.ts
│ │ │ ├── lovelace-badges.ts
│ │ │ ├── lovelace-cards.ts
│ │ │ ├── lovelace-headerfooters.ts
│ │ │ ├── lovelace-path.ts
│ │ │ ├── picture-element-editor
│ │ │ │ └── hui-picture-element-element-editor.ts
│ │ │ ├── process-editor-entities.ts
│ │ │ ├── section-editor
│ │ │ │ ├── hui-dialog-edit-section.ts
│ │ │ │ ├── hui-section-settings-editor.ts
│ │ │ │ ├── hui-section-visibility-editor.ts
│ │ │ │ └── show-edit-section-dialog.ts
│ │ │ ├── select-dashboard
│ │ │ │ ├── hui-dialog-select-dashboard.ts
│ │ │ │ └── show-select-dashboard-dialog.ts
│ │ │ ├── select-view
│ │ │ │ ├── hui-dialog-select-view.ts
│ │ │ │ └── show-select-view-dialog.ts
│ │ │ ├── show-save-config-dialog.ts
│ │ │ ├── structs
│ │ │ │ ├── action-struct.ts
│ │ │ │ ├── base-badge-struct.ts
│ │ │ │ ├── base-card-struct.ts
│ │ │ │ ├── button-entity-struct.ts
│ │ │ │ └── entities-struct.ts
│ │ │ ├── types.ts
│ │ │ ├── unused-entities
│ │ │ │ └── hui-unused-entities.ts
│ │ │ ├── view-editor
│ │ │ │ ├── hui-dialog-edit-view.ts
│ │ │ │ ├── hui-view-background-editor.ts
│ │ │ │ ├── hui-view-editor.ts
│ │ │ │ ├── hui-view-visibility-editor.ts
│ │ │ │ └── show-edit-view-dialog.ts
│ │ │ └── view-header
│ │ │ │ ├── hui-dialog-edit-view-header.ts
│ │ │ │ ├── hui-view-header-settings-editor.ts
│ │ │ │ └── show-edit-view-header-dialog.ts
│ │ ├── elements
│ │ │ ├── hui-conditional-element.ts
│ │ │ ├── hui-icon-element.ts
│ │ │ ├── hui-image-element.ts
│ │ │ ├── hui-service-button-element.ts
│ │ │ ├── hui-state-badge-element.ts
│ │ │ ├── hui-state-icon-element.ts
│ │ │ ├── hui-state-label-element.ts
│ │ │ └── types.ts
│ │ ├── entity-rows
│ │ │ ├── hui-button-entity-row.ts
│ │ │ ├── hui-climate-entity-row.ts
│ │ │ ├── hui-cover-entity-row.ts
│ │ │ ├── hui-date-entity-row.ts
│ │ │ ├── hui-datetime-entity-row.ts
│ │ │ ├── hui-event-entity-row.ts
│ │ │ ├── hui-group-entity-row.ts
│ │ │ ├── hui-humidifier-entity-row.ts
│ │ │ ├── hui-input-button-entity-row.ts
│ │ │ ├── hui-input-datetime-entity-row.ts
│ │ │ ├── hui-input-number-entity-row.ts
│ │ │ ├── hui-input-select-entity-row.ts
│ │ │ ├── hui-input-text-entity-row.ts
│ │ │ ├── hui-lock-entity-row.ts
│ │ │ ├── hui-media-player-entity-row.ts
│ │ │ ├── hui-number-entity-row.ts
│ │ │ ├── hui-scene-entity-row.ts
│ │ │ ├── hui-script-entity-row.ts
│ │ │ ├── hui-select-entity-row.ts
│ │ │ ├── hui-sensor-entity-row.ts
│ │ │ ├── hui-simple-entity-row.ts
│ │ │ ├── hui-text-entity-row.ts
│ │ │ ├── hui-time-entity-row.ts
│ │ │ ├── hui-timer-entity-row.ts
│ │ │ ├── hui-toggle-entity-row.ts
│ │ │ ├── hui-update-entity-row.ts
│ │ │ ├── hui-valve-entity-row.ts
│ │ │ ├── hui-weather-entity-row.ts
│ │ │ └── types.ts
│ │ ├── ha-panel-lovelace.ts
│ │ ├── header-footer
│ │ │ ├── hui-buttons-header-footer.ts
│ │ │ ├── hui-graph-header-footer.ts
│ │ │ ├── hui-picture-header-footer.ts
│ │ │ ├── structs.ts
│ │ │ └── types.ts
│ │ ├── heading-badges
│ │ │ ├── hui-entity-heading-badge.ts
│ │ │ ├── hui-error-heading-badge.ts
│ │ │ ├── hui-heading-badge.ts
│ │ │ └── types.ts
│ │ ├── hui-editor.ts
│ │ ├── hui-root.ts
│ │ ├── sections
│ │ │ ├── const.ts
│ │ │ ├── hui-error-section.ts
│ │ │ ├── hui-grid-section.ts
│ │ │ └── hui-section.ts
│ │ ├── special-rows
│ │ │ ├── hui-attribute-row.ts
│ │ │ ├── hui-button-row.ts
│ │ │ ├── hui-buttons-row.ts
│ │ │ ├── hui-call-service-row.ts
│ │ │ ├── hui-cast-row.ts
│ │ │ ├── hui-conditional-row.ts
│ │ │ ├── hui-divider-row.ts
│ │ │ ├── hui-section-row.ts
│ │ │ ├── hui-text-row.ts
│ │ │ └── hui-weblink-row.ts
│ │ ├── strategies
│ │ │ ├── areas
│ │ │ │ ├── area-view-strategy.ts
│ │ │ │ ├── areas-dashboard-strategy.ts
│ │ │ │ ├── areas-overview-view-strategy.ts
│ │ │ │ ├── editor
│ │ │ │ │ └── hui-areas-dashboard-strategy-editor.ts
│ │ │ │ └── helpers
│ │ │ │ │ └── areas-strategy-helper.ts
│ │ │ ├── get-strategy.ts
│ │ │ ├── iframe
│ │ │ │ ├── iframe-dashboard-strategy.ts
│ │ │ │ └── iframe-view-strategy.ts
│ │ │ ├── legacy-strategy.ts
│ │ │ ├── map
│ │ │ │ ├── map-dashboard-strategy.ts
│ │ │ │ └── map-view-strategy.ts
│ │ │ ├── original-states
│ │ │ │ ├── original-states-dashboard-strategy.ts
│ │ │ │ └── original-states-view-strategy.ts
│ │ │ └── types.ts
│ │ ├── types.ts
│ │ └── views
│ │ │ ├── const.ts
│ │ │ ├── default-section.ts
│ │ │ ├── default-view-editable.ts
│ │ │ ├── get-view-type.ts
│ │ │ ├── hui-masonry-view.ts
│ │ │ ├── hui-panel-view.ts
│ │ │ ├── hui-sections-view.ts
│ │ │ ├── hui-sidebar-view.ts
│ │ │ ├── hui-view-background.ts
│ │ │ ├── hui-view-container.ts
│ │ │ ├── hui-view-header.ts
│ │ │ └── hui-view.ts
│ ├── map
│ │ └── ha-panel-map.ts
│ ├── media-browser
│ │ ├── browser-media-player.ts
│ │ ├── ha-bar-media-player.ts
│ │ ├── ha-panel-media-browser.ts
│ │ ├── hui-dialog-web-browser-play-media.ts
│ │ └── show-media-player-dialog.ts
│ ├── my
│ │ └── ha-panel-my.ts
│ ├── profile
│ │ ├── dialog-ha-mfa-module-setup-flow.ts
│ │ ├── ha-advanced-mode-row.ts
│ │ ├── ha-change-password-card.ts
│ │ ├── ha-enable-shortcuts-row.ts
│ │ ├── ha-entity-id-picker-row.ts
│ │ ├── ha-force-narrow-row.ts
│ │ ├── ha-long-lived-access-token-dialog.ts
│ │ ├── ha-long-lived-access-tokens-card.ts
│ │ ├── ha-mfa-modules-card.ts
│ │ ├── ha-panel-profile.ts
│ │ ├── ha-pick-dashboard-row.ts
│ │ ├── ha-pick-date-format-row.ts
│ │ ├── ha-pick-first-weekday-row.ts
│ │ ├── ha-pick-language-row.ts
│ │ ├── ha-pick-number-format-row.ts
│ │ ├── ha-pick-theme-row.ts
│ │ ├── ha-pick-time-format-row.ts
│ │ ├── ha-pick-time-zone-row.ts
│ │ ├── ha-profile-section-general.ts
│ │ ├── ha-profile-section-security.ts
│ │ ├── ha-push-notifications-row.ts
│ │ ├── ha-refresh-tokens-card.ts
│ │ ├── ha-set-suspend-row.ts
│ │ ├── ha-set-vibrate-row.ts
│ │ ├── show-ha-mfa-module-setup-flow-dialog.ts
│ │ └── show-long-lived-access-token-dialog.ts
│ └── todo
│ │ ├── dialog-todo-item-editor.ts
│ │ ├── ha-panel-todo.ts
│ │ └── show-dialog-todo-item-editor.ts
├── resources
│ ├── append-ha-style.ts
│ ├── codemirror.ts
│ ├── css-variables.ts
│ ├── custom-card-support.ts
│ ├── echarts.ts
│ ├── fuse.ts
│ ├── hammer.ts
│ ├── home-assistant-logo-svg.ts
│ ├── icon-metadata.ts
│ ├── log-message.ts
│ ├── markdown-worker.ts
│ ├── particles.ts
│ ├── polyfills
│ │ ├── element-getattributenames.ts
│ │ ├── element-toggleattribute.ts
│ │ ├── intl-polyfill.ts
│ │ ├── locale-data-polyfill.ts
│ │ └── resize-observer.ts
│ ├── render-markdown.ts
│ ├── roboto.ts
│ ├── sortable.ts
│ ├── styles.ts
│ ├── svg-arc.ts
│ ├── theme
│ │ ├── color.globals.ts
│ │ ├── main.globals.ts
│ │ ├── theme.ts
│ │ └── typography.globals.ts
│ ├── translations-metadata.ts
│ └── virtualizer.ts
├── state-control
│ ├── alarm_control_panel
│ │ └── ha-state-control-alarm_control_panel-modes.ts
│ ├── climate
│ │ ├── ha-state-control-climate-humidity.ts
│ │ └── ha-state-control-climate-temperature.ts
│ ├── cover
│ │ ├── ha-state-control-cover-buttons.ts
│ │ ├── ha-state-control-cover-position.ts
│ │ ├── ha-state-control-cover-tilt-position.ts
│ │ └── ha-state-control-cover-toggle.ts
│ ├── fan
│ │ └── ha-state-control-fan-speed.ts
│ ├── ha-state-control-toggle.ts
│ ├── humidifier
│ │ └── ha-state-control-humidifier-humidity.ts
│ ├── light
│ │ └── ha-state-control-light-brightness.ts
│ ├── lock
│ │ └── ha-state-control-lock-toggle.ts
│ ├── state-control-circular-slider-style.ts
│ ├── valve
│ │ ├── ha-state-control-valve-buttons.ts
│ │ ├── ha-state-control-valve-position.ts
│ │ └── ha-state-control-valve-toggle.ts
│ └── water_heater
│ │ └── ha-state-control-water_heater-temperature.ts
├── state-display
│ ├── ha-timer-remaining-time.ts
│ └── state-display.ts
├── state-summary
│ ├── state-card-alert.ts
│ ├── state-card-button.ts
│ ├── state-card-climate.ts
│ ├── state-card-configurator.ts
│ ├── state-card-content.ts
│ ├── state-card-cover.ts
│ ├── state-card-display.ts
│ ├── state-card-event.ts
│ ├── state-card-humidifier.ts
│ ├── state-card-input_button.ts
│ ├── state-card-input_number.ts
│ ├── state-card-input_select.ts
│ ├── state-card-input_text.ts
│ ├── state-card-lawn_mower.ts
│ ├── state-card-lock.ts
│ ├── state-card-media_player.ts
│ ├── state-card-number.ts
│ ├── state-card-scene.ts
│ ├── state-card-script.ts
│ ├── state-card-select.ts
│ ├── state-card-text.ts
│ ├── state-card-timer.ts
│ ├── state-card-toggle.ts
│ ├── state-card-update.ts
│ ├── state-card-vacuum.ts
│ └── state-card-water_heater.ts
├── state
│ ├── action-mixin.ts
│ ├── auth-mixin.ts
│ ├── connection-mixin.ts
│ ├── context-mixin.ts
│ ├── dialog-manager-mixin.ts
│ ├── disconnect-toast-mixin.ts
│ ├── haptic-mixin.ts
│ ├── hass-base-mixin.ts
│ ├── hass-element.ts
│ ├── logging-mixin.ts
│ ├── more-info-mixin.ts
│ ├── notification-mixin.ts
│ ├── panel-title-mixin.ts
│ ├── quick-bar-mixin.ts
│ ├── sidebar-mixin.ts
│ ├── state-display-mixin.ts
│ ├── themes-mixin.ts
│ ├── translations-mixin.ts
│ └── url-sync-mixin.ts
├── translations
│ ├── en.json
│ └── translationMetadata.json
├── types.ts
├── types
│ ├── audio.d.ts
│ ├── echarts.d.ts
│ ├── node-vibrant.d.ts
│ └── service-worker.d.ts
└── util
│ ├── audio-recorder.ts
│ ├── brands-url.ts
│ ├── bytes-to-string.ts
│ ├── cache-manager.ts
│ ├── calculate.ts
│ ├── common-translation.ts
│ ├── custom-panel
│ ├── create-custom-panel-element.ts
│ ├── load-custom-panel.ts
│ └── set-custom-panel-properties.ts
│ ├── documentation-url.ts
│ ├── empty.js
│ ├── fetch-with-auth.ts
│ ├── file_download.ts
│ ├── ha-pref-storage.ts
│ ├── hass-call-api.ts
│ ├── hass-media-player-model.ts
│ ├── iframe.ts
│ ├── is_ios.ts
│ ├── is_mac.ts
│ ├── is_mobile.ts
│ ├── is_safari.ts
│ ├── is_touch.ts
│ ├── launch-screen.ts
│ ├── recorder-worklet.js
│ ├── register-service-worker.ts
│ ├── text.ts
│ ├── toast-saved-success.ts
│ ├── toast.ts
│ └── url.ts
├── test
├── common
│ ├── array
│ │ ├── combination.test.ts
│ │ ├── ensure-array.test.ts
│ │ └── literal-includes.test.ts
│ ├── auth
│ │ └── token_storage
│ │ │ ├── askWrite.test.ts
│ │ │ ├── saveTokens.test.ts
│ │ │ └── token_storage.test.ts
│ ├── color
│ │ ├── colors.test.ts
│ │ ├── compute-color.test.ts
│ │ ├── convert-color.test.ts
│ │ ├── convert-light-color.test.ts
│ │ ├── hex.test.ts
│ │ ├── lab.test.ts
│ │ └── rgb.test.ts
│ ├── config
│ │ ├── can_show_page.test.ts
│ │ ├── components_with_service.test.ts
│ │ ├── is_component_loaded.test.ts
│ │ ├── is_pwa.test.ts
│ │ ├── is_service_loaded.test.ts
│ │ ├── location_name.test.ts
│ │ └── version.test.ts
│ ├── datetime
│ │ ├── absolute_time.test.ts
│ │ ├── calc_date.test.ts
│ │ ├── check_valid_date.test.ts
│ │ ├── create_duration_data.test.ts
│ │ ├── duration_to_seconds.test.ts
│ │ ├── first_weekday.test.ts
│ │ ├── format_date.test.ts
│ │ ├── format_date_time.test.ts
│ │ ├── format_duration.test.ts
│ │ ├── format_time.test.ts
│ │ ├── localize_date.test.ts
│ │ ├── milliseconds_to_duration.test.ts
│ │ ├── relative_time.test.ts
│ │ ├── resolve-time-zone.test.ts
│ │ ├── seconds_to_duration.test.ts
│ │ ├── shift_date_range.test.ts
│ │ └── use_am_pm.test.ts
│ ├── entity
│ │ ├── attribute_class_names.test.ts
│ │ ├── battery_icon.test.ts
│ │ ├── can_toggle_domain.test.ts
│ │ ├── can_toggle_state.test.ts
│ │ ├── color
│ │ │ └── battery_color.test.ts
│ │ ├── compute_area_name.test.ts
│ │ ├── compute_attribute_display.test.ts
│ │ ├── compute_device_name.test.ts
│ │ ├── compute_domain.test.ts
│ │ ├── compute_entity_name.test.ts
│ │ ├── compute_floor_name.test.ts
│ │ ├── compute_state_display.test.ts
│ │ ├── compute_state_domain.test.ts
│ │ ├── compute_state_name.test.ts
│ │ ├── context
│ │ │ ├── context-mock.ts
│ │ │ ├── get_area_context.test.ts
│ │ │ ├── get_device_context.test.ts
│ │ │ └── get_entity_context.test.ts
│ │ ├── cover_icon.test.ts
│ │ ├── delete_entity.test.ts
│ │ ├── entity_domain_filter.test.ts
│ │ ├── extract_views.test.ts
│ │ ├── feature_class_names.test.ts
│ │ ├── get_group_entities.test.ts
│ │ ├── get_view_entities.test.ts
│ │ ├── has_location.test.ts
│ │ ├── split_by_groups.test.ts
│ │ ├── state_card_type.test.ts
│ │ ├── state_color.ts
│ │ ├── state_more_info_type.test.ts
│ │ ├── test_util.ts
│ │ └── timer_time_remaining.test.ts
│ ├── string
│ │ ├── format_number.test.ts
│ │ ├── get_duplicate.test.ts
│ │ ├── is_date.test.ts
│ │ ├── sequence_matching.test.ts
│ │ └── slugify.test.ts
│ └── util
│ │ └── parse_aspect_ratio.test.ts
├── data
│ ├── energy.test.ts
│ └── image_upload.test.ts
├── external_app
│ ├── external_app_entrypoint.test.ts
│ └── external_messaging.test.ts
├── hassio
│ └── create_session.test.ts
├── panels
│ └── lovelace
│ │ └── editor
│ │ └── config-util.test.ts
├── setup.ts
├── test_helper
│ ├── local-storage-fallback.test.ts
│ └── local-storage-fallback.ts
├── util
│ ├── cache-manager.test.ts
│ ├── calculate.test.ts
│ ├── generate-brands-url.test.ts
│ ├── generate-documentation-url.test.ts
│ └── ha-pref-storage.test.ts
└── vitest.config.ts
├── tsconfig.json
└── yarn.lock
/.devcontainer/Dockerfile:
--------------------------------------------------------------------------------
1 | FROM mcr.microsoft.com/devcontainers/python:1-3.13
2 |
3 | ENV \
4 | DEBIAN_FRONTEND=noninteractive \
5 | DEVCONTAINER=true \
6 | PATH=$PATH:./node_modules/.bin
7 |
--------------------------------------------------------------------------------
/.gitattributes:
--------------------------------------------------------------------------------
1 | # Ensure Docker script files uses LF to support Docker for Windows.
2 | # Ensure "git config --global core.autocrlf input" before you clone
3 | * text eol=lf
4 | *.ts whitespace=error
5 | *.js whitespace=error
6 |
7 | *.ico binary
8 | *.jpg binary
9 | *.png binary
10 | *.zip binary
11 | *.mp3 binary
12 |
13 | demo/public/api/camera_proxy_stream/* binary
14 | demo/public/api/media_player_proxy/* binary
15 |
--------------------------------------------------------------------------------
/.github/dependabot.yml:
--------------------------------------------------------------------------------
1 | version: 2
2 | updates:
3 | - package-ecosystem: "github-actions"
4 | directory: "/"
5 | schedule:
6 | interval: weekly
7 | time: "06:00"
8 | open-pull-requests-limit: 10
9 | labels:
10 | - Dependencies
11 | - GitHub Actions
12 |
--------------------------------------------------------------------------------
/.github/move.yml:
--------------------------------------------------------------------------------
1 | # Configuration for move-issues - https://github.com/dessant/move-issues
2 |
3 | # Delete the command comment. Ignored when the comment also contains other content
4 | deleteCommand: true
5 | # Close the source issue after moving
6 | closeSourceIssue: true
7 | # Lock the source issue after moving
8 | lockSourceIssue: false
9 | # Set custom aliases for targets
10 | # aliases:
11 | # r: repo
12 | # or: owner/repo
13 |
14 |
--------------------------------------------------------------------------------
/.github/release-drafter.yml:
--------------------------------------------------------------------------------
1 | categories:
2 | - title: "Dependency updates"
3 | collapse-after: 3
4 | labels:
5 | - "Dependencies"
6 | template: |
7 | ## What's Changed
8 |
9 | $CHANGES
10 |
--------------------------------------------------------------------------------
/.github/workflows/labeler.yaml:
--------------------------------------------------------------------------------
1 | name: "Pull Request Labeler"
2 |
3 | on: pull_request_target
4 |
5 | jobs:
6 | triage:
7 | permissions:
8 | contents: read
9 | pull-requests: write
10 | runs-on: ubuntu-latest
11 | steps:
12 | - name: Apply labels
13 | uses: actions/labeler@v5.0.0
14 | with:
15 | sync-labels: true
16 |
--------------------------------------------------------------------------------
/.github/workflows/lock.yml:
--------------------------------------------------------------------------------
1 | name: Lock
2 |
3 | # yamllint disable-line rule:truthy
4 | on:
5 | schedule:
6 | - cron: "0 * * * *"
7 |
8 | jobs:
9 | lock:
10 | runs-on: ubuntu-latest
11 | steps:
12 | - uses: dessant/lock-threads@v5.0.1
13 | with:
14 | github-token: ${{ github.token }}
15 | process-only: "issues, prs"
16 | issue-lock-inactive-days: "30"
17 | issue-exclude-created-before: "2020-10-01T00:00:00Z"
18 | issue-lock-reason: ""
19 | pr-lock-inactive-days: "1"
20 | pr-exclude-created-before: "2020-11-01T00:00:00Z"
21 | pr-lock-reason: ""
22 |
--------------------------------------------------------------------------------
/.github/workflows/translations.yaml:
--------------------------------------------------------------------------------
1 | name: Translations
2 |
3 | on:
4 | push:
5 | branches:
6 | - dev
7 | paths:
8 | - src/translations/en.json
9 |
10 | jobs:
11 | upload:
12 | name: Upload
13 | runs-on: ubuntu-latest
14 | steps:
15 | - name: Checkout the repository
16 | uses: actions/checkout@v4.2.2
17 |
18 | - name: Upload Translations
19 | run: |
20 | export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}"
21 | ./script/translations_upload_base
22 |
--------------------------------------------------------------------------------
/.gitmodules:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/.gitmodules
--------------------------------------------------------------------------------
/.husky/pre-commit:
--------------------------------------------------------------------------------
1 | yarn run lint-staged --relative
2 |
--------------------------------------------------------------------------------
/.nvmrc:
--------------------------------------------------------------------------------
1 | lts/iron
2 |
--------------------------------------------------------------------------------
/.prettierignore:
--------------------------------------------------------------------------------
1 | CLA.md
2 | CODE_OF_CONDUCT.md
3 | LICENSE.md
4 | PULL_REQUEST_TEMPLATE.md
--------------------------------------------------------------------------------
/.vscode/extensions.json:
--------------------------------------------------------------------------------
1 | {
2 | "recommendations": [
3 | "dbaeumer.vscode-eslint",
4 | "esbenp.prettier-vscode",
5 | "runem.lit-plugin",
6 | "github.vscode-pull-request-github",
7 | "eamodio.gitlens",
8 | "vitest.explorer",
9 | "yeion7.styled-global-variables-autocomplete"
10 | ]
11 | }
12 |
--------------------------------------------------------------------------------
/.yarnrc.yml:
--------------------------------------------------------------------------------
1 | compressionLevel: mixed
2 |
3 | defaultSemverRangePrefix: ""
4 |
5 | enableGlobalCache: false
6 |
7 | nodeLinker: node-modules
8 |
9 | yarnPath: .yarn/releases/yarn-4.9.1.cjs
10 |
--------------------------------------------------------------------------------
/MANIFEST.in:
--------------------------------------------------------------------------------
1 | include README.md
2 | graft hass_frontend
3 | graft hass_frontend_es5
4 | recursive-exclude * *.py[co]
5 |
--------------------------------------------------------------------------------
/build-scripts/eslint.config.mjs:
--------------------------------------------------------------------------------
1 | // @ts-check
2 |
3 | import tseslint from "typescript-eslint";
4 | import rootConfig from "../eslint.config.mjs";
5 |
6 | export default tseslint.config(...rootConfig, {
7 | rules: {
8 | "no-console": "off",
9 | "import/no-extraneous-dependencies": "off",
10 | "import/extensions": "off",
11 | "import/no-dynamic-require": "off",
12 | "global-require": "off",
13 | "@typescript-eslint/no-require-imports": "off",
14 | "prefer-arrow-callback": "off",
15 | },
16 | });
17 |
--------------------------------------------------------------------------------
/build-scripts/gulp/index.mjs:
--------------------------------------------------------------------------------
1 | import "./app.js";
2 | import "./cast.js";
3 | import "./clean.js";
4 | import "./compress.js";
5 | import "./demo.js";
6 | import "./download-translations.js";
7 | import "./entry-html.js";
8 | import "./fetch-nightly-translations.js";
9 | import "./gallery.js";
10 | import "./gather-static.js";
11 | import "./gen-icons-json.js";
12 | import "./hassio.js";
13 | import "./landing-page.js";
14 | import "./locale-data.js";
15 | import "./rspack.js";
16 | import "./service-worker.js";
17 | import "./translations.js";
18 |
--------------------------------------------------------------------------------
/build-scripts/removedIcons.json:
--------------------------------------------------------------------------------
1 | []
2 |
--------------------------------------------------------------------------------
/cast/public/_redirects:
--------------------------------------------------------------------------------
1 | # These redirects are handled by Netlify
2 | #
3 |
4 | # Some custom cards are not prefixing the instance URL when fetching data
5 | # and can end up fetching the data from the Cast domain instead of HA.
6 | # This will make sure that some common ones are replaced with a placeholder.
7 | /api/camera_proxy/* /images/google-nest-hub.png
8 | /api/camera_proxy_stream/* /images/google-nest-hub.png
9 | /api/media_player_proxy/* /images/google-nest-hub.png
10 |
--------------------------------------------------------------------------------
/cast/public/images/arsaboo.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/arsaboo.jpg
--------------------------------------------------------------------------------
/cast/public/images/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/favicon.ico
--------------------------------------------------------------------------------
/cast/public/images/google-nest-hub.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/google-nest-hub.png
--------------------------------------------------------------------------------
/cast/public/images/ha-cast-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/ha-cast-icon.png
--------------------------------------------------------------------------------
/cast/public/images/melody.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/melody.jpg
--------------------------------------------------------------------------------
/cast/public/images/nabu-loves-hass.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/cast/public/images/nabu-loves-hass.png
--------------------------------------------------------------------------------
/cast/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "background_color": "#FFFFFF",
3 | "description": "Show Home Assistant on your Chromecast or Google Assistant devices with a screen.",
4 | "dir": "ltr",
5 | "display": "standalone",
6 | "icons": [
7 | {
8 | "src": "/images/ha-cast-icon.png",
9 | "sizes": "512x512",
10 | "type": "image/png"
11 | }
12 | ],
13 | "lang": "en-US",
14 | "name": "Home Assistant Cast",
15 | "short_name": "HA Cast",
16 | "start_url": "/?homescreen=1",
17 | "theme_color": "#03A9F4"
18 | }
19 |
--------------------------------------------------------------------------------
/cast/public/sw-legacy.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | self.addEventListener("fetch", (event) => {
4 | event.respondWith(fetch(event.request));
5 | });
6 |
--------------------------------------------------------------------------------
/cast/public/sw-modern.js:
--------------------------------------------------------------------------------
1 | self.addEventListener("fetch", (event) => {
2 | event.respondWith(fetch(event.request));
3 | });
4 |
--------------------------------------------------------------------------------
/cast/script/build_cast:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Build the cast receiver
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp build-cast
10 |
--------------------------------------------------------------------------------
/cast/script/develop_cast:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Develop the cast receiver
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp develop-cast
10 |
--------------------------------------------------------------------------------
/cast/script/upload:
--------------------------------------------------------------------------------
1 | # Run it twice, second time we just delete.
2 | aws s3 sync dist s3://cast.home-assistant.io --acl public-read
3 | # Don't delete as it might break open sites that need to load code splitted things.
4 | # aws s3 sync dist s3://cast.home-assistant.io --acl public-read --delete
5 | # Todo : update JS first, HTML last.
6 |
--------------------------------------------------------------------------------
/cast/src/html/receiver.html.template:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | <% for (const entry of latestEntryJS) { %>
5 |
6 | <% } %>
7 | <%= renderTemplate("../../../src/html/_style_base.html.template") %>
8 |
13 |
14 |
--------------------------------------------------------------------------------
/cast/src/launcher/entrypoint.ts:
--------------------------------------------------------------------------------
1 | import "./layout/hc-connect";
2 |
3 | import("../../../src/resources/append-ha-style");
4 |
--------------------------------------------------------------------------------
/cast/src/receiver/cast_context.ts:
--------------------------------------------------------------------------------
1 | import { framework } from "./cast_framework";
2 |
3 | export const castContext = framework.CastReceiverContext.getInstance();
4 |
--------------------------------------------------------------------------------
/cast/src/receiver/cast_framework.ts:
--------------------------------------------------------------------------------
1 | import type { cast as ReceiverCast } from "chromecast-caf-receiver";
2 |
3 | export const framework = (cast as unknown as typeof ReceiverCast).framework;
4 |
--------------------------------------------------------------------------------
/cast/src/receiver/types.ts:
--------------------------------------------------------------------------------
1 | export interface ReceivedMessage {
2 | gj: boolean;
3 | data: T;
4 | senderId: string;
5 | type: "message";
6 | }
7 |
--------------------------------------------------------------------------------
/demo/public/_headers:
--------------------------------------------------------------------------------
1 | /*
2 | Cache-Control: public, max-age: 0, s-maxage=3600, must-revalidate
3 | Content-Security-Policy: form-action https:
4 | Referrer-Policy: no-referrer-when-downgrade
5 | X-Content-Type-Options: nosniff
6 | X-XSS-Protection: 1; mode=block
7 |
8 | /api/*
9 | Cache-Control: public, max-age: 604800, s-maxage=604800
10 |
11 | /assets/*
12 | Cache-Control: public, max-age: 604800, s-maxage=604800
13 |
14 | /frontend_es5/*
15 | Cache-Control: public, max-age: 604800, s-maxage=604800
16 |
17 | /frontend_latest/*
18 | Cache-Control: public, max-age: 604800, s-maxage=604800
19 |
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/floorplans/ecobee_blank.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/floorplans/ecobee_blank.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/floorplans/main.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/floorplans/main.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/floorplans/second.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/floorplans/second.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/Harmony.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/Harmony.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/abode_disabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/abode_disabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/abode_enabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/abode_enabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/automation_disabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/automation_disabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/automation_enabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/automation_enabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_backyard_recording.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_backyard_recording.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_backyard_streaming.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_backyard_streaming.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_driveway_recording.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_driveway_recording.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_driveway_streaming.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_driveway_streaming.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_patio_recording.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_patio_recording.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_patio_streaming.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_patio_streaming.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_porch_recording.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_porch_recording.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/camera_porch_streaming.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/camera_porch_streaming.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/ecobee_blank.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/ecobee_blank.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/garage_door_closed.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/garage_door_closed.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/garage_door_open.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/garage_door_open.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/light_bulb_off.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/light_bulb_off.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/light_bulb_on.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/light_bulb_on.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/light_off.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/light_off.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/light_on.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/light_on.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/security_armed_red.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/security_armed_red.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/security_disarmed.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/security_disarmed.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/tv_disabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/tv_disabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/tv_enabled.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/tv_enabled.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/tv_off2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/tv_off2.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/icons/tv_on2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/icons/tv_on2.png
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/arsaboo.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/arsaboo.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/camera.backyard.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/camera.backyard.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/camera.driveway.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/camera.driveway.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/camera.patio.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/camera.patio.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/camera.porch.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/camera.porch.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/media_player_family_room.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/media_player_family_room.jpg
--------------------------------------------------------------------------------
/demo/public/assets/arsaboo/images/melody.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/arsaboo/images/melody.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/background-15.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/background-15.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/cardbackK.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/cardbackK.png
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/home/bus_10.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/home/bus_10.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/home/git.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/home/git.png
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/home/house_4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/home/house_4.png
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/home/james_10.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/home/james_10.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/home/tina_4.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/home/tina_4.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/air_8.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/air_8.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/alarm_3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/alarm_3.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/door_3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/door_3.png
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/leak_2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/leak_2.png
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/motion_3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/motion_3.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/smoke_4.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/smoke_4.jpg
--------------------------------------------------------------------------------
/demo/public/assets/jimpower/security/window_2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/jimpower/security/window_2.jpg
--------------------------------------------------------------------------------
/demo/public/assets/kernehed/bella.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/kernehed/bella.jpg
--------------------------------------------------------------------------------
/demo/public/assets/kernehed/camera.entre.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/kernehed/camera.entre.jpg
--------------------------------------------------------------------------------
/demo/public/assets/kernehed/oscar.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/kernehed/oscar.jpg
--------------------------------------------------------------------------------
/demo/public/assets/sections/images/media_player_family_room.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/sections/images/media_player_family_room.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/House_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/House_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/Stefan_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/Stefan_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/background_square.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/background_square.png
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/cleaning_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/cleaning_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/clothes_drying_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/clothes_drying_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/dryer_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/dryer_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/folded_clothes_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/folded_clothes_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/guests_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/guests_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/isa_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/isa_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/laundry_clean_2_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/laundry_clean_2_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/laundry_running_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/laundry_running_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/mailbox_bw_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/mailbox_bw_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/mailbox_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/mailbox_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/meteogram.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/meteogram.png
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/plants.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/plants.png
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/radiator_off.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/radiator_off.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/radiator_on.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/radiator_on.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/roomba_bw_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/roomba_bw_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/roomba_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/roomba_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/trash_bear_bw_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/trash_bear_bw_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/trash_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/trash_square.jpg
--------------------------------------------------------------------------------
/demo/public/assets/teachingbirds/washer_square.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/assets/teachingbirds/washer_square.jpg
--------------------------------------------------------------------------------
/demo/public/stub_config/bedroom.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/stub_config/bedroom.png
--------------------------------------------------------------------------------
/demo/public/stub_config/floorplan.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/stub_config/floorplan.png
--------------------------------------------------------------------------------
/demo/public/stub_config/kitchen.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/stub_config/kitchen.png
--------------------------------------------------------------------------------
/demo/public/stub_config/t-shirt-promo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/demo/public/stub_config/t-shirt-promo.png
--------------------------------------------------------------------------------
/demo/public/sw-legacy.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | self.addEventListener("fetch", (event) => {
4 | event.respondWith(fetch(event.request));
5 | });
6 |
--------------------------------------------------------------------------------
/demo/public/sw-modern.js:
--------------------------------------------------------------------------------
1 | self.addEventListener("fetch", (event) => {
2 | event.respondWith(fetch(event.request));
3 | });
4 |
--------------------------------------------------------------------------------
/demo/script/build_demo:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Build the demo
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp build-demo
10 |
--------------------------------------------------------------------------------
/demo/script/develop_demo:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Develop the demo
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp develop-demo
10 |
--------------------------------------------------------------------------------
/demo/script/size_stats:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Analyze stats
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp analyze-demo
--------------------------------------------------------------------------------
/demo/src/configs/arsaboo/index.ts:
--------------------------------------------------------------------------------
1 | import type { DemoConfig } from "../types";
2 | import { demoEntitiesArsaboo } from "./entities";
3 | import { demoLovelaceArsaboo } from "./lovelace";
4 | import { demoThemeArsaboo } from "./theme";
5 |
6 | export const demoArsaboo: DemoConfig = {
7 | authorName: "Arsaboo",
8 | authorUrl: "https://github.com/arsaboo/homeassistant-config/",
9 | name: "ARS Home",
10 | lovelace: demoLovelaceArsaboo,
11 | entities: demoEntitiesArsaboo,
12 | theme: demoThemeArsaboo,
13 | };
14 |
--------------------------------------------------------------------------------
/demo/src/configs/arsaboo/theme.ts:
--------------------------------------------------------------------------------
1 | export const demoThemeArsaboo = () => ({});
2 |
--------------------------------------------------------------------------------
/demo/src/configs/jimpower/index.ts:
--------------------------------------------------------------------------------
1 | import type { DemoConfig } from "../types";
2 | import { demoEntitiesJimpower } from "./entities";
3 | import { demoLovelaceJimpower } from "./lovelace";
4 | import { demoThemeJimpower } from "./theme";
5 |
6 | export const demoJimpower: DemoConfig = {
7 | authorName: "Jimpower",
8 | authorUrl: "https://github.com/JamesMcCarthy79/Home-Assistant-Config",
9 | name: "Kingia Castle",
10 | lovelace: demoLovelaceJimpower,
11 | entities: demoEntitiesJimpower,
12 | theme: demoThemeJimpower,
13 | };
14 |
--------------------------------------------------------------------------------
/demo/src/configs/kernehed/index.ts:
--------------------------------------------------------------------------------
1 | import type { DemoConfig } from "../types";
2 | import { demoEntitiesKernehed } from "./entities";
3 | import { demoLovelaceKernehed } from "./lovelace";
4 | import { demoThemeKernehed } from "./theme";
5 |
6 | export const demoKernehed: DemoConfig = {
7 | authorName: "Kernehed",
8 | authorUrl: "https://github.com/kernehed",
9 | name: "Hem",
10 | lovelace: demoLovelaceKernehed,
11 | entities: demoEntitiesKernehed,
12 | theme: demoThemeKernehed,
13 | };
14 |
--------------------------------------------------------------------------------
/demo/src/configs/sections/index.ts:
--------------------------------------------------------------------------------
1 | import type { DemoConfig } from "../types";
2 | import { demoEntitiesSections } from "./entities";
3 | import { demoLovelaceSections } from "./lovelace";
4 |
5 | export const demoSections: DemoConfig = {
6 | authorName: "Home Assistant",
7 | authorUrl: "https://github.com/home-assistant/frontend/",
8 | name: "Home Demo",
9 | lovelace: demoLovelaceSections,
10 | entities: demoEntitiesSections,
11 | theme: () => ({}),
12 | };
13 |
--------------------------------------------------------------------------------
/demo/src/configs/teachingbirds/index.ts:
--------------------------------------------------------------------------------
1 | import type { DemoConfig } from "../types";
2 | import { demoEntitiesTeachingbirds } from "./entities";
3 | import { demoLovelaceTeachingbirds } from "./lovelace";
4 | import { demoThemeTeachingbirds } from "./theme";
5 |
6 | export const demoTeachingbirds: DemoConfig = {
7 | authorName: "Isabella Gross Alström",
8 | authorUrl: "https://github.com/isabellaalstrom/",
9 | name: "Isa's mobile friendly LL",
10 | lovelace: demoLovelaceTeachingbirds,
11 | entities: demoEntitiesTeachingbirds,
12 | theme: demoThemeTeachingbirds,
13 | };
14 |
--------------------------------------------------------------------------------
/demo/src/entrypoint.ts:
--------------------------------------------------------------------------------
1 | import "./util/is_frontpage";
2 | import "./ha-demo";
3 |
4 | import("../../src/resources/append-ha-style");
5 |
--------------------------------------------------------------------------------
/demo/src/stubs/area_registry.ts:
--------------------------------------------------------------------------------
1 | import type { AreaRegistryEntry } from "../../../src/data/area_registry";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockAreaRegistry = (
5 | hass: MockHomeAssistant,
6 | data: AreaRegistryEntry[] = []
7 | ) => {
8 | hass.mockWS("config/area_registry/list", () => data);
9 | const areas = {};
10 | data.forEach((area) => {
11 | areas[area.area_id] = area;
12 | });
13 | hass.updateHass({ areas });
14 | };
15 |
--------------------------------------------------------------------------------
/demo/src/stubs/auth.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockAuth = (hass: MockHomeAssistant) => {
4 | hass.mockWS("config/auth/list", () => []);
5 | hass.mockWS("auth/refresh_tokens", () => []);
6 | };
7 |
--------------------------------------------------------------------------------
/demo/src/stubs/config.ts:
--------------------------------------------------------------------------------
1 | import type { validateConfig } from "../../../src/data/config";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockConfig = (hass: MockHomeAssistant) => {
5 | hass.mockWS("validate_config", () => ({
6 | actions: { valid: true, error: null },
7 | conditions: { valid: true, error: null },
8 | triggers: { valid: true, error: null },
9 | }));
10 | };
11 |
--------------------------------------------------------------------------------
/demo/src/stubs/device_registry.ts:
--------------------------------------------------------------------------------
1 | import type { DeviceRegistryEntry } from "../../../src/data/device_registry";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockDeviceRegistry = (
5 | hass: MockHomeAssistant,
6 | data: DeviceRegistryEntry[] = []
7 | ) => {
8 | hass.mockWS("config/device_registry/list", () => data);
9 | const devices = {};
10 | data.forEach((device) => {
11 | devices[device.id] = device;
12 | });
13 | hass.updateHass({ devices });
14 | };
15 |
--------------------------------------------------------------------------------
/demo/src/stubs/entity_registry.ts:
--------------------------------------------------------------------------------
1 | import type { EntityRegistryEntry } from "../../../src/data/entity_registry";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockEntityRegistry = (
5 | hass: MockHomeAssistant,
6 | data: EntityRegistryEntry[] = []
7 | ) => {
8 | hass.mockWS("config/entity_registry/list", () => data);
9 | };
10 |
--------------------------------------------------------------------------------
/demo/src/stubs/events.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockEvents = (hass: MockHomeAssistant) => {
4 | hass.mockAPI("events", () => []);
5 | };
6 |
--------------------------------------------------------------------------------
/demo/src/stubs/floor_registry.ts:
--------------------------------------------------------------------------------
1 | import type { FloorRegistryEntry } from "../../../src/data/floor_registry";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockFloorRegistry = (
5 | hass: MockHomeAssistant,
6 | data: FloorRegistryEntry[] = []
7 | ) => hass.mockWS("config/floor_registry/list", () => data);
8 |
--------------------------------------------------------------------------------
/demo/src/stubs/label_registry.ts:
--------------------------------------------------------------------------------
1 | import type { LabelRegistryEntry } from "../../../src/data/label_registry";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockLabelRegistry = (
5 | hass: MockHomeAssistant,
6 | data: LabelRegistryEntry[] = []
7 | ) => hass.mockWS("config/label_registry/list", () => data);
8 |
--------------------------------------------------------------------------------
/demo/src/stubs/media_player.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockMediaPlayer = (hass: MockHomeAssistant) => {
4 | hass.mockWS("media_player_thumbnail", () => Promise.reject());
5 | };
6 |
--------------------------------------------------------------------------------
/demo/src/stubs/system_log.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockSystemLog = (hass: MockHomeAssistant) => {
4 | hass.mockAPI("error/all", () => []);
5 | };
6 |
--------------------------------------------------------------------------------
/demo/src/stubs/tags.ts:
--------------------------------------------------------------------------------
1 | import type { Tag } from "../../../src/data/tag";
2 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
3 |
4 | export const mockTags = (hass: MockHomeAssistant) => {
5 | hass.mockWS("tag/list", () => [{ id: "my-tag", name: "My Tag" }] as Tag[]);
6 | };
7 |
--------------------------------------------------------------------------------
/demo/src/stubs/template.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockTemplate = (hass: MockHomeAssistant) => {
4 | hass.mockAPI("template", () =>
5 | Promise.reject({
6 | body: { message: "Template dev tool does not work in the demo." },
7 | })
8 | );
9 | hass.mockWS("render_template", (msg, _hass, onChange) => {
10 | onChange!({
11 | result: msg.template,
12 | listeners: { all: false, domains: [], entities: [], time: false },
13 | });
14 | // eslint-disable-next-line @typescript-eslint/no-empty-function
15 | return () => {};
16 | });
17 | };
18 |
--------------------------------------------------------------------------------
/demo/src/stubs/translations.ts:
--------------------------------------------------------------------------------
1 | import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
2 |
3 | export const mockTranslations = (hass: MockHomeAssistant) => {
4 | hass.mockWS(
5 | "frontend/get_translations",
6 | (/* msg: {language: string, category: string} */) => ({ resources: {} })
7 | );
8 | };
9 |
--------------------------------------------------------------------------------
/demo/src/util/is_frontpage.ts:
--------------------------------------------------------------------------------
1 | export const isFrontpageEmbed = document.location.search === "?frontpage";
2 |
--------------------------------------------------------------------------------
/docs/screenshot.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/docs/screenshot.png
--------------------------------------------------------------------------------
/gallery/eslint.config.mjs:
--------------------------------------------------------------------------------
1 | // @ts-check
2 |
3 | import tseslint from "typescript-eslint";
4 | import rootConfig from "../eslint.config.mjs";
5 |
6 | export default tseslint.config(...rootConfig, {
7 | rules: {
8 | "no-console": "off",
9 | },
10 | });
11 |
--------------------------------------------------------------------------------
/gallery/public/api/hassio/addons/core_zwave_js/icon:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/api/hassio/addons/core_zwave_js/icon
--------------------------------------------------------------------------------
/gallery/public/api/media_player_proxy/media_player.bedroom:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/api/media_player_proxy/media_player.bedroom
--------------------------------------------------------------------------------
/gallery/public/api/media_player_proxy/media_player.living_room:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/api/media_player_proxy/media_player.living_room
--------------------------------------------------------------------------------
/gallery/public/api/media_player_proxy/media_player.walkman:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/api/media_player_proxy/media_player.walkman
--------------------------------------------------------------------------------
/gallery/public/images/album_cover.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/album_cover.jpg
--------------------------------------------------------------------------------
/gallery/public/images/album_cover_2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/album_cover_2.jpg
--------------------------------------------------------------------------------
/gallery/public/images/bed.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/bed.png
--------------------------------------------------------------------------------
/gallery/public/images/brand/README.md:
--------------------------------------------------------------------------------
1 | # Note!
2 |
3 | Note, the assets in this folder, are not part of the CC license this repository is shipped in.
4 | All rights reserved.
5 |
--------------------------------------------------------------------------------
/gallery/public/images/brand/logo-exclusion-zone.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/brand/logo-exclusion-zone.png
--------------------------------------------------------------------------------
/gallery/public/images/brand/logo-layout-variants.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/brand/logo-layout-variants.png
--------------------------------------------------------------------------------
/gallery/public/images/brand/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/brand/logo.png
--------------------------------------------------------------------------------
/gallery/public/images/clearspace.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/clearspace.png
--------------------------------------------------------------------------------
/gallery/public/images/divider.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/divider.png
--------------------------------------------------------------------------------
/gallery/public/images/floorplan.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/floorplan.png
--------------------------------------------------------------------------------
/gallery/public/images/frenck.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/frenck.jpg
--------------------------------------------------------------------------------
/gallery/public/images/kitchen.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/kitchen.png
--------------------------------------------------------------------------------
/gallery/public/images/light_bulb_off.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/light_bulb_off.png
--------------------------------------------------------------------------------
/gallery/public/images/light_bulb_on.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/light_bulb_on.png
--------------------------------------------------------------------------------
/gallery/public/images/living_room.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/living_room.png
--------------------------------------------------------------------------------
/gallery/public/images/logo-variants.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/logo-variants.png
--------------------------------------------------------------------------------
/gallery/public/images/logo-with-text.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/logo-with-text.png
--------------------------------------------------------------------------------
/gallery/public/images/netflix.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/netflix.jpg
--------------------------------------------------------------------------------
/gallery/public/images/office.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/office.jpg
--------------------------------------------------------------------------------
/gallery/public/images/paulus.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/paulus.jpg
--------------------------------------------------------------------------------
/gallery/public/images/sunflowers.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/gallery/public/images/sunflowers.jpg
--------------------------------------------------------------------------------
/gallery/script/build_gallery:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the gallery
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp build-gallery
10 |
--------------------------------------------------------------------------------
/gallery/script/develop_gallery:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the gallery
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp develop-gallery
10 |
--------------------------------------------------------------------------------
/gallery/src/data/date-options.ts:
--------------------------------------------------------------------------------
1 | import type { ControlSelectOption } from "../../../src/components/ha-control-select";
2 |
3 | export const timeOptions: ControlSelectOption[] = [
4 | {
5 | value: "now",
6 | label: "Now",
7 | },
8 | {
9 | value: "00:15:30",
10 | label: "12:15:30 AM",
11 | },
12 | {
13 | value: "06:15:30",
14 | label: "06:15:30 AM",
15 | },
16 | {
17 | value: "12:15:30",
18 | label: "12:15:30 PM",
19 | },
20 | {
21 | value: "18:15:30",
22 | label: "06:15:30 PM",
23 | },
24 | ];
25 |
--------------------------------------------------------------------------------
/gallery/src/data/traces/types.ts:
--------------------------------------------------------------------------------
1 | import type { AutomationTraceExtended } from "../../../../src/data/trace";
2 | import type { LogbookEntry } from "../../../../src/data/logbook";
3 |
4 | export interface DemoTrace {
5 | trace: AutomationTraceExtended;
6 | logbookEntries: LogbookEntry[];
7 | }
8 |
--------------------------------------------------------------------------------
/gallery/src/entrypoint.js:
--------------------------------------------------------------------------------
1 | import "./ha-gallery";
2 |
3 | import("../../src/resources/append-ha-style");
4 |
5 | document.body.appendChild(document.createElement("ha-gallery"));
6 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/describe-action.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Describe Action
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/describe-condition.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Describe Condition
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/describe-trigger.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Describe Trigger
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/editor-action.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Actions
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/editor-condition.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Conditions
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/editor-trigger.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Triggers
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/trace-timeline.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Trace Timelines
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/automation/trace.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Trace Graphs
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-bar.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Bar
3 | subtitle: Can be used to communicate progress of a task.
4 | ---
5 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-chips.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Chip
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-button.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Button
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-circular-slider.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Circular Slider
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-number-buttons.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Number Buttons
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-select-menu.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Select Menu
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-select.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Select
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-slider.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Slider
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-control-switch.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Control Switch
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-expansion-panel.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Expansion Panel
3 | ---
4 |
5 | Expansion panel following all the ARIA guidelines.
6 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-faded.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Faded Content
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-form.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Forms
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-gauge.ts:
--------------------------------------------------------------------------------
1 | import "../../../../src/components/ha-gauge";
2 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-hs-color-picker.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: HS Color Picker
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-label-badge.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Label Badge
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-select-box.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Select box
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-selector.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Selector
3 | ---
4 |
5 | See the website for [list of available selectors](https://www.home-assistant.io/docs/blueprint/selectors/).
6 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-spinner.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Spinner
3 | subtitle: Can be used to indicate an ongoing task.
4 | ---
5 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-switch.ts:
--------------------------------------------------------------------------------
1 | import "../../../../src/components/ha-switch";
2 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-tip.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Tip
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/components/ha-tooltip.ts:
--------------------------------------------------------------------------------
1 | import "../../../../src/components/ha-tooltip";
2 | import "../../../../src/components/ha-button";
3 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date-time-numeric.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date-Time Format (Numeric)
3 | ---
4 |
5 | This pages lists all supported languages with their available date-time formats.
6 |
7 | Formatting function: `const formatDateTimeNumeric: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date-time-seconds.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date-Time Format (Seconds)
3 | ---
4 |
5 | This pages lists all supported languages with their available date-time formats.
6 |
7 | Formatting function: `const formatDateTimeWithSeconds: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date-time-short-year.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date-Time Format (Short w/ Year)
3 | ---
4 |
5 | This pages lists all supported languages with their available date-time formats.
6 |
7 | Formatting function: `const formatShortDateTimeWithYear: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date-time-short.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date-Time Format (Short)
3 | ---
4 |
5 | This pages lists all supported languages with their available date-time formats.
6 |
7 | Formatting function: `const formatShortDateTime: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date-time.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date-Time Format
3 | ---
4 |
5 | This pages lists all supported languages with their available date-time formats.
6 |
7 | Formatting function: `const formatDateTime: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/date.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Date Format (Numeric)
3 | ---
4 |
5 | This pages lists all supported languages with their available (numeric) date formats.
6 |
7 | Formatting function: `const formatDateNumeric: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/time-seconds.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Time Format (Seconds)
3 | ---
4 |
5 | This pages lists all supported languages with their available time formats.
6 |
7 | Formatting function: `const formatTimeWithSeconds: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/time-weekday.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Time Format (Weekday)
3 | ---
4 |
5 | This pages lists all supported languages with their available time formats.
6 |
7 | Formatting function: `const formatTimeWeekday: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/date-time/time.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Time Format
3 | ---
4 |
5 | This pages lists all supported languages with their available time formats.
6 |
7 | Formatting function: `const formatTime: (dateObj: Date, locale: FrontendLocaleData) => string`
8 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/alarm-panel-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Alarm Panel Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/area-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Area Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/conditional-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Conditional Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/entities-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Entities Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/entity-button-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Entity Button Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/entity-filter-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Entity Filter Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/gauge-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Gauge Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/glance-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Glance Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/grid-and-stack-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Grid and Stack Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/iframe-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Website Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/introduction.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Introduction
3 | ---
4 |
5 | Dashboards have many different cards. Each card allows the user to tell
6 | a different story about what is going on in their house. These cards
7 | are very customizable, as no household is the same.
8 |
9 | This gallery helps our developers and designers to see all the
10 | different states that each card can be in.
11 |
12 | Check [the Dashboards documentation](https://www.home-assistant.io/dashboards/) for instructions on how to get started with Dashboards.
13 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/light-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Light Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/map-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Map Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/markdown-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Markdown Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/media-control-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Media Control Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/media-player-row.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Media Player Row
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/picture-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Picture Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/picture-elements-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Picture Elements Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/picture-entity-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Picture Entity Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/picture-glance-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Picture Glance Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/plant-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Plant Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/thermostat-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Thermostat Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/tile-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Tile Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/lovelace/todo-list-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Todo List Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/misc/entity-state.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Entity State
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/misc/ha-markdown.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Markdown
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/misc/integration-card.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Integration Card
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/misc/util-long-press.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Long Press Utility
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/climate.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Climate
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/cover.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Cover
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/humidifier.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Humidifier
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/input-number.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Input Number
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/input-text.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Input Text
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/light.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Light
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/lock.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Lock
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/media-player.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Media Player
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/number.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Number
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/scene.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Scene
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/timer.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Timer
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/update.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Update
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/vacuum.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Vacuum
3 | ---
4 |
--------------------------------------------------------------------------------
/gallery/src/pages/more-info/water-heater.markdown:
--------------------------------------------------------------------------------
1 | ---
2 | title: Water Heater
3 | ---
4 |
--------------------------------------------------------------------------------
/gulpfile.js:
--------------------------------------------------------------------------------
1 | import { availableParallelism } from "node:os";
2 | import "./build-scripts/gulp/index.mjs";
3 |
4 | process.env.UV_THREADPOOL_SIZE = availableParallelism();
5 |
--------------------------------------------------------------------------------
/hassio/config.cjs:
--------------------------------------------------------------------------------
1 | const path = require("path");
2 |
3 | module.exports = {
4 | // Target directory for the build.
5 | buildDir: path.resolve(__dirname, "build"),
6 | nodeDir: path.resolve(__dirname, "../node_modules"),
7 | // Path where the Hass.io frontend will be publicly available.
8 | publicPath: "/api/hassio/app",
9 | };
10 |
--------------------------------------------------------------------------------
/hassio/script/build_hassio:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Builds the Hass.io app for production
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp build-hassio
10 |
--------------------------------------------------------------------------------
/hassio/script/develop:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the Hass.io development server
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp develop-hassio
10 |
--------------------------------------------------------------------------------
/hassio/src/components/hassio-filter-addons.ts:
--------------------------------------------------------------------------------
1 | import type { IFuseOptions } from "fuse.js";
2 | import Fuse from "fuse.js";
3 | import type { StoreAddon } from "../../../src/data/supervisor/store";
4 |
5 | export function filterAndSort(addons: StoreAddon[], filter: string) {
6 | const options: IFuseOptions = {
7 | keys: ["name", "description", "slug"],
8 | isCaseSensitive: false,
9 | minMatchCharLength: Math.min(filter.length, 2),
10 | threshold: 0.2,
11 | ignoreDiacritics: true,
12 | };
13 | const fuse = new Fuse(addons, options);
14 | return fuse.search(filter).map((result) => result.item);
15 | }
16 |
--------------------------------------------------------------------------------
/hassio/src/dialogs/datadisk/show-dialog-hassio-datadisk.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../src/common/dom/fire_event";
2 | import type { Supervisor } from "../../../../src/data/supervisor/supervisor";
3 |
4 | export interface HassioDatatiskDialogParams {
5 | supervisor: Supervisor;
6 | }
7 |
8 | export const showHassioDatadiskDialog = (
9 | element: HTMLElement,
10 | dialogParams: HassioDatatiskDialogParams
11 | ): void => {
12 | fireEvent(element, "show-dialog", {
13 | dialogTag: "dialog-hassio-datadisk",
14 | dialogImport: () => import("./dialog-hassio-datadisk"),
15 | dialogParams,
16 | });
17 | };
18 |
--------------------------------------------------------------------------------
/hassio/src/dialogs/markdown/show-dialog-hassio-markdown.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../src/common/dom/fire_event";
2 |
3 | export interface HassioMarkdownDialogParams {
4 | title: string;
5 | content: string;
6 | }
7 |
8 | export const showHassioMarkdownDialog = (
9 | element: HTMLElement,
10 | dialogParams: HassioMarkdownDialogParams
11 | ): void => {
12 | fireEvent(element, "show-dialog", {
13 | dialogTag: "dialog-hassio-markdown",
14 | dialogImport: () => import("./dialog-hassio-markdown"),
15 | dialogParams,
16 | });
17 | };
18 |
--------------------------------------------------------------------------------
/landing-page/eslintrc.config.mjs:
--------------------------------------------------------------------------------
1 | import rootConfig from "../eslint.config.mjs";
2 |
3 | export default [
4 | ...rootConfig,
5 | {
6 | rules: {},
7 | },
8 | ];
9 |
--------------------------------------------------------------------------------
/landing-page/public/static/icons/favicon-192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/landing-page/public/static/icons/favicon-192x192.png
--------------------------------------------------------------------------------
/landing-page/public/static/icons/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/landing-page/public/static/icons/favicon.ico
--------------------------------------------------------------------------------
/landing-page/public/static/images/logo_discord.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/landing-page/public/static/images/logo_discord.png
--------------------------------------------------------------------------------
/landing-page/public/static/images/logo_x.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/landing-page/script/build_landing_page:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the landing-page
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp build-landing-page
10 |
--------------------------------------------------------------------------------
/landing-page/script/develop:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the landing-page
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/../.."
8 |
9 | ./node_modules/.bin/gulp develop-landing-page
10 |
--------------------------------------------------------------------------------
/landing-page/src/data/observer.ts:
--------------------------------------------------------------------------------
1 | export async function getObserverLogs() {
2 | return fetch("/observer/logs");
3 | }
4 |
5 | export const downloadUrl = "/observer/logs";
6 |
--------------------------------------------------------------------------------
/landing-page/src/entrypoint.js:
--------------------------------------------------------------------------------
1 | import "./ha-landing-page";
2 |
3 | import("../../src/resources/append-ha-style");
4 |
--------------------------------------------------------------------------------
/netlify.toml:
--------------------------------------------------------------------------------
1 | [build.environment]
2 | YARN_VERSION = "1.22.11"
3 | NODE_OPTIONS = "--max_old_space_size=6144"
4 |
--------------------------------------------------------------------------------
/prettier.config.js:
--------------------------------------------------------------------------------
1 | export default {
2 | trailingComma: "es5",
3 | };
4 |
--------------------------------------------------------------------------------
/public/__init__.py:
--------------------------------------------------------------------------------
1 | """Frontend for Home Assistant."""
2 | from pathlib import Path
3 |
4 |
5 | def where() -> Path:
6 | """Return path to the frontend."""
7 | return Path(__file__).parent
8 |
--------------------------------------------------------------------------------
/public/py.typed:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/py.typed
--------------------------------------------------------------------------------
/public/robots.txt:
--------------------------------------------------------------------------------
1 | User-agent: *
2 | Disallow: /
--------------------------------------------------------------------------------
/public/static/icons/browserconfig.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | #18bcf2
10 |
11 |
12 |
13 |
--------------------------------------------------------------------------------
/public/static/icons/favicon-1024x1024.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-1024x1024.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-16x16.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-16x16.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-192x192.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-32x32.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-32x32.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-384x384.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-384x384.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-512x512.png
--------------------------------------------------------------------------------
/public/static/icons/favicon-apple-180x180.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon-apple-180x180.png
--------------------------------------------------------------------------------
/public/static/icons/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/favicon.ico
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-128x128.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-128x128.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-192x192.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-384x384.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-384x384.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-48x48.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-48x48.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-512x512.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-72x72.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-72x72.png
--------------------------------------------------------------------------------
/public/static/icons/maskable_icon-96x96.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/maskable_icon-96x96.png
--------------------------------------------------------------------------------
/public/static/icons/tile-win-150x150.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/tile-win-150x150.png
--------------------------------------------------------------------------------
/public/static/icons/tile-win-310x150.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/tile-win-310x150.png
--------------------------------------------------------------------------------
/public/static/icons/tile-win-310x310.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/tile-win-310x310.png
--------------------------------------------------------------------------------
/public/static/icons/tile-win-70x70.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/icons/tile-win-70x70.png
--------------------------------------------------------------------------------
/public/static/images/card_media_player_bg.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/card_media_player_bg.png
--------------------------------------------------------------------------------
/public/static/images/color_wheel.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/color_wheel.png
--------------------------------------------------------------------------------
/public/static/images/config_ecobee_thermostat.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_ecobee_thermostat.png
--------------------------------------------------------------------------------
/public/static/images/config_fitbit_app.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_fitbit_app.png
--------------------------------------------------------------------------------
/public/static/images/config_flows/config_homematicip_cloud.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_flows/config_homematicip_cloud.png
--------------------------------------------------------------------------------
/public/static/images/config_freebox.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_freebox.png
--------------------------------------------------------------------------------
/public/static/images/config_icloud.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_icloud.png
--------------------------------------------------------------------------------
/public/static/images/config_insteon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_insteon.png
--------------------------------------------------------------------------------
/public/static/images/config_philips_hue.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_philips_hue.jpg
--------------------------------------------------------------------------------
/public/static/images/config_webos.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_webos.png
--------------------------------------------------------------------------------
/public/static/images/config_wink.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/config_wink.png
--------------------------------------------------------------------------------
/public/static/images/darksky/weather-cloudy.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/public/static/images/form/tile_features_position_bottom.svg:
--------------------------------------------------------------------------------
1 |
7 |
--------------------------------------------------------------------------------
/public/static/images/form/tile_features_position_bottom_dark.svg:
--------------------------------------------------------------------------------
1 |
7 |
--------------------------------------------------------------------------------
/public/static/images/form/tile_features_position_inline.svg:
--------------------------------------------------------------------------------
1 |
7 |
--------------------------------------------------------------------------------
/public/static/images/form/tile_features_position_inline_dark.svg:
--------------------------------------------------------------------------------
1 |
7 |
--------------------------------------------------------------------------------
/public/static/images/image-broken.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/public/static/images/logo_apple_home.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_apple_home.png
--------------------------------------------------------------------------------
/public/static/images/logo_automatic.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_automatic.png
--------------------------------------------------------------------------------
/public/static/images/logo_axis.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_axis.png
--------------------------------------------------------------------------------
/public/static/images/logo_deconz.jpeg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_deconz.jpeg
--------------------------------------------------------------------------------
/public/static/images/logo_discord.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_discord.png
--------------------------------------------------------------------------------
/public/static/images/logo_google_home.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_google_home.png
--------------------------------------------------------------------------------
/public/static/images/logo_nabu_casa.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_nabu_casa.png
--------------------------------------------------------------------------------
/public/static/images/logo_nabu_casa_dark.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_nabu_casa_dark.png
--------------------------------------------------------------------------------
/public/static/images/logo_philips_hue.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_philips_hue.png
--------------------------------------------------------------------------------
/public/static/images/logo_plex_mediaserver.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/logo_plex_mediaserver.png
--------------------------------------------------------------------------------
/public/static/images/logo_x.svg:
--------------------------------------------------------------------------------
1 |
4 |
--------------------------------------------------------------------------------
/public/static/images/notification-badge.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/notification-badge.png
--------------------------------------------------------------------------------
/public/static/images/screenshots/screenshot-1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/screenshots/screenshot-1.png
--------------------------------------------------------------------------------
/public/static/images/smart-tv.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/smart-tv.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/area.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/area.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/change-wake-word.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/change-wake-word.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/error.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/error.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/great-job.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/great-job.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/heart.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/heart.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/hi.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/hi.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/ok-nabu.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/ok-nabu.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/sleep.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/sleep.png
--------------------------------------------------------------------------------
/public/static/images/voice-assistant/update.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/home-assistant/frontend/02bb7086e7206f053d94953935afde678ba378a0/public/static/images/voice-assistant/update.png
--------------------------------------------------------------------------------
/script/build_frontend:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Builds the frontend for production
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | ./node_modules/.bin/gulp build-app
10 |
--------------------------------------------------------------------------------
/script/develop:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Run the frontend development server
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | ./node_modules/.bin/gulp develop-app
10 |
--------------------------------------------------------------------------------
/script/release:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Pushes a new version to PyPi.
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | # Install node modules
10 | yarn install
11 |
12 | script/build_frontend
13 |
14 | rm -rf dist home_assistant_frontend.egg-info
15 | python3 -m build
16 | python3 -m twine upload dist/*.whl --skip-existing
17 |
--------------------------------------------------------------------------------
/script/serve-config.json:
--------------------------------------------------------------------------------
1 | {
2 | "cleanUrls": false
3 | }
4 |
--------------------------------------------------------------------------------
/script/setup:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Resolve all frontend dependencies that the application requires to develop.
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | script/bootstrap
10 |
11 |
--------------------------------------------------------------------------------
/script/setup_translations:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Setup translation fetching during development
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | ./node_modules/.bin/gulp setup-and-fetch-nightly-translations
--------------------------------------------------------------------------------
/script/size_stats:
--------------------------------------------------------------------------------
1 | #!/bin/sh
2 | # Analyze stats
3 |
4 | # Stop on errors
5 | set -e
6 |
7 | cd "$(dirname "$0")/.."
8 |
9 | ./node_modules/.bin/gulp analyze-app
10 |
--------------------------------------------------------------------------------
/script/translations_download:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env bash
2 |
3 | # Safe bash settings
4 | # -e Exit on command fail
5 | # -u Exit on unset variable
6 | # -o pipefail Exit if piped command has error code
7 | set -eu -o pipefail
8 |
9 | cd "$(dirname "$0")/.."
10 |
11 | ./node_modules/.bin/gulp download-translations
--------------------------------------------------------------------------------
/src/cast/const.ts:
--------------------------------------------------------------------------------
1 | import { CAST_DEV_APP_ID } from "./dev_const";
2 |
3 | // Guard dev mode with `__dev__` so it can only ever be enabled in dev mode.
4 | export const CAST_DEV = __DEV__ && true;
5 |
6 | export const CAST_APP_ID = CAST_DEV ? CAST_DEV_APP_ID : "A078F6B0";
7 | export const CAST_NS = "urn:x-cast:com.nabucasa.hast";
8 |
--------------------------------------------------------------------------------
/src/cast/dev_const.ts:
--------------------------------------------------------------------------------
1 | // Replace this with your own unpublished cast app that points at your local dev
2 | export const CAST_DEV_APP_ID = "5FE44367";
3 |
4 | // Chromecast SDK will only load on localhost and HTTPS
5 | // So during local development we have to send our dev IP address,
6 | // but then run the UI on localhost.
7 | export const CAST_DEV_HASS_URL = "http://192.168.1.234:8123";
8 |
--------------------------------------------------------------------------------
/src/cast/types.ts:
--------------------------------------------------------------------------------
1 | export interface BaseCastMessage {
2 | type: string;
3 | senderId?: string;
4 | }
5 |
--------------------------------------------------------------------------------
/src/common/array/combinations.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Get all possible combinations of an array
3 | * @param arr - The array to get combinations of
4 | * @returns A multidimensional array of all possible combinations
5 | */
6 | export function getAllCombinations(arr: T[]) {
7 | return arr.reduce(
8 | (combinations, element) =>
9 | combinations.concat(
10 | combinations.map((combination) => [...combination, element])
11 | ),
12 | [[]]
13 | );
14 | }
15 |
--------------------------------------------------------------------------------
/src/common/array/literal-includes.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Creates a type predicate function for determining if an array literal includes a given value
3 | * @param array - The array to check
4 | * @returns A type predicate function
5 | */
6 | export const arrayLiteralIncludes =
7 | (array: T) =>
8 | (searchElement: unknown, fromIndex?: number): searchElement is T[number] =>
9 | array.includes(searchElement as T[number], fromIndex);
10 |
--------------------------------------------------------------------------------
/src/common/color/lab.ts:
--------------------------------------------------------------------------------
1 | // From https://github.com/gka/chroma.js
2 | // Copyright (c) 2011-2019, Gregor Aisch
3 |
4 | export type LabColor = [number, number, number];
5 |
6 | export const labDarken = (lab: LabColor, amount = 1): LabColor => [
7 | lab[0] - 18 * amount,
8 | lab[1],
9 | lab[2],
10 | ];
11 |
12 | export const labBrighten = (lab: LabColor, amount = 1): LabColor =>
13 | labDarken(lab, -amount);
14 |
--------------------------------------------------------------------------------
/src/common/config/components_with_service.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | /** Return an array of domains with the service. */
4 | export const componentsWithService = (
5 | hass: HomeAssistant,
6 | service: string
7 | ): string[] =>
8 | hass &&
9 | Object.keys(hass.services).filter((key) => service in hass.services[key]);
10 |
--------------------------------------------------------------------------------
/src/common/config/is_component_loaded.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | /** Return if a component is loaded. */
4 | export const isComponentLoaded = (
5 | hass: HomeAssistant,
6 | component: string
7 | ): boolean => hass && hass.config.components.includes(component);
8 |
--------------------------------------------------------------------------------
/src/common/config/is_pwa.ts:
--------------------------------------------------------------------------------
1 | /** Return if the displaymode is in standalone mode (PWA). */
2 | export default function isPwa(): boolean {
3 | return window.matchMedia("(display-mode: standalone)").matches;
4 | }
5 |
--------------------------------------------------------------------------------
/src/common/config/is_service_loaded.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | /** Return if a service is loaded. */
4 | export const isServiceLoaded = (
5 | hass: HomeAssistant,
6 | domain: string,
7 | service: string
8 | ): boolean =>
9 | hass && domain in hass.services && service in hass.services[domain];
10 |
--------------------------------------------------------------------------------
/src/common/config/location_name.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | /** Get the location name from a hass object. */
4 | export default function computeLocationName(hass: HomeAssistant): string {
5 | return hass && hass.config.location_name;
6 | }
7 |
--------------------------------------------------------------------------------
/src/common/datetime/check_valid_date.ts:
--------------------------------------------------------------------------------
1 | export default function checkValidDate(date?: Date): boolean {
2 | if (!date) {
3 | return false;
4 | }
5 |
6 | return date instanceof Date && !isNaN(date.valueOf());
7 | }
8 |
--------------------------------------------------------------------------------
/src/common/datetime/duration_to_seconds.ts:
--------------------------------------------------------------------------------
1 | export default function durationToSeconds(duration: string): number {
2 | const parts = duration.split(":").map(Number);
3 | return parts[0] * 3600 + parts[1] * 60 + parts[2];
4 | }
5 |
--------------------------------------------------------------------------------
/src/common/datetime/resolve-time-zone.ts:
--------------------------------------------------------------------------------
1 | import { TimeZone } from "../../data/translation";
2 |
3 | const RESOLVED_TIME_ZONE = Intl.DateTimeFormat?.().resolvedOptions?.().timeZone;
4 |
5 | // Browser time zone can be determined from Intl, with fallback to UTC for polyfill or no support.
6 | export const LOCAL_TIME_ZONE = RESOLVED_TIME_ZONE ?? "UTC";
7 |
8 | // Pick time zone based on user profile option. Core zone is used when local cannot be determined.
9 | export const resolveTimeZone = (option: TimeZone, serverTimeZone: string) =>
10 | option === TimeZone.local && RESOLVED_TIME_ZONE
11 | ? LOCAL_TIME_ZONE
12 | : serverTimeZone;
13 |
--------------------------------------------------------------------------------
/src/common/datetime/seconds_to_duration.ts:
--------------------------------------------------------------------------------
1 | const leftPad = (num: number) => (num < 10 ? `0${num}` : num);
2 |
3 | export default function secondsToDuration(d: number) {
4 | const h = Math.floor(d / 3600);
5 | const m = Math.floor((d % 3600) / 60);
6 | const s = Math.floor((d % 3600) % 60);
7 |
8 | if (h > 0) {
9 | return `${h}:${leftPad(m)}:${leftPad(s)}`;
10 | }
11 | if (m > 0) {
12 | return `${m}:${leftPad(s)}`;
13 | }
14 | if (s > 0) {
15 | return "" + s;
16 | }
17 | return null;
18 | }
19 |
--------------------------------------------------------------------------------
/src/common/dom/deep-active-element.ts:
--------------------------------------------------------------------------------
1 | export const deepActiveElement = (
2 | root: DocumentOrShadowRoot = document
3 | ): Element | null => {
4 | if (root.activeElement?.shadowRoot?.activeElement) {
5 | return deepActiveElement(root.activeElement.shadowRoot);
6 | }
7 | return root.activeElement;
8 | };
9 |
--------------------------------------------------------------------------------
/src/common/dom/get_main_window.ts:
--------------------------------------------------------------------------------
1 | import { MAIN_WINDOW_NAME } from "../../data/main_window";
2 |
3 | export const mainWindow = (() => {
4 | try {
5 | return window.name === MAIN_WINDOW_NAME
6 | ? window
7 | : parent.name === MAIN_WINDOW_NAME
8 | ? parent
9 | : top!;
10 | } catch {
11 | return window;
12 | }
13 | })();
14 |
--------------------------------------------------------------------------------
/src/common/dom/prevent_default.ts:
--------------------------------------------------------------------------------
1 | export const preventDefault = (ev) => ev.preventDefault();
2 |
--------------------------------------------------------------------------------
/src/common/dom/stop_propagation.ts:
--------------------------------------------------------------------------------
1 | export const stopPropagation = (ev) => ev.stopPropagation();
2 |
--------------------------------------------------------------------------------
/src/common/dom/toggle_attribute.ts:
--------------------------------------------------------------------------------
1 | // Toggle Attribute Polyfill because it's too new for some browsers
2 | export const toggleAttribute = (
3 | el: HTMLElement,
4 | name: string,
5 | force?: boolean
6 | ) => {
7 | if (force !== undefined) {
8 | force = !!force;
9 | }
10 |
11 | if (el.hasAttribute(name)) {
12 | if (force) {
13 | return true;
14 | }
15 |
16 | el.removeAttribute(name);
17 | return false;
18 | }
19 | if (force === false) {
20 | return false;
21 | }
22 |
23 | el.setAttribute(name, "");
24 | return true;
25 | };
26 |
--------------------------------------------------------------------------------
/src/common/empty_image_base64.ts:
--------------------------------------------------------------------------------
1 | /** An empty image which can be set as src of an img element. */
2 | export const emptyImageBase64 =
3 | "";
4 |
--------------------------------------------------------------------------------
/src/common/entity/attribute_class_names.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 |
3 | export const attributeClassNames = (
4 | stateObj: HassEntity,
5 | attributes: string[]
6 | ): string => {
7 | if (!stateObj) {
8 | return "";
9 | }
10 | return attributes
11 | .map((attribute) =>
12 | attribute in stateObj.attributes ? "has-" + attribute : ""
13 | )
14 | .filter((attr) => attr !== "")
15 | .join(" ");
16 | };
17 |
--------------------------------------------------------------------------------
/src/common/entity/can_toggle_domain.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | export const canToggleDomain = (hass: HomeAssistant, domain: string) => {
4 | const services = hass.services[domain];
5 | if (!services) {
6 | return false;
7 | }
8 |
9 | if (domain === "lock") {
10 | return "lock" in services;
11 | }
12 | if (domain === "cover") {
13 | return "open_cover" in services;
14 | }
15 | return "turn_on" in services;
16 | };
17 |
--------------------------------------------------------------------------------
/src/common/entity/color/battery_color.ts:
--------------------------------------------------------------------------------
1 | export const batteryStateColorProperty = (
2 | state: string
3 | ): string | undefined => {
4 | const value = Number(state);
5 | if (isNaN(value)) {
6 | return undefined;
7 | }
8 | if (value >= 70) {
9 | return "--state-sensor-battery-high-color";
10 | }
11 | if (value >= 30) {
12 | return "--state-sensor-battery-medium-color";
13 | }
14 | return "--state-sensor-battery-low-color";
15 | };
16 |
--------------------------------------------------------------------------------
/src/common/entity/compute_area_name.ts:
--------------------------------------------------------------------------------
1 | import type { AreaRegistryEntry } from "../../data/area_registry";
2 |
3 | export const computeAreaName = (area: AreaRegistryEntry): string | undefined =>
4 | area.name?.trim();
5 |
--------------------------------------------------------------------------------
/src/common/entity/compute_domain.ts:
--------------------------------------------------------------------------------
1 | export const computeDomain = (entityId: string): string =>
2 | entityId.substring(0, entityId.indexOf("."));
3 |
--------------------------------------------------------------------------------
/src/common/entity/compute_floor_name.ts:
--------------------------------------------------------------------------------
1 | import type { FloorRegistryEntry } from "../../data/floor_registry";
2 |
3 | export const computeFloorName = (floor: FloorRegistryEntry): string =>
4 | floor.name?.trim();
5 |
--------------------------------------------------------------------------------
/src/common/entity/compute_object_id.ts:
--------------------------------------------------------------------------------
1 | /** Compute the object ID of a state. */
2 | export const computeObjectId = (entityId: string): string =>
3 | entityId.substr(entityId.indexOf(".") + 1);
4 |
--------------------------------------------------------------------------------
/src/common/entity/compute_state_domain.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 | import { computeDomain } from "./compute_domain";
3 |
4 | export const computeStateDomain = (stateObj: HassEntity) =>
5 | computeDomain(stateObj.entity_id);
6 |
--------------------------------------------------------------------------------
/src/common/entity/compute_state_name.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 | import { computeObjectId } from "./compute_object_id";
3 |
4 | export const computeStateNameFromEntityAttributes = (
5 | entityId: string,
6 | attributes: Record
7 | ): string =>
8 | attributes.friendly_name === undefined
9 | ? computeObjectId(entityId).replace(/_/g, " ")
10 | : (attributes.friendly_name ?? "").toString();
11 |
12 | export const computeStateName = (stateObj: HassEntity): string =>
13 | computeStateNameFromEntityAttributes(stateObj.entity_id, stateObj.attributes);
14 |
--------------------------------------------------------------------------------
/src/common/entity/get_group_entities.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntities } from "home-assistant-js-websocket";
2 | import type { GroupEntity } from "../../data/group";
3 |
4 | export const getGroupEntities = (
5 | entities: HassEntities,
6 | group: GroupEntity
7 | ) => {
8 | const result = {};
9 |
10 | group.attributes.entity_id.forEach((entityId) => {
11 | const entity = entities[entityId];
12 |
13 | if (entity) {
14 | result[entity.entity_id] = entity;
15 | }
16 | });
17 |
18 | return result;
19 | };
20 |
--------------------------------------------------------------------------------
/src/common/entity/has_location.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 |
3 | export const hasLocation = (stateObj: HassEntity) =>
4 | "latitude" in stateObj.attributes && "longitude" in stateObj.attributes;
5 |
--------------------------------------------------------------------------------
/src/common/entity/states_sort_by_name.ts:
--------------------------------------------------------------------------------
1 | /**
2 | * Sort function to help sort states by name
3 | *
4 | * Usage:
5 | * const states = [state1, state2]
6 | * states.sort(statessortStatesByName);
7 | */
8 | import type { HassEntity } from "home-assistant-js-websocket";
9 | import { computeStateName } from "./compute_state_name";
10 |
11 | export const sortStatesByName = (entityA: HassEntity, entityB: HassEntity) => {
12 | const nameA = computeStateName(entityA);
13 | const nameB = computeStateName(entityB);
14 | if (nameA < nameB) {
15 | return -1;
16 | }
17 | if (nameA > nameB) {
18 | return 1;
19 | }
20 | return 0;
21 | };
22 |
--------------------------------------------------------------------------------
/src/common/entity/supports-feature.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 |
3 | export const supportsFeature = (
4 | stateObj: HassEntity,
5 | feature: number
6 | ): boolean => supportsFeatureFromAttributes(stateObj.attributes, feature);
7 |
8 | export const supportsFeatureFromAttributes = (
9 | attributes: Record,
10 | feature: number
11 | ): boolean =>
12 | // eslint-disable-next-line no-bitwise
13 | (attributes.supported_features! & feature) !== 0;
14 |
--------------------------------------------------------------------------------
/src/common/entity/update_icon.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntity } from "home-assistant-js-websocket";
2 | import type { UpdateEntity } from "../../data/update";
3 | import { updateIsInstalling } from "../../data/update";
4 |
5 | export const updateIcon = (stateObj: HassEntity, state?: string) => {
6 | const compareState = state ?? stateObj.state;
7 | return compareState === "on"
8 | ? updateIsInstalling(stateObj as UpdateEntity)
9 | ? "mdi:package-down"
10 | : "mdi:package-up"
11 | : "mdi:package";
12 | };
13 |
--------------------------------------------------------------------------------
/src/common/entity/valid_entity_id.ts:
--------------------------------------------------------------------------------
1 | const validEntityId = /^(\w+)\.(\w+)$/;
2 |
3 | export const isValidEntityId = (entityId: string) =>
4 | validEntityId.test(entityId);
5 |
--------------------------------------------------------------------------------
/src/common/entity/valid_service_id.ts:
--------------------------------------------------------------------------------
1 | const validServiceId = /^(\w+)\.(\w+)$/;
2 |
3 | export const isValidServiceId = (actionId: string) =>
4 | validServiceId.test(actionId);
5 |
--------------------------------------------------------------------------------
/src/common/feature-detect/support-web-components.ts:
--------------------------------------------------------------------------------
1 | export const webComponentsSupported = "attachShadow" in Element.prototype;
2 |
--------------------------------------------------------------------------------
/src/common/language/format_language.ts:
--------------------------------------------------------------------------------
1 | import memoizeOne from "memoize-one";
2 | import type { FrontendLocaleData } from "../../data/translation";
3 |
4 | export const formatLanguageCode = (
5 | languageCode: string,
6 | locale: FrontendLocaleData
7 | ) => {
8 | try {
9 | return formatLanguageCodeMem(locale)?.of(languageCode) ?? languageCode;
10 | } catch {
11 | return languageCode;
12 | }
13 | };
14 |
15 | const formatLanguageCodeMem = memoizeOne(
16 | (locale: FrontendLocaleData) =>
17 | new Intl.DisplayNames(locale.language, {
18 | type: "language",
19 | fallback: "code",
20 | })
21 | );
22 |
--------------------------------------------------------------------------------
/src/common/location/add_distance_to_coord.ts:
--------------------------------------------------------------------------------
1 | export const addDistanceToCoord = (
2 | location: [number, number],
3 | dx: number,
4 | dy: number
5 | ): [number, number] => {
6 | const rEarth = 6378000;
7 | const newLatitude = location[0] + (dy / rEarth) * (180 / Math.PI);
8 | const newLongitude =
9 | location[1] +
10 | ((dx / rEarth) * (180 / Math.PI)) / Math.cos((location[0] * Math.PI) / 180);
11 | return [newLatitude, newLongitude];
12 | };
13 |
--------------------------------------------------------------------------------
/src/common/mwc/handle-request-selected-event.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | ListItem,
3 | RequestSelectedDetail,
4 | } from "@material/mwc-list/mwc-list-item";
5 |
6 | export const shouldHandleRequestSelectedEvent = (
7 | ev: CustomEvent
8 | ): boolean => {
9 | if (!ev.detail.selected || ev.detail.source !== "property") {
10 | return false;
11 | }
12 | (ev.currentTarget as ListItem).selected = false;
13 | return true;
14 | };
15 |
--------------------------------------------------------------------------------
/src/common/number/clamp.ts:
--------------------------------------------------------------------------------
1 | export const clamp = (value: number, min: number, max: number) =>
2 | Math.min(Math.max(value, min), max);
3 |
4 | // Variant that only applies the clamping to a border if the border is defined
5 | export const conditionalClamp = (value: number, min?: number, max?: number) => {
6 | let result: number;
7 | result = min != null ? Math.max(value, min) : value;
8 | result = max != null ? Math.min(result, max) : result;
9 | return result;
10 | };
11 |
--------------------------------------------------------------------------------
/src/common/number/round.ts:
--------------------------------------------------------------------------------
1 | export const round = (value: number, precision = 2): number =>
2 | Math.round(value * 10 ** precision) / 10 ** precision;
3 |
--------------------------------------------------------------------------------
/src/common/string/capitalize-first-letter.ts:
--------------------------------------------------------------------------------
1 | export const capitalizeFirstLetter = (str: string) =>
2 | str.charAt(0).toUpperCase() + str.slice(1);
3 |
--------------------------------------------------------------------------------
/src/common/string/escape_regexp.ts:
--------------------------------------------------------------------------------
1 | export const escapeRegExp = (text: string): string =>
2 | text.replace(/[-[\]{}()*+?.,\\^$|#\s]/g, "\\$&");
3 |
--------------------------------------------------------------------------------
/src/common/string/get_duplicates.ts:
--------------------------------------------------------------------------------
1 | export function getDuplicates(array: string[]): Set {
2 | const duplicates = new Set();
3 | const seen = new Set();
4 |
5 | for (const item of array) {
6 | if (seen.has(item)) {
7 | duplicates.add(item);
8 | } else {
9 | seen.add(item);
10 | }
11 | }
12 |
13 | return duplicates;
14 | }
15 |
--------------------------------------------------------------------------------
/src/common/string/has-template.ts:
--------------------------------------------------------------------------------
1 | const isTemplateRegex = /{%|{{/;
2 |
3 | export const isTemplate = (value: string): boolean =>
4 | isTemplateRegex.test(value);
5 |
6 | export const hasTemplate = (value: unknown): boolean => {
7 | if (!value) {
8 | return false;
9 | }
10 | if (typeof value === "string") {
11 | return isTemplate(value);
12 | }
13 | if (typeof value === "object") {
14 | const values = Array.isArray(value) ? value : Object.values(value!);
15 | return values.some((val) => val && hasTemplate(val));
16 | }
17 | return false;
18 | };
19 |
--------------------------------------------------------------------------------
/src/common/string/is_date.ts:
--------------------------------------------------------------------------------
1 | // https://regex101.com/r/kc5C14/2
2 | const regExpString = "^\\d{4}-(0[1-9]|1[0-2])-([12]\\d|0[1-9]|3[01])";
3 |
4 | const regExp = new RegExp(regExpString + "$");
5 | // 2nd expression without the "end of string" enforced, so it can be used
6 | // to just verify the start of a string and then based on that result e.g.
7 | // check for a full timestamp string efficiently.
8 | const regExpNoStringEnd = new RegExp(regExpString);
9 |
10 | export const isDate = (input: string, allowCharsAfterDate = false): boolean =>
11 | allowCharsAfterDate ? regExpNoStringEnd.test(input) : regExp.test(input);
12 |
--------------------------------------------------------------------------------
/src/common/string/is_ip_address.ts:
--------------------------------------------------------------------------------
1 | const regexp =
2 | /^(?:(?: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]?)$/;
3 |
4 | export const isIPAddress = (input: string): boolean => regexp.test(input);
5 |
--------------------------------------------------------------------------------
/src/common/string/starts-with.ts:
--------------------------------------------------------------------------------
1 | export const strStartsWith = (value: string, search: string) =>
2 | value.substring(0, search.length) === search;
3 |
--------------------------------------------------------------------------------
/src/common/string/strip-diacritics.ts:
--------------------------------------------------------------------------------
1 | export const stripDiacritics = (str: string) =>
2 | str.normalize("NFD").replace(/[\u0300-\u036F]/g, "");
3 |
--------------------------------------------------------------------------------
/src/common/string/title-case.ts:
--------------------------------------------------------------------------------
1 | export const titleCase = (s) =>
2 | s.replace(/^_*(.)|_+(.)/g, (_s, c, d) =>
3 | c ? c.toUpperCase() : " " + d.toUpperCase()
4 | );
5 |
--------------------------------------------------------------------------------
/src/common/structs/is-custom-type.ts:
--------------------------------------------------------------------------------
1 | import { refine, string } from "superstruct";
2 | import { isCustomType } from "../../data/lovelace_custom_cards";
3 |
4 | export const customType = () =>
5 | refine(string(), "custom element type", isCustomType);
6 |
--------------------------------------------------------------------------------
/src/common/structs/is-icon.ts:
--------------------------------------------------------------------------------
1 | import { refine, string } from "superstruct";
2 |
3 | const isIcon = (value: string) => value.includes(":");
4 |
5 | export const icon = () => refine(string(), "icon (mdi:icon-name)", isIcon);
6 |
--------------------------------------------------------------------------------
/src/common/translations/blank_before_percent.ts:
--------------------------------------------------------------------------------
1 | import type { FrontendLocaleData } from "../../data/translation";
2 |
3 | // Logic based on https://en.wikipedia.org/wiki/Percent_sign#Form_and_spacing
4 | export const blankBeforePercent = (
5 | localeOptions: FrontendLocaleData
6 | ): string => {
7 | switch (localeOptions.language) {
8 | case "cs":
9 | case "de":
10 | case "fi":
11 | case "fr":
12 | case "sk":
13 | case "sv":
14 | return " ";
15 | default:
16 | return "";
17 | }
18 | };
19 |
--------------------------------------------------------------------------------
/src/common/translations/blank_before_unit.ts:
--------------------------------------------------------------------------------
1 | import type { FrontendLocaleData } from "../../data/translation";
2 | import { blankBeforePercent } from "./blank_before_percent";
3 |
4 | export const blankBeforeUnit = (
5 | unit: string,
6 | localeOptions: FrontendLocaleData | undefined
7 | ): string => {
8 | if (unit === "°") {
9 | return "";
10 | }
11 | if (localeOptions && unit === "%") {
12 | return blankBeforePercent(localeOptions);
13 | }
14 | return " ";
15 | };
16 |
--------------------------------------------------------------------------------
/src/common/translations/day_names.ts:
--------------------------------------------------------------------------------
1 | import { addDays, startOfWeek } from "date-fns";
2 | import type { HassConfig } from "home-assistant-js-websocket";
3 | import memoizeOne from "memoize-one";
4 | import type { FrontendLocaleData } from "../../data/translation";
5 | import { formatDateWeekday } from "../datetime/format_date";
6 |
7 | export const dayNames = memoizeOne(
8 | (locale: FrontendLocaleData, config: HassConfig): string[] =>
9 | Array.from({ length: 7 }, (_, d) =>
10 | formatDateWeekday(addDays(startOfWeek(new Date()), d), locale, config)
11 | )
12 | );
13 |
--------------------------------------------------------------------------------
/src/common/translations/markdown_support.ts:
--------------------------------------------------------------------------------
1 | import { html } from "lit";
2 | import type { LocalizeFunc } from "./localize";
3 |
4 | const MARKDOWN_SUPPORT_URL = "https://commonmark.org/help/";
5 |
6 | export const supportsMarkdownHelper = (localize: LocalizeFunc) =>
7 | localize("ui.common.supports_markdown", {
8 | markdown_help_link: html`${localize("ui.common.markdown")}`,
14 | });
15 |
--------------------------------------------------------------------------------
/src/common/translations/month_names.ts:
--------------------------------------------------------------------------------
1 | import { addMonths, startOfYear } from "date-fns";
2 | import type { HassConfig } from "home-assistant-js-websocket";
3 | import memoizeOne from "memoize-one";
4 | import type { FrontendLocaleData } from "../../data/translation";
5 | import { formatDateMonth } from "../datetime/format_date";
6 |
7 | export const monthNames = memoizeOne(
8 | (locale: FrontendLocaleData, config: HassConfig): string[] =>
9 | Array.from({ length: 12 }, (_, m) =>
10 | formatDateMonth(addMonths(startOfYear(new Date()), m), locale, config)
11 | )
12 | );
13 |
--------------------------------------------------------------------------------
/src/common/url/construct-url.ts:
--------------------------------------------------------------------------------
1 | export const constructUrlCurrentPath = (searchParams: string): string => {
2 | const base = window.location.pathname;
3 | // Prevent trailing "?" if no parameters exist
4 | return searchParams ? base + "?" + searchParams : base;
5 | };
6 |
--------------------------------------------------------------------------------
/src/common/util/copy-clipboard.ts:
--------------------------------------------------------------------------------
1 | export const copyToClipboard = async (str, rootEl?: HTMLElement) => {
2 | if (navigator.clipboard) {
3 | try {
4 | await navigator.clipboard.writeText(str);
5 | return;
6 | } catch {
7 | // just continue with the fallback coding below
8 | }
9 | }
10 |
11 | const root = rootEl ?? document.body;
12 |
13 | const el = document.createElement("textarea");
14 | el.value = str;
15 | root.appendChild(el);
16 | el.select();
17 | document.execCommand("copy");
18 | root.removeChild(el);
19 | };
20 |
--------------------------------------------------------------------------------
/src/common/util/group-by.ts:
--------------------------------------------------------------------------------
1 | export const groupBy = (
2 | list: T[],
3 | keySelector: (item: T) => string
4 | ): Record => {
5 | const result = {};
6 | for (const item of list) {
7 | const key = keySelector(item);
8 | if (key in result) {
9 | result[key].push(item);
10 | } else {
11 | result[key] = [item];
12 | }
13 | }
14 | return result;
15 | };
16 |
--------------------------------------------------------------------------------
/src/common/util/patch.ts:
--------------------------------------------------------------------------------
1 | export const applyPatch = (data, path, value): void => {
2 | if (path.length === 1) {
3 | data[path[0]] = value;
4 | return;
5 | }
6 | if (!data[path[0]]) {
7 | data[path[0]] = {};
8 | }
9 | // eslint-disable-next-line consistent-return
10 | return applyPatch(data[path[0]], path.slice(1), value);
11 | };
12 |
13 | export const getPath = (data, path): any | undefined => {
14 | if (path.length === 1) {
15 | return data[path[0]];
16 | }
17 | if (data[path[0]] === undefined) {
18 | return undefined;
19 | }
20 | return getPath(data[path[0]], path.slice(1));
21 | };
22 |
--------------------------------------------------------------------------------
/src/common/util/promise-all-settled-results.ts:
--------------------------------------------------------------------------------
1 | export const hasRejectedItems = (results: PromiseSettledResult[]) =>
2 | results.some((result) => result.status === "rejected");
3 |
4 | export const rejectedItems = (
5 | results: PromiseSettledResult[]
6 | ): PromiseRejectedResult[] =>
7 | results.filter(
8 | (result) => result.status === "rejected"
9 | ) as PromiseRejectedResult[];
10 |
--------------------------------------------------------------------------------
/src/common/util/render-status.ts:
--------------------------------------------------------------------------------
1 | export const afterNextRender = (cb: (value: unknown) => void): void => {
2 | requestAnimationFrame(() => setTimeout(cb, 0));
3 | };
4 |
5 | export const nextRender = () =>
6 | new Promise((resolve) => {
7 | afterNextRender(resolve);
8 | });
9 |
--------------------------------------------------------------------------------
/src/common/util/subscribe-one.ts:
--------------------------------------------------------------------------------
1 | import type { Connection, UnsubscribeFunc } from "home-assistant-js-websocket";
2 |
3 | export const subscribeOne = async (
4 | conn: Connection,
5 | subscribe: (
6 | conn2: Connection,
7 | onChange: (items: T) => void
8 | ) => UnsubscribeFunc
9 | ) =>
10 | new Promise((resolve) => {
11 | const unsub = subscribe(conn, (items) => {
12 | unsub();
13 | resolve(items);
14 | });
15 | });
16 |
--------------------------------------------------------------------------------
/src/common/util/subscribe-polling.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../types";
2 |
3 | export const subscribePollingCollection = (
4 | hass: HomeAssistant,
5 | updateData: (hass: HomeAssistant) => void,
6 | interval: number
7 | ) => {
8 | let timeout;
9 | const fetchData = async () => {
10 | try {
11 | await updateData(hass);
12 | } finally {
13 | timeout = setTimeout(() => fetchData(), interval);
14 | }
15 | };
16 | fetchData();
17 | return () => clearTimeout(timeout);
18 | };
19 |
--------------------------------------------------------------------------------
/src/common/util/uid.ts:
--------------------------------------------------------------------------------
1 | function s4() {
2 | return Math.floor((1 + Math.random()) * 0x10000)
3 | .toString(16)
4 | .substring(1);
5 | }
6 |
7 | export function uid() {
8 | return s4() + s4() + s4() + s4() + s4();
9 | }
10 |
--------------------------------------------------------------------------------
/src/common/util/wait.ts:
--------------------------------------------------------------------------------
1 | export const waitForMs = (ms: number) =>
2 | new Promise((resolve) => {
3 | setTimeout(resolve, ms);
4 | });
5 |
6 | export const waitForSeconds = (seconds: number) => waitForMs(seconds * 1000);
7 |
--------------------------------------------------------------------------------
/src/components/chips/ha-chip-set.ts:
--------------------------------------------------------------------------------
1 | import { MdChipSet } from "@material/web/chips/chip-set";
2 | import { customElement } from "lit/decorators";
3 |
4 | @customElement("ha-chip-set")
5 | export class HaChipSet extends MdChipSet {}
6 |
7 | declare global {
8 | interface HTMLElementTagNameMap {
9 | "ha-chip-set": HaChipSet;
10 | }
11 | }
12 |
--------------------------------------------------------------------------------
/src/components/ha-checkbox.ts:
--------------------------------------------------------------------------------
1 | import { CheckboxBase } from "@material/mwc-checkbox/mwc-checkbox-base";
2 | import { styles } from "@material/mwc-checkbox/mwc-checkbox.css";
3 | import { css } from "lit";
4 | import { customElement } from "lit/decorators";
5 |
6 | @customElement("ha-checkbox")
7 | export class HaCheckbox extends CheckboxBase {
8 | static override styles = [
9 | styles,
10 | css`
11 | :host {
12 | --mdc-theme-secondary: var(--primary-color);
13 | }
14 | `,
15 | ];
16 | }
17 |
18 | declare global {
19 | interface HTMLElementTagNameMap {
20 | "ha-checkbox": HaCheckbox;
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/src/components/ha-fade-in.ts:
--------------------------------------------------------------------------------
1 | import SlAnimation from "@shoelace-style/shoelace/dist/components/animation/animation.component";
2 | import { customElement, property } from "lit/decorators";
3 |
4 | @customElement("ha-fade-in")
5 | export class HaFadeIn extends SlAnimation {
6 | @property() public name = "fadeIn";
7 |
8 | @property() public fill: FillMode = "both";
9 |
10 | @property({ type: Boolean }) public play = true;
11 |
12 | @property({ type: Number }) public iterations = 1;
13 | }
14 |
15 | declare global {
16 | interface HTMLElementTagNameMap {
17 | "ha-fade-in": HaFadeIn;
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/components/ha-icon-next.ts:
--------------------------------------------------------------------------------
1 | import { mdiChevronLeft, mdiChevronRight } from "@mdi/js";
2 | import { customElement, property } from "lit/decorators";
3 | import { mainWindow } from "../common/dom/get_main_window";
4 | import { HaSvgIcon } from "./ha-svg-icon";
5 |
6 | @customElement("ha-icon-next")
7 | export class HaIconNext extends HaSvgIcon {
8 | @property() public override path =
9 | mainWindow.document.dir === "rtl" ? mdiChevronLeft : mdiChevronRight;
10 | }
11 |
12 | declare global {
13 | interface HTMLElementTagNameMap {
14 | "ha-icon-next": HaIconNext;
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/src/components/ha-icon-prev.ts:
--------------------------------------------------------------------------------
1 | import { mdiChevronLeft, mdiChevronRight } from "@mdi/js";
2 | import { customElement, property } from "lit/decorators";
3 | import { mainWindow } from "../common/dom/get_main_window";
4 | import { HaSvgIcon } from "./ha-svg-icon";
5 |
6 | @customElement("ha-icon-prev")
7 | export class HaIconPrev extends HaSvgIcon {
8 | @property() public override path =
9 | mainWindow.document.dir === "rtl" ? mdiChevronRight : mdiChevronLeft;
10 | }
11 |
12 | declare global {
13 | interface HTMLElementTagNameMap {
14 | "ha-icon-prev": HaIconPrev;
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/src/components/ha-list.ts:
--------------------------------------------------------------------------------
1 | import { ListBase } from "@material/mwc-list/mwc-list-base";
2 | import { styles } from "@material/mwc-list/mwc-list.css";
3 | import { customElement } from "lit/decorators";
4 |
5 | @customElement("ha-list")
6 | export class HaList extends ListBase {
7 | static styles = styles;
8 | }
9 |
10 | declare global {
11 | interface HTMLElementTagNameMap {
12 | "ha-list": HaList;
13 | }
14 | }
15 |
--------------------------------------------------------------------------------
/src/components/ha-md-divider.ts:
--------------------------------------------------------------------------------
1 | import { Divider } from "@material/web/divider/internal/divider";
2 | import { styles } from "@material/web/divider/internal/divider-styles";
3 | import { css } from "lit";
4 | import { customElement } from "lit/decorators";
5 |
6 | @customElement("ha-md-divider")
7 | export class HaMdDivider extends Divider {
8 | static override styles = [
9 | styles,
10 | css`
11 | :host {
12 | --md-divider-color: var(--divider-color);
13 | }
14 | `,
15 | ];
16 | }
17 |
18 | declare global {
19 | interface HTMLElementTagNameMap {
20 | "ha-md-divider": HaMdDivider;
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/src/components/ha-md-list.ts:
--------------------------------------------------------------------------------
1 | import { List } from "@material/web/list/internal/list";
2 | import { styles } from "@material/web/list/internal/list-styles";
3 | import { css } from "lit";
4 | import { customElement } from "lit/decorators";
5 |
6 | @customElement("ha-md-list")
7 | export class HaMdList extends List {
8 | static override styles = [
9 | styles,
10 | css`
11 | :host {
12 | --md-sys-color-surface: var(--card-background-color);
13 | }
14 | `,
15 | ];
16 | }
17 |
18 | declare global {
19 | interface HTMLElementTagNameMap {
20 | "ha-md-list": HaMdList;
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/src/components/ha-radio.ts:
--------------------------------------------------------------------------------
1 | import { RadioBase } from "@material/mwc-radio/mwc-radio-base";
2 | import { styles } from "@material/mwc-radio/mwc-radio.css";
3 | import { css } from "lit";
4 | import { customElement } from "lit/decorators";
5 |
6 | @customElement("ha-radio")
7 | export class HaRadio extends RadioBase {
8 | static override styles = [
9 | styles,
10 | css`
11 | :host {
12 | --mdc-theme-secondary: var(--primary-color);
13 | }
14 | `,
15 | ];
16 | }
17 |
18 | declare global {
19 | interface HTMLElementTagNameMap {
20 | "ha-radio": HaRadio;
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/src/components/media-player/show-join-media-players-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export interface JoinMediaPlayersDialogParams {
4 | entityId: string;
5 | }
6 |
7 | export const showJoinMediaPlayersDialog = (
8 | element: HTMLElement,
9 | dialogParams: JoinMediaPlayersDialogParams
10 | ): void => {
11 | fireEvent(element, "show-dialog", {
12 | dialogTag: "dialog-join-media-players",
13 | dialogImport: () => import("./dialog-join-media-players"),
14 | dialogParams,
15 | });
16 | };
17 |
--------------------------------------------------------------------------------
/src/components/media-player/show-media-manage-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 | import type { MediaPlayerItem } from "../../data/media-player";
3 |
4 | export interface MediaManageDialogParams {
5 | currentItem: MediaPlayerItem;
6 | onClose?: () => void;
7 | }
8 |
9 | export const showMediaManageDialog = (
10 | element: HTMLElement,
11 | dialogParams: MediaManageDialogParams
12 | ): void => {
13 | fireEvent(element, "show-dialog", {
14 | dialogTag: "dialog-media-manage",
15 | dialogImport: () => import("./dialog-media-manage"),
16 | dialogParams,
17 | });
18 | };
19 |
--------------------------------------------------------------------------------
/src/components/timezone-datalist.ts:
--------------------------------------------------------------------------------
1 | import timezones from "google-timezones-json";
2 |
3 | export const createTimezoneListEl = () => {
4 | const list = document.createElement("datalist");
5 | list.id = "timezones";
6 | Object.keys(timezones).forEach((key) => {
7 | const option = document.createElement("option");
8 | option.value = key;
9 | option.innerText = timezones[key];
10 | list.appendChild(option);
11 | });
12 | return list;
13 | };
14 |
--------------------------------------------------------------------------------
/src/components/trace/hat-graph-const.ts:
--------------------------------------------------------------------------------
1 | export const SPACING = 10;
2 | export const NODE_SIZE = 30;
3 | export const BRANCH_HEIGHT = 20;
4 |
--------------------------------------------------------------------------------
/src/data/bootstrap_integrations.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export type BootstrapIntegrationsTimings = Record;
4 |
5 | export const subscribeBootstrapIntegrations = (
6 | hass: HomeAssistant,
7 | callback: (message: BootstrapIntegrationsTimings) => void
8 | ) => {
9 | const unsubProm =
10 | hass.connection.subscribeMessage(
11 | (message) => callback(message),
12 | {
13 | type: "subscribe_bootstrap_integrations",
14 | }
15 | );
16 |
17 | return unsubProm;
18 | };
19 |
--------------------------------------------------------------------------------
/src/data/config.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | interface ValidConfig {
4 | valid: true;
5 | error: null;
6 | }
7 |
8 | interface InvalidConfig {
9 | valid: false;
10 | error: string;
11 | }
12 |
13 | type ValidKeys = "triggers" | "actions" | "conditions";
14 |
15 | export const validateConfig = >>(
16 | hass: HomeAssistant,
17 | config: T
18 | ): Promise> =>
19 | hass.callWS({
20 | type: "validate_config",
21 | ...config,
22 | });
23 |
--------------------------------------------------------------------------------
/src/data/custom_iconsets.ts:
--------------------------------------------------------------------------------
1 | import type { CustomIcon } from "./custom_icons";
2 |
3 | interface CustomIconsetsWindow {
4 | customIconsets?: Record Promise>;
5 | }
6 |
7 | const customIconsetsWindow = window as CustomIconsetsWindow;
8 |
9 | if (!("customIconsets" in customIconsetsWindow)) {
10 | customIconsetsWindow.customIconsets = {};
11 | }
12 |
13 | export const customIconsets = customIconsetsWindow.customIconsets!;
14 |
--------------------------------------------------------------------------------
/src/data/date.ts:
--------------------------------------------------------------------------------
1 | import type { HassEntityBase } from "home-assistant-js-websocket";
2 | import type { HomeAssistant } from "../types";
3 |
4 | export const stateToIsoDateString = (entityState: HassEntityBase) =>
5 | `${entityState}T00:00:00`;
6 |
7 | export const setDateValue = (
8 | hass: HomeAssistant,
9 | entityId: string,
10 | date: string | undefined = undefined
11 | ) => {
12 | const param = { entity_id: entityId, date };
13 | hass.callService("date", "set_value", param);
14 | };
15 |
--------------------------------------------------------------------------------
/src/data/datetime.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export const setDateTimeValue = (
4 | hass: HomeAssistant,
5 | entityId: string,
6 | datetime: Date
7 | ) => {
8 | hass.callService("datetime", "set_value", {
9 | entity_id: entityId,
10 | datetime: datetime.toISOString(),
11 | });
12 | };
13 |
--------------------------------------------------------------------------------
/src/data/entity.ts:
--------------------------------------------------------------------------------
1 | import { arrayLiteralIncludes } from "../common/array/literal-includes";
2 |
3 | export const UNAVAILABLE = "unavailable";
4 | export const UNKNOWN = "unknown";
5 | export const ON = "on";
6 | export const OFF = "off";
7 |
8 | export const UNAVAILABLE_STATES = [UNAVAILABLE, UNKNOWN] as const;
9 | export const OFF_STATES = [UNAVAILABLE, UNKNOWN, OFF] as const;
10 |
11 | export const isUnavailableState = arrayLiteralIncludes(UNAVAILABLE_STATES);
12 | export const isOffState = arrayLiteralIncludes(OFF_STATES);
13 |
--------------------------------------------------------------------------------
/src/data/error_log.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export interface LogProvider {
4 | key: string;
5 | name: string;
6 | }
7 |
8 | export const fetchErrorLog = (hass: HomeAssistant) =>
9 | hass.callApi("GET", "error_log");
10 |
11 | export const getErrorLogDownloadUrl = "/api/error_log";
12 |
--------------------------------------------------------------------------------
/src/data/external.ts:
--------------------------------------------------------------------------------
1 | export const isExternal =
2 | window.externalApp ||
3 | window.webkit?.messageHandlers?.getExternalAuth ||
4 | location.search.includes("external_auth=1");
5 |
--------------------------------------------------------------------------------
/src/data/graph.ts:
--------------------------------------------------------------------------------
1 | export const strokeWidth = 5;
2 |
--------------------------------------------------------------------------------
/src/data/image.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | HassEntityAttributeBase,
3 | HassEntityBase,
4 | } from "home-assistant-js-websocket";
5 |
6 | interface ImageEntityAttributes extends HassEntityAttributeBase {
7 | access_token: string;
8 | }
9 |
10 | export interface ImageEntity extends HassEntityBase {
11 | attributes: ImageEntityAttributes;
12 | }
13 |
14 | export const computeImageUrl = (entity: ImageEntity): string =>
15 | `/api/image_proxy/${entity.entity_id}?token=${entity.attributes.access_token}&state=${entity.state}`;
16 |
--------------------------------------------------------------------------------
/src/data/lovelace/action_handler.ts:
--------------------------------------------------------------------------------
1 | import type { HASSDomEvent } from "../../common/dom/fire_event";
2 |
3 | export interface ActionHandlerOptions {
4 | hasHold?: boolean;
5 | hasDoubleClick?: boolean;
6 | disabled?: boolean;
7 | }
8 |
9 | export interface ActionHandlerDetail {
10 | action: "hold" | "tap" | "double_tap";
11 | }
12 |
13 | export type ActionHandlerEvent = HASSDomEvent;
14 |
--------------------------------------------------------------------------------
/src/data/lovelace/config/card.ts:
--------------------------------------------------------------------------------
1 | import type { Condition } from "../../../panels/lovelace/common/validate-condition";
2 | import type {
3 | LovelaceGridOptions,
4 | LovelaceLayoutOptions,
5 | } from "../../../panels/lovelace/types";
6 |
7 | export interface LovelaceCardConfig {
8 | index?: number;
9 | view_index?: number;
10 | view_layout?: any;
11 | /** @deprecated Use `grid_options` instead */
12 | layout_options?: LovelaceLayoutOptions;
13 | grid_options?: LovelaceGridOptions;
14 | type: string;
15 | [key: string]: any;
16 | visibility?: Condition[];
17 | }
18 |
--------------------------------------------------------------------------------
/src/data/lovelace/config/strategy.ts:
--------------------------------------------------------------------------------
1 | export interface LovelaceStrategyConfig {
2 | type: string;
3 | [key: string]: any;
4 | }
5 |
--------------------------------------------------------------------------------
/src/data/main_window.ts:
--------------------------------------------------------------------------------
1 | export const MAIN_WINDOW_NAME = "ha-main-window";
2 |
--------------------------------------------------------------------------------
/src/data/number.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export interface NumberDeviceClassUnits {
4 | units: string[];
5 | }
6 |
7 | export const getNumberDeviceClassConvertibleUnits = (
8 | hass: HomeAssistant,
9 | deviceClass: string
10 | ): Promise =>
11 | hass.callWS({
12 | type: "number/device_class_convertible_units",
13 | device_class: deviceClass,
14 | });
15 |
--------------------------------------------------------------------------------
/src/data/panel_custom.ts:
--------------------------------------------------------------------------------
1 | import type { PanelInfo } from "../types";
2 |
3 | export interface CustomPanelConfig {
4 | name: string;
5 | embed_iframe: boolean;
6 | trust_external: boolean;
7 | js_url?: string;
8 | module_url?: string;
9 | html_url?: string;
10 | }
11 |
12 | export type CustomPanelInfo> = PanelInfo<
13 | T & { _panel_custom: CustomPanelConfig }
14 | >;
15 |
--------------------------------------------------------------------------------
/src/data/preloads.ts:
--------------------------------------------------------------------------------
1 | import type { LovelaceRawConfig } from "./lovelace/config/types";
2 | import type { LovelaceResource } from "./lovelace/resource";
3 | import type { RecorderInfo } from "./recorder";
4 |
5 | export interface WindowWithPreloads extends Window {
6 | llConfProm?: Promise;
7 | llResProm?: Promise;
8 | recorderInfoProm?: Promise;
9 | }
10 |
--------------------------------------------------------------------------------
/src/data/refresh_token.ts:
--------------------------------------------------------------------------------
1 | declare global {
2 | interface HASSDomEvents {
3 | "hass-refresh-tokens": undefined;
4 | }
5 | }
6 |
7 | export type RefreshTokenType = "normal" | "long_lived_access_token";
8 |
9 | export interface RefreshToken {
10 | client_icon?: string;
11 | client_id: string;
12 | client_name?: string;
13 | created_at: string;
14 | expire_at?: string;
15 | id: string;
16 | is_current: boolean;
17 | last_used_at?: string;
18 | last_used_ip?: string;
19 | type: RefreshTokenType;
20 | }
21 |
--------------------------------------------------------------------------------
/src/data/registry.ts:
--------------------------------------------------------------------------------
1 | export interface RegistryEntry {
2 | created_at: number;
3 | modified_at: number;
4 | }
5 |
--------------------------------------------------------------------------------
/src/data/remote.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | HassEntityAttributeBase,
3 | HassEntityBase,
4 | } from "home-assistant-js-websocket";
5 |
6 | export const REMOTE_SUPPORT_LEARN_COMMAND = 1;
7 | export const REMOTE_SUPPORT_DELETE_COMMAND = 2;
8 | export const REMOTE_SUPPORT_ACTIVITY = 4;
9 |
10 | export type RemoteEntity = HassEntityBase & {
11 | attributes: HassEntityAttributeBase & {
12 | current_activity: string | null;
13 | activity_list: string[] | null;
14 | [key: string]: any;
15 | };
16 | };
17 |
--------------------------------------------------------------------------------
/src/data/select.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | HassEntityAttributeBase,
3 | HassEntityBase,
4 | } from "home-assistant-js-websocket";
5 | import type { HomeAssistant } from "../types";
6 |
7 | interface SelectEntityAttributes extends HassEntityAttributeBase {
8 | options: string[];
9 | }
10 |
11 | export interface SelectEntity extends HassEntityBase {
12 | attributes: SelectEntityAttributes;
13 | }
14 |
15 | export const setSelectOption = (
16 | hass: HomeAssistant,
17 | entity: string,
18 | option: string
19 | ) =>
20 | hass.callService(
21 | "select",
22 | "select_option",
23 | { option },
24 | { entity_id: entity }
25 | );
26 |
--------------------------------------------------------------------------------
/src/data/siren.ts:
--------------------------------------------------------------------------------
1 | export const SirenEntityFeature = {
2 | TURN_ON: 1,
3 | TURN_OFF: 2,
4 | TONES: 4,
5 | VOLUME_SET: 8,
6 | DURATION: 16,
7 | };
8 |
--------------------------------------------------------------------------------
/src/data/text.ts:
--------------------------------------------------------------------------------
1 | import type {
2 | HassEntityAttributeBase,
3 | HassEntityBase,
4 | } from "home-assistant-js-websocket";
5 | import type { HomeAssistant } from "../types";
6 |
7 | interface TextEntityAttributes extends HassEntityAttributeBase {
8 | min?: number;
9 | max?: number;
10 | pattern?: string;
11 | mode?: "text" | "password";
12 | }
13 |
14 | export interface TextEntity extends HassEntityBase {
15 | attributes: TextEntityAttributes;
16 | }
17 |
18 | export const setValue = (hass: HomeAssistant, entity: string, value: string) =>
19 | hass.callService("text", "set_value", { value }, { entity_id: entity });
20 |
--------------------------------------------------------------------------------
/src/data/time.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export const setTimeValue = (
4 | hass: HomeAssistant,
5 | entityId: string,
6 | time: string | undefined = undefined
7 | ) => {
8 | const param = { entity_id: entityId, time: time };
9 | hass.callService("time", "set_value", param);
10 | };
11 |
--------------------------------------------------------------------------------
/src/data/usb.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export const scanUSBDevices = (hass: HomeAssistant) =>
4 | hass.callWS({ type: "usb/scan" });
5 |
--------------------------------------------------------------------------------
/src/data/wake_word.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export interface WakeWord {
4 | id: string;
5 | name: string;
6 | }
7 |
8 | export const fetchWakeWordInfo = (hass: HomeAssistant, entity_id: string) =>
9 | hass.callWS<{ wake_words: WakeWord[] }>({
10 | type: "wake_word/info",
11 | entity_id,
12 | });
13 |
--------------------------------------------------------------------------------
/src/data/webhook.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export interface Webhook {
4 | webhook_id: string;
5 | domain: string;
6 | name: string;
7 | local_only: boolean;
8 | }
9 | export interface WebhookError {
10 | code: number;
11 | message: string;
12 | }
13 |
14 | export const fetchWebhooks = (hass: HomeAssistant): Promise =>
15 | hass.callWS({
16 | type: "webhook/list",
17 | });
18 |
--------------------------------------------------------------------------------
/src/data/websocket_api.ts:
--------------------------------------------------------------------------------
1 | export const ERR_ID_REUSE = "id_reuse";
2 | export const ERR_INVALID_FORMAT = "invalid_format";
3 | export const ERR_NOT_FOUND = "not_found";
4 | export const ERR_UNKNOWN_COMMAND = "unknown_command";
5 | export const ERR_UNKNOWN_ERROR = "unknown_error";
6 | export const ERR_UNAUTHORIZED = "unauthorized";
7 |
--------------------------------------------------------------------------------
/src/data/ws-user.ts:
--------------------------------------------------------------------------------
1 | import type { Connection } from "home-assistant-js-websocket";
2 | import { getCollection, getUser } from "home-assistant-js-websocket";
3 | import type { CurrentUser } from "../types";
4 |
5 | export const userCollection = (conn: Connection) =>
6 | getCollection(
7 | conn,
8 | "_usr",
9 | () => getUser(conn) as Promise,
10 | undefined
11 | );
12 |
13 | export const subscribeUser = (
14 | conn: Connection,
15 | onChange: (user: CurrentUser) => void
16 | ) => userCollection(conn).subscribe(onChange);
17 |
--------------------------------------------------------------------------------
/src/data/wyoming.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export interface WyomingInfo {
4 | asr: WyomingAsrInfo[];
5 | handle: [];
6 | intent: [];
7 | tts: WyomingTtsInfo[];
8 | wake: [];
9 | }
10 |
11 | interface WyomingBaseInfo {
12 | name: string;
13 | version: string;
14 | attribution: Record;
15 | }
16 |
17 | interface WyomingTtsInfo extends WyomingBaseInfo {}
18 |
19 | interface WyomingAsrInfo extends WyomingBaseInfo {}
20 |
21 | export const fetchWyomingInfo = (hass: HomeAssistant) =>
22 | hass.callWS<{ info: Record }>({ type: "wyoming/info" });
23 |
--------------------------------------------------------------------------------
/src/dialogs/more-info/components/voice/show-view-voice-assistants.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 |
3 | export const loadVoiceAssistantsView = () =>
4 | import("./ha-more-info-view-voice-assistants");
5 |
6 | export const showVoiceAssistantsView = (
7 | element: HTMLElement,
8 | title: string
9 | ): void => {
10 | fireEvent(element, "show-child-view", {
11 | viewTag: "ha-more-info-view-voice-assistants",
12 | viewImport: loadVoiceAssistantsView,
13 | viewTitle: title,
14 | viewParams: {},
15 | });
16 | };
17 |
--------------------------------------------------------------------------------
/src/dialogs/more-info/show-ha-more-info-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 | import type { MoreInfoDialogParams } from "./ha-more-info-dialog";
3 |
4 | export const showMoreInfoDialog = (
5 | element: HTMLElement,
6 | params: MoreInfoDialogParams
7 | ) => fireEvent(element, "hass-more-info", params);
8 |
--------------------------------------------------------------------------------
/src/dialogs/notifications/show-notification-drawer.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export interface NotificationDrawerParams {
4 | narrow: boolean;
5 | }
6 |
7 | export const showNotificationDrawer = (
8 | element: HTMLElement,
9 | dialogParams: NotificationDrawerParams
10 | ): void => {
11 | fireEvent(element, "show-dialog", {
12 | dialogTag: "notification-drawer",
13 | dialogImport: () => import("./notification-drawer"),
14 | dialogParams,
15 | });
16 | };
17 |
--------------------------------------------------------------------------------
/src/dialogs/shortcuts/show-shortcuts-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export const showShortcutsDialog = (element: HTMLElement) =>
4 | fireEvent(element, "show-dialog", {
5 | dialogTag: "dialog-shortcuts",
6 | dialogImport: () => import("./dialog-shortcuts"),
7 | dialogParams: {},
8 | });
9 |
--------------------------------------------------------------------------------
/src/dialogs/sidebar/show-dialog-edit-sidebar.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export const loadEditSidebarDialog = () => import("./dialog-edit-sidebar");
4 |
5 | export const showEditSidebarDialog = (element: HTMLElement): void => {
6 | fireEvent(element, "show-dialog", {
7 | dialogTag: "dialog-edit-sidebar",
8 | dialogImport: loadEditSidebarDialog,
9 | dialogParams: {},
10 | });
11 | };
12 |
--------------------------------------------------------------------------------
/src/dialogs/tts-try/show-dialog-tts-try.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export interface TTSTryDialogParams {
4 | engine: string;
5 | language?: string;
6 | voice?: string;
7 | }
8 |
9 | export const loadTTSTryDialog = () => import("./dialog-tts-try");
10 |
11 | export const showTTSTryDialog = (
12 | element: HTMLElement,
13 | dialogParams: TTSTryDialogParams
14 | ): void => {
15 | fireEvent(element, "show-dialog", {
16 | addHistory: false,
17 | dialogTag: "dialog-tts-try",
18 | dialogImport: loadTTSTryDialog,
19 | dialogParams,
20 | });
21 | };
22 |
--------------------------------------------------------------------------------
/src/entrypoints/app.ts:
--------------------------------------------------------------------------------
1 | import "@webcomponents/scoped-custom-element-registry/scoped-custom-element-registry.min";
2 | import "../layouts/home-assistant";
3 |
4 | import("../resources/append-ha-style");
5 |
--------------------------------------------------------------------------------
/src/entrypoints/authorize.ts:
--------------------------------------------------------------------------------
1 | import "../auth/ha-authorize";
2 |
3 | import("../resources/append-ha-style");
4 |
--------------------------------------------------------------------------------
/src/entrypoints/onboarding.ts:
--------------------------------------------------------------------------------
1 | import "../onboarding/ha-onboarding";
2 |
3 | import("../resources/append-ha-style");
4 |
5 | declare global {
6 | interface Window {
7 | stepsPromise: Promise;
8 | }
9 | }
10 |
--------------------------------------------------------------------------------
/src/html/_header.html.template:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | <% for (const entry of latestEntryJS) { %>
5 |
6 | <% } %>
7 |
--------------------------------------------------------------------------------
/src/html/_preload_roboto.html.template:
--------------------------------------------------------------------------------
1 |
16 |
--------------------------------------------------------------------------------
/src/html/_script_loader.html.template:
--------------------------------------------------------------------------------
1 |
9 |
18 |
--------------------------------------------------------------------------------
/src/html/_style_base.html.template:
--------------------------------------------------------------------------------
1 |
2 |
13 |
--------------------------------------------------------------------------------
/src/onboarding/dialogs/show-app-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 | import type { LocalizeFunc } from "../../common/translations/localize";
3 |
4 | export const loadAppDialog = () => import("./app-dialog");
5 |
6 | export const showAppDialog = (
7 | element: HTMLElement,
8 | params: { localize: LocalizeFunc }
9 | ): void => {
10 | fireEvent(element, "show-dialog", {
11 | dialogTag: "app-dialog",
12 | dialogImport: loadAppDialog,
13 | dialogParams: params,
14 | });
15 | };
16 |
--------------------------------------------------------------------------------
/src/onboarding/dialogs/show-community-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 | import type { LocalizeFunc } from "../../common/translations/localize";
3 |
4 | export const loadCommunityDialog = () => import("./community-dialog");
5 |
6 | export const showCommunityDialog = (
7 | element: HTMLElement,
8 | params: { localize: LocalizeFunc }
9 | ): void => {
10 | fireEvent(element, "show-dialog", {
11 | dialogTag: "community-dialog",
12 | dialogImport: loadCommunityDialog,
13 | dialogParams: params,
14 | });
15 | };
16 |
--------------------------------------------------------------------------------
/src/panels/config/automation/structs.ts:
--------------------------------------------------------------------------------
1 | import { object, optional, number, string, boolean } from "superstruct";
2 |
3 | export const baseTriggerStruct = object({
4 | trigger: string(),
5 | id: optional(string()),
6 | enabled: optional(boolean()),
7 | });
8 |
9 | export const forDictStruct = object({
10 | days: optional(number()),
11 | hours: optional(number()),
12 | minutes: optional(number()),
13 | seconds: optional(number()),
14 | });
15 |
--------------------------------------------------------------------------------
/src/panels/config/backup/dialogs/show-dialog-local-backup-location.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 |
3 | export interface LocalBackupLocationDialogParams {}
4 |
5 | export const showLocalBackupLocationDialog = (
6 | element: HTMLElement,
7 | dialogParams: LocalBackupLocationDialogParams
8 | ): void => {
9 | fireEvent(element, "show-dialog", {
10 | dialogTag: "dialog-local-backup-location",
11 | dialogImport: () => import("./dialog-local-backup-location"),
12 | dialogParams,
13 | });
14 | };
15 |
--------------------------------------------------------------------------------
/src/panels/config/backup/dialogs/show-dialog-show-backup-encryption-key.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 |
3 | export interface ShowBackupEncryptionKeyDialogParams {
4 | currentKey: string;
5 | }
6 |
7 | const loadDialog = () => import("./dialog-show-backup-encryption-key");
8 |
9 | export const showShowBackupEncryptionKeyDialog = (
10 | element: HTMLElement,
11 | params?: ShowBackupEncryptionKeyDialogParams
12 | ) =>
13 | fireEvent(element, "show-dialog", {
14 | dialogTag: "ha-dialog-show-backup-encryption-key",
15 | dialogImport: loadDialog,
16 | dialogParams: params,
17 | });
18 |
--------------------------------------------------------------------------------
/src/panels/config/blueprint/show-dialog-import-blueprint.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export const loadImportBlueprintDialog = () =>
4 | import("./dialog-import-blueprint");
5 |
6 | export const showAddBlueprintDialog = (
7 | element: HTMLElement,
8 | dialogParams
9 | ): void => {
10 | fireEvent(element, "show-dialog", {
11 | dialogTag: "ha-dialog-import-blueprint",
12 | dialogImport: loadImportBlueprintDialog,
13 | dialogParams,
14 | });
15 | };
16 |
--------------------------------------------------------------------------------
/src/panels/config/cloud/account/show-dialog-cloud-support-package.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 |
3 | export const loadSupportPackageDialog = () =>
4 | import("./dialog-cloud-support-package");
5 |
6 | export const showSupportPackageDialog = (element: HTMLElement): void => {
7 | fireEvent(element, "show-dialog", {
8 | dialogTag: "dialog-cloud-support-package",
9 | dialogImport: loadSupportPackageDialog,
10 | dialogParams: {},
11 | });
12 | };
13 |
--------------------------------------------------------------------------------
/src/panels/config/cloud/account/show-dialog-cloud-tts-try.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 |
3 | export interface TryTtsDialogParams {
4 | defaultVoice: [string, string];
5 | }
6 |
7 | export const loadTryTtsDialog = () => import("./dialog-cloud-tts-try");
8 |
9 | export const showTryTtsDialog = (
10 | element: HTMLElement,
11 | dialogParams: TryTtsDialogParams
12 | ): void => {
13 | fireEvent(element, "show-dialog", {
14 | dialogTag: "dialog-cloud-try-tts",
15 | dialogImport: loadTryTtsDialog,
16 | dialogParams,
17 | });
18 | };
19 |
--------------------------------------------------------------------------------
/src/panels/config/core/updates/show-dialog-join-beta.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../common/dom/fire_event";
2 | import "./dialog-join-beta";
3 |
4 | export interface JoinBetaDialogParams {
5 | join?: () => any;
6 | cancel?: () => any;
7 | }
8 |
9 | export const showJoinBetaDialog = (
10 | element: HTMLElement,
11 | dialogParams: JoinBetaDialogParams
12 | ): void => {
13 | fireEvent(element, "show-dialog", {
14 | dialogTag: "dialog-join-beta",
15 | dialogImport: () => import("./dialog-join-beta"),
16 | dialogParams,
17 | });
18 | };
19 |
--------------------------------------------------------------------------------
/src/panels/config/entities/const.ts:
--------------------------------------------------------------------------------
1 | /** Platforms that have a settings tab. */
2 | export const PLATFORMS_WITH_SETTINGS_TAB = {
3 | input_number: "entity-settings-helper-tab",
4 | input_select: "entity-settings-helper-tab",
5 | input_text: "entity-settings-helper-tab",
6 | input_boolean: "entity-settings-helper-tab",
7 | input_datetime: "entity-settings-helper-tab",
8 | counter: "entity-settings-helper-tab",
9 | timer: "entity-settings-helper-tab",
10 | input_button: "entity-settings-helper-tab",
11 | schedule: "entity-settings-helper-tab",
12 | };
13 |
--------------------------------------------------------------------------------
/src/panels/config/hardware/show-dialog-hardware-available.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export const loadHardwareAvailableDialog = () =>
4 | import("./dialog-hardware-available");
5 |
6 | export const showhardwareAvailableDialog = (element: HTMLElement): void => {
7 | fireEvent(element, "show-dialog", {
8 | dialogTag: "ha-dialog-hardware-available",
9 | dialogImport: loadHardwareAvailableDialog,
10 | dialogParams: {},
11 | });
12 | };
13 |
--------------------------------------------------------------------------------
/src/panels/config/integrations/integration-panels/matter/show-dialog-add-matter-device.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../../../common/dom/fire_event";
2 |
3 | export const loadAddDeviceDialog = () => import("./dialog-matter-add-device");
4 |
5 | export const showMatterAddDeviceDialog = (element: HTMLElement): void => {
6 | fireEvent(element, "show-dialog", {
7 | dialogTag: "dialog-matter-add-device",
8 | dialogImport: loadAddDeviceDialog,
9 | dialogParams: {},
10 | });
11 | };
12 |
--------------------------------------------------------------------------------
/src/panels/config/logs/show-dialog-download-logs.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export interface DownloadLogsDialogParams {
4 | header?: string;
5 | provider: string;
6 | defaultLineCount?: number;
7 | boot: number;
8 | }
9 |
10 | export const showDownloadLogsDialog = (
11 | element: HTMLElement,
12 | dialogParams: DownloadLogsDialogParams
13 | ): void => {
14 | fireEvent(element, "show-dialog", {
15 | dialogTag: "dialog-download-logs",
16 | dialogImport: () => import("./dialog-download-logs"),
17 | dialogParams,
18 | });
19 | };
20 |
--------------------------------------------------------------------------------
/src/panels/config/repairs/show-integration-startup-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export const loadIntegrationStartupDialog = () =>
4 | import("./dialog-integration-startup");
5 |
6 | export const showIntegrationStartupDialog = (element: HTMLElement): void => {
7 | fireEvent(element, "show-dialog", {
8 | dialogTag: "dialog-integration-startup",
9 | dialogImport: loadIntegrationStartupDialog,
10 | dialogParams: {},
11 | });
12 | };
13 |
--------------------------------------------------------------------------------
/src/panels/config/repairs/show-system-information-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export const loadSystemInformationDialog = () =>
4 | import("./dialog-system-information");
5 |
6 | export const showSystemInformationDialog = (element: HTMLElement): void => {
7 | fireEvent(element, "show-dialog", {
8 | dialogTag: "dialog-system-information",
9 | dialogImport: loadSystemInformationDialog,
10 | dialogParams: undefined,
11 | });
12 | };
13 |
--------------------------------------------------------------------------------
/src/panels/config/storage/show-dialog-move-datadisk.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 | import type { HassioHostInfo } from "../../../data/hassio/host";
3 |
4 | export interface MoveDatadiskDialogParams {
5 | hostInfo: HassioHostInfo;
6 | }
7 |
8 | export const showMoveDatadiskDialog = (
9 | element: HTMLElement,
10 | dialogParams: MoveDatadiskDialogParams
11 | ): void => {
12 | fireEvent(element, "show-dialog", {
13 | dialogTag: "dialog-move-datadisk",
14 | dialogImport: () => import("./dialog-move-datadisk"),
15 | dialogParams,
16 | });
17 | };
18 |
--------------------------------------------------------------------------------
/src/panels/config/storage/show-dialog-view-mount.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 | import type { SupervisorMount } from "../../../data/supervisor/mounts";
3 |
4 | export interface MountViewDialogParams {
5 | mount?: SupervisorMount;
6 | reloadMounts: () => void;
7 | }
8 |
9 | export const showMountViewDialog = (
10 | element: HTMLElement,
11 | dialogParams: MountViewDialogParams
12 | ): void => {
13 | fireEvent(element, "show-dialog", {
14 | dialogTag: "dialog-mount-view",
15 | dialogImport: () => import("./dialog-mount-view"),
16 | dialogParams,
17 | });
18 | };
19 |
--------------------------------------------------------------------------------
/src/panels/config/users/show-dialog-add-user.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 | import type { User } from "../../../data/user";
3 |
4 | export interface AddUserDialogParams {
5 | userAddedCallback: (user: User) => void;
6 | name?: string;
7 | }
8 |
9 | export const loadAddUserDialog = () => import("./dialog-add-user");
10 |
11 | export const showAddUserDialog = (
12 | element: HTMLElement,
13 | dialogParams: AddUserDialogParams
14 | ): void => {
15 | fireEvent(element, "show-dialog", {
16 | dialogTag: "dialog-add-user",
17 | dialogImport: loadAddUserDialog,
18 | dialogParams,
19 | });
20 | };
21 |
--------------------------------------------------------------------------------
/src/panels/config/users/show-dialog-admin-change-password.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../../common/dom/fire_event";
2 |
3 | export interface AdminChangePasswordDialogParams {
4 | userId: string;
5 | }
6 |
7 | export const loadAdminChangePasswordDialog = () =>
8 | import("./dialog-admin-change-password");
9 |
10 | export const showAdminChangePasswordDialog = (
11 | element: HTMLElement,
12 | dialogParams: AdminChangePasswordDialogParams
13 | ): void => {
14 | fireEvent(element, "show-dialog", {
15 | dialogTag: "dialog-admin-change-password",
16 | dialogImport: loadAdminChangePasswordDialog,
17 | dialogParams,
18 | });
19 | };
20 |
--------------------------------------------------------------------------------
/src/panels/lovelace/card-features/common/filter-modes.ts:
--------------------------------------------------------------------------------
1 | export const filterModes = (
2 | supportedModes: T[] | undefined,
3 | selectedModes: T[] | undefined
4 | ): T[] =>
5 | selectedModes
6 | ? selectedModes.filter((mode) => (supportedModes || []).includes(mode))
7 | : supportedModes || [];
8 |
--------------------------------------------------------------------------------
/src/panels/lovelace/cards/hui-entity-button-card.ts:
--------------------------------------------------------------------------------
1 | import { customElement } from "lit/decorators";
2 | import { HuiButtonCard } from "./hui-button-card";
3 |
4 | @customElement("hui-entity-button-card")
5 | class HuiEntityButtonCard extends HuiButtonCard {
6 | public setConfig(config): void {
7 | if (!config.entity) {
8 | throw new Error("Entity must be specified");
9 | }
10 | super.setConfig(config);
11 | }
12 | }
13 |
14 | declare global {
15 | interface HTMLElementTagNameMap {
16 | "hui-entity-button-card": HuiEntityButtonCard;
17 | }
18 | }
19 |
--------------------------------------------------------------------------------
/src/panels/lovelace/common/entity/toggle-entity.ts:
--------------------------------------------------------------------------------
1 | import { STATES_OFF } from "../../../../common/const";
2 | import type { HomeAssistant, ServiceCallResponse } from "../../../../types";
3 | import { turnOnOffEntity } from "./turn-on-off-entity";
4 |
5 | export const toggleEntity = (
6 | hass: HomeAssistant,
7 | entityId: string
8 | ): Promise => {
9 | const turnOn = STATES_OFF.includes(hass.states[entityId].state);
10 | return turnOnOffEntity(hass, entityId, turnOn);
11 | };
12 |
--------------------------------------------------------------------------------
/src/panels/lovelace/common/has-action.ts:
--------------------------------------------------------------------------------
1 | import type { ActionConfig } from "../../../data/lovelace/config/action";
2 | import type { ConfigEntity } from "../cards/types";
3 |
4 | export function hasAction(config?: ActionConfig): boolean {
5 | return config !== undefined && config.action !== "none";
6 | }
7 |
8 | export function hasAnyAction(config: ConfigEntity): boolean {
9 | return (
10 | !config.tap_action ||
11 | hasAction(config.tap_action) ||
12 | hasAction(config.hold_action) ||
13 | hasAction(config.double_tap_action)
14 | );
15 | }
16 |
--------------------------------------------------------------------------------
/src/panels/lovelace/common/icon-condition.ts:
--------------------------------------------------------------------------------
1 | import {
2 | mdiAccount,
3 | mdiAmpersand,
4 | mdiGateOr,
5 | mdiNumeric,
6 | mdiResponsive,
7 | mdiStateMachine,
8 | } from "@mdi/js";
9 | import type { Condition } from "./validate-condition";
10 |
11 | export const ICON_CONDITION: Record = {
12 | numeric_state: mdiNumeric,
13 | state: mdiStateMachine,
14 | screen: mdiResponsive,
15 | user: mdiAccount,
16 | and: mdiAmpersand,
17 | or: mdiGateOr,
18 | };
19 |
--------------------------------------------------------------------------------
/src/panels/lovelace/common/is-valid-object.ts:
--------------------------------------------------------------------------------
1 | // Check if given obj is a JS object and optionally contains all required keys
2 | export default function isValidObject(obj, requiredKeys = []) {
3 | return (
4 | obj &&
5 | typeof obj === "object" &&
6 | !Array.isArray(obj) &&
7 | requiredKeys.every((k) => k in obj)
8 | );
9 | }
10 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/conditions/types.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../../../../types";
2 | import type { Condition } from "../../common/validate-condition";
3 |
4 | export interface LovelaceConditionEditorConstructor {
5 | defaultConfig?: Condition;
6 | validateUIConfig?: (condition: Condition, hass: HomeAssistant) => void;
7 | }
8 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/gui-support-error.ts:
--------------------------------------------------------------------------------
1 | export class GUISupportError extends Error {
2 | public warnings?: string[];
3 |
4 | public errors?: string[];
5 |
6 | constructor(message: string, warnings?: string[], errors?: string[]) {
7 | super(message);
8 | this.name = "GUISupportError";
9 | this.warnings = warnings;
10 | this.errors = errors;
11 | }
12 | }
13 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/lovelace-badges.ts:
--------------------------------------------------------------------------------
1 | import type { Badge } from "./types";
2 |
3 | export const coreBadges: Badge[] = [
4 | {
5 | type: "entity",
6 | showElement: true,
7 | },
8 | ];
9 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/lovelace-headerfooters.ts:
--------------------------------------------------------------------------------
1 | import {
2 | mdiChartBellCurveCumulative,
3 | mdiGestureTapButton,
4 | mdiImageArea,
5 | } from "@mdi/js";
6 | import type { HeaderFooter } from "./types";
7 |
8 | export const headerFooterElements: HeaderFooter[] = [
9 | { type: "graph", icon: mdiChartBellCurveCumulative },
10 | { type: "buttons", icon: mdiGestureTapButton },
11 | { type: "picture", icon: mdiImageArea },
12 | ];
13 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/process-editor-entities.ts:
--------------------------------------------------------------------------------
1 | import type { EntityConfig } from "../entity-rows/types";
2 |
3 | export function processEditorEntities(
4 | entities: (any | string)[]
5 | ): EntityConfig[] {
6 | return entities.map((entityConf) => {
7 | if (typeof entityConf === "string") {
8 | return { entity: entityConf };
9 | }
10 | return entityConf;
11 | });
12 | }
13 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/structs/base-badge-struct.ts:
--------------------------------------------------------------------------------
1 | import { object, string, any } from "superstruct";
2 |
3 | export const baseLovelaceBadgeConfig = object({
4 | type: string(),
5 | visibility: any(),
6 | });
7 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/structs/base-card-struct.ts:
--------------------------------------------------------------------------------
1 | import { object, string, any } from "superstruct";
2 |
3 | export const baseLovelaceCardConfig = object({
4 | type: string(),
5 | view_layout: any(),
6 | layout_options: any(),
7 | grid_options: any(),
8 | visibility: any(),
9 | });
10 |
--------------------------------------------------------------------------------
/src/panels/lovelace/editor/structs/button-entity-struct.ts:
--------------------------------------------------------------------------------
1 | import { boolean, object, optional, string } from "superstruct";
2 | import { actionConfigStruct } from "./action-struct";
3 |
4 | export const buttonEntityConfigStruct = object({
5 | entity: string(),
6 | name: optional(string()),
7 | icon: optional(string()),
8 | image: optional(string()),
9 | show_name: optional(boolean()),
10 | show_icon: optional(boolean()),
11 | tap_action: optional(actionConfigStruct),
12 | hold_action: optional(actionConfigStruct),
13 | double_tap_action: optional(actionConfigStruct),
14 | });
15 |
--------------------------------------------------------------------------------
/src/panels/lovelace/sections/const.ts:
--------------------------------------------------------------------------------
1 | export const GRID_SECTION_LAYOUT = "grid";
2 | export const DEFAULT_SECTION_LAYOUT = GRID_SECTION_LAYOUT;
3 |
--------------------------------------------------------------------------------
/src/panels/lovelace/views/const.ts:
--------------------------------------------------------------------------------
1 | export const MASONRY_VIEW_LAYOUT = "masonry";
2 | export const PANEL_VIEW_LAYOUT = "panel";
3 | export const SIDEBAR_VIEW_LAYOUT = "sidebar";
4 | export const SECTIONS_VIEW_LAYOUT = "sections";
5 |
6 | export const CARD_LAYOUTS = [
7 | MASONRY_VIEW_LAYOUT,
8 | PANEL_VIEW_LAYOUT,
9 | SIDEBAR_VIEW_LAYOUT,
10 | ];
11 | export const SECTION_VIEW_LAYOUTS = [SECTIONS_VIEW_LAYOUT];
12 |
--------------------------------------------------------------------------------
/src/panels/lovelace/views/default-section.ts:
--------------------------------------------------------------------------------
1 | import type { LocalizeFunc } from "../../../common/translations/localize";
2 |
3 | export const generateDefaultSection = (localize: LocalizeFunc) => ({
4 | type: "grid",
5 | cards: [
6 | {
7 | type: "heading",
8 | heading: localize(
9 | "ui.panel.lovelace.editor.section.default_section_title"
10 | ),
11 | },
12 | ],
13 | });
14 |
--------------------------------------------------------------------------------
/src/panels/lovelace/views/default-view-editable.ts:
--------------------------------------------------------------------------------
1 | // hui-view dependencies for when in edit mode.
2 | import "../../../components/ha-fab";
3 | import "../components/hui-card-options";
4 | import { importCreateCardDialog } from "../editor/card-editor/show-create-card-dialog";
5 | import { importDeleteCardDialog } from "../editor/card-editor/show-delete-card-dialog";
6 | import { importEditCardDialog } from "../editor/card-editor/show-edit-card-dialog";
7 |
8 | importCreateCardDialog();
9 | importDeleteCardDialog();
10 | importEditCardDialog();
11 |
--------------------------------------------------------------------------------
/src/panels/lovelace/views/get-view-type.ts:
--------------------------------------------------------------------------------
1 | import type { LovelaceViewConfig } from "../../../data/lovelace/config/view";
2 | import {
3 | MASONRY_VIEW_LAYOUT,
4 | PANEL_VIEW_LAYOUT,
5 | SECTIONS_VIEW_LAYOUT,
6 | } from "./const";
7 |
8 | export const getViewType = (config?: LovelaceViewConfig): string => {
9 | if (config?.type) {
10 | return config.type;
11 | }
12 | if (config?.panel) {
13 | return PANEL_VIEW_LAYOUT;
14 | }
15 | if (config?.sections) {
16 | return SECTIONS_VIEW_LAYOUT;
17 | }
18 | if (config?.cards) {
19 | return MASONRY_VIEW_LAYOUT;
20 | }
21 | return SECTIONS_VIEW_LAYOUT;
22 | };
23 |
--------------------------------------------------------------------------------
/src/panels/profile/show-long-lived-access-token-dialog.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../../common/dom/fire_event";
2 |
3 | export interface LongLivedAccessTokenDialogParams {
4 | token: string;
5 | name: string;
6 | }
7 |
8 | export const showLongLivedAccessTokenDialog = (
9 | element: HTMLElement,
10 | longLivedAccessTokenDialogParams: LongLivedAccessTokenDialogParams
11 | ): void => {
12 | fireEvent(element, "show-dialog", {
13 | dialogTag: "ha-long-lived-access-token-dialog",
14 | dialogImport: () => import("./ha-long-lived-access-token-dialog"),
15 | dialogParams: longLivedAccessTokenDialogParams,
16 | });
17 | };
18 |
--------------------------------------------------------------------------------
/src/resources/append-ha-style.ts:
--------------------------------------------------------------------------------
1 | import { themeStyles } from "./theme/theme";
2 |
3 | const styleElement = document.createElement("style");
4 | styleElement.textContent = themeStyles;
5 | document.head.append(styleElement);
6 |
--------------------------------------------------------------------------------
/src/resources/custom-card-support.ts:
--------------------------------------------------------------------------------
1 | import { css, html, LitElement } from "lit";
2 |
3 | (LitElement.prototype as any).html = html;
4 | (LitElement.prototype as any).css = css;
5 |
--------------------------------------------------------------------------------
/src/resources/hammer.ts:
--------------------------------------------------------------------------------
1 | export {
2 | DIRECTION_LEFT,
3 | DIRECTION_RIGHT,
4 | Manager,
5 | Swipe,
6 | } from "@egjs/hammerjs";
7 |
--------------------------------------------------------------------------------
/src/resources/icon-metadata.ts:
--------------------------------------------------------------------------------
1 | import * as iconMetadata_ from "../../build/mdi/iconMetadata.json";
2 | import type { IconMetaFile } from "../types.js";
3 |
4 | export const iconMetadata = (iconMetadata_ as any).default as IconMetaFile;
5 |
--------------------------------------------------------------------------------
/src/resources/polyfills/element-getattributenames.ts:
--------------------------------------------------------------------------------
1 | // Source: https://developer.mozilla.org/en-US/docs/Web/API/Element/getAttributeNames
2 | if (Element.prototype.getAttributeNames === undefined) {
3 | Element.prototype.getAttributeNames = function () {
4 | const attributes = this.attributes;
5 | const length = attributes.length;
6 | const result = new Array(length);
7 | for (let i = 0; i < length; i++) {
8 | result[i] = attributes[i].name;
9 | }
10 | return result;
11 | };
12 | }
13 |
--------------------------------------------------------------------------------
/src/resources/polyfills/element-toggleattribute.ts:
--------------------------------------------------------------------------------
1 | // Source: https://gist.github.com/rebelchris/365f26f95d7e9f432f64f21886d9b9ef
2 | if (!Element.prototype.toggleAttribute) {
3 | Element.prototype.toggleAttribute = function (name, force) {
4 | if (force !== undefined) {
5 | force = !!force;
6 | }
7 |
8 | if (this.hasAttribute(name)) {
9 | if (force) {
10 | return true;
11 | }
12 |
13 | this.removeAttribute(name);
14 | return false;
15 | }
16 | if (force === false) {
17 | return false;
18 | }
19 |
20 | this.setAttribute(name, "");
21 | return true;
22 | };
23 | }
24 |
--------------------------------------------------------------------------------
/src/resources/polyfills/resize-observer.ts:
--------------------------------------------------------------------------------
1 | if (typeof window.ResizeObserver !== "function") {
2 | window.ResizeObserver = (
3 | await import(
4 | "@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver"
5 | )
6 | ).default;
7 | }
8 |
9 | export {};
10 |
--------------------------------------------------------------------------------
/src/resources/sortable.ts:
--------------------------------------------------------------------------------
1 | import type Sortable from "sortablejs";
2 | import SortableCore, {
3 | AutoScroll,
4 | OnSpill,
5 | } from "sortablejs/modular/sortable.core.esm";
6 |
7 | SortableCore.mount(OnSpill);
8 | SortableCore.mount(new AutoScroll());
9 |
10 | export default SortableCore as typeof Sortable;
11 |
12 | export type { Sortable as SortableInstance };
13 |
--------------------------------------------------------------------------------
/src/resources/theme/theme.ts:
--------------------------------------------------------------------------------
1 | import { fontStyles } from "../roboto";
2 | import { colorDerivedVariables, colorStyles } from "./color.globals";
3 | import { mainDerivedVariables, mainStyles } from "./main.globals";
4 | import {
5 | typographyDerivedVariables,
6 | typographyStyles,
7 | } from "./typography.globals";
8 |
9 | export const themeStyles = [
10 | mainStyles.toString(),
11 | typographyStyles.toString(),
12 | colorStyles.toString(),
13 | fontStyles.toString(),
14 | ].join("");
15 |
16 | export const derivedStyles = {
17 | ...mainDerivedVariables,
18 | ...typographyDerivedVariables,
19 | ...colorDerivedVariables,
20 | };
21 |
--------------------------------------------------------------------------------
/src/resources/translations-metadata.ts:
--------------------------------------------------------------------------------
1 | import * as translationMetadata_ from "../../build/translations/translationMetadata.json";
2 | import type { TranslationMetadata } from "../types.js";
3 |
4 | export const translationMetadata = (translationMetadata_ as any)
5 | .default as TranslationMetadata;
6 |
--------------------------------------------------------------------------------
/src/resources/virtualizer.ts:
--------------------------------------------------------------------------------
1 | export const loadVirtualizer = async () => {
2 | await import("@lit-labs/virtualizer");
3 | };
4 |
--------------------------------------------------------------------------------
/src/state/notification-mixin.ts:
--------------------------------------------------------------------------------
1 | import type { Constructor } from "../types";
2 | import type { HassBaseEl } from "./hass-base-mixin";
3 |
4 | export default >(superClass: T) =>
5 | class extends superClass {
6 | protected firstUpdated(changedProps) {
7 | super.firstUpdated(changedProps);
8 | // @ts-ignore
9 | this.registerDialog({
10 | dialogShowEvent: "hass-notification",
11 | dialogTag: "notification-manager",
12 | dialogImport: () => import("../managers/notification-manager"),
13 | addHistory: false,
14 | });
15 | }
16 | };
17 |
--------------------------------------------------------------------------------
/src/types/echarts.d.ts:
--------------------------------------------------------------------------------
1 | declare module "echarts/lib/chart/graph/install" {
2 | export const install: EChartsExtensionInstaller;
3 | }
4 |
--------------------------------------------------------------------------------
/src/types/node-vibrant.d.ts:
--------------------------------------------------------------------------------
1 | // eslint-disable-next-line no-var
2 | declare var Vibrant: Any;
3 | declare module "node-vibrant" {
4 | export default Vibrant;
5 | }
6 |
--------------------------------------------------------------------------------
/src/util/bytes-to-string.ts:
--------------------------------------------------------------------------------
1 | export const bytesToString = (value = 0, decimals = 2): string => {
2 | if (value === 0) {
3 | return "0 Bytes";
4 | }
5 | const k = 1024;
6 | decimals = decimals < 0 ? 0 : decimals;
7 | const sizes = ["Bytes", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"];
8 | const i = Math.floor(Math.log(value) / Math.log(k));
9 | return `${parseFloat((value / k ** i).toFixed(decimals))} ${sizes[i]}`;
10 | };
11 |
--------------------------------------------------------------------------------
/src/util/custom-panel/create-custom-panel-element.ts:
--------------------------------------------------------------------------------
1 | export const createCustomPanelElement = (panelConfig) => {
2 | // Legacy support. Custom panels used to have to define element ha-panel-{name}
3 | const tagName =
4 | "html_url" in panelConfig
5 | ? `ha-panel-${panelConfig.name}`
6 | : panelConfig.name;
7 | return document.createElement(tagName);
8 | };
9 |
--------------------------------------------------------------------------------
/src/util/custom-panel/set-custom-panel-properties.ts:
--------------------------------------------------------------------------------
1 | export const setCustomPanelProperties = (root, properties) => {
2 | if ("setProperties" in root) {
3 | root.setProperties(properties);
4 | } else {
5 | Object.keys(properties).forEach((key) => {
6 | root[key] = properties[key];
7 | });
8 | }
9 | };
10 |
--------------------------------------------------------------------------------
/src/util/documentation-url.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 |
3 | export const documentationUrl = (hass: HomeAssistant, path: string) =>
4 | `https://${
5 | hass.config.version.includes("b")
6 | ? "rc"
7 | : hass.config.version.includes("dev")
8 | ? "next"
9 | : "www"
10 | }.home-assistant.io${path}`;
11 |
--------------------------------------------------------------------------------
/src/util/empty.js:
--------------------------------------------------------------------------------
1 | /* empty file that we alias some files to that we don't want to include */
2 |
3 | export {}; // for Babel to treat as a module
4 |
--------------------------------------------------------------------------------
/src/util/fetch-with-auth.ts:
--------------------------------------------------------------------------------
1 | import type { Auth } from "home-assistant-js-websocket";
2 |
3 | export const fetchWithAuth = async (
4 | auth: Auth,
5 | input: RequestInfo,
6 | init: RequestInit = {}
7 | ) => {
8 | if (auth.expired) {
9 | await auth.refreshAccessToken();
10 | }
11 | init.credentials = "same-origin";
12 | if (!init.headers) {
13 | init.headers = {};
14 | }
15 | if (!init.headers) {
16 | init.headers = {};
17 | }
18 | // @ts-ignore
19 | init.headers.authorization = `Bearer ${auth.accessToken}`;
20 | return fetch(input, init);
21 | };
22 |
--------------------------------------------------------------------------------
/src/util/iframe.ts:
--------------------------------------------------------------------------------
1 | export const IFRAME_SANDBOX =
2 | "allow-forms allow-popups allow-pointer-lock allow-same-origin allow-scripts allow-modals allow-downloads";
3 |
--------------------------------------------------------------------------------
/src/util/is_ios.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 | import { isSafari } from "./is_safari";
3 |
4 | export const isIosApp = (hass: HomeAssistant): boolean =>
5 | !!hass.auth.external && isSafari;
6 |
--------------------------------------------------------------------------------
/src/util/is_mac.ts:
--------------------------------------------------------------------------------
1 | export const isMac = /Mac/i.test(navigator.userAgent);
2 |
--------------------------------------------------------------------------------
/src/util/is_mobile.ts:
--------------------------------------------------------------------------------
1 | export const isMobileClient = /(?:iphone|android|ipad)/i.test(
2 | navigator.userAgent
3 | );
4 |
--------------------------------------------------------------------------------
/src/util/is_safari.ts:
--------------------------------------------------------------------------------
1 | export const isSafari = /^((?!chrome|android).)*safari/i.test(
2 | navigator.userAgent
3 | );
4 |
--------------------------------------------------------------------------------
/src/util/is_touch.ts:
--------------------------------------------------------------------------------
1 | export const isTouch =
2 | "ontouchstart" in window ||
3 | navigator.maxTouchPoints > 0 ||
4 | // @ts-ignore
5 | navigator.msMaxTouchPoints > 0;
6 |
--------------------------------------------------------------------------------
/src/util/launch-screen.ts:
--------------------------------------------------------------------------------
1 | import type { TemplateResult } from "lit";
2 | import { render } from "lit";
3 |
4 | export const removeLaunchScreen = () => {
5 | const launchScreenElement = document.getElementById("ha-launch-screen");
6 | if (launchScreenElement) {
7 | launchScreenElement.parentElement!.removeChild(launchScreenElement);
8 | }
9 | };
10 |
11 | export const renderLaunchScreenInfoBox = (content: TemplateResult) => {
12 | const infoBoxElement = document.getElementById("ha-launch-screen-info-box");
13 | if (infoBoxElement) {
14 | render(content, infoBoxElement);
15 | }
16 | };
17 |
--------------------------------------------------------------------------------
/src/util/toast-saved-success.ts:
--------------------------------------------------------------------------------
1 | import type { HomeAssistant } from "../types";
2 | import { showToast } from "./toast";
3 |
4 | export const showSaveSuccessToast = (el: HTMLElement, hass: HomeAssistant) =>
5 | showToast(el, {
6 | message: hass!.localize("ui.common.successfully_saved"),
7 | });
8 |
--------------------------------------------------------------------------------
/src/util/toast.ts:
--------------------------------------------------------------------------------
1 | import { fireEvent } from "../common/dom/fire_event";
2 | import type { ShowToastParams } from "../managers/notification-manager";
3 |
4 | export const showToast = (el: HTMLElement, params: ShowToastParams) =>
5 | fireEvent(el, "hass-notification", params);
6 |
--------------------------------------------------------------------------------
/src/util/url.ts:
--------------------------------------------------------------------------------
1 | export function obfuscateUrl(url: string) {
2 | if (url.endsWith(".ui.nabu.casa")) {
3 | return "https://•••••••••••••••••.ui.nabu.casa";
4 | }
5 | // hide any words that look like they might be a hostname or IP address
6 | return url.replace(/(?<=:\/\/)[\w-]+|(?<=\.)[\w-]+/g, (match) =>
7 | "•".repeat(match.length)
8 | );
9 | }
10 |
--------------------------------------------------------------------------------
/test/common/config/is_component_loaded.test.ts:
--------------------------------------------------------------------------------
1 | import { describe, it, expect } from "vitest";
2 | import { isComponentLoaded } from "../../../src/common/config/is_component_loaded";
3 | import type { HomeAssistant } from "../../../src/types";
4 |
5 | describe("isComponentLoaded", () => {
6 | it("should return if the component is loaded", () => {
7 | const hass = {
8 | config: { components: ["test_component"] },
9 | } as unknown as HomeAssistant;
10 | expect(isComponentLoaded(hass, "test_component")).toBe(true);
11 | expect(isComponentLoaded(hass, "other_component")).toBe(false);
12 | });
13 | });
14 |
--------------------------------------------------------------------------------
/test/common/datetime/duration_to_seconds.test.ts:
--------------------------------------------------------------------------------
1 | import { assert, describe, it } from "vitest";
2 |
3 | import durationToSeconds from "../../../src/common/datetime/duration_to_seconds";
4 |
5 | describe("durationToSeconds", () => {
6 | it("works", () => {
7 | assert.strictEqual(durationToSeconds("0:01:05"), 65);
8 | assert.strictEqual(durationToSeconds("11:01:05"), 39665);
9 | });
10 | });
11 |
--------------------------------------------------------------------------------
/test/common/datetime/resolve-time-zone.test.ts:
--------------------------------------------------------------------------------
1 | import { expect, test } from "vitest";
2 | import {
3 | LOCAL_TIME_ZONE,
4 | resolveTimeZone,
5 | } from "../../../src/common/datetime/resolve-time-zone";
6 | import { TimeZone } from "../../../src/data/translation";
7 |
8 | test("resolveTimeZone", () => {
9 | const serverTimeZone = "Vienna/Austria";
10 | expect(resolveTimeZone(TimeZone.local, serverTimeZone)).toBe(LOCAL_TIME_ZONE);
11 | expect(resolveTimeZone(TimeZone.server, serverTimeZone)).toBe(serverTimeZone);
12 | });
13 |
--------------------------------------------------------------------------------
/test/common/entity/compute_domain.test.ts:
--------------------------------------------------------------------------------
1 | import { assert, describe, it } from "vitest";
2 |
3 | import { computeDomain } from "../../../src/common/entity/compute_domain";
4 |
5 | describe("computeDomain", () => {
6 | it("Returns domains", () => {
7 | assert.equal(computeDomain("sensor.bla"), "sensor");
8 | assert.equal(computeDomain("switch.bla"), "switch");
9 | assert.equal(computeDomain("light.bla"), "light");
10 | assert.equal(
11 | computeDomain("persistent_notification.bla"),
12 | "persistent_notification"
13 | );
14 | });
15 | });
16 |
--------------------------------------------------------------------------------
/test/common/entity/compute_state_domain.test.ts:
--------------------------------------------------------------------------------
1 | import { assert, describe, it } from "vitest";
2 |
3 | import { computeStateDomain } from "../../../src/common/entity/compute_state_domain";
4 |
5 | describe("computeStateDomain", () => {
6 | it("Detects sensor domain", () => {
7 | const stateObj: any = {
8 | entity_id: "sensor.test",
9 | };
10 | assert.strictEqual(computeStateDomain(stateObj), "sensor");
11 | });
12 | });
13 |
--------------------------------------------------------------------------------
/test/common/entity/state_color.ts:
--------------------------------------------------------------------------------
1 | export {}; // for Babel to treat as a module
2 |
--------------------------------------------------------------------------------
/test/common/string/is_date.test.ts:
--------------------------------------------------------------------------------
1 | import { expect, test } from "vitest";
2 | import { isDate } from "../../../src/common/string/is_date";
3 |
4 | test("isDate", () => {
5 | expect(isDate("ABC")).toBe(false);
6 |
7 | expect(isDate("2021-02-03", false)).toBe(true);
8 | expect(isDate("2021-02-03", true)).toBe(true);
9 |
10 | expect(isDate("2021-05-25T19:23:52+00:00", true)).toBe(true);
11 | expect(isDate("2021-05-25T19:23:52+00:00", false)).toBe(false);
12 | });
13 |
--------------------------------------------------------------------------------
/test/setup.ts:
--------------------------------------------------------------------------------
1 | import { beforeAll } from "vitest";
2 |
3 | beforeAll(() => {
4 | global.window = {} as any;
5 | global.navigator = {} as any;
6 |
7 | global.__DEMO__ = false;
8 | });
9 |
--------------------------------------------------------------------------------