Extending twist
Overview
Twist can quite easily be extended to feature additional transforms. These can be tested and used on the fly with the
developer console within twist, and are encouraged to be submitted for inclusion in the live application which can be done
here, via the
Help > Submit transform code menu option, or via the link in the
developer console
In order to write new transforms for twist, familiarity with JSON and Csound is required. The UI components including parameters available to the end user are defined with JSON (detailed in the
transform definition section, and the actual audio processing is defined with Csound code (detailed in the
audio processing with Csound section. Additional twist opcodes are provided to the developer in order to ease integration.
Each transform requires a transform definition as a JSON object, and at least one Csound instrument. While each section describes the API to be used, full examples are provided in the
Csound opcode subsection.
Transform definition with JSON
The transform definition is a JSON object, which should at the very least have the keys
name and
instr defined. The possible keys for the top-level transform definition are as follows, and if a default value is applicable it is shown to the right of the equals sign in the name column:
name | Name as seen in the twist user interface |
instr | Csound named instrument, which is to be called to carry out the processing |
inputs = 1 | Number of input files the transform requires. This defaults to 1 but may be set to 2 for cross-processing transforms and such |
description = "" | Description of the transform |
author = "" | Author name and other relevant details |
parameters = [] | Parameter definitions in an array |
twine = false | Whether the transform will be available as a twine insert. Only transforms using live input can be used for this purpose |
unstable = false | Should be set to true if the transform is expected to be unstable and may cause crashes. This will result in a warning displayed when the transform is loaded |
Parameter definition
The parameter definition is a JSON object, which may have any of the following keys. If min == 0, max == 1 and step == 1, the parameter appears as a checkbox which provides the value 1 when checked and 0 when unchecked. If
options is supplied, then the parameter appears as a drop-down select box. In other cases the parameter is displayed as a range slider with an adjacent number input box.
The only required key for a parameter definition is
name. In this case, a parameter would be with a range of 0 to 1, with the default step amount and sending on the channel with the lowercase equivalent of the name.
The possible keys for a parameter definition are as follows, and if a default value is applicable it is shown to the right of the equals sign in the name column:
name | Name of the parameter to be shown in the interface |
description = "" | Description of the parameter |
channel = name.toLowerCase() | Channel name which should correspond to that which is requested by twst_param in the transform instrument. Defaults to the lowercase parameter name |
min = 0 | Numeric minimum accepted value |
max = 1 | Numeric maximum accepted value |
step = 0.0000001 | Incremental allowance of the value, should numeric |
dfault = 1 | Numeric default value |
options = null | Array containing options to be displayed in a drop-down select box. If supplied, the minimum, maximum and step values are redundant. dfault corresponds to the index of the array to be the default value. If asvalue is set, then the value supplied to Csound will be the value provided in the options array; otherwise it will be the index of the value |
asvalue = false | Whether the selected item from options should be provided to Csound as the actual value rather than the array index |
hidden = false | Whether the parameter should be hidden. May be useful passing static data from the interface to Csound |
conditions = null | An array of Condition objects which are all to be met for the parameter to be shown |
hostrange = false | For child parameters (namely those in modulations), whether the min, max, step, dfault, options and asvalue attributes should be inherited from the parent |
preset = null | The name of a preset to be used. Any definition attributes provided by the preset may be overriden |
presetgroup = null | The name of a presetgroup to be used, which will provide a number of parameters in place of the current definition
|
nameprepend = null | The string which will be prepended to parameter names, if presetgroup is specified
|
channelprepend = null | The string which will be prepended to parameter channels, if presetgroup is specified
|
Condition
The condition definition is a JSON object, which should include all of the following keys:
channel | Parameter channel to evaluate |
operator | Operator type, which may be eq (equal), neq (not equal), lt (less than), gt (greater than), le (less than or equal to) or ge (greater than or equal to) |
value | Static value to check against the above |
Presets
These are available as values to specify in the
presetgroup parameter attribute and alter set the parameter up as follows
amp | Amplitude slider with min: 0 and max: 1, channel: "amp" |
fftsize | FFT size drop down which may be transparently utilised by twst_getfinput and twst_getfcrossinput, or accessed directly via the channel "fftsize" or the specified channel name with twst_param |
wave | f-table selector which may be transparently utilised by twst_tf_getwaveform, twst_tf_getwaveformk, or accessed directly via the channel "wave" or the specified channel name with twst_param or twst_paramk |
applymode | Apply mode drop down, which may be Replace, Mix, Modulate or Demodulate. Used internally by twist at the rendering stage |
note | MIDI note number drop-down, displaying note names between MIDI note number 21 (A0) and 127 (G#9) and returning the MIDI note number to the channel |
wintype | Window type drop-down which may be utilised by twst_tf_getwintype, twst_tf_getwintypek or accessed directly via the channel "wintype" or the specified channel name with twst_param or twst_paramk |
instance | Drop down selecting a file open in twist, other than that which is currently open. Utilised interally by twst_getcrossinput and twst_getfcrossinput |
instanceloop | Drop down selecting either None, Forward, Backward or Ping-pong to denote the loop type of the other selected instance, used internally for cross-processing transforms within twst_getcrossinput, twst_getfcrossinput and twst_getfcrossdata |
Preset groups
These are available as values to specify in the
presetgroup parameter attribute.
pvanal | Provides FFT size and a frequency/phase locking checkbox, used internally in the provision of PVS stream data within twst_getfinput and twst_getfcrossinput |
pvresmode | Provides parameters which control the resynthesis approach as used by twst_tf_fresynth. A drop down permits selection between overlap-add and additive approaches, with the latter showing several further parameters when selected |
pitchscale | Provides a scaling mode drop down with semitones or ratio as options. The selected scaling is presented via twst_tf_pitchscale as a ratio |
notefreq | Shows an option of selecting a note name from a drop down, or specifying the frequency in Hz. The computed frequency is provided to Csound via twst_tf_freq and twst_tf_freqi |
Audio processing with Csound
Audio processing is carried out for the corresponding JSON transform definition by invoking the Csound instrument specified in the
instr key. Multiple Csound instruments and opcodes may be utilised, however it should be noted that the Csound instrument is called using
subinstr, and offline/commit processing is carried out using audio rate processing within a k-rate loop. The only known limitation this imposes is that additional/auxilliary instruments may not usually be called from the initial instrument in a way that would affect synchronisation of the offline processing aspect - ie, opcodes such as
schedule and
event should not be used except in careful circumstances where the synchronisation is respected - for example where the scheduled instrument only complete init time processing, or completes in a single k-cycle. However, auxilliary instruments may be called using
subinstr.
Instruments can generate audio, utilise direct feed of audio, or access table data directly. The latter is useful if the output should be a different duration to the input.
A number of opcodes to ease integration with the UI and transform definition are provided by twist, detailed below with examples in the
Opcode reference subsection.
Rules and style guide
The rules and style guide should be adhered to where appropriate, especially if opcodes are to be submitted for inclusion in the live application.
- Instruments referenced by the transform definition should be named prepended with twst_tf_ if they generate audio or use direct input (obtained with twst_getinput or twst_getfinput) - or prepended with twst_tfi_ if table access is to be used (with twst_tfi_getfn
- The first line of the instrument must be $TWST_TRANSFORM to mark it as a twist transform
- Auxilliary instruments and user-defined opcodes should be named prepended with the name of the initial instrument
- Instruments may generate audio or process audio input obtained from calls to twst_getinput, twst_getfinput or twst_tfi_getfn
- Instruments must emit stereo audio signals using the outs opcode. Depending on the processing action, either of the outputs may be a silent signal or the same as the input
- Instruments should be prepared to process or generate left and right channels according to the ileft and iright values from the call to twst_getinput, twst_getfinput or twst_tf_getstate. A channel not applicable to the request must still be emitted, but may be a silent signal or the same as the input - the audition/commit process will only use the output for the channel requested by the user in the UI
- Any global objects created must not persist after the instrument has finished. For example, ftgentmp must be used rather than ftgen - unless ftfree is used on the f-table accordingly
- print opcodes and other console output opcodes should not be used except for debugging purposes
- 0dbfs is set at 1, so anything reliant on this should adhere accordingly
- Opcodes are limited to those available in the Csound WASM build - this is generally everything, but one noted example of an exclusion is fractalnoise. If otherwise unexplainable errors are encountered, this may be due to an unavailable opcode
Opcode reference
Examples from the live application
All of the JSON transform definitions
can be seen here, under the
transforms key.
Csound code in the live application is split across several files which are as follows. Each file generally corresponds to the section in the JSON.