forked from btclock/webui
Added individual LED control
This commit is contained in:
parent
58fa1e7ecb
commit
d3c6e52f3f
8 changed files with 237 additions and 55 deletions
|
@ -62,5 +62,10 @@
|
||||||
"timer": {
|
"timer": {
|
||||||
"running": "running",
|
"running": "running",
|
||||||
"stopped": "stopped"
|
"stopped": "stopped"
|
||||||
|
},
|
||||||
|
"sections": {
|
||||||
|
"control": {
|
||||||
|
"keepSameColor": "Keep same color"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,5 +62,10 @@
|
||||||
"timer": {
|
"timer": {
|
||||||
"running": "funcionando",
|
"running": "funcionando",
|
||||||
"stopped": "detenido"
|
"stopped": "detenido"
|
||||||
|
},
|
||||||
|
"sections": {
|
||||||
|
"control": {
|
||||||
|
"keepSameColor": "Mantén el mismo color"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -61,5 +61,10 @@
|
||||||
"timer": {
|
"timer": {
|
||||||
"running": "actief",
|
"running": "actief",
|
||||||
"stopped": "gestopt"
|
"stopped": "gestopt"
|
||||||
|
},
|
||||||
|
"sections": {
|
||||||
|
"control": {
|
||||||
|
"keepSameColor": "Behoud zelfde kleur"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,6 +14,17 @@
|
||||||
fgColor: "0"
|
fgColor: "0"
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let status = writable({
|
||||||
|
data: ["L", "O", "A", "D", "I", "N", "G"],
|
||||||
|
espFreeHeap: 0,
|
||||||
|
espHeapSize: 0,
|
||||||
|
connectionStatus: {
|
||||||
|
"price": false,
|
||||||
|
"blocks": false
|
||||||
|
},
|
||||||
|
leds: []
|
||||||
|
});
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
fetch( PUBLIC_BASE_URL + `/api/settings`)
|
fetch( PUBLIC_BASE_URL + `/api/settings`)
|
||||||
.then((res) => res.json())
|
.then((res) => res.json())
|
||||||
|
@ -31,6 +42,19 @@
|
||||||
}
|
}
|
||||||
settings.set(data);
|
settings.set(data);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
fetch(`${PUBLIC_BASE_URL}/api/status`)
|
||||||
|
.then((res) => res.json())
|
||||||
|
.then((data) => {
|
||||||
|
status.set(data);
|
||||||
|
});
|
||||||
|
|
||||||
|
const evtSource = new EventSource(`${PUBLIC_BASE_URL}/events`);
|
||||||
|
|
||||||
|
evtSource.addEventListener('status', (e) => {
|
||||||
|
let dataObj = (JSON.parse(e.data));
|
||||||
|
status.set(dataObj);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -40,8 +64,8 @@
|
||||||
|
|
||||||
<Container fluid>
|
<Container fluid>
|
||||||
<Row>
|
<Row>
|
||||||
<Control bind:settings></Control>
|
<Control bind:settings bind:status></Control>
|
||||||
<Status bind:settings></Status>
|
<Status bind:settings bind:status></Status>
|
||||||
<Settings bind:settings></Settings>
|
<Settings bind:settings></Settings>
|
||||||
</Row>
|
</Row>
|
||||||
</Container>
|
</Container>
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { PUBLIC_BASE_URL } from '$env/static/public';
|
import { PUBLIC_BASE_URL } from '$env/static/public';
|
||||||
|
import { onDestroy } from 'svelte';
|
||||||
import { _ } from 'svelte-i18n';
|
import { _ } from 'svelte-i18n';
|
||||||
|
import type { Subscriber, Unsubscriber } from 'svelte/motion';
|
||||||
|
import type { Writable } from 'svelte/store';
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
ButtonGroup,
|
ButtonGroup,
|
||||||
|
@ -19,15 +22,37 @@ import {
|
||||||
export let settings = {};
|
export let settings = {};
|
||||||
export let customText:String;
|
export let customText:String;
|
||||||
export let ledColor:String = "#FFCC00";
|
export let ledColor:String = "#FFCC00";
|
||||||
|
export let status:Writable<{leds:[]}>;
|
||||||
|
let ledStatus = [];
|
||||||
|
let keepLedsSameColor = false;
|
||||||
|
|
||||||
const setCustomText = () => {
|
const setCustomText = () => {
|
||||||
fetch(`${PUBLIC_BASE_URL}/api/show/text/${customText}`).catch(err => { });
|
fetch(`${PUBLIC_BASE_URL}/api/show/text/${customText}`).catch(err => { });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const checkSyncLeds = (e:Event) => {
|
||||||
|
console.log('checksyncleds', keepLedsSameColor);
|
||||||
|
if (keepLedsSameColor) {
|
||||||
|
console.log(e.target.value);
|
||||||
|
|
||||||
|
ledStatus.forEach((element, i) => {
|
||||||
|
if (ledStatus[i].hex != e.target_value) {
|
||||||
|
ledStatus[i].hex = e.target.value;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const setLEDcolor = () => {
|
const setLEDcolor = () => {
|
||||||
console.log(`${PUBLIC_BASE_URL}/api/lights/${ledColor}`);
|
console.log(`${PUBLIC_BASE_URL}/api/lights/${ledColor}`);
|
||||||
fetch(`${PUBLIC_BASE_URL}/api/lights/${encodeURIComponent(ledColor.replace(/^#/, ''))}`).catch(err => { });
|
fetch(`${PUBLIC_BASE_URL}/api/lights`, {
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
},
|
||||||
|
method: 'PATCH',
|
||||||
|
body: JSON.stringify(ledStatus)
|
||||||
|
}
|
||||||
|
).catch(err => { });
|
||||||
};
|
};
|
||||||
|
|
||||||
const turnOffLeds = () => {
|
const turnOffLeds = () => {
|
||||||
|
@ -42,6 +67,16 @@ const forceFullRefresh = () => {
|
||||||
fetch(`${PUBLIC_BASE_URL}/api/full_refresh`).catch(err => { });
|
fetch(`${PUBLIC_BASE_URL}/api/full_refresh`).catch(err => { });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let firstLedDataSubscription = () => {};
|
||||||
|
|
||||||
|
firstLedDataSubscription = status.subscribe(async(val) => {
|
||||||
|
if (val && val.leds) {
|
||||||
|
ledStatus = val.leds.map((obj) => ({ ["hex"]: obj["hex"] }));
|
||||||
|
firstLedDataSubscription();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
onDestroy(firstLedDataSubscription);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<Col>
|
<Col>
|
||||||
|
@ -54,7 +89,7 @@ const forceFullRefresh = () => {
|
||||||
<Row>
|
<Row>
|
||||||
<Label md={6} for="customText">{ $_('section.control.text') }</Label>
|
<Label md={6} for="customText">{ $_('section.control.text') }</Label>
|
||||||
<Col md="6">
|
<Col md="6">
|
||||||
<Input type="text" id="customText" bind:value={customText} bsSize="sm" maxLength="{$settings.numScreens}"/>
|
<Input type="text" id="customText" bind:value={customText} bsSize="sm" maxLength="{$settings.numScreens}" />
|
||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
<Button color="primary" on:click={setCustomText}>{ $_('section.control.showText') }</Button>
|
<Button color="primary" on:click={setCustomText}>{ $_('section.control.showText') }</Button>
|
||||||
|
@ -66,7 +101,18 @@ const forceFullRefresh = () => {
|
||||||
<Row>
|
<Row>
|
||||||
<Label md={6} for="ledColorPicker" size="sm">{ $_('section.control.ledColor') }</Label>
|
<Label md={6} for="ledColorPicker" size="sm">{ $_('section.control.ledColor') }</Label>
|
||||||
<Col md="6">
|
<Col md="6">
|
||||||
<Input type="color" id="ledColorPicker" bind:value="{ledColor}" />
|
<Row class="justify-content-between">
|
||||||
|
{#if ledStatus}
|
||||||
|
{#each ledStatus as led, i }
|
||||||
|
<Col>
|
||||||
|
<Input type="color" id="ledColorPicker[{i}]" bind:value="{led.hex}" class="mx-auto" on:change="{checkSyncLeds}" />
|
||||||
|
</Col>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
<Col>
|
||||||
|
<Input bind:checked={keepLedsSameColor} type="switch" class="mx-auto" label="{ $_('sections.control.keepSameColor') }" />
|
||||||
|
</Col>
|
||||||
|
</Row>
|
||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
<Button color="secondary" id="turnOffLedsBtn" on:click={turnOffLeds}>{ $_('section.control.turnOff') }</Button>
|
<Button color="secondary" id="turnOffLedsBtn" on:click={turnOffLeds}>{ $_('section.control.turnOff') }</Button>
|
||||||
|
|
|
@ -3,36 +3,12 @@
|
||||||
|
|
||||||
import { onMount } from 'svelte';
|
import { onMount } from 'svelte';
|
||||||
import { _ } from 'svelte-i18n';
|
import { _ } from 'svelte-i18n';
|
||||||
import { writable } from 'svelte/store';
|
import { writable, type Writable } from 'svelte/store';
|
||||||
import { Row, Input, Button, ButtonGroup, Card, CardBody, CardHeader, Col, Progress,CardTitle } from 'sveltestrap';
|
import { Row, Input, Button, ButtonGroup, Card, CardBody, CardHeader, Col, Progress,CardTitle } from 'sveltestrap';
|
||||||
import Rendered from './Rendered.svelte';
|
import Rendered from './Rendered.svelte';
|
||||||
|
|
||||||
export let settings;
|
export let settings;
|
||||||
|
export let status:Writable<{}>;
|
||||||
const status = writable({
|
|
||||||
data: ["L", "O", "A", "D", "I", "N", "G"],
|
|
||||||
espFreeHeap: 0,
|
|
||||||
espHeapSize: 0,
|
|
||||||
connectionStatus: {
|
|
||||||
"price": false,
|
|
||||||
"blocks": false
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
onMount(() => {
|
|
||||||
fetch(`${PUBLIC_BASE_URL}/api/status`)
|
|
||||||
.then((res) => res.json())
|
|
||||||
.then((data) => {
|
|
||||||
status.set(data);
|
|
||||||
});
|
|
||||||
|
|
||||||
const evtSource = new EventSource(`${PUBLIC_BASE_URL}/events`);
|
|
||||||
|
|
||||||
evtSource.addEventListener('status', (e) => {
|
|
||||||
let dataObj = (JSON.parse(e.data));
|
|
||||||
status.set(dataObj);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const toTime = (secs:Number) => {
|
const toTime = (secs:Number) => {
|
||||||
var hours = Math.floor(secs / (60 * 60));
|
var hours = Math.floor(secs / (60 * 60));
|
||||||
|
@ -60,11 +36,11 @@
|
||||||
let memoryFreePercent:number = 50;
|
let memoryFreePercent:number = 50;
|
||||||
let lightMode:boolean = false;
|
let lightMode:boolean = false;
|
||||||
|
|
||||||
status.subscribe((value) => {
|
status.subscribe((value: {}) => {
|
||||||
memoryFreePercent = Math.round(value.espFreeHeap / value.espHeapSize * 100);
|
memoryFreePercent = Math.round(value.espFreeHeap / value.espHeapSize * 100);
|
||||||
});
|
});
|
||||||
|
|
||||||
settings.subscribe((value) => {
|
settings.subscribe((value: {}) => {
|
||||||
lightMode = value.bgColor > value.fgColor;
|
lightMode = value.bgColor > value.fgColor;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
},
|
},
|
||||||
"servers": [
|
"servers": [
|
||||||
{
|
{
|
||||||
"url": "/api/"
|
"url": "http://192.168.20.231/api/"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"paths": {
|
"paths": {
|
||||||
|
@ -216,7 +216,37 @@
|
||||||
"summary": "Get LEDs status",
|
"summary": "Get LEDs status",
|
||||||
"responses": {
|
"responses": {
|
||||||
"200": {
|
"200": {
|
||||||
"description": "successful operation"
|
"description": "successful operation",
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/ArrayOfLeds"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"patch": {
|
||||||
|
"tags": [
|
||||||
|
"lights"
|
||||||
|
],
|
||||||
|
"summary": "Set individual LEDs",
|
||||||
|
"requestBody": {
|
||||||
|
"content": {
|
||||||
|
"application/json": {
|
||||||
|
"schema": {
|
||||||
|
"$ref": "#/components/schemas/ArrayOfLedsInput"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"responses": {
|
||||||
|
"200": {
|
||||||
|
"description": "succesful operation"
|
||||||
|
},
|
||||||
|
"400": {
|
||||||
|
"description": "invalid colors or wrong amount of LEDs"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -275,24 +305,69 @@
|
||||||
},
|
},
|
||||||
"components": {
|
"components": {
|
||||||
"schemas": {
|
"schemas": {
|
||||||
"ArrayOfLeds": {
|
"RgbColorValues": {
|
||||||
"type": "array",
|
|
||||||
"items": {
|
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"properties": {
|
"properties": {
|
||||||
"red": {
|
"red": {
|
||||||
"type": "integer"
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 255,
|
||||||
|
"example": 255
|
||||||
},
|
},
|
||||||
"green": {
|
"green": {
|
||||||
"type": "integer"
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 255,
|
||||||
|
"example": 204
|
||||||
},
|
},
|
||||||
"blue": {
|
"blue": {
|
||||||
"type": "integer"
|
"type": "integer",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 255,
|
||||||
|
"example": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
"RgbColorHex": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
"hex": {
|
"hex": {
|
||||||
"type": "string"
|
"type": "string",
|
||||||
|
"pattern": "^#(?:[0-9a-fA-F]{3}){1,2}$",
|
||||||
|
"example": "#FFCC00"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"RgbColorValueAndHex": {
|
||||||
|
"allOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/components/schemas/RgbColorValues"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"$ref": "#/components/schemas/RgbColorHex"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"RgbColorValueOrHex": {
|
||||||
|
"oneOf": [
|
||||||
|
{
|
||||||
|
"$ref": "#/components/schemas/RgbColorValues"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"$ref": "#/components/schemas/RgbColorHex"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"ArrayOfLeds": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/RgbColorValueAndHex"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"ArrayOfLedsInput": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"$ref": "#/components/schemas/RgbColorValueOrHex"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"Settings": {
|
"Settings": {
|
||||||
|
|
|
@ -4,7 +4,7 @@ info:
|
||||||
version: '3.0'
|
version: '3.0'
|
||||||
description: BTClock V3 API
|
description: BTClock V3 API
|
||||||
servers:
|
servers:
|
||||||
- url: /api/
|
- url: http://192.168.20.231/api/
|
||||||
paths:
|
paths:
|
||||||
/status:
|
/status:
|
||||||
get:
|
get:
|
||||||
|
@ -135,6 +135,24 @@ paths:
|
||||||
responses:
|
responses:
|
||||||
'200':
|
'200':
|
||||||
description: successful operation
|
description: successful operation
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: '#/components/schemas/ArrayOfLeds'
|
||||||
|
patch:
|
||||||
|
tags:
|
||||||
|
- lights
|
||||||
|
summary: Set individual LEDs
|
||||||
|
requestBody:
|
||||||
|
content:
|
||||||
|
application/json:
|
||||||
|
schema:
|
||||||
|
$ref: '#/components/schemas/ArrayOfLedsInput'
|
||||||
|
responses:
|
||||||
|
"200":
|
||||||
|
description: succesful operation
|
||||||
|
"400":
|
||||||
|
description: invalid colors or wrong amount of LEDs
|
||||||
/lights/{color}:
|
/lights/{color}:
|
||||||
get:
|
get:
|
||||||
tags:
|
tags:
|
||||||
|
@ -169,19 +187,47 @@ paths:
|
||||||
description: successful operation
|
description: successful operation
|
||||||
components:
|
components:
|
||||||
schemas:
|
schemas:
|
||||||
ArrayOfLeds:
|
RgbColorValues:
|
||||||
type: array
|
|
||||||
items:
|
|
||||||
type: object
|
type: object
|
||||||
properties:
|
properties:
|
||||||
red:
|
red:
|
||||||
type: integer
|
type: integer
|
||||||
|
minimum: 0
|
||||||
|
maximum: 255
|
||||||
|
example: 255
|
||||||
green:
|
green:
|
||||||
type: integer
|
type: integer
|
||||||
|
minimum: 0
|
||||||
|
maximum: 255
|
||||||
|
example: 204
|
||||||
blue:
|
blue:
|
||||||
type: integer
|
type: integer
|
||||||
|
minimum: 0
|
||||||
|
maximum: 255
|
||||||
|
example: 0
|
||||||
|
RgbColorHex:
|
||||||
|
type: object
|
||||||
|
properties:
|
||||||
hex:
|
hex:
|
||||||
type: string
|
type: string
|
||||||
|
pattern: ^#(?:[0-9a-fA-F]{3}){1,2}$
|
||||||
|
example: "#FFCC00"
|
||||||
|
RgbColorValueAndHex:
|
||||||
|
allOf:
|
||||||
|
- $ref: '#/components/schemas/RgbColorValues'
|
||||||
|
- $ref: '#/components/schemas/RgbColorHex'
|
||||||
|
RgbColorValueOrHex:
|
||||||
|
oneOf:
|
||||||
|
- $ref: '#/components/schemas/RgbColorValues'
|
||||||
|
- $ref: '#/components/schemas/RgbColorHex'
|
||||||
|
ArrayOfLeds:
|
||||||
|
type: array
|
||||||
|
items:
|
||||||
|
$ref: '#/components/schemas/RgbColorValueAndHex'
|
||||||
|
ArrayOfLedsInput:
|
||||||
|
type: array
|
||||||
|
items:
|
||||||
|
$ref: '#/components/schemas/RgbColorValueOrHex'
|
||||||
Settings:
|
Settings:
|
||||||
type: object
|
type: object
|
||||||
properties:
|
properties:
|
Loading…
Reference in a new issue