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
12 changes: 10 additions & 2 deletions src/panels/lovelace/common/data.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { HomeAssistant } from "../../../types";
import { LovelaceConfig } from "../types";

export const getCardConfig = (
hass: HomeAssistant,
Expand All @@ -12,10 +13,17 @@ export const getCardConfig = (
export const updateCardConfig = (
hass: HomeAssistant,
cardId: string,
config: any
config: LovelaceConfig | string,
configFormat: "json" | "yaml"
): Promise<void> =>
hass!.callWS({
hass.callWS({
type: "lovelace/config/card/update",
card_id: cardId,
card_config: config,
format: configFormat,
});

export const migrateConfig = (hass: HomeAssistant): Promise<void> =>
hass.callWS({
type: "lovelace/config/migrate",
});
9 changes: 4 additions & 5 deletions src/panels/lovelace/components/hui-card-options.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,16 @@ import "@polymer/paper-button/paper-button";
import { html, LitElement, PropertyDeclarations } from "@polymer/lit-element";
import { fireEvent } from "../../../common/dom/fire_event";
import { HomeAssistant } from "../../../types";
import { LovelaceConfig } from "../types";

let registeredDialog = false;

export class HuiCardOptions extends LitElement {
public cardId?: string;
public cardConfig?: LovelaceConfig;
protected hass?: HomeAssistant;

static get properties(): PropertyDeclarations {
return {
hass: {},
};
return { hass: {} };
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

Not sure why this was changed. I dont think this is "prettier"

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.

It is, small objects will be put on a single line.

}

public connectedCallback() {
Expand Down Expand Up @@ -51,7 +50,7 @@ export class HuiCardOptions extends LitElement {
private _editCard() {
fireEvent(this, "show-edit-card", {
hass: this.hass,
cardId: this.cardId,
cardConfig: this.cardConfig,
reloadLovelace: () => fireEvent(this, "config-refresh"),
});
}
Expand Down
73 changes: 73 additions & 0 deletions src/panels/lovelace/editor/hui-card-preview.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
import "@polymer/paper-input/paper-textarea";

import createCardElement from "../common/create-card-element";
import createErrorCardConfig from "../common/create-error-card-config";
import { HomeAssistant } from "../../../types";
import { LovelaceCard, LovelaceConfig } from "../types";
import { ConfigError } from "./types";

const CUSTOM_TYPE_PREFIX = "custom:";

export class HuiCardPreview extends HTMLElement {
private _hass?: HomeAssistant;
private _element?: LovelaceCard;

set hass(value: HomeAssistant) {
this._hass = value;
if (this._element) {
this._element.hass = value;
}
}

set error(error: ConfigError) {
const configValue = createErrorCardConfig(
`${error.type}: ${error.message}`,
undefined
);

this._createCard(configValue);
}

set config(configValue: LovelaceConfig) {
if (!configValue) {
return;
}

if (!this._element) {
this._createCard(configValue);
return;
}

const tag = configValue.type.startsWith(CUSTOM_TYPE_PREFIX)
? configValue.type.substr(CUSTOM_TYPE_PREFIX.length)
: `hui-${configValue.type}-card`;

if (tag.toUpperCase() === this._element.tagName) {
this._element.setConfig(configValue);
} else {
this._createCard(configValue);
}
}

private _createCard(configValue: LovelaceConfig): void {
if (this._element) {
this.removeChild(this._element);
}

this._element = createCardElement(configValue);

if (this._hass) {
this._element!.hass = this._hass;
}

this.appendChild(this._element!);
}
}

declare global {
interface HTMLElementTagNameMap {
"hui-card-preview": HuiCardPreview;
}
}

customElements.define("hui-card-preview", HuiCardPreview);
233 changes: 28 additions & 205 deletions src/panels/lovelace/editor/hui-dialog-edit-card.ts
Original file line number Diff line number Diff line change
@@ -1,229 +1,52 @@
import { html, LitElement, PropertyDeclarations } from "@polymer/lit-element";
import yaml from "js-yaml";
import { when } from "lit-html/directives/when";
import { TemplateResult } from "lit-html";

import "@polymer/paper-button/paper-button";
import "@polymer/paper-input/paper-textarea";
import "@polymer/paper-dialog-scrollable/paper-dialog-scrollable";
import "@polymer/paper-dialog/paper-dialog";
// This is not a duplicate import, one is for types, one is for element.
// tslint:disable-next-line
import { PaperDialogElement } from "@polymer/paper-dialog/paper-dialog";
import { HomeAssistant } from "../../../types";
import { getCardConfig, updateCardConfig } from "../common/data";
import { fireEvent } from "../../../common/dom/fire_event";

import "./hui-yaml-editor";
import "./hui-yaml-card-preview";
// This is not a duplicate import, one is for types, one is for element.
// tslint:disable-next-line
import { HuiYAMLCardPreview } from "./hui-yaml-card-preview";
import { LovelaceCardEditor, LovelaceConfig } from "../types";
import { YamlChangedEvent, ConfigValue } from "./types";

const CUSTOM_TYPE_PREFIX = "custom:";
import { LovelaceConfig } from "../types";
import "./hui-edit-card";
import "./hui-migrate-config";

export class HuiDialogEditCard extends LitElement {
protected hass?: HomeAssistant;
private _cardId?: string;
private _originalConfigYaml?: string;
private _configElement?: LovelaceCardEditor | null;
protected _hass?: HomeAssistant;
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

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

Why are we making this a _ variable when its not private?

private _cardConfig?: LovelaceConfig;
private _reloadLovelace?: () => void;
private _editorToggle?: boolean;
private _configValue?: ConfigValue;

static get properties(): PropertyDeclarations {
return {
hass: {},
cardId: {
type: Number,
},
_dialogClosedCallback: {},
_configElement: {},
_editorToggle: {},
_hass: {},
_cardConfig: {},
};
}

public async showDialog({ hass, cardId, reloadLovelace }) {
this.hass = hass;
this._cardId = cardId;
public async showDialog({ hass, cardConfig, reloadLovelace }): Promise<void> {
this._hass = hass;
this._cardConfig = cardConfig;
this._reloadLovelace = reloadLovelace;
this._editorToggle = true;
this._configElement = undefined;
this._configValue = { format: "yaml", value: "" };
this._loadConfig().then(() => this._loadConfigElement());
// Wait till dialog is rendered.
await this.updateComplete;
this._dialog.open();
}

private get _dialog(): PaperDialogElement {
return this.shadowRoot!.querySelector("paper-dialog")!;
}

private get _previewEl(): HuiYAMLCardPreview {
return this.shadowRoot!.querySelector("hui-yaml-card-preview")!;
(this.shadowRoot!.children[0] as any).showDialog();
}

protected render(): TemplateResult {
return html`
<style>
paper-dialog {
width: 650px;
}
.element-editor {
margin-bottom: 16px;
}
</style>
<paper-dialog with-backdrop>
<h2>Card Configuration</h2>
<paper-dialog-scrollable>
${
this._editorToggle && this._configElement !== null
? html`
<div class="element-editor">
${
when(
this._configElement,
() => this._configElement,
() =>
html`
Loading...
`
)
}
</div>
`
: html`
<hui-yaml-editor
.yaml="${this._configValue!.value}"
@yaml-changed="${this._handleYamlChanged}"
></hui-yaml-editor>
`
}
<hui-yaml-card-preview
.hass="${this.hass}"
.value="${this._configValue}"
></hui-yaml-card-preview>
</paper-dialog-scrollable>
<div class="paper-dialog-buttons">
<paper-button @click="${this._toggleEditor}"
>Toggle Editor</paper-button
>
<paper-button @click="${this._closeDialog}">Cancel</paper-button>
<paper-button @click="${this._updateConfigInBackend}"
>Save</paper-button
>
</div>
</paper-dialog>
${
this._cardConfig!.id
? html`
<hui-edit-card
.cardConfig="${this._cardConfig}"
.hass="${this._hass}"
@reload-lovelace="${this._reloadLovelace}"
>
</hui-edit-card>
`
: html`
<hui-migrate-config
.hass="${this._hass}"
@reload-lovelace="${this._reloadLovelace}"
></hui-migrate-config>
`
}
`;
}

private _handleYamlChanged(ev: YamlChangedEvent): void {
this._configValue = { format: "yaml", value: ev.detail.yaml };
this._updatePreview(this._configValue);
}

private _handleJSConfigChanged(value: LovelaceConfig): void {
this._configElement!.setConfig(value);
this._configValue = { format: "js", value };
this._updatePreview(this._configValue);
}

private _updatePreview(value: ConfigValue) {
if (!this._previewEl) {
return;
}
this._previewEl.value = value;
}

private _closeDialog(): void {
this._dialog.close();
}

private _toggleEditor(): void {
if (this._editorToggle && this._configValue!.format === "js") {
this._configValue = {
format: "yaml",
value: yaml.safeDump(this._configValue!.value),
};
} else if (this._configElement && this._configValue!.format === "yaml") {
this._configValue = {
format: "js",
value: yaml.safeLoad(this._configValue!.value),
};
this._configElement.setConfig(this._configValue!.value as LovelaceConfig);
}
this._editorToggle = !this._editorToggle;
}

private async _loadConfig(): Promise<void> {
const cardConfig = await getCardConfig(this.hass!, this._cardId!);
this._configValue = {
format: "yaml",
value: cardConfig,
};
this._originalConfigYaml = cardConfig;

// This will center the dialog with the updated config Element
fireEvent(this._dialog, "iron-resize");
}

private async _loadConfigElement(): Promise<void> {
const conf = yaml.safeLoad(this._configValue!.value);

const tag = conf.type.startsWith(CUSTOM_TYPE_PREFIX)
? conf.type.substr(CUSTOM_TYPE_PREFIX.length)
: `hui-${conf.type}-card`;

const elClass = customElements.get(tag);
let configElement;

try {
configElement = await elClass.getConfigElement();
} catch (err) {
this._configElement = null;
return;
}

configElement.setConfig(conf);
configElement.hass = this.hass;
configElement.addEventListener("config-changed", (ev) =>
this._handleJSConfigChanged(ev.detail.config)
);
this._configValue = { format: "js", value: conf };
this._configElement = configElement;

// This will center the dialog with the updated config Element
fireEvent(this._dialog, "iron-resize");
}

private async _updateConfigInBackend(): Promise<void> {
if (this._configValue!.format === "js") {
this._configValue = {
format: "yaml",
value: yaml.safeDump(this._configValue!.value),
};
}

if (this._configValue!.value === this._originalConfigYaml) {
this._dialog.close();
return;
}

try {
await updateCardConfig(
this.hass!,
this._cardId!,
this._configValue!.value
);
this._dialog.close();
this._reloadLovelace!();
} catch (err) {
alert(`Saving failed: ${err.reason}`);
}
}
}

declare global {
Expand Down
Loading