/** * @license * Copyright 2017 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ 'use strict'; goog.provide('Blockly.TestBlocks'); Blockly.defineBlocksWithJsonArray([ // BEGIN JSON EXTRACT { "type": "test_basic_empty", "message0": "", "args0": [] }, { "type": "test_basic_stack", "message0": "stack block", "previousStatement": null, "nextStatement": null, "colour": "120" }, { "type": "test_basic_dummy", "message0": "dummy input %1", "args0": [ { "type": "input_dummy" } ], "style": "math_blocks" }, { "type": "test_basic_multiple_dummy", "message0": "first dummy %1 second dummy %2", "args0": [ { "type": "input_dummy" }, { "type": "input_dummy" } ], "style": "math_blocks" }, { "type": "test_basic_row", "message0": "row block %1", "args0": [ { "type": "input_value", "name": "INPUT" } ], "output": null, "style": "math_blocks" }, { "type": "test_basic_value_to_stack", "message0": "value to stack", "nextStatement": null, "output": null, "style": "math_blocks" }, { "type": "test_basic_value_to_statement", "message0": "value to statement %1", "args0": [ { "type": "input_statement", "name": "STATEMENT" } ], "output": null, "style": "math_blocks" }, { "type": "test_basic_limit_instances", "message0": "limit 3 instances %1 %2", "args0": [ { "type": "input_dummy" }, { "type": "input_statement", "name": "STATEMENT" } ], "previousStatement": null, "nextStatement": null, "style": "math_blocks", }, { "type": "test_basic_tooltips", "message0": "%1 %2 %3", "args0": [ { "type": "field_label", "name": "NAME", "text": "field tooltip", "tooltip": "This is a JSON tooltip for the *field*." }, { "type": "input_dummy" }, { "type": "field_label", "name": "NAME", "text": "block tooltip" } ], "tooltip": "This is a JSON tooltip for the *block*.", "style": "math_blocks" }, { "type": "test_basic_javascript", "message0": "function %1(%2) { %3 %4 return %5 }", "args0": [ "foo", "args", { "type": "input_dummy" }, { "type": "input_statement", "name": "STACK" }, { "type": "input_value", "check": "number", "align": "right", "name": "RETURN" } ], "inputsInline": true, "colour": 200, "tooltip": "Hello world." }, { "type": "test_align_dummy_right", "message0": "text right %1 long text right %2", "args0": [ { "type": "input_dummy", "align": "RIGHT", }, { "type": "input_dummy", "align": "RIGHT", }, ], "style": "math_blocks" }, { "type": "test_align_all", "message0": "text %1 long text left %2 text centre %3 much longer text right", "args0": [ { "type": "input_dummy", "align": "LEFT", }, { "type": "input_dummy", "align": "CENTRE", }, { "type": "input_dummy", "align": "RIGHT", }, ], "style": "math_blocks" }, { "type": "test_align_with_external_input", "message0": "text right %1 long text centre %2 text left %3 much longer text %4", "args0": [ { "type": "input_dummy", "align": "RIGHT", }, { "type": "input_dummy", "align": "CENTRE", }, { "type": "input_dummy", "align": "LEFT", }, { "type": "input_value", "name": "VALUE" }, ], "inputsInline": false, "style": "math_blocks" }, { "type": "test_dropdowns_long", "message0": "long: %1", "args0": [ { "type": "field_dropdown", "name": "FIELDNAME", "options": [ [ "first item", "ITEM1" ], [ "second item", "ITEM2" ], [ "third item", "ITEM3" ], [ "fourth item", "ITEM4" ], [ "fifth item", "ITEM5" ], [ "sixth item", "ITEM6" ], [ "seventh item", "ITEM7" ], [ "eighth item", "ITEM8" ], [ "ninth item", "ITEM9" ], [ "tenth item", "ITEM10" ], [ "eleventh item", "ITEM11" ], [ "twelfth item", "ITEM12" ], [ "thirteenth item", "ITEM13" ], [ "fourteenth item", "ITEM14" ], [ "fifteenth item", "ITEM15" ], [ "sixteenth item", "ITEM16" ], [ "seventeenth item", "ITEM17" ], [ "eighteenth item", "ITEM18" ], [ "nineteenth item", "ITEM19" ], [ "twentieth item", "ITEM20" ], [ "twenty-first item", "ITEM21" ], [ "twenty-second item", "ITEM22" ], [ "twenty-third item", "ITEM23" ], [ "twenty-fourth item", "ITEM24" ], [ "twenty-fifth item", "ITEM25" ], [ "twenty-sixth item", "ITEM26" ], [ "twenty-seventh item", "ITEM27" ], [ "twenty-eighth item", "ITEM28" ], [ "twenty-ninth item", "ITEM29" ], [ "thirtieth item", "ITEM30" ], [ "thirty-first item", "ITEM31" ], [ "thirty-second item", "ITEM32" ] ] } ] }, { "type": "test_dropdowns_images", "message0": "%1", "args0": [ { "NOTE": "The following paths are relative to playground.html", "type": "field_dropdown", "name": "FIELDNAME", "options": [ [{"src": "media/a.png", "width": 32, "height": 32, "alt": "A"}, "A"], [{"src": "media/b.png", "width": 32, "height": 32, "alt": "B"}, "B"], [{"src": "media/c.png", "width": 32, "height": 32, "alt": "C"}, "C"], [{"src": "media/d.png", "width": 32, "height": 32, "alt": "D"}, "D"], [{"src": "media/e.png", "width": 32, "height": 32, "alt": "E"}, "E"], [{"src": "media/f.png", "width": 32, "height": 32, "alt": "F"}, "F"], [{"src": "media/g.png", "width": 32, "height": 32, "alt": "G"}, "G"], [{"src": "media/h.png", "width": 32, "height": 32, "alt": "H"}, "H"], [{"src": "media/i.png", "width": 32, "height": 32, "alt": "I"}, "I"], [{"src": "media/j.png", "width": 32, "height": 32, "alt": "J"}, "J"], [{"src": "media/k.png", "width": 32, "height": 32, "alt": "K"}, "K"], [{"src": "media/l.png", "width": 32, "height": 32, "alt": "L"}, "L"], [{"src": "media/m.png", "width": 32, "height": 32, "alt": "M"}, "M"] ] } ] }, { "type": "test_dropdowns_images_and_text", "message0": "%1", "args0": [ { "type": "field_dropdown", "name": "FIELDNAME", "options": [ ["images and text", "IMAGES AND TEXT"], [{"src": "media/a.png", "width": 32, "height": 32, "alt": "A"}, "A"], [{"src": "media/b.png", "width": 32, "height": 32, "alt": "B"}, "B"], [{"src": "media/c.png", "width": 32, "height": 32, "alt": "C"}, "C"], [{"src": "media/d.png", "width": 32, "height": 32, "alt": "D"}, "D"], [{"src": "media/e.png", "width": 32, "height": 32, "alt": "E"}, "E"], [{"src": "media/f.png", "width": 32, "height": 32, "alt": "F"}, "F"], [{"src": "media/g.png", "width": 32, "height": 32, "alt": "G"}, "G"], [{"src": "media/h.png", "width": 32, "height": 32, "alt": "H"}, "H"], ["xyz", "LMNOP"], [{"src": "media/i.png", "width": 32, "height": 32, "alt": "I"}, "I"], [{"src": "media/j.png", "width": 32, "height": 32, "alt": "J"}, "J"], [{"src": "media/k.png", "width": 32, "height": 32, "alt": "K"}, "K"], [{"src": "media/l.png", "width": 32, "height": 32, "alt": "L"}, "L"], [{"src": "media/m.png", "width": 32, "height": 32, "alt": "M"}, "M"] ] } ] }, { "type": "test_fields_angle", "message0": "angle: %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": "90", "alt": { "type": "field_label", "text": "NO ANGLE FIELD" } } ], "style": "math_blocks", }, { "type": "test_fields_date", "message0": "date: %1", "args0": [ { "type": "field_date", "name": "FIELDNAME", "date": "2020-02-20", "alt": { "type": "field_label", "text": "NO DATE FIELD" } } ], "style": "math_blocks", }, { "type": "test_fields_text_input", "message0": "text input %1", "args0": [ { "type": "field_input", "name": "TEXT_INPUT", "text": "default" } ], "style": "math_blocks", "tooltip": "", "helpUrl": "" }, { "type": "test_fields_only_text_input", "message0": "%1", "args0": [ { "type": "field_input", "name": "TEXT_INPUT", "text": "default" } ], "style": "math_blocks", "tooltip": "", "helpUrl": "", "output": "String", "style": "textInput" }, { "type": "test_fields_multilinetext", "message0": "code %1", "args0": [ { "type": "field_multilinetext", "name": "CODE", "text": "default1\ndefault2" } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_fields_checkbox", "message0": "checkbox %1", "args0": [ { "type": "field_checkbox", "name": "CHECKBOX", "checked": true } ], "style": "math_blocks", "tooltip": "", "helpUrl": "" }, { "type": "test_fields_colour", "message0": "colour %1", "args0": [ { "type": "field_colour", "name": "COLOUR", "colour": "#ff0000" } ], "style": "math_blocks", "tooltip": "", "helpUrl": "" }, { "type": "test_fields_colour_options", "message0": "colour options %1", "args0": [ { "type": "field_colour", "name": "COLOUR", "colour": "#ff4040", "colourOptions": ['#ff4040', '#ff8080', '#ffc0c0', '#4040ff', '#8080ff', '#c0c0ff'], "colourTitles": ['dark pink', 'pink', 'light pink', 'dark blue', 'blue', 'light blue'], "columns": 3 } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_fields_variable", "message0": "variable %1", "args0": [ { "type": "field_variable", "name": "VARIABLE", "variable": "item" } ], "style": "math_blocks", "tooltip": "", "helpUrl": "" }, { "type": "test_fields_label_serializable", "message0": "label serializable %1", "args0": [ { "type": "field_label_serializable", "name": "LABEL", "text": "default" } ], "style": "math_blocks", "tooltip": "", "helpUrl": "" }, { "type": "test_fields_image", "message0": "image %1", "args0": [ { "type": "field_image", "name": "IMAGE", "src": "https://blockly-demo.appspot.com/static/tests/media/a.png", "width": 32, "height": 32, "alt": "A" } ], "colour": 230 }, { "type": "test_numbers_float", "message0": "float %1", "args0": [ { "type": "field_number", "name": "NUM", "text": "0" } ], "style": "math_blocks", "output": "Number", "tooltip": "A number." }, { "type": "test_numbers_whole", "message0": "precision 1 %1", "args0": [ { "type": "field_number", "name": "NUM", "precision": 1, "text": "0" } ], "style": "math_blocks", "output": "Number", "tooltip": "The number should be rounded to multiples of 1" }, { "type": "test_numbers_hundredths", "message0": "precision 0.01 %1", "args0": [ { "type": "field_number", "name": "NUM", "precision": 0.01, "text": "0" } ], "style": "math_blocks", "output": "Number", "tooltip": "The number should be rounded to multiples of 0.01" }, { "type": "test_numbers_halves", "message0": "precision 0.5 %1", "args0": [ { "type": "field_number", "name": "NUM", "precision": 0.5, "text": "0" } ], "style": "math_blocks", "output": "Number", "tooltip": "The number should be rounded to multiples of 0.5" }, { "type": "test_numbers_three_halves", "message0": "precision 1.5 %1", "args0": [ { "type": "field_number", "name": "NUM", "precision": 1.5, "text": "0" } ], "style": "math_blocks", "output": "Number", "tooltip": "The number should be rounded to multiples of 1.5" }, { "type": "test_numbers_whole_bounded", "message0": "midi note %1", "args0": [ { "type": "field_number", "name": "NOTE", "precision": 1, "min": 1, "max": 127, "text": "0" } ], "style": "math_blocks", "output": "Note", "tooltip": "A midi note." }, { "type": "test_angles_protractor", "message0": "protractor %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "mode": "protractor" } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_compass", "message0": "compass %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "mode": "compass" } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_clockwise", "message0": "clockwise %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "clockwise": true } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_offset", "message0": "offset 90 %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "offset": 90 } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_wrap", "message0": "wrap %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "wrap": 180 } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_round_30", "message0": "round 30 %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "round": 30 } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_angles_round_0", "message0": "no round %1", "args0": [ { "type": "field_angle", "name": "FIELDNAME", "angle": 0, "round": 0 } ], "style": "math_blocks", "tooltip": "test tooltip" }, { "type": "test_images_datauri", "message0": "Image data: URI %1", "args0": [ { "type": "field_image", "src": "", "width": 50, "height": 50, "alt": "*" } ], "style": "text_blocks" }, { "type": "test_images_small", "message0": "Image too small %1", "args0": [ { "type": "field_image", "src": "media/30px.png", "width": 50, "height": 50, "alt": "*" } ], "style": "text_blocks" }, { "type": "test_images_large", "message0": "Image too large %1", "args0": [ { "type": "field_image", "src": "media/200px.png", "width": 50, "height": 50, "alt": "*" } ], "style": "text_blocks" }, { "type": "test_images_fliprtl", "message0": "Image flipped RTL %1", "args0": [ { "type": "field_image", "src": "media/arrow.png", "width": 50, "height": 50, "alt": "*", "flipRtl": true } ], "colour": 160 }, { "type": "test_images_missing", "message0": "Image missing %1", "args0": [ { "type": "field_image", "src": "missing.png", "width": 50, "height": 50, "alt": "*" } ], "style": "text_blocks" }, { "type": "test_images_many_icons", "message0": "Lots of network icons: %1 %2 %3 %4 %5 %6 %7 %8 %9 %10 %11 %12 %13 %14 %15 %16 %17 %18", "args0": [ { "type": "input_dummy" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/a.png", "width": 32, "height": 32, "alt": "A" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/b.png", "width": 32, "height": 32, "alt": "B" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/c.png", "width": 32, "height": 32, "alt": "C" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/d.png", "width": 32, "height": 32, "alt": "D" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/e.png", "width": 32, "height": 32, "alt": "E" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/f.png", "width": 32, "height": 32, "alt": "F" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/g.png", "width": 32, "height": 32, "alt": "G" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/h.png", "width": 32, "height": 32, "alt": "H" }, { "type": "input_dummy" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/a.png", "width": 32, "height": 32, "alt": "A" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/b.png", "width": 32, "height": 32, "alt": "B" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/c.png", "width": 32, "height": 32, "alt": "C" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/d.png", "width": 32, "height": 32, "alt": "D" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/e.png", "width": 32, "height": 32, "alt": "E" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/f.png", "width": 32, "height": 32, "alt": "F" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/g.png", "width": 32, "height": 32, "alt": "G" }, { "type": "field_image", "src": "https://blockly-demo.appspot.com/static/tests/media/h.png", "width": 32, "height": 32, "alt": "H" } ], "style": "text_blocks" }, { "type": "test_style_hat", "message0": "Hat block (event)", "nextStatement": null, "style": "hat_blocks" }, { "type": "test_style_hex1", "message0": "Block color: Bright purple %1 %2 %3 %4", "args0": [ { "type": "field_input", "name": "TEXT", "text": "#992aff" }, { "type": "field_dropdown", "name": "DROPDOWN", "options": [ [ "option", "ONE" ], [ "option", "TWO" ] ] }, { "type": "field_checkbox", "name": "NAME", "checked": true }, { "type": "input_value", "name": "NAME" } ], "previousStatement": null, "nextStatement": null, "colour": "#992aff" }, { "type": "test_style_hex2", "message0": "Block color: White %1 %2 %3 %4", "args0": [ { "type": "field_input", "name": "TEXT", "text": "#fefefe" }, { "type": "field_dropdown", "name": "DROPDOWN", "options": [ [ "option", "ONE" ], [ "option", "TWO" ] ] }, { "type": "field_checkbox", "name": "NAME", "checked": true }, { "type": "input_value", "name": "NAME" } ], "previousStatement": null, "nextStatement": null, "colour": "#fefefe" }, { "type": "test_style_hex3", "message0": "Block color: Black %1 %2 %3 %4", "args0": [ { "type": "field_input", "name": "TEXT", "text": "#010101" }, { "type": "field_dropdown", "name": "DROPDOWN", "options": [ [ "option", "ONE" ], [ "option", "TWO" ] ] }, { "type": "field_checkbox", "name": "NAME", "checked": true }, { "type": "input_value", "name": "NAME" } ], "previousStatement": null, "nextStatement": null, "colour": "#010101" }, { "type": "test_style_no_colour", "message0": "Block color: unset" }, { "type": "test_style_hex4", "message0": "Block color: #RRGGBBAA (invalid)", "colour": "#992aff99" }, { "type": "test_style_hex5", "message0": "Block color: #RRGGBB (invalid)", "colour": "#NotHex" }, { "type": "test_style_emoji", "message0": "Robot Face: \uD83E\uDD16", "colour": "#AAAAAA" } ]); // END JSON EXTRACT (Do not delete this comment.) Blockly.Blocks['test_validators_text_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldTextInput("default", this.validate), "INPUT"); this.setColour(230); this.setCommentText('All input validates to null (invalid). The display' + ' text will remain the input text, but the value should be the default' + ' text. The input should be red after the first keystroke.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_text_A'] = { init: function() { this.appendDummyInput() .appendField("remove \'a\'") .appendField(new Blockly.FieldTextInput("default", this.validate), "INPUT"); this.setColour(230); this.setCommentText('All \'a\' characters are removed from field value.' + ' The display text will include invalid \'a\' characters while the' + ' field is being edited, but the value will not.'); }, validate: function(newValue) { return newValue.replace(/\a/g, ''); } }; Blockly.Blocks['test_validators_text_B'] = { init: function() { this.appendDummyInput() .appendField("\'b\' -> null") .appendField(new Blockly.FieldTextInput("default", this.validate), "INPUT"); this.setColour(230); this.setCommentText('Upon detecting a \'b\' character the input will' + ' validated to null (invalid). Upon removal it should revert to being' + ' valid. The display text will remain the input text, but if the input' + ' text is invalid the value should be the default text.'); }, validate: function(newValue) { if (newValue.indexOf('b') != -1) { return null; } return newValue; } }; Blockly.Blocks['test_validators_angle_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldAngle(90, this.validate), "INPUT"); this.setColour(230); this.setCommentText('All input validates to null (invalid). The field' + ' will display the input while the field is being edited (this' + ' includes the text and the graphic), but the value should be the' + ' default value. The input should be red after the first' + ' keystroke.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_angle_mult30_force'] = { init: function() { this.appendDummyInput() .appendField("force mult of 30") .appendField(new Blockly.FieldAngle(90, this.validate), "INPUT"); this.setColour(230); this.setCommentText('The input value will be rounded to the nearest' + ' multiple of 30. The field will display the input while the field is' + ' being edited (this includes the text and the graphic), but the value' + ' will be the validated (rounded) value. Note: If you want to do' + ' rounding this is not the proper way, use the ROUND property of the' + ' field angle instead.'); }, validate: function(newValue) { return Math.round(newValue / 30) * 30; } }; Blockly.Blocks['test_validators_angle_mult30_null'] = { init: function() { this.appendDummyInput() .appendField("not mult of 30 -> null") .appendField(new Blockly.FieldAngle(90, this.validate), "INPUT"); this.setColour(230); this.setCommentText('If the input value is not a multiple of 30, the' + ' input will validated to null (invalid). The field will display the' + ' input while the field is being edited (this includes the text and' + ' the graphic), but if the input value is invalid the value should be' + ' the default value.'); }, validate: function(newValue) { if (newValue % 30 != 0) { return null; } return newValue; } }; Blockly.Blocks['test_validators_checkbox_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldCheckbox(true, this.validate), "INPUT"); this.setColour(230); this.setCommentText('The new input always validates to null (invalid).' + ' This means that the field value should not change.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_checkbox_match'] = { init: function() { this.appendDummyInput() .appendField("force match") .appendField(new Blockly.FieldCheckbox(true), "MATCH") .appendField(new Blockly.FieldCheckbox(true, this.validate), "INPUT"); this.setColour(230); this.setCommentText('The validator for this block only works on the' + ' end-most checkbox. The validator will always return the value of the' + ' start-most checkbox. Therefor they should always match.') }, validate: function(newValue) { return this.sourceBlock_.getFieldValue('MATCH'); } }; Blockly.Blocks['test_validators_checkbox_not_match_null'] = { init: function() { this.appendDummyInput() .appendField("not match -> null") .appendField(new Blockly.FieldCheckbox(true), "MATCH") .appendField(new Blockly.FieldCheckbox(true, this.validate), "INPUT"); this.setColour(230); this.setCommentText('The validator for this block only works on the' + ' end-most checkbox. If the new value does not match the value of the' + ' start-most checkbox, it will return null (invalid), which means the' + ' field value should not change. Therfore they should always match.'); }, validate: function(newValue) { if (this.sourceBlock_.getFieldValue('MATCH') != newValue) { return null; } return newValue; } }; Blockly.Blocks['test_validators_colour_null'] = { init: function() { var colourField = new Blockly.FieldColour('#ff0000', this.validate); colourField.setColours([ '#ffffff', '#ffdcdc', '#ffb4b4','#ff8c8c','#ff6464','#ff3c3c','#ff1414', '#00ffff', '#00dcdc', '#00b4b4','#008c8c','#006464','#003c3c','#001414']); this.appendDummyInput() .appendField("always null") .appendField(colourField, "INPUT"); this.setColour(230); this.setCommentText('All input validates to null (invalid). This means' + ' the field value should not change.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_colour_force_red'] = { init: function() { var colourField = new Blockly.FieldColour('#ff0000', this.validate); colourField.setColours([ '#ffffff', '#ffdcdc', '#ffb4b4','#ff8c8c','#ff6464','#ff3c3c','#ff1414', '#00ffff', '#00dcdc', '#00b4b4','#008c8c','#006464','#003c3c','#001414']); this.appendDummyInput() .appendField("force full red") .appendField(colourField, "INPUT"); this.setColour(230); this.setCommentText('The input will have its red value replaced with' + ' full red.'); }, validate: function(newValue) { return '#ff' + newValue.substr(3, 4); } }; Blockly.Blocks['test_validators_colour_red_null'] = { init: function() { var colourField = new Blockly.FieldColour('#ff0000', this.validate); colourField.setColours([ '#ffffff', '#ffdcdc', '#ffb4b4','#ff8c8c','#ff6464','#ff3c3c','#ff1414', '#00ffff', '#00dcdc', '#00b4b4','#008c8c','#006464','#003c3c','#001414']); this.appendDummyInput() .appendField("not red -> null") .appendField(colourField, "INPUT"); this.setColour(230); this.setCommentText('If the input does not have full red, the input will' + ' validate to null (invalid). Otherwise it will return the input value'); }, validate: function(newValue) { if (newValue.substr(1, 2) != 'ff') { return null; } return newValue; } }; Blockly.Blocks['test_validators_dropdown_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldDropdown([ ["1a","1A"], ["1b","1B"], ["1c","1C"], ["2a","2A"], ["2b","2B"], ["2c","2C"]], this.validate), "INPUT"); this.setColour(230); this.setCommentText('All input validates to null (invalid). This means' + ' the field value should not change.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_dropdown_force_1s'] = { init: function() { this.appendDummyInput() .appendField("force 1s") .appendField(new Blockly.FieldDropdown([ ["1a","1A"], ["1b","1B"], ["1c","1C"], ["2a","2A"], ["2b","2B"], ["2c","2C"]], this.validate), "INPUT"); this.setColour(230); this.setCommentText('The input\'s value will always change to start with' + ' 1.'); }, validate: function(newValue) { return '1' + newValue.charAt(1); } }; Blockly.Blocks['test_validators_dropdown_1s_null'] = { init: function() { this.appendDummyInput() .appendField("not 1s -> null") .appendField(new Blockly.FieldDropdown([ ["1a","1A"], ["1b","1B"], ["1c","1C"], ["2a","2A"], ["2b","2B"], ["2c","2C"]], this.validate), "INPUT"); this.setColour(230); this.setCommentText('If the input does not start with 1, the input will' + ' validate to null (invalid). Otherwise it will return the input value.'); }, validate: function(newValue) { if (newValue.charAt(0) != '1') { return null; } return newValue; } }; Blockly.Blocks['test_validators_number_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldNumber(123, null, null, null, this.validate), "INPUT"); this.setColour(230); this.setCommentText('All input validates to null (invalid). The field' + ' will display the input while the field is being edited, but the value' + ' should be the default value. The input should be red after the first' + ' keystroke.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_number_mult10_force'] = { init: function() { this.appendDummyInput() .appendField("force mult of 10") .appendField(new Blockly.FieldNumber(123, null, null, null, this.validate), "INPUT"); this.setColour(230); this.setCommentText('Theinput value will be rounded to the nearest' + ' multiple of 10. The field will display the input while the field is' + ' being edited, but the value should be the validated (rounded) value.' + ' Note: If you want to do rounding this is not the proper way, use the' + ' precision option of the number field constructor instead.'); }, validate: function(newValue) { return Math.round(newValue / 10) * 10; } }; Blockly.Blocks['test_validators_number_mult10_null'] = { init: function() { this.appendDummyInput() .appendField("not mult of 10 -> null") .appendField(new Blockly.FieldNumber(123, null, null, null, this.validate), "INPUT"); this.setColour(230); this.setCommentText('If the input value is not a multiple of 10, the' + ' input will validate to null (invalid). The field will display the' + ' input while the field is being edited, but if the input value is' + ' invalid the value should be the default value.'); }, validate: function(newValue) { if (newValue % 10 != 0) { return null; } return newValue; } }; Blockly.Blocks['test_validators_variable_null'] = { init: function() { this.appendDummyInput() .appendField("always null") .appendField(new Blockly.FieldVariable('1a', this.validate), "INPUT"); this.setColour(230); this.setCommentText('All ids validate to null (invalid). This means' + ' the variable should not change.'); }, validate: function(newValue) { return null; } }; Blockly.Blocks['test_validators_variable_force_1s'] = { init: function() { this.appendDummyInput() .appendField("force 1s") .appendField(new Blockly.FieldVariable('1a', this.validate), "INPUT"); this.setColour(230); this.setCommentText('The id will always change to start with 1.'); }, validate: function(newValue) { return '1' + newValue.charAt(1); } }; Blockly.Blocks['test_validators_variable_1s_null'] = { init: function() { this.appendDummyInput() .appendField("not 1s -> null") .appendField(new Blockly.FieldVariable('1a', this.validate), "INPUT"); this.setColour(230); this.setCommentText('If the id does not start with 1, the id will' + ' validate to null (invalid). Otherwise it will return the id.'); }, validate: function(newValue) { if (newValue.charAt(0) != '1') { return null; } return newValue; } }; Blockly.Blocks['test_basic_empty_with_mutator'] = { init: function() { this.setMutator(new Blockly.Mutator(['math_number'])); } }; Blockly.Blocks['test_dropdowns_dynamic'] = { init: function() { var dropdown = new Blockly.FieldDropdown(this.dynamicOptions); this.appendDummyInput() .appendField('dynamic') .appendField(dropdown, 'OPTIONS'); }, dynamicOptions: function() { if (!Blockly.TestBlocks.dynamicDropdownOptions_.length) { return [['', 'OPTION0']]; } return Blockly.TestBlocks.dynamicDropdownOptions_; } }; /** * An array of options for the dynamic dropdown. * @type {!Array} * @package */ Blockly.TestBlocks.dynamicDropdownOptions_ = []; /** * Handles "add option" button in the field test category. This will prompt * the user for an option to add. * @package */ Blockly.TestBlocks.addDynamicDropdownOption_ = function() { Blockly.prompt('Add an option?', 'option ' + Blockly.TestBlocks.dynamicDropdownOptions_.length, function(text) { if (text) { // Do not remove this log! Helps you know if it was added correctly. console.log('Adding option: ' + text); // The option is an array containing human-readable text and a // language-neutral id. Blockly.TestBlocks.dynamicDropdownOptions_.push( [text, 'OPTION' + Blockly.TestBlocks.dynamicDropdownOptions_.length]); } }) }; /** * Handles "remove option" button in the field test category. This will prompt * the user for an option to remove. May remove multiple options with the * same name. * @package */ Blockly.TestBlocks.removeDynamicDropdownOption_ = function() { var defaultText = Blockly.TestBlocks.dynamicDropdownOptions_[0] ? Blockly.TestBlocks.dynamicDropdownOptions_[0][0] : ''; Blockly.prompt('Remove an option?', defaultText, function(text) { for (var i = 0, option; option = Blockly.TestBlocks.dynamicDropdownOptions_[i]; i++) { // The option is an array containing human-readable text and a // language-neutral id, we'll compare against the human-readable text. if (option[0] == text) { // Do not remove this log! Helps you know if it was removed correctly. console.log('Removing option: ' + text); Blockly.TestBlocks.dynamicDropdownOptions_.splice(i, 1); } } }) }; Blockly.Blocks['test_dropdowns_dynamic_random'] = { init: function() { var dropdown = new Blockly.FieldDropdown(this.dynamicOptions); this.appendDummyInput() .appendField('dynamic random') .appendField(dropdown, 'OPTIONS'); }, dynamicOptions: function() { var random = Math.floor(Math.random() * 10) + 1; var options = []; for (var i = 0; i < random; i++) { options.push([String(i), String(i)]); } return options; } };