mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2024-12-24 23:12:24 -05:00
Add support for "midiInstrument:" block (#1329)
* Add midiInstrument: block * Don't show midiInstrument block in palette * Lint * More lint
This commit is contained in:
parent
fa0af58c41
commit
744b01174b
2 changed files with 125 additions and 1 deletions
|
@ -535,6 +535,79 @@ class Scratch3MusicBlocks {
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An array that is a mapping from MIDI instrument numbers to Scratch instrument numbers.
|
||||||
|
* @type {number[]} an array of Scratch instrument numbers.
|
||||||
|
*/
|
||||||
|
get MIDI_INSTRUMENTS () {
|
||||||
|
return [
|
||||||
|
// Acoustic Grand, Bright Acoustic, Electric Grand, Honky-Tonk
|
||||||
|
1, 1, 1, 1,
|
||||||
|
// Electric Piano 1, Electric Piano 2, Harpsichord, Clavinet
|
||||||
|
2, 2, 4, 4,
|
||||||
|
// Celesta, Glockenspiel, Music Box, Vibraphone
|
||||||
|
17, 17, 17, 16,
|
||||||
|
// Marimba, Xylophone, Tubular Bells, Dulcimer
|
||||||
|
19, 16, 17, 17,
|
||||||
|
// Drawbar Organ, Percussive Organ, Rock Organ, Church Organ
|
||||||
|
3, 3, 3, 3,
|
||||||
|
// Reed Organ, Accordion, Harmonica, Tango Accordion
|
||||||
|
3, 3, 3, 3,
|
||||||
|
// Nylon String Guitar, Steel String Guitar, Electric Jazz Guitar, Electric Clean Guitar
|
||||||
|
4, 4, 5, 5,
|
||||||
|
// Electric Muted Guitar, Overdriven Guitar,Distortion Guitar, Guitar Harmonics
|
||||||
|
5, 5, 5, 5,
|
||||||
|
// Acoustic Bass, Electric Bass (finger), Electric Bass (pick), Fretless Bass
|
||||||
|
6, 6, 6, 6,
|
||||||
|
// Slap Bass 1, Slap Bass 2, Synth Bass 1, Synth Bass 2
|
||||||
|
6, 6, 6, 6,
|
||||||
|
// Violin, Viola, Cello, Contrabass
|
||||||
|
8, 8, 8, 8,
|
||||||
|
// Tremolo Strings, Pizzicato Strings, Orchestral Strings, Timpani
|
||||||
|
8, 7, 8, 19,
|
||||||
|
// String Ensemble 1, String Ensemble 2, SynthStrings 1, SynthStrings 2
|
||||||
|
8, 8, 8, 8,
|
||||||
|
// Choir Aahs, Voice Oohs, Synth Voice, Orchestra Hit
|
||||||
|
15, 15, 15, 19,
|
||||||
|
// Trumpet, Trombone, Tuba, Muted Trumpet
|
||||||
|
9, 9, 9, 9,
|
||||||
|
// French Horn, Brass Section, SynthBrass 1, SynthBrass 2
|
||||||
|
9, 9, 9, 9,
|
||||||
|
// Soprano Sax, Alto Sax, Tenor Sax, Baritone Sax
|
||||||
|
11, 11, 11, 11,
|
||||||
|
// Oboe, English Horn, Bassoon, Clarinet
|
||||||
|
14, 14, 14, 10,
|
||||||
|
// Piccolo, Flute, Recorder, Pan Flute
|
||||||
|
12, 12, 13, 13,
|
||||||
|
// Blown Bottle, Shakuhachi, Whistle, Ocarina
|
||||||
|
13, 13, 12, 12,
|
||||||
|
// Lead 1 (square), Lead 2 (sawtooth), Lead 3 (calliope), Lead 4 (chiff)
|
||||||
|
20, 20, 20, 20,
|
||||||
|
// Lead 5 (charang), Lead 6 (voice), Lead 7 (fifths), Lead 8 (bass+lead)
|
||||||
|
20, 20, 20, 20,
|
||||||
|
// Pad 1 (new age), Pad 2 (warm), Pad 3 (polysynth), Pad 4 (choir)
|
||||||
|
21, 21, 21, 21,
|
||||||
|
// Pad 5 (bowed), Pad 6 (metallic), Pad 7 (halo), Pad 8 (sweep)
|
||||||
|
21, 21, 21, 21,
|
||||||
|
// FX 1 (rain), FX 2 (soundtrack), FX 3 (crystal), FX 4 (atmosphere)
|
||||||
|
21, 21, 21, 21,
|
||||||
|
// FX 5 (brightness), FX 6 (goblins), FX 7 (echoes), FX 8 (sci-fi)
|
||||||
|
21, 21, 21, 21,
|
||||||
|
// Sitar, Banjo, Shamisen, Koto
|
||||||
|
4, 4, 4, 4,
|
||||||
|
// Kalimba, Bagpipe, Fiddle, Shanai
|
||||||
|
17, 14, 8, 10,
|
||||||
|
// Tinkle Bell, Agogo, Steel Drums, Woodblock
|
||||||
|
17, 17, 18, 19,
|
||||||
|
// Taiko Drum, Melodic Tom, Synth Drum, Reverse Cymbal
|
||||||
|
1, 1, 1, 1,
|
||||||
|
// Guitar Fret Noise, Breath Noise, Seashore, Bird Tweet
|
||||||
|
21, 21, 21, 21,
|
||||||
|
// Telephone Ring, Helicopter, Applause, Gunshot
|
||||||
|
21, 21, 21, 21
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The key to load & store a target's music-related state.
|
* The key to load & store a target's music-related state.
|
||||||
* @type {string}
|
* @type {string}
|
||||||
|
@ -699,6 +772,22 @@ class Scratch3MusicBlocks {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
opcode: 'midiSetInstrument',
|
||||||
|
blockType: BlockType.COMMAND,
|
||||||
|
text: formatMessage({
|
||||||
|
id: 'music.midiSetInstrument',
|
||||||
|
default: 'set instrument to [INSTRUMENT]',
|
||||||
|
description: 'set the instrument for notes played according to a mapping of MIDI codes'
|
||||||
|
}),
|
||||||
|
arguments: {
|
||||||
|
INSTRUMENT: {
|
||||||
|
type: ArgumentType.NUMBER,
|
||||||
|
defaultValue: 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
hideFromPalette: true
|
||||||
|
},
|
||||||
{
|
{
|
||||||
opcode: 'setTempo',
|
opcode: 'setTempo',
|
||||||
blockType: BlockType.COMMAND,
|
blockType: BlockType.COMMAND,
|
||||||
|
@ -1029,10 +1118,35 @@ class Scratch3MusicBlocks {
|
||||||
* @property {int} INSTRUMENT - the number of the instrument to select.
|
* @property {int} INSTRUMENT - the number of the instrument to select.
|
||||||
*/
|
*/
|
||||||
setInstrument (args, util) {
|
setInstrument (args, util) {
|
||||||
|
this._setInstrument(args.INSTRUMENT, util, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Select an instrument for playing notes according to a mapping of MIDI codes to Scratch instrument numbers.
|
||||||
|
* This block is implemented for compatibility with old Scratch projects that use the 'midiInstrument:' block.
|
||||||
|
* @param {object} args - the block arguments.
|
||||||
|
* @param {object} util - utility object provided by the runtime.
|
||||||
|
* @property {int} INSTRUMENT - the MIDI number of the instrument to select.
|
||||||
|
*/
|
||||||
|
midiSetInstrument (args, util) {
|
||||||
|
this._setInstrument(args.INSTRUMENT, util, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Internal code to select an instrument for playing notes. If mapMidi is true, set the instrument according to
|
||||||
|
* the MIDI to Scratch instrument mapping.
|
||||||
|
* @param {number} instNum - the instrument number.
|
||||||
|
* @param {object} util - utility object provided by the runtime.
|
||||||
|
* @param {boolean} mapMidi - whether or not instNum is a MIDI instrument number.
|
||||||
|
*/
|
||||||
|
_setInstrument (instNum, util, mapMidi) {
|
||||||
const musicState = this._getMusicState(util.target);
|
const musicState = this._getMusicState(util.target);
|
||||||
let instNum = Cast.toNumber(args.INSTRUMENT);
|
instNum = Cast.toNumber(instNum);
|
||||||
instNum = Math.round(instNum);
|
instNum = Math.round(instNum);
|
||||||
instNum -= 1; // instruments are one-indexed
|
instNum -= 1; // instruments are one-indexed
|
||||||
|
if (mapMidi) {
|
||||||
|
instNum = (this.MIDI_INSTRUMENTS[instNum] || 0) - 1;
|
||||||
|
}
|
||||||
instNum = MathUtil.wrapClamp(instNum, 0, this.INSTRUMENT_INFO.length - 1);
|
instNum = MathUtil.wrapClamp(instNum, 0, this.INSTRUMENT_INFO.length - 1);
|
||||||
musicState.currentInstrument = instNum;
|
musicState.currentInstrument = instNum;
|
||||||
}
|
}
|
||||||
|
|
|
@ -538,6 +538,16 @@ const specMap = {
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
'midiInstrument:': {
|
||||||
|
opcode: 'music_midiSetInstrument',
|
||||||
|
argMap: [
|
||||||
|
{
|
||||||
|
type: 'input',
|
||||||
|
inputOp: 'math_number',
|
||||||
|
inputName: 'INSTRUMENT'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
'changeVolumeBy:': {
|
'changeVolumeBy:': {
|
||||||
opcode: 'sound_changevolumeby',
|
opcode: 'sound_changevolumeby',
|
||||||
argMap: [
|
argMap: [
|
||||||
|
|
Loading…
Reference in a new issue