mirror of
https://github.com/home-assistant/frontend.git
synced 2025-07-27 11:16:35 +00:00
Add support for filtering service fields (#16004)
Co-authored-by: Paul Bottein <paul.bottein@gmail.com>
This commit is contained in:
parent
3838d76094
commit
bcdb24849d
@ -15,6 +15,7 @@ import {
|
|||||||
computeDeviceName,
|
computeDeviceName,
|
||||||
DeviceEntityLookup,
|
DeviceEntityLookup,
|
||||||
DeviceRegistryEntry,
|
DeviceRegistryEntry,
|
||||||
|
getDeviceEntityLookup,
|
||||||
subscribeDeviceRegistry,
|
subscribeDeviceRegistry,
|
||||||
} from "../../data/device_registry";
|
} from "../../data/device_registry";
|
||||||
import {
|
import {
|
||||||
@ -129,7 +130,7 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
const deviceEntityLookup: DeviceEntityLookup = {};
|
let deviceEntityLookup: DeviceEntityLookup = {};
|
||||||
|
|
||||||
if (
|
if (
|
||||||
includeDomains ||
|
includeDomains ||
|
||||||
@ -137,15 +138,7 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
includeDeviceClasses ||
|
includeDeviceClasses ||
|
||||||
entityFilter
|
entityFilter
|
||||||
) {
|
) {
|
||||||
for (const entity of entities) {
|
deviceEntityLookup = getDeviceEntityLookup(entities);
|
||||||
if (!entity.device_id) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (!(entity.device_id in deviceEntityLookup)) {
|
|
||||||
deviceEntityLookup[entity.device_id] = [];
|
|
||||||
}
|
|
||||||
deviceEntityLookup[entity.device_id].push(entity);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const areaLookup: { [areaId: string]: AreaRegistryEntry } = {};
|
const areaLookup: { [areaId: string]: AreaRegistryEntry } = {};
|
||||||
|
@ -7,14 +7,20 @@ import {
|
|||||||
import { css, CSSResultGroup, html, LitElement, PropertyValues } from "lit";
|
import { css, CSSResultGroup, html, LitElement, PropertyValues } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
|
import { ensureArray } from "../common/array/ensure-array";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { computeDomain } from "../common/entity/compute_domain";
|
import { computeDomain } from "../common/entity/compute_domain";
|
||||||
import { computeObjectId } from "../common/entity/compute_object_id";
|
import { computeObjectId } from "../common/entity/compute_object_id";
|
||||||
|
import { supportsFeature } from "../common/entity/supports-feature";
|
||||||
import {
|
import {
|
||||||
fetchIntegrationManifest,
|
fetchIntegrationManifest,
|
||||||
IntegrationManifest,
|
IntegrationManifest,
|
||||||
} from "../data/integration";
|
} from "../data/integration";
|
||||||
import { Selector } from "../data/selector";
|
import {
|
||||||
|
expandAreaTarget,
|
||||||
|
expandDeviceTarget,
|
||||||
|
Selector,
|
||||||
|
} from "../data/selector";
|
||||||
import { ValueChangedEvent, HomeAssistant } from "../types";
|
import { ValueChangedEvent, HomeAssistant } from "../types";
|
||||||
import { documentationUrl } from "../util/documentation-url";
|
import { documentationUrl } from "../util/documentation-url";
|
||||||
import "./ha-checkbox";
|
import "./ha-checkbox";
|
||||||
@ -25,6 +31,16 @@ import "./ha-settings-row";
|
|||||||
import "./ha-yaml-editor";
|
import "./ha-yaml-editor";
|
||||||
import type { HaYamlEditor } from "./ha-yaml-editor";
|
import type { HaYamlEditor } from "./ha-yaml-editor";
|
||||||
|
|
||||||
|
const attributeFilter = (values: any[], attribute: any) => {
|
||||||
|
if (typeof attribute === "object") {
|
||||||
|
if (Array.isArray(attribute)) {
|
||||||
|
return attribute.some((item) => values.includes(item));
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return values.includes(attribute);
|
||||||
|
};
|
||||||
|
|
||||||
const showOptionalToggle = (field) =>
|
const showOptionalToggle = (field) =>
|
||||||
field.selector &&
|
field.selector &&
|
||||||
!field.required &&
|
!field.required &&
|
||||||
@ -39,6 +55,10 @@ interface ExtHassService extends Omit<HassService, "fields"> {
|
|||||||
advanced?: boolean;
|
advanced?: boolean;
|
||||||
default?: any;
|
default?: any;
|
||||||
example?: any;
|
example?: any;
|
||||||
|
filter?: {
|
||||||
|
supported_features?: number[];
|
||||||
|
attribute?: Record<string, any[]>;
|
||||||
|
};
|
||||||
selector?: Selector;
|
selector?: Selector;
|
||||||
}[];
|
}[];
|
||||||
hasSelector: string[];
|
hasSelector: string[];
|
||||||
@ -213,6 +233,87 @@ export class HaServiceControl extends LitElement {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private _filterFields = memoizeOne(
|
||||||
|
(serviceData: ExtHassService | undefined, value: this["value"]) =>
|
||||||
|
serviceData?.fields?.filter(
|
||||||
|
(field) =>
|
||||||
|
!field.filter ||
|
||||||
|
this._filterField(serviceData.target, field.filter, value)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
private _filterField(
|
||||||
|
target: ExtHassService["target"],
|
||||||
|
filter: ExtHassService["fields"][number]["filter"],
|
||||||
|
value: this["value"]
|
||||||
|
) {
|
||||||
|
const targetSelector = target ? { target } : { target: {} };
|
||||||
|
const targetEntities =
|
||||||
|
ensureArray(value?.target?.entity_id || value?.data?.entity_id) || [];
|
||||||
|
const targetDevices =
|
||||||
|
ensureArray(value?.target?.device_id || value?.data?.device_id) || [];
|
||||||
|
const targetAreas = ensureArray(
|
||||||
|
value?.target?.area_id || value?.data?.area_id
|
||||||
|
);
|
||||||
|
if (targetAreas) {
|
||||||
|
targetAreas.forEach((areaId) => {
|
||||||
|
const expanded = expandAreaTarget(
|
||||||
|
this.hass,
|
||||||
|
areaId,
|
||||||
|
this.hass.devices,
|
||||||
|
this.hass.entities,
|
||||||
|
targetSelector
|
||||||
|
);
|
||||||
|
targetEntities.push(...expanded.entities);
|
||||||
|
targetDevices.push(...expanded.devices);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (targetDevices.length) {
|
||||||
|
targetDevices.forEach((deviceId) => {
|
||||||
|
targetEntities.push(
|
||||||
|
...expandDeviceTarget(
|
||||||
|
this.hass,
|
||||||
|
deviceId,
|
||||||
|
this.hass.entities,
|
||||||
|
targetSelector
|
||||||
|
).entities
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!targetEntities.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
targetEntities.some((entityId) => {
|
||||||
|
const entityState = this.hass.states[entityId];
|
||||||
|
if (!entityState) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
filter!.supported_features?.some((feature) =>
|
||||||
|
supportsFeature(entityState, feature)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
filter!.attribute &&
|
||||||
|
Object.entries(filter!.attribute).some(
|
||||||
|
([attribute, values]) =>
|
||||||
|
attribute in entityState.attributes &&
|
||||||
|
attributeFilter(values, entityState.attributes[attribute])
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
})
|
||||||
|
) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
const serviceData = this._getServiceInfo(
|
const serviceData = this._getServiceInfo(
|
||||||
this._value?.service,
|
this._value?.service,
|
||||||
@ -235,6 +336,8 @@ export class HaServiceControl extends LitElement {
|
|||||||
serviceData?.fields.some((field) => showOptionalToggle(field))
|
serviceData?.fields.some((field) => showOptionalToggle(field))
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const filteredFields = this._filterFields(serviceData, this._value);
|
||||||
|
|
||||||
return html`<ha-service-picker
|
return html`<ha-service-picker
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this._value?.service}
|
.value=${this._value?.service}
|
||||||
@ -309,7 +412,7 @@ export class HaServiceControl extends LitElement {
|
|||||||
.defaultValue=${this._value?.data}
|
.defaultValue=${this._value?.data}
|
||||||
@value-changed=${this._dataChanged}
|
@value-changed=${this._dataChanged}
|
||||||
></ha-yaml-editor>`
|
></ha-yaml-editor>`
|
||||||
: serviceData?.fields.map((dataField) => {
|
: filteredFields?.map((dataField) => {
|
||||||
const showOptional = showOptionalToggle(dataField);
|
const showOptional = showOptionalToggle(dataField);
|
||||||
return dataField.selector &&
|
return dataField.selector &&
|
||||||
(!dataField.advanced ||
|
(!dataField.advanced ||
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import chipStyles from "@material/chips/dist/mdc.chips.min.css";
|
import chipStyles from "@material/chips/dist/mdc.chips.min.css";
|
||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
@ -9,10 +10,9 @@ import {
|
|||||||
mdiSofa,
|
mdiSofa,
|
||||||
mdiUnfoldMoreVertical,
|
mdiUnfoldMoreVertical,
|
||||||
} from "@mdi/js";
|
} from "@mdi/js";
|
||||||
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
|
||||||
import { ComboBoxLightOpenedChangedEvent } from "@vaadin/combo-box/vaadin-combo-box-light";
|
import { ComboBoxLightOpenedChangedEvent } from "@vaadin/combo-box/vaadin-combo-box-light";
|
||||||
import { HassEntity, HassServiceTarget } from "home-assistant-js-websocket";
|
import { HassEntity, HassServiceTarget } from "home-assistant-js-websocket";
|
||||||
import { css, CSSResultGroup, html, LitElement, unsafeCSS, nothing } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing, unsafeCSS } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { classMap } from "lit/directives/class-map";
|
import { classMap } from "lit/directives/class-map";
|
||||||
import { ensureArray } from "../common/array/ensure-array";
|
import { ensureArray } from "../common/array/ensure-array";
|
||||||
|
@ -3,8 +3,13 @@ import { ensureArray } from "../common/array/ensure-array";
|
|||||||
import { computeStateDomain } from "../common/entity/compute_state_domain";
|
import { computeStateDomain } from "../common/entity/compute_state_domain";
|
||||||
import { supportsFeature } from "../common/entity/supports-feature";
|
import { supportsFeature } from "../common/entity/supports-feature";
|
||||||
import { UiAction } from "../panels/lovelace/components/hui-action-editor";
|
import { UiAction } from "../panels/lovelace/components/hui-action-editor";
|
||||||
import type { DeviceRegistryEntry } from "./device_registry";
|
import { HomeAssistant } from "../types";
|
||||||
import type { EntitySources } from "./entity_sources";
|
import {
|
||||||
|
DeviceRegistryEntry,
|
||||||
|
getDeviceIntegrationLookup,
|
||||||
|
} from "./device_registry";
|
||||||
|
import { EntityRegistryDisplayEntry } from "./entity_registry";
|
||||||
|
import { EntitySources } from "./entity_sources";
|
||||||
|
|
||||||
export type Selector =
|
export type Selector =
|
||||||
| ActionSelector
|
| ActionSelector
|
||||||
@ -361,10 +366,121 @@ export interface UiColorSelector {
|
|||||||
ui_color: {} | null;
|
ui_color: {} | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const expandAreaTarget = (
|
||||||
|
hass: HomeAssistant,
|
||||||
|
areaId: string,
|
||||||
|
devices: HomeAssistant["devices"],
|
||||||
|
entities: HomeAssistant["entities"],
|
||||||
|
targetSelector: TargetSelector,
|
||||||
|
entitySources?: EntitySources
|
||||||
|
) => {
|
||||||
|
const newEntities: string[] = [];
|
||||||
|
const newDevices: string[] = [];
|
||||||
|
Object.values(devices).forEach((device) => {
|
||||||
|
if (
|
||||||
|
device.area_id === areaId &&
|
||||||
|
deviceMeetsTargetSelector(
|
||||||
|
hass,
|
||||||
|
Object.values(entities),
|
||||||
|
device,
|
||||||
|
targetSelector,
|
||||||
|
entitySources
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
newDevices.push(device.id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Object.values(entities).forEach((entity) => {
|
||||||
|
if (
|
||||||
|
entity.area_id === areaId &&
|
||||||
|
entityMeetsTargetSelector(
|
||||||
|
hass.states[entity.entity_id],
|
||||||
|
targetSelector,
|
||||||
|
entitySources
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
newEntities.push(entity.entity_id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return { devices: newDevices, entities: newEntities };
|
||||||
|
};
|
||||||
|
|
||||||
|
export const expandDeviceTarget = (
|
||||||
|
hass: HomeAssistant,
|
||||||
|
deviceId: string,
|
||||||
|
entities: HomeAssistant["entities"],
|
||||||
|
targetSelector: TargetSelector,
|
||||||
|
entitySources?: EntitySources
|
||||||
|
) => {
|
||||||
|
const newEntities: string[] = [];
|
||||||
|
Object.values(entities).forEach((entity) => {
|
||||||
|
if (
|
||||||
|
entity.device_id === deviceId &&
|
||||||
|
entityMeetsTargetSelector(
|
||||||
|
hass.states[entity.entity_id],
|
||||||
|
targetSelector,
|
||||||
|
entitySources
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
newEntities.push(entity.entity_id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return { entities: newEntities };
|
||||||
|
};
|
||||||
|
|
||||||
|
const deviceMeetsTargetSelector = (
|
||||||
|
hass: HomeAssistant,
|
||||||
|
entityRegistry: EntityRegistryDisplayEntry[],
|
||||||
|
device: DeviceRegistryEntry,
|
||||||
|
targetSelector: TargetSelector,
|
||||||
|
entitySources?: EntitySources
|
||||||
|
): boolean => {
|
||||||
|
const deviceIntegrationLookup = entitySources
|
||||||
|
? getDeviceIntegrationLookup(entitySources, entityRegistry)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
if (targetSelector.target?.device) {
|
||||||
|
if (
|
||||||
|
!ensureArray(targetSelector.target.device).some((filterDevice) =>
|
||||||
|
filterSelectorDevices(filterDevice, device, deviceIntegrationLookup)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (targetSelector.target?.entity) {
|
||||||
|
const entities = entityRegistry.filter(
|
||||||
|
(reg) => reg.device_id === device.id
|
||||||
|
);
|
||||||
|
return entities.some((entity) => {
|
||||||
|
const entityState = hass.states[entity.entity_id];
|
||||||
|
return entityMeetsTargetSelector(
|
||||||
|
entityState,
|
||||||
|
targetSelector,
|
||||||
|
entitySources
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
const entityMeetsTargetSelector = (
|
||||||
|
entity: HassEntity,
|
||||||
|
targetSelector: TargetSelector,
|
||||||
|
entitySources?: EntitySources
|
||||||
|
): boolean => {
|
||||||
|
if (targetSelector.target?.entity) {
|
||||||
|
return ensureArray(targetSelector.target!.entity).some((filterEntity) =>
|
||||||
|
filterSelectorEntities(filterEntity, entity, entitySources)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
export const filterSelectorDevices = (
|
export const filterSelectorDevices = (
|
||||||
filterDevice: DeviceSelectorFilter,
|
filterDevice: DeviceSelectorFilter,
|
||||||
device: DeviceRegistryEntry,
|
device: DeviceRegistryEntry,
|
||||||
deviceIntegrationLookup: Record<string, string[]> | undefined
|
deviceIntegrationLookup?: Record<string, string[]> | undefined
|
||||||
): boolean => {
|
): boolean => {
|
||||||
const {
|
const {
|
||||||
manufacturer: filterManufacturer,
|
manufacturer: filterManufacturer,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user