// Copyright (c) IPython Development Team. // Distributed under the terms of the Modified BSD License. define([ 'base/js/namespace', 'jquery', 'codemirror/lib/codemirror', 'rsvp', ], function(IPython, $, CodeMirror, rsvp){ "use strict"; IPython.load_extensions = function () { // load one or more IPython notebook extensions with requirejs var extensions = []; var extension_names = arguments; for (var i = 0; i < extension_names.length; i++) { extensions.push("nbextensions/" + arguments[i]); } require(extensions, function () { for (var i = 0; i < arguments.length; i++) { var ext = arguments[i]; var ext_name = extension_names[i]; // success callback console.log("Loaded extension: " + ext_name); if (ext && ext.load_ipython_extension !== undefined) { ext.load_ipython_extension(); } } }, function (err) { // failure callback console.log("Failed to load extension(s):", err.requireModules, err); } ); }; //============================================================================ // Cross-browser RegEx Split //============================================================================ // This code has been MODIFIED from the code licensed below to not replace the // default browser split. The license is reproduced here. // see http://blog.stevenlevithan.com/archives/cross-browser-split for more info: /*! * Cross-Browser Split 1.1.1 * Copyright 2007-2012 Steven Levithan * Available under the MIT License * ECMAScript compliant, uniform cross-browser split method */ /** * Splits a string into an array of strings using a regex or string * separator. Matches of the separator are not included in the result array. * However, if `separator` is a regex that contains capturing groups, * backreferences are spliced into the result each time `separator` is * matched. Fixes browser bugs compared to the native * `String.prototype.split` and can be used reliably cross-browser. * @param {String} str String to split. * @param {RegExp|String} separator Regex or string to use for separating * the string. * @param {Number} [limit] Maximum number of items to include in the result * array. * @returns {Array} Array of substrings. * @example * * // Basic use * regex_split('a b c d', ' '); * // -> ['a', 'b', 'c', 'd'] * * // With limit * regex_split('a b c d', ' ', 2); * // -> ['a', 'b'] * * // Backreferences in result array * regex_split('..word1 word2..', /([a-z]+)(\d+)/i); * // -> ['..', 'word', '1', ' ', 'word', '2', '..'] */ var regex_split = function (str, separator, limit) { // If `separator` is not a regex, use `split` if (Object.prototype.toString.call(separator) !== "[object RegExp]") { return split.call(str, separator, limit); } var output = [], flags = (separator.ignoreCase ? "i" : "") + (separator.multiline ? "m" : "") + (separator.extended ? "x" : "") + // Proposed for ES6 (separator.sticky ? "y" : ""), // Firefox 3+ lastLastIndex = 0, // Make `global` and avoid `lastIndex` issues by working with a copy separator = new RegExp(separator.source, flags + "g"), separator2, match, lastIndex, lastLength; str += ""; // Type-convert var compliantExecNpcg = typeof(/()??/.exec("")[1]) === "undefined"; if (!compliantExecNpcg) { // Doesn't need flags gy, but they don't hurt separator2 = new RegExp("^" + separator.source + "$(?!\\s)", flags); } /* Values for `limit`, per the spec: * If undefined: 4294967295 // Math.pow(2, 32) - 1 * If 0, Infinity, or NaN: 0 * If positive number: limit = Math.floor(limit); if (limit > 4294967295) limit -= 4294967296; * If negative number: 4294967296 - Math.floor(Math.abs(limit)) * If other: Type-convert, then use the above rules */ limit = typeof(limit) === "undefined" ? -1 >>> 0 : // Math.pow(2, 32) - 1 limit >>> 0; // ToUint32(limit) while (match = separator.exec(str)) { // `separator.lastIndex` is not reliable cross-browser lastIndex = match.index + match[0].length; if (lastIndex > lastLastIndex) { output.push(str.slice(lastLastIndex, match.index)); // Fix browsers whose `exec` methods don't consistently return `undefined` for // nonparticipating capturing groups if (!compliantExecNpcg && match.length > 1) { match[0].replace(separator2, function () { for (var i = 1; i < arguments.length - 2; i++) { if (typeof(arguments[i]) === "undefined") { match[i] = undefined; } } }); } if (match.length > 1 && match.index < str.length) { Array.prototype.push.apply(output, match.slice(1)); } lastLength = match[0].length; lastLastIndex = lastIndex; if (output.length >= limit) { break; } } if (separator.lastIndex === match.index) { separator.lastIndex++; // Avoid an infinite loop } } if (lastLastIndex === str.length) { if (lastLength || !separator.test("")) { output.push(""); } } else { output.push(str.slice(lastLastIndex)); } return output.length > limit ? output.slice(0, limit) : output; }; //============================================================================ // End contributed Cross-browser RegEx Split //============================================================================ var uuid = function () { // http://www.ietf.org/rfc/rfc4122.txt var s = []; var hexDigits = "0123456789ABCDEF"; for (var i = 0; i < 32; i++) { s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1); } s[12] = "4"; // bits 12-15 of the time_hi_and_version field to 0010 s[16] = hexDigits.substr((s[16] & 0x3) | 0x8, 1); // bits 6-7 of the clock_seq_hi_and_reserved to 01 var uuid = s.join(""); return uuid; }; //Fix raw text to parse correctly in crazy XML function xmlencode(string) { return string.replace(/\&/g,'&'+'amp;') .replace(//g,'&'+'gt;') .replace(/\'/g,'&'+'apos;') .replace(/\"/g,'&'+'quot;') .replace(/`/g,'&'+'#96;'); } //Map from terminal commands to CSS classes var ansi_colormap = { "01":"ansibold", "30":"ansiblack", "31":"ansired", "32":"ansigreen", "33":"ansiyellow", "34":"ansiblue", "35":"ansipurple", "36":"ansicyan", "37":"ansigray", "40":"ansibgblack", "41":"ansibgred", "42":"ansibggreen", "43":"ansibgyellow", "44":"ansibgblue", "45":"ansibgpurple", "46":"ansibgcyan", "47":"ansibggray" }; function _process_numbers(attrs, numbers) { // process ansi escapes var n = numbers.shift(); if (ansi_colormap[n]) { if ( ! attrs["class"] ) { attrs["class"] = ansi_colormap[n]; } else { attrs["class"] += " " + ansi_colormap[n]; } } else if (n == "38" || n == "48") { // VT100 256 color or 24 bit RGB if (numbers.length < 2) { console.log("Not enough fields for VT100 color", numbers); return; } var index_or_rgb = numbers.shift(); var r,g,b; if (index_or_rgb == "5") { // 256 color var idx = parseInt(numbers.shift()); if (idx < 16) { // indexed ANSI // ignore bright / non-bright distinction idx = idx % 8; var ansiclass = ansi_colormap[n[0] + (idx % 8).toString()]; if ( ! attrs["class"] ) { attrs["class"] = ansiclass; } else { attrs["class"] += " " + ansiclass; } return; } else if (idx < 232) { // 216 color 6x6x6 RGB idx = idx - 16; b = idx % 6; g = Math.floor(idx / 6) % 6; r = Math.floor(idx / 36) % 6; // convert to rgb r = (r * 51); g = (g * 51); b = (b * 51); } else { // grayscale idx = idx - 231; // it's 1-24 and should *not* include black or white, // so a 26 point scale r = g = b = Math.floor(idx * 256 / 26); } } else if (index_or_rgb == "2") { // Simple 24 bit RGB if (numbers.length > 3) { console.log("Not enough fields for RGB", numbers); return; } r = numbers.shift(); g = numbers.shift(); b = numbers.shift(); } else { console.log("unrecognized control", numbers); return; } if (r !== undefined) { // apply the rgb color var line; if (n == "38") { line = "color: "; } else { line = "background-color: "; } line = line + "rgb(" + r + "," + g + "," + b + ");"; if ( !attrs.style ) { attrs.style = line; } else { attrs.style += " " + line; } } } } function ansispan(str) { // ansispan function adapted from github.com/mmalecki/ansispan (MIT License) // regular ansi escapes (using the table above) var is_open = false; return str.replace(/\033\[(0?[01]|22|39)?([;\d]+)?m/g, function(match, prefix, pattern) { if (!pattern) { // [(01|22|39|)m close spans if (is_open) { is_open = false; return ""; } else { return ""; } } else { is_open = true; // consume sequence of color escapes var numbers = pattern.match(/\d+/g); var attrs = {}; while (numbers.length > 0) { _process_numbers(attrs, numbers); } var span = ""; } }); } // Transform ANSI color escape codes into HTML tags with css // classes listed in the above ansi_colormap object. The actual color used // are set in the css file. function fixConsole(txt) { txt = xmlencode(txt); var re = /\033\[([\dA-Fa-f;]*?)m/; var opened = false; var cmds = []; var opener = ""; var closer = ""; // Strip all ANSI codes that are not color related. Matches // all ANSI codes that do not end with "m". var ignored_re = /(?=(\033\[[\d;=]*[a-ln-zA-Z]{1}))\1(?!m)/g; txt = txt.replace(ignored_re, ""); // color ansi codes txt = ansispan(txt); return txt; } // Remove chunks that should be overridden by the effect of // carriage return characters function fixCarriageReturn(txt) { var tmp = txt; do { txt = tmp; tmp = txt.replace(/\r+\n/gm, '\n'); // \r followed by \n --> newline tmp = tmp.replace(/^.*\r+/gm, ''); // Other \r --> clear line } while (tmp.length < txt.length); return txt; } // Locate any URLs and convert them to a anchor tag function autoLinkUrls(txt) { return txt.replace(/(^|\s)(https?|ftp)(:[^'">\s]+)/gi, "$1$2$3"); } var points_to_pixels = function (points) { // A reasonably good way of converting between points and pixels. var test = $('
'); $(body).append(test); var pixel_per_point = test.width()/10000; test.remove(); return Math.floor(points*pixel_per_point); }; var always_new = function (constructor) { // wrapper around contructor to avoid requiring `var a = new constructor()` // useful for passing constructors as callbacks, // not for programmer laziness. // from http://programmers.stackexchange.com/questions/118798 return function () { var obj = Object.create(constructor.prototype); constructor.apply(obj, arguments); return obj; }; }; var url_path_join = function () { // join a sequence of url components with '/' var url = ''; for (var i = 0; i < arguments.length; i++) { if (arguments[i] === '') { continue; } if (url.length > 0 && url[url.length-1] != '/') { url = url + '/' + arguments[i]; } else { url = url + arguments[i]; } } url = url.replace(/\/\/+/, '/'); return url; }; var url_path_split = function (path) { // Like os.path.split for URLs. // Always returns two strings, the directory path and the base filename var idx = path.lastIndexOf('/'); if (idx === -1) { return ['', path]; } else { return [ path.slice(0, idx), path.slice(idx + 1) ]; } }; var parse_url = function (url) { // an `a` element with an href allows attr-access to the parsed segments of a URL // a = parse_url("http://localhost:8888/path/name#hash") // a.protocol = "http:" // a.host = "localhost:8888" // a.hostname = "localhost" // a.port = 8888 // a.pathname = "/path/name" // a.hash = "#hash" var a = document.createElement("a"); a.href = url; return a; }; var encode_uri_components = function (uri) { // encode just the components of a multi-segment uri, // leaving '/' separators return uri.split('/').map(encodeURIComponent).join('/'); }; var url_join_encode = function () { // join a sequence of url components with '/', // encoding each component with encodeURIComponent return encode_uri_components(url_path_join.apply(null, arguments)); }; var splitext = function (filename) { // mimic Python os.path.splitext // Returns ['base', '.ext'] var idx = filename.lastIndexOf('.'); if (idx > 0) { return [filename.slice(0, idx), filename.slice(idx)]; } else { return [filename, '']; } }; var escape_html = function (text) { // escape text to HTML return $("
").text(text).html(); }; var get_body_data = function(key) { // get a url-encoded item from body.data and decode it // we should never have any encoded URLs anywhere else in code // until we are building an actual request return decodeURIComponent($('body').data(key)); }; var to_absolute_cursor_pos = function (cm, cursor) { // get the absolute cursor position from CodeMirror's col, ch if (!cursor) { cursor = cm.getCursor(); } var cursor_pos = cursor.ch; for (var i = 0; i < cursor.line; i++) { cursor_pos += cm.getLine(i).length + 1; } return cursor_pos; }; var from_absolute_cursor_pos = function (cm, cursor_pos) { // turn absolute cursor postion into CodeMirror col, ch cursor var i, line; var offset = 0; for (i = 0, line=cm.getLine(i); line !== undefined; i++, line=cm.getLine(i)) { if (offset + line.length < cursor_pos) { offset += line.length + 1; } else { return { line : i, ch : cursor_pos - offset, }; } } // reached end, return endpoint return { ch : line.length - 1, line : i - 1, }; }; // http://stackoverflow.com/questions/2400935/browser-detection-in-javascript var browser = (function() { if (typeof navigator === 'undefined') { // navigator undefined in node return 'None'; } var N= navigator.appName, ua= navigator.userAgent, tem; var M= ua.match(/(opera|chrome|safari|firefox|msie)\/?\s*(\.?\d+(\.\d+)*)/i); if (M && (tem= ua.match(/version\/([\.\d]+)/i)) !== null) M[2]= tem[1]; M= M? [M[1], M[2]]: [N, navigator.appVersion,'-?']; return M; })(); // http://stackoverflow.com/questions/11219582/how-to-detect-my-browser-version-and-operating-system-using-javascript var platform = (function () { if (typeof navigator === 'undefined') { // navigator undefined in node return 'None'; } var OSName="None"; if (navigator.appVersion.indexOf("Win")!=-1) OSName="Windows"; if (navigator.appVersion.indexOf("Mac")!=-1) OSName="MacOS"; if (navigator.appVersion.indexOf("X11")!=-1) OSName="UNIX"; if (navigator.appVersion.indexOf("Linux")!=-1) OSName="Linux"; return OSName; })(); var is_or_has = function (a, b) { // Is b a child of a or a itself? return a.has(b).length !==0 || a.is(b); }; var is_focused = function (e) { // Is element e, or one of its children focused? e = $(e); var target = $(document.activeElement); if (target.length > 0) { if (is_or_has(e, target)) { return true; } else { return false; } } else { return false; } }; var mergeopt = function(_class, options, overwrite){ options = options || {}; overwrite = overwrite || {}; return $.extend(true, {}, _class.options_default, options, overwrite); }; var ajax_error_msg = function (jqXHR) { // Return a JSON error message if there is one, // otherwise the basic HTTP status text. if (jqXHR.responseJSON && jqXHR.responseJSON.traceback) { return jqXHR.responseJSON.traceback; } else if (jqXHR.responseJSON && jqXHR.responseJSON.message) { return jqXHR.responseJSON.message; } else { return jqXHR.statusText; } }; var log_ajax_error = function (jqXHR, status, error) { // log ajax failures with informative messages var msg = "API request failed (" + jqXHR.status + "): "; console.log(jqXHR); msg += ajax_error_msg(jqXHR); console.log(msg); }; var requireCodeMirrorMode = function (mode, callback, errback) { // load a mode with requirejs if (typeof mode != "string") mode = mode.name; if (CodeMirror.modes.hasOwnProperty(mode)) { callback(CodeMirror.modes.mode); return; } require([ // might want to use CodeMirror.modeURL here ['codemirror/mode', mode, mode].join('/'), ], callback, errback ); }; /** Error type for wrapped XHR errors. */ var XHR_ERROR = 'XhrError'; /** * Wraps an AJAX error as an Error object. */ var wrap_ajax_error = function (jqXHR, status, error) { var wrapped_error = new Error(ajax_error_msg(jqXHR)); wrapped_error.name = XHR_ERROR; // provide xhr response wrapped_error.xhr = jqXHR; wrapped_error.xhr_status = status; wrapped_error.xhr_error = error; return wrapped_error; }; var promising_ajax = function(url, settings) { // Like $.ajax, but returning an ES6 promise. success and error settings // will be ignored. return new Promise(function(resolve, reject) { settings.success = function(data, status, jqXHR) { resolve(data); }; settings.error = function(jqXHR, status, error) { log_ajax_error(jqXHR, status, error); reject(wrap_ajax_error(jqXHR, status, error)); }; $.ajax(url, settings); }); }; var WrappedError = function(message, error){ // Wrappable Error class // The Error class doesn't actually act on `this`. Instead it always // returns a new instance of Error. Here we capture that instance so we // can apply it's properties to `this`. var tmp = Error.apply(this, [message]); // Copy the properties of the error over to this. var properties = Object.getOwnPropertyNames(tmp); for (var i = 0; i < properties.length; i++) { this[properties[i]] = tmp[properties[i]]; } // Keep a stack of the original error messages. if (error instanceof WrappedError) { this.error_stack = error.error_stack; } else { this.error_stack = [error]; } this.error_stack.push(tmp); return this; }; WrappedError.prototype = Object.create(Error.prototype, {}); var load_class = function(class_name, module_name, registry) { // Tries to load a class // // Tries to load a class from a module using require.js, if a module // is specified, otherwise tries to load a class from the global // registry, if the global registry is provided. return new rsvp.Promise(function(resolve, reject) { // Try loading the view module using require.js if (module_name) { require([module_name], function(module) { if (module[class_name] === undefined) { reject(new Error('Class '+class_name+' not found in module '+module_name)); } else { resolve(module[class_name]); } }, reject); } else { if (registry && registry[class_name]) { resolve(registry[class_name]); } else { reject(new Error('Class '+class_name+' not found in registry ')); } } }); }; var resolve_dict = function(d) { // Resolve a promiseful dictionary. // Returns a single rsvp.Promise. var keys = Object.keys(d); var values = []; keys.forEach(function(key) { values.push(d[key]); }); return rsvp.Promise.all(values).then(function(v) { d = {}; for(var i=0; i