2015-07-04 17:48:44 +00:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta charset="utf-8">
|
|
|
|
<title>VISUALIZING ENVELOPES</title>
|
|
|
|
|
2019-01-09 01:21:29 +00:00
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
|
|
|
|
<link rel="icon" type="image/png" sizes="174x174" href="./favicon.png">
|
2015-07-04 17:48:44 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.0.0/p5.js"></script>
|
2020-07-18 00:58:09 +00:00
|
|
|
<script src="https://cdnjs.cloudflare.com/ajax/libs/webcomponentsjs/2.4.3/webcomponents-bundle.js"></script>
|
2020-07-18 00:36:42 +00:00
|
|
|
<link href="https://fonts.googleapis.com/css?family=Material+Icons&display=block" rel="stylesheet"/>
|
2019-01-09 01:21:29 +00:00
|
|
|
<script src="../build/Tone.js"></script>
|
2020-07-18 00:58:09 +00:00
|
|
|
<script src="./js/tone-ui.js"></script>
|
|
|
|
<script src="./js/components.js"></script>
|
2015-07-04 17:48:44 +00:00
|
|
|
</head>
|
|
|
|
<body>
|
2020-07-18 00:36:42 +00:00
|
|
|
<tone-example label="Tone with p5.js">
|
|
|
|
<div slot="explanation">
|
2019-01-09 01:21:29 +00:00
|
|
|
Access the envelopes current value to synchronize visuals. This sketch uses <a href="https://p5js.org" target="_blank">p5.js</a> for canvas rendering.
|
|
|
|
<br><br>
|
|
|
|
Example by <a href="https://github.com/polyrhythmatic">polyrhythmatic</a>
|
2020-07-18 00:36:42 +00:00
|
|
|
</div>
|
2019-01-09 01:21:29 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
<div id="content">
|
2019-01-09 01:21:29 +00:00
|
|
|
<tone-play-toggle></tone-play-toggle>
|
2020-07-18 00:36:42 +00:00
|
|
|
</div>
|
2019-01-09 01:21:29 +00:00
|
|
|
</tone-example>
|
|
|
|
|
|
|
|
<script type="text/javascript">
|
2020-07-18 00:36:42 +00:00
|
|
|
new p5((p5) => {
|
|
|
|
|
|
|
|
class FunkyShape {
|
|
|
|
/*
|
2020-07-19 00:20:19 +00:00
|
|
|
FunkyShape init gives initial and offset values for
|
|
|
|
the perlin noise functions in update.
|
|
|
|
Giving different initial values ensures that
|
|
|
|
each funky shape follows its own funky path
|
|
|
|
*/
|
2020-07-18 00:36:42 +00:00
|
|
|
init(xInc, yInc, xOff, yOff, radius) {
|
|
|
|
this.xInc = xInc;
|
|
|
|
this.yInc = yInc;
|
|
|
|
this.xOff = xOff;
|
|
|
|
this.yOff = yOff;
|
|
|
|
this.radius = radius;
|
|
|
|
this.xPos = 0;
|
|
|
|
this.yPos = 0;
|
|
|
|
}
|
|
|
|
// updates the x, y, and radius values of the shape
|
|
|
|
update(envelope) {
|
|
|
|
this.xPos = p5.noise(this.xOff) * p5.width;
|
|
|
|
this.yPos = p5.noise(this.yOff) * p5.height;
|
|
|
|
this.xOff += this.xInc;
|
|
|
|
this.yOff += this.yInc;
|
|
|
|
this.sRadius = this.radius * envelope;
|
|
|
|
return {
|
|
|
|
xPos: this.xPos,
|
|
|
|
yPos: this.yPos,
|
|
|
|
radius: this.sRadius
|
|
|
|
};
|
|
|
|
};
|
2019-01-09 01:21:29 +00:00
|
|
|
}
|
2020-07-19 00:20:19 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
// using our FunkyShape class
|
|
|
|
// to create a funkyCircle class
|
|
|
|
const funkyCircle = new FunkyShape();
|
|
|
|
|
|
|
|
// creating an empty array
|
|
|
|
const funkySquare = [];
|
|
|
|
// and populating it with 3 FunkyShapes
|
|
|
|
for (let i = 0; i < 3; i++) {
|
|
|
|
funkySquare[i] = new FunkyShape();
|
2019-01-09 01:21:29 +00:00
|
|
|
}
|
2020-07-18 00:36:42 +00:00
|
|
|
|
|
|
|
p5.setup = () => {
|
|
|
|
// create a canvas width and height of the screen
|
|
|
|
// document.querySelector('canvas')
|
|
|
|
p5.createCanvas(300, 300);
|
|
|
|
// no fill
|
|
|
|
p5.fill(255);
|
|
|
|
p5.strokeWeight(1);
|
|
|
|
p5.rectMode(p5.CENTER);
|
|
|
|
// initializing our funky circle
|
|
|
|
funkyCircle.init(0.01, 0.02, 0.0, 0.0, 400);
|
|
|
|
// initializing our squares with random values
|
|
|
|
// to ensure they don't follow the same path
|
|
|
|
for (let i = 0; i < 3; i++) {
|
|
|
|
const xInc = Math.random() / 10;
|
|
|
|
const yInc = Math.random() / 10;
|
|
|
|
funkySquare[i].init(xInc, yInc, 0, 0, 800);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let phase = 0;
|
|
|
|
|
|
|
|
p5.draw = () => {
|
|
|
|
p5.background(255);
|
|
|
|
p5.stroke(0);
|
|
|
|
// drawing the kick wave at the bottom
|
|
|
|
// it is composed of a simple sine wave that
|
|
|
|
// changes in height with the kick envelope
|
|
|
|
for (let i = 0; i < p5.width; i++) {
|
|
|
|
// scaling kickEnvelope value by 200
|
|
|
|
// since default is 0-1
|
|
|
|
const kickValue = kickEnvelope.value * 200;
|
|
|
|
// multiplying this value to scale the sine wave
|
|
|
|
// depending on x position
|
|
|
|
const yDot = Math.sin((i / 60) + phase) * kickValue;
|
|
|
|
p5.point(i, p5.height -150 + yDot);
|
|
|
|
}
|
|
|
|
// increasing phase means that the kick wave will
|
|
|
|
// not be standing and looks more dynamic
|
|
|
|
phase += 1;
|
|
|
|
// updating circle and square positions with
|
|
|
|
// bass and bleep envelope values
|
|
|
|
const circlePos = funkyCircle.update(bassEnvelope.value);
|
|
|
|
// circlePos returns x and y positions as an object
|
|
|
|
p5.ellipse(circlePos.xPos, circlePos.yPos, circlePos.radius, circlePos.radius);
|
|
|
|
p5.stroke("red");
|
|
|
|
for (let i = 0; i < 3; i++) {
|
|
|
|
const squarePos = funkySquare[i].update(bleepEnvelope.value);
|
|
|
|
p5.rect(squarePos.xPos, squarePos.yPos, squarePos.radius, squarePos.radius);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, document.querySelector("#content"));
|
|
|
|
|
|
|
|
// filtering the hi-hats a bit
|
|
|
|
// to make them sound nicer
|
|
|
|
const lowPass = new Tone.Filter({
|
|
|
|
frequency: 14000,
|
|
|
|
}).toDestination();
|
|
|
|
|
|
|
|
// we can make our own hi hats with
|
|
|
|
// the noise synth and a sharp filter envelope
|
|
|
|
const openHiHat = new Tone.NoiseSynth({
|
|
|
|
volume: -10,
|
|
|
|
envelope: {
|
|
|
|
attack: 0.01,
|
|
|
|
decay: 0.3
|
2019-01-09 01:21:29 +00:00
|
|
|
},
|
2015-07-04 17:48:44 +00:00
|
|
|
}).connect(lowPass);
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const openHiHatPart = new Tone.Part(((time) => {
|
2015-12-05 18:20:53 +00:00
|
|
|
openHiHat.triggerAttack(time);
|
2020-07-18 00:36:42 +00:00
|
|
|
}), [{ "8n": 2 }, { "8n": 6 }]).start(0);
|
2015-12-05 18:20:53 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const closedHiHat = new Tone.NoiseSynth({
|
|
|
|
volume: -10,
|
|
|
|
envelope: {
|
|
|
|
attack: 0.01,
|
|
|
|
decay: 0.15
|
2019-01-09 01:21:29 +00:00
|
|
|
},
|
2015-07-04 17:48:44 +00:00
|
|
|
}).connect(lowPass);
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const closedHatPart = new Tone.Part(((time) => {
|
2015-12-05 18:20:53 +00:00
|
|
|
closedHiHat.triggerAttack(time);
|
2020-07-18 00:36:42 +00:00
|
|
|
}), [0, { "16n": 1 }, { "8n": 1 }, { "8n": 3 }, { "8n": 4 }, { "8n": 5 }, { "8n": 7 }, { "8n": 8 }]).start(0);
|
|
|
|
|
|
|
|
// BASS
|
|
|
|
const bassEnvelope = new Tone.AmplitudeEnvelope({
|
|
|
|
attack: 0.01,
|
|
|
|
decay: 0.2,
|
|
|
|
sustain: 0,
|
|
|
|
}).toDestination();
|
|
|
|
|
|
|
|
const bassFilter = new Tone.Filter({
|
|
|
|
frequency: 600,
|
|
|
|
Q: 8
|
2015-07-04 17:48:44 +00:00
|
|
|
});
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const bass = new Tone.PulseOscillator("A2", 0.4).chain(bassFilter, bassEnvelope);
|
2015-07-04 17:48:44 +00:00
|
|
|
bass.start();
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const bassPart = new Tone.Part(((time, note) => {
|
2015-12-05 18:20:53 +00:00
|
|
|
bass.frequency.setValueAtTime(note, time);
|
2019-01-09 01:21:29 +00:00
|
|
|
bassEnvelope.triggerAttack(time);
|
2020-07-18 00:36:42 +00:00
|
|
|
}), [["0:0", "A1"],
|
2015-12-05 18:20:53 +00:00
|
|
|
["0:2", "G1"],
|
|
|
|
["0:2:2", "C2"],
|
|
|
|
["0:3:2", "A1"]]).start(0);
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
// BLEEP
|
|
|
|
const bleepEnvelope = new Tone.AmplitudeEnvelope({
|
|
|
|
attack: 0.01,
|
|
|
|
decay: 0.4,
|
|
|
|
sustain: 0,
|
|
|
|
}).toDestination();
|
2015-07-04 17:48:44 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const bleep = new Tone.Oscillator("A4").connect(bleepEnvelope);
|
2015-07-04 17:48:44 +00:00
|
|
|
bleep.start();
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const bleepLoop = new Tone.Loop(((time) => {
|
2020-07-19 00:20:19 +00:00
|
|
|
bleepEnvelope.triggerAttack(time);
|
2020-07-18 00:36:42 +00:00
|
|
|
}), "2n").start(0);
|
|
|
|
|
|
|
|
// KICK
|
|
|
|
const kickEnvelope = new Tone.AmplitudeEnvelope({
|
|
|
|
attack: 0.01,
|
|
|
|
decay: 0.2,
|
|
|
|
sustain: 0,
|
|
|
|
}).toDestination();
|
|
|
|
|
|
|
|
const kick = new Tone.Oscillator("A2").connect(kickEnvelope).start();
|
|
|
|
|
|
|
|
const kickSnapEnv = new Tone.FrequencyEnvelope({
|
|
|
|
attack: 0.005,
|
|
|
|
decay: 0.01,
|
|
|
|
sustain: 0,
|
|
|
|
baseFrequency: "A2",
|
|
|
|
octaves: 2.7
|
2015-07-04 17:48:44 +00:00
|
|
|
}).connect(kick.frequency);
|
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
const kickPart = new Tone.Part(((time) => {
|
2015-12-05 18:20:53 +00:00
|
|
|
kickEnvelope.triggerAttack(time);
|
|
|
|
kickSnapEnv.triggerAttack(time);
|
2020-07-18 00:36:42 +00:00
|
|
|
}), ["0", "0:0:3", "0:2:0", "0:3:1"]).start(0);
|
2015-07-04 17:48:44 +00:00
|
|
|
|
2020-07-18 00:36:42 +00:00
|
|
|
// TRANSPORT
|
2015-07-04 17:48:44 +00:00
|
|
|
Tone.Transport.loopStart = 0;
|
|
|
|
Tone.Transport.loopEnd = "1:0";
|
|
|
|
Tone.Transport.loop = true;
|
|
|
|
|
2020-07-19 00:20:19 +00:00
|
|
|
// bind the interface
|
2020-07-18 00:36:42 +00:00
|
|
|
document.querySelector("tone-play-toggle").addEventListener("start", e => Tone.Transport.start());
|
|
|
|
document.querySelector("tone-play-toggle").addEventListener("stop", e => Tone.Transport.stop());
|
|
|
|
|
|
|
|
const controls = drawer({
|
|
|
|
parent: document.body,
|
|
|
|
open: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
controls.folder({
|
|
|
|
name: "Hihat"
|
|
|
|
}).add({
|
|
|
|
tone: lowPass,
|
|
|
|
}).add({
|
|
|
|
name: "Open Hihat",
|
|
|
|
tone: openHiHat,
|
|
|
|
}).add({
|
|
|
|
name: "Closed Hihat",
|
|
|
|
tone: closedHiHat
|
|
|
|
});
|
|
|
|
|
|
|
|
controls.folder({
|
|
|
|
name: "Bass"
|
|
|
|
}).add({
|
|
|
|
tone: bassFilter,
|
|
|
|
}).add({
|
|
|
|
tone: bass,
|
|
|
|
}).add({
|
|
|
|
tone: bassEnvelope
|
|
|
|
});
|
|
|
|
|
|
|
|
controls.folder({
|
|
|
|
name: "Bleep"
|
|
|
|
}).add({
|
|
|
|
tone: bleep,
|
|
|
|
}).add({
|
|
|
|
tone: bleepEnvelope,
|
|
|
|
});
|
|
|
|
|
|
|
|
controls.folder({
|
|
|
|
name: "Kick"
|
|
|
|
}).add({
|
|
|
|
tone: kick,
|
|
|
|
}).add({
|
|
|
|
tone: kickEnvelope,
|
|
|
|
}).add({
|
|
|
|
tone: kickSnapEnv,
|
|
|
|
});
|
2015-07-04 17:48:44 +00:00
|
|
|
</script>
|
2017-03-26 20:39:19 +00:00
|
|
|
|
2015-07-04 17:48:44 +00:00
|
|
|
</body>
|
2017-03-18 16:35:37 +00:00
|
|
|
</html>
|