/** * Visual Blocks Language * * Copyright 2012 Google Inc. * http://blockly.googlecode.com/ * * 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 Generating JavaScript for text blocks. * @author fraser@google.com (Neil Fraser) */ 'use strict'; goog.provide('Blockly.JavaScript.text'); goog.require('Blockly.JavaScript'); Blockly.JavaScript['text'] = function(block) { // Text value. var code = Blockly.JavaScript.quote_(block.getTitleValue('TEXT')); return [code, Blockly.JavaScript.ORDER_ATOMIC]; }; Blockly.JavaScript['text_join'] = function(block) { // Create a string made up of any number of elements of any type. var code; if (block.itemCount_ == 0) { return ['\'\'', Blockly.JavaScript.ORDER_ATOMIC]; } else if (block.itemCount_ == 1) { var argument0 = Blockly.JavaScript.valueToCode(block, 'ADD0', Blockly.JavaScript.ORDER_NONE) || '\'\''; code = 'String(' + argument0 + ')'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; } else if (block.itemCount_ == 2) { var argument0 = Blockly.JavaScript.valueToCode(block, 'ADD0', Blockly.JavaScript.ORDER_NONE) || '\'\''; var argument1 = Blockly.JavaScript.valueToCode(block, 'ADD1', Blockly.JavaScript.ORDER_NONE) || '\'\''; code = 'String(' + argument0 + ') + String(' + argument1 + ')'; return [code, Blockly.JavaScript.ORDER_ADDITION]; } else { code = new Array(block.itemCount_); for (var n = 0; n < block.itemCount_; n++) { code[n] = Blockly.JavaScript.valueToCode(block, 'ADD' + n, Blockly.JavaScript.ORDER_COMMA) || '\'\''; } code = '[' + code.join(',') + '].join(\'\')'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; } }; Blockly.JavaScript['text_append'] = function(block) { // Append to a variable in place. var varName = Blockly.JavaScript.variableDB_.getName( block.getTitleValue('VAR'), Blockly.Variables.NAME_TYPE); var argument0 = Blockly.JavaScript.valueToCode(block, 'TEXT', Blockly.JavaScript.ORDER_NONE) || '\'\''; return varName + ' = String(' + varName + ') + String(' + argument0 + ');\n'; }; Blockly.JavaScript['text_length'] = function(block) { // String length. var argument0 = Blockly.JavaScript.valueToCode(block, 'VALUE', Blockly.JavaScript.ORDER_FUNCTION_CALL) || '\'\''; return [argument0 + '.length', Blockly.JavaScript.ORDER_MEMBER]; }; Blockly.JavaScript['text_isEmpty'] = function(block) { // Is the string null? var argument0 = Blockly.JavaScript.valueToCode(block, 'VALUE', Blockly.JavaScript.ORDER_MEMBER) || '\'\''; return ['!' + argument0, Blockly.JavaScript.ORDER_LOGICAL_NOT]; }; Blockly.JavaScript['text_indexOf'] = function(block) { // Search the text for a substring. var operator = block.getTitleValue('END') == 'FIRST' ? 'indexOf' : 'lastIndexOf'; var argument0 = Blockly.JavaScript.valueToCode(block, 'FIND', Blockly.JavaScript.ORDER_NONE) || '\'\''; var argument1 = Blockly.JavaScript.valueToCode(block, 'VALUE', Blockly.JavaScript.ORDER_MEMBER) || '\'\''; var code = argument1 + '.' + operator + '(' + argument0 + ') + 1'; return [code, Blockly.JavaScript.ORDER_MEMBER]; }; Blockly.JavaScript['text_charAt'] = function(block) { // Get letter at index. // Note: Until January 2013 this block did not have the WHERE input. var where = block.getTitleValue('WHERE') || 'FROM_START'; var at = Blockly.JavaScript.valueToCode(block, 'AT', Blockly.JavaScript.ORDER_UNARY_NEGATION) || '1'; var text = Blockly.JavaScript.valueToCode(block, 'VALUE', Blockly.JavaScript.ORDER_MEMBER) || '\'\''; switch (where) { case 'FIRST': var code = text + '.charAt(0)'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; case 'LAST': var code = text + '.slice(-1)'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; case 'FROM_START': // Blockly uses one-based indicies. if (Blockly.isNumber(at)) { // If the index is a naked number, decrement it right now. at = parseFloat(at) - 1; } else { // If the index is dynamic, decrement it in code. at += ' - 1'; } var code = text + '.charAt(' + at + ')'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; case 'FROM_END': var code = text + '.slice(-' + at + ').charAt(0)'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; case 'RANDOM': if (!Blockly.JavaScript.definitions_['text_random_letter']) { var functionName = Blockly.JavaScript.variableDB_.getDistinctName( 'text_random_letter', Blockly.Generator.NAME_TYPE); Blockly.JavaScript.text_charAt.text_random_letter = functionName; var func = []; func.push('function ' + functionName + '(text) {'); func.push(' var x = Math.floor(Math.random() * text.length);'); func.push(' return text[x];'); func.push('}'); Blockly.JavaScript.definitions_['text_random_letter'] = func.join('\n'); } code = Blockly.JavaScript.text_charAt.text_random_letter + '(' + text + ')'; return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; } throw 'Unhandled option (text_charAt).'; }; Blockly.JavaScript['text_getSubstring'] = function(block) { // Get substring. var text = Blockly.JavaScript.valueToCode(block, 'STRING', Blockly.JavaScript.ORDER_MEMBER) || '[]'; var where1 = block.getTitleValue('WHERE1'); var where2 = block.getTitleValue('WHERE2'); var at1 = Blockly.JavaScript.valueToCode(block, 'AT1', Blockly.JavaScript.ORDER_NONE) || '1'; var at2 = Blockly.JavaScript.valueToCode(block, 'AT2', Blockly.JavaScript.ORDER_NONE) || '1'; if (where1 == 'FIRST' && where2 == 'LAST') { var code = text; } else { if (!Blockly.JavaScript.definitions_['text_get_substring']) { var functionName = Blockly.JavaScript.variableDB_.getDistinctName( 'text_get_substring', Blockly.Generator.NAME_TYPE); Blockly.JavaScript.text_getSubstring.func = functionName; var func = []; func.push('function ' + functionName + '(text, where1, at1, where2, at2) {'); func.push(' function getAt(where, at) {'); func.push(' if (where == \'FROM_START\') {'); func.push(' at--;'); func.push(' } else if (where == \'FROM_END\') {'); func.push(' at = text.length - at;'); func.push(' } else if (where == \'FIRST\') {'); func.push(' at = 0;'); func.push(' } else if (where == \'LAST\') {'); func.push(' at = text.length - 1;'); func.push(' } else {'); func.push(' throw \'Unhandled option (text_getSubstring).\';'); func.push(' }'); func.push(' return at;'); func.push(' }'); func.push(' at1 = getAt(where1, at1);'); func.push(' at2 = getAt(where2, at2) + 1;'); func.push(' return text.slice(at1, at2);'); func.push('}'); Blockly.JavaScript.definitions_['text_get_substring'] = func.join('\n'); } var code = Blockly.JavaScript.text_getSubstring.func + '(' + text + ', \'' + where1 + '\', ' + at1 + ', \'' + where2 + '\', ' + at2 + ')'; } return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; }; Blockly.JavaScript['text_changeCase'] = function(block) { // Change capitalization. var OPERATORS = { UPPERCASE: '.toUpperCase()', LOWERCASE: '.toLowerCase()', TITLECASE: null }; var operator = OPERATORS[block.getTitleValue('CASE')]; var code; if (operator) { // Upper and lower case are functions built into JavaScript. var argument0 = Blockly.JavaScript.valueToCode(block, 'TEXT', Blockly.JavaScript.ORDER_MEMBER) || '\'\''; code = argument0 + operator; } else { if (!Blockly.JavaScript.definitions_['text_toTitleCase']) { // Title case is not a native JavaScript function. Define one. var functionName = Blockly.JavaScript.variableDB_.getDistinctName( 'text_toTitleCase', Blockly.Generator.NAME_TYPE); Blockly.JavaScript.text_changeCase.toTitleCase = functionName; var func = []; func.push('function ' + functionName + '(str) {'); func.push(' return str.replace(/\\S+/g,'); func.push(' function(txt) {return txt[0].toUpperCase() + ' + 'txt.substring(1).toLowerCase();});'); func.push('}'); Blockly.JavaScript.definitions_['text_toTitleCase'] = func.join('\n'); } var argument0 = Blockly.JavaScript.valueToCode(block, 'TEXT', Blockly.JavaScript.ORDER_NONE) || '\'\''; code = Blockly.JavaScript.text_changeCase.toTitleCase + '(' + argument0 + ')'; } return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; }; Blockly.JavaScript['text_trim'] = function(block) { // Trim spaces. var OPERATORS = { LEFT: '.trimLeft()', RIGHT: '.trimRight()', BOTH: '.trim()' }; var operator = OPERATORS[block.getTitleValue('MODE')]; var argument0 = Blockly.JavaScript.valueToCode(block, 'TEXT', Blockly.JavaScript.ORDER_MEMBER) || '\'\''; return [argument0 + operator, Blockly.JavaScript.ORDER_FUNCTION_CALL]; }; Blockly.JavaScript['text_print'] = function(block) { // Print statement. var argument0 = Blockly.JavaScript.valueToCode(block, 'TEXT', Blockly.JavaScript.ORDER_NONE) || '\'\''; return 'window.alert(' + argument0 + ');\n'; }; Blockly.JavaScript['text_prompt'] = function(block) { // Prompt function. var msg = Blockly.JavaScript.quote_(block.getTitleValue('TEXT')); var code = 'window.prompt(' + msg + ')'; var toNumber = block.getTitleValue('TYPE') == 'NUMBER'; if (toNumber) { code = 'parseFloat(' + code + ')'; } return [code, Blockly.JavaScript.ORDER_FUNCTION_CALL]; };