diff --git a/src/Loop.ts b/src/Loop.ts
index 978e33d..50c5f8a 100644
--- a/src/Loop.ts
+++ b/src/Loop.ts
@@ -8,10 +8,7 @@ type LoopEvent = {
type LoopState = 'readyToRecord' | 'recording' | 'readyToPlay' | 'playing';
export class Loop {
- constructor(private trackIndex: number) {
- this.log('Loop created');
- }
-
+ private static masterLoop: Loop | null = null;
private state: LoopState = 'readyToRecord';
private events: LoopEvent[] = [];
private eventEmitter = new Phaser.Events.EventEmitter();
@@ -19,7 +16,10 @@ export class Loop {
private startPlayingTime = 0;
private currentLoopIndex = 0;
private loopTimeout: number | null = null;
- private static masterLoop: Loop | null = null;
+
+ constructor(private trackIndex: number) {
+ this.log('Loop created');
+ }
getStartPlayingTime() {
return this.startPlayingTime;
@@ -141,7 +141,7 @@ export class Loop {
} else {
console.log(this.events);
const masterLoopLength = Loop.masterLoop?.getLoopLength();
- if (!masterLoopLength){
+ if (!masterLoopLength) {
throw new Error('masterLoopLength is not set');
}
// check how many times the master loop fits in this loop so that it is always in sync
diff --git a/src/main.tsx b/src/main.tsx
index e63eef4..d5e2cc9 100644
--- a/src/main.tsx
+++ b/src/main.tsx
@@ -4,6 +4,6 @@ import App from './App.tsx'
ReactDOM.createRoot(document.getElementById('root')!).render(
-
+
,
)
diff --git a/src/samples/drums/hihat.ts b/src/samples/drums/hihat.ts
index f36db74..6d007aa 100644
--- a/src/samples/drums/hihat.ts
+++ b/src/samples/drums/hihat.ts
@@ -1,4 +1,3 @@
-
// Improvements:
// Shorter Noise Burst: The noise buffer duration is now much shorter (0.05 seconds), which results in a sharper, crisper hi-hat sound.
// High-Pass Filter Tweaks: The cutoff frequency is set to 8000 Hz to eliminate more of the low-end and leave the high frequencies that give the hi-hat its metallic sound.
diff --git a/src/samples/play-sample.ts b/src/samples/play-sample.ts
index 6afc9ea..829cfc2 100644
--- a/src/samples/play-sample.ts
+++ b/src/samples/play-sample.ts
@@ -22,7 +22,7 @@ const sampleToAudioFn: Record void> = {
export const playSample = (sample: Sample) => {
try {
sampleToAudioFn[sample]();
- } catch(e) {
+ } catch (e) {
console.error(`Error playing ${sample}`, e);
resetAudioContext();
sampleToAudioFn[sample]();
@@ -30,4 +30,4 @@ export const playSample = (sample: Sample) => {
}
export type Sample = 'hihat' | 'hihat-open' | 'ride' | 'crash'
- | 'snare' | 'kick' | 'tom-low' | 'tom-high';
+ | 'snare' | 'kick' | 'tom-low' | 'tom-high';
diff --git a/src/samples/synth-frequencies.ts b/src/samples/synth-frequencies.ts
index 3d8c7a2..e5b2cf9 100644
--- a/src/samples/synth-frequencies.ts
+++ b/src/samples/synth-frequencies.ts
@@ -10,7 +10,7 @@ export type NoteFreq = {
function createNoteTable() {
const noteFreq: Partial[] = [];
- for (let i=0; i< 9; i++) {
+ for (let i = 0; i < 9; i++) {
noteFreq[i] = {};
}
diff --git a/src/scenes/DrumsScene.ts b/src/scenes/DrumsScene.ts
index 0c52db0..95419a8 100644
--- a/src/scenes/DrumsScene.ts
+++ b/src/scenes/DrumsScene.ts
@@ -5,8 +5,8 @@ import {LoopTracksScene} from './LoopTracksScene.ts';
import {rotateArray} from '../utils/math.ts';
type Pad = {
- instrument: Sample,
- button: Phaser.GameObjects.Rectangle,
+ instrument: Sample,
+ button: Phaser.GameObjects.Rectangle,
}
const padColors: Record = {
@@ -20,7 +20,7 @@ const padColors: Record = {
'tom-high': '#9B59B6',
};
-type DrumsType= 'drums' | 'other';
+type DrumsType = 'drums' | 'other';
export class DrumsScene extends Phaser.Scene {
@@ -28,7 +28,7 @@ export class DrumsScene extends Phaser.Scene {
super();
}
- create({ type}: { type: DrumsType }) {
+ create({type}: { type: DrumsType }) {
if (type) {
this.type = type;
}
@@ -58,9 +58,9 @@ export class DrumsScene extends Phaser.Scene {
const width = isPortrait ? window.innerWidth / colNumber : (window.innerWidth - LoopTracksScene.sceneWidthHeight) / colNumber;
const height = isPortrait ? (window.innerHeight - LoopTracksScene.sceneWidthHeight) / rowNumber : window.innerHeight / rowNumber;
- const currentPads = isPortrait ? rotateArray(pads, rowNumber, colNumber): pads;
+ const currentPads = isPortrait ? rotateArray(pads, rowNumber, colNumber) : pads;
- currentPads.forEach(({ button }, index) => {
+ currentPads.forEach(({button}, index) => {
const x = (index % colNumber) * width;
const y = Math.floor(index / colNumber) * height;
const offsetX = isPortrait ? 0 : LoopTracksScene.sceneWidthHeight;
diff --git a/src/scenes/GiberishScene.ts b/src/scenes/GiberishScene.ts
index 8855964..cc7336a 100644
--- a/src/scenes/GiberishScene.ts
+++ b/src/scenes/GiberishScene.ts
@@ -4,8 +4,8 @@ import {LoopTracksScene} from './LoopTracksScene.ts';
import {rotateArray} from '../utils/math.ts';
type Pad = {
- instrument: number,
- button: Phaser.GameObjects.Rectangle,
+ instrument: number,
+ button: Phaser.GameObjects.Rectangle,
}
declare const Freeverb: any, Bus2: any, Gibberish: any, Synth: any, Add: any, Sine: any, Sequencer: any;
@@ -24,33 +24,33 @@ const testNote = () => {
const beat = 22050
// global reverb object
- const verb = Freeverb({ input:Bus2(), roomSize:.975, damping:.5 }).connect()
+ const verb = Freeverb({input: Bus2(), roomSize: .975, damping: .5}).connect()
/*** bassline ***/
const bass = Synth({
- gain:.15,
- attack:44,
+ gain: .15,
+ attack: 44,
decay: 5512,
- Q:.8, // CAREFUL!!!
- filterType:2,
- saturation:2,
- filterMult:3.25,
- antialias:true,
- cutoff: Add( 1, Sine({ frequency:.1, gain:.75 }) )
+ Q: .8, // CAREFUL!!!
+ filterType: 2,
+ saturation: 2,
+ filterMult: 3.25,
+ antialias: true,
+ cutoff: Add(1, Sine({frequency: .1, gain: .75}))
})
- .connect( Gibberish.output )
- .connect( verb.input, .5 )
+ .connect(Gibberish.output)
+ .connect(verb.input, .5)
- const bassNotes = [55,110,165,220]
- bassSeq = Sequencer.make( [55,110,165,220], [beat/4], bass, 'note' ).start()
+ const bassNotes = [55, 110, 165, 220]
+ bassSeq = Sequencer.make([55, 110, 165, 220], [beat / 4], bass, 'note').start()
noteSeq = Sequencer.make(
[
- bassNotes.map( v=>v*1.25 ),
- bassNotes.map( v=>v*1.25*.8 ),
- bassNotes.map( v=>v*1.25*.8*.8 ),
- bassNotes.map( v=>v*1.25*.8*.8*1.25 ),
+ bassNotes.map(v => v * 1.25),
+ bassNotes.map(v => v * 1.25 * .8),
+ bassNotes.map(v => v * 1.25 * .8 * .8),
+ bassNotes.map(v => v * 1.25 * .8 * .8 * 1.25),
],
- [beat*16],
+ [beat * 16],
bassSeq,
'values'
);
@@ -65,7 +65,7 @@ export class GibberishScene extends Phaser.Scene {
super();
}
- create({ numberOfPads = 8 }: { numberOfPads: number}) {
+ create({numberOfPads = 8}: { numberOfPads: number }) {
this.createPads(numberOfPads);
this.scene.get(LoopTracksScene.key).events.emit('track-selected');
}
@@ -83,9 +83,9 @@ export class GibberishScene extends Phaser.Scene {
const width = isPortrait ? window.innerWidth / colNumber : (window.innerWidth - LoopTracksScene.sceneWidthHeight) / colNumber;
const height = isPortrait ? (window.innerHeight - LoopTracksScene.sceneWidthHeight) / rowNumber : window.innerHeight / rowNumber;
- const currentPads = isPortrait ? rotateArray(pads, rowNumber, colNumber): pads;
+ const currentPads = isPortrait ? rotateArray(pads, rowNumber, colNumber) : pads;
- currentPads.forEach(({ button }, index) => {
+ currentPads.forEach(({button}, index) => {
const x = (index % colNumber) * width;
const y = Math.floor(index / colNumber) * height;
const offsetX = isPortrait ? 0 : LoopTracksScene.sceneWidthHeight;
diff --git a/src/scenes/LoopTracksScene.ts b/src/scenes/LoopTracksScene.ts
index 806d0d5..188451e 100644
--- a/src/scenes/LoopTracksScene.ts
+++ b/src/scenes/LoopTracksScene.ts
@@ -34,6 +34,7 @@ type Track = {
export class LoopTracksScene extends Phaser.Scene {
static key = 'LoopTracksScene';
+ static numTracks = 5;
private static tracks: Track[]
private static instance: LoopTracksScene;
@@ -42,8 +43,6 @@ export class LoopTracksScene extends Phaser.Scene {
LoopTracksScene.instance = this;
}
- static numTracks = 5;
-
static get sceneWidthHeight() {
const width = window.innerWidth;
const height = window.innerHeight;
@@ -61,6 +60,10 @@ export class LoopTracksScene extends Phaser.Scene {
}
}
+ public static getTrackSceneKey(index: number) {
+ return `track_scene_${index}`;
+ }
+
create() {
this.cameras.main
.setOrigin(0, 0)
@@ -84,6 +87,33 @@ export class LoopTracksScene extends Phaser.Scene {
}
}
+ public getTrackScene(index: number) {
+ return this.scene.get(LoopTracksScene.getTrackSceneKey(index));
+ }
+
+ updateProgressArc(track: Track) {
+ const loopLength = track.loop.getLoopLength();
+ if (!loopLength) {
+ throw new Error('Loop length is not defined');
+ }
+
+ const elapsed = Date.now() - track.loop.getStartPlayingTime();
+ const progress = elapsed / loopLength; // 0 to 1 based on loop progress
+
+ // Clear previous arc and redraw
+ track.loopProgressArc.clear();
+ track.loopProgressArc.lineStyle(4, 0x00FF00, 1); // green color with 4px thickness
+
+ // Calculate the angle for the arc based on progress
+ const startAngle = Phaser.Math.DegToRad(-90); // start from the top
+ const endAngle = startAngle + Phaser.Math.DegToRad(360 * progress);
+
+ // Draw the arc around the button
+ track.loopProgressArc.beginPath();
+ track.loopProgressArc.arc(track.buttonText.x, track.buttonText.y, track.buttonSelectedCircle.width / 2, startAngle, endAngle, false); // radius slightly larger than button
+ track.loopProgressArc.strokePath();
+ }
+
private createTracks() {
LoopTracksScene.tracks = new Array(LoopTracksScene.numTracks).fill(null).map((_, index) => {
return {
@@ -195,14 +225,6 @@ export class LoopTracksScene extends Phaser.Scene {
}
}
- public static getTrackSceneKey(index: number) {
- return `track_scene_${index}`;
- }
-
- public getTrackScene(index: number) {
- return this.scene.get(LoopTracksScene.getTrackSceneKey(index));
- }
-
private updateControlsState() {
LoopTracksScene.tracks.forEach((track, index) => {
track.button.setFillStyle(hexToColor(trackColorsState.unselected));
@@ -228,7 +250,7 @@ export class LoopTracksScene extends Phaser.Scene {
.setColor(track.loop.isRecording() ? controlColors.recording : controlColors.idle)
.setFontSize(FontSize.big)
.setPosition(
- isPortrait ? track.controlIcon.x: track.controlIcon.x + 5,
+ isPortrait ? track.controlIcon.x : track.controlIcon.x + 5,
track.controlIcon.y + 10
)
track.buttonSelectedCircle.setVisible(false);
@@ -243,27 +265,4 @@ export class LoopTracksScene extends Phaser.Scene {
}
});
}
-
- updateProgressArc(track: Track) {
- const loopLength = track.loop.getLoopLength();
- if (!loopLength) {
- throw new Error('Loop length is not defined');
- }
-
- const elapsed = Date.now() - track.loop.getStartPlayingTime();
- const progress = elapsed / loopLength; // 0 to 1 based on loop progress
-
- // Clear previous arc and redraw
- track.loopProgressArc.clear();
- track.loopProgressArc.lineStyle(4, 0x00FF00, 1); // green color with 4px thickness
-
- // Calculate the angle for the arc based on progress
- const startAngle = Phaser.Math.DegToRad(-90); // start from the top
- const endAngle = startAngle + Phaser.Math.DegToRad(360 * progress);
-
- // Draw the arc around the button
- track.loopProgressArc.beginPath();
- track.loopProgressArc.arc(track.buttonText.x, track.buttonText.y, track.buttonSelectedCircle.width/2, startAngle, endAngle, false); // radius slightly larger than button
- track.loopProgressArc.strokePath();
- }
}
diff --git a/src/utils/colors.ts b/src/utils/colors.ts
index 6a095af..a4e8be2 100644
--- a/src/utils/colors.ts
+++ b/src/utils/colors.ts
@@ -2,7 +2,7 @@ import Phaser from 'phaser';
export type HexaColor = `#${string}`;
-export const hexToColor = (hex: HexaColor, isDarkMode= false) => {
+export const hexToColor = (hex: HexaColor, isDarkMode = false) => {
return Phaser.Display.Color.HexStringToColor(hex)
.darken(isDarkMode ? 75 : 0)
.color;
diff --git a/src/utils/fonts.ts b/src/utils/fonts.ts
index 48068fe..aff549d 100644
--- a/src/utils/fonts.ts
+++ b/src/utils/fonts.ts
@@ -12,7 +12,7 @@ export const FontColor = {
white: '#FFF',
}
-export const loadFonts = async() => {
+export const loadFonts = async () => {
const newFontFace = new FontFace(FontFamily.Icons, 'url(./fonts/material.woff2)');
document.fonts.add(newFontFace);
return newFontFace.load().catch(error => console.error(error));