PageRenderTime 33ms CodeModel.GetById 18ms RepoModel.GetById 0ms app.codeStats 0ms

/build/pojoviz.js

https://github.com/artlantis/PojoViz
JavaScript | 1402 lines | 969 code | 178 blank | 255 comment | 99 complexity | b9bb55e10e271b1fcfe24df3da907f2f MD5 | raw file
  1. !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var o;"undefined"!=typeof window?o=window:"undefined"!=typeof global?o=global:"undefined"!=typeof self&&(o=self),o.pojoviz=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"dagre":[function(_dereq_,module,exports){
  2. module.exports=_dereq_('JWa/F1');
  3. },{}],"lodash":[function(_dereq_,module,exports){
  4. module.exports=_dereq_('K2RcUv');
  5. },{}],"q":[function(_dereq_,module,exports){
  6. module.exports=_dereq_('qLuPo1');
  7. },{}],4:[function(_dereq_,module,exports){
  8. var HashMap = _dereq_('./util/HashMap'),
  9. hashKey = _dereq_('./util/hashKey'),
  10. _ = _dereq_('lodash');
  11. // utils
  12. function eachObjectAndPrototype(obj, fn) {
  13. fn(obj);
  14. if (obj.hasOwnProperty('prototype')) {
  15. fn(obj.prototype);
  16. }
  17. }
  18. /**
  19. * Wraps a function with another
  20. * @param {Function} fn
  21. * @param {Function} wrapper
  22. * @return {*}
  23. */
  24. function wrapFn(fn, wrapper) {
  25. return function () {
  26. // NOTE: `this` will be the instance
  27. wrapper.call(this);
  28. var args = [].slice.call(arguments);
  29. return fn.apply(this, args);
  30. };
  31. }
  32. function isObjectOrFunction(v) {
  33. return !!(v && (typeof v === 'object' ||
  34. typeof v === 'function'));
  35. }
  36. /**
  37. * Properties forbidden in strict mode
  38. * @type {Array}
  39. */
  40. var forbiddenInStrictMode = [
  41. 'callee', 'caller', 'arguments'
  42. ];
  43. /**
  44. * @constructor
  45. * Object analyz
  46. * @param {[type]} config [description]
  47. */
  48. function Analyzer(config) {
  49. if (!(this instanceof Analyzer)) {
  50. return new Analyzer(config);
  51. }
  52. config = config || {};
  53. /**
  54. * Objects registered in this instance
  55. * @type {HashMap}
  56. */
  57. this.objects = config.objects || new HashMap();
  58. /**
  59. * Forbidden objects
  60. * @type {HashMap}
  61. */
  62. this.forbidden = config.forbidden || new HashMap();
  63. /**
  64. * Cache of properties
  65. * @type {Object}
  66. */
  67. this.__cacheObjects = {};
  68. /**
  69. * Cache of links
  70. * @type {Object}
  71. */
  72. this.__cacheLinks = {};
  73. /**
  74. * Dfs levels
  75. * @type {number}
  76. */
  77. this.levels = Infinity;
  78. /**
  79. * If the analyzer is dirty then it has some pending work
  80. * to do
  81. * @type {boolean}
  82. */
  83. this.dirty = true;
  84. /**
  85. * True to save the properties of the objects analyzed in an
  86. * internal cache
  87. * @type {Boolean}
  88. */
  89. this.cache =
  90. config.hasOwnProperty('cache') ?
  91. config.cache : true;
  92. /**
  93. * True to include function constructors in the analysis graph
  94. * i.e. the functions that have a prototype
  95. * @type {boolean}
  96. */
  97. this.functionConstructors =
  98. config.hasOwnProperty('functionConstructors') ?
  99. config.functionConstructors : false;
  100. /**
  101. * True to include all the functions in the analysis graph
  102. * @type {boolean}
  103. */
  104. this.allFunctions =
  105. config.hasOwnProperty('allFunctions') ?
  106. config.allFunctions : false;
  107. /**
  108. * True to allow HTML nodes
  109. * @type {boolean}
  110. */
  111. this.htmlNode =
  112. config.hasOwnProperty('htmlNode') ?
  113. config.htmlNode : false;
  114. }
  115. Analyzer.prototype = {
  116. constructor: Analyzer,
  117. /**
  118. * Checks if an object is in the forbidden hash
  119. * @param {Object} obj
  120. * @return {boolean}
  121. */
  122. isForbidden: function (obj) {
  123. return this.forbidden.get(obj);
  124. },
  125. isLinkable: function (key, obj) {
  126. if (!obj) {
  127. return false;
  128. }
  129. var v = typeof obj === 'object';
  130. // if (v) {
  131. // if (!this.htmlNode && v instanceof Node) { return false; }
  132. // return true;
  133. // }
  134. // if (!this.htmlNode) {
  135. // v = v && !(v instanceof Node);
  136. // }fdeq1`
  137. // typeof obj === 'function' &&
  138. // console.log(Object.getOwnPropertyNames(obj));
  139. if (!v && this.allFunctions) {
  140. // minimize the nodes created by considering functions
  141. // with more properties than the usual ones
  142. v = typeof obj === 'function';
  143. v = v && Object.getOwnPropertyNames(obj).length > 5;
  144. }
  145. if (!v && this.functionConstructors) {
  146. v = typeof obj === 'function';
  147. v = v && (
  148. obj.name &&
  149. obj.name[0].match(/^[A-Z]/) ||
  150. key[0].match(/^[A-Z]/)
  151. );
  152. }
  153. return v;
  154. },
  155. /**
  156. * Gets the enumerable properties an object discarding
  157. * forbidden ones
  158. *
  159. * @param {Object} obj
  160. * @return {Array} Array of objects, each object has the following
  161. * properties:
  162. *
  163. * - name
  164. * - cls
  165. * - type
  166. * - linkeable (if it's an object this property is set to true)
  167. */
  168. getProperties: function (obj, linkableOnly) {
  169. var me = this,
  170. hk = hashKey(obj),
  171. properties;
  172. if (!obj) {
  173. throw 'this method needs an object to analyze';
  174. }
  175. if (this.cache) {
  176. if (!linkableOnly && this.__cacheObjects[hk]) {
  177. // console.log('objects from cache :)');
  178. return this.__cacheObjects[hk];
  179. }
  180. }
  181. properties = Object.getOwnPropertyNames(obj);
  182. function forbiddenKey(v) {
  183. // forbidden in strict mode
  184. return ~forbiddenInStrictMode.indexOf(v) ||
  185. v.match(/^__.*?__$/) ||
  186. v.match(/^\$\$.*?\$\$$/) ||
  187. v.match(/[:+~!><=//\[\]@\. ]/);
  188. }
  189. properties = _.filter(properties, function (v) {
  190. var good = typeof v === 'string' && !forbiddenKey(v),
  191. r;
  192. if (linkableOnly) {
  193. try {
  194. r = good && me.isLinkable(v, obj[v]);
  195. } catch (e) {
  196. r = false;
  197. // uncomment to see why obj[v] is not allowed
  198. // console.log(e);
  199. } finally {
  200. return r;
  201. }
  202. }
  203. return good;
  204. }).map(function (v) {
  205. var type,
  206. linkeable;
  207. try {
  208. // type = null|string|undefined|number|object
  209. type = typeof obj[v];
  210. linkeable = isObjectOrFunction(obj[v]);
  211. } catch(e) {
  212. type = 'undefined';
  213. linkeable = false;
  214. }
  215. return {
  216. // parent: hashKey(obj),
  217. name: v,
  218. type: type,
  219. linkeable: linkeable
  220. };
  221. });
  222. // special properties
  223. var proto = Object.getPrototypeOf(obj);
  224. if (proto) {
  225. properties.push({
  226. name: '[[Prototype]]',
  227. // cls: hashKey(obj),
  228. type: 'object',
  229. linkeable: true,
  230. hidden: true
  231. });
  232. }
  233. var constructor = obj.hasOwnProperty &&
  234. obj.hasOwnProperty('constructor') &&
  235. typeof obj.constructor === 'function';
  236. if (constructor &&
  237. _.findIndex(properties, { name: 'constructor' }) === -1) {
  238. properties.push({
  239. // cls: hashKey(obj),
  240. name: 'constructor',
  241. type: 'function',
  242. linkeable: true
  243. });
  244. }
  245. if (this.cache && !linkableOnly) {
  246. this.__cacheObjects[hk] = properties;
  247. }
  248. // console.log(properties);
  249. return properties;
  250. },
  251. /**
  252. * Analyzes a list of objects recursively
  253. * @param {Array} objects Array of objects
  254. * @param {number} currentLevel Current dfs level
  255. */
  256. analyzeObjects: function (objects, currentLevel) {
  257. var me = this;
  258. objects.forEach(function (v) {
  259. if (currentLevel <= me.levels && // dfs level
  260. !me.objects.get(v) && // already registered
  261. !me.isForbidden(v) // forbidden check
  262. ) {
  263. // add to the registered object pool
  264. me.objects.put(v);
  265. // dfs to the next level
  266. me.analyzeObjects(
  267. me.getOwnLinks(v).map(function (link) {
  268. return link.to;
  269. }),
  270. currentLevel + 1
  271. );
  272. }
  273. });
  274. },
  275. /**
  276. * Returns a list of links, each link is an object which has the
  277. * following properties:
  278. *
  279. * - from
  280. * - to
  281. * - property (string)
  282. *
  283. * @param {Object} obj
  284. * @return {Array}
  285. */
  286. getOwnLinks: function (obj) {
  287. var me = this,
  288. links = [],
  289. properties,
  290. name = hashKey(obj);
  291. if (this.__cacheLinks[name]) {
  292. // console.log('links from cache :)');
  293. return this.__cacheLinks[name];
  294. }
  295. properties = me.getProperties(obj, true);
  296. function getAugmentedHash(obj, name) {
  297. if (!hashKey.has(obj) &&
  298. name !== 'prototype' &&
  299. name !== 'constructor') {
  300. hashKey.createHashKeysFor(obj, name);
  301. }
  302. return hashKey(obj);
  303. }
  304. if (!name) {
  305. throw 'the object needs to have a hashkey';
  306. }
  307. _.forEach(properties, function (v) {
  308. var ref = obj[v.name];
  309. // because of the levels a reference might not exist
  310. if (!ref) {
  311. return;
  312. }
  313. // if the object doesn't have a hashkey
  314. // let's give it a name equal to the property
  315. // being analyzed
  316. getAugmentedHash(ref, v.name);
  317. if (!me.isForbidden(ref)) {
  318. links.push({
  319. from: obj,
  320. fromHash: hashKey(obj),
  321. to: ref,
  322. toHash: hashKey(ref),
  323. property: v.name
  324. });
  325. }
  326. });
  327. var proto = Object.getPrototypeOf(obj);
  328. if (proto && !me.isForbidden(proto)) {
  329. links.push({
  330. from: obj,
  331. fromHash: hashKey(obj),
  332. to: proto,
  333. toHash: hashKey(proto),
  334. property: '[[Prototype]]'
  335. });
  336. }
  337. if (this.cache) {
  338. this.__cacheLinks[name] = links;
  339. }
  340. return links;
  341. },
  342. makeDirty: function () {
  343. this.dirty = true;
  344. },
  345. setLevels: function (l) {
  346. this.levels = l;
  347. },
  348. setDirty: function (d) {
  349. this.dirty = d;
  350. },
  351. setFunctionConstructors: function (v) {
  352. this.functionConstructors = v;
  353. },
  354. getObjects: function () {
  355. return this.objects;
  356. },
  357. /**
  358. * Stringifies an object properties
  359. * @param obj
  360. * @param toString
  361. * @return {Array}
  362. */
  363. stringifyObjectProperties: function (obj) {
  364. return this.getProperties(obj);
  365. },
  366. /**
  367. * Returns a representation of the links of
  368. * an object
  369. * @return {Array}
  370. */
  371. stringifyObjectLinks: function (obj) {
  372. var me = this;
  373. return me.getOwnLinks(obj).map(function (link) {
  374. // discarded: from, to
  375. return {
  376. from: link.fromHash,
  377. to: link.toHash,
  378. property: link.property
  379. };
  380. });
  381. },
  382. /**
  383. * Stringifies the objects saved in this analyzer
  384. * @return {Object}
  385. */
  386. stringify: function () {
  387. var me = this,
  388. nodes = {},
  389. edges = {};
  390. console.time('stringify');
  391. _(this.objects).forOwn(function (v) {
  392. nodes[hashKey(v)] = me.stringifyObjectProperties(v);
  393. edges[hashKey(v)] = me.stringifyObjectLinks(v);
  394. });
  395. console.timeEnd('stringify');
  396. return {
  397. nodes: nodes,
  398. edges: edges
  399. };
  400. }
  401. };
  402. // aditional objects that need the prototype to exist
  403. var aProto = Analyzer.prototype;
  404. _.merge(aProto, {
  405. /**
  406. * Adds a list of objects to analyze and make the analyzer dirty
  407. * @param {Array<Objects>} objects
  408. * @return {this}
  409. */
  410. add: wrapFn(function (objects) {
  411. console.time('analyze');
  412. this.analyzeObjects(objects, 0);
  413. console.timeEnd('analyze');
  414. return this;
  415. }, aProto.makeDirty),
  416. /**
  417. * Removes a list of objects, if `withPrototype` is true then
  418. * also the prototype is removed
  419. * @param {Array<Objects>} objects
  420. * @param {boolean} withPrototype
  421. * @return {this}
  422. */
  423. remove: wrapFn(function (objects, withPrototype) {
  424. var me = this;
  425. objects.forEach(function (obj) {
  426. me.objects.remove(obj);
  427. if (withPrototype && obj.hasOwnProperty('prototype')) {
  428. me.objects.remove(obj.prototype);
  429. }
  430. });
  431. return me;
  432. }, aProto.makeDirty),
  433. /**
  434. * Forbids a list of objects, if `withPrototype` is true then
  435. * also the prototype is forbidden
  436. * @param {Array<Objects>} objects
  437. * @param {boolean} withPrototype
  438. * @return {this}
  439. */
  440. forbid: wrapFn(function (objects, withPrototype) {
  441. var me = this;
  442. me.remove(objects, withPrototype);
  443. objects.forEach(function (obj) {
  444. me.forbidden.put(obj);
  445. if (withPrototype && obj.hasOwnProperty('prototype')) {
  446. me.forbidden.put(obj.prototype);
  447. }
  448. });
  449. }, aProto.makeDirty),
  450. /**
  451. * Unforbids a list of objects, if `withPrototype` is true then
  452. * also the prototype is unforbidden
  453. * @param {Array<Objects>} objects
  454. * @param {boolean} withPrototype
  455. * @return {this}
  456. */
  457. unforbid: wrapFn(function (objects, withPrototype) {
  458. var me = this;
  459. objects.forEach(function (obj) {
  460. me.forbidden.remove(obj);
  461. if (withPrototype && obj.hasOwnProperty('prototype')) {
  462. me.forbidden.remove(obj.prototype);
  463. }
  464. });
  465. }, aProto.makeDirty)
  466. });
  467. module.exports = Analyzer;
  468. },{"./util/HashMap":12,"./util/hashKey":13,"lodash":"K2RcUv"}],5:[function(_dereq_,module,exports){
  469. 'use strict';
  470. var _ = _dereq_('lodash'),
  471. Generic = _dereq_('./analyzer/GenericAnalyzer'),
  472. Angular = _dereq_('./analyzer/Angular'),
  473. Window = _dereq_('./analyzer/Window'),
  474. PObject = _dereq_('./analyzer/Object'),
  475. BuiltIn = _dereq_('./analyzer/BuiltIn');
  476. var libraries;
  477. var proto = {
  478. createNew: function (global, options) {
  479. console.log('creating a generic container for: ' + global, options);
  480. return (libraries[global] = new Generic(options));
  481. },
  482. all: function (fn) {
  483. _.forOwn(libraries, fn);
  484. },
  485. markDirty: function () {
  486. proto.all(function (v) {
  487. v.markDirty();
  488. });
  489. return proto;
  490. },
  491. setFunctionConstructors: function (newValue) {
  492. proto.all(function (v) {
  493. // this only works on the generic analyzers
  494. if (!v._hasfc) {
  495. v.analyzer.setFunctionConstructors(newValue);
  496. }
  497. });
  498. return proto;
  499. }
  500. };
  501. libraries = Object.create(proto);
  502. _.merge(libraries, {
  503. object: new PObject(),
  504. builtIn: new BuiltIn(),
  505. window: new Window(),
  506. // popular
  507. angular: new Angular(),
  508. // mine
  509. // t3: new Generic({ global: 't3' }),
  510. // huge
  511. three: new Generic({
  512. global: 'THREE',
  513. rendereachtime: true
  514. })
  515. });
  516. // console.log(libraries);
  517. // win max level initially is 0
  518. // libraries.win.preRender = function () {
  519. // libraries.win.getObjects().empty();
  520. // libraries.win.analyzeObjects([window], 0);
  521. // };
  522. // console.log(builtIn.getObjects());
  523. // console.log(win.getObjects());
  524. // console.log(user.getObjects());
  525. module.exports = libraries;
  526. },{"./analyzer/Angular":6,"./analyzer/BuiltIn":7,"./analyzer/GenericAnalyzer":8,"./analyzer/Object":9,"./analyzer/Window":10,"lodash":"K2RcUv"}],6:[function(_dereq_,module,exports){
  527. 'use strict';
  528. var GenericAnalyzer = _dereq_('./GenericAnalyzer'),
  529. hashKey = _dereq_('../util/hashKey');
  530. function Angular() {
  531. GenericAnalyzer.call(this, {
  532. global: 'angular',
  533. displayname: 'AngularJS',
  534. rendereachtime: true
  535. });
  536. this.services = [
  537. '$animate',
  538. '$cacheFactory',
  539. '$compile',
  540. '$controller',
  541. // '$document',
  542. '$exceptionHandler',
  543. '$filter',
  544. '$http',
  545. '$httpBackend',
  546. '$interpolate',
  547. '$interval',
  548. '$locale',
  549. '$log',
  550. '$parse',
  551. '$q',
  552. '$rootScope',
  553. '$sce',
  554. '$sceDelegate',
  555. '$templateCache',
  556. '$timeout',
  557. // '$window'
  558. ].map(function (v) {
  559. return { checked: true, name: v };
  560. });
  561. }
  562. Angular.prototype = Object.create(GenericAnalyzer.prototype);
  563. Angular.prototype.getSelectedServices = function () {
  564. var me = this,
  565. toAnalyze = [];
  566. window.angular.module('app', ['ng']);
  567. this.injector = window.angular.injector(['app']);
  568. me.services.forEach(function (s) {
  569. if (s.checked) {
  570. var obj = me.injector.get(s.name);
  571. hashKey.createHashKeysFor(obj, s.name);
  572. toAnalyze.push(obj);
  573. }
  574. });
  575. return toAnalyze;
  576. };
  577. Angular.prototype.inspectSelf = function () {
  578. console.log('inspecting angular');
  579. hashKey.createHashKeysFor(window.angular, 'angular');
  580. this.analyzer.getObjects().empty();
  581. this.analyzer.add(
  582. [window.angular].concat(this.getSelectedServices())
  583. );
  584. };
  585. module.exports = Angular;
  586. },{"../util/hashKey":13,"./GenericAnalyzer":8}],7:[function(_dereq_,module,exports){
  587. 'use strict';
  588. var GenericAnalyzer = _dereq_('./GenericAnalyzer'),
  589. utils = _dereq_('../util');
  590. var toInspect = [
  591. Object, Function,
  592. Array, Date, Boolean, Number, Math, String, RegExp, JSON,
  593. Error
  594. ];
  595. function BuiltIn() {
  596. GenericAnalyzer.call(this);
  597. }
  598. BuiltIn.prototype = Object.create(GenericAnalyzer.prototype);
  599. BuiltIn.prototype.inspectSelf = function () {
  600. console.log('inspecting builtIn objects');
  601. this.analyzer.add(this.getObjects());
  602. };
  603. BuiltIn.prototype.getObjects = function () {
  604. return toInspect;
  605. };
  606. BuiltIn.prototype.showSearch = function (nodeName, nodeProperty) {
  607. var url = 'https://developer.mozilla.org/en-US/search?' +
  608. utils.toQueryString({
  609. q: encodeURIComponent(nodeName + ' ' + nodeProperty),
  610. });
  611. window.open(url);
  612. };
  613. module.exports = BuiltIn;
  614. },{"../util":14,"./GenericAnalyzer":8}],8:[function(_dereq_,module,exports){
  615. 'use strict';
  616. var Q = _dereq_('q'),
  617. _ = _dereq_('lodash'),
  618. utils = _dereq_('../util/'),
  619. hashKey = _dereq_('../util/hashKey'),
  620. analyzer = _dereq_('../ObjectAnalyzer');
  621. var searchEngine = 'https://duckduckgo.com/?q=';
  622. function GenericAnalyzer(options) {
  623. options = options || {};
  624. // if (!name) {
  625. // throw 'name needs to be defined';
  626. // }
  627. this.global = options.global;
  628. this.displayname = options.displayname;
  629. this.levels = options.hasOwnProperty('levels') ? options.levels : 10;
  630. this.forbidden = options.forbidden || [];
  631. this.src = options.src;
  632. this._hasfc = options.hasOwnProperty('functionconstructors');
  633. this.functionconstructors = this._hasfc ?
  634. options.functionconstructors : GenericAnalyzer.SHOW_FUNCTION_CONSTRUCTORS;
  635. this.rendereachtime = options.hasOwnProperty('rendereachtime') ?
  636. options.rendereachtime : false;
  637. this.allfunctions = options.hasOwnProperty('allfunctions') ?
  638. options.allfunctions : false;
  639. this.inspected = false;
  640. // parse forbid string to array
  641. this.parse();
  642. this.analyzer = analyzer({
  643. functionConstructors: this.functionconstructors,
  644. allFunctions: this.allfunctions
  645. });
  646. }
  647. GenericAnalyzer.SHOW_BUILTIN = false;
  648. GenericAnalyzer.SHOW_FUNCTION_CONSTRUCTORS = true;
  649. GenericAnalyzer.FORBIDDEN = 'pojoviz:window,pojoviz:builtIn,document';
  650. GenericAnalyzer.prototype.init = function () {
  651. var me = this;
  652. console.log('%cPojoViz', 'font-size: 15px; color: ');
  653. return me.fetch()
  654. .then(function () {
  655. if (me.rendereachtime || !me.inspected) {
  656. me.inspect();
  657. }
  658. return me;
  659. });
  660. };
  661. GenericAnalyzer.prototype.parse = function () {
  662. if (typeof this.forbidden === 'string') {
  663. this.forbidden = this.forbidden.split(',');
  664. }
  665. if (typeof this.functionconstructors === 'string') {
  666. this.functionconstructors = this.functionconstructors === 'true';
  667. }
  668. if (typeof this.rendereachtime === 'string') {
  669. this.rendereachtime = this.rendereachtime === 'true';
  670. }
  671. if (typeof this.allfunctions === 'string') {
  672. this.allfunctions = this.allfunctions === 'true';
  673. }
  674. };
  675. GenericAnalyzer.prototype.markDirty = function () {
  676. this.inspected = false;
  677. };
  678. GenericAnalyzer.prototype.inspectSelf = function () {
  679. console.log('analyzing window.' + this.global);
  680. var me = this,
  681. analyzer = this.analyzer,
  682. forbidden = [].concat(this.forbidden);
  683. // set a predefied global
  684. hashKey.createHashKeysFor(window[this.global], this.global);
  685. // clean
  686. analyzer.getObjects().empty();
  687. analyzer.forbidden.empty();
  688. analyzer.setLevels(this.levels);
  689. // settings > show links to built in objects
  690. if (!GenericAnalyzer.SHOW_BUILTIN) {
  691. forbidden = forbidden.concat(
  692. GenericAnalyzer.FORBIDDEN.split(',')
  693. );
  694. }
  695. forbidden.forEach(function(f) {
  696. var arr,
  697. tokens;
  698. if (!f.indexOf('pojoviz:')) {
  699. tokens = f.split(':');
  700. arr = _dereq_('../ObjectHashes')[tokens[1]].getObjects();
  701. } else {
  702. arr = [window[f]];
  703. }
  704. console.log('forbidding: ', arr);
  705. analyzer.forbid(arr, true);
  706. });
  707. analyzer.add([window[this.global]]);
  708. };
  709. GenericAnalyzer.prototype.markInspected = function () {
  710. // mark this container as inspected
  711. this.inspected = true;
  712. return this;
  713. };
  714. GenericAnalyzer.prototype.inspect = function () {
  715. this
  716. .markInspected()
  717. .inspectSelf();
  718. };
  719. GenericAnalyzer.prototype.preRender = function () {
  720. };
  721. GenericAnalyzer.prototype.fetch = function () {
  722. var me = this,
  723. script;
  724. function getValue() {
  725. return window[me.global];
  726. }
  727. function promisify(v) {
  728. return function () {
  729. utils.notification('fetching script ' + v, true);
  730. var deferred = Q.defer();
  731. script = document.createElement('script');
  732. script.src = v;
  733. script.onload = function () {
  734. utils.notification('completed script ' + v, true);
  735. deferred.resolve(getValue());
  736. };
  737. document.head.appendChild(script);
  738. return deferred.promise;
  739. };
  740. }
  741. if (this.src) {
  742. if (getValue()) {
  743. console.log('resource already fetched ' + this.src);
  744. } else {
  745. var srcs = this.src.split('|');
  746. return srcs.reduce(function (prev, current) {
  747. return prev.then(promisify(current));
  748. }, Q('reduce'));
  749. }
  750. }
  751. return Q(true);
  752. };
  753. GenericAnalyzer.prototype.showSearch = function (nodeName, nodeProperty) {
  754. var me = this;
  755. window.open(
  756. _.template('${searchEngine}${lucky}${libraryName} ${nodeName} ${nodeProperty}', {
  757. searchEngine: searchEngine,
  758. lucky: GenericAnalyzer.lucky ? '!ducky' : '',
  759. libraryName: me.displayname || me.global,
  760. nodeName: nodeName,
  761. nodeProperty: nodeProperty
  762. })
  763. );
  764. };
  765. module.exports = GenericAnalyzer;
  766. },{"../ObjectAnalyzer":4,"../ObjectHashes":5,"../util/":14,"../util/hashKey":13,"lodash":"K2RcUv","q":"qLuPo1"}],9:[function(_dereq_,module,exports){
  767. 'use strict';
  768. var GenericAnalyzer = _dereq_('./GenericAnalyzer'),
  769. utils = _dereq_('../util');
  770. function PObject() {
  771. GenericAnalyzer.call(this);
  772. }
  773. PObject.prototype = Object.create(GenericAnalyzer.prototype);
  774. PObject.prototype.inspectSelf = function () {
  775. console.log('inspecting Object objects');
  776. this.analyzer.add(this.getObjects());
  777. };
  778. PObject.prototype.getObjects = function () {
  779. return [Object];
  780. };
  781. module.exports = PObject;
  782. },{"../util":14,"./GenericAnalyzer":8}],10:[function(_dereq_,module,exports){
  783. 'use strict';
  784. var _ = _dereq_('lodash'),
  785. hashKey = _dereq_('../util/hashKey'),
  786. GenericAnalyzer = _dereq_('./GenericAnalyzer');
  787. var toInspect = [window];
  788. function Window() {
  789. GenericAnalyzer.call(this, {
  790. levels: 1,
  791. rendereachtime: true,
  792. functionconstructors: false
  793. });
  794. }
  795. Window.prototype = Object.create(GenericAnalyzer.prototype);
  796. Window.prototype.getObjects = function () {
  797. return toInspect;
  798. };
  799. Window.prototype.inspectSelf = function () {
  800. console.log('inspecting window');
  801. var me = this,
  802. hashes = _dereq_('../ObjectHashes');
  803. _.forOwn(hashes, function (v, k) {
  804. if (v.global && window[v.global]) {
  805. me.analyzer.forbid([window[v.global]], true);
  806. }
  807. });
  808. this.analyzer.getObjects().empty();
  809. this.analyzer.setLevels(this.levels);
  810. this.analyzer.add(me.getObjects());
  811. };
  812. module.exports = Window;
  813. },{"../ObjectHashes":5,"../util/hashKey":13,"./GenericAnalyzer":8,"lodash":"K2RcUv"}],11:[function(_dereq_,module,exports){
  814. var _ = _dereq_('lodash'),
  815. Q = _dereq_('q'),
  816. dagre = _dereq_('dagre'),
  817. utils = _dereq_('./util/'),
  818. ObjectHashes = _dereq_('./ObjectHashes');
  819. // enable long stacks
  820. Q.longStackSupport = true;
  821. var container,
  822. oldContainer,
  823. oldRenderer,
  824. renderer,
  825. pojoviz; // namespace
  826. function process() {
  827. var g = new dagre.Digraph(),
  828. properties,
  829. node,
  830. library = container.analyzer,
  831. str = library.stringify(),
  832. libraryNodes = str.nodes,
  833. libraryEdges = str.edges;
  834. // create the graph
  835. // each element of the graph has
  836. // - label
  837. // - width
  838. // - height
  839. // - properties
  840. _.forOwn(libraryNodes, function (properties, k) {
  841. var label = k.match(/\S*?-(.*)/)[1];
  842. // console.log(k, label.length);
  843. node = {
  844. label: k,
  845. width: label.length * 10
  846. };
  847. // lines + header + padding bottom
  848. node.height = properties.length * 15 + 50;
  849. node.properties = properties;
  850. properties.forEach(function (v) {
  851. node.width = Math.max(node.width, v.name.length * 10);
  852. });
  853. g.addNode(k, node);
  854. });
  855. // build the edges from node to node
  856. _.forOwn(libraryEdges, function (links) {
  857. links.forEach(function (link) {
  858. if (g.hasNode(link.from) && g.hasNode(link.to)) {
  859. g.addEdge(null, link.from, link.to);
  860. }
  861. });
  862. });
  863. // layout of the graph
  864. var layout = dagre.layout()
  865. .nodeSep(30)
  866. // .rankSep(70)
  867. // .rankDir('TB')
  868. .run(g);
  869. var nodes = [],
  870. edges = [],
  871. center = {x: 0, y: 0},
  872. mn = {x: Infinity, y: Infinity},
  873. mx = {x: -Infinity, y: -Infinity},
  874. total = g.nodes().length;
  875. // update the node info of the node adding:
  876. // - x
  877. // - y
  878. // - predecessors
  879. // - successors
  880. layout.eachNode(function (k, layoutInfo) {
  881. var x = layoutInfo.x;
  882. var y = layoutInfo.y;
  883. node = g.node(k);
  884. node.x = x;
  885. node.y = y;
  886. node.predecessors = g.predecessors(k);
  887. node.successors = g.successors(k);
  888. nodes.push(node);
  889. // calculate the bbox of the graph to center the graph
  890. var mnx = x - node.width / 2;
  891. var mny = y - node.height / 2;
  892. var mxx = x + node.width / 2;
  893. var mxy = y + node.height / 2;
  894. center.x += x;
  895. center.y += y;
  896. mn.x = Math.min(mn.x, mnx);
  897. mn.y = Math.min(mn.y, mny);
  898. // console.log(x, y, ' dim ', node.width, node.height);
  899. mx.x = Math.max(mx.x, mxx);
  900. mx.y = Math.max(mx.y, mxy);
  901. });
  902. center.x /= (total || 1);
  903. center.y /= (total || 1);
  904. // create the edges from property to node
  905. _(libraryEdges).forOwn(function (links) {
  906. links.forEach(function (link) {
  907. if (g.hasNode(link.from) && g.hasNode(link.to)) {
  908. edges.push(link);
  909. }
  910. });
  911. });
  912. return {
  913. edges: edges,
  914. nodes: nodes,
  915. center: center,
  916. mn: mn,
  917. mx: mx
  918. };
  919. }
  920. // render
  921. function render() {
  922. var data;
  923. if (container === oldContainer) {
  924. return;
  925. }
  926. utils.notification('processing ' + container.global);
  927. // pre render
  928. oldRenderer && oldRenderer.clean();
  929. renderer.clean();
  930. setTimeout(function () {
  931. container.preRender();
  932. console.log('process & render start: ', new Date());
  933. // data has
  934. // - edges (property -> node)
  935. // - nodes
  936. // - center
  937. //
  938. console.time('process');
  939. data = process();
  940. console.timeEnd('process');
  941. utils.notification('rendering ' + container.global);
  942. console.time('render');
  943. renderer.render(data);
  944. console.timeEnd('render');
  945. utils.notification('complete!');
  946. }, 0);
  947. }
  948. // public api
  949. pojoviz = {
  950. renderers: {},
  951. addRenderers: function (newRenderers) {
  952. _.merge(pojoviz.renderers, newRenderers);
  953. },
  954. nullifyContainer: function () {
  955. oldContainer = container;
  956. container = null;
  957. },
  958. getContainer: function () {
  959. return container;
  960. },
  961. setContainer: function (containerName, options) {
  962. oldContainer = container;
  963. container = ObjectHashes[containerName];
  964. if (!container) {
  965. container = ObjectHashes.createNew(containerName, options);
  966. } else {
  967. // required to fetch external resources
  968. container.src = options.src;
  969. }
  970. return container.init();
  971. },
  972. setRenderer: function (r) {
  973. oldRenderer = renderer;
  974. renderer = pojoviz.renderers[r];
  975. },
  976. getRenderer: function () {
  977. return renderer;
  978. },
  979. render: render,
  980. // expose inner modules
  981. ObjectHashes: _dereq_('./ObjectHashes'),
  982. ObjectAnalyzer: _dereq_('./ObjectAnalyzer'),
  983. analyzer: {
  984. GenericAnalyzer: _dereq_('./analyzer/GenericAnalyzer')
  985. },
  986. utils: _dereq_('./util'),
  987. // user vars
  988. userVariables: []
  989. };
  990. // custom events
  991. document.addEventListener('property-click', function (e) {
  992. var detail = e.detail;
  993. pojoviz
  994. .getContainer()
  995. .showSearch(detail.name, detail.property);
  996. });
  997. module.exports = pojoviz;
  998. },{"./ObjectAnalyzer":4,"./ObjectHashes":5,"./analyzer/GenericAnalyzer":8,"./util":14,"./util/":14,"dagre":"JWa/F1","lodash":"K2RcUv","q":"qLuPo1"}],12:[function(_dereq_,module,exports){
  999. 'use strict';
  1000. var hashKey = _dereq_('./hashKey');
  1001. function HashMap() {
  1002. }
  1003. HashMap.prototype = {
  1004. put: function (key, value) {
  1005. this[hashKey(key)] = (value || key);
  1006. },
  1007. get: function (key) {
  1008. return this[hashKey(key)];
  1009. },
  1010. remove: function (key) {
  1011. var v = this[hashKey(key)];
  1012. delete this[hashKey(key)];
  1013. return v;
  1014. },
  1015. empty: function () {
  1016. var p,
  1017. me = this;
  1018. for (p in me) {
  1019. if (me.hasOwnProperty(p)) {
  1020. delete this[p];
  1021. }
  1022. }
  1023. }
  1024. };
  1025. module.exports = HashMap;
  1026. },{"./hashKey":13}],13:[function(_dereq_,module,exports){
  1027. 'use strict';
  1028. var _ = _dereq_('lodash'),
  1029. assert = _dereq_('./').assert,
  1030. me, hashKey;
  1031. function isObjectOrFunction(v) {
  1032. return v && (typeof v === 'object' || typeof v === 'function');
  1033. }
  1034. /**
  1035. * Gets a store hashkey only if it's an object
  1036. * @param {[type]} obj
  1037. * @return {[type]}
  1038. */
  1039. function get(obj) {
  1040. assert(isObjectOrFunction(obj), 'obj must be an object|function');
  1041. return obj.hasOwnProperty &&
  1042. obj.hasOwnProperty(me.hiddenKey) &&
  1043. obj[me.hiddenKey];
  1044. }
  1045. /**
  1046. * Sets a key on an object
  1047. * @param {[type]} obj [description]
  1048. * @param {[type]} key [description]
  1049. */
  1050. function set(obj, key) {
  1051. assert(isObjectOrFunction(obj), 'obj must be an object|function');
  1052. assert(
  1053. key && typeof key === 'string',
  1054. 'The key needs to be a valid string'
  1055. );
  1056. if (!get(obj)) {
  1057. Object.defineProperty(obj, me.hiddenKey, {
  1058. value: typeof obj + '-' + key
  1059. });
  1060. }
  1061. return me;
  1062. }
  1063. me = hashKey = function (v) {
  1064. var value = v,
  1065. uid = v;
  1066. if (isObjectOrFunction(v)) {
  1067. if (!get(v)) {
  1068. me.createHashKeysFor(v);
  1069. }
  1070. uid = get(v);
  1071. if (!uid) {
  1072. console.err('no hashkey :(', v);
  1073. }
  1074. assert(uid, 'error getting the key');
  1075. return uid;
  1076. }
  1077. // v is a primitive
  1078. return typeof v + '-' + uid;
  1079. };
  1080. me.hiddenKey = '__pojoVizKey__';
  1081. me.createHashKeysFor = function (obj, name) {
  1082. function localToString(obj) {
  1083. var match;
  1084. try {
  1085. match = {}.toString.call(obj).match(/^\[object (\S*?)\]/);
  1086. } catch (e) {
  1087. match = false;
  1088. }
  1089. return match && match[1];
  1090. }
  1091. /**
  1092. * Analyze the internal property [[Class]] to guess the name
  1093. * of this object, e.g. [object Date], [object Math]
  1094. * Many object will give false positives (they will match [object Object])
  1095. * so let's consider Object as the name only if it's equal to
  1096. * Object.prototype
  1097. * @param {Object} obj
  1098. * @return {Boolean}
  1099. */
  1100. function hasAClassName(obj) {
  1101. var match = localToString(obj);
  1102. if (match === 'Object') {
  1103. return obj === Object.prototype && 'Object';
  1104. }
  1105. return match;
  1106. }
  1107. function getName(obj) {
  1108. var name, className;
  1109. // return the already generated hashKey
  1110. if (get(obj)) {
  1111. return get(obj);
  1112. }
  1113. // generate a new key based on
  1114. // - the name if it's a function
  1115. // - a unique id
  1116. name = typeof obj === 'function' &&
  1117. typeof obj.name === 'string' &&
  1118. obj.name;
  1119. className = hasAClassName(obj);
  1120. if (!name && className) {
  1121. name = className;
  1122. }
  1123. name = name || _.uniqueId();
  1124. return name;
  1125. }
  1126. // the name is equal to the passed name or the
  1127. // generated name
  1128. name = name || getName(obj);
  1129. name = name.replace(/[\. ]/img, '-');
  1130. // if the obj is a prototype then try to analyze
  1131. // the constructor first so that the prototype becomes
  1132. // [name].prototype
  1133. // special case: object.constructor = object
  1134. if (obj.hasOwnProperty &&
  1135. obj.hasOwnProperty('constructor') &&
  1136. typeof obj.constructor === 'function' &&
  1137. obj.constructor !== obj) {
  1138. me.createHashKeysFor(obj.constructor);
  1139. }
  1140. // set name on self
  1141. set(obj, name);
  1142. // set name on the prototype
  1143. if (typeof obj === 'function' &&
  1144. obj.hasOwnProperty('prototype')) {
  1145. set(obj.prototype, name + '-prototype');
  1146. }
  1147. };
  1148. me.has = function (v) {
  1149. return v.hasOwnProperty &&
  1150. v.hasOwnProperty(me.hiddenKey);
  1151. };
  1152. module.exports = me;
  1153. },{"./":14,"lodash":"K2RcUv"}],14:[function(_dereq_,module,exports){
  1154. 'use strict';
  1155. var _ = _dereq_('lodash');
  1156. var propertiesTransformation = {
  1157. '[[Prototype]]': '__proto__'
  1158. };
  1159. var utils = {
  1160. assert: function (v, message) {
  1161. if (!v) {
  1162. throw message || 'error';
  1163. }
  1164. },
  1165. translate: function (x, y) {
  1166. return 'translate(' + (x || 0) + ', ' + (y || 0) + ')';
  1167. },
  1168. scale: function (s) {
  1169. return 'scale(' + (s || 1) + ')';
  1170. },
  1171. transform: function (obj) {
  1172. var t = [];
  1173. _.forOwn(obj, function (v, k) {
  1174. t.push(utils[k].apply(utils, v));
  1175. });
  1176. return t.join(' ');
  1177. },
  1178. prefixer: function () {
  1179. var args = [].slice.call(arguments);
  1180. args.unshift('pv');
  1181. return args.join('-');
  1182. },
  1183. transformProperty: function (v) {
  1184. if (propertiesTransformation.hasOwnProperty(v)) {
  1185. return propertiesTransformation[v];
  1186. }
  1187. return v;
  1188. },
  1189. escapeCls: function(v) {
  1190. return v.replace(/\$/g, '_');
  1191. },
  1192. toQueryString: function (obj) {
  1193. var s = '',
  1194. i = 0;
  1195. _.forOwn(obj, function (v, k) {
  1196. if (i) {
  1197. s += '&';
  1198. }
  1199. s += k + '=' + v;
  1200. i += 1;
  1201. });
  1202. return s;
  1203. },
  1204. createEvent: function (eventName, details) {
  1205. return new CustomEvent(eventName, {
  1206. detail: details
  1207. });
  1208. },
  1209. notification: function (message, consoleToo) {
  1210. var ev = utils.createEvent('pojoviz-notification', message);
  1211. consoleToo && console.log(message);
  1212. document.dispatchEvent(ev);
  1213. },
  1214. createJsonpCallback: function (url) {
  1215. var script = document.createElement('script');
  1216. script.src = url;
  1217. document.head.appendChild(script);
  1218. }
  1219. };
  1220. module.exports = utils;
  1221. },{"lodash":"K2RcUv"}]},{},[11])
  1222. //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/ObjectAnalyzer.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/ObjectHashes.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/analyzer/Angular.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/analyzer/BuiltIn.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/analyzer/GenericAnalyzer.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/analyzer/Object.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/analyzer/Window.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/index.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/util/HashMap.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/util/hashKey.js","/Users/mauricio/Documents/web/maurizzzio.me/pojoviz/src/util/index.js"],"names":[],"mappings":"AAAA;;;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACngBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var HashMap = require('./util/HashMap'),\n  hashKey = require('./util/hashKey'),\n  _ = require('lodash');\n\n// utils\nfunction eachObjectAndPrototype(obj, fn) {\n  fn(obj);\n  if (obj.hasOwnProperty('prototype')) {\n    fn(obj.prototype);\n  }\n}\n\n/**\n * Wraps a function with another\n * @param  {Function} fn\n * @param  {Function}   wrapper\n * @return {*}\n */\nfunction wrapFn(fn, wrapper) {\n  return function () {\n    // NOTE: `this` will be the instance\n    wrapper.call(this);\n    var args = [].slice.call(arguments);\n    return fn.apply(this, args);\n  };\n}\n\nfunction isObjectOrFunction(v) {\n  return !!(v && (typeof v === 'object' ||\n    typeof v === 'function'));\n}\n\n/**\n * Properties forbidden in strict mode\n * @type {Array}\n */\nvar forbiddenInStrictMode = [\n  'callee', 'caller', 'arguments'\n];\n\n/**\n * @constructor\n * Object analyz\n * @param {[type]} config [description]\n */\nfunction Analyzer(config) {\n  if (!(this instanceof Analyzer)) {\n    return new Analyzer(config);\n  }\n  config = config || {};\n\n  /**\n   * Objects registered in this instance\n   * @type {HashMap}\n   */\n  this.objects = config.objects || new HashMap();\n  /**\n   * Forbidden objects\n   * @type {HashMap}\n   */\n  this.forbidden = config.forbidden || new HashMap();\n\n  /**\n   * Cache of properties\n   * @type {Object}\n   */\n  this.__cacheObjects = {};\n\n  /**\n   * Cache of links\n   * @type {Object}\n   */\n  this.__cacheLinks = {};\n\n  /**\n   * Dfs levels\n   * @type {number}\n   */\n  this.levels = Infinity;\n  /**\n   * If the analyzer is dirty then it has some pending work\n   * to do\n   * @type {boolean}\n   */\n  this.dirty = true;\n\n  /**\n   * True to save the properties of the objects analyzed in an\n   * internal cache\n   * @type {Boolean}\n   */\n  this.cache =\n    config.hasOwnProperty('cache') ?\n    config.cache : true;\n  /**\n   * True to include function constructors in the analysis graph\n   * i.e. the functions that have a prototype\n   * @type {boolean}\n   */\n  this.functionConstructors =\n    config.hasOwnProperty('functionConstructors') ?\n    config.functionConstructors : false;\n  /**\n   * True to include all the functions in the analysis graph\n   * @type {boolean}\n   */\n  this.allFunctions =\n    config.hasOwnProperty('allFunctions') ?\n    config.allFunctions : false;\n  /**\n   * True to allow HTML nodes\n   * @type {boolean}\n   */\n  this.htmlNode =\n    config.hasOwnProperty('htmlNode') ?\n    config.htmlNode : false;\n}\n\nAnalyzer.prototype = {\n  constructor: Analyzer,\n\n  /**\n   * Checks if an object is in the forbidden hash\n   * @param  {Object}  obj\n   * @return {boolean}\n   */\n  isForbidden: function (obj) {\n    return this.forbidden.get(obj);\n  },\n\n  isLinkable: function (key, obj) {\n    if (!obj) {\n      return false;\n    }\n\n    var v = typeof obj === 'object';\n\n    // if (v) {\n    //   if (!this.htmlNode && v instanceof Node) { return false; }\n    //   return true;\n    // }\n\n    // if (!this.htmlNode) {\n    //   v = v && !(v instanceof Node);\n    // }fdeq1`\n\n    // typeof obj === 'function' &&\n    //   console.log(Object.getOwnPropertyNames(obj));\n    if (!v && this.allFunctions) {\n      // minimize the nodes created by considering functions\n      // with more properties than the usual ones\n      v = typeof obj === 'function';\n      v = v && Object.getOwnPropertyNames(obj).length > 5;\n    }\n    if (!v && this.functionConstructors) {\n      v = typeof obj === 'function';\n      v = v && (\n        obj.name &&\n        obj.name[0].match(/^[A-Z]/) ||\n        key[0].match(/^[A-Z]/)\n      );\n    }\n    return v;\n  },\n\n  /**\n   * Gets the enumerable properties an object discarding\n   * forbidden ones\n   *\n   * @param  {Object} obj\n   * @return {Array} Array of objects, each object has the following\n   * properties:\n   *\n   * - name\n   * - cls\n   * - type\n   * - linkeable (if it's an object this property is set to true)\n   */\n  getProperties: function (obj, linkableOnly) {\n    var me = this,\n      hk = hashKey(obj),\n      properties;\n\n    if (!obj) {\n      throw 'this method needs an object to analyze';\n    }\n\n    if (this.cache) {\n      if (!linkableOnly && this.__cacheObjects[hk]) {\n        // console.log('objects from cache :)');\n        return this.__cacheObjects[hk];\n      }\n    }\n\n    properties = Object.getOwnPropertyNames(obj);\n\n    function forbiddenKey(v) {\n      // forbidden in strict mode\n      return ~forbiddenInStrictMode.indexOf(v) ||\n        v.match(/^__.*?__$/) ||\n        v.match(/^\\$\\$.*?\\$\\$$/) ||\n        v.match(/[:+~!><=//\\[\\]@\\. ]/);\n    }\n\n    properties = _.filter(properties, function (v) {\n      var good = typeof v === 'string' && !forbiddenKey(v),\n          r;\n      if (linkableOnly) {\n        try {\n          r = good && me.isLinkable(v, obj[v]);\n        } catch (e) {\n          r = false;\n          // uncomment to see why obj[v] is not allowed\n          // console.log(e);\n        } finally {\n          return r;\n        }\n      }\n      return good;\n    }).map(function (v) {\n      var type,\n        linkeable;\n      try {\n        // type = null|string|undefined|number|object\n        type = typeof obj[v];\n        linkeable = isObjectOrFunction(obj[v]);\n      } catch(e) {\n        type = 'undefined';\n        linkeable = false;\n      }\n\n      return {\n        // parent: hashKey(obj),\n        name: v,\n        type: type,\n        linkeable: linkeable\n      };\n    });\n\n    // special properties\n    var proto = Object.getPrototypeOf(obj);\n    if (proto) {\n      properties.push({\n        name: '[[Prototype]]',\n        // cls: hashKey(obj),\n        type: 'object',\n        linkeable: true,\n        hidden: true\n      });\n    }\n    var constructor = obj.hasOwnProperty &&\n      obj.hasOwnProperty('constructor') &&\n      typeof obj.constructor === 'function';\n    if (constructor &&\n        _.findIndex(properties, { name: 'constructor' }) === -1) {\n      properties.push({\n        // cls: hashKey(obj),\n        name: 'constructor',\n        type: 'function',\n        linkeable: true\n      });\n    }\n\n    if (this.cache && !linkableOnly) {\n      this.__cacheObjects[hk] = properties;\n    }\n\n    // console.log(properties);\n    return properties;\n  },\n\n  /**\n   * Analyzes a list of objects recursively\n   * @param  {Array} objects      Array of objects\n   * @param  {number} currentLevel Current dfs level\n   */\n  analyzeObjects: function (objects, currentLevel) {\n    var me = this;\n    objects.forEach(function (v) {\n      if (currentLevel <= me.levels &&    // dfs level\n          !me.objects.get(v) &&         // already registered\n          !me.isForbidden(v)              // forbidden check\n          ) {\n\n        // add to the registered object pool\n        me.objects.put(v);\n\n        // dfs to the next level\n        me.analyzeObjects(\n          me.getOwnLinks(v).map(function (link) {\n            return link.to;\n          }),\n          currentLevel + 1\n        );\n      }\n    });\n  },\n\n  /**\n   * Returns a list of links, each link is an object which has the\n   * following properties:\n   *\n   * - from\n   * - to\n   * - property (string)\n   *\n   * @param  {Object} obj\n   * @return {Array}\n   */\n  getOwnLinks: function (obj) {\n    var me = this,\n        links = [],\n        properties,\n        name = hashKey(obj);\n\n    if (this.__cacheLinks[name]) {\n      // console.log('links from cache :)');\n      return this.__cacheLinks[name];\n    }\n\n    properties = me.getProperties(obj, true);\n\n    function getAugmentedHash(obj, name) {\n      if (!hashKey.has(obj) &&\n          name !== 'prototype' &&\n          name !== 'constructor') {\n        hashKey.createHashKeysFor(obj, name);\n      }\n      return hashKey(obj);\n    }\n\n    if (!name) {\n      throw 'the object needs to have a hashkey';\n    }\n\n    _.forEach(properties, function (v) {\n      var ref = obj[v.name];\n      // because of the levels a reference might not exist\n      if (!ref) {\n        return;\n      }\n\n      // if the object doesn't have a hashkey\n      // let's give it a name equal to the property\n      // being analyzed\n      getAugmentedHash(ref, v.name);\n\n      if (!me.isForbidden(ref)) {\n        links.push({\n          from: obj,\n          fromHash: hashKey(obj),\n          to: ref,\n          toHash: hashKey(ref),\n          property: v.name\n        });\n      }\n    });\n\n    var proto = Object.getPrototypeOf(obj);\n    if (proto && !me.isForbidden(proto)) {\n      links.push({\n        from: obj,\n        fromHash: hashKey(obj),\n        to: proto,\n        toHash: hashKey(proto),\n        property: '[[Prototype]]'\n      });\n    }\n\n    if (this.cache) {\n      this.__cacheLinks[name] = links;\n    }\n\n    return links;\n  },\n\n  makeDirty: function () {\n    this.dirty = true;\n  },\n\n  setLevels: function (l) {\n    this.levels = l;\n  },\n\n  setDirty: function (d) {\n    this.dirty = d;\n  },\n\n  setFunctionConstructors: function (v) {\n    this.functionConstructors = v;\n  },\n\n  getObjects: function () {\n    return this.objects;\n  },\n\n  /**\n   * Stringifies an object properties\n   * @param  obj\n   * @param  toString\n   * @return {Array}\n   */\n  stringifyObjectProperties: function (obj) {\n    return this.getProperties(obj);\n  },\n\n  /**\n   * Returns a representation of the links of\n   * an object\n   * @return {Array}\n   */\n  stringifyObjectLinks: function (obj) {\n    var me = this;\n    return me.getOwnLinks(obj).map(function (link) {\n      // discarded: from, to\n      return {\n        from: link.fromHash,\n        to: link.toHash,\n        property: link.property\n      };\n    });\n  },\n\n  /**\n   * Stringifies the objects saved in this analyzer\n   * @return {Object}\n   */\n  stringify: function () {\n    var me = this,\n      nodes = {},\n      edges = {};\n    console.time('stringify');\n    _(this.objects).forOwn(function (v) {\n      nodes[hashKey(v)] = me.stringifyObjectProperties(v);\n      edges[hashKey(v)] = me.stringifyObjectLinks(v);\n    });\n    console.timeEnd('stringify');\n    return {\n      nodes: nodes,\n      edges: edges\n    };\n  }\n};\n\n// aditional objects that need the prototype to exist\nvar aProto = Analyzer.prototype;\n_.merge(aProto, {\n\n  /**\n   * Adds a list of objects to analyze and make the analyzer dirty\n   * @param  {Array<Objects>} objects\n   * @return {this}\n   */\n  add: wrapFn(function (objects) {\n    console.time('analyze');\n    this.analyzeObjects(objects, 0);\n    console.timeEnd('analyze');\n    return this;\n  }, aProto.makeDirty),\n\n  /**\n   * Removes a list of objects, if `withPrototype` is true then\n   * also the prototype is removed\n   * @param  {Array<Objects>} objects\n   * @param  {boolean} withPrototype\n   * @return {this}\n   */\n  remove: wrapFn(function (objects, withPrototype) {\n    var me = this;\n    objects.forEach(function (obj) {\n      me.objects.remove(obj);\n      if (withPrototype && obj.hasOwnProperty('prototype')) {\n        me.objects.remove(obj.prototype);\n      }\n    });\n    return me;\n  }, aProto.makeDirty),\n\n  /**\n   * Forbids a list of objects, if `withPrototype` is true then\n   * also the prototype is forbidden\n   * @param  {Array<Objects>} objects\n   * @param  {boolean} withPrototype\n   * @return {this}\n   */\n  forbid: wrapFn(function (objects, withPrototype) {\n    var me = this;\n    me.remove(objects, withPrototype);\n    objects.forEach(function (obj) {\n      me.forbidden.put(obj);\n      if (withPrototype && obj.hasOwnProperty('prototype')) {\n        me.forbidden.put(obj.prototype);\n      }\n    });\n  }, aProto.makeDirty),\n\n  /**\n   * Unforbids a list of objects, if `withPrototype` is true then\n   * also the prototype is unforbidden\n   * @param  {Array<Objects>} objects\n   * @param  {boolean} withPrototype\n   * @return {this}\n   */\n  unforbid: wrapFn(function (objects, withPrototype) {\n    var me = this;\n    objects.forEach(function (obj) {\n      me.forbidden.remove(obj);\n      if (withPrototype && obj.hasOwnProperty('prototype')) {\n        me.forbidden.remove(obj.prototype);\n      }\n    });\n  }, aProto.makeDirty)\n});\n\nmodule.exports = Analyzer;\n","'use strict';\n\nvar _ = require('lodash'),\n  Generic = require('./analyzer/GenericAnalyzer'),\n  Angular = require('./analyzer/Angular'),\n  Window = require('./analyzer/Window'),\n  PObject = require('./analyzer/Object'),\n  BuiltIn = require('./analyzer/BuiltIn');\n\nvar libraries;\n\nvar proto = {\n  createNew: function (global, options) {\n    console.log('creating a generic container for: ' + global, options);\n    return (libraries[global] = new Generic(options));\n  },\n  all: function (fn) {\n    _.forOwn(libraries, fn);\n  },\n  markDirty: function () {\n    proto.all(function (v) {\n      v.markDirty();\n    });\n    return proto;\n  },\n  setFunctionConstructors: function (newValue) {\n    proto.all(function (v) {\n      // this only works on the generic analyzers\n      if (!v._hasfc) {\n        v.analyzer.setFunctionConstructors(newValue);\n      }\n    });\n    return proto;\n  }\n};\n\nlibraries = Object.create(proto);\n_.merge(libraries, {\n  object: new PObject(),\n  builtIn: new BuiltIn(),\n  window: new Window(),\n  // popular\n  angular: new Angular(),\n  // mine\n  // t3: new Generic({ global: 't3' }),\n  // huge\n  three: new Generic({\n    global: 'THREE',\n    rendereachtime: true\n  })\n});\n\n// console.log(libraries);\n\n// win max level initially is 0\n// libraries.win.preRender = function () {\n//   libraries.win.getObjects().empty();\n//   libraries.win.analyzeObjects([window], 0);\n// };\n\n// console.log(builtIn.getObjects());\n// console.log(win.getObjects());\n// console.log(user.getObjects());\n\nmodule.exports = libraries;","'use strict';\n\nvar GenericAnalyzer = require('./GenericAnalyzer'),\n  hashKey = require('../util/hashKey');\n\nfunction Angular() {\n  GenericAnalyzer.call(this, {\n    global: 'angular',\n    displayname: 'AngularJS',\n    rendereachtime: true\n  });\n\n  this.services = [\n    '$animate',\n    '$cacheFactory',\n    '$compile',\n    '$controller',\n    // '$document',\n    '$exceptionHandler',\n    '$filter',\n    '$http',\n    '$httpBackend',\n    '$interpolate',\n    '$interval',\n    '$locale',\n    '$log',\n    '$parse',\n    '$q',\n    '$rootScope',\n    '$sce',\n    '$sceDelegate',\n    '$templateCache',\n    '$timeout',\n    // '$window'\n  ].map(function (v) {\n    return { checked: true, name: v };\n  });\n}\n\nAngular.prototype = Object.create(GenericAnalyzer.prototype);\n\nAngular.prototype.getSelectedServices = function () {\n  var me = this,\n    toAnalyze = [];\n\n  window.angular.module('app', ['ng']);\n  this.injector = window.angular.injector(['app']);\n\n  me.services.forEach(function (s) {\n    if (s.checked) {\n      var obj = me.injector.get(s.name);\n      hashKey.createHashKeysFor(obj, s.name);\n      toAnalyze.push(obj);\n    }\n  });\n  return toAnalyze;\n};\n\nAngular.prototype.inspectSelf = function () {\n  console.log('inspecting angular');\n  hashKey.createHashKeysFor(window.angular, 'angular');\n  this.analyzer.getObjects().empty();\n  this.analyzer.add(\n    [window.angular].concat(this.getSelectedServices())\n  );\n};\n\nmodule.exports = Angular;","'use strict';\n\nvar GenericAnalyzer = require('./GenericAnalyzer'),\n  utils = require('../util');\n\nvar toInspect = [\n  Object, Function,\n  Array, Date, Boolean, Number, Math, String, RegExp, JSON,\n  Error\n];\n\nfunction BuiltIn() {\n  GenericAnalyzer.call(this);\n}\n\nBuiltIn.prototype = Object.create(GenericAnalyzer.prototype);\n\nBuiltIn.prototype.inspectSelf = function () {\n  console.log('inspecting builtIn objects');\n  this.analyzer.add(this.getObjects());\n};\n\nBuiltIn.prototype.getObjects = function () {\n  return toInspect;\n};\n\nBuiltIn.prototype.showSearch = function (nodeName, nodeProperty) {\n  var url = 'https://developer.mozilla.org/en-US/search?' +\n    utils.toQueryString({\n      q: encodeURIComponent(nodeName + ' ' + nodeProperty),\n    });\n  window.open(url);\n};\n\nmodule.exports = BuiltIn;","'use strict';\n\nvar Q = require('q'),\n  _ = require('lodash'),\n  utils = require('../util/'),\n  hashKey = require('../util/hashKey'),\n  analyzer = require('../ObjectAnalyzer');\n\nvar searchEngine = 'https://duckduckgo.com/?q=';\n\nfunction GenericAnalyzer(options) {\n  options = options || {};\n  // if (!name) {\n  //   throw 'name needs to be defined';\n  // }\n  this.global = options.global;\n  this.displayname = options.displayname;\n  this.levels = options.hasOwnProperty('levels') ? options.levels : 10;\n  this.forbidden = options.forbidden || [];\n  this.src = options.src;\n  this._hasfc = options.hasOwnProperty('functionconstructors');\n  this.functionconstructors = this._hasfc ?\n    options.functionconstructors : GenericAnalyzer.SHOW_FUNCTION_CONSTRUCTORS;\n  this.rendereachtime = options.hasOwnProperty('rendereachtime') ?\n    options.rendereachtime : false;\n  this.allfunctions = options.hasOwnProperty('allfunctions') ?\n    options.allfunctions : false;\n\n  this.inspected = false;\n\n  // parse forbid string to array\n  this.parse();\n\n  this.analyzer = analyzer({\n    functionConstructors: this.functionconstructors,\n    allFunctions: this.allfunctions\n  });\n}\n\nGenericAnalyzer.SHOW_BUILTIN = false;\nGenericAnalyzer.SHOW_FUNCTION_CONSTRUCTORS = true;\nGenericAnalyzer.FORBIDDEN = 'pojoviz:window,pojoviz:builtIn,document';\n\nGenericAnalyzer.prototype.init = function () {\n  var me = this;\n  console.log('%cPojoViz', 'font-size: 15px; color: ');\n  return me.fetch()\n    .then(function () {\n      if (me.rendereachtime || !me.inspected) {\n        me.inspect();\n      }\n      return me;\n    });\n};\n\nGenericAnalyzer.prototype.parse = function () {\n  if (typeof this.forbidden === 'string') {\n    this.forbidden = this.forbidden.split(',');\n  }\n  if (typeof this.functionconstructors === 'string') {\n    this.functionconstructors = this.functionconstructors === 'true';\n  }\n  if (typeof this.rendereachtime === 'string') {\n    this.rendereachtime = this.rendereachtime === 'true';\n  }\n  if (typeof this.allfunctions === 'string') {\n    this.allfunctions = this.allfunctions === 'true';\n  }\n};\n\nGenericAnalyzer.prototype.markDirty = function () {\n  this.inspected = false;\n};\n\nGenericAnalyzer.prototype.inspectSelf = function () {\n  console.log('analyzing window.' + this.global);\n  var me = this,\n    analyzer = this.analyzer,\n    forbidden = [].concat(this.forbidden);\n  // set a predefied global\n  hashKey.createHashKeysFor(window[this.global], this.global);\n  // clean\n  analyzer.getObjects().empty();\n  analyzer.forbidden.empty();\n  analyzer.setLevels(this.levels);\n\n  // settings > show links to built in objects\n  if (!GenericAnalyzer.SHOW_BUILTIN) {\n    forbidden = forbidden.concat(\n      GenericAnalyzer.FORBIDDEN.split(',')\n    );\n  }\n\n  forbidden.forEach(function(f) {\n    var arr,\n      tokens;\n    if (!f.indexOf('pojoviz:')) {\n      tokens = f.split(':');\n      arr = require('../ObjectHashes')[tokens[1]].getObjects();\n    } else {\n      arr = [window[f]];\n    }\n    console.log('forbidding: ', arr);\n    analyzer.forbid(arr, true);\n  });\n\n  analyzer.add([window[this.global]]);\n\n};\n\nGenericAnalyzer.prototype.markInspected = function () {\n  // mark this container as inspected\n  this.inspected = true;\n  return this;\n};\n\nGenericAnalyzer.prototype.inspect = function () {\n  this\n    .markInspected()\n    .inspectSelf();\n};\n\nGenericAnalyzer.prototype.preRender = function () {\n};\n\nGenericAnalyzer.prototype.fetch = function () {\n  var me = this,\n    script;\n\n  function getValue() {\n    return window[me.global];\n  }\n\n  function promisify(v) {\n    return function () {\n      utils.notification('fetching script ' + v, true);\n      var deferred = Q.defer();\n      script = document.createElement('script');\n      script.src = v;\n      script.onload = function () {\n        utils.notification('completed script ' + v, true);\n        deferred.resolve(getValue());\n      };\n      document.head.appendChild(script);\n      return deferred.promise;\n    };\n  }\n\n  if (this.src) {\n    if (getValue()) {\n      console.log('resource already fetched ' + this.src);\n    } else {\n      var srcs = this.src.split('|');\n      return srcs.reduce(function (prev, current) {\n        return prev.then(promisify(current));\n      }, Q('reduce'));\n    }\n  }\n\n  return Q(true);\n};\n\nGenericAnalyzer.prototype.showSearch = function (nodeName, nodeProperty) {\n  var me = this;\n  window.open(\n    _.template('${searchEngine}${lucky}${libraryName} ${nodeName} ${nodeProperty}', {\n      searchEngine: searchEngine,\n      lucky: GenericAnalyzer.lucky ? '!ducky' : '',\n      libraryName: me.displayname || me.global,\n      nodeName: nodeName,\n      nodeProperty: nodeProperty\n    })\n  );\n};\n\nmodule.exports = GenericAnalyzer;","'use strict';\n\nvar GenericAnalyzer = require('./GenericAnalyzer'),\n  utils = require('../util');\n\nfunction PObject() {\n  GenericAnalyzer.call(this);\n}\n\nPObject.prototype = Object.create(GenericAnalyzer.prototype);\n\nPObject.prototype.inspectSelf = function () {\n  console.log('inspecting Object objects');\n  this.analyzer.add(this.getObjects());\n};\n\nPObject.prototype.getObjects = function () {\n  return [Object];\n};\n\nmodule.exports = PObject;","'use strict';\n\nvar _ = require('lodash'),\n  hashKey = require('../util/hashKey'),\n  GenericAnalyzer = require('./GenericAnalyzer');\n\nvar toInspect = [window];\n\nfunction Window() {\n  GenericAnalyzer.call(this, {\n    levels: 1,\n    rendereachtime: true,\n    functionconstructors: false\n  });\n}\n\nWindow.prototype = Object.create(GenericAnalyzer.prototype);\n\nWindow.prototype.getObjects = function () {\n  return toInspect;\n};\n\nWindow.prototype.inspectSelf = function () {\n  console.log('inspecting window');\n  var me = this,\n    hashes = require('../ObjectHashes');\n\n  _.forOwn(hashes, function (v, k) {\n    if (v.global && window[v.global]) {\n      me.analyzer.forbid([window[v.global]], true);\n    }\n  });\n  this.analyzer.getObjects().empty();\n  this.analyzer.setLevels(this.levels);\n  this.analyzer.add(me.getObjects());\n};\n\nmodule.exports = Window;","var _ = require('lodash'),\n  Q = require('q'),\n  dagre = require('dagre'),\n  utils = require('./util/'),\n  ObjectHashes = require('./ObjectHashes');\n\n// enable long stacks\nQ.longStackSupport = true;\n\nvar container,\n  oldContainer,\n  oldRenderer,\n  renderer,\n  pojoviz;      // namespace\n\nfunction process() {\n  var g = new dagre.Digraph(),\n      properties,\n      node,\n      library = container.analyzer,\n      str = library.stringify(),\n      libraryNodes = str.nodes,\n      libraryEdges = str.edges;\n\n  // create the graph\n  // each element of the graph has\n  // - label\n  // - width\n  // - height\n  // - properties\n  _.forOwn(libraryNodes, function (properties, k) {\n    var label = k.match(/\\S*?-(.*)/)[1];\n    // console.log(k, label.length);\n    node = {\n      label: k,\n      width: label.length * 10\n    };\n    // lines + header + padding bottom\n    node.height = properties.length * 15 + 50;\n    node.properties = properties;\n    properties.forEach(function (v) {\n      node.width = Math.max(node.width, v.name.length * 10);\n    });\n    g.addNode(k, node);\n  });\n\n  // build the edges from node to node\n  _.forOwn(libraryEdges, function (links) {\n    links.forEach(function (link) {\n      if (g.hasNode(link.from) && g.hasNode(link.to)) {\n        g.addEdge(null, link.from, link.to);\n      }\n    });\n  });\n\n  // layout of the graph\n  var layout = dagre.layout()\n    .nodeSep(30)\n    // .rankSep(70)\n    // .rankDir('TB')\n    .run(g);\n\n  var nodes = [],\n      edges = [],\n      center = {x: 0, y: 0},\n      mn = {x: Infinity, y: Infinity},\n      mx = {x: -Infinity, y: -Infinity},\n      total = g.nodes().length;\n\n  // update the node info of the node adding:\n  // - x\n  // - y\n  // - predecessors\n  // - successors\n  layout.eachNode(function (k, layoutInfo) {\n    var x = layoutInfo.x;\n    var y = layoutInfo.y;\n\n    node = g.node(k);\n    node.x = x;\n    node.y = y;\n    node.predecessors = g.predecessors(k);\n    node.successors = g.successors(k);\n    nodes.push(node);\n\n    // calculate the bbox of the graph to center the graph\n    var mnx = x - node.width / 2;\n    var mny = y - node.height / 2;\n    var mxx = x + node.width / 2;\n    var mxy = y + node.height / 2;\n\n    center.x += x;\n    center.y += y;\n    mn.x = Math.min(mn.x, mnx);\n    mn.y = Math.min(mn.y, mny);\n    // console.log(x, y, ' dim ', node.width, node.height);\n    mx.x = Math.max(mx.x, mxx);\n    mx.y = Math.max(mx.y, mxy);\n  });\n\n  center.x /= (total || 1);\n  center.y /= (total || 1);\n\n  // create the edges from property to node\n  _(libraryEdges).forOwn(function (links) {\n    links.forEach(function (link) {\n      if (g.hasNode(link.from) && g.hasNode(link.to)) {\n        edges.push(link);\n      }\n    });\n  });\n\n  return {\n    edges: edges,\n    nodes: nodes,\n    center: center,\n    mn: mn,\n    mx: mx\n  };\n}\n\n// render\nfunction render() {\n  var data;\n\n  if (container === oldContainer) {\n    return;\n  }\n\n  utils.notification('processing ' + container.global);\n\n  // pre render\n  oldRenderer && oldRenderer.clean();\n  renderer.clean();\n\n  setTimeout(function () {\n    container.preRender();\n    console.log('process & render start: ', new Date());\n    // data has\n    // - edges (property -> node)\n    // - nodes\n    // - center\n    //\n    console.time('process');\n    data = process();\n    console.timeEnd('process');\n\n    utils.notification('rendering ' + container.global);\n\n    console.time('render');\n    renderer.render(data);\n    console.timeEnd('render');\n\n    utils.notification('complete!');\n  }, 0);\n}\n\n// public api\npojoviz = {\n  renderers: {},\n  addRenderers: function (newRenderers) {\n    _.merge(pojoviz.renderers, newRenderers);\n  },\n  nullifyContainer: function () {\n    oldContainer = container;\n    container = null;\n  },\n  getContainer: function () {\n    return container;\n  },\n  setContainer: function (containerName, options) {\n    oldContainer = container;\n    container = ObjectHashes[containerName];\n\n    if (!container) {\n      container = ObjectHashes.createNew(containerName, options);\n    } else {\n      // required to fetch external resources\n      container.src = options.src;\n    }\n\n    return container.init();\n  },\n  setRenderer: function (r) {\n    oldRenderer = renderer;\n    renderer = pojoviz.renderers[r];\n  },\n  getRenderer: function () {\n    return renderer;\n  },\n  render: render,\n\n  // expose inner modules\n  ObjectHashes: require('./ObjectHashes'),\n  ObjectAnalyzer: require('./ObjectAnalyzer'),\n  analyzer: {\n    GenericAnalyzer: require('./analyzer/GenericAnalyzer')\n  },\n  utils: require('./util'),\n\n  // user vars\n  userVariables: []\n};\n\n// custom events\ndocument.addEventListener('property-click', function (e) {\n  var detail = e.detail;\n  pojoviz\n    .getContainer()\n    .showSearch(detail.name, detail.property);\n});\n\nmodule.exports = pojoviz;","'use strict';\n\nvar hashKey = require('./hashKey');\n\nfunction HashMap() {\n}\n\nHashMap.prototype = {\n  put: function (key, value) {\n    this[hashKey(key)] = (value || key);\n  },\n  get: function (key) {\n    return this[hashKey(key)];\n  },\n  remove: function (key) {\n    var v = this[hashKey(key)];\n    delete this[hashKey(key)];\n    return v;\n  },\n  empty: function () {\n    var p,\n        me = this;\n    for (p in me) {\n      if (me.hasOwnProperty(p)) {\n        delete this[p];\n      }\n    }\n  }\n};\n\nmodule.exports = HashMap;","'use strict';\n\nvar _ = require('lodash'),\n  assert = require('./').assert,\n  me, hashKey;\n\nfunction isObjectOrFunction(v) {\n  return v && (typeof v === 'object' || typeof v === 'function');\n}\n\n/**\n * Gets a store hashkey only if it's an object\n * @param  {[type]} obj\n * @return {[type]}\n */\nfunction get(obj) {\n  assert(isObjectOrFunction(obj), 'obj must be an object|function');\n  return obj.hasOwnProperty &&\n    obj.hasOwnProperty(me.hiddenKey) &&\n    obj[me.hiddenKey];\n}\n\n/**\n * Sets a key on an object\n * @param {[type]} obj [description]\n * @param {[type]} key [description]\n */\nfunction set(obj, key) {\n  assert(isObjectOrFunction(obj), 'obj must be an object|function');\n  assert(\n    key && typeof key === 'string',\n    'The key needs to be a valid string'\n  );\n  if (!get(obj)) {\n    Object.defineProperty(obj, me.hiddenKey, {\n      value: typeof obj + '-' + key\n    });\n  }\n  return me;\n}\n\nme = hashKey = function (v) {\n  var value = v,\n      uid = v;\n\n  if (isObjectOrFunction(v)) {\n    if (!get(v)) {\n      me.createHashKeysFor(v);\n    }\n    uid = get(v);\n    if (!uid) {\n      console.err('no hashkey :(', v);\n    }\n    assert(uid, 'error getting the key');\n    return uid;\n  }\n\n  // v is a primitive\n  return typeof v + '-' + uid;\n};\nme.hiddenKey = '__pojoVizKey__';\n\nme.createHashKeysFor = function (obj, name) {\n\n  function localToString(obj) {\n    var match;\n    try {\n      match = {}.toString.call(obj).match(/^\\[object (\\S*?)\\]/);\n    } catch (e) {\n      match = false;\n    }\n    return match && match[1];\n  }\n\n  /**\n   * Analyze the internal property [[Class]] to guess the name\n   * of this object, e.g. [object Date], [object Math]\n   * Many object will give false positives (they will match [object Object])\n   * so let's consider Object as the name only if it's equal to\n   * Object.prototype\n   * @param  {Object}  obj\n   * @return {Boolean}\n   */\n  function hasAClassName(obj) {\n    var match = localToString(obj);\n    if (match === 'Object') {\n      return obj === Object.prototype && 'Object';\n    }\n    return match;\n  }\n\n  function getName(obj) {\n    var name, className;\n\n    // return the already generated hashKey\n    if (get(obj)) {\n      return get(obj);\n    }\n\n    // generate a new key based on\n    // - the name if it's a function\n    // - a unique id\n    name = typeof obj === 'function' &&\n      typeof obj.name === 'string' &&\n      obj.name;\n\n    className = hasAClassName(obj);\n    if (!name && className) {\n      name = className;\n    }\n\n    name = name || _.uniqueId();\n    return name;\n  }\n\n  // the name is equal to the passed name or the\n  // generated name\n  name = name || getName(obj);\n  name = name.replace(/[\\. ]/img, '-');\n\n  // if the obj is a prototype then try to analyze\n  // the constructor first so that the prototype becomes\n  // [name].prototype\n  // special case: object.constructor = object\n  if (obj.hasOwnProperty &&\n      obj.hasOwnProperty('constructor') &&\n      typeof obj.constructor === 'function' &&\n      obj.constructor !== obj) {\n    me.createHashKeysFor(obj.constructor);\n  }\n\n  // set name on self\n  set(obj, name);\n\n  // set name on the prototype\n  if (typeof obj === 'function' &&\n      obj.hasOwnProperty('prototype')) {\n    set(obj.prototype, name + '-prototype');\n  }\n};\n\nme.has = function (v) {\n  return v.hasOwnProperty &&\n    v.hasOwnProperty(me.hiddenKey);\n};\n\nmodule.exports = me;","'use strict';\n\nvar _ = require('lodash');\n\nvar propertiesTransformation = {\n  '[[Prototype]]': '__proto__'\n};\n\nvar utils = {\n  assert: function (v, message) {\n    if (!v) {\n      throw message || 'error';\n    }\n  },\n  translate: function (x, y) {\n    return 'translate(' + (x || 0) + ', ' + (y || 0) + ')';\n  },\n  scale: function (s) {\n    return 'scale(' + (s || 1) + ')';\n  },\n  transform: function (obj) {\n    var t = [];\n    _.forOwn(obj, function (v, k) {\n      t.push(utils[k].apply(utils, v));\n    });\n    return t.join(' ');\n  },\n  prefixer: function () {\n    var args = [].slice.call(arguments);\n    args.unshift('pv');\n    return args.join('-');\n  },\n  transformProperty: function (v) {\n    if (propertiesTransformation.hasOwnProperty(v)) {\n      return propertiesTransformation[v];\n    }\n    return v;\n  },\n  escapeCls: function(v) {\n    return v.replace(/\\$/g, '_');\n  },\n  toQueryString: function (obj) {\n    var s = '',\n        i = 0;\n    _.forOwn(obj, function (v, k) {\n      if (i) {\n        s += '&';\n      }\n      s += k + '=' + v;\n      i += 1;\n    });\n    return s;\n  },\n  createEvent: function (eventName, details) {\n    return new CustomEvent(eventName, {\n      detail: details\n    });\n  },\n  notification: function (message, consoleToo) {\n    var ev = utils.createEvent('pojoviz-notification', message);\n    consoleToo && console.log(message);\n    document.dispatchEvent(ev);\n  },\n  createJsonpCallback: function (url) {\n    var script = document.createElement('script');\n    script.src = url;\n    document.head.appendChild(script);\n  }\n};\n\nmodule.exports = utils;"]}
  1223. (11)
  1224. });