2019-03-09 03:31:29 +00:00
|
|
|
'use strict';
|
|
|
|
const BLACK = '0,0,0';
|
|
|
|
const WHITE = '255,255,255';
|
|
|
|
const CMD = [
|
2019-07-26 21:28:27 +00:00
|
|
|
'CF',
|
|
|
|
'PF',
|
|
|
|
'BF',
|
|
|
|
'CB',
|
|
|
|
'PB',
|
|
|
|
'BB'
|
2019-03-09 03:31:29 +00:00
|
|
|
];
|
|
|
|
const ALTS = {
|
2019-07-26 21:28:27 +00:00
|
|
|
'CF': ['CAMERA FORWARD', 'CAM FORWARD'],
|
|
|
|
'PF': ['PROJECTOR FORWARD', 'PROJ FORWARD'],
|
|
|
|
'BF': ['BLACK FORWARD', 'BLACK', 'BLANK FORWARD', 'BLANK'],
|
|
|
|
'CB': ['CAMERA BACKWARD', 'CAM BACKWARD', 'CAMERA BACK', 'CAM BACK'],
|
|
|
|
'PB': ['PROJECTOR FORWARD', 'PROJ FORWARD', 'PROJECTOR BACK', 'PROJ BACK'],
|
|
|
|
'BB': ['BLACK BACKWARD', 'BLACK BACK', 'BLANK BACK'],
|
|
|
|
'L ': ['LIGHT', 'COLOR', 'LAMP'],
|
|
|
|
'F ': ['FADE']
|
2019-03-09 03:31:29 +00:00
|
|
|
};
|
2019-07-29 16:45:13 +00:00
|
|
|
const PAUSE = 'PAUSE';
|
|
|
|
const ALERT = 'ALERT';
|
2019-03-09 03:31:29 +00:00
|
|
|
/** helper functions */
|
2019-07-26 21:28:27 +00:00
|
|
|
/** startswith function from lodash, do not want the entire lib for this
|
|
|
|
* @param str {string} Text to evaluate
|
|
|
|
* @param target {string} Text to compare string against
|
|
|
|
* @param position {integer} Position in the string to make comparison at
|
|
|
|
*
|
|
|
|
* @returns {boolean} True for match, false for no match
|
|
|
|
**/
|
|
|
|
function startsWith(str, target, position) {
|
|
|
|
const { length } = str;
|
|
|
|
position = position == null ? 0 : position;
|
|
|
|
if (position < 0) {
|
|
|
|
position = 0;
|
|
|
|
}
|
|
|
|
else if (position > length) {
|
|
|
|
position = length;
|
|
|
|
}
|
|
|
|
target = `${target}`;
|
|
|
|
return str.slice(position, position + target.length) == target;
|
2019-03-09 03:31:29 +00:00
|
|
|
}
|
|
|
|
/** class Mscript */
|
|
|
|
class Mscript {
|
2019-07-26 22:12:06 +00:00
|
|
|
/**
|
|
|
|
* @constructor
|
|
|
|
* Create new Mscript interpreter
|
|
|
|
**/
|
2019-07-26 21:28:27 +00:00
|
|
|
constructor() {
|
|
|
|
this.output = {};
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Clear the state of the script
|
|
|
|
*/
|
|
|
|
clear() {
|
|
|
|
this.lines = [];
|
|
|
|
this.cam = 0;
|
|
|
|
this.proj = 0;
|
|
|
|
this.color = '';
|
|
|
|
this.loops = [];
|
|
|
|
this.rec = -1;
|
|
|
|
this.two = '';
|
|
|
|
this.arr = [];
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta = [];
|
2019-07-26 21:28:27 +00:00
|
|
|
this.target = 0; //move to target using CAM # or PROJ #
|
|
|
|
this.dist = 0;
|
|
|
|
this.variables = {};
|
|
|
|
this.output = {};
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Main function, accepts multi-line string, parses into lines
|
|
|
|
* and interprets the instructions from the text. Returns an array
|
2019-07-26 22:12:06 +00:00
|
|
|
* of steps to be fed into the mcopy sequence.
|
2019-07-26 21:28:27 +00:00
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} text Mscript text to interpret
|
|
|
|
* @param {function} callback Function to call when string is interpreted
|
2019-07-26 21:28:27 +00:00
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @returns {object} if callback is not provided
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
interpret(text, callback) {
|
|
|
|
this.clear();
|
|
|
|
if (typeof text === 'undefined') {
|
|
|
|
return this.fail('No input');
|
|
|
|
}
|
|
|
|
//split string into lines, each containing a command
|
|
|
|
this.lines = text.split('\n');
|
|
|
|
this.lines = this.lines.map(line => {
|
|
|
|
line = line.replace(/\t+/g, ''); //strip tabs
|
|
|
|
line = line.trim(); //remove excess whitespace before and after command
|
|
|
|
line = line.toUpperCase();
|
|
|
|
return line;
|
|
|
|
});
|
|
|
|
for (let line of this.lines) {
|
|
|
|
this.two = line.substring(0, 2);
|
|
|
|
if (CMD.indexOf(this.two) !== -1) {
|
|
|
|
this.basic_cmd(line);
|
|
|
|
}
|
2019-07-29 16:45:13 +00:00
|
|
|
else if (startsWith(line, PAUSE)) {
|
|
|
|
this.pause(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, ALERT)) {
|
|
|
|
this.alert(line);
|
|
|
|
}
|
2019-07-26 21:28:27 +00:00
|
|
|
else if (startsWith(line, '@') || line.indexOf('@') !== -1) {
|
|
|
|
this.variable(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'LOOP')) {
|
|
|
|
this.new_loop(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'L ')) {
|
|
|
|
this.light_state(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'F ')) {
|
|
|
|
this.new_loop(line, true);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'END')) {
|
|
|
|
this.end_loop(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'CAM')) { //directly go to that frame (black?)
|
|
|
|
this.move_cam(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'PROJ')) { //directly go to that frame
|
|
|
|
this.move_proj(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'SET')) { //set that state
|
|
|
|
this.set_state(line);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, '#') || startsWith(line, '//')) {
|
|
|
|
//comments
|
|
|
|
//ignore while parsing
|
|
|
|
}
|
2019-07-26 22:12:06 +00:00
|
|
|
else if (startsWith(line, 'ALERT')) {
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'PAUSE')) {
|
2019-07-26 23:54:22 +00:00
|
|
|
this.pause(line);
|
2019-07-26 22:12:06 +00:00
|
|
|
}
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
this.output.success = true;
|
|
|
|
this.output.arr = this.arr; //all instructions
|
2019-07-26 23:54:22 +00:00
|
|
|
this.output.meta = this.meta; //all metadata for instructions
|
2019-07-26 21:28:27 +00:00
|
|
|
this.output.cam = this.cam;
|
|
|
|
this.output.proj = this.proj;
|
|
|
|
if (typeof callback !== 'undefined') {
|
|
|
|
//should only be invoked by running mscript.tests()
|
|
|
|
callback(this.output);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return this.output;
|
|
|
|
}
|
|
|
|
}
|
2019-07-26 22:12:06 +00:00
|
|
|
/**
|
|
|
|
* Interprets variables for complex sequence behavior.
|
|
|
|
* TODO: Fully implement, add test coverage
|
|
|
|
*
|
|
|
|
* @param {string} line Line containing a variable assignment
|
|
|
|
*
|
|
|
|
**/
|
2019-07-26 21:28:27 +00:00
|
|
|
variable(line) {
|
|
|
|
let parts = line.split('=');
|
|
|
|
let key = parts[0];
|
|
|
|
let value = parts[1];
|
|
|
|
let update = false;
|
|
|
|
if (value && value.indexOf('#') !== -1) {
|
|
|
|
value = value.split('#')[0];
|
|
|
|
}
|
|
|
|
if (value && value.indexOf('//') !== -1) {
|
|
|
|
value = value.split('//')[0];
|
|
|
|
}
|
|
|
|
if (value && value.indexOf('+') !== -1) {
|
|
|
|
if (value)
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
if (line.indexOf('-') !== -1) {
|
|
|
|
update = true;
|
|
|
|
}
|
|
|
|
if (line.indexOf(',') === -1) { //if not color string
|
|
|
|
try {
|
|
|
|
value = parseInt(value);
|
|
|
|
}
|
|
|
|
catch (err) {
|
|
|
|
//supress parsing error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//console.dir(parts)
|
|
|
|
if (!this.variables[key] || update) {
|
|
|
|
this.variables[key] = value;
|
|
|
|
}
|
|
|
|
//console.dir(this.variables)
|
|
|
|
}
|
2019-07-26 22:12:06 +00:00
|
|
|
/**
|
|
|
|
* Replace variable with value at time of interpretation
|
|
|
|
* TODO: Implement this please
|
|
|
|
*
|
|
|
|
* @param {string} line Line containing variable to be replaced with value
|
|
|
|
*
|
|
|
|
* @returns {string} New string to be interpreted
|
|
|
|
**/
|
2019-07-26 21:28:27 +00:00
|
|
|
variable_replace(line) {
|
2019-07-26 22:12:06 +00:00
|
|
|
return line;
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Interpret a basic two character command
|
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} line Line of script to interpret
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
basic_cmd(line) {
|
|
|
|
if (this.rec !== -1) {
|
|
|
|
//hold generated arr in state loop array
|
|
|
|
this.loops[this.rec].arr
|
|
|
|
.push.apply(this.loops[this.rec].arr, this.str_to_arr(line, this.two));
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec].meta
|
|
|
|
.push.apply(this.loops[this.rec].meta, this.light_to_arr(line, this.two));
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.arr.push.apply(this.arr, this.str_to_arr(line, this.two));
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push.apply(this.meta, this.light_to_arr(line, this.two));
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Start a new loop
|
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} line Line to evaluate as either loop or fade
|
|
|
|
* @param {boolean} fade Flag as boolean if true
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
new_loop(line, fade) {
|
|
|
|
this.rec++;
|
|
|
|
this.loops[this.rec] = {
|
|
|
|
arr: [],
|
2019-07-26 23:54:22 +00:00
|
|
|
meta: [],
|
2019-07-26 21:28:27 +00:00
|
|
|
cam: 0,
|
|
|
|
proj: 0,
|
|
|
|
cmd: line + ''
|
|
|
|
};
|
|
|
|
if (fade) {
|
|
|
|
this.fade(line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Close the most recent loop
|
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} line Line to interpret
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
end_loop(line) {
|
2019-07-26 23:54:22 +00:00
|
|
|
let meta_arr;
|
2019-07-26 21:28:27 +00:00
|
|
|
let start;
|
|
|
|
let end;
|
|
|
|
let len;
|
|
|
|
for (let x = 0; x < this.loop_count(this.loops[this.rec].cmd); x++) {
|
2019-07-26 23:54:22 +00:00
|
|
|
meta_arr = this.loops[this.rec].meta;
|
2019-07-26 21:28:27 +00:00
|
|
|
if (this.loops[this.rec].fade) {
|
|
|
|
start = this.loops[this.rec].start;
|
|
|
|
end = this.loops[this.rec].end;
|
|
|
|
len = this.loops[this.rec].fade_len;
|
2019-07-26 23:54:22 +00:00
|
|
|
meta_arr = meta_arr.map(l => {
|
2019-07-26 21:28:27 +00:00
|
|
|
return this.fade_rgb(start, end, len, x);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (this.rec === 0) {
|
|
|
|
this.arr.push.apply(this.arr, this.loops[this.rec].arr);
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push.apply(this.meta, meta_arr);
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
else if (this.rec >= 1) {
|
|
|
|
this.loops[this.rec - 1].arr
|
|
|
|
.push.apply(this.loops[this.rec - 1].arr, this.loops[this.rec].arr);
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec - 1].meta
|
|
|
|
.push.apply(this.loops[this.rec - 1].meta, meta_arr);
|
2019-07-26 21:28:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this.update('END', this.loop_count(this.loops[this.rec].cmd));
|
|
|
|
delete this.loops[this.rec];
|
|
|
|
this.rec--;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Move camera to explicitly-defined frame
|
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} line Line to interpret with camera move statement
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
move_cam(line) {
|
|
|
|
this.target = parseInt(line.split('CAM ')[1]);
|
|
|
|
if (this.rec !== -1) {
|
|
|
|
if (this.target > this.cam) {
|
|
|
|
this.dist = this.target - this.cam;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.loops[this.rec].arr.push('BF');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec].meta.push(BLACK);
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('BF');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.dist = this.cam - this.target;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.loops[this.rec].arr.push('BB');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec].meta.push(BLACK);
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('BB');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (this.target > this.cam) {
|
|
|
|
this.dist = this.target - this.cam;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.arr.push('BF');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push(BLACK);
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('BF');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.dist = this.cam - this.target;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.arr.push('BB');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push(BLACK);
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('BB');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Move projector to explicitly-defined frame
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} line Line containing `move` statement to interpret
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
move_proj(line) {
|
|
|
|
this.target = parseInt(line.split('PROJ ')[1]);
|
|
|
|
if (this.rec !== -1) {
|
|
|
|
if (this.target > this.proj) {
|
|
|
|
this.dist = this.target - this.proj;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.loops[this.rec].arr.push('PF');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec].meta.push('');
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('PF');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.dist = this.proj - this.target;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.loops[this.rec].arr.push('PB');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.loops[this.rec].meta.push('');
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('PB');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (this.target > this.proj) {
|
|
|
|
this.dist = this.target - this.proj;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.arr.push('PF');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push('');
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('PF');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.dist = this.proj - this.target;
|
|
|
|
for (let x = 0; x < this.dist; x++) {
|
|
|
|
this.arr.push('PB');
|
2019-07-26 23:54:22 +00:00
|
|
|
this.meta.push('');
|
2019-07-26 21:28:27 +00:00
|
|
|
this.update('PB');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Set the state of either the cam or projector
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param line {string} String containing set statement
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
set_state(line) {
|
|
|
|
if (startsWith(line, 'SET CAM')) {
|
|
|
|
this.cam = parseInt(line.split('SET CAM')[1]);
|
|
|
|
}
|
|
|
|
else if (startsWith(line, 'SET PROJ')) {
|
|
|
|
this.proj = parseInt(line.split('SET PROJ')[1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Return the last loop
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @returns {object}
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
last_loop() {
|
|
|
|
return this.loops[this.loops.length - 1];
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Return the second-last loop
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @returns {object} Loop array
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
parent_loop() {
|
|
|
|
return this.loops[this.loops.length - 2];
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Extract the loop count integer from a LOOP cmd
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @returns {integer} Loop count in string parsed into integer
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
loop_count(str) {
|
|
|
|
return parseInt(str.split(' ')[1]);
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Execute a fade of frame length, from color to another color
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} line Line containing a fade initiator
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
fade(line) {
|
|
|
|
let len = this.fade_count(line);
|
|
|
|
let start = this.fade_start(line);
|
|
|
|
let end = this.fade_end(line);
|
|
|
|
this.loops[this.rec].start = start;
|
|
|
|
this.loops[this.rec].end = end;
|
|
|
|
this.loops[this.rec].fade = true;
|
|
|
|
this.loops[this.rec].fade_count = 0;
|
|
|
|
this.loops[this.rec].fade_len = len;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Extract the fade length integer from a FADE cmd
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} str Line containing the length of fade in frames
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
fade_count(str) {
|
|
|
|
return parseInt(str.split(' ')[1]);
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Extract the start color from a string
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} str Line containing the start color value in a fade initiator
|
|
|
|
*
|
|
|
|
* @returns {array} Array containing RGB color values
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
fade_start(str) {
|
|
|
|
let color = str.split(' ')[2];
|
|
|
|
return this.rgb(color.trim());
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Extract the end color from a string
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} str Line containing the end color value in a fade initiator
|
|
|
|
*
|
|
|
|
* @returns {array} Array containing RGB color values
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
fade_end(str) {
|
|
|
|
let color = str.split(' ')[3];
|
|
|
|
return this.rgb(color.trim());
|
|
|
|
}
|
2019-07-26 22:57:17 +00:00
|
|
|
/**
|
|
|
|
* Determine the state of a fade at a particular frame in the sequence, x
|
|
|
|
*
|
|
|
|
* @param {array} start Color the fade starts at
|
|
|
|
* @param {array} end Color the fade finishes at
|
|
|
|
* @param {integer} len Total length of the fade in frames
|
|
|
|
* @param {integer} x Position of the fade to get color value of
|
|
|
|
*
|
|
|
|
* @returns {array} Array containing RGB color values
|
|
|
|
*/
|
2019-07-26 21:28:27 +00:00
|
|
|
fade_rgb(start, end, len, x) {
|
|
|
|
let cur = [];
|
|
|
|
let diff;
|
|
|
|
for (let i = 0; i < 3; i++) {
|
|
|
|
if (x === len - 1) {
|
|
|
|
cur[i] = end[i];
|
|
|
|
}
|
|
|
|
else if (start[i] >= end[i]) {
|
|
|
|
diff = start[i] - end[i];
|
|
|
|
cur[i] = start[i] - Math.round((diff / (len - 1)) * x);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
diff = end[i] - start[i];
|
|
|
|
cur[i] = start[i] + Math.round((diff / (len - 1)) * x);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this.rgb_str(cur);
|
|
|
|
}
|
2019-07-26 22:12:06 +00:00
|
|
|
/**
|
|
|
|
* Parse string into array of RGB color values. 0-255 octet.
|
|
|
|
*
|
|
|
|
* @param {string} str String containing only color values as `#,#,#`
|
|
|
|
**/
|
2019-07-26 21:28:27 +00:00
|
|
|
rgb(str) {
|
|
|
|
let rgb = str.split(',');
|
|
|
|
return rgb.map((char) => {
|
|
|
|
return parseInt(char);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
/**
|
2019-07-26 22:12:06 +00:00
|
|
|
* Cast RGB color values as string
|
|
|
|
*
|
|
|
|
* @param {array} arr Array to join into string
|
2019-07-26 21:28:27 +00:00
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @returns {string} String of RGB values
|
2019-07-26 21:28:27 +00:00
|
|
|
**/
|
|
|
|
rgb_str(arr) {
|
|
|
|
return arr.join(',');
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Increase the state of a specific object, such as the camera/projector,
|
2019-07-26 22:12:06 +00:00
|
|
|
* by the value defined in val.
|
|
|
|
*
|
|
|
|
* @param {string} cmd String representing command to interpret and update state
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
update(cmd, val = 1) {
|
|
|
|
if (cmd === 'END') {
|
|
|
|
//I don't understand this loop
|
|
|
|
for (let i = 0; i < val; i++) {
|
|
|
|
if (this.rec === 0) {
|
|
|
|
this.cam += this.loops[this.rec].cam;
|
|
|
|
this.proj += this.loops[this.rec].proj;
|
|
|
|
}
|
|
|
|
else if (this.rec >= 1) {
|
|
|
|
this.loops[this.rec - 1].cam += this.loops[this.rec].cam;
|
|
|
|
this.loops[this.rec - 1].proj += this.loops[this.rec].proj;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'CF') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.cam += val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].cam += val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'CB') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.cam -= val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].cam--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'PF') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.proj += val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].proj += val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'PB') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.proj -= val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].proj--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'BF') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.cam += val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].cam += val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'BB') {
|
|
|
|
if (this.rec === -1) {
|
|
|
|
this.cam -= val;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.loops[this.rec].cam -= val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (cmd === 'L ') {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
2019-07-26 22:12:06 +00:00
|
|
|
* Split string on command, turn into array of commands
|
|
|
|
* as long as count variable. Default 1.
|
|
|
|
*
|
|
|
|
* @param {string} str String to split
|
|
|
|
* @param {string} cmd String representing command to split at
|
|
|
|
*
|
|
|
|
* @returns {array} Array containing commands
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
str_to_arr(str, cmd) {
|
|
|
|
const cnt = str.split(cmd);
|
|
|
|
let c = parseInt(cnt[1]);
|
|
|
|
let arr = [];
|
|
|
|
if (cnt[1] === '') {
|
|
|
|
c = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
c = parseInt(cnt[1]);
|
|
|
|
}
|
|
|
|
arr = new Array(c).fill(cmd);
|
|
|
|
this.update(cmd, c);
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Split a string on a command to extract data for light array
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} str String containing light command
|
|
|
|
* @param {string} cmd String representing command
|
|
|
|
*
|
|
|
|
* @returns {array} An RGB array containing the color values
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
light_to_arr(str, cmd) {
|
|
|
|
const cnt = str.split(cmd);
|
|
|
|
let c = parseInt(cnt[1]);
|
|
|
|
let arr = [];
|
|
|
|
if (cnt[1] === '') {
|
|
|
|
c = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
c = parseInt(cnt[1]);
|
|
|
|
}
|
|
|
|
for (let i = 0; i < c; i++) {
|
|
|
|
if (cmd === 'CF'
|
|
|
|
|| cmd === 'CB') {
|
|
|
|
arr.push(this.color);
|
|
|
|
}
|
|
|
|
else if (cmd === 'BF'
|
|
|
|
|| cmd === 'BB') {
|
|
|
|
arr.push(BLACK);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
arr.push('');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Split a string to extract an rgb color value
|
2019-07-26 22:12:06 +00:00
|
|
|
*
|
|
|
|
* @param {string} Color string assign to color property
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
light_state(str) {
|
|
|
|
//add parsers for other color spaces
|
|
|
|
const color = str.replace('L ', '').trim();
|
|
|
|
this.color = color;
|
|
|
|
}
|
2019-07-26 23:54:22 +00:00
|
|
|
/**
|
|
|
|
* Interpret a pause command
|
|
|
|
*
|
|
|
|
* @param {string} line String containing pause command
|
|
|
|
**/
|
|
|
|
pause(line) {
|
2019-07-29 16:45:13 +00:00
|
|
|
let lenStr = line.split(' ')[1] || '';
|
|
|
|
let len;
|
|
|
|
lenStr = lenStr.trim();
|
|
|
|
try {
|
|
|
|
len = parseInt(lenStr, 10); //clean up string or fail
|
|
|
|
}
|
|
|
|
catch (err) {
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
if (isNaN(len)) {
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
lenStr = String(len);
|
|
|
|
if (this.rec !== -1) {
|
|
|
|
//hold generated arr in state loop array
|
|
|
|
this.loops[this.rec].arr
|
|
|
|
.push('PA');
|
|
|
|
this.loops[this.rec].meta
|
|
|
|
.push(lenStr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.arr.push('AL');
|
|
|
|
this.meta.push(lenStr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Interpret an alert command
|
|
|
|
*
|
|
|
|
* @param {string} line String containing pause command
|
|
|
|
**/
|
|
|
|
alert(line) {
|
|
|
|
let msg = line.split(' ')[1] || '';
|
|
|
|
msg = msg.trim();
|
|
|
|
if (this.rec !== -1) {
|
|
|
|
//hold generated arr in state loop array
|
|
|
|
this.loops[this.rec].arr
|
|
|
|
.push('AL');
|
|
|
|
this.loops[this.rec].meta
|
|
|
|
.push(msg);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.arr.push('AL');
|
|
|
|
this.meta.push(msg);
|
|
|
|
}
|
2019-07-26 23:54:22 +00:00
|
|
|
}
|
2019-07-26 21:28:27 +00:00
|
|
|
/**
|
|
|
|
* Throw an error with specific message
|
|
|
|
*
|
2019-07-26 22:12:06 +00:00
|
|
|
* @param {string} msg Error message to print
|
2019-07-26 21:28:27 +00:00
|
|
|
*/
|
|
|
|
fail(msg) {
|
|
|
|
throw new Error(msg);
|
|
|
|
}
|
2019-03-09 03:31:29 +00:00
|
|
|
}
|
|
|
|
module.exports = Mscript;
|
2019-07-26 21:28:27 +00:00
|
|
|
//# sourceMappingURL=index.js.map
|