From 855e61083a281427485802593de097155697c987 Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 18:32:54 +0100 Subject: [PATCH 1/6] feat(device): add new `novation-launchpad-mini` --- .../devices/known-devices/known-devices.ts | 2 + .../known-devices/novation-launchpad-mini.ts | 73 +++++++++++++++++++ 2 files changed, 75 insertions(+) create mode 100644 src/app/devices/known-devices/novation-launchpad-mini.ts diff --git a/src/app/devices/known-devices/known-devices.ts b/src/app/devices/known-devices/known-devices.ts index eb45171..2acbc26 100644 --- a/src/app/devices/known-devices/known-devices.ts +++ b/src/app/devices/known-devices/known-devices.ts @@ -1,8 +1,10 @@ import { Controller, Selector } from '../device/device.types'; import { novationLaunchpadX } from './novation-launchpad-x'; import { novationLaunchControlXl } from './novation-launch-control-xl'; +import { novationLaunchpadMini } from './novation-launchpad-mini'; export const knownDevices: (Selector|Controller)[] = [ novationLaunchpadX, novationLaunchControlXl, + novationLaunchpadMini, ]; diff --git a/src/app/devices/known-devices/novation-launchpad-mini.ts b/src/app/devices/known-devices/novation-launchpad-mini.ts new file mode 100644 index 0000000..aef2157 --- /dev/null +++ b/src/app/devices/known-devices/novation-launchpad-mini.ts @@ -0,0 +1,73 @@ +import { Selector, DeviceCategory } from '../device/device.types'; + +/** + * Novation Launchpad Mini + * + * @see http://leemans.ch/latex/doc_launchpad-programmers-reference.pdf + */ +export const novationLaunchpadMini: Selector = { + category: DeviceCategory.select, + manufacturer: 'Focusrite A.E. Ltd', + name: 'Launchpad Mini', + channels: { + input: 'all', + output: 1, + }, + lights: { + first: 0, + last: 120, + }, + grid: [ + // describe columns + // top down then left to right + [0, 16, 32, 48, 64, 80, 96, 112], + [1, 17, 33, 49, 65, 81, 97, 113], + [2, 18, 34, 50, 66, 82, 98, 114], + [3, 19, 35, 51, 67, 83, 99, 115], + [4, 20, 36, 52, 68, 84, 100, 116], + [5, 21, 37, 53, 69, 85, 101, 117], + [6, 22, 38, 54, 70, 86, 102, 118], + [7, 23, 39, 55, 71, 87, 103, 119], + ], + functionKeys: { + firstRow: [91, 92, 93, 94, 95, 96, 97, 98], // not available + lastColumn: [8, 24, 40, 56, 72, 88, 104, 120], + }, + colors: { + black: 0, + off: 0, + red: 15, + amber: 63, + yellow: 62, + green: 60, + }, + get colorByState () { + return { + inputOn: this.colors.red, + inputOff: this.colors.black, + neuronOn: this.colors.green, + neuronOff: this.colors.black, + neuronSelected: this.colors.yellow, + outputWeightOn: this.colors.red, + outputWeightOff: this.colors.black, + playbackOn: this.colors.green, + playbackOff: this.colors.red, + layerOn: this.colors.amber, + layerOff: this.colors.black, + }; + }, + time: { + wait: 200, + defaultDuration: 500, + longClick: 400, + }, + get bootSequence () { + return { + color: this.colors.red, + sysex: { + manufacturer: 0, + data: [32, 41, 2, 12, 14, 1], + }, + }; + }, +}; From 666981e571ea3d5f0bc8209bdb98a9279f63c468 Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 18:58:53 +0100 Subject: [PATCH 2/6] fix: make the controller update when mouse click on layer button --- src/app/ui/playground.ui.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/app/ui/playground.ui.ts b/src/app/ui/playground.ui.ts index abcb286..bf21280 100644 --- a/src/app/ui/playground.ui.ts +++ b/src/app/ui/playground.ui.ts @@ -4,6 +4,7 @@ import { mappingsUi } from './mappings.ui'; import { networkState } from '../state/network.state'; import { layerCardUi } from './layer-card.ui'; import { selectorDevice } from '../devices/selector.device'; +import { controllerDevice } from '../devices/controller.device'; export const playgroundUi = Object.create (null); @@ -120,6 +121,7 @@ playgroundUi.attachLayers = function () { this.renderLayers (); layerCardUi.updateCard (); selectorDevice.renderLayers (); + controllerDevice.updateMode (); }; }); }; From e024c73ff3284d36be100db3fa0ca108742dbb1c Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 18:59:27 +0100 Subject: [PATCH 3/6] fix(selector): add self check before rendering layer buttons --- src/app/devices/selector.device.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/app/devices/selector.device.ts b/src/app/devices/selector.device.ts index 2dd99dd..ced70ea 100644 --- a/src/app/devices/selector.device.ts +++ b/src/app/devices/selector.device.ts @@ -369,6 +369,10 @@ selectorDevice.updateLightPlayback = function () { }; selectorDevice.renderLayers = function () { + if (!this.isInitialized) { + return; + } + const layerPads = this.settings.functionKeys.firstRow.slice (1, -1); const index = networkState.selectedLayerIndex; From a7db018f8fa9c29f72e429bc85d27d95b57915de Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 19:02:40 +0100 Subject: [PATCH 4/6] perf(controller/layer-mode): add green color buttons for neurons that are selected within the selected layer --- src/app/devices/controller.device.ts | 57 ++++++++++++++++++++++++++-- 1 file changed, 53 insertions(+), 4 deletions(-) diff --git a/src/app/devices/controller.device.ts b/src/app/devices/controller.device.ts index 7a68a6d..241759d 100644 --- a/src/app/devices/controller.device.ts +++ b/src/app/devices/controller.device.ts @@ -450,19 +450,68 @@ Object.defineProperty (controllerDevice, 'isMultipleMode', { }); controllerDevice.setLayerMode = function () { - this.attachButtonsToNeuron (); + this.initializeLayerMode (); + this.attachButtonsToLayer (); this.attachControlsToLayer (); }; -controllerDevice.attachControlsToLayer = function (): void { - const neurons = networkState.neurons[networkState.selectedLayerIndex]; - +controllerDevice.initializeLayerMode = function () { + // all nodes this.playNotes ({ firstNote: this.settings.lights.first, lastNote: this.settings.lights.last, color: this.settings.colorByState.layerMode, }); + // selected nodes in green + setTimeout (() => { + playgroundFacade.selectedNodes.forEach ((nodeIndex) => { + const { layerIndex: layerNumber, neuronIndex: neuronNumber } = networkState.getNeuronAndLayerIndexes (nodeIndex); + const layerIndex = layerNumber - 1; + const neuronIndex = neuronNumber - 1; + if (layerIndex === networkState.selectedLayerIndex) { + this.playNote ({ + note: this.settings.rows.firstButtons[neuronIndex], + color: this.settings.colorByState.selectMode, + }); + this.playNote ({ + note: this.settings.rows.secondButtons[neuronIndex], + color: this.settings.colorByState.selectMode, + }); + } + }); + }, this.settings.time.wait); +}; + +controllerDevice.attachButtonsToLayer = function (): void { + this.addNoteListener ('on', (e) => { + const inputNote = parseInt (e.note.number); + const index = this.settings.rows.secondButtons.indexOf (inputNote); + if (index !== -1) { + this.shifted[index] = true; + this.playNote ({ + note: inputNote, + color: this.settings.colorByState.shift, + }); + } + }); + + this.addNoteListener ('off', (e) => { + const inputNote = parseInt (e.note.number); + const index = this.settings.rows.secondButtons.indexOf (inputNote); + if (index !== -1) { + this.shifted[index] = false; + this.playNote ({ + note: inputNote, + color: this.settings.colorByState.layerMode, + }); + } + }); +}; + +controllerDevice.attachControlsToLayer = function (): void { + const neurons = networkState.neurons[networkState.selectedLayerIndex]; + this.addControlListener ((e) => { const inputNote = e.controller.number; From aa8d1e5969b5d778fe623f0f4a7ddc8b41b51846 Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 19:09:49 +0100 Subject: [PATCH 5/6] fix(selector): should exit layer mode when long pressing a layer button --- src/app/devices/selector.device.ts | 8 +++++++- src/app/state/network.state.ts | 4 ++++ 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/src/app/devices/selector.device.ts b/src/app/devices/selector.device.ts index ced70ea..9b624f1 100644 --- a/src/app/devices/selector.device.ts +++ b/src/app/devices/selector.device.ts @@ -336,8 +336,14 @@ selectorDevice.attachLayers = function () { // clear clearTimeout (clickTimer); clickTimer = null; - // payload networkState.toggleLayer (index); + if (networkState.selectedLayerIndex !== null) { + networkState.resetLayerSelection (); + this.renderLayers (); + controllerDevice.updateMode (); + layerCardUi.updateCard (); + playgroundUi.renderLayers (); + } }, this.settings.time.longClick); } diff --git a/src/app/state/network.state.ts b/src/app/state/network.state.ts index 97c7192..adfca3f 100644 --- a/src/app/state/network.state.ts +++ b/src/app/state/network.state.ts @@ -317,3 +317,7 @@ networkState.setLayer = function (index: number) { } return this.selectedLayerIndex; }; + +networkState.resetLayerSelection = function () { + this.selectedLayerIndex = null; +}; From a03eff3fe6a6eb0b5d5e5a098a449f0f2e1d4ee2 Mon Sep 17 00:00:00 2001 From: Bamdad Sabbagh Date: Tue, 16 Nov 2021 19:24:52 +0100 Subject: [PATCH 6/6] fix(controller): in layer mode, show active neurons in green instead of selected --- src/app/devices/controller.device.ts | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/src/app/devices/controller.device.ts b/src/app/devices/controller.device.ts index 241759d..c3b18e8 100644 --- a/src/app/devices/controller.device.ts +++ b/src/app/devices/controller.device.ts @@ -463,19 +463,18 @@ controllerDevice.initializeLayerMode = function () { color: this.settings.colorByState.layerMode, }); - // selected nodes in green + // color enabled neurons in green setTimeout (() => { - playgroundFacade.selectedNodes.forEach ((nodeIndex) => { - const { layerIndex: layerNumber, neuronIndex: neuronNumber } = networkState.getNeuronAndLayerIndexes (nodeIndex); - const layerIndex = layerNumber - 1; - const neuronIndex = neuronNumber - 1; - if (layerIndex === networkState.selectedLayerIndex) { + const neurons = networkState.neurons[networkState.selectedLayerIndex]; + neurons.forEach ((neuron) => { + if (neuron.isEnabled === true) { + const { neuronIndex } = networkState.getNeuronAndLayerIndexes (parseInt (neuron.id)); this.playNote ({ - note: this.settings.rows.firstButtons[neuronIndex], + note: this.settings.rows.firstButtons[neuronIndex - 1], color: this.settings.colorByState.selectMode, }); this.playNote ({ - note: this.settings.rows.secondButtons[neuronIndex], + note: this.settings.rows.secondButtons[neuronIndex - 1], color: this.settings.colorByState.selectMode, }); } @@ -533,7 +532,7 @@ controllerDevice.attachControlsToLayer = function (): void { ); const learningRate = selectCardUi.options.learningRate[learningRateOptionIndex]; - + if (learningRate !== neurons[index].learningRate) { networkState.setLearningRate (parseInt (neurons[index].id), learningRate); layerCardUi.setLearningRate (index, learningRate);