Skip to content

Commit

Permalink
Fix trailing energy gaps, refactor chart options (#19117)
Browse files Browse the repository at this point in the history
* Fix trailing gap on energy graph cards

* Use date methods instead of unix time calculation

* Fix trailing energy gaps, refactor chart options

---------

Co-authored-by: Till Fleisch <[email protected]>
  • Loading branch information
karwosts and TillFleisch authored Dec 27, 2023
1 parent bded31b commit df54687
Show file tree
Hide file tree
Showing 5 changed files with 231 additions and 520 deletions.
168 changes: 168 additions & 0 deletions src/panels/lovelace/cards/energy/common/energy-chart-options.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,168 @@
import { ChartOptions } from "chart.js";
import { HassConfig } from "home-assistant-js-websocket";
import {
addHours,
subHours,
differenceInDays,
differenceInHours,
} from "date-fns/esm";
import { FrontendLocaleData } from "../../../../../data/translation";
import {
formatNumber,
numberFormatToLocale,
} from "../../../../../common/number/format_number";
import { formatDateVeryShort } from "../../../../../common/datetime/format_date";
import { formatTime } from "../../../../../common/datetime/format_time";

export function getSuggestedMax(dayDifference: number, end: Date): number {
let suggestedMax = new Date(end);

// Sometimes around DST we get a time of 0:59 instead of 23:59 as expected.
// Correct for this when showing days/months so we don't get an extra day.
if (dayDifference > 2 && suggestedMax.getHours() === 0) {
suggestedMax = subHours(suggestedMax, 1);
}

suggestedMax.setMinutes(0, 0, 0);
if (dayDifference > 35) {
suggestedMax.setDate(1);
}
if (dayDifference > 2) {
suggestedMax.setHours(0);
}
return suggestedMax.getTime();
}

export function getCommonOptions(
start: Date,
end: Date,
locale: FrontendLocaleData,
config: HassConfig,
unit?: string,
compareStart?: Date,
compareEnd?: Date
): ChartOptions {
const dayDifference = differenceInDays(end, start);
const compare = compareStart !== undefined && compareEnd !== undefined;
if (compare && dayDifference <= 35) {
const difference = differenceInHours(end, start);
const differenceCompare = differenceInHours(compareEnd!, compareStart!);
// If the compare period doesn't match the main period, adjust them to match
if (differenceCompare > difference) {
end = addHours(end, differenceCompare - difference);
} else if (difference > differenceCompare) {
compareEnd = addHours(compareEnd!, difference - differenceCompare);
}
}

const options: ChartOptions = {
parsing: false,
animation: false,
interaction: {
mode: "x",
},
scales: {
x: {
type: "time",
suggestedMin: start.getTime(),
max: getSuggestedMax(dayDifference, end),
adapters: {
date: {
locale,
config,
},
},
ticks: {
maxRotation: 0,
sampleSize: 5,
autoSkipPadding: 20,
font: (context) =>
context.tick && context.tick.major
? ({ weight: "bold" } as any)
: {},
},
time: {
tooltipFormat:
dayDifference > 35
? "monthyear"
: dayDifference > 7
? "date"
: dayDifference > 2
? "weekday"
: dayDifference > 0
? "datetime"
: "hour",
minUnit:
dayDifference > 35 ? "month" : dayDifference > 2 ? "day" : "hour",
},
},
y: {
stacked: true,
type: "linear",
title: {
display: true,
text: unit,
},
ticks: {
beginAtZero: true,
callback: (value) => formatNumber(Math.abs(value), locale),
},
},
},
plugins: {
tooltip: {
position: "nearest",
filter: (val) => val.formattedValue !== "0",
itemSort: function (a, b) {
return b.datasetIndex - a.datasetIndex;
},
callbacks: {
title: (datasets) => {
if (dayDifference > 0) {
return datasets[0].label;
}
const date = new Date(datasets[0].parsed.x);
return `${
compare ? `${formatDateVeryShort(date, locale, config)}: ` : ""
}${formatTime(date, locale, config)}${formatTime(
addHours(date, 1),
locale,
config
)}`;
},
label: (context) =>
`${context.dataset.label}: ${formatNumber(
context.parsed.y,
locale
)} ${unit}`,
},
},
filler: {
propagate: false,
},
legend: {
display: false,
labels: {
usePointStyle: true,
},
},
},
elements: {
bar: { borderWidth: 1.5, borderRadius: 4 },
point: {
hitRadius: 50,
},
},
// @ts-expect-error
locale: numberFormatToLocale(locale),
};
if (compare) {
options.scales!.xAxisCompare = {
...(options.scales!.x as Record<string, any>),
suggestedMin: compareStart!.getTime(),
max: getSuggestedMax(dayDifference, compareEnd!),
display: false,
};
}
return options;
}
150 changes: 16 additions & 134 deletions src/panels/lovelace/cards/energy/hui-energy-gas-graph-card.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,14 +4,7 @@ import {
ChartOptions,
ScatterDataPoint,
} from "chart.js";
import {
addHours,
differenceInDays,
differenceInHours,
endOfToday,
isToday,
startOfToday,
} from "date-fns";
import { endOfToday, isToday, startOfToday } from "date-fns";
import { HassConfig, UnsubscribeFunc } from "home-assistant-js-websocket";
import {
css,
Expand All @@ -31,12 +24,7 @@ import {
rgb2lab,
} from "../../../../common/color/convert-color";
import { labBrighten, labDarken } from "../../../../common/color/lab";
import { formatDateVeryShort } from "../../../../common/datetime/format_date";
import { formatTime } from "../../../../common/datetime/format_time";
import {
formatNumber,
numberFormatToLocale,
} from "../../../../common/number/format_number";
import { formatNumber } from "../../../../common/number/format_number";
import "../../../../components/chart/ha-chart-base";
import "../../../../components/ha-card";
import {
Expand All @@ -56,6 +44,7 @@ import { HomeAssistant } from "../../../../types";
import { LovelaceCard } from "../../types";
import { EnergyGasGraphCardConfig } from "../types";
import { hasConfigChanged } from "../../common/has-changed";
import { getCommonOptions } from "./common/energy-chart-options";

@customElement("hui-energy-gas-graph-card")
export class HuiEnergyGasGraphCard
Expand Down Expand Up @@ -159,105 +148,23 @@ export class HuiEnergyGasGraphCard
compareStart?: Date,
compareEnd?: Date
): ChartOptions => {
const dayDifference = differenceInDays(end, start);
const compare = compareStart !== undefined && compareEnd !== undefined;
if (compare) {
const difference = differenceInHours(end, start);
const differenceCompare = differenceInHours(compareEnd!, compareStart!);
// If the compare period doesn't match the main period, adjust them to match
if (differenceCompare > difference) {
end = addHours(end, differenceCompare - difference);
} else if (difference > differenceCompare) {
compareEnd = addHours(compareEnd!, difference - differenceCompare);
}
}

const commonOptions = getCommonOptions(
start,
end,
locale,
config,
unit,
compareStart,
compareEnd
);
const options: ChartOptions = {
parsing: false,
animation: false,
interaction: {
mode: "x",
},
scales: {
x: {
type: "time",
suggestedMin: start.getTime(),
suggestedMax: end.getTime(),
adapters: {
date: {
locale,
config,
},
},
ticks: {
maxRotation: 0,
sampleSize: 5,
autoSkipPadding: 20,
font: (context) =>
context.tick && context.tick.major
? ({ weight: "bold" } as any)
: {},
},
time: {
tooltipFormat:
dayDifference > 35
? "monthyear"
: dayDifference > 7
? "date"
: dayDifference > 2
? "weekday"
: dayDifference > 0
? "datetime"
: "hour",
minUnit:
dayDifference > 35
? "month"
: dayDifference > 2
? "day"
: "hour",
},
offset: true,
},
y: {
stacked: true,
type: "linear",
title: {
display: true,
text: unit,
},
ticks: {
beginAtZero: true,
},
},
},
...commonOptions,
plugins: {
...commonOptions.plugins,
tooltip: {
position: "nearest",
filter: (val) => val.formattedValue !== "0",
itemSort: function (a, b) {
return b.datasetIndex - a.datasetIndex;
},
...commonOptions.plugins!.tooltip,
callbacks: {
title: (datasets) => {
if (dayDifference > 0) {
return datasets[0].label;
}
const date = new Date(datasets[0].parsed.x);
return `${
compare
? `${formatDateVeryShort(date, locale, config)}: `
: ""
}${formatTime(date, locale, config)}${formatTime(
addHours(date, 1),
locale,
config
)}`;
},
label: (context) =>
`${context.dataset.label}: ${formatNumber(
context.parsed.y,
locale
)} ${unit}`,
...commonOptions.plugins!.tooltip!.callbacks,
footer: (contexts) => {
if (contexts.length < 2) {
return [];
Expand All @@ -278,33 +185,8 @@ export class HuiEnergyGasGraphCard
},
},
},
filler: {
propagate: false,
},
legend: {
display: false,
labels: {
usePointStyle: true,
},
},
},
elements: {
bar: { borderWidth: 1.5, borderRadius: 4 },
point: {
hitRadius: 50,
},
},
// @ts-expect-error
locale: numberFormatToLocale(locale),
};
if (compare) {
options.scales!.xAxisCompare = {
...(options.scales!.x as Record<string, any>),
suggestedMin: compareStart!.getTime(),
suggestedMax: compareEnd!.getTime(),
display: false,
};
}
return options;
}
);
Expand Down
Loading

0 comments on commit df54687

Please sign in to comment.