From 030fbe34e2d8acd11c65746c81afb22282a6b29d Mon Sep 17 00:00:00 2001 From: krippix Date: Mon, 5 Jun 2023 02:35:57 +0200 Subject: [PATCH 1/6] removed unneeded code --- html/assets/scripts/midi.js | 10 ++-------- html/assets/scripts/render.js | 2 -- 2 files changed, 2 insertions(+), 10 deletions(-) diff --git a/html/assets/scripts/midi.js b/html/assets/scripts/midi.js index fd8b7ad..abb3628 100644 --- a/html/assets/scripts/midi.js +++ b/html/assets/scripts/midi.js @@ -2,11 +2,9 @@ var keyboard = document.getElementById("piano"); var svgDoc; // loaded keyboard svg const keysB = [2,5,7,10,12,14,17,19,22,24,26,29,31,34,36,38,41,43,46,48,50,53,55,58,60,62,65,67,70,72,74,77,79,82,84,86] -const keyTo = {} window.addEventListener("load", function() { svgDoc = keyboard.contentDocument; - console.log("image loaded"); }, false) // press or unpress keyboard button @@ -22,15 +20,11 @@ function setKeyState(note, state) { } } - // Check if browser supports MIDI -if (navigator.requestMIDIAccess) { - console.log('This browser supports WebMIDI!'); -} else { - console.log('WebMIDI is not supported in this browser.'); +if (!navigator.requestMIDIAccess) { + console.log('WebMIDI is not supported in this browser. Or Site was not opened via HTTPS'); } - // Attempt to establish a connection to the MIDI device navigator.requestMIDIAccess() .then(onMIDISuccess, onMIDIFailure); diff --git a/html/assets/scripts/render.js b/html/assets/scripts/render.js index d11aaa4..48e793e 100644 --- a/html/assets/scripts/render.js +++ b/html/assets/scripts/render.js @@ -46,10 +46,8 @@ class Notation { to_set.forEach( note => { if (note.clef == "treble") { - console.log("amogus"); this.notesTop.push(new Vex.Flow.StaveNote(note)); } else { - console.log("amogus"); this.notesBottom.push(new Vex.Flow.StaveNote(note)); } }); From 772d23b02ca469c6b2d13657704bd7208f20e7a7 Mon Sep 17 00:00:00 2001 From: krippix Date: Mon, 5 Jun 2023 02:36:25 +0200 Subject: [PATCH 2/6] added map from str -> number --- html/assets/scripts/map.js | 161 +++++++++++++++++++++++++++++++++++++ html/index.html | 3 + 2 files changed, 164 insertions(+) create mode 100644 html/assets/scripts/map.js diff --git a/html/assets/scripts/map.js b/html/assets/scripts/map.js new file mode 100644 index 0000000..9a12695 --- /dev/null +++ b/html/assets/scripts/map.js @@ -0,0 +1,161 @@ +// access ex: noteMap.get("C0"); +// Maps key names to key number +{ + let values = [ + ["A0" , 1], + ["A#0", 2], + ["Bb0", 2], + ["B0" , 3], + ["B#0", 4], + ["Cb1", 3], + ["C1" , 4], + ["C#1", 5], + ["Db1", 5], + ["D1" , 6], + ["D#1", 7], + ["Eb1", 7], + ["E1" , 8], + ["E#1", 9], + ["Fb1", 8], + ["F1" , 9], + ["F#1",10], + ["Gb1",10], + ["G1" ,11], + ["G#1",12], + ["Ab1",12], + ["A1" ,13], + ["A#1",14], + ["Bb1",14], + ["B1" ,15], + ["B#1",16], + ["Cb2",15], + ["C2" ,16], + ["C#2",17], + ["Db2",17], + ["D2" ,18], + ["D#2",19], + ["Eb2",19], + ["E2" ,20], + ["E#2",21], + ["Fb2",20], + ["F2" ,21], + ["F#2",22], + ["Gb2",22], + ["G2" ,23], + ["G#2",24], + ["Ab2",24], + ["A2" ,25], + ["A#2",26], + ["Bb2",26], + ["B2" ,27], + ["B#2",28], + ["Cb3",27], + ["C3" ,28], + ["C#3",29], + ["Db3",29], + ["D3" ,30], + ["D#3",31], + ["Eb3",31], + ["E3" ,32], + ["E#3",33], + ["Fb3",32], + ["F3" ,33], + ["F#3",34], + ["Gb3",34], + ["G3" ,35], + ["G#3",36], + ["Ab3",36], + ["A3" ,37], + ["A#3",38], + ["Bb3",38], + ["B3" ,39], + ["B#3",40], + ["Cb4",39], + ["C4" ,40], + ["C#4",41], + ["Db4",41], + ["D4" ,42], + ["D#4",43], + ["Eb4",43], + ["E4" ,44], + ["E#4",45], + ["Fb4",44], + ["F4" ,45], + ["F#4",46], + ["Gb4",46], + ["G4" ,47], + ["G#4",48], + ["Ab4",48], + ["A4" ,49], + ["A#4",50], + ["Bb4",50], + ["B4" ,51], + ["B#4",52], + ["Cb5",51], + ["C5" ,52], + ["C#5",53], + ["Db5",53], + ["D5" ,54], + ["D#5",55], + ["Eb5",55], + ["E5" ,56], + ["E#5",57], + ["Fb5",56], + ["F5" ,57], + ["F#5",58], + ["Gb5",58], + ["G5" ,59], + ["G#5",60], + ["Ab5",60], + ["A5" ,61], + ["A#5",62], + ["Bb5",62], + ["B5" ,63], + ["B#5",64], + ["Cb6",63], + ["C6" ,64], + ["C#6",65], + ["Db6",65], + ["D6" ,66], + ["D#6",67], + ["Eb6",67], + ["E6" ,68], + ["E#6",69], + ["Fb6",68], + ["F6" ,69], + ["F#6",70], + ["Gb6",70], + ["G6" ,71], + ["G#6",72], + ["Ab6",72], + ["A6" ,73], + ["A#6",74], + ["Bb6",74], + ["B6" ,75], + ["B#6",76], + ["Cb7",75], + ["C7" ,76], + ["C#7",77], + ["Db7",77], + ["D7" ,78], + ["D#7",79], + ["Eb7",79], + ["E7" ,80], + ["E#7",81], + ["Fb7",80], + ["F7" ,81], + ["F#7",82], + ["Gb7",82], + ["G7" ,83], + ["G#7",84], + ["Ab7",84], + ["A7" ,85], + ["A#7",86], + ["Bb7",86], + ["B7" ,87], + ["B#7",88], + ["Cb8",87], + ["C8", 88] + ]; + var noteMap = new Map(values); +} \ No newline at end of file diff --git a/html/index.html b/html/index.html index 5ea925d..50b46bb 100644 --- a/html/index.html +++ b/html/index.html @@ -1,3 +1,4 @@ + MidiTrainer @@ -9,6 +10,8 @@ + + \ No newline at end of file From 3fbb9df8c7054dfe7a50d1ca54ca02077975a73f Mon Sep 17 00:00:00 2001 From: krippix Date: Mon, 5 Jun 2023 17:47:24 +0200 Subject: [PATCH 3/6] added favicon.ico --- html/favicon.ico | Bin 0 -> 1094 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 html/favicon.ico diff --git a/html/favicon.ico b/html/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..5272c72721efe6d15360c4ce1ee618c4e9722564 GIT binary patch literal 1094 zcmeIvu?>ST5QX7$f^=oNv}sdj0~jTnzy?{#*~C&7pmbs7^4=XOkRtj>fUFPWA0&)O z!Z{2F_tuG=MdXSxU`!ZX-#W}}Yk9q Date: Mon, 5 Jun 2023 17:47:46 +0200 Subject: [PATCH 4/6] split notes up into categories --- html/assets/scripts/map.js | 326 +++++++++++++++++++------------------ 1 file changed, 167 insertions(+), 159 deletions(-) diff --git a/html/assets/scripts/map.js b/html/assets/scripts/map.js index 9a12695..2595ba1 100644 --- a/html/assets/scripts/map.js +++ b/html/assets/scripts/map.js @@ -1,161 +1,169 @@ // access ex: noteMap.get("C0"); // Maps key names to key number -{ - let values = [ - ["A0" , 1], - ["A#0", 2], - ["Bb0", 2], - ["B0" , 3], - ["B#0", 4], - ["Cb1", 3], - ["C1" , 4], - ["C#1", 5], - ["Db1", 5], - ["D1" , 6], - ["D#1", 7], - ["Eb1", 7], - ["E1" , 8], - ["E#1", 9], - ["Fb1", 8], - ["F1" , 9], - ["F#1",10], - ["Gb1",10], - ["G1" ,11], - ["G#1",12], - ["Ab1",12], - ["A1" ,13], - ["A#1",14], - ["Bb1",14], - ["B1" ,15], - ["B#1",16], - ["Cb2",15], - ["C2" ,16], - ["C#2",17], - ["Db2",17], - ["D2" ,18], - ["D#2",19], - ["Eb2",19], - ["E2" ,20], - ["E#2",21], - ["Fb2",20], - ["F2" ,21], - ["F#2",22], - ["Gb2",22], - ["G2" ,23], - ["G#2",24], - ["Ab2",24], - ["A2" ,25], - ["A#2",26], - ["Bb2",26], - ["B2" ,27], - ["B#2",28], - ["Cb3",27], - ["C3" ,28], - ["C#3",29], - ["Db3",29], - ["D3" ,30], - ["D#3",31], - ["Eb3",31], - ["E3" ,32], - ["E#3",33], - ["Fb3",32], - ["F3" ,33], - ["F#3",34], - ["Gb3",34], - ["G3" ,35], - ["G#3",36], - ["Ab3",36], - ["A3" ,37], - ["A#3",38], - ["Bb3",38], - ["B3" ,39], - ["B#3",40], - ["Cb4",39], - ["C4" ,40], - ["C#4",41], - ["Db4",41], - ["D4" ,42], - ["D#4",43], - ["Eb4",43], - ["E4" ,44], - ["E#4",45], - ["Fb4",44], - ["F4" ,45], - ["F#4",46], - ["Gb4",46], - ["G4" ,47], - ["G#4",48], - ["Ab4",48], - ["A4" ,49], - ["A#4",50], - ["Bb4",50], - ["B4" ,51], - ["B#4",52], - ["Cb5",51], - ["C5" ,52], - ["C#5",53], - ["Db5",53], - ["D5" ,54], - ["D#5",55], - ["Eb5",55], - ["E5" ,56], - ["E#5",57], - ["Fb5",56], - ["F5" ,57], - ["F#5",58], - ["Gb5",58], - ["G5" ,59], - ["G#5",60], - ["Ab5",60], - ["A5" ,61], - ["A#5",62], - ["Bb5",62], - ["B5" ,63], - ["B#5",64], - ["Cb6",63], - ["C6" ,64], - ["C#6",65], - ["Db6",65], - ["D6" ,66], - ["D#6",67], - ["Eb6",67], - ["E6" ,68], - ["E#6",69], - ["Fb6",68], - ["F6" ,69], - ["F#6",70], - ["Gb6",70], - ["G6" ,71], - ["G#6",72], - ["Ab6",72], - ["A6" ,73], - ["A#6",74], - ["Bb6",74], - ["B6" ,75], - ["B#6",76], - ["Cb7",75], - ["C7" ,76], - ["C#7",77], - ["Db7",77], - ["D7" ,78], - ["D#7",79], - ["Eb7",79], - ["E7" ,80], - ["E#7",81], - ["Fb7",80], - ["F7" ,81], - ["F#7",82], - ["Gb7",82], - ["G7" ,83], - ["G#7",84], - ["Ab7",84], - ["A7" ,85], - ["A#7",86], - ["Bb7",86], - ["B7" ,87], - ["B#7",88], - ["Cb8",87], - ["C8", 88] - ]; - var noteMap = new Map(values); -} \ No newline at end of file + +var nat_values = [ + ["C/2" ,16], + ["D/2" ,18], + ["E/2" ,20], + ["F/2" ,21], + ["G/2" ,23], + ["A/2" ,25], + ["B/2" ,27], + ["C/3" ,28], + ["D/3" ,30], + ["E/3" ,32], + ["F/3" ,33], + ["G/3" ,35], + ["A/3" ,37], + ["B/3" ,39], + ["C/4" ,40], + ["D/4" ,42], + ["E/4" ,44], + ["F/4" ,45], + ["G/4" ,47], + ["A/4" ,49], + ["B/4" ,51], + ["C/5" ,52], + ["D/5" ,54], + ["E/5" ,56], + ["F/5" ,57], + ["G/5" ,59], + ["A/5" ,61], + ["B/5" ,63], + ["C/6" ,64] +]; +var nat_values_ex = [ + ["A/0" , 1], + ["B/0" , 3], + ["C/1" , 4], + ["D/1" , 6], + ["E/1" , 8], + ["F/1" , 9], + ["G/1" ,11], + ["A/1" ,13], + ["B/1" ,15], + ["D/6" ,66], + ["E/6" ,68], + ["F/6" ,69], + ["G/6" ,71], + ["A/6" ,73], + ["B/6" ,75], + ["C/7" ,76], + ["D/7" ,78], + ["E/7" ,80], + ["F/7" ,81], + ["G/7" ,83], + ["A/7" ,85], + ["B/7" ,87], + ["C/8", 88] +]; +var shp_values = [ + ["C#/2",17], + ["D#/2",19], + ["E#/2",21], + ["F#/2",22], + ["G#/2",24], + ["A#/2",26], + ["B#/2",28], + ["C#/3",29], + ["D#/3",31], + ["E#/3",33], + ["F#/3",34], + ["G#/3",36], + ["A#/3",38], + ["B#/3",40], + ["C#/4",41], + ["D#/4",43], + ["E#/4",45], + ["F#/4",46], + ["G#/4",48], + ["A#/4",50], + ["B#/4",52], + ["C#/5",53], + ["D#/5",55], + ["E#/5",57], + ["F#/5",58], + ["G#/5",60], + ["A#/5",62], + ["B#/5",64], + ["C#/6",65] +]; +var shp_values_ex = [ + ["A#/0", 2], + ["B#/0", 4], + ["C#/1", 5], + ["D#/1", 7], + ["E#/1", 9], + ["F#/1",10], + ["G#/1",12], + ["A#/1",14], + ["B#/1",16], + ["D#/6",67], + ["E#/6",69], + ["F#/6",70], + ["G#/6",72], + ["A#/6",74], + ["B#/6",76], + ["C#/7",77], + ["D#/7",79], + ["E#/7",81], + ["F#/7",82], + ["G#/7",84], + ["A#/7",86], + ["B#/7",88] +]; +var flt_values = [ + ["Cb/2",15], + ["Db/2",17], + ["Eb/2",19], + ["Fb/2",20], + ["Gb/2",22], + ["Ab/2",24], + ["Bb/2",26], + ["Cb/3",27], + ["Db/3",29], + ["Eb/3",31], + ["Fb/3",32], + ["Gb/3",34], + ["Ab/3",36], + ["Bb/3",38], + ["Cb/4",39], + ["Db/4",41], + ["Eb/4",43], + ["Fb/4",44], + ["Gb/4",46], + ["Ab/4",48], + ["Bb/4",50], + ["Cb/5",51], + ["Db/5",53], + ["Eb/5",55], + ["Fb/5",56], + ["Gb/5",58], + ["Ab/5",60], + ["Bb/5",62], + ["Cb/6",63] +]; +flt_values_ex = [ + ["Bb/0", 2], + ["Cb/1", 3], + ["Db/1", 5], + ["Eb/1", 7], + ["Fb/1", 8], + ["Gb/1",10], + ["Ab/1",12], + ["Bb/1",14], + ["Db/6",65], + ["Eb/6",67], + ["Fb/6",68], + ["Gb/6",70], + ["Ab/6",72], + ["Bb/6",74], + ["Cb/7",75], + ["Db/7",77], + ["Eb/7",79], + ["Fb/7",80], + ["Gb/7",82], + ["Ab/7",84], + ["Bb/7",86], + ["Cb/8",87] +] \ No newline at end of file From 637edb030213446b3aa8b0677892da2087d1cffe Mon Sep 17 00:00:00 2001 From: krippix Date: Mon, 5 Jun 2023 20:09:37 +0200 Subject: [PATCH 5/6] implemented single note practice --- html/assets/css/base.css | 7 +- html/assets/scripts/midi.js | 8 +- html/assets/scripts/practice.js | 167 ++++++++++++++++++++++++++++++++ html/assets/scripts/render.js | 112 +++++++++++---------- html/index.html | 6 ++ 5 files changed, 243 insertions(+), 57 deletions(-) create mode 100644 html/assets/scripts/practice.js diff --git a/html/assets/css/base.css b/html/assets/css/base.css index b8d9ed7..09a6fee 100644 --- a/html/assets/css/base.css +++ b/html/assets/css/base.css @@ -25,4 +25,9 @@ h1 { #note-render { display: inline-block; margin: 0 auto; -} \ No newline at end of file +} +.checkboxes { + font-family: 'arial'; + margin-top: 10px; + margin-bottom: 10px; +} diff --git a/html/assets/scripts/midi.js b/html/assets/scripts/midi.js index abb3628..63d2097 100644 --- a/html/assets/scripts/midi.js +++ b/html/assets/scripts/midi.js @@ -10,7 +10,6 @@ window.addEventListener("load", function() { // press or unpress keyboard button function setKeyState(note, state) { let key = svgDoc.getElementById(String(note)); - console.log(key); if (state) { key.style.fill = "red"; } else if (keysB.includes(note)) { @@ -50,13 +49,10 @@ function getMIDIMessage(midiMessage) { console.log(note); if (velocity == 0) { setKeyState(note, false); + task.changeKeyState(note,false); } else { setKeyState(note, true); + task.changeKeyState(note,true); } } - - if (command == 144 && velocity != 0) { - - } else if (command == 144 && velocity == 0) - console.log(command,note,velocity); } \ No newline at end of file diff --git a/html/assets/scripts/practice.js b/html/assets/scripts/practice.js new file mode 100644 index 0000000..640b441 --- /dev/null +++ b/html/assets/scripts/practice.js @@ -0,0 +1,167 @@ +class Task { + /** + * + * @param {Array} keysig list of keysignatures to use (int 0-11) where 0 = C || a, 1 = G || e ... + * @param {boolean} natural enable/disable naturals + * @param {boolean} sharp enable/disable sharps + * @param {boolean} flat enable/disable flats + * @param {boolean} chord enable/disable chords (it's either chords or single notes) + * @param {boolean} extend enable/disable notes above C6 and below C2 + */ + constructor() { + this.render = new Notation(); + this.retrieve_settings(); + + this.solution = []; // keys to solve current problem + this.activeKeys = []; // currently pressed keys + this.active = true; // true while task can be solved + this.success = false; + + this.generateTask(); + } + + generateTask() { + this.retrieve_settings(); + if (this.chord) { + this.generateChord(); + } else { + this.generateNote(); + } + } + + generateNote() { + let pickFrom = []; + + // TODO: choose signature + + // TODO: maybe weight note likelyhood + + if (this.natural) { + pickFrom.push(...nat_values); + if (this.extend) { + pickFrom.push(...nat_values_ex); + } + } + if (this.sharp) { + pickFrom.push(...shp_values); + if (this.extend) { + pickFrom.push(...shp_values_ex); + } + } + if (this.flat) { + pickFrom.push(...flt_values); + if (this.extend) { + pickFrom.push(...flt_values_ex); + } + } + + // pick from generated pool + var randomResult = Math.floor(Math.random() * pickFrom.length); + let chosenNote = pickFrom[randomResult]; + this.solution = [chosenNote[1]]; + + let clef = ""; + + // choose clef + if (chosenNote[1] == 40) { + Math.random() < 0.5 ? clef = "bass" : clef = "treble"; + } else if (chosenNote[1] < 40) { + clef = "bass"; + } else { + clef = "treble"; + } + + this.solution = [chosenNote[1]]; + this.render.set_notes([new Noteset(clef, [chosenNote[0]])]); + } + + generateChord() { + + } + + on_keyStateChange() { + if (this.active) { + this.checkActiveKeys(); + } else { + console.log("Can't continue, still keys pressed."); + return; + } + if (this.success) { + this.generateTask(); + } else if (this.activeKeys.length == 0) { + this.active = true; + } + } + + /** + * Keeps Track of pressed keys + * @param {number} key + * @param {boolean} state + */ + changeKeyState(key, state) { + if (state) { + // start new timer + if (this.activeKeys.length == 0) { + this.active = true; + this.starttime = Date.now(); + } + this.activeKeys.push(key); + } else { + let index = this.activeKeys.indexOf(key); + if (index != -1) { + this.activeKeys.splice(index, 1); + } + } + this.on_keyStateChange() + } + + /** + * Checks if activeKeys fulfill current task + */ + checkActiveKeys() { + this.activeKeys.forEach( key => { + if (!(key in this.solution)) { + this.result_fail(); + return; + } + }); + if (this.activeKeys.length == this.solution.length) { + if (Date.now() - this.starttime <= 500) { + this.result_success(); + return; + } else { + this.result_fail(); + return; + } + } + } + + result_fail() { + console.log("Wrong!"); + this.active = false; + } + + result_success() { + console.log("Correct!"); + this.solution = []; + this.active = false; + this.success = true; + } + + retrieve_settings() { + let cbx_naturals = document.getElementById("naturals"); + let cbx_sharps = document.getElementById("sharps"); + let cbx_flats = document.getElementById("flats"); + let cbx_extended = document.getElementById("extended"); + + this.keysig = [0]; + this.natural = cbx_naturals.checked; + this.sharp = cbx_sharps.checked; + this.flat = cbx_flats.checked; + this.chord = false; + this.extend = cbx_extended.checked; + } +} + +// Starts the task loop +var task = new Task(); diff --git a/html/assets/scripts/render.js b/html/assets/scripts/render.js index 48e793e..21fc3fa 100644 --- a/html/assets/scripts/render.js +++ b/html/assets/scripts/render.js @@ -2,14 +2,44 @@ const { Renderer, Stave, StaveNote, Voice, Formatter, TickContext } = Vex.Flow; class Notation { constructor() { - // Create an SVG renderer and attach it to the DIV element - this.div = document.getElementById("note-render"); - this.renderer = new Renderer(this.div, Renderer.Backends.SVG); + // init notelist + this.notesTop = []; + this.notesBottom = []; - // Configure the rendering context. - this.renderer.resize(321, 500); - this.ctx = this.renderer.getContext(); + // create voice + this.voiceTop = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); + this.voiceBottom = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); + this.draw(); + } + /** + * Takes list of notes ex: [{clef: "bass/treble", keys: ["c/4,"e/4"], duration: "w"},{clef: "bass/treble", keys: ["c/4,"e/4"], duration: "w"}] + * @param {Array[Noteset]} notes + */ + set_notes(to_set) { + // create voice + this.voiceTop = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); + this.voiceBottom = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); + //this.voiceTop.tickables = []; + //this.voiceBottom.tickables = []; + + this.notesTop = []; + this.notesBottom = []; + + to_set.forEach( note => { + if (note.clef == "treble") { + this.notesTop.push(note.staveNote); + } else { + this.notesBottom.push(note.staveNote); + } + }); + + this.voiceTop.addTickables(this.notesTop); + this.voiceBottom.addTickables(this.notesBottom); + this.draw(); + } + + draw() { // Create the staves this.topStaff = new Vex.Flow.Stave(20, 100, 300); this.bottomStaff = new Vex.Flow.Stave(20, 200, 300); @@ -22,49 +52,17 @@ class Notation { this.brace = new Vex.Flow.StaveConnector(this.topStaff, this.bottomStaff).setType(3); // 3 = brace this.lineLeft = new Vex.Flow.StaveConnector(this.topStaff, this.bottomStaff).setType(1); this.lineRight = new Vex.Flow.StaveConnector(this.topStaff, this.bottomStaff).setType(6); + + // Create an SVG renderer and attach it to the DIV element + this.div = document.getElementById("note-render"); + this.div.innerHTML = ""; - // create voice - this.voiceTop = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); - this.voiceBottom = new Vex.Flow.Voice({num_beats: 4, beat_value: 4, resolution: Vex.Flow.RESOLUTION}); + this.renderer = new Renderer(this.div, Renderer.Backends.SVG); - // init notelist - this.notesTop = []; - this.notesBottom = []; + // Configure the rendering context. + this.renderer.resize(321, 500); + this.ctx = this.renderer.getContext(); - this.draw(); - } - /** - * Takes list of notes ex: [{clef: "bass/treble", keys: ["c/4,"e/4"], duration: "w"},{clef: "bass/treble", keys: ["c/4,"e/4"], duration: "w"}] - * @param {*} notes - */ - set_notes(to_set) { - this.voiceTop.tickables = []; - this.voiceBottom.tickables = []; - - this.notesTop = []; - this.notesBottom = []; - - to_set.forEach( note => { - if (note.clef == "treble") { - this.notesTop.push(new Vex.Flow.StaveNote(note)); - } else { - this.notesBottom.push(new Vex.Flow.StaveNote(note)); - } - }); - - for (let note of this.notesTop) { - note.x_shift = 100; - } - for (let note of this.notesBottom) { - note.x_shift = 100; - } - - this.voiceTop.addTickables(this.notesTop); - this.voiceBottom.addTickables(this.notesBottom); - this.draw(); - } - - draw() { // draw background this.topStaff.setContext(this.ctx).draw(); this.bottomStaff.setContext(this.ctx).draw(); @@ -86,12 +84,26 @@ class Notation { class Noteset { // syntax for notes: https://github.com/0xfe/vexflow/blob/master/src/tables.ts + /** + * + * @param {String} clef bass or treble + * @param {Array[String]} keys array of string + */ constructor(clef, keys) { this.clef = clef; this.keys = keys; - this.duration = "w"; + this.staveNote = new StaveNote({clef: clef, keys: keys, duration: "w"}); + this.addAccidentals(); + } + addAccidentals() { + for (var i=0; i < this.keys.length; i++) { + if (this.keys[i].includes("b")) { + this.staveNote.addModifier(new Vex.Flow.Accidental("b"),i) + } + if (this.keys[i].includes("#")) { + this.staveNote.addModifier(new Vex.Flow.Accidental("#"),i); + } + } + //this.staveNote.setXShift(80); } } - -notator = new Notation(); -notator.set_notes([new Noteset("treble",["c/5","c/6"]),new Noteset("bass",["c/3"])]); diff --git a/html/index.html b/html/index.html index 50b46bb..27b574f 100644 --- a/html/index.html +++ b/html/index.html @@ -7,6 +7,12 @@
+
+ + + + +
From 5ad46c8c7ffd820dae882e28a4abdb3fb683b275 Mon Sep 17 00:00:00 2001 From: krippix Date: Mon, 5 Jun 2023 20:52:19 +0200 Subject: [PATCH 6/6] fixed check if note was played correctly --- html/assets/scripts/practice.js | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/html/assets/scripts/practice.js b/html/assets/scripts/practice.js index 640b441..94ab50f 100644 --- a/html/assets/scripts/practice.js +++ b/html/assets/scripts/practice.js @@ -22,6 +22,7 @@ class Task { generateTask() { this.retrieve_settings(); + this.success = false; if (this.chord) { this.generateChord(); } else { @@ -119,14 +120,23 @@ class Task { * Checks if activeKeys fulfill current task */ checkActiveKeys() { + let failed = false; this.activeKeys.forEach( key => { - if (!(key in this.solution)) { + if (!(this.solution.includes(key))) { this.result_fail(); - return; + console.log("failed key: "+key); + failed = true; } }); + if (failed) { + console.log("should have pressed: "+this.solution); + console.log("actually pressed: "+this.activeKeys); + return; + } if (this.activeKeys.length == this.solution.length) { if (Date.now() - this.starttime <= 500) { + console.log(this.solution); + console.log(this.activeKeys); this.result_success(); return; } else {