module SurfaceMaster module Launchpad # Low-level interface to Novation Launchpad Mark 2 control surface. class Device < SurfaceMaster::Device include MIDICodes # TODO: Rename scenes to match Mk2 CODE_NOTE_TO_TYPE = Hash.new { |*_| :grid } .merge([Status::ON, Scene::SCENE1] => :scene1, [Status::ON, Scene::SCENE2] => :scene2, [Status::ON, Scene::SCENE3] => :scene3, [Status::ON, Scene::SCENE4] => :scene4, [Status::ON, Scene::SCENE5] => :scene5, [Status::ON, Scene::SCENE6] => :scene6, [Status::ON, Scene::SCENE7] => :scene7, [Status::ON, Scene::SCENE8] => :scene8, [Status::CC, Control::UP] => :up, [Status::CC, Control::DOWN] => :down, [Status::CC, Control::LEFT] => :left, [Status::CC, Control::RIGHT] => :right, [Status::CC, Control::SESSION] => :session, [Status::CC, Control::USER1] => :user1, [Status::CC, Control::USER2] => :user2, [Status::CC, Control::MIXER] => :mixer) .freeze TYPE_TO_NOTE = { up: Control::UP, down: Control::DOWN, left: Control::LEFT, right: Control::RIGHT, session: Control::SESSION, user1: Control::USER1, user2: Control::USER2, mixer: Control::MIXER, scene1: Scene::SCENE1, # Volume scene2: Scene::SCENE2, # Pan scene3: Scene::SCENE3, # Send A scene4: Scene::SCENE4, # Send B scene5: Scene::SCENE5, # Stop scene6: Scene::SCENE6, # Mute scene7: Scene::SCENE7, # Solo scene8: Scene::SCENE8 }.freeze # Record Arm def initialize(opts = nil) @name = "Launchpad MK2" super(opts) reset! if output_enabled? end def reset # TODO: Suss out what this should be for the Mark 2. layout!(0x00) output!(Status::CC, Status::NIL, Status::NIL) end # TODO: Support more of the LaunchPad Mark 2's functionality. def change(opts = nil) fail NoOutputAllowedError unless output_enabled? opts ||= {} command, payload = color_payload(opts) sysex!(command, payload[:led], payload[:color]) nil end def changes(values) fail NoOutputAllowedError unless output_enabled? organize_commands(values).each do |command, payloads| # The documented batch size for RGB LED updates is 80. The docs lie, at least on my # current firmware version -- anything above 62 crashes the device hard. while (slice = payloads.shift(62)).length > 0 messages = slice.map { |payload| [payload[:led], payload[:color]] } sysex!(command, *messages) end end nil end def read fail NoInputAllowedError unless input_enabled? super.collect do |input| note = input.delete(:note) input[:type] = CODE_NOTE_TO_TYPE[[input.delete(:code), note]] || :grid input[:x], input[:y] = decode_grid_coord(note) if input[:type] == :grid input.delete(:velocity) input end end protected def organize_commands(values) msg_by_command = {} values.each do |value| command, payload = color_payload(value) (msg_by_command[command] ||= []) << payload end msg_by_command end def decode_grid_coord(note) note -= 11 x = note % 10 y = note / 10 [x, y] end def layout!(mode); sysex!(0x22, mode); end def sysex_prefix; @sysex_prefix ||= super + [0x00, 0x20, 0x29, 0x02, 0x18]; end def decode_led(opts) case when opts[:cc] then [:cc, TYPE_TO_NOTE[opts[:cc]]] when opts[:grid] then decode_grid_led(opts) when opts[:column] then [:column, opts[:column]] when opts[:row] then [:row, opts[:row]] end rescue raise SurfaceMaster::Launchpad::NoValidGridCoordinatesError end def decode_grid_led(opts) if opts[:grid] == :all [:all, nil] else check_xy_values!(opts[:grid]) [:grid, (opts[:grid][1] * 10) + opts[:grid][0] + 11] end end def check_xy_values!(xy_pair) x = xy_pair[0] y = xy_pair[1] return unless xy_pair.length != 2 || !coord_in_range?(x) || !coord_in_range?(y) fail SurfaceMaster::Launchpad::NoValidGridCoordinatesError end def coord_in_range?(val); val && val >= 0 && val <= 7; end def color_payload(opts) # Hard-coded to single-LED RGB update right now. # For paletted changes, commands available include: # 0x0C -> Column # 0x0D -> Row # 0x0E -> All LEDs [0x0B, { led: decode_led(opts), color: [opts[:red] || 0x00, opts[:green] || 0x00, opts[:blue] || 0x00] }] end def output!(status, data1, data2) outputs!(message(status, data1, data2)) end def outputs!(*messages) messages = Array(messages) if @output.nil? logger.error "trying to write to device not open for output" fail SurfaceMaster::NoOutputAllowedError end logger.debug "writing messages to launchpad:\n #{messages.join("\n ")}" if logger.debug? @output.write(messages) nil end end end end