2021-04-06 16:41:49 +02:00
|
|
|
const logger = require('./logger');
|
|
|
|
const blinkstick = require('blinkstick');
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
const LED_ANIMATED = 0;
|
|
|
|
|
|
|
|
const LEDS_ALL = 'all';
|
2021-04-06 16:41:49 +02:00
|
|
|
|
2022-02-22 17:00:30 +01:00
|
|
|
const MODE_SET = 'set';
|
|
|
|
const MODE_MORPH = 'morph';
|
2022-02-23 17:08:41 +01:00
|
|
|
const MODE_BLINK = 'blink';
|
2022-02-22 17:00:30 +01:00
|
|
|
const MODE_PULSE = 'pulse';
|
|
|
|
const MODE_POWEROFF = 'poweroff';
|
|
|
|
|
2022-02-22 16:42:49 +01:00
|
|
|
const LEDS = new Map();
|
2021-04-06 16:41:49 +02:00
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
let stop = false;
|
|
|
|
|
|
|
|
// find a connected blinkstick
|
2021-04-06 16:41:49 +02:00
|
|
|
function findBlinkstick() {
|
2022-02-21 00:48:29 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2022-02-23 17:08:41 +01:00
|
|
|
let led = blinkstick.findFirst();
|
|
|
|
if (led === undefined) {
|
|
|
|
reject('could not find any blinkstick');
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
resolve(led);
|
2021-04-06 16:41:49 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
// simple animation (set the color / morph to color)
|
|
|
|
async function simple(config) {
|
|
|
|
try {
|
|
|
|
await forceStop();
|
|
|
|
config.timestamp = new Date().getTime();
|
|
|
|
let indices = getIndices(config);
|
|
|
|
for (let index = 0; index < indices.length; index++) {
|
|
|
|
await setLedState(indices[index], LED_ANIMATED);
|
|
|
|
await singleAnimation(JSON.parse(JSON.stringify(config)), indices[index]);
|
|
|
|
await clearLedState(indices[index]);
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
return {
|
|
|
|
status: 'ok',
|
|
|
|
color: config.color,
|
|
|
|
indices: indices,
|
|
|
|
time: (new Date().getTime() - config.timestamp) + 'ms'
|
|
|
|
};
|
|
|
|
} catch (err) {
|
|
|
|
logger.error(err);
|
|
|
|
return err;
|
2022-02-22 17:00:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
// complex animation (pulse / blink)
|
|
|
|
async function complex(config) {
|
|
|
|
try {
|
|
|
|
if (config.timestamp === undefined) {
|
|
|
|
await powerOff({id: Math.random(), mode: MODE_POWEROFF, color: '#000000', options: {index: LEDS_ALL}});
|
|
|
|
config.timestamp = new Date().getTime();
|
|
|
|
}
|
|
|
|
let indices = getIndices(config);
|
|
|
|
for (let index = 0; index < indices.length; index++) {
|
|
|
|
if (shouldLedFinish(config)) {
|
|
|
|
return {status: 'ok', time: (new Date().getTime() - config.timestamp) + 'ms'};
|
|
|
|
}
|
|
|
|
await setLedState(indices[index], LED_ANIMATED);
|
|
|
|
await singleAnimation(JSON.parse(JSON.stringify(config)), indices[index]);
|
|
|
|
await clearLedState(indices[index]);
|
|
|
|
config.repetitions.done++;
|
|
|
|
}
|
|
|
|
return complex(config);
|
|
|
|
} catch (err) {
|
|
|
|
logger.error(err);
|
|
|
|
return err;
|
2022-02-22 17:00:30 +01:00
|
|
|
}
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
// power the blinkstick (or just a specific led) off
|
|
|
|
async function powerOff(config) {
|
|
|
|
try {
|
|
|
|
await forceStop();
|
|
|
|
config.timestamp = new Date().getTime();
|
|
|
|
let indices = getIndices(config);
|
|
|
|
for (let index = 0; index < indices.length; index++) {
|
2022-02-22 16:42:49 +01:00
|
|
|
await singleAnimation(JSON.parse(JSON.stringify(config)), indices[index]);
|
2022-02-23 17:08:41 +01:00
|
|
|
logger.info('led \'' + indices[index] + '\' powered off');
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
if (config.options.index === LEDS_ALL) {
|
|
|
|
const blinkstick = await findBlinkstick();
|
|
|
|
blinkstick.stop();
|
|
|
|
blinkstick.turnOff();
|
|
|
|
logger.info('blinkstick powered off');
|
|
|
|
}
|
|
|
|
return {status: 'ok', indices: indices, time: (new Date().getTime() - config.timestamp) + 'ms'};
|
|
|
|
} catch (err) {
|
|
|
|
return err;
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// animations
|
2022-02-23 17:08:41 +01:00
|
|
|
async function singleAnimation(config, index) {
|
|
|
|
try {
|
2022-02-22 16:42:49 +01:00
|
|
|
config.options.index = index;
|
2022-02-23 17:08:41 +01:00
|
|
|
const blinkstick = await findBlinkstick();
|
|
|
|
return await new Promise((resolve, reject) => {
|
|
|
|
logger.debug('changing color of led \'' + config.options.index + '\' to \'' + config.color + '\' (mode: ' + config.mode + ')...');
|
|
|
|
switch (config.mode) {
|
|
|
|
case MODE_MORPH:
|
|
|
|
blinkstick.morph(config.color, config.options, callback);
|
|
|
|
break;
|
|
|
|
case MODE_BLINK:
|
|
|
|
blinkstick.blink(config.color, config.options, callback);
|
|
|
|
break;
|
|
|
|
case MODE_PULSE:
|
|
|
|
blinkstick.pulse(config.color, config.options, callback);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
blinkstick.setColor(config.color, config.options, callback);
|
|
|
|
break;
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
2022-02-22 17:00:30 +01:00
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
function callback(err) {
|
|
|
|
if (err) {
|
|
|
|
reject('changing color of led \'' + config.options.index + '\' to \'' + config.color + '\' encountered an error > ' + err);
|
|
|
|
}
|
|
|
|
logger.debug('changed color of led \'' + config.options.index + '\' to \'' + config.color + '\' (mode: ' + config.mode + ')');
|
|
|
|
resolve();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch (err) {
|
|
|
|
return err;
|
|
|
|
}
|
2022-02-22 17:00:30 +01:00
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
async function forceStop() {
|
2022-02-21 00:48:29 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
2022-02-23 17:08:41 +01:00
|
|
|
stop = true;
|
|
|
|
waitForAllAnimationsEnd(() => {
|
|
|
|
stop = false;
|
|
|
|
resolve();
|
2022-02-22 16:42:49 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
function shouldLedFinish(config) {
|
|
|
|
if (stop) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (config.mode === MODE_BLINK && config.mode === MODE_PULSE) {
|
|
|
|
return config.repetitions.max !== 0 && config.repetitions.done >= config.repetitions.max
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-02-22 16:42:49 +01:00
|
|
|
// led / index helper functions
|
|
|
|
function getIndices(blinkstickConfig) {
|
|
|
|
if (blinkstickConfig.options.index === LEDS_ALL) {
|
|
|
|
return [0, 1, 2, 3, 4, 5, 6, 7];
|
|
|
|
}
|
|
|
|
return [blinkstickConfig.options.index];
|
|
|
|
}
|
|
|
|
|
|
|
|
async function setLedState(index, state) {
|
|
|
|
await stopLedAnimation(index);
|
|
|
|
LEDS.set(index, state);
|
|
|
|
}
|
|
|
|
|
|
|
|
function clearLedState(index) {
|
2022-02-23 17:08:41 +01:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
LEDS.delete(index);
|
|
|
|
resolve();
|
|
|
|
});
|
2022-02-22 16:42:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function isLedAnimated(index) {
|
|
|
|
return (LEDS.get(index) || LEDS.get(LEDS_ALL)) !== undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
function stopLedAnimation(index) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
waitForAnimationEnd(index, () => {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function waitForAnimationEnd(index, callback) {
|
|
|
|
if (!isLedAnimated(index)) {
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
setTimeout(() => {
|
|
|
|
waitForAnimationEnd(index, callback);
|
|
|
|
}, 100);
|
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
function waitForAllAnimationsEnd(callback) {
|
|
|
|
if (LEDS.size === 0) {
|
|
|
|
return callback();
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
setTimeout(() => {
|
|
|
|
waitForAllAnimationsEnd(callback);
|
|
|
|
}, 100);
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
|
|
|
|
2022-02-23 17:08:41 +01:00
|
|
|
function isInfiniteAnimation(config) {
|
|
|
|
if (config.mode !== MODE_BLINK && config.mode !== MODE_PULSE) {
|
|
|
|
return false;
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
if (config.options.index === LEDS_ALL) {
|
|
|
|
return config.repetitions.max === 0;
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
2022-02-23 17:08:41 +01:00
|
|
|
return config.options.repeats === 0;
|
2021-04-06 16:41:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// exports
|
|
|
|
module.exports = {
|
2022-02-23 17:08:41 +01:00
|
|
|
simple,
|
|
|
|
complex,
|
2022-02-22 16:42:49 +01:00
|
|
|
powerOff,
|
2022-02-23 17:08:41 +01:00
|
|
|
isInfiniteAnimation,
|
2022-02-22 17:00:30 +01:00
|
|
|
LEDS_ALL,
|
|
|
|
MODE_SET,
|
|
|
|
MODE_MORPH,
|
2022-02-23 17:08:41 +01:00
|
|
|
MODE_BLINK,
|
2022-02-22 17:00:30 +01:00
|
|
|
MODE_PULSE,
|
|
|
|
MODE_POWEROFF
|
2021-04-06 16:41:49 +02:00
|
|
|
};
|