Alda REPL: "Play right away" offset calculation doesn't work with voices
daveyarwood opened this issue ยท 1 comments
๐ Bug report ๐
Description
In the Alda REPL use case, the user is appending additional events to the score, and as they're doing that, we want to play the new notes immediately, not wait until they would come up in the score. There shouldn't be any delay between entering a line of code in the REPL and hearing it.
To support this use case, we play some tricks in the note scheduling code, where the caller can pass in a map of "sync offsets" for each part, and we will subtract the offset for each part from all of its notes. As a result, we end up with notes that would be played right away, not however far into the score they would be if this were an Alda score file and not a REPL session.
This has been working great, but I just noticed that for some reason, it doesn't seem to work if you use voices.
Steps to Reproduce
- Start a REPL session (
alda repl
) - Type in
piano: V1: o4 c d e V2: o5 c d e
and press Enter. - Press the Up arrow to recall the previous line and press Enter to run it again.
- Repeat step 3 several times.
Expected Behavior
Every time you enter this line of code in the REPL, you hear the result immediately.
Actual Behavior
The first time, you hear the result immediately.
The second time, the result is delayed by 3 beats.
The third time, the result is delayed by 6 beats, etc.
Environment
Operating system and version: Ubuntu 20.04
Alda version:
$ alda version
alda 2.2.1
$ alda-player info
alda-player 2.2.1
log path: /home/dave/.cache/alda/logs
Health check:
$ alda doctor
(all passing)
Logs: not relevant here
My hunch is that this has something to do with the way Part instances are cloned in the voices implementation.
On this line could maybe try using Part.origin
instead of Part
. May need to make origin
public, except that that might have unintended consequences, but we could add an Origin()
getter method to get around that. Assuming that I'm even on the right track with this.
(In the "sync offsets" we are passing into the transmitter, I see that we are using part.origin already, which I think is correct.)