Last updated:
0 purchases
flutter sequencer
flutter_sequencer #
This Flutter plugin lets you set up sampler instruments and create multi-track sequences of notes
that play on those instruments. You can specify a loop range for a sequence and schedule volume
automations.
It uses the sfizz SFZ player on both Android and iOS. The
SFZ format can be used to create high-quality sample-based instruments and
do subtractive synthesis. These instruments can have modulation parameters that can be controlled by
your Flutter UI. The plugin also supports playing SF2 (SoundFont) files on both platforms, and on
iOS, you can load any AudioUnit instrument.
The example app is a drum machine, but there are many things you could build. The plugin is not
limited to "step" based sequencing. You could make a whole sample-based DAW. You could use it to
make a cross-platform app to host your SFZ instrument. You could also use it for game sound effects,
or even to generate a dynamic game soundtrack.
How to use #
Create the sequence #
final sequence = Sequence(tempo: 120.0, endBeat: 8.0);
copied to clipboard
You need to set the tempo and the end beat when you create the sequence.
Create instruments #
final instruments = [
Sf2Instrument(path: "assets/sf2/TR-808.sf2", isAsset: true),
SfzInstrument(
path: "assets/sfz/GMPiano.sfz",
isAsset: true,
tuningPath: "assets/sfz/meanquar.scl",
),
RuntimeSfzInstrument(
id: "Sampled Synth",
sampleRoot: "assets/wav",
isAsset: true,
sfz: Sfz(
groups: [
SfzGroup(
regions: [
SfzRegion(sample: "D3.wav", noteNumber: 62),
SfzRegion(sample: "F3.wav", noteNumber: 65),
SfzRegion(sample: "Gsharp3.wav", noteNumber: 68),
],
),
],
),
),
RuntimeSfzInstrument(
id: "Generated Synth",
// This SFZ doesn't use any sample files, so just put "/" as a placeholder.
sampleRoot: "/",
isAsset: false,
// Based on the Unison Oscillator example here:
// https://sfz.tools/sfizz/quick_reference#unison-oscillator
sfz: Sfz(
groups: [
SfzGroup(
regions: [
SfzRegion(
sample: "*saw",
otherOpcodes: {
"oscillator_multi": "5",
"oscillator_detune": "50",
}
)
]
)
]
)
),
];
copied to clipboard
An instrument can be used to create one or more tracks.
There are four instruments:
SfzInstrument, to load a .sfz file and the samples it refers to.
On iOS and Android, it will be played by sfizz
As far as I know, sfizz is the most complete and most frequently updated SFZ player library
with a license that permits commercial use.
Sfizz supports .wav and .flac sample files, among others. I recommend using .flac when
possible, since it supports lossless compression. It's easy to convert audio files to FLAC
format with ffmpeg.
You can also create an SFZ that doesn't use any sample files by setting sample to a
predefined waveform, such as *sine, *saw, *square, *triangle, or *noise.
Check which SFZ opcodes are supported by sfizz here:
https://sfz.tools/sfizz/development/status/opcodes/
Learn more about the SFZ format here: https://sfzformat.com
RuntimeSfzInstrument, to build an SFZ at runtime.
This will also be played by sfizz.
Instead of a file, you pass an Sfz object to the constructor. See the example.
You might use this so that your app can build a synth using selected oscillators, or a sampler
using user-provided samples.
Sf2Instrument, to load a .sf2 SoundFont file.
On iOS, it will be played by the built-in Apple MIDI synth AudioUnit
On Android, it will be played by tinysoundfont
I recommend using SFZ format, since sfizz can stream samples from disk. This way you can load
bigger sound fonts without running out of RAM.
You can easily convert SF2 to SFZ with Polyphone. Just
open the SF2, click the menu icon at the top right, and click "Export Soundfonts." Change the
format to SFZ. The other options shouldn't matter.
AudioUnitInstrument, to load an AudioUnit
This will only work on iOS
You might use this if you are making a DAW type of app.
For an SF2 or SFZ instrument, pass isAsset: true to load a path in the Flutter assets directory.
You should use assets for "factory preset" sounds. To load user-provided or downloaded sounds
from the filesystem, pass isAsset: false.
Important notes about isAsset: true #
Note that on Android, SFZ files and samples that are loaded with isAsset: true will be extracted
from the bundle into the application files directory (context.filesDir), since sfizz cannot read
directly from Android assets. This means they will exist in two places on the device - in the
APK in compressed (zipped) form and in the files directory in uncompressed form. So I only recommend
using isAsset: true if your samples are small. If you want to include high-quality soundfonts in
your app, your app should download them at runtime.
Note that on either platform, Flutter asset paths get URL-encoded. For example, if you put a
file called "Piano G#5.wav" in your assets folder, it will end up being called "Piano%20G%235.wav"
on the device. So if you are bundling an SFZ file as an asset, make sure that you either remove any
special characters and spaces from the sample file names, or update the SFZ file to refer to the
URL-encoded sample paths. I recommend just getting rid of any spaces and special characters.
Optional: keep engine running #
GlobalState().setKeepEngineRunning(true);
copied to clipboard
This will keep the audio engine running even when all sequences are paused. Set this to true if you
need to trigger sounds when the sequence is paused. Don't do it otherwise, since it will increase
energy usage.
Create tracks #
sequence.createTracks(instruments).then((tracks) {
setState(() {
this.tracks = tracks;
...
});
});
copied to clipboard
createTracks returns Future<List>. You probably want to store the value it completes with in
your widget's state.
Schedule events on the tracks #
track.addNote(noteNumber: 60, velocity: 0.7, startBeat: 0.0, durationBeats: 2.0);
copied to clipboard
This will add middle C (MIDI note number 60) to the sequence, starting from beat 0, and stopping
after 2 beats.
track.addVolumeChange(volume: 0.75, beat: 2.0);
copied to clipboard
This will schedule a volume change. It can be used to do volume automation. Note that track volume
is on a linear scale, not a logarithmic scale. You may want to use a logarithmic scale and convert
to linear.
track.addMidiCC(ccNumber: 127, ccValue: 127, beat: 2.0);
copied to clipboard
This will schedule a MIDI CC event. These can be used to change parameters in a sound font. For
example, in an SFZ, you can use the "cutoff_cc1" and "cutoff" opcodes to define a filter where the
cutoff can be changed with MIDI CC events. There are many other parameters that can be controlled
by CC as well, such as amp envelope, filter envelope, sample offset, and EQ parameters. For more
information, see how to do modulations in SFZ
and how to use filter cutoff in SFZ.
track.addMidiPitchBend(value: 1.0, beat: 2.0);
copied to clipboard
This will schedule a MIDI pitch bend event. The value can be from -1.0 to 1.0. Note that the value
is NOT the number of semitones. The sound font defines how many semitones the bend range is. For
example, in an SFZ, you can use the "bend_down" and "bend_up" opcodes to define how many cents the
pitch will be changed when the bend value is set to -1.0 and 1.0, respectively.
Control playback #
sequence.play();
copied to clipboard
sequence.pause();
copied to clipboard
sequence.stop();
copied to clipboard
Start, pause, or stop the sequence.
sequence.setBeat(double beat);
copied to clipboard
Set the playback position in the sequence, in beats.
sequence.setEndBeat(double beat);
copied to clipboard
Set the length of the sequence in beats.
sequence.setTempo(120.0);
copied to clipboard
Set the tempo in beats per minute.
sequence.setLoop(double loopStartBeat, double loopEndBeat);
copied to clipboard
Enable looping.
sequence.unsetLoop();
copied to clipboard
Disable looping.
Get real-time information about the sequence #
You can use SingleTickerProviderStateMixin to make these calls on every frame.
sequence.getPosition(); // double
copied to clipboard
Gets the playback position of the sequence, in beats.
sequence.getIsPlaying(); // bool
copied to clipboard
Gets whether or not the sequence is playing. It may have stopped if it reached the end.
track.getVolume();
copied to clipboard
Gets the volume of a track. A VolumeEvent may have changed it during playback.
Real-time playback #
track.startNoteNow(noteNumber: 60, velocity: 0.75);
copied to clipboard
Send a MIDI Note On message to the track immediately.
track.stopNoteNow(noteNumber: 60);
copied to clipboard
Send a MIDI Note Off message to the track immediately.
track.changeVolumeNow(volume: 0.5);
copied to clipboard
Change the track's volume immediately. Note that this is linear gain, not logarithmic.
How it works #
The Android and iOS backends start their respective audio engines. The iOS one adds an AudioUnit
for each track to an AVAudioEngine and connects it to a Mixer AudioUnit. The Android one has to
do all of the rendering manually. Both of them share a "BaseScheduler", which can be found under the
ios directory.
The backend has no concept of the sequence, the position in a sequence, the loop state, or even time
as measured in seconds or beats. It just maintains a map of tracks, and it triggers events on those
tracks when the appropriate number of frames have been rendered by the audio engine. The
BaseScheduler has a Buffer for each track that holds its scheduled events. The Buffer is supposed to
be thread-safe for one reader and one writer and real-time safe (i.e. it will not allocate memory,
so it can be used on the audio render thread.)
The Sequence lives on the Dart front end. A Sequence has Tracks. Each Track is backed by a Buffer on
the backend. When you add a note or a volume change to the track, it schedules an event on the
Buffer at the appropriate frame, based on the tempo and sample rate.
The buffer might not be big enough to hold all the events. Also, when looping is enabled, events
will occur indefinitely, so the buffer will never be big enough. To deal with this, the frontend
will periodically "top off" each track's buffer.
Development instructions #
Note that the Android build uses several third party libraries, including sfizz. The Gradle build
will download them from GitHub into the android/third_party directory.
The iOS also uses the sfizz library. It will be downloaded by the prepare.sh script which CocoaPods
will run.
To build the C++ tests on Mac OS, go into the cpp_test directory, and run
cmake .
make
copied to clipboard
Then, to run the tests,
./build/sequencer_test
copied to clipboard
I haven't tried it on Windows or Linux, but it should work without too many changes.
To Do #
PRs are welcome! If you use this plugin in your project, please consider contributing by fixing a
bug or by tackling one of these to-do items.
Difficulty: Easy #
Change position_frame_t to 64-bit integer to avoid overflow errors
Make constants configurable (TOP_OFF_PERIOD_MS and LEAD_FRAMES)
Support federated plugins
Difficulty: Medium #
Start using Dart null safety
Support tempo automation
MIDI Out instrument
Create an instrument that doesn't make any sounds on its own and just sends MIDI
events to a designated MIDI output port.
Refactoring
GlobalState, Sequence, and Track should have a clear separation of responsibilities
Engine, Sequencer, Mixer, Scheduler do too
"Track index" and "Track ID" are used interchangeably, "Track ID" should be used everywhere
Make the names and organization of the different files (Plugin/Engine/Scheduler) consistent
Support MacOS
Most of the iOS code should be able to be reused
Add more C++ tests
For BaseScheduler, specifically.
Add Dart tests
Test loop edge cases
Ensure there are no memory leaks or concurrency issues
Difficulty: Hard #
Support Windows
Record audio output
Support React Native?
Could use dart2js
Difficulty: Very Hard
(important) Audio graph
Create a graph of tracks and effects where any output can be connected to any input.
At that point, the most of the audio engine can be cross-platform. On iOS, it could be
wrapped in one AudioUnit, and external AudioUnit instruments and effects could be connected to it
via input and output buses.
LabSound seems like the library to use
It's based on WebAudio, so there could also be a web backend
Can be used to add some key effects like reverb, delay, and compression
Supports audio input, writing output to file, and other DAW stuff
Support Web
For personal and professional use. You cannot resell or redistribute these repositories in their original state.
There are no reviews.