import Basic from "helper/Basic"; import Part from "Tone/event/Part"; import Tone from "Tone/core/Tone"; import Transport from "Tone/core/Transport"; import Event from "Tone/event/Event"; import Offline from "helper/Offline"; import Test from "helper/Test"; import Time from "Tone/type/Time"; describe("Part", function(){ Basic(Part); context("Constructor", function(){ it("takes a callback and an array of values", function(){ return Offline(function(){ var callback = function(){}; var part = new Part(callback, [0, 1, 2]); expect(part.callback).to.equal(callback); expect(part.length).to.equal(3); part.dispose(); }); }); it("can be constructed with no arguments", function(){ return Offline(function(){ var part = new Part(); expect(part.length).to.equal(0); part.dispose(); }); }); it("can pass in arguments in options object", function(){ return Offline(function(){ var callback = function(){}; var part = new Part({ "callback" : callback, "humanize" : true, "events" : [0, 1, 2], "loop" : true, "loopEnd" : "4n", "probability" : 0.3 }); expect(part.callback).to.equal(callback); expect(part.length).to.equal(3); expect(part.loop).to.be.true; expect(part.loopEnd).to.equal(Time("4n").valueOf()); expect(part.probability).to.equal(0.3); expect(part.humanize).to.be.true; part.dispose(); }); }); }); context("Adding / Removing / Getting Events", function(){ it("can take events in the constructor as an array of times", function(){ return Offline(function(){ var part = new Part(function(){}, ["0", "8n", "4n"]); expect(part.length).to.equal(3); part.dispose(); }); }); it("can take events in the constructor as an array of times and values", function(){ return Offline(function(){ var part = new Part(function(){}, [["0", "C4"], ["8n", "D3"], ["4n", "E4"]]); expect(part.length).to.equal(3); part.dispose(); }); }); it("can retrieve an event using 'at'", function(){ return Offline(function(){ var part = new Part(function(){}, [["0", 0], ["8n", "C2"], ["4n", 2]]); expect(part.length).to.equal(3); expect(part.at(0)).to.be.instanceof(Event); expect(part.at(0).value).to.equal(0); expect(part.at("8n").value).to.equal("C2"); expect(part.at("4n").value).to.equal(2); expect(part.at("2n")).to.be.null; part.dispose(); }); }); it("can set the value of an existing event with 'at'", function(){ return Offline(function(){ var part = new Part({ "events" : [[0, "C3"]] }); expect(part.length).to.equal(1); expect(part.at(0).value).to.equal("C3"); part.at(0, "C4"); expect(part.at(0).value).to.equal("C4"); part.dispose(); }); }); it("can take events in the constructor as an array of objects", function(){ return Offline(function(){ var part = new Part(function(){}, [{ "time" : 0.3, "note" : "C3" }, { "time" : 1, "note" : "D3" }]); expect(part.length).to.equal(2); expect(part.at(0.3).value).to.be.object; expect(part.at(0.3).value.note).to.equal("C3"); part.dispose(); }); }); it("can cancel event changes", function(){ var count = 0; return Offline(function(Transport){ var part = new Part(function(time){ count++; }, [{ "time" : 0, "note" : "C3" }, { "time" : 0.2, "note" : "D3" }]).start(0).stop(0.1); part.cancel(0.1); Transport.start(0); }, 0.3).then(function(){ expect(count).to.equal(2); }); }); it("can add an event as a time and value", function(){ return Offline(function(){ var part = new Part(); expect(part.length).to.equal(0); part.add(1, "D3"); expect(part.length).to.equal(1); expect(part.at(1).value).to.equal("D3"); part.dispose(); }); }); it("can add an event as an object", function(){ return Offline(function(){ var part = new Part(); expect(part.length).to.equal(0); part.add({ "time" : 0.5, "note" : "D4", "duration" : "8n" }); expect(part.length).to.equal(1); expect(part.at(0.5).value).to.be.object; expect(part.at(0.5).value.duration).to.deep.equal("8n"); expect(part.at(0.5).value.note).to.deep.equal("D4"); part.dispose(); }); }); it("can add another part", function(){ return Offline(function(){ var part = new Part(); expect(part.length).to.equal(0); var subPart = new Part({ "events" : [0, 0.5] }); part.add(0.2, subPart); expect(part.length).to.equal(1); expect(part.at(0.2)).to.equal(subPart); part.dispose(); }); }); it("can remove an event by time", function(){ return Offline(function(){ var part = new Part({ "events" : [[0.2, "C3"], [0.2, "C4"]] }); expect(part.length).to.equal(2); part.remove(0.2); expect(part.length).to.equal(0); part.dispose(); }); }); it("can remove an event by time and value", function(){ return Offline(function(){ var secondEvent = { time : 0.2, note : "C4" }; var part = new Part({ "events" : [[0.2, "C2"], secondEvent] }); expect(part.length).to.equal(2); part.remove(0.2, "C2"); expect(part.length).to.equal(1); part.remove(secondEvent); expect(part.length).to.equal(0); part.dispose(); }); }); it("added events have the same settings as the parent", function(){ return Offline(function(){ var part = new Part({ "loopEnd" : "1m", "loopStart" : "4n", "probability" : 0.2, "events" : [[0.2, "C3"], [0.3, "C4"]] }); part.humanize = 0.1; var firstEvent = part.at(0.2); expect(firstEvent.humanize).to.equal(0.1); expect(firstEvent.probability).to.equal(0.2); //loop duration is the same expect(firstEvent.loopEnd).to.equal(Time("1m").valueOf()); expect(firstEvent.loopStart).to.equal(Time("4n").valueOf()); var secondEvent = part.at(0.3); expect(secondEvent.humanize).to.equal(0.1); expect(secondEvent.probability).to.equal(0.2); //loop duration is the same expect(secondEvent.loopEnd).to.equal(Time("1m").valueOf()); expect(secondEvent.loopStart).to.equal(Time("4n").valueOf()); part.dispose(); }); }); it("will create an event using at if one wasn't there at that time", function(){ return Offline(function(){ var part = new Part(); expect(part.length).to.equal(0); expect(part.at(0.1, "C4").value).to.equal("C4"); expect(part.length).to.equal(1); part.dispose(); }); }); it("can remove all of the events", function(){ return Offline(function(){ var part = new Part(function(){}, [0, 1, 2, 3, 4, 5]); expect(part.length).to.equal(6); part.removeAll(); expect(part.length).to.equal(0); part.dispose(); }); }); }); context("Part callback", function(){ it("does not invoke get invoked until started", function(){ return Offline(function(Transport){ new Part(function(){ throw new Error("shouldn't call this callback"); }, [0, 0.4]); Transport.start(); }, 0.5); }); it("is invoked after it's started", function(){ var invokations = 0; return Offline(function(Transport){ new Part(function(){ invokations++; }, [0, 0.1]).start(0); Transport.start(); }, 0.2).then(function(){ expect(invokations).to.equal(2); }); }); it("passes in the scheduled time to the callback", function(){ var invoked = false; return Offline(function(Transport){ var startTime = 0.1; var part = new Part(function(time){ expect(time).to.be.a.number; expect(time - startTime).to.be.closeTo(0.5, 0.01); invoked = true; }, [0.3]); part.start(0.2); Transport.start(startTime); }, 0.62).then(function(){ expect(invoked).to.be.true; }); }); it("passes in the value to the callback", function(){ var invoked = false; return Offline(function(Transport){ var part = new Part(function(time, thing){ expect(time).to.be.a.number; expect(thing).to.equal("thing"); part.dispose(); invoked = true; }, [[0, "thing"]]).start(); Transport.start(); }, 0.6).then(function(){ expect(invoked).to.be.true; }); }); it("can mute the callback", function(){ return Offline(function(){ var part = new Part(function(){ throw new Error("shouldn't call this callback"); }, [0, 0.1, 0.2, 0.3]).start(); part.mute = true; expect(part.mute).to.be.true; Transport.start(); }, 0.5); }); it("can trigger with some probability", function(){ return Offline(function(){ var part = new Part(function(){ throw new Error("shouldn't call this callback"); }, [0, 0.1, 0.2, 0.3]).start(); part.probability = 0; expect(part.probability).to.equal(0); Transport.start(); }, 0.4); }); it("invokes all of the scheduled events", function(){ var count = 0; return Offline(function(Transport){ new Part(function(){ count++; }, [0, 0.1, 0.2, 0.3]).start(); Transport.start(); }, 0.4).then(function(){ expect(count).to.equal(4); }); }); it("invokes all of the scheduled events at the correct times", function(){ var count = 0; return Offline(function(Transport){ var now = Transport.now() + 0.1; new Part(function(time, value){ count++; expect(time - now).to.be.closeTo(value, 0.01); }, [[0, 0], [0.1, 0.1], [0.2, 0.2]]).start(); Transport.start(now); }, 0.4).then(function(){ expect(count).to.equal(3); }); }); it("starts an event added after the part was started", function(){ var invoked = false; return Offline(function(Transport){ var part = new Part({ "loopEnd" : 0.2, "loop" : true, "callback" : function(time, value){ if (value === 1){ invoked = true; } }, "events" : [[0, 0]] }).start(0); Transport.start(); return Test.atTime(0.1, function(){ part.add(0.1, 1); }); }, 0.6).then(function(){ expect(invoked).to.be.true; }); }); it("can schedule a subpart", function(){ var invokations = 0; return Offline(function(Transport){ var startTime = 0.1; var subPart = new Part({ "events" : [[0, 1], [0.3, 2]] }); var part = new Part(function(time, value){ invokations++; if (value === 0){ expect(time - startTime).to.be.closeTo(0, 0.01); } else if (value === 1){ expect(time - startTime).to.be.closeTo(0.2, 0.01); } else if (value === 2){ expect(time - startTime).to.be.closeTo(0.5, 0.01); part.dispose(); } }).add(0.2, subPart).add(0, 0).start(0); Transport.start(startTime); }, 0.7).then(function(){ expect(invokations).to.equal(3); }); }); it("can start with an offset", function(){ var invoked = false; return Offline(function(Transport){ var startTime = 0.1; var part = new Part(function(time, number){ expect(time - startTime).to.be.closeTo(0.1, 0.01); expect(number).to.equal(1); invoked = true; }, [[0, 0], [1, 1]]).start(0, 0.9); Transport.start(startTime); }, 0.3).then(function(){ expect(invoked).to.be.true; }); }); }); context("Looping", function(){ it("can be set to loop", function(){ var callCount = 0; return Offline(function(Transport){ new Part({ "loopEnd" : 0.2, "loop" : true, "callback" : function(){ callCount++; }, "events" : [[0, 1], [0.1, 2]] }).start(0); Transport.start(); }, 0.55).then(function(){ expect(callCount).to.equal(6); }); }); it("can be set to loop at a specific interval", function(){ var invoked = false; return Offline(function(Transport){ var lastCall; var part = new Part({ "loopEnd" : 0.25, "loop" : true, "callback" : function(time){ if (lastCall){ invoked = true; expect(time - lastCall).to.be.closeTo(0.25, 0.01); } lastCall = time; }, "events" : [0] }).start(0); Transport.start(); }, 0.7).then(function(){ expect(invoked).to.be.true; }); }); it("a started part will be stopped if it is after the loopEnd", function(){ var invoked = true; return Offline(function(Transport){ var switched = false; var part = new Part({ "loopEnd" : 0.5, "loop" : true, "callback" : function(time, value){ if (value === 1 && !switched){ switched = true; part.loopEnd = 0.2; } else if (switched){ expect(value).to.equal(0); invoked = true; } }, "events" : [[0, 0], [0.25, 1]] }).start(0); Transport.start(); }, 0.7).then(function(){ expect(invoked).to.be.true; }); }); it("a started part will be stopped if it is before the loopStart", function(){ var invoked = false; return Offline(function(Transport){ var switched = false; var part = new Part({ "loopEnd" : 0.5, "loop" : true, "callback" : function(time, value){ if (value === 1 && !switched){ switched = true; part.loopStart = 0.2; } else if (switched){ expect(value).to.equal(1); invoked = true; } }, "events" : [[0, 0], [0.25, 1]] }).start(0); Transport.start(); }, 0.7).then(function(){ expect(invoked).to.be.true; }); }); it("can loop a specific number of times", function(){ var callCount = 0; return Offline(function(Transport){ new Part({ "loopEnd" : 0.125, "loop" : 3, "callback" : function(){ callCount++; }, "events" : [0, 0.1] }).start(0.1); Transport.start(); }, 0.8).then(function(){ expect(callCount).to.equal(6); }); }); it("can loop between loopStart and loopEnd", function(){ var invoked = false; return Offline(function(Transport){ new Part({ "loopStart" : "8n", "loopEnd" : "4n", "loop" : true, "callback" : function(time, value){ expect(value).to.be.at.least(1); expect(value).to.be.at.most(2); invoked = true; }, "events" : [[0, 0], ["8n", 1], ["8n + 16n", 2], ["4n", 3]] }).start(0); Transport.start(); }, 0.8).then(function(){ expect(invoked).to.be.true; }); }); it("can be started and stopped multiple times", function(){ var eventTimeIndex = 0; return Offline(function(Transport){ var eventTimes = [[0.5, 0], [0.6, 1], [1.1, 0], [1.2, 1], [1.3, 2], [1.4, 0], [1.5, 1], [1.6, 2]]; new Part({ "loopEnd" : 0.3, "loopStart" : 0, "loop" : true, "callback" : function(time, value){ expect(eventTimes.length).to.be.gt(eventTimeIndex); expect(eventTimes[eventTimeIndex][0]).to.be.closeTo(time, 0.05); expect(eventTimes[eventTimeIndex][1]).to.equal(value); eventTimeIndex++; }, "events" : [[0, 0], [0.1, 1], [0.2, 2]] }).start(0.3).stop(0.81); Transport.start(0.2).stop(0.61).start(0.8); }, 2).then(function(){ expect(eventTimeIndex).to.equal(8); }); }); it("can adjust the loopEnd times", function(){ var eventTimeIndex = 0; return Offline(function(Transport){ var eventTimes = [[0.5, 0], [0.6, 1], [1.1, 0], [1.2, 1], [1.3, 2], [1.4, 0], [1.5, 1], [1.6, 2]]; var part = new Part({ "loopEnd" : 0.2, "loopStart" : 0, "loop" : true, "callback" : function(time, value){ expect(eventTimes.length).to.be.gt(eventTimeIndex); expect(eventTimes[eventTimeIndex][0]).to.be.closeTo(time, 0.05); expect(eventTimes[eventTimeIndex][1]).to.equal(value); eventTimeIndex++; }, "events" : [[0, 0], [0.1, 1], [0.2, 2]] }).start(0.3).stop(0.81); part.loopEnd = 0.4; part.loopEnd = 0.3; Transport.start(0.2).stop(0.61).start(0.8); }, 2).then(function(){ expect(eventTimeIndex).to.equal(8); }); }); it("reports the progress of the loop", function(){ var callCount = 0; return Offline(function(Transport){ var part = new Part({ "loopStart" : 0, "loopEnd" : 1, "loop" : true, "callback" : function(){ callCount++; }, "events" : [0] }).start(0); Transport.start(0); return function(time){ expect(part.progress).to.be.closeTo(time, 0.01); }; }, 0.8).then(function(){ expect(callCount).to.equal(1); }); }); it("can start a loop with an offset", function(){ var iteration = 0; return Offline(function(Transport){ var now = Tone.now(); var part = new Part(function(time, number){ if (iteration === 0){ expect(number).to.equal(1); expect(time - now).to.be.closeTo(0.2, 0.05); } else if (iteration === 1){ expect(number).to.equal(0); } iteration++; }, [[0, 0], [0.25, 1]]); part.loop = true; part.loopEnd = 0.5; part.start(0, 1.05); Transport.start(0); }, 0.6).then(function(){ expect(iteration).to.equal(2); }); }); it("can start a loop with an offset before loop start", function(){ var iteration = 0; return Offline(function(Transport){ var part = new Part(function(time, number){ if (iteration === 0){ expect(number).to.equal(0); } else if (iteration === 1){ expect(number).to.equal(1); } else if (iteration === 2){ expect(number).to.equal(2); } else if (iteration === 3){ expect(number).to.equal(1); } else if (iteration === 4){ expect(number).to.equal(2); } iteration++; }, [[0, 0], [0.25, 1], [0.30, 2]]); part.loop = true; part.loopStart = 0.25; part.loopEnd = 0.5; part.start(0, 0); Transport.start(Tone.now()); }, 0.7).then(function(){ expect(iteration).to.equal(5); }); }); }); context("playbackRate", function(){ it("can adjust the playbackRate", function(){ var invoked = false; return Offline(function(Transport){ var lastCall; new Part({ "playbackRate" : 2, "loopEnd" : 1, "loop" : true, "events" : [0, 0.5], "callback" : function(time){ if (lastCall){ invoked = true; expect(time - lastCall).to.be.closeTo(0.25, 0.01); } lastCall = time; } }).start(0); Transport.start(0); }, 0.7).then(function(){ expect(invoked).to.be.true; }); }); it("can adjust the playbackRate after starting", function(){ var invoked = false; return Offline(function(Transport){ var lastCall; var part = new Part({ "playbackRate" : 1, "loopEnd" : 0.5, "loop" : true, "events" : [0, 0.25], "callback" : function(time){ if (lastCall){ expect(time - lastCall).to.be.closeTo(0.5, 0.01); } else { invoked = true; part.playbackRate = 0.5; } lastCall = time; } }).start(0); Transport.start(0); }, 0.8).then(function(){ expect(invoked).to.be.true; }); }); }); });