This will be a step sequencer for Renoise using the Launchpad.
git flow release start
- remove all print functions (comment them out)
- update version in manifest.xmlandRakefile
rake package
- copy xrnx to renoise
git flow release finish
Layers should represent something that exists and you can talk to.
Like the Launchpad or an OscClient.
Should prevent you from magic numbers.
Should hold all the stuff you need to set up the system.
Everything from main.lua will call stuff from in here.
A module is something that is used by other Modules. Most of the time that is something that writes something to the Launchpad.
Sub-Modules are no real data-type. Because Modules become hard to manage over time, I split them up in multiple files.
The folder always has a file-named like the folder require-ing all the other sub-module.
Every Sub-Module should have a __init_SUBMODULE __activate_SUBMODULE and __deactivate_SUBMODULE function, even if
they are empty. These functions will be called in the Modules init _activate and _deactivate function.
Is some kind of abstract Module, to toggle other module on and off.
self.is_active and self.is_not_active will give you information if this module is activated or not.
Use them in your notifies and callbacks!
self.is_first_run and self.first_run will always be true till the first time self._activate() is called.
After self._activate() has finished, those values will be false.
While the first time self._activate() is called those values will be true.
Use them to control the wiring against notifies.
Modules have them too, and the convention is <ModuleName>Data.
It holds the Constant values for the module or this artifact (Color or Note for example).
Its a dictionary (table).
For more complex objects it has a access key (containing the indices to access parts of the complex object)
It would be best to use the native Observables but they are strictly bound to a type.
In a lot of cases we want to publish changes of whole objects, like in the Pagination module.
So we stick to the manual callback process. Here is a brief description on how this is to implement.
You have two objects LayerObject and ModuleObject.
The LayerObject holds variables which should trigger update routines in the ModuleObject if changing.
Create the callback hook.
  function LayerObject:__init( )
    ...
    self.update_callbacks = {}
  end
  function LayerObject:register_update_callback( callback )
    table.insert( self.update_callbacks, callback )
  end
and the update function
  function LayerObject:update_callbacks( )
    local myUpdate = {
      foo = self.value_1,
      bar = self.value_2,
    }
    for _, callback in ipairs(self.update_callbacks) do
        callback(myUpdate)
    end
  end
After that you can hook in the LayerObject
  function ModuleObject:hook( layerObject )
    local callback = function ( update )
      print( update.foo )
      print( update.bar )
    end
    layerObject:register_update_callback( callback )
  end
It is a lot of work you have to do to set something up like this.
But you can pass around object, which is in most cases more readable.
It might be possible that you dont need that at all, and you are fine with just the
Observables given for primitive data types.
(much more readable)
function LayerObject:__init()
    self.value = {
        foo = "foo",
        bar = "baz"
    }
    self.value_observable = renoise.Document.ObservableBang
end
function LayerObject:change()
    self.value = {
        foo = "bar",
        bar = "baz"
    }
    self.value_observable:bang()
end
Hook to it
function ModuleObject:wire_layer(layer)
    self.layer = layer
    add_notifier( self.layer.value_observable, function() 
        print(self.layer.value.foo)
        print(self.layer.value.bar)
    end)
end