310 lines
8.0 KiB
TypeScript
310 lines
8.0 KiB
TypeScript
import { mdiRefresh } from "@mdi/js";
|
|
import "@polymer/app-layout/app-header/app-header";
|
|
import "@polymer/app-layout/app-toolbar/app-toolbar";
|
|
import {
|
|
addDays,
|
|
endOfToday,
|
|
endOfWeek,
|
|
endOfYesterday,
|
|
startOfToday,
|
|
startOfWeek,
|
|
startOfYesterday,
|
|
} from "date-fns/esm";
|
|
import { css, html, LitElement, PropertyValues } from "lit";
|
|
import { property, state } from "lit/decorators";
|
|
import { navigate } from "../../common/navigate";
|
|
import {
|
|
createSearchParam,
|
|
extractSearchParam,
|
|
} from "../../common/url/search-params";
|
|
import { computeRTL } from "../../common/util/compute_rtl";
|
|
import "../../components/chart/state-history-charts";
|
|
import "../../components/entity/ha-entity-picker";
|
|
import "../../components/ha-circular-progress";
|
|
import "../../components/ha-date-range-picker";
|
|
import type { DateRangePickerRanges } from "../../components/ha-date-range-picker";
|
|
import "../../components/ha-icon-button";
|
|
import "../../components/ha-menu-button";
|
|
import { computeHistory, fetchDateWS } from "../../data/history";
|
|
import "../../layouts/ha-app-layout";
|
|
import { haStyle } from "../../resources/styles";
|
|
import { HomeAssistant } from "../../types";
|
|
|
|
class HaPanelHistory extends LitElement {
|
|
@property() hass!: HomeAssistant;
|
|
|
|
@property({ reflect: true, type: Boolean }) narrow!: boolean;
|
|
|
|
@property() _startDate: Date;
|
|
|
|
@property() _endDate: Date;
|
|
|
|
@property() _entityId = "";
|
|
|
|
@property() _isLoading = false;
|
|
|
|
@property() _stateHistory?;
|
|
|
|
@property({ reflect: true, type: Boolean }) rtl = false;
|
|
|
|
@state() private _ranges?: DateRangePickerRanges;
|
|
|
|
public constructor() {
|
|
super();
|
|
|
|
const start = new Date();
|
|
start.setHours(start.getHours() - 2, 0, 0, 0);
|
|
this._startDate = start;
|
|
|
|
const end = new Date();
|
|
end.setHours(end.getHours() + 1, 0, 0, 0);
|
|
this._endDate = end;
|
|
}
|
|
|
|
protected render() {
|
|
return html`
|
|
<ha-app-layout>
|
|
<app-header slot="header" fixed>
|
|
<app-toolbar>
|
|
<ha-menu-button
|
|
.hass=${this.hass}
|
|
.narrow=${this.narrow}
|
|
></ha-menu-button>
|
|
<div main-title>${this.hass.localize("panel.history")}</div>
|
|
<ha-icon-button
|
|
@click=${this._refreshHistory}
|
|
.disabled=${this._isLoading}
|
|
.path=${mdiRefresh}
|
|
.label=${this.hass.localize("ui.common.refresh")}
|
|
></ha-icon-button>
|
|
</app-toolbar>
|
|
</app-header>
|
|
|
|
<div class="filters">
|
|
<ha-date-range-picker
|
|
.hass=${this.hass}
|
|
?disabled=${this._isLoading}
|
|
.startDate=${this._startDate}
|
|
.endDate=${this._endDate}
|
|
.ranges=${this._ranges}
|
|
@change=${this._dateRangeChanged}
|
|
></ha-date-range-picker>
|
|
|
|
<ha-entity-picker
|
|
.hass=${this.hass}
|
|
.value=${this._entityId}
|
|
.label=${this.hass.localize(
|
|
"ui.components.entity.entity-picker.entity"
|
|
)}
|
|
.disabled=${this._isLoading}
|
|
@change=${this._entityPicked}
|
|
></ha-entity-picker>
|
|
</div>
|
|
${this._isLoading
|
|
? html`<div class="progress-wrapper">
|
|
<ha-circular-progress
|
|
active
|
|
alt=${this.hass.localize("ui.common.loading")}
|
|
></ha-circular-progress>
|
|
</div>`
|
|
: html`
|
|
<state-history-charts
|
|
virtualize
|
|
.hass=${this.hass}
|
|
.historyData=${this._stateHistory}
|
|
.endTime=${this._endDate}
|
|
.narrow=${this.narrow}
|
|
no-single
|
|
>
|
|
</state-history-charts>
|
|
`}
|
|
</ha-app-layout>
|
|
`;
|
|
}
|
|
|
|
protected firstUpdated(changedProps: PropertyValues) {
|
|
super.firstUpdated(changedProps);
|
|
|
|
const today = new Date();
|
|
const weekStart = startOfWeek(today);
|
|
const weekEnd = endOfWeek(today);
|
|
|
|
this._ranges = {
|
|
[this.hass.localize("ui.components.date-range-picker.ranges.today")]: [
|
|
startOfToday(),
|
|
endOfToday(),
|
|
],
|
|
[this.hass.localize("ui.components.date-range-picker.ranges.yesterday")]:
|
|
[startOfYesterday(), endOfYesterday()],
|
|
[this.hass.localize("ui.components.date-range-picker.ranges.this_week")]:
|
|
[weekStart, weekEnd],
|
|
[this.hass.localize("ui.components.date-range-picker.ranges.last_week")]:
|
|
[addDays(weekStart, -7), addDays(weekEnd, -7)],
|
|
};
|
|
|
|
this._entityId = extractSearchParam("entity_id") ?? "";
|
|
|
|
const startDate = extractSearchParam("start_date");
|
|
if (startDate) {
|
|
this._startDate = new Date(startDate);
|
|
}
|
|
const endDate = extractSearchParam("end_date");
|
|
if (endDate) {
|
|
this._endDate = new Date(endDate);
|
|
}
|
|
}
|
|
|
|
protected updated(changedProps: PropertyValues) {
|
|
if (
|
|
changedProps.has("_startDate") ||
|
|
changedProps.has("_endDate") ||
|
|
changedProps.has("_entityId")
|
|
) {
|
|
this._getHistory();
|
|
}
|
|
|
|
if (changedProps.has("hass")) {
|
|
const oldHass = changedProps.get("hass") as HomeAssistant | undefined;
|
|
if (!oldHass || oldHass.language !== this.hass.language) {
|
|
this.rtl = computeRTL(this.hass);
|
|
}
|
|
}
|
|
}
|
|
|
|
private _refreshHistory() {
|
|
this._getHistory();
|
|
}
|
|
|
|
private async _getHistory() {
|
|
this._isLoading = true;
|
|
const dateHistory = await fetchDateWS(
|
|
this.hass,
|
|
this._startDate,
|
|
this._endDate,
|
|
this._entityId
|
|
);
|
|
this._stateHistory = computeHistory(
|
|
this.hass,
|
|
dateHistory,
|
|
this.hass.localize
|
|
);
|
|
this._isLoading = false;
|
|
}
|
|
|
|
private _dateRangeChanged(ev) {
|
|
this._startDate = ev.detail.startDate;
|
|
const endDate = ev.detail.endDate;
|
|
if (endDate.getHours() === 0 && endDate.getMinutes() === 0) {
|
|
endDate.setDate(endDate.getDate() + 1);
|
|
endDate.setMilliseconds(endDate.getMilliseconds() - 1);
|
|
}
|
|
this._endDate = endDate;
|
|
|
|
this._updatePath();
|
|
}
|
|
|
|
private _entityPicked(ev) {
|
|
this._entityId = ev.target.value;
|
|
|
|
this._updatePath();
|
|
}
|
|
|
|
private _updatePath() {
|
|
const params: Record<string, string> = {};
|
|
|
|
if (this._entityId) {
|
|
params.entity_id = this._entityId;
|
|
}
|
|
|
|
if (this._startDate) {
|
|
params.start_date = this._startDate.toISOString();
|
|
}
|
|
|
|
if (this._endDate) {
|
|
params.end_date = this._endDate.toISOString();
|
|
}
|
|
|
|
navigate(`/history?${createSearchParam(params)}`, { replace: true });
|
|
}
|
|
|
|
static get styles() {
|
|
return [
|
|
haStyle,
|
|
css`
|
|
.content {
|
|
padding: 0 16px 16px;
|
|
}
|
|
|
|
state-history-charts {
|
|
height: calc(100vh - 136px);
|
|
}
|
|
|
|
:host([narrow]) state-history-charts {
|
|
height: calc(100vh - 198px);
|
|
}
|
|
|
|
.progress-wrapper {
|
|
height: calc(100vh - 136px);
|
|
}
|
|
|
|
:host([narrow]) .progress-wrapper {
|
|
height: calc(100vh - 198px);
|
|
}
|
|
|
|
:host([virtualize]) {
|
|
height: 100%;
|
|
}
|
|
|
|
.progress-wrapper {
|
|
position: relative;
|
|
}
|
|
|
|
.filters {
|
|
display: flex;
|
|
align-items: flex-end;
|
|
padding: 8px 16px 0;
|
|
}
|
|
|
|
:host([narrow]) .filters {
|
|
flex-wrap: wrap;
|
|
}
|
|
|
|
ha-date-range-picker {
|
|
margin-right: 16px;
|
|
margin-inline-end: 16px;
|
|
margin-inline-start: initial;
|
|
max-width: 100%;
|
|
direction: var(--direction);
|
|
}
|
|
|
|
:host([narrow]) ha-date-range-picker {
|
|
margin-right: 0;
|
|
margin-inline-end: 0;
|
|
margin-inline-start: initial;
|
|
direction: var(--direction);
|
|
}
|
|
|
|
ha-circular-progress {
|
|
position: absolute;
|
|
left: 50%;
|
|
top: 50%;
|
|
transform: translate(-50%, -50%);
|
|
}
|
|
|
|
ha-entity-picker {
|
|
display: inline-block;
|
|
flex-grow: 1;
|
|
max-width: 400px;
|
|
}
|
|
|
|
:host([narrow]) ha-entity-picker {
|
|
max-width: none;
|
|
width: 100%;
|
|
}
|
|
`,
|
|
];
|
|
}
|
|
}
|
|
|
|
customElements.define("ha-panel-history", HaPanelHistory);
|