xxxxxxxxxx
1985
let uiSize;
let uiSpacing;
let uiElementBeat;
let uiElementMelody;
let uiElementAtmo;
let hasStarted = false;
let isLoading = true;
let updateCountdown = 0;
let beatGenePool = [];
let melodyGenePool = []
let atmoGenePool = []
let selectedGenes = [];
let activeBeat;
let activeMelody;
let activeAtmo;
let BeatSynth1;
let BeatSynth2;
let BeatSynth3;
let MelodySynth;
let AtmoSynth;
let sequenzer;
let step = 0;
let isLooping = false;
let beatMuted = false;
let melodyMuted = false;
let atmoMuted = false;
let beatSelected = false;
let melodySelected = false;
let atmoSelected = false;
let selectionAmmount = 3;
function setup() {
let canvas = createCanvas(displayWidth, 1080);
canvas.parent('Canvas');
background(0);
uiSize = createVector(200, 100);
uiSpacing = 60;
//Instruments-------------------------------------------------------------------------
BeatSynth1 = new Synthesizer();
BeatSynth2 = new Synthesizer();
BeatSynth3 = new Synthesizer();
MelodySynth = new Synthesizer();
AtmoSynth = new Synthesizer();
sequenzer = new Sequenzer();
//UI----------------------------------------------------------------------------
textFont('Courier New');
UiElementBeat = new BeatVisuals(createVector(width / 2, height / 2 - uiSize.y - uiSpacing), uiSize,
'Beat');
UiElementMelody = new MelodyVisuals(createVector(width / 2, height / 2), uiSize,
'Melody');
UiElementAtmo = new AtmoVisuals(createVector(width / 2, height / 2 + uiSize.y + uiSpacing), uiSize,
'Atmosphere');
externalUi = new UI();
genePoolVisuals = new BgVisuals();
//init GenePool------------------------------
fetchGenePool();
}
function windowResized() {
resizeCanvas(windowWidth, 1080);
background(0);
UiElementBeat.pos = createVector(width / 2, height / 2 - uiSize.y - uiSpacing);
UiElementMelody.pos = createVector(width / 2, height / 2);
UiElementAtmo.pos = createVector(width / 2, height / 2 + uiSize.y + uiSpacing);
}
function draw() {
if (!isLoading) {
updateCountdown += deltaTime;
if (updateCountdown > 30000) {
updateCountdown = 0;
pushDNA();
}
}
if (!hasStarted) {
background(0);
introUI();
}
if (hasStarted) {
background(0, 0, 0, 20);
UiElementBeat.draw();
UiElementMelody.draw();
UiElementAtmo.draw();
genePoolVisuals.draw();
}
externalUi.draw();
}
function keyPressed() {
if (selectedGenes.length < 30) {
if (key == "b" && !beatSelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeBeat);
}
beatSelected = true;
}
if (key == "m" && !melodySelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeMelody);
}
melodySelected = true;
}
if (key == "a" && !atmoSelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeAtmo);
}
atmoSelected = true;
}
}
}
function mousePressed() {
externalUi.click();
}
//------------------------------ui
function introUI() {
fill(255);
textAlign(CENTER);
textSize(24);
text("Welcome to Lyra. This is a synthesizer based on tone.js learning to play it self through mashine learning and everyone visiting this website is training it! It consists of 3 instruments. Hover with your mouse over one of them to only hear that Instrument on its own. If you like what one of them is doing click on it or press 'b' for beat, 'm' for melody or 'a' for Atmosphere to encourage the machine to explore more into that direction. Press 'next' to load a new arangement or press ''autoplay to automaticaly loop through them. Click on the button below to start.", width / 2 - 350, height / 2 - 300, 700, 500);
if (!isLoading) {
let highlight = 0;
if (mouseX > width / 2 - 100 && mouseX < width / 2 + 100 && mouseY > height / 2 + 300 && mouseY < height / 2 + 400) {
highlight = 255;
if (mouseIsPressed) {
hasStarted = true;
sequenzer.start();
}
}
fill(highlight);
stroke(255);
rect(width / 2 - 100, height / 2 + 300, 200, 100);
fill(255);
textAlign(CENTER);
fill(255 - highlight);
textSize(34);
text("PLAY", width / 2, height / 2 + 360)
} else {
textAlign(CENTER);
fill(255);
textSize(24);
text("Loaing genepool...", width / 2, height / 2 + 360)
for (let i = 0; i < 50; i++) {
let x = width / 2 + i * 2 - 50;
let y = height / 2 + 400 + sin(i / 5 + millis() / 80) * 10;
fill(255);
stroke(255);
ellipse(x, y, 2);
}
}
}
class UiElement {
constructor(position, size, name) {
this.size = size;
this.pos = position;
this.name = name
this.isSelected = false;
}
draw() {
if (mouseX > this.pos.x - this.size.x / 2 &&
mouseX < this.pos.x + this.size.x / 2 &&
mouseY > this.pos.y &&
mouseY < this.pos.y + this.size.y
) {
if (!this.isSelected) {
this.muteOther();
this.isSelected = true;
}
if (mouseIsPressed && selectedGenes.length < 30) {
if (this.name == "Beat" && !beatSelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeBeat);
}
beatSelected = true;
}
if (this.name == "Melody" && !melodySelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeMelody);
}
melodySelected = true;
}
if (this.name == "Atmosphere" && !atmoSelected) {
for (let i = 0; i < selectionAmmount; i++) {
selectedGenes.push(activeAtmo);
}
atmoSelected = true;
}
}
} else if (this.isSelected) {
this.unMuteOther();
this.isSelected = false;
}
let borderCol = color(255);
if (this.name == "Beat" && beatMuted) borderCol = color(255, 0, 0);
if (this.name == "Melody" && melodyMuted) borderCol = color(255, 0, 0);
if (this.name == "Atmosphere" && atmoMuted) borderCol = color(255, 0, 0);
//box
noFill();
stroke(borderCol);
rect(this.pos.x - this.size.x / 2, this.pos.y, this.size.x, this.size.y);
//Name
fill(255);
stroke(255);
textSize(18);
textAlign(LEFT);
text(this.name, this.pos.x - this.size.x / 2, this.pos.y - 10);
//has been Added
textAlign(RIGHT);
textSize(12);
noStroke();
if (this.name == "Beat" && beatSelected) {
fill(0, 255, 0);
//stroke(0, 255, 0);
text("selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
} else if (this.name == "Beat") {
fill(255);
//stroke(255);
text("not selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
}
if (this.name == "Melody" && melodySelected) {
fill(0, 255, 0);
//stroke(0, 255, 0);
text("selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
} else if (this.name == "Melody") {
fill(255);
//stroke(255);
text("not selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
}
if (this.name == "Atmosphere" && atmoSelected) {
fill(0, 255, 0);
//stroke(0, 255, 0);
text("selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
} else if (this.name == "Atmosphere") {
fill(255);
//stroke(255);
text("not selected", this.pos.x + this.size.x / 2, this.pos.y - 10);
}
this.visualisation();
return this;
}
visualisation() {
}
muteOther() {}
unMuteOther() {}
}
class BeatVisuals extends UiElement {
constructor(position, size, name) {
super(position, size, name);
}
visualisation1() {
fill(255, 0, 0);
stroke(255, 0, 0);
ellipse(this.pos.x, this.pos.y + this.size.y / 2, this.size.y - this.size.y / 2);
}
visualisation2() {
noFill();
stroke(0, 255, 0);
strokeWeight(5);
ellipse(this.pos.x, this.pos.y + this.size.y / 2, this.size.y - this.size.y / 3 + 5);
strokeWeight(1);
}
visualisation3() {
noFill();
stroke(0, 0, 255);
strokeWeight(5);
ellipse(this.pos.x, this.pos.y + this.size.y / 2, this.size.y - 10);
strokeWeight(1);
}
muteOther() {
melodyMuted = true;
atmoMuted = true;
}
unMuteOther() {
melodyMuted = false;
atmoMuted = false;
}
}
class MelodyVisuals extends UiElement {
constructor(position, size, name) {
super(position, size, name);
}
visualisation() {
let buffer = MelodySynth.wave.getValue(0);
let start = 0;
for (let i = 0; i < buffer.length; i++) {
if (buffer[i - 1] < 0 && buffer[i] >= 0) {
start = i;
break;
}
}
let end = start + buffer.length / 4;
for (let i = start; i < end; i++) {
let x = map(i, start, end, this.pos.x - this.size.x / 2 + 2, this.pos.x + this.size.x / 2 - 2);
let y = map(buffer[i], -2, 2, this.pos.y + this.size.y - 2, this.pos.y + 2);
fill(255);
stroke(255);
ellipse(x, y, 2);
}
}
muteOther() {
beatMuted = true;
atmoMuted = true;
}
unMuteOther() {
beatMuted = false;
atmoMuted = false;
}
}
class AtmoVisuals extends UiElement {
constructor(position, size, name) {
super(position, size, name);
}
visualisation() {
let buffer = AtmoSynth.wave.getValue(0);
let start = 0;
for (let i = 0; i < buffer.length; i++) {
if (buffer[i - 1] < 0 && buffer[i] >= 0) {
start = i;
break;
}
}
let end = start + buffer.length / 4;
for (let i = start; i < end; i++) {
let x = map(i, start, end, this.pos.x - this.size.x / 2 + 2, this.pos.x + this.size.x / 2 - 2);
let y = map(buffer[i], -2, 2, this.pos.y + this.size.y - 2, this.pos.y + 2);
fill(255);
stroke(255);
ellipse(x, y, 2);
}
}
muteOther() {
melodyMuted = true;
beatMuted = true;
}
unMuteOther() {
melodyMuted = false;
beatMuted = false;
}
}
class UI {
constructor() {
let windowWasOpened = false;
}
draw() {
//autoplay options
if (hasStarted) {
//autoplay
let autoplayFill = color(255);
let nextFill = color(255);
if (mouseX < width / 2 + 100 && mouseX > width / 2 - 100 && mouseY < height / 2 + 300 + 18 && mouseY > height / 2 + 300 - 28) {
autoplayFill = color(150);
}
textAlign(CENTER);
textSize(18);
//stroke(autoplayFill);
noStroke();
fill(autoplayFill);
if (isLooping) text("Autoplay: On", width / 2, height / 2 + 300);
if (!isLooping) text("Autoplay: Off", width / 2, height / 2 + 300);
if (mouseX < width / 2 + 100 && mouseX > width / 2 - 100 && mouseY < height / 2 + 350 + 18 && mouseY > height / 2 + 350 - 28) {
nextFill = color(150);
}
//stroke(nextFill);
noStroke();
fill(nextFill);
if (!isLooping) text("Next", width / 2, height / 2 + 350);
}
//Logo
textSize(12);
strokeWeight(1);
stroke(255);
fill(255);
text("/", width / 2 - 24, height - 10);
text("/", width / 2 - 12, height - 10);
stroke(255, 0, 0);
fill(255, 0, 0);
text("j", width / 2, height - 10);
stroke(0, 255, 0);
fill(0, 255, 0);
text("j", width / 2 + 12, height - 10);
stroke(0, 0, 255);
fill(0, 0, 255);
text("j", width / 2 + 24, height - 10);
if (mouseX > width / 2 - 24 && mouseX < width / 2 + 24 &&
mouseY > height - 26 && mouseIsPressed
) {
if (!this.windowWasOpened) {
this.windowWasOpened = true;
window.open('https://jonasjoergenjacobsen.com');
}
} else if (this.windowWasOpened && !mouseIsPressed) this.windowWasOpened = false;
}
//handle mouse clicks
click() {
if (hasStarted) {
if (mouseX < width / 2 + 100 && mouseX > width / 2 - 100 && mouseY < height / 2 + 300 + 18 && mouseY > height / 2 + 300 - 28) {
isLooping = !isLooping;
}
if (mouseX < width / 2 + 100 && mouseX > width / 2 - 100 && mouseY < height / 2 + 350 + 18 && mouseY > height / 2 + 350 - 28 && !isLooping) {
beatSelected = false;
melodySelected = false;
atmoSelected = false;
sequenzer.setNewBar();
genePoolVisuals.generatePairs();
}
}
}
}
class BgVisuals {
constructor() {
this.randomPairs = [];
this.connectedPairs = [];
}
draw() {
fill(0);
noStroke();
rect(0, 0, width / 2 - 100, height);
rect(width / 2 + 100, 0, width - width / 2 + 100, height);
if ( this.randomPairs.length == 0 || this.connectedPairs.length == 0) {
this.generatePairs();
}
stroke(255);
fill(255);
textAlign(LEFT);
// draw connected Pairs
for (let i = 0; i < this.connectedPairs.length; i++) {
let pX = map(this.randomPairs[i].posX, 0, 1, 0, width, true);
let pY = map(this.connectedPairs[i].posY, 0, 1, 0, height) + this.connectedPairs[i].directionY * cos(millis() / 10000);
let c = this.connectedPairs[i].col;
let s = this.connectedPairs[i].size;
let t = this.connectedPairs[i].text;
let cX;
let cY = 0;
if (pX > width / 2) {
textAlign(LEFT);
cX = width / 2 + 100;
if (pX < width / 2 + 200) pX = width / 2 + 200;
} else {
textAlign(RIGHT);
cX = width / 2 - 100;
if (pX > width / 2 - 200) pX = width / 2 - 200;
}
if (this.connectedPairs[i].type == "beat") cY = UiElementBeat.pos.y + UiElementBeat.size.y / 2;
if (this.connectedPairs[i].type == "melody") cY = UiElementMelody.pos.y + UiElementMelody.size.y / 2;
if (this.connectedPairs[i].type == "atmo") cY = UiElementAtmo.pos.y + UiElementAtmo.size.y / 2;
stroke(c);
fill(c);
textSize(s);
text(t, pX, pY);
line(pX, pY - s / 3, cX, cY);
}
// draw random Pairs
for (let i = 0; i < this.randomPairs.length; i++) {
let pX = map(this.randomPairs[i].posX, 0, 1, 0, width, true);
let pY = map(this.randomPairs[i].posY, 0, 1, 0, height) + this.randomPairs[i].directionY * cos(millis() / 10000);
let s = this.randomPairs[i].size;
let t = this.randomPairs[i].text;
if (pX > width / 2) {
textAlign(LEFT);
if (pX < width / 2 + 200) pX = width / 2 + 200;
} else {
textAlign(RIGHT);
if (pX > width / 2 - 200) pX = width / 2 - 200;
}
stroke(255);
fill(255);
textSize(s);
text(t, pX, pY);
}
}
generatePairs() {
this.randomPairs = [];
this.connectedPairs = [];
let ammountConnected = int(random(8, 15));
let ammountRandom = int(random(30, 50));
//fill connected Pairs
for (let i = 0; i < ammountConnected; i++) {
let r = random([activeBeat, activeMelody, activeAtmo]);
let t = random(Object.keys(r));
let x;
if (random(0, 1) < 0.5) {
x = random(0, 0.4);
} else {
x = random(0.4, 1);
}
let n = {
type: r.type,
text: t + " : " + r[t],
posX: x,
posY: random(0, 1),
col: random([color(255, 0, 0), color(0, 255, 0), color(0, 0, 255)]),
size: random(8, 20),
directionX: random(-100, 100),
directionY: random(-100, 100),
connectTp: r.type
}
this.connectedPairs.push(n);
}
//fill random Pairs
for (let i = 0; i < ammountRandom; i++) {
let rB =random(beatGenePool);
let rM = random(melodyGenePool);
let rA = random(atmoGenePool);
let r = random([rB, rM , rA]);
let t = random(Object.keys(r));
let x;
if (random(0, 1) < 0.5) {
x = random(0, 0.4);
} else {
x = random(0.4, 1);
}
let n = {
text: t + " : " + r[t],
posX: x,
posY: random(0, 1),
size: random(8, 20),
directionX: random(-100, 100),
directionY: random(-100, 100),
}
this.randomPairs.push(n);
}
}
}
//------------------------------Sound
class Sequenzer {
constructor() {
this.tones = [264, 396, 440, 330];
}
start() {
this.setNewBar();
let loop = new Tone.Loop(this.callback, "16n").start(0);
Tone.Transport.start();
}
callback(time) {
this.tone = 220;
if (step < 16) this.tone = 264;
else if (step < 16 * 2) this.tone = 396;
else if (step < 16 * 3) this.tone = 440;
else this.tone = 330;
//beat synth
for (let i = 0; i < activeBeat.sequenzer1.length; i++) {
if (activeBeat.sequenzer1[i].deploy == step) {
if (!beatMuted) {
BeatSynth1.playTone(activeBeat.frequency1, "16n", time);
UiElementBeat.visualisation1();
}
}
}
for (let i = 0; i < activeBeat.sequenzer2.length; i++) {
if (activeBeat.sequenzer2[i].deploy == step) {
if (!beatMuted) {
BeatSynth2.playTone(activeBeat.frequency2, "16n", time);
UiElementBeat.visualisation2();
}
}
}
for (let i = 0; i < activeBeat.sequenzer3.length; i++) {
if (activeBeat.sequenzer3[i].deploy == step) {
if (!beatMuted) {
BeatSynth3.playTone(activeBeat.frequency3, "16n", time);
UiElementBeat.visualisation3();
}
}
}
// melody synth
for (let i = 0; i < activeMelody.sequenzer.length; i++) {
if (activeMelody.sequenzer[i].deploy == step) {
if (!melodyMuted) {
MelodySynth.playTone(this.tone + this.tone * activeMelody.frequency[activeMelody.sequenzer[i].note],
activeMelody.sequenzer[i].duration,
time);
}
}
}
// atmo synth
for (let i = 0; i < activeAtmo.sequenzer.length; i++) {
if (activeAtmo.sequenzer[i].deploy == step) {
if (!atmoMuted) {
AtmoSynth.playTone(this.tone + this.tone * activeAtmo.frequency[activeAtmo.sequenzer[i].note],
activeAtmo.sequenzer[i].duration,
time);
}
}
}
// next line
if (step == 16 * 4 && isLooping) {
beatSelected = false;
melodySelected = false;
atmoSelected = false;
sequenzer.setNewBar();
genePoolVisuals.generatePairs();
step = 0;
} else if (step == 16 * 4) step = 0;
else step++;
}
setNewBar() {
dnaSynthesis();
//-----------------------------beat Synth---------------------------
BeatSynth1.osc1.type = activeBeat.instruments[0].osc1Shape;
BeatSynth1.osc1Octave = activeBeat.instruments[0].osc1Octave;
BeatSynth1.osc1.volume.value = activeBeat.instruments[0].osc1Amp;
BeatSynth1.osc2.type = activeBeat.instruments[0].osc2Shape;
BeatSynth1.osc2Octave = activeBeat.instruments[0].osc2Octave;
BeatSynth1.osc2.volume.value = activeBeat.instruments[0].osc2Amp;
BeatSynth1.osc2Detune = activeBeat.instruments[0].detune;
if (activeBeat.instruments[0].noise) BeatSynth1.noise.volume.value = 0;
if (activeBeat.instruments[0].noise) BeatSynth1.noise.volume.value = activeBeat.instruments[0].noiseAmp;
BeatSynth1.envelope.attack = activeBeat.instruments[0].attack;
BeatSynth1.envelope.decay = activeBeat.instruments[0].decay;
BeatSynth1.envelope.sustain = activeBeat.instruments[0].sustain;
BeatSynth1.envelope.release = activeBeat.instruments[0].release;
//effects
BeatSynth2.osc1.type = activeBeat.instruments[1].osc1Shape;
//MelodySynth.osc1octave
BeatSynth2.osc1.volume.value = activeBeat.instruments[1].osc1Amp;
BeatSynth2.osc2.type = activeBeat.instruments[1].osc2Shape;
//MelodySynth.osc1octave
BeatSynth2.osc2.volume.value = activeBeat.instruments[1].osc2Amp;
//MelodySynth.detune
if (activeBeat.instruments[1].noise) BeatSynth2.noise.volume.value = 0;
if (activeBeat.instruments[1].noise) BeatSynth2.noise.volume.value = activeBeat.instruments[1].noiseAmp;
BeatSynth2.envelope.attack = activeBeat.instruments[2].attack;
BeatSynth2.envelope.decay = activeBeat.instruments[2].decay;
BeatSynth2.envelope.sustain = activeBeat.instruments[2].sustain;
BeatSynth2.envelope.release = activeBeat.instruments[2].release;
//effects
BeatSynth3.osc1.type = activeBeat.instruments[2].osc1Shape;
BeatSynth3.osc1Octave = activeBeat.instruments[2].osc1Octave;
BeatSynth3.osc1.volume.value = activeBeat.instruments[2].osc1Amp;
BeatSynth3.osc2.type = activeBeat.instruments[2].osc2Shape;
BeatSynth3.osc2Octave = activeBeat.instruments[2].osc1Octave;
BeatSynth3.osc2.volume.value = activeBeat.instruments[2].osc2Amp;
//MelodySynth.detune
if (activeBeat.instruments[2].noise) BeatSynth3.noise.volume.value = 0;
if (activeBeat.instruments[2].noise) BeatSynth3.noise.volume.value = activeBeat.instruments[2].noiseAmp;
BeatSynth3.envelope.attack = activeBeat.instruments[2].attack;
BeatSynth3.envelope.decay = activeBeat.instruments[2].decay;
BeatSynth3.envelope.sustain = activeBeat.instruments[2].sustain;
BeatSynth3.envelope.release = activeBeat.instruments[2].release;
//effects
//-----------------------------melody Synth---------------------------
MelodySynth.osc1.type = activeMelody.osc1Shape;
MelodySynth.osc1Octave = activeMelody.osc1Octave;
MelodySynth.osc1.volume.value = activeMelody.osc1Amp;
MelodySynth.osc2.type = activeMelody.osc2Shape;
MelodySynth.osc2Octave = activeMelody.osc2Octave;
MelodySynth.osc2.volume.value = activeMelody.osc2Amp;
MelodySynth.osc2Detune = activeMelody.detune;
if (!activeMelody.noise) MelodySynth.noise.volume.value = -100;
if (activeMelody.noise) MelodySynth.noise.volume.value = activeMelody.noiseAmp;
MelodySynth.envelope.attack = activeMelody.attack;
MelodySynth.envelope.decay = activeMelody.decay;
MelodySynth.envelope.sustain = activeMelody.sustain;
MelodySynth.envelope.release = activeMelody.release;
//effects
this.addEffects(MelodySynth, activeMelody);
//-----------------------------atmo Synth---------------------------
AtmoSynth.osc1.type = activeAtmo.osc1Shape;
AtmoSynth.osc1Octave = activeAtmo.osc1Octave;
AtmoSynth.osc1.volume.value = activeAtmo.osc1Amp;
AtmoSynth.osc2.type = activeAtmo.osc2Shape;
AtmoSynth.osc2Octave = activeAtmo.osc1Octave;
AtmoSynth.osc2.volume.value = activeAtmo.osc2Amp;
AtmoSynth.osc1Detune = activeAtmo.osc1Detune;
if (!activeAtmo.noise) AtmoSynth.noise.volume.value = -100;
if (activeAtmo.noise) AtmoSynth.noise.volume.value = activeAtmo.noiseAmp;
AtmoSynth.envelope.attack = activeAtmo.attack;
AtmoSynth.envelope.decay = activeAtmo.decay;
AtmoSynth.envelope.sustain = activeAtmo.sustain;
AtmoSynth.envelope.release = activeAtmo.release;
//effects
this.addEffects(AtmoSynth, activeAtmo);
}
addEffects(synth, data) {
let endPoint = synth.envelope;
endPoint.disconnect();
for (let i = 0; i < synth.effects.length; i++) {
if (synth.effects[i] != null) {
synth.effects[i].disconnect();
synth.effects[i].dispose();
}
}
synth.effects = [];
for (let i = 0; i < 2; i++) {
if (data.effects[i].type == "none") {
synth.effects.push(null);
} else if (data.effects[i].type == "delay") {
synth.effects.push(new Tone.FeedbackDelay(data.effects[i].delayTime + "n", data.effects[i].feedback));
synth.effects[i].toDestination()
} else if (data.effects[i].type == "distortion") {
synth.effects.push(new Tone.Distortion(data.effects[i].ammount));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
} else if (data.effects[i].type == "wah") {
synth.effects.push(new Tone.AutoWah(data.effects[i].baseFrequency,
data.effects[i].octaves,
data.effects[i].sensitivity));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
} else if (data.effects[i].type == "bitcrusher") {
synth.effects.push(new Tone.BitCrusher(data.effects[i].bits));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
} else if (data.effects[i].type == "chorus") {
synth.effects.push(new Tone.Chorus(data.effects[i].frequency,
data.effects[i].delayTime,
data.effects[i].depth));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
} else if (data.effects[i].type == "phaser") {
synth.effects.push(new Tone.Phaser({
frequency: data.effects[i].frequency,
octaves: data.effects[i].octaves,
baseFrequency: data.effects[i].baseFrequency
}));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
} else if (data.effects[i].type == "autofilter") {
synth.effects.push(new Tone.AutoFilter(data.effects[i].frequency + "n"));
endPoint.connect(synth.effects[i]);
endPoint = synth.effects[i];
}
}
endPoint.toDestination();
endPoint.connect(synth.wave);
}
}
class Synthesizer {
constructor() {
//this.delay = new Tone.FeedbackDelay("8n", 0.5).toDestination();
/*this.pingPong = new Tone.PingPongDelay( 0).connect(this.channel);
this.chorus = new Tone.Chorus(2, 0, 0).connect(this.channel);
this.chorus.connect(this.pingPong);
this.dist = new Tone.Distortion(1).toDestination();
*/
this.envelope = new Tone.AmplitudeEnvelope({
attack: 0.2,
decay: 0.4,
sustain: 1,
release: 10
}).toDestination();
this.channel = new Tone.Channel().connect(this.envelope);
this.osc1 = new Tone.Oscillator();
this.osc1.type = "sine";
this.osc1.connect(this.channel);
this.osc1.start();
this.osc1Octave = 1;
this.osc2 = new Tone.Oscillator();
this.osc2.type = "sawtooth";
this.osc2.connect(this.channel);
this.osc2.start();
this.osc2Octave = 1;
this.osc2Detune = 0;
this.noise = new Tone.Noise("pink").connect(this.channel).start();
this.effects = [];
this.wave = new Tone.Waveform();
// this.envelope.connect(this.wave);
}
playTone(note, duration, t) {
this.osc1.frequency.value = note * this.osc1Octave;
this.osc2.frequency.value = note * this.osc1Octave + this.osc2Detune;
this.envelope.triggerAttackRelease(duration, t);
}
}
//------------------------------DNA
//fetch DNA
function fetchGenePool() {
isLoading = true;
let ajax = new XMLHttpRequest();
let url = 'https://jonasjoergenjacobsen.com/content/lyra/api.php';
ajax.onreadystatechange = function() {
if (this.readyState == 4 && this.status == 200) {
gotData(JSON.parse(this.response));
print(JSON.parse(this.response));
}
}
ajax.open("GET", url);
ajax.send();
}
function gotData(data) {
fillDNAPool(data);
}
function fillDNAPool(data) {
let beatData = data.beatGenes;
for (let i = 0; i < beatData.length; i++) {
beatGenePool.push(beatData[i]);
}
let melodyData = data.melodyGenes;
for (let i = 0; i < melodyData.length; i++) {
melodyGenePool.push(melodyData[i]);
}
let atmoData = data.atmoGenes;
for (let i = 0; i < atmoData.length; i++) {
atmoGenePool.push(atmoData[i]);
}
if (beatGenePool.length < 3000) {
let missing = beatGenePool.length;
for (let i = 0; i < 3000 - missing; i++) {
beatGenePool.push(createRandomDNA("beat"));
}
}
if (melodyGenePool.length < 3000) {
let missing = melodyGenePool.length;
for (let i = 0; i < 3000 - missing; i++) {
melodyGenePool.push(createRandomDNA("melody"));
}
}
if (atmoGenePool.length < 3000) {
let missing = atmoGenePool.length;
for (let i = 0; i < 3000 - missing; i++) {
atmoGenePool.push(createRandomDNA("atmo"));
}
}
isLoading = false;
//!!!!!!!!!!!!!!!!!!!!!!!!!!!!!Debug!!!!!!!!!!!!!!!
//dnaSynthesis();
}
function createRandomDNA(type) {
let build;
if (type == "beat") {
build = {
type: "beat",
instruments: [{
osc1Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc1Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc1Amp: random(-5, 1),
osc2Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc2Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc2Amp: random(-10, 1),
detune: random(-10, 10),
noise: random([true, false]),
noiseAmp: random(-20, +1),
attack: random(0.001, 0.5),
decay: random(0.001, 0.5),
sustain: random(0.001, 0.1),
release: random(0.01, 5),
},
{
osc1Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc1Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc1Amp: random(-5, 1),
osc2Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc2Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc2Amp: random(-10, 1),
detune: random(-10, 10),
noise: random([true, false]),
noiseAmp: random(-10, 1),
attack: random(0.001, 0.5),
decay: random(0.001, 0.5),
sustain: random(0.001, 0.1),
release: random(0.01, 5),
},
{
osc1Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc1Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc1Amp: random(-5, 1),
osc2Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc2Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc2Amp: random(-10, 1),
detune: random(-10, 10),
noise: random([true, false]),
noiseAmp: random(-20, 1),
attack: random(0.001, 0.5),
decay: random(0.001, 0.5),
sustain: random(0.001, 0.1),
release: random(0.01, 5),
}
],
effects: [
],
frequency1: random(40, 200),
frequency2: random(40, 1000),
frequency3: random(40, 20000),
sequenzer1: [],
sequenzer2: [],
sequenzer3: []
};
//add effects
for (let i = 0; i < 2; i++) {
let r = random([{
type: "none"
},
{
type: "none"
},
{
type: "none"
},
{
type: "delay",
delayTime: int(random(1, 33)),
feedback: random(0.01, 0.9)
},
{
type: "distortion",
ammount: random(0.01, 5)
},
{
type: "wah",
baseFrequency: random(0.2, 100),
octaves: random(2, 10),
sensitivity: random(-50, 10)
},
{
type: "bitcrusher",
bits: random(2, 13)
},
{
type: "chorus",
frequency: random(0.1, 5),
delayTime: random(0.1, 50),
depth: random(0, 20)
},
{
type: "phaser",
frequency: random(1, 11),
octaves: random(1, 7),
baseFrequency: random(20, 3000)
},
{
type: "autofilter",
frequency: int(random(0, 32))
}
]);
build.effects.push(r);
}
//create sequence
for (let j = 1; j <= 3; j++) {
for (let i = 0; i < 16 * 4; i++) {
if (random(0, 1) < 0.2) {
//let d = int(random(0,16));
//i += d;
let n = {
deploy: i,
//duration: d + "n",
//note: int(random(0, 14))
};
if (j == 1) build.sequenzer1.push(n);
if (j == 2) build.sequenzer2.push(n);
if (j == 3) build.sequenzer3.push(n);
}
}
}
}
if (type == "melody") {
build = {
type: "melody",
osc1Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc1Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc1Amp: random(-10, 0),
osc2Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc2Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc2Amp: random(-10, 0),
detune: random(-10, 10),
noise: random([true, false, false]),
noiseAmp: random(-20, 0),
attack: random(0.005, 1),
decay: random(0.005, 1),
sustain: random(0.1, 1),
release: random(0.005, 5),
effects: [
],
frequency: [
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1)
],
sequenzer: []
};
//add effects
for (let i = 0; i < 2; i++) {
let r = random([{
type: "none"
},
{
type: "none"
},
{
type: "none"
},
{
type: "delay",
delayTime: int(random(1, 33)),
feedback: random(0.01, 1)
},
{
type: "distortion",
ammount: random(0.01, 5)
},
{
type: "wah",
baseFrequency: random(0.2, 100),
octaves: random(2, 10),
sensitivity: random(-50, 10)
},
{
type: "bitcrusher",
bits: random(2, 13)
},
{
type: "chorus",
frequency: random(0.1, 5),
delayTime: random(0.1, 50),
depth: random(0, 20)
},
{
type: "phaser",
frequency: random(1, 11),
octaves: random(1, 7),
baseFrequency: random(20, 3000)
},
{
type: "autofilter",
frequency: int(random(0, 32))
}
]);
build.effects.push(r);
}
//create sequence
for (let i = 0; i < 16 * 4; i++) {
if (random(0, 1) < 0.2) {
let d = int(random(1, 16));
if (i + d > 16 * 4) d = 16 * 4 - i;
i += d;
let n1 = {
deploy: i,
duration: 17 - d + "n",
note: int(random(0, 7))
};
build.sequenzer.push(n1);
}
}
}
if (type == "atmo") {
build = {
type: "atmo",
osc1Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc1Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc1Amp: random(-10, 0),
osc2Shape: random(["sine", "sawtooth", "square", "triangle"]),
osc2Octave: random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]),
osc2Amp: random(-10, 0),
detune: random(-10, 10),
noise: random([true, false]),
noiseAmp: random(-20, 0),
attack: random(0.005, 1),
decay: random(0.005, 1),
sustain: random(0.1, 1),
release: random(0.005, 10),
effects: [
],
frequency: [
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1),
random(0, 1)
],
sequenzer: []
};
//add effects
for (let i = 0; i < 2; i++) {
let r = random([{
type: "none"
},
{
type: "none"
},
{
type: "none"
},
{
type: "delay",
delayTime: int(random(1, 33)),
feedback: random(0.01, 1)
},
{
type: "distortion",
ammount: random(0.01, 5)
},
{
type: "wah",
baseFrequency: random(0.2, 100),
octaves: random(2, 10),
sensitivity: random(-50, 10)
},
{
type: "bitcrusher",
bits: random(2, 13)
},
{
type: "chorus",
frequency: random(0.1, 5),
delayTime: random(0.1, 50),
depth: random(0, 20)
},
{
type: "phaser",
frequency: random(1, 11),
octaves: random(1, 7),
baseFrequency: random(20, 3000)
},
{
type: "autofilter",
frequency: int(random(0, 32))
}
]);
build.effects.push(r);
}
//create sequence
for (let i = 0; i < 16 * 4; i++) {
if (random(0, 1) < 0.2) {
let d = int(random(1, 17));
if (i + d > 16 * 4) d = 16 * 4 - i;
i += d;
let n1 = {
deploy: i,
duration: 17 - d + "n",
note: int(random(0, 7))
};
build.sequenzer.push(n1);
}
}
}
return build;
}
//push DNA
function pushDNA() {
if (selectedGenes.length > 0) {
let ajax = new XMLHttpRequest();
let url = 'https://jonasjoergenjacobsen.com/content/lyra/api.php';
ajax.onreadystatechange = function() {
if (this.readyState == 4 && this.status == 200) {
// print(JSON.parse(this.response));
}
}
ajax.open("POST", url);
ajax.send(JSON.stringify(selectedGenes));
}
selectedGenes = [];
fetchGenePool();
}
//----------------------DNA Synthesis
function dnaSynthesis() {
let p1Beat;
let p2Beat;
let p1Melody;
let p2Melody
let p1Atmo;
let p2Atmo
let combinedBeat;
let combinedMelody;
let combinedAtmo;
if (random(0, 1) > 0.01) {
p1Beat = random(beatGenePool);
} else p1Beat = createRandomDNA("beat");
if (random(0, 1) > 0.01) {
p2Beat = random(beatGenePool);
} else p2Beat = createRandomDNA("beat");
if (random(0, 1) > 0.01) {
p1Melody = random(melodyGenePool);
} else p1Melody = createRandomDNA("melody");
if (random(0, 1) > 0.01) {
p2Melody = random(melodyGenePool);
} else p2Melody = createRandomDNA("melody");
if (random(0, 1) > 0.01) {
p1Atmo = random(atmoGenePool);
} else p1Atmo = createRandomDNA("atmo");
if (random(0, 1) > 0.01) {
p2Atmo = random(atmoGenePool);
} else p2Atmo = createRandomDNA("atmo");
//combine beat
combinedBeat = p1Beat;
for (let i = 0; i < 3; i++) {
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc1Shape = p1Beat.instruments[i].osc1Shape;
else combinedBeat.instruments[i].osc1Shape = p2Beat.instruments[i].osc1Shape;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc1Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc1Octave = p1Beat.instruments[i].osc1Octave;
else combinedBeat.instruments[i].osc1Octave = p2Beat.instruments[i].osc1Octave;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc1Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc1Amp = p1Beat.instruments[i].osc1Amp;
else combinedBeat.instruments[i].osc1Amp = p2Beat.instruments[i].osc1Amp;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc1Amp = combinedBeat.instruments[i].osc1Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc2Shape = p1Beat.instruments[i].osc2Shape;
else combinedBeat.instruments[i].osc2Shape = p2Beat.instruments[i].osc2Shape;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc2Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc2Octave = p1Beat.instruments[i].osc2Octave;
else combinedBeat.instruments[i].osc2Octave = p2Beat.instruments[i].osc2Octave;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc2Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].osc2Amp = p1Beat.instruments[i].osc2Amp;
else combinedBeat.instruments[i].osc2Amp = p2Beat.instruments[i].osc2Amp;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].osc2Amp = combinedBeat.instruments[i].osc2Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].detune = p1Beat.instruments[i].detune;
else combinedBeat.instruments[i].detune = p2Beat.instruments[i].detune;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].detune = combinedBeat.instruments[i].detune + random(-5, 5);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].noise = p1Beat.instruments[i].noise;
else combinedBeat.instruments[i].noise = p2Beat.instruments[i].noise;
if (random(0, 1) > 0.5) combinedBeat.instruments[i].noiseAmp = p1Beat.instruments[i].noiseAmp;
else combinedBeat.instruments[i].noiseAmp = p2Beat.instruments[i].noiseAmp;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].noise = combinedBeat.instruments[i].noise + random(-5, 5);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].attack = p1Beat.instruments[i].attack;
else combinedBeat.instruments[i].attack = p2Beat.instruments[i].attack;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].attack = constrain(combinedBeat.instruments[i].attack + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].decay = p1Beat.instruments[i].decay;
else combinedBeat.instruments[i].decay = p2Beat.instruments[i].decay;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].decay = constrain(combinedBeat.instruments[i].decay + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].sustain = p1Beat.instruments[i].sustain;
else combinedBeat.instruments[i].sustain = p2Beat.instruments[i].sustain;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].sustain = constrain(combinedBeat.instruments[i].sustain + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedBeat.instruments[i].release = p1Beat.instruments[i].release;
else combinedBeat.instruments[i].release = p2Beat.instruments[i].release;
if (random(0, 1) < 0.05) combinedBeat.instruments[i].release = constrain(combinedBeat.instruments[i].release + random(-0.1, 0.1), 0, 1);
}
//no effects
if (random(0, 1) > 0.5) combinedBeat.frequency1 = p1Beat.frequency1;
else combinedBeat.frequency1 = p2Beat.frequency1;
if (random(0, 1) > 0.5) combinedBeat.frequency2 = p1Beat.frequency2;
else combinedBeat.frequency2 = p2Beat.frequency2;
if (random(0, 1) > 0.5) combinedBeat.frequency3 = p1Beat.frequency3;
else combinedBeat.frequency3 = p2Beat.frequency3;
//sequencer1
if (random(0, 1) > 0.5) combinedBeat.sequenzer1 = p1Beat.sequenzer1;
else combinedBeat.sequenzer1 = p2Beat.sequenzer1;
if (random(0, 1) < 0.05) {
for (let i = 0; i < combinedBeat.sequenzer1.length; i++) {
let r = random(0, 1);
if (r < 0.1) combinedBeat.sequenzer1.splice(i, 1);
else if (r < 0.2) {
let lastNote;
if (i > 0) lastNote = combinedBeat.sequenzer1[i - 1];
else lastNote = {
deploy: 0
};
let nextNote;
if (i != combinedBeat.sequenzer1.length - 1) nextNote = combinedBeat.sequenzer1[i + 1];
else nextNote = {
deploy: 16 * 4 + 1
};
let n = {
deploy: int(random(lastNote.deploy + 1, nextNote.deploy))
};
combinedBeat.sequenzer1.splice(i, 0, n);
} else if (r < 0.3) {
combinedBeat.sequenzer1[i].note = int(random(0, 7));
}
}
}
// sequencer2
if (random(0, 1) > 0.5) combinedBeat.sequenzer2 = p1Beat.sequenzer2;
else combinedBeat.sequenzer2 = p2Beat.sequenzer2;
if (random(0, 1) < 0.05) {
for (let i = 0; i < combinedBeat.sequenzer2.length; i++) {
let r = random(0, 1);
if (r < 0.1) combinedBeat.sequenzer2.splice(i, 1);
else if (r < 0.2) {
let lastNote;
if (i > 0) lastNote = combinedBeat.sequenzer2[i - 1];
else lastNote = {
deploy: 0
};
let nextNote;
if (i != combinedBeat.sequenzer2.length - 1) nextNote = combinedBeat.sequenzer2[i + 1];
else nextNote = {
deploy: 16 * 4 + 1
};
let n = {
deploy: int(random(lastNote.deploy + 1, nextNote.deploy))
};
combinedBeat.sequenzer2.splice(i, 0, n);
} else if (r < 0.3) {
combinedBeat.sequenzer2[i].note = int(random(0, 7));
}
}
}
// sequencer3
if (random(0, 1) > 0.5) combinedBeat.sequenzer3 = p1Beat.sequenzer3;
else combinedBeat.sequenzer3 = p2Beat.sequenzer3;
if (random(0, 1) < 0.05) {
for (let i = 0; i < combinedBeat.sequenzer3.length; i++) {
let r = random(0, 1);
if (r < 0.1) combinedBeat.sequenzer3.splice(i, 1);
else if (r < 0.2) {
let lastNote;
if (i > 0) lastNote = combinedBeat.sequenzer3[i - 1];
else lastNote = {
deploy: 0
};
let nextNote;
if (i != combinedBeat.sequenzer3.length - 1) nextNote = combinedBeat.sequenzer3[i + 1];
else nextNote = {
deploy: 16 * 4 + 1
};
let n = {
deploy: int(random(lastNote.deploy + 1, nextNote.deploy))
};
combinedBeat.sequenzer3.splice(i, 0, n);
} else if (r < 0.3) {
combinedBeat.sequenzer3[i].note = int(random(0, 7));
}
}
}
//----------------end Combine Beat
// combine Melody
combinedMelody = p1Melody;
if (random(0, 1) > 0.5) combinedMelody.osc1Shape = p1Melody.osc1Shape;
else combinedMelody.osc1Shape = p2Melody.osc1Shape;
if (random(0, 1) < 0.05) combinedMelody.osc1Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedMelody.osc1Octave = p1Melody.osc1Octave;
else combinedMelody.osc1Octave = p2Melody.osc1Octave;
if (random(0, 1) < 0.05) combinedMelody.osc1Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedMelody.osc1Amp = p1Melody.osc1Amp;
else combinedMelody.osc1Amp = p2Melody.osc1Amp;
if (random(0, 1) < 0.05) combinedMelody.osc1Amp = combinedMelody.osc1Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedMelody.osc2Shape = p1Melody.osc2Shape;
else combinedMelody.osc2Shape = p2Melody.osc2Shape;
if (random(0, 1) < 0.05) combinedMelody.osc2Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedMelody.osc2Octave = p1Melody.osc2Octave;
else combinedMelody.osc2Octave = p2Melody.osc2Octave;
if (random(0, 1) < 0.05) combinedMelody.osc2Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedMelody.osc2Amp = p1Melody.osc2Amp;
else combinedMelody.osc2Amp = p2Melody.osc2Amp;
if (random(0, 1) < 0.05) combinedMelody.osc2Amp = combinedMelody.osc2Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedMelody.detune = p1Melody.detune;
else combinedMelody.detune = p2Melody.detune;
if (random(0, 1) < 0.05) combinedMelody.detune = combinedMelody.detune + random(-5, 5);
if (random(0, 1) > 0.5) combinedMelody.noise = p1Melody.noise;
else combinedMelody.noise = p2Melody.noise;
if (random(0, 1) > 0.5) combinedMelody.noiseAmp = p1Melody.noiseAmp;
else combinedMelody.noiseAmp = p2Melody.noiseAmp;
if (random(0, 1) < 0.05) combinedMelody.noise = combinedMelody.noise + random(-5, 5);
if (random(0, 1) > 0.5) combinedMelody.attack = p1Melody.attack;
else combinedMelody.attack = p2Melody.attack;
if (random(0, 1) < 0.05) combinedMelody.attack = constrain(combinedMelody.attack + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedMelody.decay = p1Melody.decay;
else combinedMelody.decay = p2Melody.decay;
if (random(0, 1) < 0.05) combinedMelody.decay = constrain(combinedMelody.decay + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedMelody.sustain = p1Melody.sustain;
else combinedMelody.sustain = p2Melody.sustain;
if (random(0, 1) < 0.05) combinedMelody.sustain = constrain(combinedMelody.sustain + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedMelody.release = p1Melody.release;
else combinedMelody.release = p2Melody.release;
if (random(0, 1) < 0.05) combinedMelody.release = constrain(combinedMelody.release + random(-0.1, 0.1), 0, 10);
//effects
for (let i = 0; i < 2; i++) {
if (random(0, 1) < 0.05) combinedMelody.effects[i] = p1Melody.effects[i];
else combinedMelody.effects[i] = p2Melody.effects[i];
if (random(0, 1) < 0.05) {
if (combinedMelody.effects[i].type == "delay") {
combinedMelody.effects[i].delayTime = constrain(combinedMelody.effects[i].delayTime + random(-5, 5), 1, 33);
combinedMelody.effects[i].feedback = constrain(combinedMelody.effects[i].feedback + random(-0.1, 0.1), 0.01, 0.9);
}
if (combinedMelody.effects[i].type == "distortion") {
combinedMelody.effects[i].ammount = constrain(combinedMelody.effects[i].ammount + random(-0.1, 0.1), 0.01, 5);
}
if (combinedMelody.effects[i].type == "wah") {
combinedMelody.effects[i].baseFrequency = constrain(combinedMelody.effects[i].baseFrequency + random(-10, 10), 0.2, 100);
combinedMelody.effects[i].octaves = constrain(combinedMelody.effects[i].octaves + random(-1, 1), 2, 10);
combinedMelody.effects[i].sensitivity = constrain(combinedMelody.effects[i].sensitivity + random(-5, 5), -50, 10);
}
if (combinedMelody.effects[i].type == "bitcrusher") {
combinedMelody.effects[i].bits = constrain(combinedMelody.effects[i].bits + random(-2, 2), 2, 13);
}
if (combinedMelody.effects[i].type == "chorus") {
combinedMelody.effects[i].frequency = constrain(combinedMelody.effects[i].frequency + random(-1, 1), 0.1, 5);
combinedMelody.effects[i].delayTime = constrain(combinedMelody.effects[i].delayTime + random(-5, 5), 0.1, 50);
combinedMelody.effects[i].depth = constrain(combinedMelody.effects[i].depth + random(-2, 2), 0, 20);
}
if (combinedMelody.effects[i].type == "phaser") {
combinedMelody.effects[i].frequency = constrain(combinedMelody.effects[i].frequency + random(-1, 1), 1, 11);
combinedMelody.effects[i].octaves = constrain(combinedMelody.effects[i].octaves + random(-1, 1), 1, 7);
combinedMelody.effects[i].baseFrequency = constrain(combinedMelody.effects[i].baseFrequency + random(-200, 200), 20, 3000);
}
if (combinedMelody.effects[i].type == "autofilter") {
combinedMelody.effects[i].frequency = constrain(combinedMelody.effects[i].frequency + random(-5, 5), 0, 32);
}
}
}
//frequencys
for (let i = 0; i < 7; i++) {
if (random(0, 1) > 0.5) combinedMelody.frequency[i] = p1Melody.frequency[i];
else combinedMelody.frequency[i] = p2Melody.frequency[i];
if (random(0, 1) < 0.05) combinedMelody.frequency[i] = constrain(combinedMelody.frequency[i] + random(-0.1, 0.1), 0, 1);
}
//sequencer
if (random(0, 1) > 0.5) combinedMelody.sequenzer = p1Melody.sequenzer;
else combinedMelody.sequenzer = p2Melody.sequenzer;
if (random(0, 1) < 0.05) {
for (let i = 0; i < combinedMelody.sequenzer.length; i++) {
let r = random(0, 1);
if (r < 0.1) combinedMelody.sequenzer.splice(i, 1);
else if (r < 0.2) {
let lastNote;
if (i > 0) lastNote = combinedMelody.sequenzer[i - 1];
else lastNote = {
deploy: 0,
duration: 1 + "n",
note: int(random(0, 7))
};
let nextNote;
if (i != combinedMelody.sequenzer.length - 1) nextNote = combinedMelody.sequenzer[i + 1];
else nextNote = {
deploy: 16 * 4 + 1,
duration: 1 + "n",
note: int(random(0, 7))
};
let n = {
deploy: int(random(lastNote.deploy + 1, nextNote.deploy)),
duration: 17 - int(random(1, nextNote.deploy - combinedMelody.sequenzer[i].deploy)) + "n",
note: int(random(0, 7))
};
combinedMelody.sequenzer.splice(i, 0, n);
} else if (r < 0.3) {
combinedMelody.sequenzer[i].note = int(random(0, 7));
}
}
}
//----------------end CombineMelody
// combine Atmo
combinedAtmo = p1Atmo;
if (random(0, 1) > 0.5) combinedAtmo.osc1Shape = p1Atmo.osc1Shape;
else combinedAtmo.osc1Shape = p2Atmo.osc1Shape;
if (random(0, 1) < 0.05) combinedAtmo.osc1Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedAtmo.osc1Octave = p1Atmo.osc1Octave;
else combinedAtmo.osc1Octave = p2Atmo.osc1Octave;
if (random(0, 1) < 0.05) combinedAtmo.osc1Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedAtmo.osc1Amp = p1Atmo.osc1Amp;
else combinedAtmo.osc1Amp = p2Atmo.osc1Amp;
if (random(0, 1) < 0.05) combinedAtmo.osc1Amp = combinedAtmo.osc1Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedAtmo.osc2Shape = p1Atmo.osc2Shape;
else combinedAtmo.osc2Shape = p2Atmo.osc2Shape;
if (random(0, 1) < 0.05) combinedAtmo.osc2Shape = random(["sine", "sawtooth", "square", "triangle"]);
if (random(0, 1) > 0.5) combinedAtmo.osc2Octave = p1Atmo.osc2Octave;
else combinedAtmo.osc2Octave = p2Atmo.osc2Octave;
if (random(0, 1) < 0.05) combinedAtmo.osc2Octave = random([1 / 4, 1 / 3, 1 / 2, 1, 2, 3]);
if (random(0, 1) > 0.5) combinedAtmo.osc2Amp = p1Atmo.osc2Amp;
else combinedAtmo.osc2Amp = p2Atmo.osc2Amp;
if (random(0, 1) < 0.05) combinedAtmo.osc2Amp = combinedAtmo.osc2Amp + random(-5, 5);
if (random(0, 1) > 0.5) combinedAtmo.detune = p1Atmo.detune;
else combinedAtmo.detune = p2Atmo.detune;
if (random(0, 1) < 0.05) combinedAtmo.detune = combinedAtmo.detune + random(-5, 5);
if (random(0, 1) > 0.5) combinedAtmo.noise = p1Atmo.noise;
else combinedAtmo.noise = p2Atmo.noise;
if (random(0, 1) > 0.5) combinedAtmo.noiseAmp = p1Atmo.noiseAmp;
else combinedAtmo.noiseAmp = p2Atmo.noiseAmp;
if (random(0, 1) < 0.05) combinedAtmo.noise = combinedAtmo.noise + random(-5, 5);
if (random(0, 1) > 0.5) combinedAtmo.attack = p1Atmo.attack;
else combinedAtmo.attack = p2Atmo.attack;
if (random(0, 1) < 0.05) combinedAtmo.attack = constrain(combinedAtmo.attack + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedAtmo.decay = p1Atmo.decay;
else combinedAtmo.decay = p2Atmo.decay;
if (random(0, 1) < 0.05) combinedAtmo.decay = constrain(combinedAtmo.decay + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedAtmo.sustain = p1Atmo.sustain;
else combinedAtmo.sustain = p2Atmo.sustain;
if (random(0, 1) < 0.05) combinedAtmo.sustain = constrain(combinedAtmo.sustain + random(-0.1, 0.1), 0, 1);
if (random(0, 1) > 0.5) combinedAtmo.release = p1Atmo.release;
else combinedAtmo.release = p2Atmo.release;
if (random(0, 1) < 0.05) combinedAtmo.release = constrain(combinedAtmo.release + random(-0.1, 0.1), 0, 10);
//effects
for (let i = 0; i < 2; i++) {
if (random(0, 1) < 0.05) combinedAtmo.effects[i] = p1Atmo.effects[i];
else combinedAtmo.effects[i] = p2Atmo.effects[i];
if (random(0, 1) < 0.05) {
if (combinedAtmo.effects[i].type == "delay") {
combinedAtmo.effects[i].delayTime = constrain(combinedAtmo.effects[i].delayTime + random(-5, 5), 1, 33);
combinedAtmo.effects[i].feedback = constrain(combinedAtmo.effects[i].feedback + random(-0.1, 0.1), 0.01, 0.9);
}
if (combinedAtmo.effects[i].type == "distortion") {
combinedAtmo.effects[i].ammount = constrain(combinedAtmo.effects[i].ammount + random(-0.1, 0.1), 0.01, 5);
}
if (combinedAtmo.effects[i].type == "wah") {
combinedAtmo.effects[i].baseFrequency = constrain(combinedAtmo.effects[i].baseFrequency + random(-10, 10), 0.2, 100);
combinedAtmo.effects[i].octaves = constrain(combinedAtmo.effects[i].octaves + random(-1, 1), 2, 10);
combinedAtmo.effects[i].sensitivity = constrain(combinedAtmo.effects[i].sensitivity + random(-5, 5), -50, 10);
}
if (combinedAtmo.effects[i].type == "bitcrusher") {
combinedAtmo.effects[i].bits = constrain(combinedAtmo.effects[i].bits + random(-2, 2), 2, 13);
}
if (combinedAtmo.effects[i].type == "chorus") {
combinedAtmo.effects[i].frequency = constrain(combinedAtmo.effects[i].frequency + random(-1, 1), 0.1, 5);
combinedAtmo.effects[i].delayTime = constrain(combinedAtmo.effects[i].delayTime + random(-5, 5), 0.1, 50);
combinedAtmo.effects[i].depth = constrain(combinedAtmo.effects[i].depth + random(-2, 2), 0, 20);
}
if (combinedAtmo.effects[i].type == "phaser") {
combinedAtmo.effects[i].frequency = constrain(combinedAtmo.effects[i].frequency + random(-1, 1), 1, 11);
combinedAtmo.effects[i].octaves = constrain(combinedAtmo.effects[i].octaves + random(-1, 1), 1, 7);
combinedAtmo.effects[i].baseFrequency = constrain(combinedAtmo.effects[i].baseFrequency + random(-200, 200), 20, 3000);
}
if (combinedAtmo.effects[i].type == "autofilter") {
combinedAtmo.effects[i].frequency = constrain(combinedAtmo.effects[i].frequency + random(-5, 5), 0, 32);
}
}
}
//frequencys
for (let i = 0; i < 7; i++) {
if (random(0, 1) > 0.5) combinedAtmo.frequency[i] = p1Atmo.frequency[i];
else combinedAtmo.frequency[i] = p2Atmo.frequency[i];
if (random(0, 1) < 0.05) combinedAtmo.frequency[i] = constrain(combinedAtmo.frequency[i] + random(-0.1, 0.1), 0, 1);
}
//sequencer
if (random(0, 1) > 0.5) combinedAtmo.sequenzer = p1Atmo.sequenzer;
else combinedAtmo.sequenzer = p2Atmo.sequenzer;
if (random(0, 1) < 0.05) {
for (let i = 0; i < combinedAtmo.sequenzer.length; i++) {
let r = random(0, 1);
if (r < 0.1) combinedAtmo.sequenzer.splice(i, 1);
else if (r < 0.2) {
let lastNote;
if (i > 0) lastNote = combinedAtmo.sequenzer[i - 1];
else lastNote = {
deploy: 0,
duration: 1 + "n",
note: int(random(0, 7))
};
let nextNote;
if (i != combinedAtmo.sequenzer.length - 1) nextNote = combinedAtmo.sequenzer[i + 1];
else nextNote = {
deploy: 16 * 4 + 1,
duration: 1 + "n",
note: int(random(0, 7))
};
let n = {
deploy: int(random(lastNote.deploy + 1, nextNote.deploy)),
duration: 17 - int(random(1, nextNote.deploy - combinedAtmo.sequenzer[i].deploy)) + "n",
note: int(random(0, 7))
};
combinedAtmo.sequenzer.splice(i, 0, n);
} else if (r < 0.3) {
combinedAtmo.sequenzer[i].note = int(random(0, 7));
}
}
}
activeBeat = combinedBeat;
activeMelody = combinedMelody;
activeAtmo = combinedAtmo;
}