mirror of
https://github.com/home-assistant/frontend.git
synced 2025-07-23 17:26:42 +00:00
Picture Glance Conversion to TS (#2029)
* First Commit * Convert to TS * Extract entity render to own function * Making it one function like not an idiot * Addressing Reviews
This commit is contained in:
parent
2058e0d3fb
commit
ef2aa2ea6f
@ -1,201 +0,0 @@
|
|||||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
|
|
||||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
|
||||||
|
|
||||||
import "../../../components/ha-card";
|
|
||||||
import "../../../components/ha-icon";
|
|
||||||
import "../components/hui-image";
|
|
||||||
|
|
||||||
import computeStateDisplay from "../../../common/entity/compute_state_display";
|
|
||||||
import computeStateName from "../../../common/entity/compute_state_name";
|
|
||||||
import { DOMAINS_TOGGLE } from "../../../common/const";
|
|
||||||
import stateIcon from "../../../common/entity/state_icon";
|
|
||||||
import toggleEntity from "../common/entity/toggle-entity";
|
|
||||||
import processConfigEntities from "../common/process-config-entities";
|
|
||||||
|
|
||||||
import EventsMixin from "../../../mixins/events-mixin";
|
|
||||||
import LocalizeMixin from "../../../mixins/localize-mixin";
|
|
||||||
import NavigateMixin from "../../../mixins/navigate-mixin";
|
|
||||||
import computeDomain from "../../../common/entity/compute_domain";
|
|
||||||
|
|
||||||
const STATES_OFF = new Set(["closed", "locked", "not_home", "off"]);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @appliesMixin EventsMixin
|
|
||||||
* @appliesMixin LocalizeMixin
|
|
||||||
* @appliesMixin NavigateMixin
|
|
||||||
*/
|
|
||||||
class HuiPictureGlanceCard extends NavigateMixin(
|
|
||||||
LocalizeMixin(EventsMixin(PolymerElement))
|
|
||||||
) {
|
|
||||||
static get template() {
|
|
||||||
return html`
|
|
||||||
<style>
|
|
||||||
ha-card {
|
|
||||||
position: relative;
|
|
||||||
min-height: 48px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
hui-image.clickable {
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
.box {
|
|
||||||
@apply --paper-font-common-nowrap;
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: 0;
|
|
||||||
background-color: rgba(0, 0, 0, 0.3);
|
|
||||||
padding: 4px 8px;
|
|
||||||
font-size: 16px;
|
|
||||||
line-height: 40px;
|
|
||||||
color: white;
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
}
|
|
||||||
.box .title {
|
|
||||||
font-weight: 500;
|
|
||||||
margin-left: 8px;
|
|
||||||
}
|
|
||||||
ha-icon {
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 8px;
|
|
||||||
color: #a9a9a9;
|
|
||||||
}
|
|
||||||
ha-icon.state-on {
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
||||||
<ha-card>
|
|
||||||
<hui-image
|
|
||||||
class$="[[_computeImageClass(_config)]]"
|
|
||||||
on-click="_handleImageClick"
|
|
||||||
hass="[[hass]]"
|
|
||||||
image="[[_config.image]]"
|
|
||||||
state-image="[[_config.state_image]]"
|
|
||||||
camera-image="[[_config.camera_image]]"
|
|
||||||
entity="[[_config.entity]]"
|
|
||||||
aspect-ratio="[[_config.aspect_ratio]]"
|
|
||||||
></hui-image>
|
|
||||||
<div class="box">
|
|
||||||
<template is="dom-if" if="[[_config.title]]">
|
|
||||||
<div class="title">[[_config.title]]</div>
|
|
||||||
</template>
|
|
||||||
<div>
|
|
||||||
<template
|
|
||||||
is="dom-repeat"
|
|
||||||
items="[[_computeVisible(_entitiesDialog, hass.states)]]"
|
|
||||||
>
|
|
||||||
<ha-icon
|
|
||||||
on-click="_openDialog"
|
|
||||||
class$="[[_computeButtonClass(item.entity, hass.states)]]"
|
|
||||||
icon="[[_computeIcon(item, hass.states)]]"
|
|
||||||
title="[[_computeTooltip(item.entity, hass.states)]]"
|
|
||||||
></ha-icon>
|
|
||||||
</template>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<template
|
|
||||||
is="dom-repeat"
|
|
||||||
items="[[_computeVisible(_entitiesToggle, hass.states)]]"
|
|
||||||
>
|
|
||||||
<ha-icon
|
|
||||||
on-click="_callService"
|
|
||||||
class$="[[_computeButtonClass(item.entity, hass.states)]]"
|
|
||||||
icon="[[_computeIcon(item, hass.states)]]"
|
|
||||||
title="[[_computeTooltip(item.entity, hass.states)]]"
|
|
||||||
></ha-icon>
|
|
||||||
</template>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ha-card>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get properties() {
|
|
||||||
return {
|
|
||||||
hass: Object,
|
|
||||||
_config: Object,
|
|
||||||
_entitiesDialog: Array,
|
|
||||||
_entitiesToggle: Array,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
getCardSize() {
|
|
||||||
return 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
setConfig(config) {
|
|
||||||
if (
|
|
||||||
!config ||
|
|
||||||
!config.entities ||
|
|
||||||
!Array.isArray(config.entities) ||
|
|
||||||
!(config.image || config.camera_image || config.state_image) ||
|
|
||||||
(config.state_image && !config.entity)
|
|
||||||
) {
|
|
||||||
throw new Error("Invalid card configuration");
|
|
||||||
}
|
|
||||||
const entities = processConfigEntities(config.entities);
|
|
||||||
const dialog = [];
|
|
||||||
const toggle = [];
|
|
||||||
|
|
||||||
entities.forEach((item) => {
|
|
||||||
if (
|
|
||||||
config.force_dialog ||
|
|
||||||
!DOMAINS_TOGGLE.has(computeDomain(item.entity))
|
|
||||||
) {
|
|
||||||
dialog.push(item);
|
|
||||||
} else {
|
|
||||||
toggle.push(item);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
this.setProperties({
|
|
||||||
_config: config,
|
|
||||||
_entitiesDialog: dialog,
|
|
||||||
_entitiesToggle: toggle,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
_computeVisible(collection, states) {
|
|
||||||
return collection.filter((el) => el.entity in states);
|
|
||||||
}
|
|
||||||
|
|
||||||
_computeIcon(item, states) {
|
|
||||||
return item.icon || stateIcon(states[item.entity]);
|
|
||||||
}
|
|
||||||
|
|
||||||
_computeButtonClass(entityId, states) {
|
|
||||||
return STATES_OFF.has(states[entityId].state) ? "" : "state-on";
|
|
||||||
}
|
|
||||||
|
|
||||||
_computeTooltip(entityId, states) {
|
|
||||||
return `${computeStateName(states[entityId])}: ${computeStateDisplay(
|
|
||||||
this.localize,
|
|
||||||
states[entityId]
|
|
||||||
)}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
_computeImageClass(config) {
|
|
||||||
return config.navigation_path || config.camera_image ? "clickable" : "";
|
|
||||||
}
|
|
||||||
|
|
||||||
_openDialog(ev) {
|
|
||||||
this.fire("hass-more-info", { entityId: ev.model.item.entity });
|
|
||||||
}
|
|
||||||
|
|
||||||
_callService(ev) {
|
|
||||||
toggleEntity(this.hass, ev.model.item.entity);
|
|
||||||
}
|
|
||||||
|
|
||||||
_handleImageClick() {
|
|
||||||
if (this._config.navigation_path) {
|
|
||||||
this.navigate(this._config.navigation_path);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this._config.camera_image) {
|
|
||||||
this.fire("hass-more-info", { entityId: this._config.camera_image });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
customElements.define("hui-picture-glance-card", HuiPictureGlanceCard);
|
|
236
src/panels/lovelace/cards/hui-picture-glance-card.ts
Normal file
236
src/panels/lovelace/cards/hui-picture-glance-card.ts
Normal file
@ -0,0 +1,236 @@
|
|||||||
|
import { html, LitElement, PropertyDeclarations } from "@polymer/lit-element";
|
||||||
|
import { classMap } from "lit-html/directives/classMap";
|
||||||
|
import { TemplateResult } from "lit-html";
|
||||||
|
|
||||||
|
import { hassLocalizeLitMixin } from "../../../mixins/lit-localize-mixin";
|
||||||
|
import { fireEvent } from "../../../common/dom/fire_event";
|
||||||
|
import { DOMAINS_TOGGLE } from "../../../common/const";
|
||||||
|
import { LovelaceCard, LovelaceConfig } from "../types";
|
||||||
|
import { EntityConfig } from "../entity-rows/types";
|
||||||
|
import { navigate } from "../../../common/navigate";
|
||||||
|
import { HomeAssistant } from "../../../types";
|
||||||
|
|
||||||
|
import computeStateDisplay from "../../../common/entity/compute_state_display";
|
||||||
|
import computeStateName from "../../../common/entity/compute_state_name";
|
||||||
|
import processConfigEntities from "../common/process-config-entities";
|
||||||
|
import computeDomain from "../../../common/entity/compute_domain";
|
||||||
|
import stateIcon from "../../../common/entity/state_icon";
|
||||||
|
import toggleEntity from "../common/entity/toggle-entity";
|
||||||
|
|
||||||
|
import "../../../components/ha-card";
|
||||||
|
import "../../../components/ha-icon";
|
||||||
|
import "../components/hui-image";
|
||||||
|
|
||||||
|
const STATES_OFF = new Set(["closed", "locked", "not_home", "off"]);
|
||||||
|
|
||||||
|
interface Config extends LovelaceConfig {
|
||||||
|
entities: EntityConfig[];
|
||||||
|
title?: string;
|
||||||
|
navigation_path?: string;
|
||||||
|
image?: string;
|
||||||
|
camera_image?: string;
|
||||||
|
state_image?: {};
|
||||||
|
aspect_ratio?: string;
|
||||||
|
entity?: string;
|
||||||
|
force_dialog?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
class HuiPictureGlanceCard extends hassLocalizeLitMixin(LitElement)
|
||||||
|
implements LovelaceCard {
|
||||||
|
public hass?: HomeAssistant;
|
||||||
|
private _config?: Config;
|
||||||
|
private _entitiesDialog?: EntityConfig[];
|
||||||
|
private _entitiesToggle?: EntityConfig[];
|
||||||
|
|
||||||
|
static get properties(): PropertyDeclarations {
|
||||||
|
return {
|
||||||
|
hass: {},
|
||||||
|
_config: {},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public getCardSize(): number {
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
public setConfig(config: Config): void {
|
||||||
|
if (
|
||||||
|
!config ||
|
||||||
|
!config.entities ||
|
||||||
|
!Array.isArray(config.entities) ||
|
||||||
|
!(config.image || config.camera_image || config.state_image) ||
|
||||||
|
(config.state_image && !config.entity)
|
||||||
|
) {
|
||||||
|
throw new Error("Invalid card configuration");
|
||||||
|
}
|
||||||
|
|
||||||
|
const entities = processConfigEntities(config.entities);
|
||||||
|
this._entitiesDialog = [];
|
||||||
|
this._entitiesToggle = [];
|
||||||
|
|
||||||
|
entities.forEach((item) => {
|
||||||
|
if (
|
||||||
|
config.force_dialog ||
|
||||||
|
!DOMAINS_TOGGLE.has(computeDomain(item.entity))
|
||||||
|
) {
|
||||||
|
this._entitiesDialog!.push(item);
|
||||||
|
} else {
|
||||||
|
this._entitiesToggle!.push(item);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this._config = config;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this._config || !this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
const isClickable =
|
||||||
|
this._config.navigation_path || this._config.camera_image;
|
||||||
|
|
||||||
|
return html`
|
||||||
|
${this.renderStyle()}
|
||||||
|
<ha-card>
|
||||||
|
<hui-image
|
||||||
|
class="${
|
||||||
|
classMap({
|
||||||
|
clickable: Boolean(isClickable),
|
||||||
|
})
|
||||||
|
}"
|
||||||
|
@click="${this._handleImageClick}"
|
||||||
|
.hass="${this.hass}"
|
||||||
|
.image="${this._config.image}"
|
||||||
|
.stateImage="${this._config.state_image}"
|
||||||
|
.cameraImage="${this._config.camera_image}"
|
||||||
|
.entity="${this._config.entity}"
|
||||||
|
.aspectRatio="${this._config.aspect_ratio}"
|
||||||
|
></hui-image>
|
||||||
|
<div class="box">
|
||||||
|
${
|
||||||
|
this._config.title
|
||||||
|
? html`
|
||||||
|
<div class="title">${this._config.title}</div>
|
||||||
|
`
|
||||||
|
: ""
|
||||||
|
}
|
||||||
|
<div>
|
||||||
|
${
|
||||||
|
this._entitiesDialog!.map((entityConf) =>
|
||||||
|
this.renderEntity(entityConf, true)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
${
|
||||||
|
this._entitiesToggle!.map((entityConf) =>
|
||||||
|
this.renderEntity(entityConf, false)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderEntity(
|
||||||
|
entityConf: EntityConfig,
|
||||||
|
dialog: boolean
|
||||||
|
): TemplateResult {
|
||||||
|
const stateObj = this.hass!.states[entityConf.entity];
|
||||||
|
|
||||||
|
if (!stateObj) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<ha-icon
|
||||||
|
.entity="${stateObj.entity_id}"
|
||||||
|
@click="${dialog ? this._openDialog : this._callService}"
|
||||||
|
class="${
|
||||||
|
classMap({
|
||||||
|
"state-on": !STATES_OFF.has(stateObj.state),
|
||||||
|
})
|
||||||
|
}"
|
||||||
|
.icon="${entityConf.icon || stateIcon(stateObj)}"
|
||||||
|
title="${
|
||||||
|
`
|
||||||
|
${computeStateName(stateObj)} : ${computeStateDisplay(
|
||||||
|
this.localize,
|
||||||
|
stateObj,
|
||||||
|
this.hass!.language
|
||||||
|
)}
|
||||||
|
`
|
||||||
|
}"
|
||||||
|
></ha-icon>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _openDialog(ev: MouseEvent): void {
|
||||||
|
fireEvent(this, "hass-more-info", { entityId: (ev.target as any).entity });
|
||||||
|
}
|
||||||
|
|
||||||
|
private _callService(ev: MouseEvent): void {
|
||||||
|
toggleEntity(this.hass, (ev.target as any).entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleImageClick(): void {
|
||||||
|
if (this._config!.navigation_path) {
|
||||||
|
navigate(this, this._config!.navigation_path!);
|
||||||
|
} else if (this._config!.camera_image) {
|
||||||
|
fireEvent(this, "hass-more-info", {
|
||||||
|
entityId: this._config!.camera_image,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderStyle(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<style>
|
||||||
|
ha-card {
|
||||||
|
position: relative;
|
||||||
|
min-height: 48px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
hui-image.clickable {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.box {
|
||||||
|
@apply --paper-font-common-nowrap;
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.3);
|
||||||
|
padding: 4px 8px;
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 40px;
|
||||||
|
color: white;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
.box .title {
|
||||||
|
font-weight: 500;
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
ha-icon {
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 8px;
|
||||||
|
color: #a9a9a9;
|
||||||
|
}
|
||||||
|
ha-icon.state-on {
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"hui-picture-glance-card": HuiPictureGlanceCard;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
customElements.define("hui-picture-glance-card", HuiPictureGlanceCard);
|
Loading…
x
Reference in New Issue
Block a user