mirror of
https://github.com/home-assistant/frontend.git
synced 2025-07-27 11:16:35 +00:00
Merge pull request #12233 from home-assistant/dev
Co-authored-by: Joakim Sørensen <ludeeus@ludeeus.dev> Co-authored-by: Philip Allgaier <mail@spacegaier.de> Co-authored-by: Zack Barett <zackbarett@hey.com>
This commit is contained in:
commit
d02cd122a9
@ -261,6 +261,8 @@ class DemoHaSelector extends LitElement implements ProvideHassElement {
|
|||||||
|
|
||||||
@state() private _required = false;
|
@state() private _required = false;
|
||||||
|
|
||||||
|
@state() private _helper = false;
|
||||||
|
|
||||||
@state() private _label = true;
|
@state() private _label = true;
|
||||||
|
|
||||||
private data = SCHEMAS.map(() => ({}));
|
private data = SCHEMAS.map(() => ({}));
|
||||||
@ -418,6 +420,13 @@ class DemoHaSelector extends LitElement implements ProvideHassElement {
|
|||||||
@change=${this._handleOptionChange}
|
@change=${this._handleOptionChange}
|
||||||
></ha-switch>
|
></ha-switch>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
|
<ha-formfield label="Helper text">
|
||||||
|
<ha-switch
|
||||||
|
.name=${"helper"}
|
||||||
|
.checked=${this._helper}
|
||||||
|
@change=${this._handleOptionChange}
|
||||||
|
></ha-switch>
|
||||||
|
</ha-formfield>
|
||||||
</div>
|
</div>
|
||||||
${SCHEMAS.map((info, idx) => {
|
${SCHEMAS.map((info, idx) => {
|
||||||
const data = this.data[idx];
|
const data = this.data[idx];
|
||||||
@ -446,6 +455,7 @@ class DemoHaSelector extends LitElement implements ProvideHassElement {
|
|||||||
.disabled=${this._disabled}
|
.disabled=${this._disabled}
|
||||||
.required=${this._required}
|
.required=${this._required}
|
||||||
@value-changed=${valueChanged}
|
@value-changed=${valueChanged}
|
||||||
|
.helper=${this._helper ? "Helper text" : undefined}
|
||||||
></ha-selector>
|
></ha-selector>
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
`
|
`
|
||||||
@ -466,7 +476,8 @@ class DemoHaSelector extends LitElement implements ProvideHassElement {
|
|||||||
width: 60;
|
width: 60;
|
||||||
}
|
}
|
||||||
.options {
|
.options {
|
||||||
padding: 16px 48px;
|
max-width: 800px;
|
||||||
|
margin: 16px auto;
|
||||||
}
|
}
|
||||||
.options ha-formfield {
|
.options ha-formfield {
|
||||||
margin-right: 16px;
|
margin-right: 16px;
|
||||||
|
@ -180,7 +180,7 @@ export class SupervisorBackupContent extends LitElement {
|
|||||||
>
|
>
|
||||||
<ha-checkbox
|
<ha-checkbox
|
||||||
.checked=${this.homeAssistant}
|
.checked=${this.homeAssistant}
|
||||||
@click=${this.toggleHomeAssistant}
|
@change=${this.toggleHomeAssistant}
|
||||||
>
|
>
|
||||||
</ha-checkbox>
|
</ha-checkbox>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[metadata]
|
[metadata]
|
||||||
name = home-assistant-frontend
|
name = home-assistant-frontend
|
||||||
version = 20220401.0
|
version = 20220405.0
|
||||||
author = The Home Assistant Authors
|
author = The Home Assistant Authors
|
||||||
author_email = hello@home-assistant.io
|
author_email = hello@home-assistant.io
|
||||||
license = Apache-2.0
|
license = Apache-2.0
|
||||||
|
@ -52,6 +52,8 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public value?: string;
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property() public devices?: DeviceRegistryEntry[];
|
@property() public devices?: DeviceRegistryEntry[];
|
||||||
|
|
||||||
@property() public areas?: AreaRegistryEntry[];
|
@property() public areas?: AreaRegistryEntry[];
|
||||||
@ -269,6 +271,7 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
? this.hass.localize("ui.components.device-picker.device")
|
? this.hass.localize("ui.components.device-picker.device")
|
||||||
: this.label}
|
: this.label}
|
||||||
.value=${this._value}
|
.value=${this._value}
|
||||||
|
.helper=${this.helper}
|
||||||
.renderer=${rowRenderer}
|
.renderer=${rowRenderer}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
@ -11,6 +11,8 @@ class HaDevicesPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public value?: string[];
|
@property() public value?: string[];
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required?: boolean;
|
@property({ type: Boolean }) public required?: boolean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -64,6 +66,7 @@ class HaDevicesPicker extends LitElement {
|
|||||||
<div>
|
<div>
|
||||||
<ha-device-picker
|
<ha-device-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.helper=${this.helper}
|
||||||
.includeDomains=${this.includeDomains}
|
.includeDomains=${this.includeDomains}
|
||||||
.excludeDomains=${this.excludeDomains}
|
.excludeDomains=${this.excludeDomains}
|
||||||
.includeDeviceClasses=${this.includeDeviceClasses}
|
.includeDeviceClasses=${this.includeDeviceClasses}
|
||||||
|
@ -16,6 +16,8 @@ class HaEntitiesPickerLight extends LitElement {
|
|||||||
|
|
||||||
@property({ type: Boolean }) public required?: boolean;
|
@property({ type: Boolean }) public required?: boolean;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show entities from specific domains.
|
* Show entities from specific domains.
|
||||||
* @type {string}
|
* @type {string}
|
||||||
@ -110,6 +112,7 @@ class HaEntitiesPickerLight extends LitElement {
|
|||||||
.includeUnitOfMeasurement=${this.includeUnitOfMeasurement}
|
.includeUnitOfMeasurement=${this.includeUnitOfMeasurement}
|
||||||
.entityFilter=${this._entityFilter}
|
.entityFilter=${this._entityFilter}
|
||||||
.label=${this.pickEntityLabel}
|
.label=${this.pickEntityLabel}
|
||||||
|
.helper=${this.helper}
|
||||||
.required=${this.required && !currentEntities.length}
|
.required=${this.required && !currentEntities.length}
|
||||||
@value-changed=${this._addEntity}
|
@value-changed=${this._addEntity}
|
||||||
></ha-entity-picker>
|
></ha-entity-picker>
|
||||||
|
@ -28,6 +28,8 @@ class HaEntityAttributePicker extends LitElement {
|
|||||||
|
|
||||||
@property() public value?: string;
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) private _opened = false;
|
@property({ type: Boolean }) private _opened = false;
|
||||||
|
|
||||||
@query("ha-combo-box", true) private _comboBox!: HaComboBox;
|
@query("ha-combo-box", true) private _comboBox!: HaComboBox;
|
||||||
@ -64,6 +66,7 @@ class HaEntityAttributePicker extends LitElement {
|
|||||||
)}
|
)}
|
||||||
.disabled=${this.disabled || !this.entityId}
|
.disabled=${this.disabled || !this.entityId}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
.helper=${this.helper}
|
||||||
.allowCustomValue=${this.allowCustomValue}
|
.allowCustomValue=${this.allowCustomValue}
|
||||||
item-value-path="value"
|
item-value-path="value"
|
||||||
item-label-path="label"
|
item-label-path="label"
|
||||||
|
@ -48,6 +48,8 @@ export class HaEntityPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public value?: string;
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show entities from specific domains.
|
* Show entities from specific domains.
|
||||||
* @type {Array}
|
* @type {Array}
|
||||||
@ -304,6 +306,7 @@ export class HaEntityPicker extends LitElement {
|
|||||||
.label=${this.label === undefined
|
.label=${this.label === undefined
|
||||||
? this.hass.localize("ui.components.entity.entity-picker.entity")
|
? this.hass.localize("ui.components.entity.entity-picker.entity")
|
||||||
: this.label}
|
: this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.allowCustomValue=${this.allowCustomEntity}
|
.allowCustomValue=${this.allowCustomEntity}
|
||||||
.filteredItems=${this._states}
|
.filteredItems=${this._states}
|
||||||
.renderer=${rowRenderer}
|
.renderer=${rowRenderer}
|
||||||
|
@ -29,6 +29,8 @@ class HaAddonPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public value = "";
|
@property() public value = "";
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@state() private _addons?: HassioAddonInfo[];
|
@state() private _addons?: HassioAddonInfo[];
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
@ -62,6 +64,7 @@ class HaAddonPicker extends LitElement {
|
|||||||
.value=${this._value}
|
.value=${this._value}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
|
.helper=${this.helper}
|
||||||
.renderer=${rowRenderer}
|
.renderer=${rowRenderer}
|
||||||
.items=${this._addons}
|
.items=${this._addons}
|
||||||
item-value-path="slug"
|
item-value-path="slug"
|
||||||
|
@ -49,6 +49,8 @@ export class HaAreaPicker extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public value?: string;
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property() public placeholder?: string;
|
@property() public placeholder?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, attribute: "no-add" })
|
@property({ type: Boolean, attribute: "no-add" })
|
||||||
@ -312,6 +314,7 @@ export class HaAreaPicker extends SubscribeMixin(LitElement) {
|
|||||||
return html`
|
return html`
|
||||||
<ha-combo-box
|
<ha-combo-box
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.helper=${this.helper}
|
||||||
item-value-path="area_id"
|
item-value-path="area_id"
|
||||||
item-id-path="area_id"
|
item-id-path="area_id"
|
||||||
item-label-path="name"
|
item-label-path="name"
|
||||||
|
@ -15,6 +15,8 @@ export class HaAreasPicker extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public value?: string[];
|
@property() public value?: string[];
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property() public placeholder?: string;
|
@property() public placeholder?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, attribute: "no-add" })
|
@property({ type: Boolean, attribute: "no-add" })
|
||||||
@ -90,6 +92,7 @@ export class HaAreasPicker extends SubscribeMixin(LitElement) {
|
|||||||
.noAdd=${this.noAdd}
|
.noAdd=${this.noAdd}
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.label=${this.pickAreaLabel}
|
.label=${this.pickAreaLabel}
|
||||||
|
.helper=${this.helper}
|
||||||
.includeDomains=${this.includeDomains}
|
.includeDomains=${this.includeDomains}
|
||||||
.excludeDomains=${this.excludeDomains}
|
.excludeDomains=${this.excludeDomains}
|
||||||
.includeDeviceClasses=${this.includeDeviceClasses}
|
.includeDeviceClasses=${this.includeDeviceClasses}
|
||||||
|
@ -5,6 +5,7 @@ import { fireEvent } from "../common/dom/fire_event";
|
|||||||
import { stopPropagation } from "../common/dom/stop_propagation";
|
import { stopPropagation } from "../common/dom/stop_propagation";
|
||||||
import "./ha-select";
|
import "./ha-select";
|
||||||
import "./ha-textfield";
|
import "./ha-textfield";
|
||||||
|
import "./ha-input-helper-text";
|
||||||
|
|
||||||
export interface TimeChangedEvent {
|
export interface TimeChangedEvent {
|
||||||
days?: number;
|
days?: number;
|
||||||
@ -253,7 +254,9 @@ export class HaBaseTimeInput extends LitElement {
|
|||||||
<mwc-list-item value="PM">PM</mwc-list-item>
|
<mwc-list-item value="PM">PM</mwc-list-item>
|
||||||
</ha-select>`}
|
</ha-select>`}
|
||||||
</div>
|
</div>
|
||||||
${this.helper ? html`<div class="helper">${this.helper}</div>` : ""}
|
${this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: ""}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -350,13 +353,6 @@ export class HaBaseTimeInput extends LitElement {
|
|||||||
color: var(--mdc-theme-text-primary-on-background, rgba(0, 0, 0, 0.87));
|
color: var(--mdc-theme-text-primary-on-background, rgba(0, 0, 0, 0.87));
|
||||||
padding-left: 4px;
|
padding-left: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.helper {
|
|
||||||
color: var(--mdc-text-field-label-ink-color, rgba(0, 0, 0, 0.6));
|
|
||||||
font-size: 0.75rem;
|
|
||||||
padding-left: 16px;
|
|
||||||
padding-right: 16px;
|
|
||||||
}
|
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -64,6 +64,8 @@ export class HaComboBox extends LitElement {
|
|||||||
|
|
||||||
@property() public validationMessage?: string;
|
@property() public validationMessage?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ attribute: "error-message" }) public errorMessage?: string;
|
@property({ attribute: "error-message" }) public errorMessage?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public invalid?: boolean;
|
@property({ type: Boolean }) public invalid?: boolean;
|
||||||
@ -147,6 +149,8 @@ export class HaComboBox extends LitElement {
|
|||||||
.suffix=${html`<div style="width: 28px;"></div>`}
|
.suffix=${html`<div style="width: 28px;"></div>`}
|
||||||
.icon=${this.icon}
|
.icon=${this.icon}
|
||||||
.invalid=${this.invalid}
|
.invalid=${this.invalid}
|
||||||
|
.helper=${this.helper}
|
||||||
|
helperPersistent
|
||||||
>
|
>
|
||||||
<slot name="icon" slot="leadingIcon"></slot>
|
<slot name="icon" slot="leadingIcon"></slot>
|
||||||
</ha-textfield>
|
</ha-textfield>
|
||||||
|
@ -39,11 +39,15 @@ export class HaDateInput extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return html`<ha-textfield
|
return html`<ha-textfield
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
iconTrailing
|
iconTrailing
|
||||||
|
helperPersistent
|
||||||
@click=${this._openDialog}
|
@click=${this._openDialog}
|
||||||
.value=${this.value
|
.value=${this.value
|
||||||
? formatDateNumeric(new Date(this.value), this.locale)
|
? formatDateNumeric(new Date(this.value), this.locale)
|
||||||
|
@ -77,7 +77,7 @@ export class HaForm extends LitElement implements HaFormElement {
|
|||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<div class="root">
|
<div class="root" part="root">
|
||||||
${this.error && this.error.base
|
${this.error && this.error.base
|
||||||
? html`
|
? html`
|
||||||
<ha-alert alert-type="error">
|
<ha-alert alert-type="error">
|
||||||
@ -173,7 +173,6 @@ export class HaForm extends LitElement implements HaFormElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
// .root has overflow: auto to avoid margin collapse
|
|
||||||
return css`
|
return css`
|
||||||
.root {
|
.root {
|
||||||
margin-bottom: -24px;
|
margin-bottom: -24px;
|
||||||
|
@ -31,6 +31,8 @@ export class HaIconPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property() public placeholder?: string;
|
@property() public placeholder?: string;
|
||||||
|
|
||||||
@property() public fallbackPath?: string;
|
@property() public fallbackPath?: string;
|
||||||
@ -57,6 +59,7 @@ export class HaIconPicker extends LitElement {
|
|||||||
allow-custom-value
|
allow-custom-value
|
||||||
.filteredItems=${iconItems}
|
.filteredItems=${iconItems}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
.placeholder=${this.placeholder}
|
.placeholder=${this.placeholder}
|
||||||
|
25
src/components/ha-input-helper-text.ts
Normal file
25
src/components/ha-input-helper-text.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { css, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement } from "lit/decorators";
|
||||||
|
|
||||||
|
@customElement("ha-input-helper-text")
|
||||||
|
class InputHelperText extends LitElement {
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`<slot></slot>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static styles = css`
|
||||||
|
:host {
|
||||||
|
display: block;
|
||||||
|
color: var(--mdc-text-field-label-ink-color, rgba(0, 0, 0, 0.6));
|
||||||
|
font-size: 0.75rem;
|
||||||
|
padding-left: 16px;
|
||||||
|
padding-right: 16px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-input-helper-text": InputHelperText;
|
||||||
|
}
|
||||||
|
}
|
@ -46,6 +46,9 @@ class HaLabeledSlider extends PolymerElement {
|
|||||||
value="{{value}}"
|
value="{{value}}"
|
||||||
></ha-slider>
|
></ha-slider>
|
||||||
</div>
|
</div>
|
||||||
|
<template is="dom-if" if="[[helper]]">
|
||||||
|
<ha-input-helper-text>[[helper]]</ha-input-helper-text>
|
||||||
|
</template>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -62,6 +65,7 @@ class HaLabeledSlider extends PolymerElement {
|
|||||||
max: Number,
|
max: Number,
|
||||||
pin: Boolean,
|
pin: Boolean,
|
||||||
step: Number,
|
step: Number,
|
||||||
|
helper: String,
|
||||||
|
|
||||||
extra: {
|
extra: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
|
@ -3,7 +3,6 @@ import { UnsubscribeFunc } from "home-assistant-js-websocket";
|
|||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { computeDomain } from "../common/entity/compute_domain";
|
|
||||||
import { subscribeNotifications } from "../data/persistent_notification";
|
import { subscribeNotifications } from "../data/persistent_notification";
|
||||||
import { HomeAssistant } from "../types";
|
import { HomeAssistant } from "../types";
|
||||||
import "./ha-icon-button";
|
import "./ha-icon-button";
|
||||||
@ -43,18 +42,15 @@ class HaMenuButton extends LitElement {
|
|||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
const hasNotifications =
|
const hasNotifications =
|
||||||
(this.narrow || this.hass.dockedSidebar === "always_hidden") &&
|
this._hasNotifications &&
|
||||||
(this._hasNotifications ||
|
(this.narrow || this.hass.dockedSidebar === "always_hidden");
|
||||||
Object.keys(this.hass.states).some(
|
|
||||||
(entityId) => computeDomain(entityId) === "configurator"
|
|
||||||
));
|
|
||||||
return html`
|
return html`
|
||||||
<ha-icon-button
|
<ha-icon-button
|
||||||
.label=${this.hass.localize("ui.sidebar.sidebar_toggle")}
|
.label=${this.hass.localize("ui.sidebar.sidebar_toggle")}
|
||||||
.path=${mdiMenu}
|
.path=${mdiMenu}
|
||||||
@click=${this._toggleMenu}
|
@click=${this._toggleMenu}
|
||||||
></ha-icon-button>
|
></ha-icon-button>
|
||||||
${hasNotifications ? html` <div class="dot"></div> ` : ""}
|
${hasNotifications ? html`<div class="dot"></div>` : ""}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,6 +14,8 @@ export class HaAddonSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -23,6 +25,7 @@ export class HaAddonSelector extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
allow-custom-entity
|
allow-custom-entity
|
||||||
|
@ -18,6 +18,8 @@ export class HaAreaSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@state() public _configEntries?: ConfigEntry[];
|
@state() public _configEntries?: ConfigEntry[];
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
@ -47,6 +49,7 @@ export class HaAreaSelector extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
no-add
|
no-add
|
||||||
.deviceFilter=${this._filterDevices}
|
.deviceFilter=${this._filterDevices}
|
||||||
.entityFilter=${this._filterEntities}
|
.entityFilter=${this._filterEntities}
|
||||||
@ -66,6 +69,7 @@ export class HaAreaSelector extends LitElement {
|
|||||||
<ha-areas-picker
|
<ha-areas-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
|
.helper=${this.helper}
|
||||||
.pickAreaLabel=${this.label}
|
.pickAreaLabel=${this.label}
|
||||||
no-add
|
no-add
|
||||||
.deviceFilter=${this._filterDevices}
|
.deviceFilter=${this._filterDevices}
|
||||||
|
@ -16,6 +16,8 @@ export class HaSelectorAttribute extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -32,6 +34,7 @@ export class HaSelectorAttribute extends SubscribeMixin(LitElement) {
|
|||||||
this.context?.filter_entity}
|
this.context?.filter_entity}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
allow-custom-value
|
allow-custom-value
|
||||||
|
@ -4,6 +4,7 @@ import { fireEvent } from "../../common/dom/fire_event";
|
|||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import "../ha-formfield";
|
import "../ha-formfield";
|
||||||
import "../ha-switch";
|
import "../ha-switch";
|
||||||
|
import "../ha-input-helper-text";
|
||||||
|
|
||||||
@customElement("ha-selector-boolean")
|
@customElement("ha-selector-boolean")
|
||||||
export class HaBooleanSelector extends LitElement {
|
export class HaBooleanSelector extends LitElement {
|
||||||
@ -13,16 +14,23 @@ export class HaBooleanSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
return html`<ha-formfield alignEnd spaceBetween .label=${this.label}>
|
return html`
|
||||||
<ha-switch
|
<ha-formfield alignEnd spaceBetween .label=${this.label}>
|
||||||
.checked=${this.value}
|
<ha-switch
|
||||||
@change=${this._handleChange}
|
.checked=${this.value}
|
||||||
.disabled=${this.disabled}
|
@change=${this._handleChange}
|
||||||
></ha-switch>
|
.disabled=${this.disabled}
|
||||||
</ha-formfield>`;
|
></ha-switch>
|
||||||
|
</ha-formfield>
|
||||||
|
${this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: ""}
|
||||||
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleChange(ev) {
|
private _handleChange(ev) {
|
||||||
@ -35,12 +43,10 @@ export class HaBooleanSelector extends LitElement {
|
|||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
|
||||||
height: 56px;
|
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
ha-formfield {
|
ha-formfield {
|
||||||
width: 100%;
|
display: flex;
|
||||||
|
height: 56px;
|
||||||
|
align-items: center;
|
||||||
--mdc-typography-body2-font-size: 1em;
|
--mdc-typography-body2-font-size: 1em;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
@ -16,6 +16,8 @@ export class HaColorRGBSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -24,9 +26,11 @@ export class HaColorRGBSelector extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<ha-textfield
|
<ha-textfield
|
||||||
type="color"
|
type="color"
|
||||||
|
helperPersistent
|
||||||
.value=${this.value ? rgb2hex(this.value as any) : ""}
|
.value=${this.value ? rgb2hex(this.value as any) : ""}
|
||||||
.label=${this.label || ""}
|
.label=${this.label || ""}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
.helper=${this.helper}
|
||||||
.disalbled=${this.disabled}
|
.disalbled=${this.disabled}
|
||||||
@change=${this._valueChanged}
|
@change=${this._valueChanged}
|
||||||
></ha-textfield>
|
></ha-textfield>
|
||||||
|
@ -15,6 +15,8 @@ export class HaColorTempSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -29,6 +31,7 @@ export class HaColorTempSelector extends LitElement {
|
|||||||
.max=${this.selector.color_temp.max_mireds ?? 500}
|
.max=${this.selector.color_temp.max_mireds ?? 500}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
|
.helper=${this.helper}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
@change=${this._valueChanged}
|
@change=${this._valueChanged}
|
||||||
></ha-labeled-slider>
|
></ha-labeled-slider>
|
||||||
|
@ -14,6 +14,8 @@ export class HaDateSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -26,6 +28,7 @@ export class HaDateSelector extends LitElement {
|
|||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
.helper=${this.helper}
|
||||||
>
|
>
|
||||||
</ha-date-input>
|
</ha-date-input>
|
||||||
`;
|
`;
|
||||||
|
@ -6,6 +6,7 @@ import type { HomeAssistant } from "../../types";
|
|||||||
import "../ha-date-input";
|
import "../ha-date-input";
|
||||||
import type { HaDateInput } from "../ha-date-input";
|
import type { HaDateInput } from "../ha-date-input";
|
||||||
import "../ha-time-input";
|
import "../ha-time-input";
|
||||||
|
import "../ha-input-helper-text";
|
||||||
import type { HaTimeInput } from "../ha-time-input";
|
import type { HaTimeInput } from "../ha-time-input";
|
||||||
|
|
||||||
@customElement("ha-selector-datetime")
|
@customElement("ha-selector-datetime")
|
||||||
@ -18,6 +19,8 @@ export class HaDateTimeSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -30,23 +33,28 @@ export class HaDateTimeSelector extends LitElement {
|
|||||||
const values = this.value?.split(" ");
|
const values = this.value?.split(" ");
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
<ha-date-input
|
<div class="input">
|
||||||
.label=${this.label}
|
<ha-date-input
|
||||||
.locale=${this.hass.locale}
|
.label=${this.label}
|
||||||
.disabled=${this.disabled}
|
.locale=${this.hass.locale}
|
||||||
.required=${this.required}
|
.disabled=${this.disabled}
|
||||||
.value=${values?.[0]}
|
.required=${this.required}
|
||||||
@value-changed=${this._valueChanged}
|
.value=${values?.[0]}
|
||||||
>
|
@value-changed=${this._valueChanged}
|
||||||
</ha-date-input>
|
>
|
||||||
<ha-time-input
|
</ha-date-input>
|
||||||
enable-second
|
<ha-time-input
|
||||||
.value=${values?.[1] || "0:00:00"}
|
enable-second
|
||||||
.locale=${this.hass.locale}
|
.value=${values?.[1] || "0:00:00"}
|
||||||
.disabled=${this.disabled}
|
.locale=${this.hass.locale}
|
||||||
.required=${this.required}
|
.disabled=${this.disabled}
|
||||||
@value-changed=${this._valueChanged}
|
.required=${this.required}
|
||||||
></ha-time-input>
|
@value-changed=${this._valueChanged}
|
||||||
|
></ha-time-input>
|
||||||
|
</div>
|
||||||
|
${this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: ""}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -58,7 +66,7 @@ export class HaDateTimeSelector extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static styles = css`
|
static styles = css`
|
||||||
:host {
|
.input {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
@ -17,6 +17,8 @@ export class HaDeviceSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@state() public _configEntries?: ConfigEntry[];
|
@state() public _configEntries?: ConfigEntry[];
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
@ -43,6 +45,7 @@ export class HaDeviceSelector extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.deviceFilter=${this._filterDevices}
|
.deviceFilter=${this._filterDevices}
|
||||||
.includeDeviceClasses=${this.selector.device.entity?.device_class
|
.includeDeviceClasses=${this.selector.device.entity?.device_class
|
||||||
? [this.selector.device.entity.device_class]
|
? [this.selector.device.entity.device_class]
|
||||||
@ -62,6 +65,7 @@ export class HaDeviceSelector extends LitElement {
|
|||||||
<ha-devices-picker
|
<ha-devices-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
|
.helper=${this.helper}
|
||||||
.includeDeviceClasses=${this.selector.device.entity?.device_class
|
.includeDeviceClasses=${this.selector.device.entity?.device_class
|
||||||
? [this.selector.device.entity.device_class]
|
? [this.selector.device.entity.device_class]
|
||||||
: undefined}
|
: undefined}
|
||||||
|
@ -23,6 +23,8 @@ export class HaEntitySelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -33,6 +35,7 @@ export class HaEntitySelector extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.includeEntities=${this.selector.entity.include_entities}
|
.includeEntities=${this.selector.entity.include_entities}
|
||||||
.excludeEntities=${this.selector.entity.exclude_entities}
|
.excludeEntities=${this.selector.entity.exclude_entities}
|
||||||
.entityFilter=${this._filterEntities}
|
.entityFilter=${this._filterEntities}
|
||||||
@ -47,6 +50,7 @@ export class HaEntitySelector extends LitElement {
|
|||||||
<ha-entities-picker
|
<ha-entities-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
|
.helper=${this.helper}
|
||||||
.entityFilter=${this._filterEntities}
|
.entityFilter=${this._filterEntities}
|
||||||
.includeEntities=${this.selector.entity.include_entities}
|
.includeEntities=${this.selector.entity.include_entities}
|
||||||
.excludeEntities=${this.selector.entity.exclude_entities}
|
.excludeEntities=${this.selector.entity.exclude_entities}
|
||||||
|
@ -15,6 +15,8 @@ export class HaIconSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = true;
|
@property({ type: Boolean }) public required = true;
|
||||||
@ -26,6 +28,7 @@ export class HaIconSelector extends LitElement {
|
|||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
|
.helper=${this.helper}
|
||||||
.fallbackPath=${this.selector.icon.fallbackPath}
|
.fallbackPath=${this.selector.icon.fallbackPath}
|
||||||
.placeholder=${this.selector.icon.placeholder}
|
.placeholder=${this.selector.icon.placeholder}
|
||||||
@value-changed=${this._valueChanged}
|
@value-changed=${this._valueChanged}
|
||||||
|
@ -20,6 +20,8 @@ export class HaLocationSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
@ -27,6 +29,7 @@ export class HaLocationSelector extends LitElement {
|
|||||||
<ha-locations-editor
|
<ha-locations-editor
|
||||||
class="flex"
|
class="flex"
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.helper=${this.helper}
|
||||||
.locations=${this._location(this.selector, this.value)}
|
.locations=${this._location(this.selector, this.value)}
|
||||||
@location-updated=${this._locationChanged}
|
@location-updated=${this._locationChanged}
|
||||||
@radius-updated=${this._radiusChanged}
|
@radius-updated=${this._radiusChanged}
|
||||||
|
@ -33,6 +33,8 @@ export class HaMediaSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public disabled = false;
|
@property({ type: Boolean, reflect: true }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public required = true;
|
@property({ type: Boolean, reflect: true }) public required = true;
|
||||||
@ -86,6 +88,7 @@ export class HaMediaSelector extends LitElement {
|
|||||||
.label=${this.label ||
|
.label=${this.label ||
|
||||||
this.hass.localize("ui.components.selectors.media.pick_media_player")}
|
this.hass.localize("ui.components.selectors.media.pick_media_player")}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
|
.helper=${this.helper}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
include-domains='["media_player"]'
|
include-domains='["media_player"]'
|
||||||
allow-custom-entity
|
allow-custom-entity
|
||||||
|
@ -6,6 +6,7 @@ import { NumberSelector } from "../../data/selector";
|
|||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import "../ha-slider";
|
import "../ha-slider";
|
||||||
import "../ha-textfield";
|
import "../ha-textfield";
|
||||||
|
import "../ha-input-helper-text";
|
||||||
|
|
||||||
@customElement("ha-selector-number")
|
@customElement("ha-selector-number")
|
||||||
export class HaNumberSelector extends LitElement {
|
export class HaNumberSelector extends LitElement {
|
||||||
@ -26,8 +27,13 @@ export class HaNumberSelector extends LitElement {
|
|||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
return html`${this.selector.number.mode !== "box"
|
const isBox = this.selector.number.mode === "box";
|
||||||
? html`${this.label}${this.required ? "*" : ""}<ha-slider
|
|
||||||
|
return html`
|
||||||
|
${this.label}${this.required ? "*" : ""}
|
||||||
|
<div class="input">
|
||||||
|
${!isBox
|
||||||
|
? html`<ha-slider
|
||||||
.min=${this.selector.number.min}
|
.min=${this.selector.number.min}
|
||||||
.max=${this.selector.number.max}
|
.max=${this.selector.number.max}
|
||||||
.value=${this._value}
|
.value=${this._value}
|
||||||
@ -39,28 +45,33 @@ export class HaNumberSelector extends LitElement {
|
|||||||
@change=${this._handleSliderChange}
|
@change=${this._handleSliderChange}
|
||||||
>
|
>
|
||||||
</ha-slider>`
|
</ha-slider>`
|
||||||
|
: ""}
|
||||||
|
<ha-textfield
|
||||||
|
inputMode="numeric"
|
||||||
|
pattern="[0-9]+([\\.][0-9]+)?"
|
||||||
|
.label=${this.selector.number.mode !== "box" ? undefined : this.label}
|
||||||
|
.placeholder=${this.placeholder}
|
||||||
|
class=${classMap({ single: this.selector.number.mode === "box" })}
|
||||||
|
.min=${this.selector.number.min}
|
||||||
|
.max=${this.selector.number.max}
|
||||||
|
.value=${this.value ?? ""}
|
||||||
|
.step=${this.selector.number.step ?? 1}
|
||||||
|
helperPersistent
|
||||||
|
.helper=${isBox ? this.helper : undefined}
|
||||||
|
.disabled=${this.disabled}
|
||||||
|
.required=${this.required}
|
||||||
|
.suffix=${this.selector.number.unit_of_measurement}
|
||||||
|
type="number"
|
||||||
|
autoValidate
|
||||||
|
?no-spinner=${this.selector.number.mode !== "box"}
|
||||||
|
@input=${this._handleInputChange}
|
||||||
|
>
|
||||||
|
</ha-textfield>
|
||||||
|
</div>
|
||||||
|
${!isBox && this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
: ""}
|
: ""}
|
||||||
<ha-textfield
|
`;
|
||||||
inputMode="numeric"
|
|
||||||
pattern="[0-9]+([\\.][0-9]+)?"
|
|
||||||
.label=${this.selector.number.mode !== "box" ? undefined : this.label}
|
|
||||||
.placeholder=${this.placeholder}
|
|
||||||
class=${classMap({ single: this.selector.number.mode === "box" })}
|
|
||||||
.min=${this.selector.number.min}
|
|
||||||
.max=${this.selector.number.max}
|
|
||||||
.value=${this.value ?? ""}
|
|
||||||
.step=${this.selector.number.step ?? 1}
|
|
||||||
helperPersistent
|
|
||||||
.helper=${this.helper}
|
|
||||||
.disabled=${this.disabled}
|
|
||||||
.required=${this.required}
|
|
||||||
.suffix=${this.selector.number.unit_of_measurement}
|
|
||||||
type="number"
|
|
||||||
autoValidate
|
|
||||||
?no-spinner=${this.selector.number.mode !== "box"}
|
|
||||||
@input=${this._handleInputChange}
|
|
||||||
>
|
|
||||||
</ha-textfield>`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private get _value() {
|
private get _value() {
|
||||||
@ -92,7 +103,7 @@ export class HaNumberSelector extends LitElement {
|
|||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
.input {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
@ -58,6 +58,7 @@ export class HaSelectSelector extends LitElement {
|
|||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
${this._renderHelper()}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -76,6 +77,7 @@ export class HaSelectSelector extends LitElement {
|
|||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
${this._renderHelper()}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,6 +109,7 @@ export class HaSelectSelector extends LitElement {
|
|||||||
item-label-path="label"
|
item-label-path="label"
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required && !value.length}
|
.required=${this.required && !value.length}
|
||||||
.value=${this._filter}
|
.value=${this._filter}
|
||||||
@ -131,6 +134,7 @@ export class HaSelectSelector extends LitElement {
|
|||||||
item-label-path="label"
|
item-label-path="label"
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
|
.helper=${this.helper}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
.items=${options}
|
.items=${options}
|
||||||
@ -161,6 +165,12 @@ export class HaSelectSelector extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _renderHelper() {
|
||||||
|
return this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: "";
|
||||||
|
}
|
||||||
|
|
||||||
private get _mode(): "list" | "dropdown" {
|
private get _mode(): "list" | "dropdown" {
|
||||||
return (
|
return (
|
||||||
this.selector.select.mode ||
|
this.selector.select.mode ||
|
||||||
|
@ -26,6 +26,8 @@ export class HaTargetSelector extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@state() private _entityPlaformLookup?: Record<string, string>;
|
@state() private _entityPlaformLookup?: Record<string, string>;
|
||||||
|
|
||||||
@state() private _configEntries?: ConfigEntry[];
|
@state() private _configEntries?: ConfigEntry[];
|
||||||
@ -64,6 +66,7 @@ export class HaTargetSelector extends SubscribeMixin(LitElement) {
|
|||||||
return html`<ha-target-picker
|
return html`<ha-target-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
|
.helper=${this.helper}
|
||||||
.deviceFilter=${this._filterDevices}
|
.deviceFilter=${this._filterDevices}
|
||||||
.entityRegFilter=${this._filterRegEntities}
|
.entityRegFilter=${this._filterRegEntities}
|
||||||
.entityFilter=${this._filterEntities}
|
.entityFilter=${this._filterEntities}
|
||||||
|
@ -14,6 +14,8 @@ export class HaTimeSelector extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = false;
|
@property({ type: Boolean }) public required = false;
|
||||||
@ -25,6 +27,7 @@ export class HaTimeSelector extends LitElement {
|
|||||||
.locale=${this.hass.locale}
|
.locale=${this.hass.locale}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
.helper=${this.helper}
|
||||||
.label=${this.label}
|
.label=${this.label}
|
||||||
enable-second
|
enable-second
|
||||||
></ha-time-input>
|
></ha-time-input>
|
||||||
|
@ -36,10 +36,9 @@ import memoizeOne from "memoize-one";
|
|||||||
import { LocalStorage } from "../common/decorators/local-storage";
|
import { LocalStorage } from "../common/decorators/local-storage";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { toggleAttribute } from "../common/dom/toggle_attribute";
|
import { toggleAttribute } from "../common/dom/toggle_attribute";
|
||||||
import { computeDomain } from "../common/entity/compute_domain";
|
|
||||||
import { computeStateDomain } from "../common/entity/compute_state_domain";
|
|
||||||
import { stringCompare } from "../common/string/compare";
|
import { stringCompare } from "../common/string/compare";
|
||||||
import { computeRTL } from "../common/util/compute_rtl";
|
import { computeRTL } from "../common/util/compute_rtl";
|
||||||
|
import { throttle } from "../common/util/throttle";
|
||||||
import { ActionHandlerDetail } from "../data/lovelace";
|
import { ActionHandlerDetail } from "../data/lovelace";
|
||||||
import {
|
import {
|
||||||
PersistentNotification,
|
PersistentNotification,
|
||||||
@ -294,11 +293,7 @@ class HaSidebar extends LitElement {
|
|||||||
toggleAttribute(this, "rtl", computeRTL(this.hass));
|
toggleAttribute(this, "rtl", computeRTL(this.hass));
|
||||||
}
|
}
|
||||||
|
|
||||||
this._updatesCount = Object.values(this.hass.states).filter(
|
this._calculateCounts();
|
||||||
(entity) =>
|
|
||||||
computeStateDomain(entity) === "update" &&
|
|
||||||
updateCanInstall(entity as UpdateEntity)
|
|
||||||
).length;
|
|
||||||
|
|
||||||
if (!SUPPORT_SCROLL_IF_NEEDED) {
|
if (!SUPPORT_SCROLL_IF_NEEDED) {
|
||||||
return;
|
return;
|
||||||
@ -312,6 +307,21 @@ class HaSidebar extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _calculateCounts = throttle(() => {
|
||||||
|
let updateCount = 0;
|
||||||
|
|
||||||
|
for (const entityId of Object.keys(this.hass.states)) {
|
||||||
|
if (
|
||||||
|
entityId.startsWith("update.") &&
|
||||||
|
updateCanInstall(this.hass.states[entityId] as UpdateEntity)
|
||||||
|
) {
|
||||||
|
updateCount++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this._updatesCount = updateCount;
|
||||||
|
}, 5000);
|
||||||
|
|
||||||
private _renderHeader() {
|
private _renderHeader() {
|
||||||
return html`<div
|
return html`<div
|
||||||
class="menu"
|
class="menu"
|
||||||
@ -519,14 +529,9 @@ class HaSidebar extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _renderNotifications() {
|
private _renderNotifications() {
|
||||||
let notificationCount = this._notifications
|
const notificationCount = this._notifications
|
||||||
? this._notifications.length
|
? this._notifications.length
|
||||||
: 0;
|
: 0;
|
||||||
for (const entityId in this.hass.states) {
|
|
||||||
if (computeDomain(entityId) === "configurator") {
|
|
||||||
notificationCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return html`<div
|
return html`<div
|
||||||
class="notifications-container"
|
class="notifications-container"
|
||||||
|
@ -43,6 +43,7 @@ import type { HaEntityPickerEntityFilterFunc } from "./entity/ha-entity-picker";
|
|||||||
import "./ha-area-picker";
|
import "./ha-area-picker";
|
||||||
import "./ha-icon-button";
|
import "./ha-icon-button";
|
||||||
import "./ha-svg-icon";
|
import "./ha-svg-icon";
|
||||||
|
import "./ha-input-helper-text";
|
||||||
|
|
||||||
@customElement("ha-target-picker")
|
@customElement("ha-target-picker")
|
||||||
export class HaTargetPicker extends SubscribeMixin(LitElement) {
|
export class HaTargetPicker extends SubscribeMixin(LitElement) {
|
||||||
@ -52,6 +53,8 @@ export class HaTargetPicker extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show only targets with entities from specific domains.
|
* Show only targets with entities from specific domains.
|
||||||
* @type {Array}
|
* @type {Array}
|
||||||
@ -213,7 +216,11 @@ export class HaTargetPicker extends SubscribeMixin(LitElement) {
|
|||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>`;
|
</div>
|
||||||
|
|
||||||
|
${this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: ""} `;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _showPicker(ev) {
|
private async _showPicker(ev) {
|
||||||
|
@ -14,6 +14,8 @@ export class HaTimeInput extends LitElement {
|
|||||||
|
|
||||||
@property() public label?: string;
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) public required = false;
|
@property({ type: Boolean }) public required = false;
|
||||||
@ -46,6 +48,7 @@ export class HaTimeInput extends LitElement {
|
|||||||
@value-changed=${this._timeChanged}
|
@value-changed=${this._timeChanged}
|
||||||
.enableSecond=${this.enableSecond}
|
.enableSecond=${this.enableSecond}
|
||||||
.required=${this.required}
|
.required=${this.required}
|
||||||
|
.helper=${this.helper}
|
||||||
></ha-base-time-input>
|
></ha-base-time-input>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,7 @@ import type { LeafletModuleType } from "../../common/dom/setup-leaflet-map";
|
|||||||
import type { HomeAssistant } from "../../types";
|
import type { HomeAssistant } from "../../types";
|
||||||
import "./ha-map";
|
import "./ha-map";
|
||||||
import type { HaMap } from "./ha-map";
|
import type { HaMap } from "./ha-map";
|
||||||
|
import "../ha-input-helper-text";
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
// for fire event
|
// for fire event
|
||||||
@ -50,6 +51,8 @@ export class HaLocationsEditor extends LitElement {
|
|||||||
|
|
||||||
@property({ attribute: false }) public locations?: MarkerLocation[];
|
@property({ attribute: false }) public locations?: MarkerLocation[];
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public autoFit = false;
|
@property({ type: Boolean }) public autoFit = false;
|
||||||
|
|
||||||
@property({ type: Number }) public zoom = 16;
|
@property({ type: Number }) public zoom = 16;
|
||||||
@ -102,13 +105,18 @@ export class HaLocationsEditor extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`<ha-map
|
return html`
|
||||||
.hass=${this.hass}
|
<ha-map
|
||||||
.layers=${this._getLayers(this._circles, this._locationMarkers)}
|
.hass=${this.hass}
|
||||||
.zoom=${this.zoom}
|
.layers=${this._getLayers(this._circles, this._locationMarkers)}
|
||||||
.autoFit=${this.autoFit}
|
.zoom=${this.zoom}
|
||||||
.darkMode=${this.darkMode}
|
.autoFit=${this.autoFit}
|
||||||
></ha-map>`;
|
.darkMode=${this.darkMode}
|
||||||
|
></ha-map>
|
||||||
|
${this.helper
|
||||||
|
? html`<ha-input-helper-text>${this.helper}</ha-input-helper-text>`
|
||||||
|
: ""}
|
||||||
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _getLayers = memoizeOne(
|
private _getLayers = memoizeOne(
|
||||||
@ -287,13 +295,10 @@ export class HaLocationsEditor extends LitElement {
|
|||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
ha-map {
|
||||||
display: block;
|
display: block;
|
||||||
height: 300px;
|
height: 300px;
|
||||||
}
|
}
|
||||||
ha-map {
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -29,6 +29,7 @@ import {
|
|||||||
DeviceRegistryEntry,
|
DeviceRegistryEntry,
|
||||||
subscribeDeviceRegistry,
|
subscribeDeviceRegistry,
|
||||||
} from "../../data/device_registry";
|
} from "../../data/device_registry";
|
||||||
|
import { fetchIntegrationManifest } from "../../data/integration";
|
||||||
import { haStyleDialog } from "../../resources/styles";
|
import { haStyleDialog } from "../../resources/styles";
|
||||||
import type { HomeAssistant } from "../../types";
|
import type { HomeAssistant } from "../../types";
|
||||||
import { documentationUrl } from "../../util/documentation-url";
|
import { documentationUrl } from "../../util/documentation-url";
|
||||||
@ -43,10 +44,10 @@ import "./step-flow-create-entry";
|
|||||||
import "./step-flow-external";
|
import "./step-flow-external";
|
||||||
import "./step-flow-form";
|
import "./step-flow-form";
|
||||||
import "./step-flow-loading";
|
import "./step-flow-loading";
|
||||||
|
import "./step-flow-menu";
|
||||||
import "./step-flow-pick-flow";
|
import "./step-flow-pick-flow";
|
||||||
import "./step-flow-pick-handler";
|
import "./step-flow-pick-handler";
|
||||||
import "./step-flow-progress";
|
import "./step-flow-progress";
|
||||||
import "./step-flow-menu";
|
|
||||||
|
|
||||||
let instance = 0;
|
let instance = 0;
|
||||||
|
|
||||||
@ -237,22 +238,32 @@ class DataEntryFlowDialog extends LitElement {
|
|||||||
""
|
""
|
||||||
: html`
|
: html`
|
||||||
<div class="dialog-actions">
|
<div class="dialog-actions">
|
||||||
${["form", "menu", "external"].includes(
|
${([
|
||||||
this._step?.type as any
|
"form",
|
||||||
)
|
"menu",
|
||||||
|
"external",
|
||||||
|
"progress",
|
||||||
|
"data_entry_flow_progressed",
|
||||||
|
].includes(this._step?.type as any) &&
|
||||||
|
this._params.manifest?.is_built_in) ||
|
||||||
|
this._params.manifest?.documentation
|
||||||
? html`
|
? html`
|
||||||
<a
|
<a
|
||||||
href=${documentationUrl(
|
href=${this._params.manifest.is_built_in
|
||||||
this.hass,
|
? documentationUrl(
|
||||||
`/integrations/${this._step!.handler}`
|
this.hass,
|
||||||
)}
|
`/integrations/${this._params.manifest.domain}`
|
||||||
|
)
|
||||||
|
: this._params?.manifest?.documentation}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer noopener"
|
rel="noreferrer noopener"
|
||||||
><ha-icon-button
|
>
|
||||||
|
<ha-icon-button
|
||||||
.label=${this.hass.localize("ui.common.help")}
|
.label=${this.hass.localize("ui.common.help")}
|
||||||
.path=${mdiHelpCircle}
|
.path=${mdiHelpCircle}
|
||||||
?rtl=${computeRTL(this.hass)}
|
?rtl=${computeRTL(this.hass)}
|
||||||
></ha-icon-button
|
>
|
||||||
|
</ha-icon-button
|
||||||
></a>
|
></a>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@ -427,6 +438,17 @@ class DataEntryFlowDialog extends LitElement {
|
|||||||
this._handler = undefined;
|
this._handler = undefined;
|
||||||
}
|
}
|
||||||
this._processStep(step);
|
this._processStep(step);
|
||||||
|
if (this._params!.manifest === undefined) {
|
||||||
|
try {
|
||||||
|
this._params!.manifest = await fetchIntegrationManifest(
|
||||||
|
this.hass,
|
||||||
|
this._params?.domain || step.handler
|
||||||
|
);
|
||||||
|
} catch (_) {
|
||||||
|
// No manifest
|
||||||
|
this._params!.manifest = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this._step = null;
|
this._step = null;
|
||||||
this._flowsInProgress = flowsInProgress;
|
this._flowsInProgress = flowsInProgress;
|
||||||
|
@ -10,6 +10,7 @@ import {
|
|||||||
DataEntryFlowStepMenu,
|
DataEntryFlowStepMenu,
|
||||||
DataEntryFlowStepProgress,
|
DataEntryFlowStepProgress,
|
||||||
} from "../../data/data_entry_flow";
|
} from "../../data/data_entry_flow";
|
||||||
|
import { IntegrationManifest } from "../../data/integration";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
|
|
||||||
export interface FlowHandlers {
|
export interface FlowHandlers {
|
||||||
@ -122,6 +123,8 @@ export interface DataEntryFlowDialogParams {
|
|||||||
startFlowHandler?: string;
|
startFlowHandler?: string;
|
||||||
searchQuery?: string;
|
searchQuery?: string;
|
||||||
continueFlowId?: string;
|
continueFlowId?: string;
|
||||||
|
manifest?: IntegrationManifest | null;
|
||||||
|
domain?: string;
|
||||||
dialogClosedCallback?: (params: {
|
dialogClosedCallback?: (params: {
|
||||||
flowFinished: boolean;
|
flowFinished: boolean;
|
||||||
entryId?: string;
|
entryId?: string;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { html } from "lit";
|
import { html } from "lit";
|
||||||
import { ConfigEntry } from "../../data/config_entries";
|
import { ConfigEntry } from "../../data/config_entries";
|
||||||
import { domainToName } from "../../data/integration";
|
import { domainToName, IntegrationManifest } from "../../data/integration";
|
||||||
import {
|
import {
|
||||||
createOptionsFlow,
|
createOptionsFlow,
|
||||||
deleteOptionsFlow,
|
deleteOptionsFlow,
|
||||||
@ -16,12 +16,15 @@ export const loadOptionsFlowDialog = loadDataEntryFlowDialog;
|
|||||||
|
|
||||||
export const showOptionsFlowDialog = (
|
export const showOptionsFlowDialog = (
|
||||||
element: HTMLElement,
|
element: HTMLElement,
|
||||||
configEntry: ConfigEntry
|
configEntry: ConfigEntry,
|
||||||
|
manifest?: IntegrationManifest | null
|
||||||
): void =>
|
): void =>
|
||||||
showFlowDialog(
|
showFlowDialog(
|
||||||
element,
|
element,
|
||||||
{
|
{
|
||||||
startFlowHandler: configEntry.entry_id,
|
startFlowHandler: configEntry.entry_id,
|
||||||
|
domain: configEntry.domain,
|
||||||
|
manifest,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
loadDevicesAndAreas: false,
|
loadDevicesAndAreas: false,
|
||||||
|
@ -190,6 +190,10 @@ class StepFlowForm extends LitElement {
|
|||||||
margin-top: 24px;
|
margin-top: 24px;
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
h2 {
|
||||||
|
word-break: break-word;
|
||||||
|
padding-right: 72px;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@ -35,6 +35,7 @@ class MoreInfoFan extends LocalizeMixin(EventsMixin(PolymerElement)) {
|
|||||||
.has-direction .container-direction,
|
.has-direction .container-direction,
|
||||||
.has-oscillating .container-oscillating {
|
.has-oscillating .container-oscillating {
|
||||||
display: block;
|
display: block;
|
||||||
|
margin-top: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
ha-select {
|
ha-select {
|
||||||
|
@ -243,6 +243,10 @@ class MoreInfoUpdate extends LitElement {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
mwc-linear-progress {
|
||||||
|
margin-bottom: -10px;
|
||||||
|
margin-top: -10px;
|
||||||
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -165,6 +165,9 @@ export class HaChooseAction extends LitElement implements ActionElement {
|
|||||||
right: 0;
|
right: 0;
|
||||||
padding: 4px;
|
padding: 4px;
|
||||||
}
|
}
|
||||||
|
ha-form::part(root) {
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@ -442,6 +442,9 @@ export default class HaAutomationTriggerRow extends LitElement {
|
|||||||
z-index: 3;
|
z-index: 3;
|
||||||
--mdc-theme-text-primary-on-background: var(--primary-text-color);
|
--mdc-theme-text-primary-on-background: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
|
.rtl .card-menu {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
.triggered {
|
.triggered {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@ -470,9 +473,6 @@ export default class HaAutomationTriggerRow extends LitElement {
|
|||||||
background-color: var(--accent-color);
|
background-color: var(--accent-color);
|
||||||
color: var(--text-accent-color, var(--text-primary-color));
|
color: var(--text-accent-color, var(--text-primary-color));
|
||||||
}
|
}
|
||||||
.rtl .card-menu {
|
|
||||||
float: left;
|
|
||||||
}
|
|
||||||
mwc-list-item[disabled] {
|
mwc-list-item[disabled] {
|
||||||
--mdc-theme-text-primary-on-background: var(--disabled-text-color);
|
--mdc-theme-text-primary-on-background: var(--disabled-text-color);
|
||||||
}
|
}
|
||||||
|
@ -11,10 +11,13 @@ import {
|
|||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../common/dom/fire_event";
|
import { fireEvent } from "../../../common/dom/fire_event";
|
||||||
import { stopPropagation } from "../../../common/dom/stop_propagation";
|
import { stopPropagation } from "../../../common/dom/stop_propagation";
|
||||||
import { computeDomain } from "../../../common/entity/compute_domain";
|
import { computeDomain } from "../../../common/entity/compute_domain";
|
||||||
import { domainIcon } from "../../../common/entity/domain_icon";
|
import { domainIcon } from "../../../common/entity/domain_icon";
|
||||||
|
import { stringCompare } from "../../../common/string/compare";
|
||||||
|
import { LocalizeFunc } from "../../../common/translations/localize";
|
||||||
import "../../../components/ha-alert";
|
import "../../../components/ha-alert";
|
||||||
import "../../../components/ha-area-picker";
|
import "../../../components/ha-area-picker";
|
||||||
import "../../../components/ha-expansion-panel";
|
import "../../../components/ha-expansion-panel";
|
||||||
@ -96,7 +99,7 @@ const OVERRIDE_SENSOR_UNITS = {
|
|||||||
pressure: ["hPa", "Pa", "kPa", "bar", "cbar", "mbar", "mmHg", "inHg", "psi"],
|
pressure: ["hPa", "Pa", "kPa", "bar", "cbar", "mbar", "mmHg", "inHg", "psi"],
|
||||||
};
|
};
|
||||||
|
|
||||||
const SWITCH_AS_DOMAINS = ["light", "lock", "cover", "fan", "siren"];
|
const SWITCH_AS_DOMAINS = ["cover", "fan", "light", "lock", "siren"];
|
||||||
|
|
||||||
@customElement("entity-registry-settings")
|
@customElement("entity-registry-settings")
|
||||||
export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
||||||
@ -273,22 +276,29 @@ export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
|||||||
@selected=${this._deviceClassChanged}
|
@selected=${this._deviceClassChanged}
|
||||||
@closed=${stopPropagation}
|
@closed=${stopPropagation}
|
||||||
>
|
>
|
||||||
${this._deviceClassOptions[0].map(
|
${this._deviceClassesSorted(
|
||||||
(deviceClass: string) => html`
|
domain,
|
||||||
<mwc-list-item .value=${deviceClass}>
|
this._deviceClassOptions[0],
|
||||||
${this.hass.localize(
|
this.hass.localize
|
||||||
`ui.dialogs.entity_registry.editor.device_classes.${domain}.${deviceClass}`
|
).map(
|
||||||
)}
|
(entry) => html`
|
||||||
|
<mwc-list-item .value=${entry.deviceClass}>
|
||||||
|
${entry.label}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
<li divider role="separator"></li>
|
${this._deviceClassOptions[0].length &&
|
||||||
${this._deviceClassOptions[1].map(
|
this._deviceClassOptions[1].length
|
||||||
(deviceClass: string) => html`
|
? html`<li divider role="separator"></li>`
|
||||||
<mwc-list-item .value=${deviceClass}>
|
: ""}
|
||||||
${this.hass.localize(
|
${this._deviceClassesSorted(
|
||||||
`ui.dialogs.entity_registry.editor.device_classes.${domain}.${deviceClass}`
|
domain,
|
||||||
)}
|
this._deviceClassOptions[1],
|
||||||
|
this.hass.localize
|
||||||
|
).map(
|
||||||
|
(entry) => html`
|
||||||
|
<mwc-list-item .value=${entry.deviceClass}>
|
||||||
|
${entry.label}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
@ -296,9 +306,9 @@ export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
|||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
${this._deviceClass &&
|
${this._deviceClass &&
|
||||||
stateObj.attributes.unit_of_measurement &&
|
stateObj?.attributes.unit_of_measurement &&
|
||||||
OVERRIDE_SENSOR_UNITS[this._deviceClass]?.includes(
|
OVERRIDE_SENSOR_UNITS[this._deviceClass]?.includes(
|
||||||
stateObj.attributes.unit_of_measurement
|
stateObj?.attributes.unit_of_measurement
|
||||||
)
|
)
|
||||||
? html`
|
? html`
|
||||||
<ha-select
|
<ha-select
|
||||||
@ -332,10 +342,14 @@ export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
|||||||
<mwc-list-item value="switch" selected>
|
<mwc-list-item value="switch" selected>
|
||||||
${domainToName(this.hass.localize, "switch")}</mwc-list-item
|
${domainToName(this.hass.localize, "switch")}</mwc-list-item
|
||||||
>
|
>
|
||||||
${SWITCH_AS_DOMAINS.map(
|
<li divider role="separator"></li>
|
||||||
(as_domain) => html`
|
${this._switchAsDomainsSorted(
|
||||||
<mwc-list-item .value=${as_domain}>
|
SWITCH_AS_DOMAINS,
|
||||||
${domainToName(this.hass.localize, as_domain)}
|
this.hass.localize
|
||||||
|
).map(
|
||||||
|
(entry) => html`
|
||||||
|
<mwc-list-item .value=${entry.domain}>
|
||||||
|
${entry.label}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
@ -716,9 +730,31 @@ export class EntityRegistrySettings extends SubscribeMixin(LitElement) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _showOptionsFlow() {
|
private async _showOptionsFlow() {
|
||||||
showOptionsFlowDialog(this, this._helperConfigEntry!);
|
showOptionsFlowDialog(this, this._helperConfigEntry!, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _switchAsDomainsSorted = memoizeOne(
|
||||||
|
(domains: string[], localize: LocalizeFunc) =>
|
||||||
|
domains
|
||||||
|
.map((entry) => ({
|
||||||
|
domain: entry,
|
||||||
|
label: domainToName(localize, entry),
|
||||||
|
}))
|
||||||
|
.sort((a, b) => stringCompare(a.label, b.label))
|
||||||
|
);
|
||||||
|
|
||||||
|
private _deviceClassesSorted = memoizeOne(
|
||||||
|
(domain: string, deviceClasses: string[], localize: LocalizeFunc) =>
|
||||||
|
deviceClasses
|
||||||
|
.map((entry) => ({
|
||||||
|
deviceClass: entry,
|
||||||
|
label: localize(
|
||||||
|
`ui.dialogs.entity_registry.editor.device_classes.${domain}.${entry}`
|
||||||
|
),
|
||||||
|
}))
|
||||||
|
.sort((a, b) => stringCompare(a.label, b.label))
|
||||||
|
);
|
||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
|
@ -700,6 +700,7 @@ class HaConfigIntegrations extends SubscribeMixin(LitElement) {
|
|||||||
this._handleFlowUpdated();
|
this._handleFlowUpdated();
|
||||||
},
|
},
|
||||||
startFlowHandler: domain,
|
startFlowHandler: domain,
|
||||||
|
manifest: this._manifests[domain],
|
||||||
showAdvanced: this.hass.userData?.showAdvanced,
|
showAdvanced: this.hass.userData?.showAdvanced,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -482,7 +482,11 @@ export class HaIntegrationCard extends LitElement {
|
|||||||
);
|
);
|
||||||
|
|
||||||
private _showOptions(ev) {
|
private _showOptions(ev) {
|
||||||
showOptionsFlowDialog(this, ev.target.closest("ha-card").configEntry);
|
showOptionsFlowDialog(
|
||||||
|
this,
|
||||||
|
ev.target.closest("ha-card").configEntry,
|
||||||
|
this.manifest
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleRename(ev: CustomEvent<RequestSelectedDetail>): void {
|
private _handleRename(ev: CustomEvent<RequestSelectedDetail>): void {
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { mdiChevronRight } from "@mdi/js";
|
import { mdiChevronRight } from "@mdi/js";
|
||||||
|
import formatISO9075 from "date-fns/formatISO9075";
|
||||||
import {
|
import {
|
||||||
css,
|
css,
|
||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
@ -69,12 +70,11 @@ export class DialogStatisticsFixUnsupportedUnitMetadata extends LitElement {
|
|||||||
|
|
||||||
public showDialog(params: DialogStatisticsAdjustSumParams): void {
|
public showDialog(params: DialogStatisticsAdjustSumParams): void {
|
||||||
this._params = params;
|
this._params = params;
|
||||||
|
// moment is in format YYYY-MM-DD HH:mm:ss because of selector
|
||||||
|
// Here we create a date with minutes set to %5
|
||||||
const now = new Date();
|
const now = new Date();
|
||||||
this._moment = `${now.getFullYear()}-${
|
now.setMinutes(now.getMinutes() - (now.getMinutes() % 5), 0);
|
||||||
now.getMonth() + 1
|
this._moment = formatISO9075(now);
|
||||||
}-${now.getDate()} ${now.getHours()}:${
|
|
||||||
now.getMinutes() - (now.getMinutes() % 5)
|
|
||||||
}:00`;
|
|
||||||
this._fetchStats();
|
this._fetchStats();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,6 +167,9 @@ export class DialogStatisticsFixUnsupportedUnitMetadata extends LitElement {
|
|||||||
time. This can mess up your beautiful graphs! Select a time below to
|
time. This can mess up your beautiful graphs! Select a time below to
|
||||||
find the bad moment and adjust the data.
|
find the bad moment and adjust the data.
|
||||||
</div>
|
</div>
|
||||||
|
<div class="text-content">
|
||||||
|
<b>Statistic:</b> ${this._params!.statistic.statistic_id}
|
||||||
|
</div>
|
||||||
<ha-selector-datetime
|
<ha-selector-datetime
|
||||||
label="Pick a time"
|
label="Pick a time"
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
@ -191,7 +194,7 @@ export class DialogStatisticsFixUnsupportedUnitMetadata extends LitElement {
|
|||||||
private _renderAdjustStat() {
|
private _renderAdjustStat() {
|
||||||
return html`
|
return html`
|
||||||
<div class="text-content">
|
<div class="text-content">
|
||||||
${this._params!.statistic.name || this._params!.statistic.statistic_id}
|
<b>Statistic:</b> ${this._params!.statistic.statistic_id}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="table-row">
|
<div class="table-row">
|
||||||
@ -250,7 +253,10 @@ export class DialogStatisticsFixUnsupportedUnitMetadata extends LitElement {
|
|||||||
this._stats5min = undefined;
|
this._stats5min = undefined;
|
||||||
this._statsHour = undefined;
|
this._statsHour = undefined;
|
||||||
const statId = this._params!.statistic.statistic_id;
|
const statId = this._params!.statistic.statistic_id;
|
||||||
const moment = new Date(this._moment!);
|
|
||||||
|
// moment is in format YYYY-MM-DD HH:mm:ss because of selector
|
||||||
|
// Here we convert it to an ISO string.
|
||||||
|
const moment = new Date(this._moment!.replace(" ", "T"));
|
||||||
|
|
||||||
// Search 3 hours before and 3 hours after chosen time
|
// Search 3 hours before and 3 hours after chosen time
|
||||||
const hourStatStart = new Date(moment.getTime());
|
const hourStatStart = new Date(moment.getTime());
|
||||||
|
@ -253,6 +253,10 @@ class HaPanelHistory extends LitElement {
|
|||||||
padding: 8px 16px 0;
|
padding: 8px 16px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:host([narrow]) .filters {
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
ha-date-range-picker {
|
ha-date-range-picker {
|
||||||
margin-right: 16px;
|
margin-right: 16px;
|
||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
|
@ -21,6 +21,9 @@ export const turnOnOffEntity = (
|
|||||||
case "input_button":
|
case "input_button":
|
||||||
service = "press";
|
service = "press";
|
||||||
break;
|
break;
|
||||||
|
case "scene":
|
||||||
|
service = "turn_on";
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
service = turnOn ? "turn_on" : "turn_off";
|
service = turnOn ? "turn_on" : "turn_off";
|
||||||
}
|
}
|
||||||
|
@ -15,6 +15,7 @@ import type { LovelaceRowEditor } from "../../types";
|
|||||||
import { entitiesConfigStruct } from "../structs/entities-struct";
|
import { entitiesConfigStruct } from "../structs/entities-struct";
|
||||||
|
|
||||||
const SecondaryInfoValues: { [key: string]: { domains?: string[] } } = {
|
const SecondaryInfoValues: { [key: string]: { domains?: string[] } } = {
|
||||||
|
none: {},
|
||||||
"entity-id": {},
|
"entity-id": {},
|
||||||
"last-changed": {},
|
"last-changed": {},
|
||||||
"last-updated": {},
|
"last-updated": {},
|
||||||
|
Loading…
x
Reference in New Issue
Block a user