diff --git a/media/test_200px.png b/media/test_200px.png
new file mode 100644
index 000000000..a08e2fa68
Binary files /dev/null and b/media/test_200px.png differ
diff --git a/media/test_30px.png b/media/test_30px.png
new file mode 100644
index 000000000..bf2532b79
Binary files /dev/null and b/media/test_30px.png differ
diff --git a/media/test_50px.png b/media/test_50px.png
new file mode 100644
index 000000000..f09f54313
Binary files /dev/null and b/media/test_50px.png differ
diff --git a/media/test_a.png b/media/test_a.png
new file mode 100644
index 000000000..b0706daac
Binary files /dev/null and b/media/test_a.png differ
diff --git a/media/test_b.png b/media/test_b.png
new file mode 100644
index 000000000..3798fe746
Binary files /dev/null and b/media/test_b.png differ
diff --git a/media/test_c.png b/media/test_c.png
new file mode 100644
index 000000000..86b5c834e
Binary files /dev/null and b/media/test_c.png differ
diff --git a/media/test_d.png b/media/test_d.png
new file mode 100644
index 000000000..40eab0f47
Binary files /dev/null and b/media/test_d.png differ
diff --git a/media/test_e.png b/media/test_e.png
new file mode 100644
index 000000000..2070b06b0
Binary files /dev/null and b/media/test_e.png differ
diff --git a/media/test_f.png b/media/test_f.png
new file mode 100644
index 000000000..beb8b9e86
Binary files /dev/null and b/media/test_f.png differ
diff --git a/media/test_g.png b/media/test_g.png
new file mode 100644
index 000000000..5a6c6eb26
Binary files /dev/null and b/media/test_g.png differ
diff --git a/media/test_h.png b/media/test_h.png
new file mode 100644
index 000000000..33cbb44c8
Binary files /dev/null and b/media/test_h.png differ
diff --git a/media/test_i.png b/media/test_i.png
new file mode 100644
index 000000000..cf4d90c8b
Binary files /dev/null and b/media/test_i.png differ
diff --git a/media/test_j.png b/media/test_j.png
new file mode 100644
index 000000000..81da54444
Binary files /dev/null and b/media/test_j.png differ
diff --git a/media/test_k.png b/media/test_k.png
new file mode 100644
index 000000000..5f80a8230
Binary files /dev/null and b/media/test_k.png differ
diff --git a/media/test_l.png b/media/test_l.png
new file mode 100644
index 000000000..c43626e09
Binary files /dev/null and b/media/test_l.png differ
diff --git a/media/test_m.png b/media/test_m.png
new file mode 100644
index 000000000..b9b3385bd
Binary files /dev/null and b/media/test_m.png differ
diff --git a/tests/blocks/test_blocks.js b/tests/blocks/test_blocks.js
new file mode 100644
index 000000000..57ed9a037
--- /dev/null
+++ b/tests/blocks/test_blocks.js
@@ -0,0 +1,493 @@
+/**
+ * @license
+ * Blockly Tests
+ *
+ * Copyright 2017 Google Inc.
+ * https://developers.google.com/blockly/
+ *
+ * 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';
+
+Blockly.defineBlocksWithJsonArray([ // BEGIN JSON EXTRACT
+ {
+ "type": "empty_block",
+ "message0": "",
+ "args0": []
+ },
+ {
+ "type": "example_dropdown_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": "example_dropdown_images",
+ "message0": "%1",
+ "args0": [
+ {
+ "type": "field_dropdown",
+ "name": "FIELDNAME",
+ "options": [
+ [{"src": "../media/test_a.png", "width": 32, "height": 32, "alt": "A"}, "A"],
+ [{"src": "../media/test_b.png", "width": 32, "height": 32, "alt": "B"}, "B"],
+ [{"src": "../media/test_c.png", "width": 32, "height": 32, "alt": "C"}, "C"],
+ [{"src": "../media/test_d.png", "width": 32, "height": 32, "alt": "D"}, "D"],
+ [{"src": "../media/test_e.png", "width": 32, "height": 32, "alt": "E"}, "E"],
+ [{"src": "../media/test_f.png", "width": 32, "height": 32, "alt": "F"}, "F"],
+ [{"src": "../media/test_g.png", "width": 32, "height": 32, "alt": "G"}, "G"],
+ [{"src": "../media/test_h.png", "width": 32, "height": 32, "alt": "H"}, "H"],
+ [{"src": "../media/test_i.png", "width": 32, "height": 32, "alt": "I"}, "I"],
+ [{"src": "../media/test_j.png", "width": 32, "height": 32, "alt": "J"}, "J"],
+ [{"src": "../media/test_k.png", "width": 32, "height": 32, "alt": "K"}, "K"],
+ [{"src": "../media/test_l.png", "width": 32, "height": 32, "alt": "L"}, "L"],
+ [{"src": "../media/test_m.png", "width": 32, "height": 32, "alt": "M"}, "M"]
+ ]
+ }
+ ]
+ },
+ {
+ "type": "example_angle",
+ "message0": "angle: %1",
+ "args0": [
+ {
+ "type": "field_angle",
+ "name": "FIELDNAME",
+ "angle": "90",
+ "alt":
+ {
+ "type": "field_label",
+ "text": "NO ANGLE FIELD"
+ }
+ }
+ ]
+ },
+ {
+ "type": "example_date",
+ "message0": "date: %1",
+ "args0": [
+ {
+ "type": "field_date",
+ "name": "FIELDNAME",
+ "date": "2020-02-20",
+ "alt":
+ {
+ "type": "field_label",
+ "text": "NO DATE FIELD"
+ }
+ }
+ ]
+ },
+ {
+ "type": "test_number",
+ "message0": "float %1",
+ "args0": [
+ {
+ "type": "field_number",
+ "name": "NUM",
+ "text": "0"
+ }
+ ],
+ "colour": 230,
+ "output": "Number",
+ "tooltip": "A number."
+ },
+ {
+ "type": "test_integer",
+ "message0": "integer %1",
+ "args0": [
+ {
+ "type": "field_number",
+ "name": "NUM",
+ "precision": 1,
+ "text": "0"
+ }
+ ],
+ "colour": 230,
+ "output": "Number",
+ "tooltip": "An integer."
+ },
+ {
+ "type": "test_number_hundredths",
+ "message0": "$ %1",
+ "args0": [
+ {
+ "type": "field_number",
+ "name": "NUM",
+ "precision": 0.01,
+ "text": "0"
+ }
+ ],
+ "colour": 230,
+ "output": "Number",
+ "tooltip": "A dollar amount."
+ },
+ {
+ "type": "test_integer_bounded",
+ "message0": "midi note %1",
+ "args0": [
+ {
+ "type": "field_number",
+ "name": "NOTE",
+ "precision": 1,
+ "min": 1,
+ "max": 127,
+ "text": "0"
+ }
+ ],
+ "colour": 230,
+ "output": "Note",
+ "tooltip": "A midi note."
+ },
+ {
+ "type": "image_datauri",
+ "message0": "Image data: URI %1",
+ "args0": [
+ {
+ "type": "field_image",
+ "src": "",
+ "width": 50,
+ "height": 50,
+ "alt": "*"
+ }
+ ],
+ "colour": 160
+ },
+ {
+ "type": "image_small",
+ "message0": "Image too small %1",
+ "args0": [
+ {
+ "type": "field_image",
+ "src": "../media/test_30px.png",
+ "width": 50,
+ "height": 50,
+ "alt": "*"
+ }
+ ],
+ "colour": 160
+ },
+ {
+ "type": "image_large",
+ "message0": "Image too large %1",
+ "args0": [
+ {
+ "type": "field_image",
+ "src": "../media/test_200px.png",
+ "width": 50,
+ "height": 50,
+ "alt": "*"
+ }
+ ],
+ "colour": 160
+ },
+ {
+ "type": "image_missing",
+ "message0": "Image missing %1",
+ "args0": [
+ {
+ "type": "field_image",
+ "src": "missing.png",
+ "width": 50,
+ "height": 50,
+ "alt": "*"
+ }
+ ],
+ "colour": 160
+ },
+ {
+ "type": "test_with_lots_of_network_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/media/test_a.png",
+ "width": 32,
+ "height": 32,
+ "alt": "A"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_b.png",
+ "width": 32,
+ "height": 32,
+ "alt": "B"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_c.png",
+ "width": 32,
+ "height": 32,
+ "alt": "C"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_d.png",
+ "width": 32,
+ "height": 32,
+ "alt": "D"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_e.png",
+ "width": 32,
+ "height": 32,
+ "alt": "E"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_f.png",
+ "width": 32,
+ "height": 32,
+ "alt": "F"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_g.png",
+ "width": 32,
+ "height": 32,
+ "alt": "G"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_h.png",
+ "width": 32,
+ "height": 32,
+ "alt": "H"
+ },
+ {
+ "type": "input_dummy"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_a.png",
+ "width": 32,
+ "height": 32,
+ "alt": "A"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_b.png",
+ "width": 32,
+ "height": 32,
+ "alt": "B"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_c.png",
+ "width": 32,
+ "height": 32,
+ "alt": "C"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_d.png",
+ "width": 32,
+ "height": 32,
+ "alt": "D"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_e.png",
+ "width": 32,
+ "height": 32,
+ "alt": "E"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_f.png",
+ "width": 32,
+ "height": 32,
+ "alt": "F"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_g.png",
+ "width": 32,
+ "height": 32,
+ "alt": "G"
+ },
+ {
+ "type": "field_image",
+ "src": "https://blockly-demo.appspot.com/static/media/test_h.png",
+ "width": 32,
+ "height": 32,
+ "alt": "H"
+ }
+ ],
+ "colour": 160
+ },
+ {
+ "type": "styled_event_cap",
+ "message0": "Hat block (event)",
+ "nextStatement": null,
+ "colour": 330,
+ "style": {
+ "hat": "cap"
+ }
+ },
+ {
+ "type": "block_colour_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": "block_colour_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": "block_colour_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": "block_no_colour",
+ "message0": "Block color: unset"
+ },
+ {
+ "type": "block_colour_hex4",
+ "message0": "Block color: #RRGGBBAA (invalid)",
+ "colour": "#992aff99"
+ },
+ {
+ "type": "block_colour_hex5",
+ "message0": "Block color: #RRGGBB (invalid)",
+ "colour": "#NotHex"
+ }
+]); // END JSON EXTRACT (Do not delete this comment.)
+
+Blockly.Blocks['empty_block_with_mutator'] = {
+ init: function() {
+ this.setMutator(new Blockly.Mutator(['math_number']));
+ }
+};
\ No newline at end of file
diff --git a/tests/playground.html b/tests/playground.html
index 596e8f373..6bbe2b43f 100644
--- a/tests/playground.html
+++ b/tests/playground.html
@@ -64,6 +64,7 @@
+