visualizer-on-tabs 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2016 cheminfo
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,113 @@
1
+ # visualizer-on-tabs
2
+
3
+ Builds a static website that has multiple instances of the visualizer that can communicate with each other.
4
+
5
+ ## CLI usage
6
+
7
+ ```bash
8
+ npx visualizer-on-tabs --config=./config.json --outDir=./out
9
+ ```
10
+
11
+ ## Configuration
12
+
13
+ Example: https://github.com/cheminfo/cheminfo-server-setup/blob/master/doc/on-tabs/config.json
14
+
15
+ ```js
16
+ const config = {
17
+ // Title of the single page app
18
+ title: 'My app',
19
+ // List of default views to load
20
+ possibleViews: {
21
+ Home: {
22
+ url: 'https://couch.cheminfo.org/cheminfo-public/158ef2f0cc85bfc5b4f2d88cff473e83/view.json',
23
+ },
24
+ },
25
+ // Rules on how visualizer view URLs should be rewritten when a tab is opened
26
+ rewriteRules: [
27
+ {
28
+ reg: '^([a-z0-9]+)\\?(.*)$',
29
+ replace: 'https://couch.cheminfo.org/cheminfo-public/$1/view.json?$2',
30
+ },
31
+ {
32
+ reg: '^[a-z0-9]+$',
33
+ replace: 'https://couch.cheminfo.org/cheminfo-public/$&/view.json',
34
+ },
35
+ {
36
+ reg: '^[a-z0-9]+/view.json\\?.*',
37
+ replace: 'https://couch.cheminfo.org/cheminfo-public/$&',
38
+ },
39
+ ],
40
+ // Setting this to true loads all the tabs (in possibleViews) on page load
41
+ // It is discouraged to do this because loading hidden iframes
42
+ // lead to layout issues. Especially in Firefox.
43
+ // When false, only the selected tab is loaded.
44
+ loadHidden: false,
45
+ // The visualizer configuration object that will be passed to each visualizer instance
46
+ visualizerConfig: undefined,
47
+ // The version of the visualizer to load. By default it 'auto', which uses
48
+ // the version stored in the loaded view.
49
+ visualizerVersion: 'auto',
50
+ // Options passed to `makeVisualizerPage`, see https://github.com/cheminfo/react-visualizer
51
+ // Respectively `fallbackVersion` and `cdn`.
52
+ visualizerFallbackVersion: undefined,
53
+ visualizerCDN: undefined,
54
+ };
55
+ ```
56
+
57
+ ## Dev setup
58
+
59
+ Here is how you can test your changes in the visualizer-on-tabs react app.
60
+
61
+ ### Build the page with a custom configuration
62
+
63
+ There is a dev configuration in `./dev.json`, which is used by local scripts to build a working visualizer-on-tabs app.
64
+
65
+ To build in dev mode with automatic rebuild, run:
66
+
67
+ ```bash
68
+ npm run build:dev
69
+ ```
70
+
71
+ To test the production build, run:
72
+
73
+ ```bash
74
+ npm run build
75
+ ```
76
+
77
+ To serve the files produced by the build, run:
78
+
79
+ ```bash
80
+ npm run serve
81
+ ```
82
+
83
+ ## Install and configure visualizer-on-tabs
84
+
85
+ ### Configure a flavor to deploy on-tabs
86
+
87
+ Edit the `flavorLayouts`(/usr/local/flavor-builder/config.json) to specify a deployment method for your flavor. For this, you need to add a new entry which key is your flavor name and value is `visualizer-on-tabs`. Example:
88
+
89
+ ```
90
+ ...
91
+ "flavorLayouts": {
92
+ "720p": "minimal-simple-menu",
93
+ "myflavor":"visualizer-on-tabs"
94
+ }
95
+ ...
96
+ ```
97
+
98
+ Add a new rewriteRule
99
+
100
+ ```
101
+ "visualizerOnTabs": {
102
+ "_default": {
103
+ "rocLogin": {
104
+ "url": "https://myloginserver"
105
+ },
106
+ "rewriteRules": [
107
+ {"reg": "^[^/]+$", "replace": "http://myserver.org/rest-on-couch/db/visualizer/$&/view.json"}
108
+ ]
109
+ }
110
+ ...
111
+ ```
112
+
113
+ You would need to edit a view in this flavor, or launch the build manually with the `--forceUpdate` option.
package/bin/build.js ADDED
@@ -0,0 +1,39 @@
1
+ #!/usr/bin/env node
2
+
3
+ /* eslint-disable no-console */
4
+
5
+ import fs from 'node:fs';
6
+ import path from 'node:path';
7
+ import process from 'node:process';
8
+
9
+ import minimist from 'minimist';
10
+
11
+ import build from '../main/index.js';
12
+
13
+ const argv = minimist(process.argv.slice(2));
14
+
15
+ if (!argv.outDir) {
16
+ console.log(`CLI args:
17
+ --outDir - output directory (required)
18
+ --dev - development mode with file watching
19
+ --config - path to JSON config file
20
+ `);
21
+ throw new Error('The --outDir option is required.');
22
+ }
23
+
24
+ const mode = argv.dev ? 'development' : 'production';
25
+ const watch = !!argv.dev;
26
+ const outDir = argv.outDir;
27
+ let config = {};
28
+ if (argv.config) {
29
+ const configFile = path.resolve(argv.config);
30
+ config = JSON.parse(fs.readFileSync(configFile, 'utf8'));
31
+ }
32
+
33
+ const cleanup = await build({ mode, watch, outDir, config });
34
+
35
+ process.on('SIGINT', async () => {
36
+ console.log('Build cancelled on SIGINT');
37
+ cleanup().catch(console.error);
38
+ process.exit(1);
39
+ });
@@ -0,0 +1,8 @@
1
+ // v1.0.0 taken from http://www.lactame.com/lib/iframe-bridge/HEAD/iframe-bridge.js on 2016-12-19
2
+ // Conversion to base64 with http://decodebase64.com/
3
+ // Added //# sourceURL=iframe-bridge-browser.js at the end of the file to help developer tools
4
+ const iframeBridge =
5
+ 'data:application/javascript;base64,' +
6
+ '/**
 * iframe-bridge - Communicate between iframes and a control page
 * @version v1.0.0
 * @link https://github.com/cheminfo-js/iframe-bridge
 * @license MIT
 */
(function webpackUniversalModuleDefinition(root, factory) {
	if(typeof exports === 'object' && typeof module === 'object')
		module.exports = factory();
	else if(typeof define === 'function' && define.amd)
		define([], factory);
	else if(typeof exports === 'object')
		exports["IframeBridge"] = factory();
	else
		root["IframeBridge"] = factory();
})(this, function() {
return /******/ (function(modules) { // webpackBootstrap
/******/ 	// The module cache
/******/ 	var installedModules = {};
/******/
/******/ 	// The require function
/******/ 	function __webpack_require__(moduleId) {
/******/
/******/ 		// Check if module is in cache
/******/ 		if(installedModules[moduleId]) {
/******/ 			return installedModules[moduleId].exports;
/******/ 		}
/******/ 		// Create a new module (and put it into the cache)
/******/ 		var module = installedModules[moduleId] = {
/******/ 			i: moduleId,
/******/ 			l: false,
/******/ 			exports: {}
/******/ 		};
/******/
/******/ 		// Execute the module function
/******/ 		modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/
/******/ 		// Flag the module as loaded
/******/ 		module.l = true;
/******/
/******/ 		// Return the exports of the module
/******/ 		return module.exports;
/******/ 	}
/******/
/******/
/******/ 	// expose the modules object (__webpack_modules__)
/******/ 	__webpack_require__.m = modules;
/******/
/******/ 	// expose the module cache
/******/ 	__webpack_require__.c = installedModules;
/******/
/******/ 	// define getter function for harmony exports
/******/ 	__webpack_require__.d = function(exports, name, getter) {
/******/ 		if(!__webpack_require__.o(exports, name)) {
/******/ 			Object.defineProperty(exports, name, {
/******/ 				configurable: false,
/******/ 				enumerable: true,
/******/ 				get: getter
/******/ 			});
/******/ 		}
/******/ 	};
/******/
/******/ 	// getDefaultExport function for compatibility with non-harmony modules
/******/ 	__webpack_require__.n = function(module) {
/******/ 		var getter = module && module.__esModule ?
/******/ 			function getDefault() { return module['default']; } :
/******/ 			function getModuleExports() { return module; };
/******/ 		__webpack_require__.d(getter, 'a', getter);
/******/ 		return getter;
/******/ 	};
/******/
/******/ 	// Object.prototype.hasOwnProperty.call
/******/ 	__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
/******/
/******/ 	// __webpack_public_path__
/******/ 	__webpack_require__.p = "";
/******/
/******/ 	// Load entry module and return exports
/******/ 	return __webpack_require__(__webpack_require__.s = 1);
/******/ })
/************************************************************************/
/******/ ([
/* 0 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


// Copyright Joyent, Inc. and other Node contributors.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to permit
// persons to whom the Software is furnished to do so, subject to the
// following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
// USE OR OTHER DEALINGS IN THE SOFTWARE.

function EventEmitter() {
  this._events = this._events || {};
  this._maxListeners = this._maxListeners || undefined;
}
module.exports = EventEmitter;

// Backwards-compat with node 0.10.x
EventEmitter.EventEmitter = EventEmitter;

EventEmitter.prototype._events = undefined;
EventEmitter.prototype._maxListeners = undefined;

// By default EventEmitters will print a warning if more than 10 listeners are
// added to it. This is a useful default which helps finding memory leaks.
EventEmitter.defaultMaxListeners = 10;

// Obviously not all Emitters should be limited to 10. This function allows
// that to be increased. Set to zero for unlimited.
EventEmitter.prototype.setMaxListeners = function (n) {
  if (!isNumber(n) || n < 0 || isNaN(n)) throw TypeError('n must be a positive number');
  this._maxListeners = n;
  return this;
};

EventEmitter.prototype.emit = function (type) {
  var er, handler, len, args, i, listeners;

  if (!this._events) this._events = {};

  // If there is no 'error' event listener then throw.
  if (type === 'error') {
    if (!this._events.error || isObject(this._events.error) && !this._events.error.length) {
      er = arguments[1];
      if (er instanceof Error) {
        throw er; // Unhandled 'error' event
      } else {
        // At least give some kind of context to the user
        var err = new Error('Uncaught, unspecified "error" event. (' + er + ')');
        err.context = er;
        throw err;
      }
    }
  }

  handler = this._events[type];

  if (isUndefined(handler)) return false;

  if (isFunction(handler)) {
    switch (arguments.length) {
      // fast cases
      case 1:
        handler.call(this);
        break;
      case 2:
        handler.call(this, arguments[1]);
        break;
      case 3:
        handler.call(this, arguments[1], arguments[2]);
        break;
      // slower
      default:
        args = Array.prototype.slice.call(arguments, 1);
        handler.apply(this, args);
    }
  } else if (isObject(handler)) {
    args = Array.prototype.slice.call(arguments, 1);
    listeners = handler.slice();
    len = listeners.length;
    for (i = 0; i < len; i++) {
      listeners[i].apply(this, args);
    }
  }

  return true;
};

EventEmitter.prototype.addListener = function (type, listener) {
  var m;

  if (!isFunction(listener)) throw TypeError('listener must be a function');

  if (!this._events) this._events = {};

  // To avoid recursion in the case that type === "newListener"! Before
  // adding it to the listeners, first emit "newListener".
  if (this._events.newListener) this.emit('newListener', type, isFunction(listener.listener) ? listener.listener : listener);

  if (!this._events[type])
    // Optimize the case of one listener. Don't need the extra array object.
    this._events[type] = listener;else if (isObject(this._events[type]))
    // If we've already got an array, just append.
    this._events[type].push(listener);else
    // Adding the second element, need to change to array.
    this._events[type] = [this._events[type], listener];

  // Check for listener leak
  if (isObject(this._events[type]) && !this._events[type].warned) {
    if (!isUndefined(this._maxListeners)) {
      m = this._maxListeners;
    } else {
      m = EventEmitter.defaultMaxListeners;
    }

    if (m && m > 0 && this._events[type].length > m) {
      this._events[type].warned = true;
      console.error('(node) warning: possible EventEmitter memory ' + 'leak detected. %d listeners added. ' + 'Use emitter.setMaxListeners() to increase limit.', this._events[type].length);
      if (typeof console.trace === 'function') {
        // not supported in IE 10
        console.trace();
      }
    }
  }

  return this;
};

EventEmitter.prototype.on = EventEmitter.prototype.addListener;

EventEmitter.prototype.once = function (type, listener) {
  if (!isFunction(listener)) throw TypeError('listener must be a function');

  var fired = false;

  function g() {
    this.removeListener(type, g);

    if (!fired) {
      fired = true;
      listener.apply(this, arguments);
    }
  }

  g.listener = listener;
  this.on(type, g);

  return this;
};

// emits a 'removeListener' event iff the listener was removed
EventEmitter.prototype.removeListener = function (type, listener) {
  var list, position, length, i;

  if (!isFunction(listener)) throw TypeError('listener must be a function');

  if (!this._events || !this._events[type]) return this;

  list = this._events[type];
  length = list.length;
  position = -1;

  if (list === listener || isFunction(list.listener) && list.listener === listener) {
    delete this._events[type];
    if (this._events.removeListener) this.emit('removeListener', type, listener);
  } else if (isObject(list)) {
    for (i = length; i-- > 0;) {
      if (list[i] === listener || list[i].listener && list[i].listener === listener) {
        position = i;
        break;
      }
    }

    if (position < 0) return this;

    if (list.length === 1) {
      list.length = 0;
      delete this._events[type];
    } else {
      list.splice(position, 1);
    }

    if (this._events.removeListener) this.emit('removeListener', type, listener);
  }

  return this;
};

EventEmitter.prototype.removeAllListeners = function (type) {
  var key, listeners;

  if (!this._events) return this;

  // not listening for removeListener, no need to emit
  if (!this._events.removeListener) {
    if (arguments.length === 0) this._events = {};else if (this._events[type]) delete this._events[type];
    return this;
  }

  // emit removeListener for all listeners on all events
  if (arguments.length === 0) {
    for (key in this._events) {
      if (key === 'removeListener') continue;
      this.removeAllListeners(key);
    }
    this.removeAllListeners('removeListener');
    this._events = {};
    return this;
  }

  listeners = this._events[type];

  if (isFunction(listeners)) {
    this.removeListener(type, listeners);
  } else if (listeners) {
    // LIFO order
    while (listeners.length) {
      this.removeListener(type, listeners[listeners.length - 1]);
    }
  }
  delete this._events[type];

  return this;
};

EventEmitter.prototype.listeners = function (type) {
  var ret;
  if (!this._events || !this._events[type]) ret = [];else if (isFunction(this._events[type])) ret = [this._events[type]];else ret = this._events[type].slice();
  return ret;
};

EventEmitter.prototype.listenerCount = function (type) {
  if (this._events) {
    var evlistener = this._events[type];

    if (isFunction(evlistener)) return 1;else if (evlistener) return evlistener.length;
  }
  return 0;
};

EventEmitter.listenerCount = function (emitter, type) {
  return emitter.listenerCount(type);
};

function isFunction(arg) {
  return typeof arg === 'function';
}

function isNumber(arg) {
  return typeof arg === 'number';
}

function isObject(arg) {
  return typeof arg === 'object' && arg !== null;
}

function isUndefined(arg) {
  return arg === void 0;
}

/***/ }),
/* 1 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


var debug = __webpack_require__(2)('iframe-bridge:iframe');
var MessageHandler = __webpack_require__(6);

var messageHandler = new MessageHandler();

messageHandler.init(window.parent);
window.addEventListener('message', function (event) {
    try {
        var data = JSON.parse(event.data);
        debug('message received', data);
        messageHandler.handleMessage(data);
    } catch (e) {}
});

exports.postMessage = function (type, message) {
    return messageHandler.postMessage(type, message);
};

exports.onMessage = function (cb) {
    messageHandler.on('message', cb);
};

exports.ready = function () {
    messageHandler.handlePendingMessages();
};

/***/ }),
/* 2 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";
/* WEBPACK VAR INJECTION */(function(process) {

/**
 * This is the web browser implementation of `debug()`.
 *
 * Expose `debug()` as the module.
 */

exports = module.exports = __webpack_require__(4);
exports.log = log;
exports.formatArgs = formatArgs;
exports.save = save;
exports.load = load;
exports.useColors = useColors;
exports.storage = 'undefined' != typeof chrome && 'undefined' != typeof chrome.storage ? chrome.storage.local : localstorage();

/**
 * Colors.
 */

exports.colors = ['lightseagreen', 'forestgreen', 'goldenrod', 'dodgerblue', 'darkorchid', 'crimson'];

/**
 * Currently only WebKit-based Web Inspectors, Firefox >= v31,
 * and the Firebug extension (any Firefox version) are known
 * to support "%c" CSS customizations.
 *
 * TODO: add a `localStorage` variable to explicitly enable/disable colors
 */

function useColors() {
  // NB: In an Electron preload script, document will be defined but not fully
  // initialized. Since we know we're in Chrome, we'll just detect this case
  // explicitly
  if (typeof window !== 'undefined' && window.process && window.process.type === 'renderer') {
    return true;
  }

  // is webkit? http://stackoverflow.com/a/16459606/376773
  // document is undefined in react-native: https://github.com/facebook/react-native/pull/1632
  return typeof document !== 'undefined' && document.documentElement && document.documentElement.style && document.documentElement.style.WebkitAppearance ||
  // is firebug? http://stackoverflow.com/a/398120/376773
  typeof window !== 'undefined' && window.console && (window.console.firebug || window.console.exception && window.console.table) ||
  // is firefox >= v31?
  // https://developer.mozilla.org/en-US/docs/Tools/Web_Console#Styling_messages
  typeof navigator !== 'undefined' && navigator.userAgent && navigator.userAgent.toLowerCase().match(/firefox\/(\d+)/) && parseInt(RegExp.$1, 10) >= 31 ||
  // double check webkit in userAgent just in case we are in a worker
  typeof navigator !== 'undefined' && navigator.userAgent && navigator.userAgent.toLowerCase().match(/applewebkit\/(\d+)/);
}

/**
 * Map %j to `JSON.stringify()`, since no Web Inspectors do that by default.
 */

exports.formatters.j = function (v) {
  try {
    return JSON.stringify(v);
  } catch (err) {
    return '[UnexpectedJSONParseError]: ' + err.message;
  }
};

/**
 * Colorize log arguments if enabled.
 *
 * @api public
 */

function formatArgs(args) {
  var useColors = this.useColors;

  args[0] = (useColors ? '%c' : '') + this.namespace + (useColors ? ' %c' : ' ') + args[0] + (useColors ? '%c ' : ' ') + '+' + exports.humanize(this.diff);

  if (!useColors) return;

  var c = 'color: ' + this.color;
  args.splice(1, 0, c, 'color: inherit');

  // the final "%c" is somewhat tricky, because there could be other
  // arguments passed either before or after the %c, so we need to
  // figure out the correct index to insert the CSS into
  var index = 0;
  var lastC = 0;
  args[0].replace(/%[a-zA-Z%]/g, function (match) {
    if ('%%' === match) return;
    index++;
    if ('%c' === match) {
      // we only are interested in the *last* %c
      // (the user may have provided their own)
      lastC = index;
    }
  });

  args.splice(lastC, 0, c);
}

/**
 * Invokes `console.log()` when available.
 * No-op when `console.log` is not a "function".
 *
 * @api public
 */

function log() {
  // this hackery is required for IE8/9, where
  // the `console.log` function doesn't have 'apply'
  return 'object' === typeof console && console.log && Function.prototype.apply.call(console.log, console, arguments);
}

/**
 * Save `namespaces`.
 *
 * @param {String} namespaces
 * @api private
 */

function save(namespaces) {
  try {
    if (null == namespaces) {
      exports.storage.removeItem('debug');
    } else {
      exports.storage.debug = namespaces;
    }
  } catch (e) {}
}

/**
 * Load `namespaces`.
 *
 * @return {String} returns the previously persisted debug modes
 * @api private
 */

function load() {
  var r;
  try {
    r = exports.storage.debug;
  } catch (e) {}

  // If debug isn't set in LS, and we're in Electron, try to load $DEBUG
  if (!r && typeof process !== 'undefined' && 'env' in process) {
    r = process.env.DEBUG;
  }

  return r;
}

/**
 * Enable namespaces listed in `localStorage.debug` initially.
 */

exports.enable(load());

/**
 * Localstorage attempts to return the localstorage.
 *
 * This is necessary because safari throws
 * when a user disables cookies/localstorage
 * and you attempt to access it.
 *
 * @return {LocalStorage}
 * @api private
 */

function localstorage() {
  try {
    return window.localStorage;
  } catch (e) {}
}
/* WEBPACK VAR INJECTION */}.call(exports, __webpack_require__(3)))

/***/ }),
/* 3 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


// shim for using process in browser
var process = module.exports = {};

// cached from whatever global is present so that test runners that stub it
// don't break things.  But we need to wrap it in a try catch in case it is
// wrapped in strict mode code which doesn't define any globals.  It's inside a
// function because try/catches deoptimize in certain engines.

var cachedSetTimeout;
var cachedClearTimeout;

function defaultSetTimout() {
    throw new Error('setTimeout has not been defined');
}
function defaultClearTimeout() {
    throw new Error('clearTimeout has not been defined');
}
(function () {
    try {
        if (typeof setTimeout === 'function') {
            cachedSetTimeout = setTimeout;
        } else {
            cachedSetTimeout = defaultSetTimout;
        }
    } catch (e) {
        cachedSetTimeout = defaultSetTimout;
    }
    try {
        if (typeof clearTimeout === 'function') {
            cachedClearTimeout = clearTimeout;
        } else {
            cachedClearTimeout = defaultClearTimeout;
        }
    } catch (e) {
        cachedClearTimeout = defaultClearTimeout;
    }
})();
function runTimeout(fun) {
    if (cachedSetTimeout === setTimeout) {
        //normal enviroments in sane situations
        return setTimeout(fun, 0);
    }
    // if setTimeout wasn't available but was latter defined
    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {
        cachedSetTimeout = setTimeout;
        return setTimeout(fun, 0);
    }
    try {
        // when when somebody has screwed with setTimeout but no I.E. maddness
        return cachedSetTimeout(fun, 0);
    } catch (e) {
        try {
            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally
            return cachedSetTimeout.call(null, fun, 0);
        } catch (e) {
            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error
            return cachedSetTimeout.call(this, fun, 0);
        }
    }
}
function runClearTimeout(marker) {
    if (cachedClearTimeout === clearTimeout) {
        //normal enviroments in sane situations
        return clearTimeout(marker);
    }
    // if clearTimeout wasn't available but was latter defined
    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {
        cachedClearTimeout = clearTimeout;
        return clearTimeout(marker);
    }
    try {
        // when when somebody has screwed with setTimeout but no I.E. maddness
        return cachedClearTimeout(marker);
    } catch (e) {
        try {
            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally
            return cachedClearTimeout.call(null, marker);
        } catch (e) {
            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.
            // Some versions of I.E. have different rules for clearTimeout vs setTimeout
            return cachedClearTimeout.call(this, marker);
        }
    }
}
var queue = [];
var draining = false;
var currentQueue;
var queueIndex = -1;

function cleanUpNextTick() {
    if (!draining || !currentQueue) {
        return;
    }
    draining = false;
    if (currentQueue.length) {
        queue = currentQueue.concat(queue);
    } else {
        queueIndex = -1;
    }
    if (queue.length) {
        drainQueue();
    }
}

function drainQueue() {
    if (draining) {
        return;
    }
    var timeout = runTimeout(cleanUpNextTick);
    draining = true;

    var len = queue.length;
    while (len) {
        currentQueue = queue;
        queue = [];
        while (++queueIndex < len) {
            if (currentQueue) {
                currentQueue[queueIndex].run();
            }
        }
        queueIndex = -1;
        len = queue.length;
    }
    currentQueue = null;
    draining = false;
    runClearTimeout(timeout);
}

process.nextTick = function (fun) {
    var args = new Array(arguments.length - 1);
    if (arguments.length > 1) {
        for (var i = 1; i < arguments.length; i++) {
            args[i - 1] = arguments[i];
        }
    }
    queue.push(new Item(fun, args));
    if (queue.length === 1 && !draining) {
        runTimeout(drainQueue);
    }
};

// v8 likes predictible objects
function Item(fun, array) {
    this.fun = fun;
    this.array = array;
}
Item.prototype.run = function () {
    this.fun.apply(null, this.array);
};
process.title = 'browser';
process.browser = true;
process.env = {};
process.argv = [];
process.version = ''; // empty string to avoid regexp issues
process.versions = {};

function noop() {}

process.on = noop;
process.addListener = noop;
process.once = noop;
process.off = noop;
process.removeListener = noop;
process.removeAllListeners = noop;
process.emit = noop;
process.prependListener = noop;
process.prependOnceListener = noop;

process.listeners = function (name) {
    return [];
};

process.binding = function (name) {
    throw new Error('process.binding is not supported');
};

process.cwd = function () {
    return '/';
};
process.chdir = function (dir) {
    throw new Error('process.chdir is not supported');
};
process.umask = function () {
    return 0;
};

/***/ }),
/* 4 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


/**
 * This is the common logic for both the Node.js and web browser
 * implementations of `debug()`.
 *
 * Expose `debug()` as the module.
 */

exports = module.exports = createDebug.debug = createDebug['default'] = createDebug;
exports.coerce = coerce;
exports.disable = disable;
exports.enable = enable;
exports.enabled = enabled;
exports.humanize = __webpack_require__(5);

/**
 * The currently active debug mode names, and names to skip.
 */

exports.names = [];
exports.skips = [];

/**
 * Map of special "%n" handling functions, for the debug "format" argument.
 *
 * Valid key names are a single, lower or upper-case letter, i.e. "n" and "N".
 */

exports.formatters = {};

/**
 * Previous log timestamp.
 */

var prevTime;

/**
 * Select a color.
 * @param {String} namespace
 * @return {Number}
 * @api private
 */

function selectColor(namespace) {
  var hash = 0,
      i;

  for (i in namespace) {
    hash = (hash << 5) - hash + namespace.charCodeAt(i);
    hash |= 0; // Convert to 32bit integer
  }

  return exports.colors[Math.abs(hash) % exports.colors.length];
}

/**
 * Create a debugger with the given `namespace`.
 *
 * @param {String} namespace
 * @return {Function}
 * @api public
 */

function createDebug(namespace) {

  function debug() {
    // disabled?
    if (!debug.enabled) return;

    var self = debug;

    // set `diff` timestamp
    var curr = +new Date();
    var ms = curr - (prevTime || curr);
    self.diff = ms;
    self.prev = prevTime;
    self.curr = curr;
    prevTime = curr;

    // turn the `arguments` into a proper Array
    var args = new Array(arguments.length);
    for (var i = 0; i < args.length; i++) {
      args[i] = arguments[i];
    }

    args[0] = exports.coerce(args[0]);

    if ('string' !== typeof args[0]) {
      // anything else let's inspect with %O
      args.unshift('%O');
    }

    // apply any `formatters` transformations
    var index = 0;
    args[0] = args[0].replace(/%([a-zA-Z%])/g, function (match, format) {
      // if we encounter an escaped % then don't increase the array index
      if (match === '%%') return match;
      index++;
      var formatter = exports.formatters[format];
      if ('function' === typeof formatter) {
        var val = args[index];
        match = formatter.call(self, val);

        // now we need to remove `args[index]` since it's inlined in the `format`
        args.splice(index, 1);
        index--;
      }
      return match;
    });

    // apply env-specific formatting (colors, etc.)
    exports.formatArgs.call(self, args);

    var logFn = debug.log || exports.log || console.log.bind(console);
    logFn.apply(self, args);
  }

  debug.namespace = namespace;
  debug.enabled = exports.enabled(namespace);
  debug.useColors = exports.useColors();
  debug.color = selectColor(namespace);

  // env-specific initialization logic for debug instances
  if ('function' === typeof exports.init) {
    exports.init(debug);
  }

  return debug;
}

/**
 * Enables a debug mode by namespaces. This can include modes
 * separated by a colon and wildcards.
 *
 * @param {String} namespaces
 * @api public
 */

function enable(namespaces) {
  exports.save(namespaces);

  exports.names = [];
  exports.skips = [];

  var split = (typeof namespaces === 'string' ? namespaces : '').split(/[\s,]+/);
  var len = split.length;

  for (var i = 0; i < len; i++) {
    if (!split[i]) continue; // ignore empty strings
    namespaces = split[i].replace(/\*/g, '.*?');
    if (namespaces[0] === '-') {
      exports.skips.push(new RegExp('^' + namespaces.substr(1) + '$'));
    } else {
      exports.names.push(new RegExp('^' + namespaces + '$'));
    }
  }
}

/**
 * Disable debug output.
 *
 * @api public
 */

function disable() {
  exports.enable('');
}

/**
 * Returns true if the given mode name is enabled, false otherwise.
 *
 * @param {String} name
 * @return {Boolean}
 * @api public
 */

function enabled(name) {
  var i, len;
  for (i = 0, len = exports.skips.length; i < len; i++) {
    if (exports.skips[i].test(name)) {
      return false;
    }
  }
  for (i = 0, len = exports.names.length; i < len; i++) {
    if (exports.names[i].test(name)) {
      return true;
    }
  }
  return false;
}

/**
 * Coerce `val`.
 *
 * @param {Mixed} val
 * @return {Mixed}
 * @api private
 */

function coerce(val) {
  if (val instanceof Error) return val.stack || val.message;
  return val;
}

/***/ }),
/* 5 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


/**
 * Helpers.
 */

var s = 1000;
var m = s * 60;
var h = m * 60;
var d = h * 24;
var y = d * 365.25;

/**
 * Parse or format the given `val`.
 *
 * Options:
 *
 *  - `long` verbose formatting [false]
 *
 * @param {String|Number} val
 * @param {Object} [options]
 * @throws {Error} throw an error if val is not a non-empty string or a number
 * @return {String|Number}
 * @api public
 */

module.exports = function (val, options) {
  options = options || {};
  var type = typeof val;
  if (type === 'string' && val.length > 0) {
    return parse(val);
  } else if (type === 'number' && isNaN(val) === false) {
    return options.long ? fmtLong(val) : fmtShort(val);
  }
  throw new Error('val is not a non-empty string or a valid number. val=' + JSON.stringify(val));
};

/**
 * Parse the given `str` and return milliseconds.
 *
 * @param {String} str
 * @return {Number}
 * @api private
 */

function parse(str) {
  str = String(str);
  if (str.length > 100) {
    return;
  }
  var match = /^((?:\d+)?\.?\d+) *(milliseconds?|msecs?|ms|seconds?|secs?|s|minutes?|mins?|m|hours?|hrs?|h|days?|d|years?|yrs?|y)?$/i.exec(str);
  if (!match) {
    return;
  }
  var n = parseFloat(match[1]);
  var type = (match[2] || 'ms').toLowerCase();
  switch (type) {
    case 'years':
    case 'year':
    case 'yrs':
    case 'yr':
    case 'y':
      return n * y;
    case 'days':
    case 'day':
    case 'd':
      return n * d;
    case 'hours':
    case 'hour':
    case 'hrs':
    case 'hr':
    case 'h':
      return n * h;
    case 'minutes':
    case 'minute':
    case 'mins':
    case 'min':
    case 'm':
      return n * m;
    case 'seconds':
    case 'second':
    case 'secs':
    case 'sec':
    case 's':
      return n * s;
    case 'milliseconds':
    case 'millisecond':
    case 'msecs':
    case 'msec':
    case 'ms':
      return n;
    default:
      return undefined;
  }
}

/**
 * Short format for `ms`.
 *
 * @param {Number} ms
 * @return {String}
 * @api private
 */

function fmtShort(ms) {
  if (ms >= d) {
    return Math.round(ms / d) + 'd';
  }
  if (ms >= h) {
    return Math.round(ms / h) + 'h';
  }
  if (ms >= m) {
    return Math.round(ms / m) + 'm';
  }
  if (ms >= s) {
    return Math.round(ms / s) + 's';
  }
  return ms + 'ms';
}

/**
 * Long format for `ms`.
 *
 * @param {Number} ms
 * @return {String}
 * @api private
 */

function fmtLong(ms) {
  return plural(ms, d, 'day') || plural(ms, h, 'hour') || plural(ms, m, 'minute') || plural(ms, s, 'second') || ms + ' ms';
}

/**
 * Pluralization helper.
 */

function plural(ms, n, name) {
  if (ms < n) {
    return;
  }
  if (ms < n * 1.5) {
    return Math.floor(ms / n) + ' ' + name;
  }
  return Math.ceil(ms / n) + ' ' + name + 's';
}

/***/ }),
/* 6 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


var Message = __webpack_require__(7);

var EventEmitter = __webpack_require__(0);

var idCounter = 0;
var postedMessages = new Map();

class MessageHandler extends EventEmitter {
    constructor() {
        super();
        this.readyToPost = false;
        this.readyToHandle = false;
        this.windowID = null;
        this.messageSource = null;
        this.pendingMessages = [];
        this.unhandledMessages = [];
    }

    _postMessageToSource(message) {
        this.messageSource.postMessage(JSON.stringify(message), '*');
    }

    init(messageSource) {
        this.readyToPost = true;
        this.windowID = Date.now();
        this.messageSource = messageSource;
        this._postMessageToSource({
            type: 'admin.connect',
            windowID: this.windowID
        });

        window.addEventListener('unload', () => {
            this.messageSource.postMessage(JSON.stringify({
                type: 'admin.disconnect',
                windowID: this.windowID
            }), '*');
        });

        this.postPendingMessages();
    }

    postMessage(type, message) {
        var id = ++idCounter;
        var toPost = {
            type,
            message,
            messageID: id,
            windowID: this.windowID
        };
        var theMessage = new Message(id, toPost);
        if (this.readyToPost) {
            this._postMessageToSource(toPost);
            postedMessages.set(id, theMessage);
        } else {
            this.pendingMessages.push(theMessage);
        }
        return theMessage;
    }

    postPendingMessages() {
        for (var message of this.pendingMessages) {
            message.data.windowID = this.windowID;
            this._postMessageToSource(message.data);
            postedMessages.set(message.id, message);
        }
    }

    handlePendingMessages() {
        this.readyToHandle = true;
        for (var i = 0; i < this.unhandledMessages.length; i++) {
            this.handleMessage(this.unhandledMessages[i]);
        }
        this.unhandledMessages = [];
    }

    handleMessage(data) {
        if (!this.readyToHandle) {
            this.unhandledMessages.push(data);
            return;
        }
        if (!postedMessages.has(data.messageID)) {
            return this.emit('message', data);
        }
        var message = postedMessages.get(data.messageID);
        if (data.status) {
            if (data.status === 'error') {
                message._reject(data);
                postedMessages.delete(data.messageID);
            } else if (data.status === 'success') {
                message._resolve(data);
                postedMessages.delete(data.messageID);
            } else {
                message.emit(data.status, data);
            }
        } else {
            // no status is considered a success
            message._resolve(data);
            postedMessages.delete(data.messageID);
        }
    }
}

module.exports = MessageHandler;

/***/ }),
/* 7 */
/***/ (function(module, exports, __webpack_require__) {

"use strict";


var EventEmitter = __webpack_require__(0).EventEmitter;
var promise = Symbol();

class Message extends EventEmitter {
    constructor(id, data) {
        super();
        this.id = id;
        this.data = data;
        this[promise] = new Promise((resolve, reject) => {
            this._resolve = resolve;
            this._reject = reject;
        });
    }

    then(onResolve, onReject) {
        return this[promise].then(onResolve, onReject);
    }

    catch(onReject) {
        return this[promise].catch(onReject);
    }
}

module.exports = Message;

/***/ })
/******/ ]);
});
//# sourceMappingURL=iframe-bridge.js.map';
7
+
8
+ export default iframeBridge;
package/main/index.js ADDED
@@ -0,0 +1,174 @@
1
+ /* eslint-disable no-console */
2
+
3
+ import fs from 'node:fs/promises';
4
+ import path from 'node:path';
5
+ import { fileURLToPath } from 'node:url';
6
+
7
+ import _ from 'lodash';
8
+ import visualizer from 'react-visualizer';
9
+ import webpack from 'webpack';
10
+
11
+ import iframeBridge from './iframe-bridge.js';
12
+
13
+ const __dirname = import.meta.dirname;
14
+
15
+ const defaultConfig = {
16
+ title: 'visualizer-on-tabs',
17
+ };
18
+
19
+ async function buildApp(options, outDir, cleanup) {
20
+ const { promise, resolve, reject } = Promise.withResolvers();
21
+ const entries = [{ file: 'app.js' }];
22
+ for (const entry of entries) {
23
+ let config = {
24
+ mode: options.mode === 'development' ? 'development' : 'production',
25
+ context: path.resolve(__dirname, '../'),
26
+ entry: path.resolve(__dirname, '../src', entry.file),
27
+ output: {
28
+ path: outDir,
29
+ filename: entry.file,
30
+ },
31
+ devtool: 'source-map',
32
+ module: {
33
+ rules: [
34
+ {
35
+ test: /\.js$/,
36
+ include: [
37
+ path.resolve(__dirname, '../src'),
38
+ path.resolve(__dirname, '../node_modules/iframe-bridge'),
39
+ ],
40
+ loader: 'babel-loader',
41
+ options: {
42
+ cwd: path.join(__dirname, '..'),
43
+ presets: ['@babel/env', '@babel/react'],
44
+ },
45
+ },
46
+ ],
47
+ },
48
+ };
49
+
50
+ function handleError(err, stats) {
51
+ if (err) {
52
+ if (options.watch) {
53
+ console.error(err.stack, err.message);
54
+ } else {
55
+ reject(err);
56
+ }
57
+ } else {
58
+ // TODO: use node's util.debuglog here and in flavor-builder
59
+ const statsJson = stats.toJson();
60
+
61
+ if (statsJson.errors.length > 0) {
62
+ for (let error of statsJson.errors) {
63
+ console.error(error.message);
64
+ }
65
+ if (!options.watch) {
66
+ reject(
67
+ new Error(
68
+ `Build failed with ${statsJson.errors.length} error(s)`,
69
+ ),
70
+ );
71
+ }
72
+ }
73
+ if (statsJson.warnings.length > 0) {
74
+ for (let warning of statsJson.warnings) {
75
+ console.warn(warning.message);
76
+ }
77
+ }
78
+ console.log(`Build of ${entry.file} successful`);
79
+ if (!options.watch) {
80
+ cleanup()
81
+ .catch(console.error)
82
+ .then(() => resolve(cleanup));
83
+ }
84
+ }
85
+ }
86
+ const instance = webpack(config);
87
+ if (options.watch) {
88
+ instance.watch({ aggregateTimeout: 200 }, handleError);
89
+ // In watch mode we resolve before the first build is done.
90
+ resolve(cleanup);
91
+ } else {
92
+ // With a single run, the handler will resolve / reject the promise.
93
+ instance.run(handleError);
94
+ }
95
+ }
96
+ return promise;
97
+ }
98
+
99
+ export default async (options) => {
100
+ Object.assign(options.config, defaultConfig);
101
+
102
+ const outDir = path.resolve(options.outDir);
103
+ await fs.mkdir(outDir, { recursive: true });
104
+
105
+ const confPath = path.join(__dirname, '../src/config/custom.json');
106
+ console.log('Copying files');
107
+ await Promise.all([
108
+ fs.writeFile(confPath, JSON.stringify(options.config)),
109
+ copyBootstrap(options),
110
+ copyContent(options),
111
+ addIndex(options),
112
+ addVisualizer(options),
113
+ ]);
114
+
115
+ async function cleanup() {
116
+ console.log('Cleaning up');
117
+ // Normally, the file should exist when this is called.
118
+ await fs.unlink(confPath);
119
+ }
120
+
121
+ console.log('Building app');
122
+ await buildApp(options, outDir, cleanup);
123
+
124
+ return cleanup;
125
+ };
126
+
127
+ function copyContent(options) {
128
+ return fs.cp(
129
+ path.join(__dirname, '../src/static'),
130
+ path.join(options.outDir, 'static'),
131
+ {
132
+ recursive: true,
133
+ },
134
+ );
135
+ }
136
+
137
+ async function copyBootstrap(options) {
138
+ const bootstrapCss = fileURLToPath(
139
+ import.meta.resolve('bootstrap/dist/css/bootstrap.min.css'),
140
+ );
141
+ await fs.mkdir(path.join(options.outDir, 'static'), { recursive: true });
142
+ return fs.copyFile(
143
+ bootstrapCss,
144
+ path.join(options.outDir, 'static/bootstrap.min.css'),
145
+ );
146
+ }
147
+
148
+ async function addIndex(options) {
149
+ const content = await fs.readFile(
150
+ path.join(__dirname, '../src/template/index.html'),
151
+ 'utf8',
152
+ );
153
+ const tpl = _.template(content);
154
+ return fs.writeFile(
155
+ path.join(options.outDir, 'index.html'),
156
+ tpl({
157
+ title: options.config.title,
158
+ uniqid: Date.now(),
159
+ }),
160
+ );
161
+ }
162
+
163
+ function addVisualizer(options) {
164
+ const page = visualizer.makeVisualizerPage({
165
+ cdn: options.config.visualizerCDN,
166
+ fallbackVersion: options.config.visualizerFallbackVersion,
167
+ scripts: [
168
+ {
169
+ url: iframeBridge,
170
+ },
171
+ ],
172
+ });
173
+ return fs.writeFile(path.join(options.outDir, 'visualizer.html'), page);
174
+ }
package/package.json ADDED
@@ -0,0 +1,55 @@
1
+ {
2
+ "name": "visualizer-on-tabs",
3
+ "version": "1.0.0",
4
+ "description": "visualizer-on-tabs webpack builder",
5
+ "type": "module",
6
+ "exports": {
7
+ ".": "./main/index.js"
8
+ },
9
+ "bin": "bin/build.js",
10
+ "files": [
11
+ "main",
12
+ "src",
13
+ "bin",
14
+ "!**/*.test.js"
15
+ ],
16
+ "browserslist": [
17
+ "defaults"
18
+ ],
19
+ "scripts": {
20
+ "build": "npx --no-install visualizer-on-tabs --config=./dev.json --outDir=./out",
21
+ "build:dev": "npx --no-install visualizer-on-tabs --dev=1 --config=./dev.json --outDir=./out",
22
+ "clean": "rimraf out",
23
+ "eslint": "eslint bin main src",
24
+ "prettier": "prettier --check ./",
25
+ "prettier-write": "prettier --write ./",
26
+ "serve": "npx serve out/",
27
+ "test": "npm run test-only && npm run eslint && npm run prettier",
28
+ "test-only": "node --test"
29
+ },
30
+ "dependencies": {
31
+ "@babel/core": "^7.28.3",
32
+ "@babel/preset-env": "^7.28.3",
33
+ "@babel/preset-react": "^7.27.1",
34
+ "babel-loader": "^10.0.0",
35
+ "bootstrap": "^5.3.3",
36
+ "iframe-bridge": "^3.0.2",
37
+ "lockr": "^0.8.5",
38
+ "lodash": "^4.17.21",
39
+ "minimist": "^1.2.6",
40
+ "react": "^18.3.1",
41
+ "react-bootstrap": "^3.0.0-beta.3",
42
+ "react-dom": "^18.3.1",
43
+ "react-visualizer": "^3.0.1",
44
+ "webpack": "^5.91.0"
45
+ },
46
+ "devDependencies": {
47
+ "eslint": "^9.34.0",
48
+ "eslint-config-zakodium": "^16.0.0",
49
+ "prettier": "^3.6.2",
50
+ "rimraf": "^6.0.1"
51
+ },
52
+ "volta": {
53
+ "node": "24.7.0"
54
+ }
55
+ }
package/src/app.js ADDED
@@ -0,0 +1,20 @@
1
+ import React from 'react';
2
+ import ReactDOM from 'react-dom/client';
3
+
4
+ import App from './components/App.js';
5
+
6
+ const loc = window.location;
7
+ let hash = loc.hash.slice(1);
8
+ if (!hash) {
9
+ hash = 'main';
10
+ }
11
+
12
+ const props = {
13
+ origin: loc.origin,
14
+ path: hash,
15
+ };
16
+
17
+ const element = React.createElement(App, props);
18
+ ReactDOM.createRoot(document.getElementById('visualizer-on-tabs')).render(
19
+ element,
20
+ );
@@ -0,0 +1,373 @@
1
+ /* eslint-disable no-await-in-loop */
2
+
3
+ import { postMessage, registerHandler } from 'iframe-bridge/main';
4
+ import React from 'react';
5
+ import Tab from 'react-bootstrap/Tab';
6
+ import BTabs from 'react-bootstrap/Tabs';
7
+ import { Visualizer } from 'react-visualizer';
8
+
9
+ import { getConfig } from '../config/config.js';
10
+ import customConfig from '../config/custom.json' with { type: 'json' };
11
+ import Tabs from '../main/Tabs.js';
12
+ import iframeMessageHandler from '../main/iframeMessageHandler.js';
13
+ import * as tabStorage from '../main/tabStorage.js';
14
+ import { rewriteURL } from '../util.js';
15
+
16
+ import Login from './Login.js';
17
+ import TabTitle from './TabTitle.js';
18
+
19
+ const config = getConfig(customConfig);
20
+ let tabInit = Promise.resolve();
21
+ let currentIframe;
22
+
23
+ const pageURL = new URL(window.location);
24
+ const pageQueryParameters = (() => {
25
+ let params = {};
26
+ for (let key of pageURL.searchParams.keys()) {
27
+ params[key] = pageURL.searchParams.get(key);
28
+ }
29
+ return params;
30
+ })();
31
+
32
+ const iframeStyle = { position: 'static', flex: 2, border: 'none' };
33
+
34
+ class App extends React.Component {
35
+ constructor(props) {
36
+ super(props);
37
+ for (let key in config.possibleViews) {
38
+ config.possibleViews[key].id = key;
39
+ }
40
+ this.onActiveTab = this.onActiveTab.bind(this);
41
+
42
+ registerHandler('tab', iframeMessageHandler);
43
+ registerHandler('admin', (data, [level2]) => {
44
+ if (level2 === 'connect' && data.windowID !== undefined) {
45
+ if (!currentIframe) {
46
+ // The iframe was refreshed
47
+ config.possibleViews[this.state.activeTabKey].windowID =
48
+ data.windowID;
49
+ this.sendData(this.state.activeTabKey);
50
+ } else {
51
+ config.possibleViews[currentIframe.id].windowID = data.windowID;
52
+ currentIframe.resolve();
53
+ currentIframe = null;
54
+ }
55
+ }
56
+ });
57
+
58
+ Tabs.on('openTab', (obj) => {
59
+ const options = {};
60
+ ['noFocus', 'noFocusEvent', 'noData', 'load'].forEach((prop) => {
61
+ options[prop] = obj[prop];
62
+ delete obj[prop];
63
+ });
64
+ this.doTab(obj, options);
65
+ });
66
+ Tabs.on('status', this.setTabStatus.bind(this));
67
+ Tabs.on('message', this.sendTabMessage.bind(this));
68
+ Tabs.on('focus', this.focusTab.bind(this));
69
+
70
+ this.visualizerVersion = pageURL.searchParams.get('v');
71
+
72
+ this.state = {
73
+ viewsList: [],
74
+ activeTabKey: 0,
75
+ isConfigLoaded: false,
76
+ };
77
+
78
+ void this.loadTabs();
79
+ }
80
+
81
+ async loadTabs() {
82
+ let firstTab;
83
+ const loadTab = async (view) => {
84
+ if (!firstTab) firstTab = view.id;
85
+ await this.doTab(view, {
86
+ noFocus: true,
87
+ load: config.loadHidden,
88
+ noFocusEvent: true,
89
+ });
90
+ };
91
+ const data = tabStorage.load();
92
+ // Load possible views first
93
+ for (let key in config.possibleViews) {
94
+ let saved;
95
+ if ((saved = data.find((el) => el.id === key))) {
96
+ await loadTab(saved);
97
+ } else {
98
+ await loadTab(config.possibleViews[key]);
99
+ }
100
+ }
101
+
102
+ for (let i = 0; i < data.length; i++) {
103
+ if (!config.possibleViews[data[i].id]) {
104
+ await loadTab(data[i]);
105
+ }
106
+ }
107
+
108
+ // Nothing is focused at this point
109
+ const lastSelected = tabStorage.getSelected();
110
+ if (this.state.viewsList.find((el) => el.id === lastSelected)) {
111
+ await this.showTab(lastSelected);
112
+ } else {
113
+ await this.showTab(firstTab);
114
+ }
115
+ }
116
+
117
+ setTabStatus(data) {
118
+ // Find view with given window ID
119
+ const ids = Object.keys(config.possibleViews);
120
+ let id = ids.find(
121
+ (id) => config.possibleViews[id].windowID === data.windowID,
122
+ );
123
+ if (!id) return;
124
+ let view = config.possibleViews[id];
125
+
126
+ view = this.state.viewsList.find((el) => el.id === view.id);
127
+ if (!view) return;
128
+
129
+ view.status = { ...view.status, ...data.message };
130
+ this.setState((state) => ({
131
+ viewsList: state.viewsList,
132
+ }));
133
+ }
134
+
135
+ sendTabMessage(data) {
136
+ const viewInfo = config.possibleViews[data.id];
137
+ if (viewInfo) {
138
+ postMessage('tab.message', data.message, viewInfo.windowID);
139
+ }
140
+ }
141
+
142
+ async focusTab(tabId) {
143
+ if (this.state.viewsList.find((el) => el.id === tabId)) {
144
+ await this.showTab(tabId, {
145
+ noData: true,
146
+ });
147
+ }
148
+ }
149
+
150
+ async doTab(obj, options) {
151
+ if (!config.possibleViews[obj.id]) {
152
+ config.possibleViews[obj.id] = {
153
+ id: obj.id,
154
+ url: obj.url,
155
+ data: obj.data,
156
+ closable: obj.closable,
157
+ rawIframe: obj.rawIframe,
158
+ };
159
+ } else {
160
+ config.possibleViews[obj.id].data = obj.data;
161
+ }
162
+
163
+ if (config.rewriteRules) {
164
+ let newURL = rewriteURL(
165
+ config.rewriteRules,
166
+ config.possibleViews[obj.id].url,
167
+ );
168
+ if (newURL) {
169
+ config.possibleViews[obj.id].rewrittenUrl = newURL;
170
+ }
171
+ }
172
+
173
+ await this.showTab(obj.id, options);
174
+ }
175
+
176
+ async showTab(id, options) {
177
+ options = options || {};
178
+ const sameTab = this.state.activeTabKey === id;
179
+ if (sameTab && !options.force) return;
180
+
181
+ const focusedTabId = options.noFocus ? undefined : id;
182
+ let viewFromList = this.state.viewsList.find((el) => el.id === id);
183
+ const newTab = !viewFromList;
184
+ const viewInfo = config.possibleViews[id];
185
+
186
+ if (!viewInfo) throw new Error('unreachable');
187
+ if (!viewFromList) {
188
+ viewFromList = {
189
+ id,
190
+ url: viewInfo.url,
191
+ rewrittenUrl: viewInfo.rewrittenUrl,
192
+ closable: viewInfo.closable,
193
+ rawIframe: viewInfo.rawIframe,
194
+ };
195
+ this.state.viewsList.push(viewFromList);
196
+ }
197
+ const firstRender =
198
+ (options.load || !options.noFocus) && (newTab || !viewFromList.rendered);
199
+ await tabInit;
200
+ // First render means we expect the render function to initialize a new iframe
201
+ // We need to get the IframeBridge ID of that frame and prevent any other iframes
202
+ // to load during that time
203
+ if (firstRender) {
204
+ // TODO: there is probably a cleaner way than a global promise
205
+ // eslint-disable-next-line require-atomic-updates
206
+ tabInit = new Promise((resolve) => {
207
+ viewFromList.rendered = true;
208
+ this.setState((state) => ({
209
+ activeTabKey: focusedTabId,
210
+ viewsList: state.viewsList,
211
+ }));
212
+
213
+ setTimeout(() => {
214
+ // This will have an effect only if Promise is not yet resolved
215
+ // It prevents completely blocking the interface if there is a problem
216
+ // with that tab
217
+ return resolve();
218
+ }, 3000);
219
+ currentIframe = {
220
+ resolve,
221
+ id,
222
+ };
223
+ });
224
+ await tabInit;
225
+ } else {
226
+ this.setState((state) => ({
227
+ activeTabKey: focusedTabId,
228
+ viewsList: state.viewsList,
229
+ }));
230
+ }
231
+
232
+ // always send data on first render
233
+ if (!options.noData || firstRender) {
234
+ this.sendData(id);
235
+ }
236
+ tabStorage.save(id, viewInfo);
237
+ if (!options.noFocus) {
238
+ tabStorage.saveSelected(id);
239
+ }
240
+ if (!options.noFocusEvent && !sameTab) {
241
+ this.sendTabFocusEvent(focusedTabId);
242
+ }
243
+ }
244
+
245
+ sendData(id) {
246
+ const viewInfo = config.possibleViews[id];
247
+ postMessage(
248
+ 'tab.data',
249
+ { ...viewInfo.data, queryParameters: pageQueryParameters },
250
+ viewInfo.windowID,
251
+ );
252
+ }
253
+
254
+ async removeTab(id) {
255
+ const forbiddenPossibleViews = Object.keys(config.possibleViews);
256
+ tabStorage.remove(id);
257
+ if (forbiddenPossibleViews.indexOf(id) === -1) {
258
+ delete config.possibleViews[id];
259
+ }
260
+ let idx = this.state.viewsList.findIndex((el) => el.id === id);
261
+ if (idx === -1) return;
262
+ this.state.viewsList.splice(idx, 1);
263
+
264
+ let newActiveTab;
265
+ if (id !== this.state.activeTabKey) {
266
+ newActiveTab = this.state.activeTabKey;
267
+ } else {
268
+ const viewsLength = this.state.viewsList.length;
269
+ // Set next active tab
270
+ if (viewsLength > 0) {
271
+ if (idx < viewsLength) {
272
+ newActiveTab = this.state.viewsList[idx].id;
273
+ } else {
274
+ newActiveTab = this.state.viewsList[viewsLength - 1].id;
275
+ }
276
+ }
277
+ }
278
+
279
+ await this.showTab(newActiveTab, {
280
+ noData: true,
281
+ force: true,
282
+ });
283
+ }
284
+
285
+ sendTabFocusEvent(key) {
286
+ if (config.possibleViews[key]) {
287
+ postMessage('tab.focus', {}, config.possibleViews[key].windowID);
288
+ }
289
+ }
290
+
291
+ async onActiveTab(key) {
292
+ await this.showTab(key, {
293
+ noData: true,
294
+ });
295
+ }
296
+
297
+ render() {
298
+ const arr = [];
299
+ for (let view of this.state.viewsList) {
300
+ const closable = view.closable === undefined ? true : view.closable;
301
+ const saved =
302
+ !view.status || view.status.saved === undefined
303
+ ? true
304
+ : view.status.saved;
305
+
306
+ const textStyle = {};
307
+ if (!saved) {
308
+ textStyle.color = 'red';
309
+ }
310
+ const shouldRender = view.rendered || view.id === this.state.activeTabKey;
311
+ let viewPage;
312
+ if (shouldRender) {
313
+ if (view.rawIframe) {
314
+ viewPage = (
315
+ <iframe
316
+ allow="fullscreen; clipboard-read; clipboard-write;"
317
+ src={view.rewrittenUrl || view.url}
318
+ style={iframeStyle}
319
+ />
320
+ );
321
+ } else {
322
+ viewPage = (
323
+ <Visualizer
324
+ url="visualizer.html"
325
+ viewURL={view.rewrittenUrl || view.url}
326
+ version={
327
+ this.visualizerVersion || config.visualizerVersion || 'auto'
328
+ }
329
+ config={config.visualizerConfig}
330
+ style={iframeStyle}
331
+ />
332
+ );
333
+ }
334
+ } else {
335
+ viewPage = <div>Not rendered</div>;
336
+ }
337
+ arr.push(
338
+ <Tab
339
+ title={
340
+ <TabTitle
341
+ textTitle={saved ? null : 'Not saved'}
342
+ textStyle={textStyle}
343
+ name={view.id}
344
+ onTabClosed={closable ? () => this.removeTab(view.id) : null}
345
+ />
346
+ }
347
+ key={view.id}
348
+ eventKey={view.id}
349
+ >
350
+ {viewPage}
351
+ </Tab>,
352
+ );
353
+ }
354
+
355
+ return (
356
+ <div className="visualizer-on-tabs-app">
357
+ <Login config={config} />
358
+ <div className="visualizer-on-tabs-content d-flex flex-column">
359
+ <BTabs
360
+ id="visualizer-on-tabs-tab"
361
+ transition={false}
362
+ activeKey={this.state.activeTabKey}
363
+ onSelect={this.onActiveTab}
364
+ >
365
+ {arr}
366
+ </BTabs>
367
+ </div>
368
+ </div>
369
+ );
370
+ }
371
+ }
372
+
373
+ export default App;
@@ -0,0 +1,92 @@
1
+ import React from 'react';
2
+
3
+ const styles = {
4
+ position: 'fixed',
5
+ right: 20,
6
+ top: 10,
7
+ };
8
+
9
+ class Login extends React.Component {
10
+ constructor(props) {
11
+ super(props);
12
+ const { config } = props;
13
+ this.state = {};
14
+ this.logout = this.logout.bind(this);
15
+ if (!config.rocLogin) return;
16
+
17
+ if (config.rocLogin.urlAbsolute) {
18
+ this.loginUrl = config.rocLogin.urlAbsolute;
19
+ } else {
20
+ this.loginUrl = `${config.rocLogin.url}/auth/login?continue=${
21
+ config.rocLogin.redirect || window.location.href
22
+ }`;
23
+ }
24
+ }
25
+
26
+ componentDidMount() {
27
+ void this.session();
28
+ }
29
+
30
+ async session() {
31
+ if (!this.props.config.rocLogin) return;
32
+ const login = this.props.config.rocLogin;
33
+ const response = await fetch(`${login.url}/auth/session`, {
34
+ credentials: 'include',
35
+ });
36
+ if (response.ok) {
37
+ const body = await response.json();
38
+ if (
39
+ login.auto &&
40
+ (!body.authenticated || (login.user && body.username !== login.user))
41
+ ) {
42
+ window.location.href = this.loginUrl;
43
+ }
44
+ this.setState({
45
+ user: body.username,
46
+ });
47
+ } else {
48
+ this.setState({
49
+ user: null,
50
+ });
51
+ }
52
+ }
53
+
54
+ async logout() {
55
+ if (!this.props.config.rocLogin) return;
56
+ const response = await fetch(
57
+ `${this.props.config.rocLogin.url}/auth/logout`,
58
+ {
59
+ credentials: 'include',
60
+ },
61
+ );
62
+ if (!response.ok) {
63
+ throw new Error(`Unexpected logout response: ${response.statusText}`);
64
+ }
65
+ void this.session();
66
+ }
67
+
68
+ render() {
69
+ if (!this.props.config.rocLogin) {
70
+ return <div />;
71
+ }
72
+ if (!this.state.user || this.state.user === 'anonymous') {
73
+ return (
74
+ <div style={styles}>
75
+ <a href={this.loginUrl}>Login</a>
76
+ </div>
77
+ );
78
+ } else {
79
+ return (
80
+ <div style={styles}>
81
+ {this.state.user}
82
+ &nbsp;
83
+ <a href="#" onClick={this.logout}>
84
+ Logout
85
+ </a>
86
+ </div>
87
+ );
88
+ }
89
+ }
90
+ }
91
+
92
+ export default Login;
@@ -0,0 +1,41 @@
1
+ import React from 'react';
2
+
3
+ class TabTitle extends React.Component {
4
+ constructor(props) {
5
+ super(props);
6
+ this.onClosedClicked = this.onClosedClicked.bind(this);
7
+ }
8
+ onClosedClicked(event) {
9
+ event.stopPropagation();
10
+ event.preventDefault();
11
+ if (this.props.onTabClosed) {
12
+ this.props.onTabClosed.call();
13
+ }
14
+ }
15
+
16
+ render() {
17
+ let closeHandle;
18
+ if (this.props.onTabClosed) {
19
+ closeHandle = (
20
+ <span className="close-tab-glyph" onClick={this.onClosedClicked}>
21
+ ×
22
+ </span>
23
+ );
24
+ } else {
25
+ closeHandle = '';
26
+ }
27
+
28
+ return (
29
+ <div
30
+ title={this.props.textTitle}
31
+ style={this.props.textStyle}
32
+ className="d-flex flex-row gap-2 align-items-baseline"
33
+ >
34
+ {this.props.name}
35
+ {closeHandle}
36
+ </div>
37
+ );
38
+ }
39
+ }
40
+
41
+ export default TabTitle;
@@ -0,0 +1,10 @@
1
+ import defaultConfig from './default.js';
2
+
3
+ export function getConfig(customConfig) {
4
+ const config = { ...defaultConfig, ...customConfig };
5
+ if (config.rocLogin && config.rocLogin.url) {
6
+ // Remove trailing slash
7
+ config.rocLogin.url = config.rocLogin.url.replace(/\/$/, '');
8
+ }
9
+ return config;
10
+ }
@@ -0,0 +1,15 @@
1
+ export default {
2
+ visualizerConfig: {
3
+ debugLevel: 0,
4
+ filters: [],
5
+ modules: {
6
+ folders: ['modules/types'],
7
+ },
8
+ header: false,
9
+ },
10
+ possibleViews: {
11
+ Home: {
12
+ url: '',
13
+ },
14
+ },
15
+ };
@@ -0,0 +1,21 @@
1
+ import { EventEmitter } from 'events';
2
+
3
+ const Tabs = new EventEmitter();
4
+
5
+ Tabs.openTab = function openTab(data) {
6
+ Tabs.emit('openTab', data);
7
+ };
8
+
9
+ Tabs.status = function status(data) {
10
+ Tabs.emit('status', data);
11
+ };
12
+
13
+ Tabs.sendMessage = function sendMessage(message) {
14
+ Tabs.emit('message', message);
15
+ };
16
+
17
+ Tabs.focus = function focus(message) {
18
+ Tabs.emit('focus', message);
19
+ };
20
+
21
+ export default Tabs;
@@ -0,0 +1 @@
1
+ export const version = 1;
@@ -0,0 +1,42 @@
1
+ import Tabs from './Tabs.js';
2
+
3
+ export default function iframeMessageHandler(data, [level2]) {
4
+ let prom;
5
+ switch (level2) {
6
+ case 'open':
7
+ Tabs.openTab(data.message);
8
+ prom = Promise.resolve('done');
9
+ break;
10
+ case 'status':
11
+ Tabs.status(data);
12
+ prom = Promise.resolve('done');
13
+ break;
14
+ case 'message':
15
+ Tabs.sendMessage(data.message);
16
+ prom = Promise.resolve('done');
17
+ break;
18
+ case 'focus':
19
+ Tabs.focus(data.message);
20
+ prom = Promise.resolve('done');
21
+ break;
22
+ default:
23
+ prom = Promise.reject(new Error(`Unknown action: ${level2}}`));
24
+ break;
25
+ }
26
+
27
+ prom.then(
28
+ (message) => {
29
+ data.message = message.data;
30
+ // the iframeMessageHandler callback's `this` is bound by the iframe-bridge library
31
+ // eslint-disable-next-line no-invalid-this
32
+ this.postMessage(data);
33
+ },
34
+ (error) => {
35
+ data.status = 'error';
36
+ data.message = error;
37
+ // the iframeMessageHandler callback's `this` is bound by the iframe-bridge library
38
+ // eslint-disable-next-line no-invalid-this
39
+ this.postMessage(data);
40
+ },
41
+ );
42
+ }
@@ -0,0 +1,63 @@
1
+ import lockr from 'lockr';
2
+
3
+ import { version } from './constants.js';
4
+
5
+ const LOCAL_STORAGE_TAB_DATA = 'vweb-';
6
+ const LOCAL_STORAGE_TAB_IDS = 'vweb1-tab-ids';
7
+ const LOCAL_STORAGE_LAST_TAB = 'vweb1-selected-tab';
8
+
9
+ const storage = {};
10
+
11
+ function isVersionOK(v) {
12
+ return v === undefined || v === version;
13
+ }
14
+
15
+ export function save(tabId, data) {
16
+ if (!tabId) return;
17
+ lockr.sadd(LOCAL_STORAGE_TAB_IDS, tabId);
18
+ let key = LOCAL_STORAGE_TAB_DATA + tabId;
19
+ data.version = version;
20
+ lockr.set(key, data);
21
+ }
22
+
23
+ export function saveSelected(tabId) {
24
+ if (!tabId) return;
25
+ lockr.set(LOCAL_STORAGE_LAST_TAB, tabId);
26
+ }
27
+
28
+ export function getSelected() {
29
+ return lockr.get(LOCAL_STORAGE_LAST_TAB);
30
+ }
31
+
32
+ export function load() {
33
+ let ids = lockr.smembers(LOCAL_STORAGE_TAB_IDS);
34
+ if (!ids) return [];
35
+
36
+ let data = ids.map((id) => {
37
+ return lockr.get(LOCAL_STORAGE_TAB_DATA + id);
38
+ });
39
+
40
+ data.forEach((entry) => {
41
+ if (!isVersionOK(entry.version)) {
42
+ storage.remove(entry.id);
43
+ }
44
+ });
45
+
46
+ data = data.filter((entry) => isVersionOK(entry.version));
47
+
48
+ data.sort(function sortData(a, b) {
49
+ let idxA = ids.indexOf(a.id);
50
+ let idxB = ids.indexOf(b.id);
51
+ if (idxA < idxB) return -1;
52
+ else if (idxB < idxA) return 1;
53
+ else return 0;
54
+ });
55
+ return data;
56
+ }
57
+
58
+ export function remove(id) {
59
+ lockr.srem(LOCAL_STORAGE_TAB_IDS, id);
60
+ lockr.rm(LOCAL_STORAGE_TAB_DATA + id);
61
+ }
62
+
63
+ export default storage;
@@ -0,0 +1,7 @@
1
+ const config = '../config.json';
2
+
3
+ const defaultConfig = {};
4
+
5
+ export function getConfig() {
6
+ return config.visualizerConfig || defaultConfig;
7
+ }
@@ -0,0 +1,75 @@
1
+ html,
2
+ body,
3
+ #visualizer-on-tabs,
4
+ .visualizer-on-tabs-app,
5
+ .visualizer-on-tabs-splash {
6
+ margin: 0;
7
+ padding: 0;
8
+ width: 100%;
9
+ height: 100%;
10
+ font-size: 14px;
11
+ }
12
+
13
+ .visualizer-on-tabs-app {
14
+ display: flex;
15
+ flex-flow: column;
16
+ position: relative;
17
+ }
18
+
19
+ .visualizer-on-tabs-splash {
20
+ position: absolute;
21
+ height: calc(100% - 3px);
22
+ display: flex;
23
+ flex-flow: column;
24
+ background: url(../images/splash.png) no-repeat center center fixed;
25
+ background-size: cover;
26
+ z-index: 1000;
27
+ }
28
+
29
+ .visualizer-on-tabs-login {
30
+ width: 500px;
31
+ margin: auto;
32
+ }
33
+
34
+ .visualizer-on-tabs-login > .form-group {
35
+ display: inline-block;
36
+ }
37
+
38
+ .visualizer-on-tabs-content,
39
+ .tab-content,
40
+ .tab-content > .active,
41
+ .visualizer-on-tabs-splash-bg {
42
+ display: flex;
43
+ flex-direction: column;
44
+ flex: 2;
45
+ }
46
+
47
+ .visualizer-on-tabs-header {
48
+ height: 40px;
49
+ }
50
+
51
+ .visualizer-on-tabs-footer {
52
+ height: 20px;
53
+ }
54
+
55
+ .visualizer-on-tabs-progress {
56
+ width: 250px;
57
+ }
58
+
59
+ .close-tab-glyph {
60
+ font-size: 18px;
61
+ font-weight: bold;
62
+ line-height: 0;
63
+ }
64
+
65
+ .close-tab-glyph:hover {
66
+ color: #ff4136;
67
+ }
68
+
69
+ .nav-tabs > li > a {
70
+ outline: none;
71
+ }
72
+
73
+ .sync-progress-bar {
74
+ background-color: #ff4136;
75
+ }
@@ -0,0 +1,15 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <link rel="stylesheet" href="static/bootstrap.min.css" />
7
+ <link rel="stylesheet" href="static/style.css" />
8
+ <title><%- title %></title>
9
+ </head>
10
+
11
+ <body>
12
+ <div id="visualizer-on-tabs"></div>
13
+ <script src="app.js?_=<%= uniqid %>"></script>
14
+ </body>
15
+ </html>
package/src/util.js ADDED
@@ -0,0 +1,29 @@
1
+ export function rewriteURL(rewriteRules, url) {
2
+ for (let i = 0; i < rewriteRules.length; i++) {
3
+ let rewriteRule = rewriteRules[i];
4
+ if (Array.isArray(rewriteRule)) {
5
+ let tmpUrl = url;
6
+ for (let j = 0; j < rewriteRule.length; j++) {
7
+ let reg = new RegExp(rewriteRule[j].reg);
8
+ let replace = rewriteRule[j].replace;
9
+ let optional = rewriteRule[j].optionalMatch;
10
+ let lastRule = j === rewriteRule.length - 1;
11
+ if (tmpUrl.match(reg) && lastRule) {
12
+ return tmpUrl.replace(reg, replace);
13
+ } else if (tmpUrl.match(reg)) {
14
+ tmpUrl = tmpUrl.replace(reg, replace);
15
+ } else if (!optional) {
16
+ break;
17
+ } else if (optional && lastRule) {
18
+ return tmpUrl;
19
+ }
20
+ }
21
+ } else {
22
+ let reg = new RegExp(rewriteRule.reg);
23
+ if (url.match(reg)) {
24
+ return url.replace(reg, rewriteRule.replace);
25
+ }
26
+ }
27
+ }
28
+ return null;
29
+ }