| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523 | /** * @license * Visual Blocks Editor * * Copyright 2012 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. *//** * @fileoverview Logic blocks for Blockly. * @author q.neutron@gmail.com (Quynh Neutron) */'use strict';goog.provide('Blockly.Blocks.logic');goog.require('Blockly.Blocks');/** * Common HSV hue for all blocks in this category. */Blockly.HSV_SATURATION = 0.7;Blockly.HSV_VALUE = 0.9;Blockly.Blocks.logic.HUE = 214;Blockly.Blocks['controls_if'] = {  /**   * Block for if/elseif/else condition.   * @this Blockly.Block   */  init: function() {    this.setHelpUrl(Blockly.Msg.CONTROLS_IF_HELPURL);    this.setColour(Blockly.Blocks.logic.HUE);    this.appendValueInput('IF0')        .setCheck('Boolean')        .appendField(Blockly.Msg.CONTROLS_IF_MSG_IF);    this.appendStatementInput('DO0')        .appendField(Blockly.Msg.CONTROLS_IF_MSG_THEN);    this.setPreviousStatement(true);    this.setNextStatement(true);    this.setMutator(new Blockly.Mutator(['controls_if_elseif',                                         'controls_if_else']));    // Assign 'this' to a variable for use in the tooltip closure below.    var thisBlock = this;    this.setTooltip(function() {      if (!thisBlock.elseifCount_ && !thisBlock.elseCount_) {        return Blockly.Msg.CONTROLS_IF_TOOLTIP_1;      } else if (!thisBlock.elseifCount_ && thisBlock.elseCount_) {        return Blockly.Msg.CONTROLS_IF_TOOLTIP_2;      } else if (thisBlock.elseifCount_ && !thisBlock.elseCount_) {        return Blockly.Msg.CONTROLS_IF_TOOLTIP_3;      } else if (thisBlock.elseifCount_ && thisBlock.elseCount_) {        return Blockly.Msg.CONTROLS_IF_TOOLTIP_4;      }      return '';    });    this.elseifCount_ = 0;    this.elseCount_ = 0;  },  /**   * Create XML to represent the number of else-if and else inputs.   * @return {Element} XML storage element.   * @this Blockly.Block   */  mutationToDom: function() {    if (!this.elseifCount_ && !this.elseCount_) {      return null;    }    var container = document.createElement('mutation');    if (this.elseifCount_) {      container.setAttribute('elseif', this.elseifCount_);    }    if (this.elseCount_) {      container.setAttribute('else', 1);    }    return container;  },  /**   * Parse XML to restore the else-if and else inputs.   * @param {!Element} xmlElement XML storage element.   * @this Blockly.Block   */  domToMutation: function(xmlElement) {    this.elseifCount_ = parseInt(xmlElement.getAttribute('elseif'), 10) || 0;    this.elseCount_ = parseInt(xmlElement.getAttribute('else'), 10) || 0;    this.updateShape_();  },  /**   * Populate the mutator's dialog with this block's components.   * @param {!Blockly.Workspace} workspace Mutator's workspace.   * @return {!Blockly.Block} Root block in mutator.   * @this Blockly.Block   */  decompose: function(workspace) {    var containerBlock = workspace.newBlock('controls_if_if');    containerBlock.initSvg();    var connection = containerBlock.nextConnection;    for (var i = 1; i <= this.elseifCount_; i++) {      var elseifBlock = workspace.newBlock('controls_if_elseif');      elseifBlock.initSvg();      connection.connect(elseifBlock.previousConnection);      connection = elseifBlock.nextConnection;    }    if (this.elseCount_) {      var elseBlock = workspace.newBlock('controls_if_else');      elseBlock.initSvg();      connection.connect(elseBlock.previousConnection);    }    return containerBlock;  },  /**   * Reconfigure this block based on the mutator dialog's components.   * @param {!Blockly.Block} containerBlock Root block in mutator.   * @this Blockly.Block   */  compose: function(containerBlock) {    var clauseBlock = containerBlock.nextConnection.targetBlock();    // Count number of inputs.    this.elseifCount_ = 0;    this.elseCount_ = 0;    var valueConnections = [null];    var statementConnections = [null];    var elseStatementConnection = null;    while (clauseBlock) {      switch (clauseBlock.type) {        case 'controls_if_elseif':          this.elseifCount_++;          valueConnections.push(clauseBlock.valueConnection_);          statementConnections.push(clauseBlock.statementConnection_);          break;        case 'controls_if_else':          this.elseCount_++;          elseStatementConnection = clauseBlock.statementConnection_;          break;        default:          throw 'Unknown block type.';      }      clauseBlock = clauseBlock.nextConnection &&          clauseBlock.nextConnection.targetBlock();    }    this.updateShape_();    // Reconnect any child blocks.    for (var i = 1; i <= this.elseifCount_; i++) {      Blockly.Mutator.reconnect(valueConnections[i], this, 'IF' + i);      Blockly.Mutator.reconnect(statementConnections[i], this, 'DO' + i);    }    Blockly.Mutator.reconnect(elseStatementConnection, this, 'ELSE');  },  /**   * Store pointers to any connected child blocks.   * @param {!Blockly.Block} containerBlock Root block in mutator.   * @this Blockly.Block   */  saveConnections: function(containerBlock) {    var clauseBlock = containerBlock.nextConnection.targetBlock();    var i = 1;    while (clauseBlock) {      switch (clauseBlock.type) {        case 'controls_if_elseif':          var inputIf = this.getInput('IF' + i);          var inputDo = this.getInput('DO' + i);          clauseBlock.valueConnection_ =              inputIf && inputIf.connection.targetConnection;          clauseBlock.statementConnection_ =              inputDo && inputDo.connection.targetConnection;          i++;          break;        case 'controls_if_else':          var inputDo = this.getInput('ELSE');          clauseBlock.statementConnection_ =              inputDo && inputDo.connection.targetConnection;          break;        default:          throw 'Unknown block type.';      }      clauseBlock = clauseBlock.nextConnection &&          clauseBlock.nextConnection.targetBlock();    }  },  /**   * Modify this block to have the correct number of inputs.   * @private   * @this Blockly.Block   */  updateShape_: function() {    // Delete everything.    if (this.getInput('ELSE')) {      this.removeInput('ELSE');    }    var i = 1;    while (this.getInput('IF' + i)) {      this.removeInput('IF' + i);      this.removeInput('DO' + i);      i++;    }    // Rebuild block.    for (var i = 1; i <= this.elseifCount_; i++) {      this.appendValueInput('IF' + i)          .setCheck('Boolean')          .appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSEIF);      this.appendStatementInput('DO' + i)          .appendField(Blockly.Msg.CONTROLS_IF_MSG_THEN);    }    if (this.elseCount_) {      this.appendStatementInput('ELSE')          .appendField(Blockly.Msg.CONTROLS_IF_MSG_ELSE);    }  }};Blockly.Blocks['controls_if_if'] = {  /**   * Mutator block for if container.   * @this Blockly.Block   */  init: function() {    this.setColour(Blockly.Blocks.logic.HUE);    this.appendDummyInput()        .appendField(Blockly.Msg.CONTROLS_IF_IF_TITLE_IF);    this.setNextStatement(true);    this.setTooltip(Blockly.Msg.CONTROLS_IF_IF_TOOLTIP);    this.contextMenu = false;  }};Blockly.Blocks['controls_if_elseif'] = {  /**   * Mutator bolck for else-if condition.   * @this Blockly.Block   */  init: function() {    this.setColour(Blockly.Blocks.logic.HUE);    this.appendDummyInput()        .appendField(Blockly.Msg.CONTROLS_IF_ELSEIF_TITLE_ELSEIF);    this.setPreviousStatement(true);    this.setNextStatement(true);    this.setTooltip(Blockly.Msg.CONTROLS_IF_ELSEIF_TOOLTIP);    this.contextMenu = false;  }};Blockly.Blocks['controls_if_else'] = {  /**   * Mutator block for else condition.   * @this Blockly.Block   */  init: function() {    this.setColour(Blockly.Blocks.logic.HUE);    this.appendDummyInput()        .appendField(Blockly.Msg.CONTROLS_IF_ELSE_TITLE_ELSE);    this.setPreviousStatement(true);    this.setTooltip(Blockly.Msg.CONTROLS_IF_ELSE_TOOLTIP);    this.contextMenu = false;  }};Blockly.Blocks['logic_compare'] = {  /**   * Block for comparison operator.   * @this Blockly.Block   */  init: function() {    var rtlOperators = [      ['==', 'EQ'],      ['!=', 'NEQ'],      ['>', 'LT'],      ['>=', 'LTE'],      ['<', 'GT'],      ['<=', 'GTE']    ];    var ltrOperators = [      ['==', 'EQ'],      ['!=', 'NEQ'],      ['<', 'LT'],      ['<=', 'LTE'],      ['>', 'GT'],      ['>=', 'GTE']    ];    var OPERATORS = this.RTL ? rtlOperators : ltrOperators;    this.setHelpUrl(Blockly.Msg.LOGIC_COMPARE_HELPURL);    this.setColour(Blockly.Blocks.logic.HUE);    this.setOutput(true, 'Boolean');    this.appendValueInput('A');    this.appendValueInput('B')        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');    this.setInputsInline(true);    // Assign 'this' to a variable for use in the tooltip closure below.    var thisBlock = this;    this.setTooltip(function() {      var op = thisBlock.getFieldValue('OP');      var TOOLTIPS = {        'EQ': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_EQ,        'NEQ': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_NEQ,        'LT': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_LT,        'LTE': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_LTE,        'GT': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_GT,        'GTE': Blockly.Msg.LOGIC_COMPARE_TOOLTIP_GTE      };      return TOOLTIPS[op];    });    this.prevBlocks_ = [null, null];  },  /**   * Called whenever anything on the workspace changes.   * Prevent mismatched types from being compared.   * @param {!Blockly.Events.Abstract} e Change event.   * @this Blockly.Block   */  onchange: function(e) {    var blockA = this.getInputTargetBlock('A');    var blockB = this.getInputTargetBlock('B');    // Disconnect blocks that existed prior to this change if they don't match.    if (blockA && blockB &&        !blockA.outputConnection.checkType_(blockB.outputConnection)) {      // Mismatch between two inputs.  Disconnect previous and bump it away.      // Ensure that any disconnections are grouped with the causing event.      Blockly.Events.setGroup(e.group);      for (var i = 0; i < this.prevBlocks_.length; i++) {        var block = this.prevBlocks_[i];        if (block === blockA || block === blockB) {          block.unplug();          block.bumpNeighbours_();        }      }      Blockly.Events.setGroup(false);    }    this.prevBlocks_[0] = blockA;    this.prevBlocks_[1] = blockB;  }};Blockly.Blocks['logic_operation'] = {  /**   * Block for logical operations: 'and', 'or'.   * @this Blockly.Block   */  init: function() {    var OPERATORS =        [[Blockly.Msg.LOGIC_OPERATION_AND, 'AND'],         [Blockly.Msg.LOGIC_OPERATION_OR, 'OR']];    this.setHelpUrl(Blockly.Msg.LOGIC_OPERATION_HELPURL);    this.setColour(Blockly.Blocks.logic.HUE);    this.setOutput(true, 'Boolean');    this.appendValueInput('A')        .setCheck('Boolean');    this.appendValueInput('B')        .setCheck('Boolean')        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');    this.setInputsInline(true);    // Assign 'this' to a variable for use in the tooltip closure below.    var thisBlock = this;    this.setTooltip(function() {      var op = thisBlock.getFieldValue('OP');      var TOOLTIPS = {        'AND': Blockly.Msg.LOGIC_OPERATION_TOOLTIP_AND,        'OR': Blockly.Msg.LOGIC_OPERATION_TOOLTIP_OR      };      return TOOLTIPS[op];    });  }};Blockly.Blocks['logic_negate'] = {  /**   * Block for negation.   * @this Blockly.Block   */  init: function() {    this.jsonInit({      "message0": Blockly.Msg.LOGIC_NEGATE_TITLE,      "args0": [        {          "type": "input_value",          "name": "BOOL",          "check": "Boolean"        }      ],      "output": "Boolean",      "colour": Blockly.Blocks.logic.HUE,      "tooltip": Blockly.Msg.LOGIC_NEGATE_TOOLTIP,      "helpUrl": Blockly.Msg.LOGIC_NEGATE_HELPURL    });  }};Blockly.Blocks['logic_boolean'] = {  /**   * Block for boolean data type: true and false.   * @this Blockly.Block   */  init: function() {    this.jsonInit({      "message0": "%1",      "args0": [        {          "type": "field_dropdown",          "name": "BOOL",          "options": [            [Blockly.Msg.LOGIC_BOOLEAN_TRUE, "TRUE"],            [Blockly.Msg.LOGIC_BOOLEAN_FALSE, "FALSE"]          ]        }      ],      "output": "Boolean",      "colour": Blockly.Blocks.logic.HUE,      "tooltip": Blockly.Msg.LOGIC_BOOLEAN_TOOLTIP,      "helpUrl": Blockly.Msg.LOGIC_BOOLEAN_HELPURL    });  }};Blockly.Blocks['logic_null'] = {  /**   * Block for null data type.   * @this Blockly.Block   */  init: function() {    this.jsonInit({      "message0": Blockly.Msg.LOGIC_NULL,      "output": null,      "colour": Blockly.Blocks.logic.HUE,      "tooltip": Blockly.Msg.LOGIC_NULL_TOOLTIP,      "helpUrl": Blockly.Msg.LOGIC_NULL_HELPURL    });  }};Blockly.Blocks['logic_ternary'] = {  /**   * Block for ternary operator.   * @this Blockly.Block   */  init: function() {    this.setHelpUrl(Blockly.Msg.LOGIC_TERNARY_HELPURL);    this.setColour(Blockly.Blocks.logic.HUE);    this.appendValueInput('IF')        .setCheck('Boolean')        .appendField(Blockly.Msg.LOGIC_TERNARY_CONDITION);    this.appendValueInput('THEN')        .appendField(Blockly.Msg.LOGIC_TERNARY_IF_TRUE);    this.appendValueInput('ELSE')        .appendField(Blockly.Msg.LOGIC_TERNARY_IF_FALSE);    this.setOutput(true);    this.setTooltip(Blockly.Msg.LOGIC_TERNARY_TOOLTIP);    this.prevParentConnection_ = null;  },  /**   * Called whenever anything on the workspace changes.   * Prevent mismatched types.   * @param {!Blockly.Events.Abstract} e Change event.   * @this Blockly.Block   */  onchange: function(e) {    var blockA = this.getInputTargetBlock('THEN');    var blockB = this.getInputTargetBlock('ELSE');    var parentConnection = this.outputConnection.targetConnection;    // Disconnect blocks that existed prior to this change if they don't match.    if ((blockA || blockB) && parentConnection) {      for (var i = 0; i < 2; i++) {        var block = (i == 1) ? blockA : blockB;        if (block && !block.outputConnection.checkType_(parentConnection)) {          // Ensure that any disconnections are grouped with the causing event.          Blockly.Events.setGroup(e.group);          if (parentConnection === this.prevParentConnection_) {            this.unplug();            parentConnection.getSourceBlock().bumpNeighbours_();          } else {            block.unplug();            block.bumpNeighbours_();          }          Blockly.Events.setGroup(false);        }      }    }    this.prevParentConnection_ = parentConnection;  }};Blockly.Blocks['logic_isIn'] = {  /**   * Block for testing if something contains something.   * @this Blockly.Block   */  init: function() {    var OPERATORS =        [["is in", 'IN'],         ["is not in", 'NOTIN']];    this.setColour(Blockly.Blocks.logic.HUE);    this.setOutput(true, 'Boolean');    this.appendValueInput('ITEM');    this.appendValueInput('LIST')        .setCheck(['Array', 'String'])        .appendField(new Blockly.FieldDropdown(OPERATORS), 'OP');    this.setInputsInline(true);  }};Blockly.Blocks['logic_none'] = {  /**   * Block for testing if something contains something.   * @this Blockly.Block   */  init: function() {    this.appendDummyInput()      .appendField("None");    this.setColour(Blockly.Blocks.logic.HUE);    this.setOutput(true);  }};
 |