mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-01-11 23:30:09 -05:00
0162bfc71e
There are parts of the extension registration process which rely on the `info` field being non-null, even for block separators. At one point during development I broke that, so here's a test to hopefully prevent someone else from getting bitten by the same thing.
185 lines
6.7 KiB
JavaScript
185 lines
6.7 KiB
JavaScript
const test = require('tap').test;
|
|
|
|
const ArgumentType = require('../../src/extension-support/argument-type');
|
|
const BlockType = require('../../src/extension-support/block-type');
|
|
const Runtime = require('../../src/engine/runtime');
|
|
const ScratchBlocksConstants = require('../../src/engine/scratch-blocks-constants');
|
|
|
|
/**
|
|
* @type {ExtensionMetadata}
|
|
*/
|
|
const testExtensionInfo = {
|
|
id: 'test',
|
|
name: 'fake test extension',
|
|
blocks: [
|
|
{
|
|
func: 'MAKE_A_VARIABLE',
|
|
blockType: BlockType.BUTTON,
|
|
text: 'this is a button'
|
|
},
|
|
{
|
|
opcode: 'reporter',
|
|
blockType: BlockType.REPORTER,
|
|
text: 'simple text'
|
|
},
|
|
'---', // separator between groups of blocks in an extension
|
|
{
|
|
opcode: 'command',
|
|
blockType: BlockType.COMMAND,
|
|
text: 'text with [ARG]',
|
|
arguments: {
|
|
ARG: {
|
|
type: ArgumentType.STRING
|
|
}
|
|
}
|
|
},
|
|
{
|
|
opcode: 'ifElse',
|
|
blockType: BlockType.CONDITIONAL,
|
|
branchCount: 2,
|
|
text: [
|
|
'test if [THING] is spiffy and if so then',
|
|
'or elsewise'
|
|
],
|
|
arguments: {
|
|
THING: {
|
|
type: ArgumentType.BOOLEAN
|
|
}
|
|
}
|
|
},
|
|
{
|
|
opcode: 'loop',
|
|
blockType: BlockType.LOOP, // implied branchCount of 1 unless otherwise stated
|
|
isTerminal: true,
|
|
text: [
|
|
'loopty [MANY] loops'
|
|
],
|
|
arguments: {
|
|
MANY: {
|
|
type: ArgumentType.NUMBER
|
|
}
|
|
}
|
|
}
|
|
]
|
|
};
|
|
|
|
const testButton = function (t, button) {
|
|
t.same(button.json, null); // should be null or undefined
|
|
t.equal(button.xml, '<button text="this is a button" callbackKey="MAKE_A_VARIABLE"></button>');
|
|
};
|
|
|
|
const testReporter = function (t, reporter) {
|
|
t.equal(reporter.json.type, 'test_reporter');
|
|
t.equal(reporter.json.outputShape, ScratchBlocksConstants.OUTPUT_SHAPE_ROUND);
|
|
t.equal(reporter.json.output, 'String');
|
|
t.notOk(reporter.json.hasOwnProperty('previousStatement'));
|
|
t.notOk(reporter.json.hasOwnProperty('nextStatement'));
|
|
t.equal(reporter.json.message0, 'simple text');
|
|
t.notOk(reporter.json.hasOwnProperty('message1'));
|
|
t.notOk(reporter.json.hasOwnProperty('args0'));
|
|
t.notOk(reporter.json.hasOwnProperty('args1'));
|
|
t.equal(reporter.xml, '<block type="test_reporter"></block>');
|
|
};
|
|
|
|
const testSeparator = function (t, separator) {
|
|
t.same(separator.json, null); // should be null or undefined
|
|
t.equal(separator.xml, '<sep gap="36"/>');
|
|
};
|
|
|
|
const testCommand = function (t, command) {
|
|
t.equal(command.json.type, 'test_command');
|
|
t.equal(command.json.outputShape, ScratchBlocksConstants.OUTPUT_SHAPE_SQUARE);
|
|
t.assert(command.json.hasOwnProperty('previousStatement'));
|
|
t.assert(command.json.hasOwnProperty('nextStatement'));
|
|
t.equal(command.json.message0, 'text with %1');
|
|
t.notOk(command.json.hasOwnProperty('message1'));
|
|
t.strictSame(command.json.args0[0], {
|
|
type: 'input_value',
|
|
name: 'ARG'
|
|
});
|
|
t.notOk(command.json.hasOwnProperty('args1'));
|
|
t.equal(command.xml,
|
|
'<block type="test_command"><value name="ARG"><shadow type="text"><field name="TEXT">' +
|
|
'</field></shadow></value></block>');
|
|
};
|
|
|
|
const testConditional = function (t, conditional) {
|
|
t.equal(conditional.json.type, 'test_ifElse');
|
|
t.equal(conditional.json.outputShape, ScratchBlocksConstants.OUTPUT_SHAPE_SQUARE);
|
|
t.ok(conditional.json.hasOwnProperty('previousStatement'));
|
|
t.ok(conditional.json.hasOwnProperty('nextStatement'));
|
|
t.equal(conditional.json.message0, 'test if %1 is spiffy and if so then');
|
|
t.equal(conditional.json.message1, '%1'); // placeholder for substack #1
|
|
t.equal(conditional.json.message2, 'or elsewise');
|
|
t.equal(conditional.json.message3, '%1'); // placeholder for substack #2
|
|
t.notOk(conditional.json.hasOwnProperty('message4'));
|
|
t.strictSame(conditional.json.args0[0], {
|
|
type: 'input_value',
|
|
name: 'THING',
|
|
check: 'Boolean'
|
|
});
|
|
t.strictSame(conditional.json.args1[0], {
|
|
type: 'input_statement',
|
|
name: 'SUBSTACK'
|
|
});
|
|
t.notOk(conditional.json.hasOwnProperty(conditional.json.args2));
|
|
t.strictSame(conditional.json.args3[0], {
|
|
type: 'input_statement',
|
|
name: 'SUBSTACK2'
|
|
});
|
|
t.notOk(conditional.json.hasOwnProperty('args4'));
|
|
t.equal(conditional.xml, '<block type="test_ifElse"><value name="THING"></value></block>');
|
|
};
|
|
|
|
const testLoop = function (t, loop) {
|
|
t.equal(loop.json.type, 'test_loop');
|
|
t.equal(loop.json.outputShape, ScratchBlocksConstants.OUTPUT_SHAPE_SQUARE);
|
|
t.ok(loop.json.hasOwnProperty('previousStatement'));
|
|
t.notOk(loop.json.hasOwnProperty('nextStatement')); // isTerminal is set on this block
|
|
t.equal(loop.json.message0, 'loopty %1 loops');
|
|
t.equal(loop.json.message1, '%1'); // placeholder for substack
|
|
t.equal(loop.json.message2, '%1'); // placeholder for loop arrow
|
|
t.notOk(loop.json.hasOwnProperty('message3'));
|
|
t.strictSame(loop.json.args0[0], {
|
|
type: 'input_value',
|
|
name: 'MANY'
|
|
});
|
|
t.strictSame(loop.json.args1[0], {
|
|
type: 'input_statement',
|
|
name: 'SUBSTACK'
|
|
});
|
|
t.equal(loop.json.lastDummyAlign2, 'RIGHT'); // move loop arrow to right side
|
|
t.equal(loop.json.args2[0].type, 'field_image');
|
|
t.equal(loop.json.args2[0].flip_rtl, true);
|
|
t.notOk(loop.json.hasOwnProperty('args3'));
|
|
t.equal(loop.xml,
|
|
'<block type="test_loop"><value name="MANY"><shadow type="math_number"><field name="NUM">' +
|
|
'</field></shadow></value></block>');
|
|
};
|
|
|
|
test('registerExtensionPrimitives', t => {
|
|
const runtime = new Runtime();
|
|
|
|
runtime.on(Runtime.EXTENSION_ADDED, blocksInfo => {
|
|
t.equal(blocksInfo.length, testExtensionInfo.blocks.length);
|
|
|
|
blocksInfo.forEach(blockInfo => {
|
|
// `true` here means "either an object or a non-empty string but definitely not null or undefined"
|
|
t.true(blockInfo.info, 'Every block and pseudo-block must have a non-empty "info" field');
|
|
});
|
|
|
|
// Note that this also implicitly tests that block order is preserved
|
|
const [button, reporter, separator, command, conditional, loop] = blocksInfo;
|
|
|
|
testButton(t, button);
|
|
testReporter(t, reporter);
|
|
testSeparator(t, separator);
|
|
testCommand(t, command);
|
|
testConditional(t, conditional);
|
|
testLoop(t, loop);
|
|
|
|
t.end();
|
|
});
|
|
|
|
runtime._registerExtensionPrimitives(testExtensionInfo);
|
|
});
|