| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 | /** * Blockly Demos: Code * * 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 JavaScript for Blockly's Code demo. * @author fraser@google.com (Neil Fraser) */'use strict';/** * Create a namespace for the application. */var Code = {};/** * Lookup for names of supported languages.  Keys should be in ISO 639 format. */Code.LANGUAGE_NAME = {  'ar': 'العربية',  'be-tarask': 'Taraškievica',  'br': 'Brezhoneg',  'ca': 'Català',  'cs': 'Česky',  'da': 'Dansk',  'de': 'Deutsch',  'el': 'Ελληνικά',  'en': 'English',  'es': 'Español',  'et': 'Eesti',  'fa': 'فارسی',  'fr': 'Français',  'he': 'עברית',  'hrx': 'Hunsrik',  'hu': 'Magyar',  'ia': 'Interlingua',  'is': 'Íslenska',  'it': 'Italiano',  'ja': '日本語',  'ko': '한국어',  'mk': 'Македонски',  'ms': 'Bahasa Melayu',  'nb': 'Norsk Bokmål',  'nl': 'Nederlands, Vlaams',  'oc': 'Lenga d\'òc',  'pl': 'Polski',  'pms': 'Piemontèis',  'pt-br': 'Português Brasileiro',  'ro': 'Română',  'ru': 'Русский',  'sc': 'Sardu',  'sk': 'Slovenčina',  'sr': 'Српски',  'sv': 'Svenska',  'ta': 'தமிழ்',  'th': 'ภาษาไทย',  'tlh': 'tlhIngan Hol',  'tr': 'Türkçe',  'uk': 'Українська',  'vi': 'Tiếng Việt',  'zh-hans': '简体中文',  'zh-hant': '正體中文'};/** * List of RTL languages. */Code.LANGUAGE_RTL = ['ar', 'fa', 'he', 'lki'];/** * Blockly's main workspace. * @type {Blockly.WorkspaceSvg} */Code.workspace = null;/** * Extracts a parameter from the URL. * If the parameter is absent default_value is returned. * @param {string} name The name of the parameter. * @param {string} defaultValue Value to return if paramater not found. * @return {string} The parameter value or the default value if not found. */Code.getStringParamFromUrl = function(name, defaultValue) {  var val = location.search.match(new RegExp('[?&]' + name + '=([^&]+)'));  return val ? decodeURIComponent(val[1].replace(/\+/g, '%20')) : defaultValue;};/** * Get the language of this user from the URL. * @return {string} User's language. */Code.getLang = function() {  var lang = Code.getStringParamFromUrl('lang', '');  if (Code.LANGUAGE_NAME[lang] === undefined) {    // Default to English.    lang = 'en';  }  return lang;};/** * Is the current language (Code.LANG) an RTL language? * @return {boolean} True if RTL, false if LTR. */Code.isRtl = function() {  return Code.LANGUAGE_RTL.indexOf(Code.LANG) != -1;};/** * Load blocks saved on App Engine Storage or in session/local storage. * @param {string} defaultXml Text representation of default blocks. */Code.loadBlocks = function(defaultXml) {  try {    var loadOnce = window.sessionStorage.loadOnceBlocks;  } catch(e) {    // Firefox sometimes throws a SecurityError when accessing sessionStorage.    // Restarting Firefox fixes this, so it looks like a bug.    var loadOnce = null;  }  if ('BlocklyStorage' in window && window.location.hash.length > 1) {    // An href with #key trigers an AJAX call to retrieve saved blocks.    BlocklyStorage.retrieveXml(window.location.hash.substring(1));  } else if (loadOnce) {    // Language switching stores the blocks during the reload.    delete window.sessionStorage.loadOnceBlocks;    var xml = Blockly.Xml.textToDom(loadOnce);    Blockly.Xml.domToWorkspace(xml, Code.workspace);  } else if (defaultXml) {    // Load the editor with default starting blocks.    var xml = Blockly.Xml.textToDom(defaultXml);    Blockly.Xml.domToWorkspace(xml, Code.workspace);  } else if ('BlocklyStorage' in window) {    // Restore saved blocks in a separate thread so that subsequent    // initialization is not affected from a failed load.    window.setTimeout(BlocklyStorage.restoreBlocks, 0);  }};/** * Save the blocks and reload with a different language. */Code.changeLanguage = function() {  // Store the blocks for the duration of the reload.  // This should be skipped for the index page, which has no blocks and does  // not load Blockly.  // MSIE 11 does not support sessionStorage on file:// URLs.  if (typeof Blockly != 'undefined' && window.sessionStorage) {    var xml = Blockly.Xml.workspaceToDom(Code.workspace);    var text = Blockly.Xml.domToText(xml);    window.sessionStorage.loadOnceBlocks = text;  }  var languageMenu = document.getElementById('languageMenu');  var newLang = encodeURIComponent(      languageMenu.options[languageMenu.selectedIndex].value);  var search = window.location.search;  if (search.length <= 1) {    search = '?lang=' + newLang;  } else if (search.match(/[?&]lang=[^&]*/)) {    search = search.replace(/([?&]lang=)[^&]*/, '$1' + newLang);  } else {    search = search.replace(/\?/, '?lang=' + newLang + '&');  }  window.location = window.location.protocol + '//' +      window.location.host + window.location.pathname + search;};/** * Bind a function to a button's click event. * On touch enabled browsers, ontouchend is treated as equivalent to onclick. * @param {!Element|string} el Button element or ID thereof. * @param {!Function} func Event handler to bind. */Code.bindClick = function(el, func) {  if (typeof el == 'string') {    el = document.getElementById(el);  }  el.addEventListener('click', func, true);  el.addEventListener('touchend', func, true);};/** * Load the Prettify CSS and JavaScript. */Code.importPrettify = function() {  //<link rel="stylesheet" href="../prettify.css">  //<script src="../prettify.js"></script>  var link = document.createElement('link');  link.setAttribute('rel', 'stylesheet');  link.setAttribute('href', '../prettify.css');  document.head.appendChild(link);  var script = document.createElement('script');  script.setAttribute('src', '../prettify.js');  document.head.appendChild(script);};/** * Compute the absolute coordinates and dimensions of an HTML element. * @param {!Element} element Element to match. * @return {!Object} Contains height, width, x, and y properties. * @private */Code.getBBox_ = function(element) {  var height = element.offsetHeight;  var width = element.offsetWidth;  var x = 0;  var y = 0;  do {    x += element.offsetLeft;    y += element.offsetTop;    element = element.offsetParent;  } while (element);  return {    height: height,    width: width,    x: x,    y: y  };};/** * User's language (e.g. "en"). * @type {string} */Code.LANG = Code.getLang();/** * List of tab names. * @private */Code.TABS_ = ['blocks', 'javascript', 'php', 'python', 'dart', 'lua', 'xml'];Code.selected = 'blocks';/** * Switch the visible pane when a tab is clicked. * @param {string} clickedName Name of tab clicked. */Code.tabClick = function(clickedName) {  // If the XML tab was open, save and render the content.  if (document.getElementById('tab_xml').className == 'tabon') {    var xmlTextarea = document.getElementById('content_xml');    var xmlText = xmlTextarea.value;    var xmlDom = null;    try {      xmlDom = Blockly.Xml.textToDom(xmlText);    } catch (e) {      var q =          window.confirm(MSG['badXml'].replace('%1', e));      if (!q) {        // Leave the user on the XML tab.        return;      }    }    if (xmlDom) {      Code.workspace.clear();      Blockly.Xml.domToWorkspace(xmlDom, Code.workspace);    }  }  if (document.getElementById('tab_blocks').className == 'tabon') {    Code.workspace.setVisible(false);  }  // Deselect all tabs and hide all panes.  for (var i = 0; i < Code.TABS_.length; i++) {    var name = Code.TABS_[i];    document.getElementById('tab_' + name).className = 'taboff';    document.getElementById('content_' + name).style.visibility = 'hidden';  }  // Select the active tab.  Code.selected = clickedName;  document.getElementById('tab_' + clickedName).className = 'tabon';  // Show the selected pane.  document.getElementById('content_' + clickedName).style.visibility =      'visible';  Code.renderContent();  if (clickedName == 'blocks') {    Code.workspace.setVisible(true);  }  Blockly.svgResize(Code.workspace);};/** * Populate the currently selected pane with content generated from the blocks. */Code.renderContent = function() {  var content = document.getElementById('content_' + Code.selected);  // Initialize the pane.  if (content.id == 'content_xml') {    var xmlTextarea = document.getElementById('content_xml');    var xmlDom = Blockly.Xml.workspaceToDom(Code.workspace);    var xmlText = Blockly.Xml.domToPrettyText(xmlDom);    xmlTextarea.value = xmlText;    xmlTextarea.focus();  } else if (content.id == 'content_javascript') {    var code = Blockly.JavaScript.workspaceToCode(Code.workspace);    content.textContent = code;    if (typeof prettyPrintOne == 'function') {      code = content.textContent;      code = prettyPrintOne(code, 'js');      content.innerHTML = code;    }  } else if (content.id == 'content_python') {    code = Blockly.Python.workspaceToCode(Code.workspace);    content.textContent = code;    if (typeof prettyPrintOne == 'function') {      code = content.textContent;      code = prettyPrintOne(code, 'py');      content.innerHTML = code;    }  } else if (content.id == 'content_php') {    code = Blockly.PHP.workspaceToCode(Code.workspace);    content.textContent = code;    if (typeof prettyPrintOne == 'function') {      code = content.textContent;      code = prettyPrintOne(code, 'php');      content.innerHTML = code;    }  } else if (content.id == 'content_dart') {    code = Blockly.Dart.workspaceToCode(Code.workspace);    content.textContent = code;    if (typeof prettyPrintOne == 'function') {      code = content.textContent;      code = prettyPrintOne(code, 'dart');      content.innerHTML = code;    }  } else if (content.id == 'content_lua') {    code = Blockly.Lua.workspaceToCode(Code.workspace);    content.textContent = code;    if (typeof prettyPrintOne == 'function') {      code = content.textContent;      code = prettyPrintOne(code, 'lua');      content.innerHTML = code;    }  }};/** * Initialize Blockly.  Called on page load. */Code.init = function() {  Code.initLanguage();  var rtl = Code.isRtl();  var container = document.getElementById('content_area');  var onresize = function(e) {    var bBox = Code.getBBox_(container);    for (var i = 0; i < Code.TABS_.length; i++) {      var el = document.getElementById('content_' + Code.TABS_[i]);      el.style.top = bBox.y + 'px';      el.style.left = bBox.x + 'px';      // Height and width need to be set, read back, then set again to      // compensate for scrollbars.      el.style.height = bBox.height + 'px';      el.style.height = (2 * bBox.height - el.offsetHeight) + 'px';      el.style.width = bBox.width + 'px';      el.style.width = (2 * bBox.width - el.offsetWidth) + 'px';    }    // Make the 'Blocks' tab line up with the toolbox.    if (Code.workspace && Code.workspace.toolbox_.width) {      document.getElementById('tab_blocks').style.minWidth =          (Code.workspace.toolbox_.width - 38) + 'px';          // Account for the 19 pixel margin and on each side.    }  };  window.addEventListener('resize', onresize, false);  // Interpolate translated messages into toolbox.  var toolboxText = document.getElementById('toolbox').outerHTML;  toolboxText = toolboxText.replace(/{(\w+)}/g,      function(m, p1) {return MSG[p1]});  var toolboxXml = Blockly.Xml.textToDom(toolboxText);  Code.workspace = Blockly.inject('content_blocks',      {grid:          {spacing: 25,           length: 3,           colour: '#ccc',           snap: true},       media: '../../media/',       rtl: rtl,       toolbox: toolboxXml,       zoom:           {controls: true,            wheel: true}      });  // Add to reserved word list: Local variables in execution environment (runJS)  // and the infinite loop detection function.  Blockly.JavaScript.addReservedWords('code,timeouts,checkTimeout');  Code.loadBlocks('');  if ('BlocklyStorage' in window) {    // Hook a save function onto unload.    BlocklyStorage.backupOnUnload(Code.workspace);  }  Code.tabClick(Code.selected);  Code.bindClick('trashButton',      function() {Code.discard(); Code.renderContent();});  Code.bindClick('runButton', Code.runJS);  // Disable the link button if page isn't backed by App Engine storage.  var linkButton = document.getElementById('linkButton');  if ('BlocklyStorage' in window) {    BlocklyStorage['HTTPREQUEST_ERROR'] = MSG['httpRequestError'];    BlocklyStorage['LINK_ALERT'] = MSG['linkAlert'];    BlocklyStorage['HASH_ERROR'] = MSG['hashError'];    BlocklyStorage['XML_ERROR'] = MSG['xmlError'];    Code.bindClick(linkButton,        function() {BlocklyStorage.link(Code.workspace);});  } else if (linkButton) {    linkButton.className = 'disabled';  }  for (var i = 0; i < Code.TABS_.length; i++) {    var name = Code.TABS_[i];    Code.bindClick('tab_' + name,        function(name_) {return function() {Code.tabClick(name_);};}(name));  }  onresize();  Blockly.svgResize(Code.workspace);  // Lazy-load the syntax-highlighting.  window.setTimeout(Code.importPrettify, 1);};/** * Initialize the page language. */Code.initLanguage = function() {  // Set the HTML's language and direction.  var rtl = Code.isRtl();  document.dir = rtl ? 'rtl' : 'ltr';  document.head.parentElement.setAttribute('lang', Code.LANG);  // Sort languages alphabetically.  var languages = [];  for (var lang in Code.LANGUAGE_NAME) {    languages.push([Code.LANGUAGE_NAME[lang], lang]);  }  var comp = function(a, b) {    // Sort based on first argument ('English', 'Русский', '简体字', etc).    if (a[0] > b[0]) return 1;    if (a[0] < b[0]) return -1;    return 0;  };  languages.sort(comp);  // Populate the language selection menu.  var languageMenu = document.getElementById('languageMenu');  languageMenu.options.length = 0;  for (var i = 0; i < languages.length; i++) {    var tuple = languages[i];    var lang = tuple[tuple.length - 1];    var option = new Option(tuple[0], lang);    if (lang == Code.LANG) {      option.selected = true;    }    languageMenu.options.add(option);  }  languageMenu.addEventListener('change', Code.changeLanguage, true);  // Inject language strings.  document.title += ' ' + MSG['title'];  document.getElementById('title').textContent = MSG['title'];  document.getElementById('tab_blocks').textContent = MSG['blocks'];  document.getElementById('linkButton').title = MSG['linkTooltip'];  document.getElementById('runButton').title = MSG['runTooltip'];  document.getElementById('trashButton').title = MSG['trashTooltip'];};/** * Execute the user's code. * Just a quick and dirty eval.  Catch infinite loops. */Code.runJS = function() {  Blockly.JavaScript.INFINITE_LOOP_TRAP = '  checkTimeout();\n';  var timeouts = 0;  var checkTimeout = function() {    if (timeouts++ > 1000000) {      throw MSG['timeout'];    }  };  var code = Blockly.JavaScript.workspaceToCode(Code.workspace);  Blockly.JavaScript.INFINITE_LOOP_TRAP = null;  try {    eval(code);  } catch (e) {    alert(MSG['badCode'].replace('%1', e));  }};/** * Discard all blocks from the workspace. */Code.discard = function() {  var count = Code.workspace.getAllBlocks().length;  if (count < 2 ||      window.confirm(Blockly.Msg.DELETE_ALL_BLOCKS.replace('%1', count))) {    Code.workspace.clear();    if (window.location.hash) {      window.location.hash = '';    }  }};// Load the Code demo's language strings.document.write('<script src="msg/' + Code.LANG + '.js"></script>\n');// Load Blockly's language strings.document.write('<script src="../../msg/js/' + Code.LANG + '.js"></script>\n');window.addEventListener('load', Code.init);
 |