mirror of
https://github.com/joBr99/nspanel-lovelace-ui.git
synced 2025-12-22 15:34:26 +01:00
With the change, the offColor can also be used if, for example, the sockets are to be displayed in red and green
891 lines
32 KiB
TypeScript
891 lines
32 KiB
TypeScript
var Icons = new IconsSelector();
|
|
|
|
const Months = ["Januar", "Februar", "März", "April", "Mai", "Juni", "Juli", "August", "September", "Oktober", "November", "Dezember"];
|
|
const Days = ["Sonntag", "Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag"];
|
|
const Red: RGB = { red: 255, green: 0, blue: 0 };
|
|
const White: RGB = { red: 255, green: 255, blue: 255 };
|
|
const Yellow: RGB = { red: 253, green: 216, blue: 53 };
|
|
const Green: RGB = { red: 96, green: 176, blue: 62 }
|
|
const Brown: RGB = { red: 120, green: 80, blue: 59 }
|
|
const Gray: RGB = { red: 69, green: 69, blue: 69 }
|
|
const Off: RGB = { red: 68, green: 115, blue: 158 };
|
|
const On: RGB = { red: 253, green: 216, blue: 53 };
|
|
const BatteryFull: RGB = { red: 96, green: 176, blue: 62 }
|
|
const BatteryEmpty: RGB = { red: 179, green: 45, blue: 25 }
|
|
|
|
var Wohnen: PageEntities =
|
|
{
|
|
"type": "cardEntities",
|
|
"heading": "Haus",
|
|
"useColor": true,
|
|
"items": [
|
|
<PageItem>{ id: "alias.0.Stern", name: "Sternsteckdose"},
|
|
<PageItem>{ id: "alias.0.Erker"},
|
|
<PageItem>{ id: "alias.0.Küche", interpolateColor: true },
|
|
<PageItem>{ id: "alias.0.Wand" }
|
|
]
|
|
};
|
|
|
|
var Strom: PageEntities =
|
|
{
|
|
"type": "cardEntities",
|
|
"heading": "Strom",
|
|
"useColor": true,
|
|
"items": [
|
|
<PageItem>{ id: "alias.0.Netz", icon: "flash", interpolateColor: true, offColor: BatteryFull, onColor: Red, minValue: -1000, maxValue: 1000 },
|
|
<PageItem>{ id: "alias.0.Hausverbrauch", icon: "flash", interpolateColor: true, offColor: BatteryFull, onColor: Red, maxValue: 1000 },
|
|
<PageItem>{ id: "alias.0.Pv", icon: "solar-power", interpolateColor: true, offColor: Off, onColor: BatteryFull, maxValue: 1000 },
|
|
<PageItem>{ id: "alias.0.Batterie", icon: "battery-medium", interpolateColor: true, offColor: BatteryEmpty, onColor: BatteryFull }
|
|
]
|
|
};
|
|
|
|
var Müll: PageEntities =
|
|
{
|
|
"type": "cardEntities",
|
|
"heading": "Müllkalender",
|
|
"useColor": true,
|
|
"items": [
|
|
<PageItem>{ id: "alias.0.WzNsPanel.Müll.Bio_Tonne" ,unit:"Tage", icon: "trash-can",onColor: Brown},
|
|
<PageItem>{ id: "alias.0.WzNsPanel.Müll.Graue_Tonne" ,unit:"Tage",icon: "trash-can",onColor: Gray},
|
|
<PageItem>{ id: "alias.0.WzNsPanel.Müll.Grüne_Tonne" ,unit:"Tage",icon: "trash-can",onColor: Green},
|
|
<PageItem>{ id: "alias.0.WzNsPanel.Müll.Gelbe_Tonne" ,unit:"Tage",icon: "trash-can",onColor: Yellow}
|
|
]
|
|
};
|
|
|
|
|
|
var button1Page: PageGrid =
|
|
{
|
|
"type": "cardGrid",
|
|
"heading": "Radio",
|
|
"useColor": true,
|
|
"items": [
|
|
<PageItem>{ id: "alias.0.Radio.NJoy" },
|
|
<PageItem>{ id: "alias.0.Radio.Delta_Radio" },
|
|
<PageItem>{ id: "alias.0.Radio.NDR2" },
|
|
]
|
|
};
|
|
|
|
|
|
var button2Page: PageEntities =
|
|
{
|
|
"type": "cardEntities",
|
|
"heading": "Knopf2",
|
|
"useColor": true,
|
|
"items": [
|
|
<PageItem>{ id: "alias.0.Schlafen" },
|
|
<PageItem>{ id: "alias.0.Stern" }
|
|
]
|
|
};
|
|
|
|
export const config: Config = {
|
|
panelRecvTopic: "mqtt.0.tele.WzDisplay.RESULT",
|
|
panelSendTopic: "mqtt.0.cmnd.WzDisplay.CustomSend",
|
|
firstScreensaverEntity: { ScreensaverEntity: "alias.0.Wetter.HUMIDITY", ScreensaverEntityIcon: "water-percent", ScreensaverEntityText: "Luft", ScreensaverEntityUnitText: "%" },
|
|
secondScreensaverEntity: { ScreensaverEntity: "alias.0.Wetter.PRECIPITATION_CHANCE", ScreensaverEntityIcon: "weather-pouring", ScreensaverEntityText: "Regen", ScreensaverEntityUnitText: "%" },
|
|
thirdScreensaverEntity: { ScreensaverEntity: "alias.0.Batterie.ACTUAL", ScreensaverEntityIcon: "battery-medium", ScreensaverEntityText: "Batterie", ScreensaverEntityUnitText: "%" },
|
|
fourthScreensaverEntity: { ScreensaverEntity: "alias.0.Pv.ACTUAL", ScreensaverEntityIcon: "solar-power", ScreensaverEntityText: "PV", ScreensaverEntityUnitText: "W" },
|
|
timeoutScreensaver: 15,
|
|
dimmode: 8,
|
|
screenSaverDoubleClick: false,
|
|
locale: "de_DE",
|
|
timeFormat: "%H:%M",
|
|
dateFormat: "%A, %d. %B %Y",
|
|
weatherEntity: "alias.0.Wetter",
|
|
defaultOffColor: Off,
|
|
defaultOnColor: On,
|
|
defaultColor: Off,
|
|
temperatureUnit: "°C",
|
|
pages: [Wohnen, Strom, Müll,
|
|
{
|
|
"type": "cardThermo",
|
|
"heading": "Thermostat",
|
|
"useColor": true,
|
|
"items": [<PageItem>{ id: "alias.0.WzNsPanel", name: "Wohnzimmer" }]
|
|
}
|
|
],
|
|
button1Page: button1Page,
|
|
button2Page: button2Page
|
|
};
|
|
|
|
var subscriptions: any = {};
|
|
var screensaverEnabled : boolean = false;
|
|
var pageId = 0;
|
|
|
|
schedule("* * * * *", function () {
|
|
SendTime();
|
|
});
|
|
schedule("0 * * * *", function () {
|
|
SendDate();
|
|
});
|
|
|
|
|
|
// Only monitor the extra nodes if present
|
|
var updateArray: string[] = [];
|
|
if (config.firstScreensaverEntity !== null && config.firstScreensaverEntity.ScreensaverEntity != null && existsState(config.firstScreensaverEntity.ScreensaverEntity)) {
|
|
updateArray.push(config.firstScreensaverEntity.ScreensaverEntity)
|
|
}
|
|
if (config.secondScreensaverEntity !== null && config.secondScreensaverEntity.ScreensaverEntity != null && existsState(config.secondScreensaverEntity.ScreensaverEntity)) {
|
|
updateArray.push(config.secondScreensaverEntity.ScreensaverEntity)
|
|
}
|
|
if (config.thirdScreensaverEntity !== null && config.thirdScreensaverEntity.ScreensaverEntity != null && existsState(config.thirdScreensaverEntity.ScreensaverEntity)) {
|
|
updateArray.push(config.thirdScreensaverEntity.ScreensaverEntity)
|
|
}
|
|
if (config.fourthScreensaverEntity !== null && config.fourthScreensaverEntity.ScreensaverEntity != null && existsState(config.fourthScreensaverEntity.ScreensaverEntity)) {
|
|
updateArray.push(config.fourthScreensaverEntity.ScreensaverEntity)
|
|
}
|
|
|
|
if (updateArray.length > 0) {
|
|
on(updateArray, function () {
|
|
HandleScreensaverUpdate();
|
|
})
|
|
}
|
|
on({ id: config.panelRecvTopic }, function (obj) {
|
|
if (obj.state.val.startsWith('\{"CustomRecv":')) {
|
|
var json = JSON.parse(obj.state.val);
|
|
|
|
var split = json.CustomRecv.split(",");
|
|
HandleMessage(split[0], split[1], parseInt(split[2]), split);
|
|
}
|
|
});
|
|
|
|
function SendToPanel(val: Payload | Payload[]): void {
|
|
if (Array.isArray(val)) {
|
|
val.forEach(function (id, i) {
|
|
setState(config.panelSendTopic, id.payload);
|
|
});
|
|
}
|
|
else
|
|
setState(config.panelSendTopic, val.payload);
|
|
}
|
|
|
|
function HandleMessage(typ: string, method: string, page: number, words: Array<string>): void {
|
|
if (typ == "event") {
|
|
switch (method) {
|
|
case "startup":
|
|
screensaverEnabled = false;
|
|
UnsubscribeWatcher();
|
|
HandleStartupProcess();
|
|
pageId = 0;
|
|
GeneratePage(config.pages[0]);
|
|
break;
|
|
case "sleepReached":
|
|
screensaverEnabled = true;
|
|
if(pageId < 0)
|
|
pageId = 0;
|
|
HandleScreensaver();
|
|
break;
|
|
case "pageOpenDetail":
|
|
screensaverEnabled = false;
|
|
UnsubscribeWatcher();
|
|
let pageItem = config.pages[pageId].items.find(e => e.id === words[3]);
|
|
if (pageItem !== undefined)
|
|
SendToPanel(GenerateDetailPage(words[2], pageItem));
|
|
case "buttonPress2":
|
|
screensaverEnabled = false;
|
|
HandleButtonEvent(words);
|
|
break;
|
|
case "button1":
|
|
case "button2":
|
|
screensaverEnabled = false;
|
|
HandleHardwareButton(method);
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function GeneratePage(page: Page): void {
|
|
switch (page.type) {
|
|
case "cardEntities":
|
|
SendToPanel(GenerateEntitiesPage(<PageEntities>page));
|
|
break;
|
|
case "cardThermo":
|
|
SendToPanel(GenerateThermoPage(<PageThermo>page));
|
|
break;
|
|
case "cardGrid":
|
|
SendToPanel(GenerateGridPage(<PageGrid>page));
|
|
break;
|
|
}
|
|
}
|
|
|
|
function HandleHardwareButton(method: string): void {
|
|
let page: (PageThermo | PageEntities | PageGrid);
|
|
if (config.button1Page !== null && method == "button1") {
|
|
page = config.button1Page;
|
|
pageId = -1;
|
|
}
|
|
else if (config.button2Page !== null && method == "button2") {
|
|
page = config.button2Page;
|
|
pageId = -2;
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
GeneratePage(page);
|
|
}
|
|
|
|
function HandleStartupProcess(): void {
|
|
SendDate();
|
|
SendTime();
|
|
SendToPanel({ payload: "timeout~" + config.timeoutScreensaver });
|
|
SendToPanel({ payload: "dimmode~" + config.dimmode });
|
|
}
|
|
|
|
function SendDate(): void {
|
|
var d = new Date();
|
|
var day = Days[d.getDay()];
|
|
var date = d.getDate();
|
|
var month = Months[d.getMonth()];
|
|
var year = d.getFullYear();
|
|
var _sendDate = "date~" + day + " " + date + " " + month + " " + year;
|
|
SendToPanel(<Payload>{ payload: _sendDate });
|
|
}
|
|
|
|
function SendTime(): void {
|
|
var d = new Date();
|
|
var hr = d.getHours().toString();
|
|
var min = d.getMinutes().toString();
|
|
|
|
if (d.getHours() < 10) {
|
|
hr = "0" + d.getHours().toString();
|
|
}
|
|
if (d.getMinutes() < 10) {
|
|
min = "0" + d.getMinutes().toString();
|
|
}
|
|
SendToPanel(<Payload>{ payload: "time~" + hr + ":" + min });
|
|
}
|
|
|
|
function GenerateEntitiesPage(page: PageEntities): Payload[] {
|
|
var out_msgs: Array<Payload> = [];
|
|
out_msgs = [{ payload: "pageType~cardEntities" }]
|
|
out_msgs.push({ payload: GeneratePageElements(page) });
|
|
return out_msgs
|
|
}
|
|
|
|
function GenerateGridPage(page: PageGrid): Payload[] {
|
|
var out_msgs: Array<Payload> = [];
|
|
out_msgs = [{ payload: "pageType~cardGrid" }]
|
|
out_msgs.push({ payload: GeneratePageElements(page) });
|
|
return out_msgs
|
|
}
|
|
|
|
function GeneratePageElements(page: Page): string {
|
|
let maxItems = 0;
|
|
switch (page.type) {
|
|
case "cardThermo":
|
|
maxItems = 1;
|
|
break;
|
|
case "cardEntities":
|
|
maxItems = 4;
|
|
break;
|
|
case "cardGrid":
|
|
maxItems = 6;
|
|
break;
|
|
}
|
|
let pageData = "entityUpd~" + page.heading + "~" + GetNavigationString(pageId)
|
|
for (let index = 0; index < maxItems; index++) {
|
|
if (page.items[index] !== undefined) {
|
|
pageData += CreateEntity(page.items[index], index + 1, page.useColor);
|
|
}
|
|
else {
|
|
pageData += CreateEntity(<PageItem>{ id: "delete" }, index + 1);
|
|
}
|
|
}
|
|
return pageData;
|
|
}
|
|
|
|
function CreateEntity(pageItem: PageItem, placeId: number, useColors: boolean = false): string {
|
|
var iconId = "0"
|
|
if (pageItem.id == "delete") {
|
|
return "~delete~~~~~"
|
|
}
|
|
var name: string;
|
|
var type: string;
|
|
// ioBroker
|
|
if (existsObject(pageItem.id)) {
|
|
let o = getObject(pageItem.id)
|
|
var val = null;
|
|
name = pageItem.name !== undefined ? pageItem.name : o.common.name.de
|
|
|
|
if (existsState(pageItem.id + ".GET")) {
|
|
val = getState(pageItem.id + ".GET").val;
|
|
RegisterEntityWatcher(pageItem.id + ".GET");
|
|
}
|
|
else if (existsState(pageItem.id + ".SET")) {
|
|
val = getState(pageItem.id + ".SET").val;
|
|
RegisterEntityWatcher(pageItem.id + ".SET");
|
|
}
|
|
var iconColor = rgb_dec565(config.defaultColor);
|
|
|
|
switch (o.common.role) {
|
|
case "socket":
|
|
case "light":
|
|
type = "light"
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : o.common.role == "socket" ? Icons.GetIcon("power-socket-de") : Icons.GetIcon("lightbulb");
|
|
var optVal = "0"
|
|
|
|
if (val === true || val === "true") {
|
|
optVal = "1"
|
|
iconColor = GetIconColor(pageItem, true, useColors);
|
|
} else {
|
|
iconColor = GetIconColor(pageItem, false, useColors);
|
|
}
|
|
|
|
return "~" + type + "~" + pageItem.id + "~" + iconId + "~" + iconColor + "~" + name + "~" + optVal;
|
|
|
|
case "dimmer":
|
|
type = "light"
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : Icons.GetIcon("lightbulb");
|
|
var optVal = "0"
|
|
if (existsState(pageItem.id + ".ON_ACTUAL")) {
|
|
val = getState(pageItem.id + ".ON_ACTUAL").val;
|
|
RegisterEntityWatcher(pageItem.id + ".ON_ACTUAL");
|
|
}
|
|
else if (existsState(pageItem.id + ".ON_SET")) {
|
|
val = getState(pageItem.id + ".ON_SET").val;
|
|
RegisterEntityWatcher(pageItem.id + ".ON_SET");
|
|
}
|
|
if (val === true || val === "true") {
|
|
optVal = "1"
|
|
iconColor = GetIconColor(pageItem, existsState(pageItem.id + ".ACTUAL") ? getState(pageItem.id + ".ACTUAL").val : true, useColors);
|
|
}
|
|
|
|
return "~" + type + "~" + pageItem.id + "~" + iconId + "~" + iconColor + "~" + name + "~" + optVal;
|
|
|
|
case "blind":
|
|
type = "shutter"
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : Icons.GetIcon("window-open");
|
|
iconColor = GetIconColor(pageItem, existsState(pageItem.id + ".ACTUAL") ? getState(pageItem.id + ".ACTUAL").val : true, useColors);
|
|
return "~" + type + "~" + pageItem.id + "~" + iconId + "~" + iconColor + "~" + name + "~"
|
|
|
|
case "info":
|
|
case "value.temperature":
|
|
case "thermostat":
|
|
type = "text";
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : o.common.role == "value.temperature" || o.common.role == "thermostat" ? Icons.GetIcon("thermometer") : Icons.GetIcon("information-outline");
|
|
let unit = "";
|
|
var optVal = "0"
|
|
if (existsState(pageItem.id + ".ON_ACTUAL")) {
|
|
optVal = getState(pageItem.id + ".ON_ACTUAL").val;
|
|
unit = pageItem.unit !== undefined ? pageItem.unit : GetUnitOfMeasurement(pageItem.id + ".ON_ACTUAL");
|
|
RegisterEntityWatcher(pageItem.id + ".ON_ACTUAL");
|
|
}
|
|
else if (existsState(pageItem.id + ".ACTUAL")) {
|
|
optVal = getState(pageItem.id + ".ACTUAL").val;
|
|
unit = pageItem.unit !== undefined ? pageItem.unit : GetUnitOfMeasurement(pageItem.id + ".ACTUAL");
|
|
RegisterEntityWatcher(pageItem.id + ".ACTUAL");
|
|
}
|
|
|
|
if (o.common.role == "value.temperature") {
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : Icons.GetIcon("thermometer");
|
|
}
|
|
|
|
iconColor = GetIconColor(pageItem, parseInt(optVal), useColors);
|
|
|
|
return "~" + type + "~" + pageItem.id + "~" + iconId + "~" + iconColor + "~" + name + "~" + optVal + " " + unit;
|
|
|
|
case "button":
|
|
type = "button";
|
|
iconId = pageItem.icon !== undefined ? Icons.GetIcon(pageItem.icon) : Icons.GetIcon("gesture-tap-button");
|
|
let buttonText = pageItem.name !== undefined ? pageItem.name : "PRESS";
|
|
iconColor = GetIconColor(pageItem, true, useColors);
|
|
return "~" + type + "~" + pageItem.id + "~" + iconId + "~" + + iconColor + "~" + name + "~" + buttonText;
|
|
|
|
default:
|
|
return "~delete~~~~~";
|
|
}
|
|
}
|
|
|
|
return "~delete~~~~~"
|
|
}
|
|
|
|
function GetIconColor(pageItem: PageItem, value: (boolean | number), useColors: boolean): number {
|
|
// dimmer
|
|
if ((pageItem.useColor || useColors) && pageItem.interpolateColor && typeof (value) === "number") {
|
|
let maxValue = pageItem.maxValue !== undefined ? pageItem.maxValue : 100;
|
|
let minValue = pageItem.minValue !== undefined ? pageItem.minValue : 0;
|
|
value = value > maxValue ? maxValue : value;
|
|
value = value < minValue ? minValue : value;
|
|
return rgb_dec565(
|
|
Interpolate(
|
|
pageItem.offColor !== undefined ? pageItem.offColor : config.defaultOffColor,
|
|
pageItem.onColor !== undefined ? pageItem.onColor : config.defaultOnColor,
|
|
scale(value, minValue, maxValue, 0, 1)
|
|
));
|
|
}
|
|
|
|
if ((pageItem.useColor || useColors) && ((typeof (value) === "boolean" && value) || value > (pageItem.minValue !== undefined ? pageItem.minValue : 0))) {
|
|
return rgb_dec565(pageItem.onColor !== undefined ? pageItem.onColor : config.defaultOnColor)
|
|
}
|
|
|
|
return rgb_dec565(pageItem.offColor !== undefined ? pageItem.offColor : config.defaultOffColor);
|
|
}
|
|
|
|
function RegisterEntityWatcher(id: string): void {
|
|
if (subscriptions.hasOwnProperty(id)) {
|
|
return;
|
|
}
|
|
subscriptions[id] = (on({ id: id, change: 'any' }, function (data) {
|
|
if(pageId >= 0)
|
|
SendToPanel({ payload: GeneratePageElements(config.pages[pageId]) });
|
|
if(pageId == -1 && config.button1Page != undefined)
|
|
SendToPanel({ payload: GeneratePageElements(config.button1Page) });
|
|
if(pageId == -2 && config.button2Page != undefined)
|
|
SendToPanel({ payload: GeneratePageElements(config.button2Page) });
|
|
}))
|
|
}
|
|
|
|
|
|
function RegisterDetailEntityWatcher(id: string, pageItem: PageItem, type: string): void {
|
|
if (subscriptions.hasOwnProperty(id)) {
|
|
return;
|
|
}
|
|
subscriptions[id] = (on({ id: id, change: 'any' }, function () {
|
|
SendToPanel(GenerateDetailPage(type, pageItem));
|
|
}))
|
|
}
|
|
|
|
function GetUnitOfMeasurement(id: string): string {
|
|
if (!existsObject(id))
|
|
return "";
|
|
|
|
let obj = getObject(id);
|
|
if (typeof obj.common.unit !== 'undefined') {
|
|
return obj.common.unit
|
|
}
|
|
|
|
if (typeof obj.common.alias !== 'undefined' && typeof obj.common.alias.id !== 'undefined') {
|
|
return GetUnitOfMeasurement(obj.common.alias.id);
|
|
}
|
|
return "";
|
|
}
|
|
|
|
function GenerateThermoPage(page: PageThermo): Payload[] {
|
|
var id = page.items[0].id
|
|
var out_msgs: Array<Payload> = [];
|
|
out_msgs.push({ payload: "pageType~cardThermo" });
|
|
|
|
// ioBroker
|
|
if (existsObject(id)) {
|
|
let o = getObject(id)
|
|
let name = page.items[0].name !== undefined ? page.items[0].name : o.common.name.de
|
|
let currentTemp = 0;
|
|
if (existsState(id + ".ACTUAL"))
|
|
currentTemp = (Math.round(parseFloat(getState(id + ".ACTUAL").val) * 10)/10)*10;
|
|
|
|
let destTemp = 0;
|
|
if (existsState(id + ".SET")) {
|
|
destTemp = getState(id + ".SET").val.toFixed(2) * 10;
|
|
log(id + ".SET " + destTemp)
|
|
}
|
|
|
|
|
|
let status = ""
|
|
if (existsState(id + ".MODE"))
|
|
status = getState(id + ".MODE").val;
|
|
let minTemp = 180
|
|
let maxTemp = 300
|
|
let stepTemp = 5
|
|
|
|
out_msgs.push({ payload: "entityUpd~" + name + "~" + GetNavigationString(pageId) + "~" + id + "~" + currentTemp + "~" + destTemp + "~" + status + "~" + minTemp + "~" + maxTemp + "~" + stepTemp })
|
|
}
|
|
|
|
return out_msgs
|
|
}
|
|
|
|
|
|
function setIfExists(id: string, value: any, type: string | null = null): boolean {
|
|
if (type === null) {
|
|
if (existsState(id)) {
|
|
setState(id, value);
|
|
return true;
|
|
}
|
|
}
|
|
else {
|
|
let obj = getObject(id);
|
|
if (existsState(id) && obj.common.type !== undefined && obj.common.type === type) {
|
|
setState(id, value);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function toggleState(id: string): boolean {
|
|
let obj = getObject(id);
|
|
if (existsState(id) && obj.common.type !== undefined && obj.common.type === "boolean") {
|
|
setState(id, !getState(id).val);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function HandleButtonEvent(words): void {
|
|
let id = words[2]
|
|
let buttonAction = words[3];
|
|
|
|
switch (buttonAction) {
|
|
case "bNext":
|
|
var pageNum = ((pageId + 1) % config.pages.length);
|
|
pageId = Math.abs(pageNum);
|
|
UnsubscribeWatcher();
|
|
GeneratePage(config.pages[pageId]);
|
|
break;
|
|
case "bPrev":
|
|
var pageNum = ((pageId - 1) % config.pages.length);
|
|
pageId = Math.abs(pageNum);
|
|
UnsubscribeWatcher();
|
|
GeneratePage(config.pages[pageId]);
|
|
break;
|
|
case "bExit":
|
|
if (config.screenSaverDoubleClick) {
|
|
if (words[4] == 2)
|
|
GeneratePage(config.pages[pageId]);
|
|
}
|
|
else
|
|
GeneratePage(config.pages[pageId]);
|
|
break;
|
|
case "OnOff":
|
|
if (existsObject(id)) {
|
|
var action = false
|
|
if (words[4] == "1")
|
|
action = true
|
|
let o = getObject(id)
|
|
switch (o.common.role) {
|
|
case "light":
|
|
setIfExists(id + ".SET", action);
|
|
break;
|
|
case "dimmer":
|
|
setIfExists(id + ".ON_SET", action) ? true : setIfExists(id + ".ON_ACTUAL", action);
|
|
}
|
|
}
|
|
break;
|
|
case "up":
|
|
setIfExists(id + ".OPEN", true)
|
|
break;
|
|
case "stop":
|
|
setIfExists(id + ".STOP", true)
|
|
break;
|
|
case "down":
|
|
setIfExists(id + ".CLOSE", true)
|
|
break;
|
|
case "button":
|
|
toggleState(id + ".SET") ? true : toggleState(id + ".ON_SET")
|
|
break;
|
|
case "positionSlider":
|
|
case "brightnessSlider":
|
|
setIfExists(id + ".SET", parseInt(words[4])) ? true : setIfExists(id + ".ACTUAL", parseInt(words[4]));
|
|
break;
|
|
case "tempUpd":
|
|
setIfExists(id + ".SET", parseInt(words[4]) / 10)
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
function GetNavigationString(pageId: number): string {
|
|
switch (pageId) {
|
|
case 0:
|
|
return "0|1";
|
|
case config.pages.length - 1:
|
|
return "1|0";
|
|
case -1:
|
|
return "0|0";
|
|
default:
|
|
return "1|1";
|
|
}
|
|
}
|
|
|
|
|
|
function GenerateDetailPage(type: string, pageItem: PageItem): Payload[] {
|
|
|
|
var out_msgs: Array<Payload> = [];
|
|
let id = pageItem.id
|
|
if (existsObject(id)) {
|
|
var o = getObject(id)
|
|
var val: (boolean | number) = 0;
|
|
let icon = Icons.GetIcon("lightbulb");
|
|
var iconColor = rgb_dec565(config.defaultColor);
|
|
if (type == "popupLight") {
|
|
let switchVal = "0"
|
|
if (o.common.role == "light") {
|
|
if (existsState(id + ".GET")) {
|
|
val = getState(id + ".GET").val;
|
|
RegisterDetailEntityWatcher(id + ".GET", pageItem, type);
|
|
}
|
|
else if (existsState(id + ".SET")) {
|
|
val = getState(id + ".SET").val;
|
|
RegisterDetailEntityWatcher(id + ".SET", pageItem, type);
|
|
}
|
|
|
|
if (val) {
|
|
switchVal = "1";
|
|
iconColor = GetIconColor(pageItem, true, false);
|
|
}
|
|
|
|
out_msgs.push({ payload: "entityUpdateDetail~" + icon + "~" + + iconColor + "~" + switchVal + ",disable,disable,disable" })
|
|
}
|
|
|
|
if (o.common.role == "dimmer") {
|
|
if (existsState(id + ".ON_ACTUAL")) {
|
|
val = getState(id + ".ON_ACTUAL").val;
|
|
RegisterDetailEntityWatcher(id + ".ON_ACTUAL", pageItem, type);
|
|
}
|
|
|
|
else if (existsState(id + ".ON_SET")) {
|
|
val = getState(id + ".ON_SET").val;
|
|
RegisterDetailEntityWatcher(id + ".ON_SET", pageItem, type);
|
|
}
|
|
|
|
if (val === true) {
|
|
var iconColor = GetIconColor(pageItem, val, false);
|
|
switchVal = "1"
|
|
}
|
|
let brightness = 0;
|
|
if (existsState(id + ".ACTUAL")) {
|
|
brightness = Math.trunc(scale(getState(id + ".ACTUAL").val, 0, 100, 0, 100))
|
|
iconColor = GetIconColor(pageItem, brightness, false);
|
|
RegisterDetailEntityWatcher(id + ".ACTUAL", pageItem, type);
|
|
}
|
|
let colorTemp = "disable"
|
|
let colorMode = "disable"
|
|
//let attr_support_color = attr.supported_color_modes
|
|
//if (attr_support_color.includes("color_temp"))
|
|
// colortemp = Math.trunc(scale(attr.color_temp, attr.min_mireds, attr.max_mireds, 0, 100))
|
|
|
|
out_msgs.push({ payload: "entityUpdateDetail~" + icon + "~" + iconColor + "~" + switchVal + "~" + brightness + "~" + colorTemp + "~" + colorMode })
|
|
}
|
|
|
|
}
|
|
|
|
if (type == "popupShutter") {
|
|
if (existsState(id + ".ACTUAL"))
|
|
val = getState(id + ".ACTUAL").val;
|
|
else if (existsState(id + ".SET"))
|
|
val = getState(id + ".SET").val;
|
|
out_msgs.push({ payload: "entityUpdateDetail," + val })
|
|
}
|
|
}
|
|
return out_msgs
|
|
}
|
|
|
|
function scale(number: number, inMin: number, inMax: number, outMin: number, outMax: number): number {
|
|
return (number - inMin) * (outMax - outMin) / (inMax - inMin) + outMin;
|
|
}
|
|
|
|
function UnsubscribeWatcher(): void {
|
|
for (const [key, value] of Object.entries(subscriptions)) {
|
|
unsubscribe(value);
|
|
delete subscriptions[key]
|
|
}
|
|
}
|
|
|
|
function HandleScreensaver(): void {
|
|
SendToPanel({ payload: "pageType~screensaver"})
|
|
UnsubscribeWatcher();
|
|
HandleScreensaverUpdate();
|
|
}
|
|
|
|
function HandleScreensaverUpdate(): void {
|
|
if (screensaverEnabled && config.weatherEntity != null && existsObject(config.weatherEntity)) {
|
|
var icon = getState(config.weatherEntity + ".ICON").val;
|
|
|
|
let temperature: string =
|
|
existsState(config.weatherEntity + ".ACTUAL") ? getState(config.weatherEntity + ".ACTUAL").val :
|
|
existsState(config.weatherEntity + ".TEMP") ? getState(config.weatherEntity + ".TEMP").val : "null";
|
|
|
|
let payloadString =
|
|
"weatherUpdate~" + Icons.GetIcon(GetAccuWeatherIcon(parseInt(icon))) + "~"
|
|
+ temperature + " " + config.temperatureUnit + "~"
|
|
|
|
payloadString += GetScreenSaverEntityString(config.firstScreensaverEntity);
|
|
payloadString += GetScreenSaverEntityString(config.secondScreensaverEntity);
|
|
payloadString += GetScreenSaverEntityString(config.thirdScreensaverEntity);
|
|
payloadString += GetScreenSaverEntityString(config.fourthScreensaverEntity);
|
|
|
|
SendToPanel(<Payload>{ payload: payloadString });
|
|
}
|
|
}
|
|
|
|
function GetScreenSaverEntityString(configElement: ScreenSaverElement | null): string {
|
|
if (configElement != null && configElement.ScreensaverEntity != null && existsState(configElement.ScreensaverEntity)) {
|
|
let u1 = getState(configElement.ScreensaverEntity).val;
|
|
return configElement.ScreensaverEntityText + "~" + Icons.GetIcon(configElement.ScreensaverEntityIcon) + "~" + u1 + " " + configElement.ScreensaverEntityUnitText + "~";
|
|
}
|
|
else {
|
|
return "~~~";
|
|
}
|
|
}
|
|
|
|
function GetAccuWeatherIcon(icon: number): string {
|
|
switch (icon) {
|
|
case 24: // Ice
|
|
case 30: // Hot
|
|
case 31: // Cold
|
|
return "window-open"; // exceptional
|
|
|
|
case 7: // Cloudy
|
|
case 8: // Dreary (Overcast)
|
|
case 38: // Mostly Cloudy
|
|
return "weather-cloudy"; // cloudy
|
|
|
|
case 11: // fog
|
|
return "weather-fog"; // fog
|
|
|
|
case 25: // Sleet
|
|
return "weather-hail"; // Hail
|
|
|
|
case 15: // T-Storms
|
|
return "weather-lightning"; // lightning
|
|
|
|
case 16: // Mostly Cloudy w/ T-Storms
|
|
case 17: // Partly Sunny w/ T-Storms
|
|
case 41: // Partly Cloudy w/ T-Storms
|
|
case 42: // Mostly Cloudy w/ T-Storms
|
|
return "weather-lightning-rainy"; // lightning-rainy
|
|
|
|
case 33: // Clear
|
|
case 34: // Mostly Clear
|
|
case 37: // Hazy Moonlight
|
|
return "weather-night";
|
|
|
|
case 3: // Partly Sunny
|
|
case 4: // Intermittent Clouds
|
|
case 6: // Mostly Cloudy
|
|
case 35: // Partly Cloudy
|
|
case 36: // Intermittent Clouds
|
|
return "weather-partly-cloudy"; // partlycloudy
|
|
|
|
case 18: // pouring
|
|
return "weather-pouring"; // pouring
|
|
|
|
case 12: // Showers
|
|
case 13: // Mostly Cloudy w/ Showers
|
|
case 14: // Partly Sunny w/ Showers
|
|
case 26: // Freezing Rain
|
|
case 39: // Partly Cloudy w/ Showers
|
|
case 40: // Mostly Cloudy w/ Showers
|
|
return "weather-rainy"; // rainy
|
|
|
|
case 19: // Flurries
|
|
case 20: // Mostly Cloudy w/ Flurries
|
|
case 21: // Partly Sunny w/ Flurries
|
|
case 22: // Snow
|
|
case 23: // Mostly Cloudy w/ Snow
|
|
case 43: // Mostly Cloudy w/ Flurries
|
|
case 44: // Mostly Cloudy w/ Snow
|
|
return "weather-snowy"; // snowy
|
|
|
|
case 29: // Rain and Snow
|
|
return "weather-snowy-rainy"; // snowy-rainy
|
|
|
|
case 1: // Sunny
|
|
case 2: // Mostly Sunny
|
|
case 5: // Hazy Sunshine
|
|
return "weather-sunny"; // sunny
|
|
|
|
case 32: // windy
|
|
return "weather-windy"; // windy
|
|
|
|
default:
|
|
return "alert-circle-outline";
|
|
}
|
|
}
|
|
|
|
function GetBlendedColor(percentage: number): RGB {
|
|
if (percentage < 50)
|
|
return Interpolate(config.defaultOffColor, config.defaultOnColor, percentage / 50.0);
|
|
return Interpolate(Red, White, (percentage - 50) / 50.0);
|
|
}
|
|
|
|
function Interpolate(color1: RGB, color2: RGB, fraction: number): RGB {
|
|
var r: number = InterpolateNum(color1.red, color2.red, fraction);
|
|
var g: number = InterpolateNum(color1.green, color2.green, fraction);
|
|
var b: number = InterpolateNum(color1.blue, color2.blue, fraction);
|
|
return <RGB>{ red: Math.round(r), green: Math.round(g), blue: Math.round(b) };
|
|
}
|
|
|
|
function InterpolateNum(d1: number, d2: number, fraction: number): number {
|
|
return d1 + (d2 - d1) * fraction;
|
|
}
|
|
|
|
function rgb_dec565(rgb: RGB): number {
|
|
return ((Math.floor(rgb.red / 255 * 31) << 11) | (Math.floor(rgb.green / 255 * 63) << 5) | (Math.floor(rgb.blue / 255 * 31)));
|
|
}
|
|
|
|
type RGB = {
|
|
red: number,
|
|
green: number,
|
|
blue: number
|
|
};
|
|
|
|
type Payload = {
|
|
payload: string;
|
|
};
|
|
|
|
type Page = {
|
|
type: string,
|
|
heading: string,
|
|
items: PageItem[],
|
|
useColor: boolean
|
|
};
|
|
|
|
interface PageEntities extends Page {
|
|
type: "cardEntities",
|
|
items: PageItem[],
|
|
|
|
};
|
|
interface PageGrid extends Page {
|
|
type: "cardGrid",
|
|
items: PageItem[],
|
|
};
|
|
|
|
interface PageThermo extends Page {
|
|
type: "cardThermo",
|
|
items: PageItem[],
|
|
};
|
|
|
|
type PageItem = {
|
|
id: string,
|
|
icon: (string | undefined),
|
|
onColor: (RGB | undefined),
|
|
offColor: (RGB | undefined),
|
|
useColor: (boolean | undefined),
|
|
interpolateColor: (boolean | undefined),
|
|
minValue: (number | undefined),
|
|
maxValue: (number | undefined),
|
|
name: (string | undefined),
|
|
unit: (string | undefined)
|
|
}
|
|
|
|
type Config = {
|
|
panelRecvTopic: string,
|
|
panelSendTopic: string,
|
|
timeoutScreensaver: number,
|
|
dimmode: number,
|
|
//brightnessScreensaver:
|
|
locale: string,
|
|
timeFormat: string,
|
|
dateFormat: string,
|
|
weatherEntity: string | null,
|
|
screenSaverDoubleClick: boolean,
|
|
temperatureUnit: string,
|
|
firstScreensaverEntity: ScreenSaverElement | null,
|
|
secondScreensaverEntity: ScreenSaverElement | null,
|
|
thirdScreensaverEntity: ScreenSaverElement | null,
|
|
fourthScreensaverEntity: ScreenSaverElement | null,
|
|
defaultColor: RGB,
|
|
defaultOnColor: RGB,
|
|
defaultOffColor: RGB,
|
|
pages: (PageThermo | PageEntities | PageGrid)[],
|
|
button1Page: (PageThermo | PageEntities | PageGrid | null),
|
|
button2Page: (PageThermo | PageEntities | PageGrid | null),
|
|
};
|
|
|
|
type ScreenSaverElement = {
|
|
ScreensaverEntity: string | null,
|
|
ScreensaverEntityIcon: string | null,
|
|
ScreensaverEntityText: string | null,
|
|
ScreensaverEntityUnitText: string | null,
|
|
}
|