123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199 |
- import defineDynamicBlock from '../../../src/lib/define-dynamic-block';
- import BlockType from 'scratch-vm/src/extension-support/block-type';
- const MockScratchBlocks = {
- OUTPUT_SHAPE_HEXAGONAL: 1,
- OUTPUT_SHAPE_ROUND: 2,
- OUTPUT_SHAPE_SQUARE: 3
- };
- const categoryInfo = {
- name: 'test category',
- color1: '#111',
- color2: '#222',
- color3: '#333'
- };
- const penIconURI = '_pen_icon_svg_base64_data';
- const testBlockInfo = {
- commandWithIcon: {
- blockType: BlockType.COMMAND,
- blockIconURI: penIconURI,
- text: 'command with icon'
- },
- commandWithoutIcon: {
- blockType: BlockType.COMMAND,
- text: 'command without icon'
- },
- terminalCommand: {
- blockType: BlockType.COMMAND,
- isTerminal: true,
- text: 'terminal command'
- },
- reporter: {
- blockType: BlockType.REPORTER,
- text: 'reporter'
- },
- boolean: {
- blockType: BlockType.BOOLEAN,
- text: 'Boolean'
- },
- hat: {
- blockType: BlockType.HAT,
- text: 'hat'
- }
- };
- // similar to goog.mixin from the Closure library
- const mixin = function (target, source) {
- for (const x in source) {
- target[x] = source[x];
- }
- };
- class MockBlock {
- constructor (blockInfo, extendedOpcode) {
- // mimic Closure-style inheritance by mixing in `defineDynamicBlock` output as this instance's prototype
- // see also the `Blockly.Block` constructor
- const prototype = defineDynamicBlock(MockScratchBlocks, categoryInfo, blockInfo, extendedOpcode);
- mixin(this, prototype);
- this.init();
- // bootstrap the mutation<->DOM cycle
- this.blockInfoText = JSON.stringify(blockInfo);
- const xmlElement = this.mutationToDom();
- // parse blockInfo from XML to fill dynamic properties
- this.domToMutation(xmlElement);
- }
- jsonInit (json) {
- this.result = Object.assign({}, json);
- }
- interpolate_ () {
- // TODO: add tests for this?
- }
- setCheckboxInFlyout (isEnabled) {
- this.result.checkboxInFlyout_ = isEnabled;
- }
- setOutput (isEnabled) {
- this.result.outputConnection = isEnabled; // Blockly calls `makeConnection_` here
- }
- setOutputShape (outputShape) {
- this.result.outputShape_ = outputShape;
- }
- setNextStatement (isEnabled) {
- this.result.nextConnection = isEnabled; // Blockly calls `makeConnection_` here
- }
- setPreviousStatement (isEnabled) {
- this.result.previousConnection = isEnabled; // Blockly calls `makeConnection_` here
- }
- }
- describe('defineDynamicBlock', () => {
- test('is a function', () => {
- expect(typeof defineDynamicBlock).toBe('function');
- });
- test('can define a command block with an icon', () => {
- const extendedOpcode = 'test.commandWithIcon';
- const block = new MockBlock(testBlockInfo.commandWithIcon, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- extensions: ['scratch_extension'],
- inputsInline: true,
- nextConnection: true,
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_SQUARE,
- previousConnection: true,
- type: extendedOpcode
- });
- });
- test('can define a command block without an icon', () => {
- const extendedOpcode = 'test.commandWithoutIcon';
- const block = new MockBlock(testBlockInfo.commandWithoutIcon, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- // extensions: undefined, // no icon means no extension
- inputsInline: true,
- nextConnection: true,
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_SQUARE,
- previousConnection: true,
- type: extendedOpcode
- });
- });
- test('can define a terminal command', () => {
- const extendedOpcode = 'test.terminal';
- const block = new MockBlock(testBlockInfo.terminalCommand, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- // extensions: undefined, // no icon means no extension
- inputsInline: true,
- nextConnection: false, // terminal
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_SQUARE,
- previousConnection: true,
- type: extendedOpcode
- });
- });
- test('can define a reporter', () => {
- const extendedOpcode = 'test.reporter';
- const block = new MockBlock(testBlockInfo.reporter, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- checkboxInFlyout_: true,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- // extensions: undefined, // no icon means no extension
- inputsInline: true,
- // nextConnection: undefined, // reporter
- outputConnection: true, // reporter
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_ROUND, // reporter
- // previousConnection: undefined, // reporter
- type: extendedOpcode
- });
- });
- test('can define a Boolean', () => {
- const extendedOpcode = 'test.boolean';
- const block = new MockBlock(testBlockInfo.boolean, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- // checkboxInFlyout_: undefined,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- // extensions: undefined, // no icon means no extension
- inputsInline: true,
- // nextConnection: undefined, // reporter
- outputConnection: true, // reporter
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_HEXAGONAL, // Boolean
- // previousConnection: undefined, // reporter
- type: extendedOpcode
- });
- });
- test('can define a hat', () => {
- const extendedOpcode = 'test.hat';
- const block = new MockBlock(testBlockInfo.hat, extendedOpcode);
- expect(block.result).toEqual({
- category: categoryInfo.name,
- colour: categoryInfo.color1,
- colourSecondary: categoryInfo.color2,
- colourTertiary: categoryInfo.color3,
- // extensions: undefined, // no icon means no extension
- inputsInline: true,
- nextConnection: true,
- outputShape_: MockScratchBlocks.OUTPUT_SHAPE_SQUARE,
- // previousConnection: undefined, // hat
- type: extendedOpcode
- });
- });
- });
|