feat(sointu): add RowsPerBeat so non-power of 2 beat divisions are easier

This commit is contained in:
vsariola
2021-02-01 18:33:52 +02:00
parent 3cf2fc70a8
commit 3f494661e3
95 changed files with 132 additions and 6 deletions

View File

@ -40,7 +40,7 @@ func TestOscillatSine(t *testing.T) {
sointu.Unit{Type: "out", Parameters: map[string]int{"stereo": 1, "gain": 128}},
}}}}
tracks := []sointu.Track{{NumVoices: 1, Sequence: []byte{0}, Patterns: [][]byte{{64, 0, 68, 0, 32, 0, 0, 0, 75, 0, 78, 0, 0, 0, 0, 0}}}}
song := sointu.Song{BPM: 100, RowsPerPattern: 16, Tracks: tracks, Patch: patch}
song := sointu.Song{BPM: 100, RowsPerPattern: 16, RowsPerBeat: 4, Tracks: tracks, Patch: patch}
synth, err := bridge.Synth(patch)
if err != nil {
t.Fatalf("Compiling patch failed: %v", err)

View File

@ -119,6 +119,9 @@ func main() {
return fmt.Errorf("song could not be unmarshaled as a .json (%v) or .yml (%v)", errJSON, errYaml)
}
}
if song.RowsPerBeat == 0 {
song.RowsPerBeat = 4
}
var compiledPlayer map[string]string
if compile {
var err error

View File

@ -257,6 +257,7 @@ var UnitTypes = map[string]([]UnitParameter){
type Song struct {
BPM int
RowsPerPattern int
RowsPerBeat int
Tracks []Track
Patch Patch
}
@ -266,7 +267,7 @@ func (s *Song) Copy() Song {
for i, t := range s.Tracks {
tracks[i] = t.Copy()
}
return Song{BPM: s.BPM, RowsPerPattern: s.RowsPerPattern, Tracks: tracks, Patch: s.Patch.Copy()}
return Song{BPM: s.BPM, RowsPerPattern: s.RowsPerPattern, RowsPerBeat: s.RowsPerBeat, Tracks: tracks, Patch: s.Patch.Copy()}
}
func (s *Song) SequenceLength() int {
@ -278,7 +279,7 @@ func (s *Song) TotalRows() int {
}
func (s *Song) SamplesPerRow() int {
return 44100 * 60 / (s.BPM * 4)
return 44100 * 60 / (s.BPM * s.RowsPerBeat)
}
func (s *Song) FirstTrackVoice(track int) int {

View File

@ -7,10 +7,11 @@
#define SU_SAMPLE_RATE 44100
#define SU_BPM {{.Song.BPM}}
#define SU_ROWS_PER_BEAT {{.Song.RowsPerBeat}}
#define SU_PATTERN_SIZE {{.Song.RowsPerPattern}}
#define SU_MAX_PATTERNS {{.Song.SequenceLength}}
#define SU_TOTAL_ROWS (SU_MAX_PATTERNS*SU_PATTERN_SIZE)
#define SU_SAMPLES_PER_ROW (SU_SAMPLE_RATE*4*60/(SU_BPM*16))
#define SU_SAMPLES_PER_ROW (SU_SAMPLE_RATE*60/(SU_BPM*SU_ROWS_PER_BEAT))
#include <stdint.h>
#if UINTPTR_MAX == 0xffffffff

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 8
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [1, 0, 2, 0, 3, 0, 4, 0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 8
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [1, 0, 2, 0, 3, 0, 4, 0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 2
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0, 0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -1,5 +1,6 @@
bpm: 100
rowsperpattern: 16
rowsperbeat: 4
tracks:
- numvoices: 1
sequence: [0]

View File

@ -15,6 +15,7 @@ var defaultInstrument = sointu.Instrument{
var defaultSong = sointu.Song{
BPM: 100,
RowsPerPattern: 16,
RowsPerBeat: 4,
Tracks: []sointu.Track{
{NumVoices: 2, Sequence: []byte{0, 0, 0, 1}, Patterns: [][]byte{{64, 0, 68, 0, 32, 0, 0, 0, 75, 0, 78, 0, 0, 0, 0, 0}, {64, 0, 68, 0, 32, 0, 0, 0, 75, 0, 75, 0, 75, 0, 80, 0}}},
{NumVoices: 2, Sequence: []byte{0, 0, 0, 1}, Patterns: [][]byte{{0, 0, 64, 0, 68, 0, 32, 0, 0, 0, 75, 0, 78, 0, 0, 0}, {32, 0, 64, 0, 68, 0, 32, 0, 0, 0, 75, 0, 68, 0, 68, 0}}},

View File

@ -110,5 +110,19 @@ func (t *Tracker) layoutSongOptions(gtx C) D {
}),
)
}),
layout.Rigid(func(gtx C) D {
return layout.Flex{Axis: layout.Horizontal}.Layout(gtx,
layout.Rigid(Label("RPB:", white)),
layout.Rigid(func(gtx layout.Context) layout.Dimensions {
t.RowsPerBeat.Value = t.song.RowsPerBeat
numStyle := NumericUpDown(t.Theme, t.RowsPerBeat, 1, 32)
gtx.Constraints.Min.Y = gtx.Px(unit.Dp(20))
gtx.Constraints.Min.X = gtx.Px(unit.Dp(70))
dims := in.Layout(gtx, numStyle.Layout)
t.SetRowsPerBeat(t.RowsPerBeat.Value)
return dims
}),
)
}),
)
}

View File

@ -30,6 +30,7 @@ type Tracker struct {
Octave *NumberInput
BPM *NumberInput
RowsPerPattern *NumberInput
RowsPerBeat *NumberInput
NewTrackBtn *widget.Clickable
NewInstrumentBtn *widget.Clickable
DeleteInstrumentBtn *widget.Clickable
@ -102,7 +103,7 @@ func (t *Tracker) sequencerLoop(closer <-chan struct{}) {
panic("cannot create a synth with the default patch")
}
curVoices := make([]int, 32)
t.sequencer = NewSequencer(synth, 44100*60/(4*t.song.BPM), func() ([]Note, bool) {
t.sequencer = NewSequencer(synth, t.song.SamplesPerRow(), func() ([]Note, bool) {
t.playRowPatMutex.Lock()
if !t.Playing {
t.playRowPatMutex.Unlock()
@ -172,7 +173,23 @@ func (t *Tracker) SetBPM(value int) bool {
if value != int(t.song.BPM) {
t.SaveUndo()
t.song.BPM = value
t.sequencer.SetRowLength(44100 * 60 / (4 * t.song.BPM))
t.sequencer.SetRowLength(t.song.SamplesPerRow())
return true
}
return false
}
func (t *Tracker) SetRowsPerBeat(value int) bool {
if value < 1 {
value = 1
}
if value > 32 {
value = 32
}
if value != int(t.song.RowsPerBeat) {
t.SaveUndo()
t.song.RowsPerBeat = value
t.sequencer.SetRowLength(t.song.SamplesPerRow())
return true
}
return false
@ -362,6 +379,7 @@ func New(audioContext sointu.AudioContext) *Tracker {
Octave: new(NumberInput),
SongLength: new(NumberInput),
RowsPerPattern: new(NumberInput),
RowsPerBeat: new(NumberInput),
NewTrackBtn: new(widget.Clickable),
NewInstrumentBtn: new(widget.Clickable),
DeleteInstrumentBtn: new(widget.Clickable),