diff --git a/static/extensions/orengdog/Music to numbers.js b/static/extensions/orengdog/Music to numbers.js new file mode 100644 index 00000000..98d79eb3 --- /dev/null +++ b/static/extensions/orengdog/Music to numbers.js @@ -0,0 +1,323 @@ +(async function(Scratch) { + const variables = { + volume: 1.0, + pitchShift: 0, + leftRightValue: 0, + selectedEffect: "none", + isPlaying: false, + instrument: "sine", + instrumentID: 0, + instrumentName: "without instrument", + pauseDuration: 0.5 + }; + + class MusicGenerator { + constructor() { + this.audioContext = new (window.AudioContext || window.webkitAudioContext)(); + this.digitToFrequency = { + "0": 261.63, // C4 + "1": 293.66, // D4 + "2": 329.63, // E4 + "3": 349.23, // F4 + "4": 392.00, // G4 + "5": 440.00, // A4 + "6": 493.88, // B4 + "7": 523.25, // C5 + "8": 587.33, // D5 + "9": 659.25 // E5 + }; + this.activeOscillators = []; + } + + getFrequencyWithEffect(frequency) { + if (variables.selectedEffect === "pitch") { + return frequency * Math.pow(2, variables.pitchShift / 12); + } + return frequency; + } + + playContinuousSequence(sequence, noteDuration = 0.3) { + let currentTime = this.audioContext.currentTime; + variables.isPlaying = true; + + sequence.split('').forEach((char) => { + if (this.digitToFrequency[char] !== undefined) { + const frequency = this.getFrequencyWithEffect(this.digitToFrequency[char]); + this.playNoteAtTime(frequency, currentTime, noteDuration); + currentTime += noteDuration; + } else if (char === ' ' || char === '_' || char === '.') { + currentTime += variables.pauseDuration; + } + }); + + setTimeout(() => { variables.isPlaying = false; }, (currentTime - this.audioContext.currentTime) * 1000); + } + + playNoteAtTime(frequency, startTime, duration) { + const oscillator = this.audioContext.createOscillator(); + oscillator.type = variables.instrument; + oscillator.frequency.setValueAtTime(frequency, startTime); + + const gainNode = this.audioContext.createGain(); + gainNode.gain.setValueAtTime(variables.volume * 0.1, startTime); + + let panNode; + if (variables.selectedEffect === "left-right") { + panNode = this.audioContext.createStereoPanner(); + panNode.pan.setValueAtTime(variables.leftRightValue, startTime); + oscillator.connect(panNode); + panNode.connect(gainNode); + } else { + oscillator.connect(gainNode); + } + + gainNode.connect(this.audioContext.destination); + oscillator.start(startTime); + oscillator.stop(startTime + duration); + this.activeOscillators.push(oscillator); + } + + stopAllMusic() { + this.activeOscillators.forEach(oscillator => oscillator.stop()); + this.activeOscillators = []; + variables.isPlaying = false; + } + } + + const musicGen = new MusicGenerator(); + + const extension = { + getInfo() { + return { + blockIconURI: "", + id: "musicDigitGenerator", + name: "Music to Numbers", + color1: "#0088ff", + color2: "#0063ba", + blocks: [ + { + opcode: 'playDigitSequence', + blockType: Scratch.BlockType.COMMAND, + text: 'play note sequence for [sequence]', + arguments: { + sequence: { + type: Scratch.ArgumentType.STRING, + defaultValue: '1 2 3 . 4 _ 5' + } + } + }, + { + opcode: 'setVolume', + blockType: Scratch.BlockType.COMMAND, + text: 'set volume to [volume] (1-100)', + arguments: { + volume: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100 + } + } + }, + { + opcode: 'setEffect', + blockType: Scratch.BlockType.COMMAND, + text: 'set [effectType] effect value to [value]', + arguments: { + effectType: { + type: Scratch.ArgumentType.STRING, + menu: 'effectTypes', + defaultValue: 'pitch' + }, + value: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0 + } + } + }, + { + opcode: 'changeEffect', + blockType: Scratch.BlockType.COMMAND, + text: 'change [effectType] effect by [amount]', + arguments: { + effectType: { + type: Scratch.ArgumentType.STRING, + menu: 'effectTypes', + defaultValue: 'pitch' + }, + amount: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1 + } + } + }, + { + opcode: 'getCurrentEffect', + blockType: Scratch.BlockType.REPORTER, + text: 'current [effectType] effect', + arguments: { + effectType: { + type: Scratch.ArgumentType.STRING, + menu: 'effectTypes', + defaultValue: 'pitch' + } + } + }, + { + opcode: 'setPauseDuration', + blockType: Scratch.BlockType.COMMAND, + text: 'set pause to [duration] seconds', + arguments: { + duration: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0.5 + } + } + }, + { + opcode: 'changePauseDuration', + blockType: Scratch.BlockType.COMMAND, + text: 'change pause by [amount] seconds', + arguments: { + amount: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0.1 + } + } + }, + { + opcode: 'getCurrentPause', + blockType: Scratch.BlockType.REPORTER, + text: 'current pause duration', + }, + { + opcode: 'setInstrument', + blockType: Scratch.BlockType.COMMAND, + text: 'set instrument to [instrument]', + arguments: { + instrument: { + type: Scratch.ArgumentType.STRING, + menu: 'instruments', + defaultValue: 'without instrument' + } + } + }, + { + opcode: 'getCurrentInstrument', + blockType: Scratch.BlockType.REPORTER, + text: 'current instrument [returnType]', + arguments: { + returnType: { + type: Scratch.ArgumentType.STRING, + menu: 'returnTypes', + defaultValue: 'name' + } + } + }, + { + opcode: 'isMusicPlaying', + blockType: Scratch.BlockType.BOOLEAN, + text: 'music playing now?' + }, + { + opcode: 'stopMusic', + blockType: Scratch.BlockType.COMMAND, + text: 'stop music' + } + ], + menus: { + effectTypes: { + acceptReporters: true, + items: ['pitch', 'left-right'] + }, + instruments: { + acceptReporters: true, + items: ['without instrument', 'piano', 'organ'] + }, + returnTypes: { + acceptReporters: true, + items: ['name', 'number'] + } + } + }; + }, + + playDigitSequence(args) { + const sequence = args.sequence; + musicGen.playContinuousSequence(sequence); + }, + + setVolume(args) { + variables.volume = Math.max(0, Math.min(1, args.volume / 100)); + }, + + setEffect(args) { + variables.selectedEffect = args.effectType; + if (args.effectType === "pitch") { + variables.pitchShift = args.value; + } else if (args.effectType === "left-right") { + variables.leftRightValue = Math.max(-1, Math.min(1, args.value)); + } + }, + + changeEffect(args) { + if (args.effectType === "pitch") { + variables.pitchShift += args.amount; + } else if (args.effectType === "left-right") { + variables.leftRightValue = Math.max(-1, Math.min(1, variables.leftRightValue + args.amount)); + } + }, + + getCurrentEffect(args) { + if (args.effectType === "pitch") { + return variables.pitchShift; + } else if (args.effectType === "left-right") { + return variables.leftRightValue; + } + return 0; + }, + + setPauseDuration(args) { + variables.pauseDuration = Math.max(0, args.duration); + }, + + changePauseDuration(args) { + variables.pauseDuration = Math.max(0, variables.pauseDuration + args.amount); + }, + + getCurrentPause() { + return variables.pauseDuration; + }, + + getCurrentInstrument(args) { + return args.returnType === "name" ? variables.instrumentName : variables.instrumentID; + }, + + setInstrument(args) { + switch (args.instrument) { + case 'piano': + variables.instrument = 'triangle'; + variables.instrumentID = 1; + variables.instrumentName = 'piano'; + break; + case 'organ': + variables.instrument = 'square'; + variables.instrumentID = 2; + variables.instrumentName = 'organ'; + break; + default: + variables.instrument = 'sine'; + variables.instrumentID = 0; + variables.instrumentName = 'without instrument'; + } + }, + + isMusicPlaying() { + return variables.isPlaying; + }, + + stopMusic() { + musicGen.stopAllMusic(); + } + }; + + Scratch.extensions.register(extension); +})(Scratch); diff --git a/static/images/orengdog/Music to numbers.png b/static/images/orengdog/Music to numbers.png new file mode 100644 index 00000000..a89661cf Binary files /dev/null and b/static/images/orengdog/Music to numbers.png differ