Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 10 additions & 1 deletion src/data/device_automation.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,15 @@ export const fetchDeviceTriggers = (hass: HomeAssistant, deviceId: string) =>
device_id: deviceId,
});

export const fetchDeviceActionCapabilities = (
hass: HomeAssistant,
action: DeviceAction
) =>
hass.callWS<DeviceAction[]>({
type: "device_automation/action/capabilities",
action,
});

export const fetchDeviceConditionCapabilities = (
hass: HomeAssistant,
condition: DeviceCondition
Expand All @@ -57,7 +66,7 @@ export const fetchDeviceTriggerCapabilities = (
trigger,
});

const whitelist = ["above", "below", "for"];
const whitelist = ["above", "below", "code", "for"];
Copy link
Copy Markdown
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This might get a bit out of hand..
Maybe it would be better to pass the automation's extra_fields list to deviceAutomationsEqual instead?

Copy link
Copy Markdown
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, agree

Copy link
Copy Markdown
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The problem is just that HaDeviceAutomationPicker doesn't know about the extra fields right now.


export const deviceAutomationsEqual = (
a: DeviceAutomation,
Expand Down
2 changes: 1 addition & 1 deletion src/panels/config/js/condition/device.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ export default class DeviceCondition extends Component<any, any> {
}

public componentDidUpdate(prevProps) {
if (prevProps.condition !== this.props.condition) {
if (!deviceAutomationsEqual(prevProps.condition, this.props.condition)) {
this._getCapabilities();
}
}
Expand Down
86 changes: 82 additions & 4 deletions src/panels/config/js/script/device.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,14 @@ import { h, Component } from "preact";

import "../../../../components/device/ha-device-picker";
import "../../../../components/device/ha-device-action-picker";
import { HomeAssistant } from "../../../../types";
import "../../../../components/ha-form";

import {
fetchDeviceActionCapabilities,
deviceAutomationsEqual,
} from "../../../../data/device_automation";
import { DeviceAction } from "../../../../data/script";
import { HomeAssistant } from "../../../../types";

export default class DeviceActionEditor extends Component<
{
Expand All @@ -14,6 +20,7 @@ export default class DeviceActionEditor extends Component<
},
{
device_id: string | undefined;
capabilities: any | undefined;
}
> {
public static defaultConfig: DeviceAction = {
Expand All @@ -22,16 +29,26 @@ export default class DeviceActionEditor extends Component<
entity_id: "",
};

private _origAction;

constructor() {
super();
this.devicePicked = this.devicePicked.bind(this);
this.deviceActionPicked = this.deviceActionPicked.bind(this);
this.state = { device_id: undefined };
this._extraFieldsChanged = this._extraFieldsChanged.bind(this);
this.state = { device_id: undefined, capabilities: undefined };
}

public render() {
const { action, hass } = this.props;
const deviceId = this.state.device_id || action.device_id;
const capabilities = this.state.capabilities;
const extraFieldsData =
capabilities && capabilities.extra_fields
? capabilities.extra_fields.map((item) => {
return { [item.name]: this.props.action[item.name] };
})
: undefined;

return (
<div>
Expand All @@ -48,16 +65,77 @@ export default class DeviceActionEditor extends Component<
hass={hass}
label="Action"
/>
{extraFieldsData && (
<ha-form
data={Object.assign({}, ...extraFieldsData)}
onData-changed={this._extraFieldsChanged}
schema={this.state.capabilities.extra_fields}
computeLabel={this._extraFieldsComputeLabelCallback(hass.localize)}
/>
)}
</div>
);
}

public componentDidMount() {
if (!this.state.capabilities) {
this._getCapabilities();
}
if (this.props.action) {
this._origAction = this.props.action;
}
}

public componentDidUpdate(prevProps) {
if (!deviceAutomationsEqual(prevProps.action, this.props.action)) {
this._getCapabilities();
}
}

private devicePicked(ev) {
this.setState({ device_id: ev.target.value });
this.setState({ ...this.state, device_id: ev.target.value });
}

private deviceActionPicked(ev) {
const deviceAction = { ...ev.target.value };
let deviceAction = ev.target.value;
if (
this._origAction &&
deviceAutomationsEqual(this._origAction, deviceAction)
) {
deviceAction = this._origAction;
}
this.props.onChange(this.props.index, deviceAction);
}

private async _getCapabilities() {
const action = this.props.action;

const capabilities = action.domain
? await fetchDeviceActionCapabilities(this.props.hass, action)
: null;
this.setState({ ...this.state, capabilities });
}

private _extraFieldsChanged(ev) {
if (!ev.detail.path) {
return;
}
const item = ev.detail.path.replace("data.", "");
const value = ev.detail.value || undefined;

this.props.onChange(this.props.index, {
...this.props.action,
[item]: value,
});
}

private _extraFieldsComputeLabelCallback(localize) {
// Returns a callback for ha-form to calculate labels per schema object
return (schema) =>
localize(
`ui.panel.config.automation.editor.actions.type.device_id.extra_fields.${
schema.name
}`
) || schema.name;
}
}
2 changes: 1 addition & 1 deletion src/panels/config/js/trigger/device.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ export default class DeviceTrigger extends Component<any, any> {
}

public componentDidUpdate(prevProps) {
if (prevProps.trigger !== this.props.trigger) {
if (!deviceAutomationsEqual(prevProps.trigger, this.props.trigger)) {
this._getCapabilities();
}
}
Expand Down
5 changes: 4 additions & 1 deletion src/translations/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -908,7 +908,10 @@
"service_data": "[%key:ui::panel::config::automation::editor::actions::type::service::service_data%]"
},
"device_id": {
"label": "Device"
"label": "Device",
"extra_fields": {
"code": "Code"
}
},
"scene": {
"label": "Activate scene"
Expand Down