Chromium Code Reviews| Index: tools/binary_size/template/D3SymbolTreeMap.js |
| diff --git a/tools/binary_size/template/D3SymbolTreeMap.js b/tools/binary_size/template/D3SymbolTreeMap.js |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..83487e8269d30ff67cd0456c70bf45c4313ea707 |
| --- /dev/null |
| +++ b/tools/binary_size/template/D3SymbolTreeMap.js |
| @@ -0,0 +1,1037 @@ |
| +// Copyright 2014 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +// TODO: |
| +// 1. Visibility functions: base on boxPadding.t, not 15 |
| +// 2. Track a maxDisplayDepth that is user-settable: |
| +// maxDepth == currentRoot.depth + maxDisplayDepth |
| +function D3SymbolTreeMap(mapWidth, mapHeight, levelsToShow) { |
| + this._mapContainer = undefined; |
|
bulach
2014/04/16 16:32:16
nit: I think indent is 2 rather than 4
|
| + this._mapWidth = mapWidth; |
| + this._mapHeight = mapHeight; |
| + this.boxPadding = {'l': 5, 'r': 5, 't': 20, 'b': 5}; |
| + this.infobox = undefined; |
| + this._maskContainer = undefined; |
| + this._highlightContainer = undefined; |
| + // Transition in this order: |
| + // 1. Exiting items go away. |
| + // 2. Updated items move. |
| + // 3. New items enter. |
| + this._exitDuration=500; |
| + this._updateDuration=500; |
| + this._enterDuration=500; |
| + this._firstTransition=true; |
| + this._layout = undefined; |
| + this._currentRoot = undefined; |
| + this._currentNodes = undefined; |
| + this._treeData = undefined; |
| + this._maxLevelsToShow = levelsToShow; |
| + this._currentMaxDepth = this._maxLevelsToShow; |
| +} |
| + |
| +/** |
| + * Make a number pretty, with comma separators. |
| + */ |
| +D3SymbolTreeMap._pretty = function(num) { |
| + var asString = String(num); |
| + var result = ''; |
| + var counter = 0; |
| + for (var x=asString.length - 1; x >= 0; x--) { |
| + counter++; |
| + if (counter === 4) { |
| + result = ',' + result; |
| + counter = 1; |
| + } |
| + result = asString.charAt(x) + result; |
| + } |
| + return result; |
| +} |
| + |
| +/** |
| + * Express a number in terms of KiB, MiB, GiB, etc. |
| + * Note that these are powers of 2, not of 10. |
| + */ |
| +D3SymbolTreeMap._byteify = function(num) { |
| + var suffix; |
| + if (num >= 1024) { |
| + if (num >= 1024*1024*1024) { |
| + suffix = 'GiB'; |
| + num = num / (1024*1024*1024); |
| + } else if (num >= 1024*1024) { |
| + suffix = 'MiB'; |
| + num = num / (1024*1024); |
| + } else if (num >= 1024) { |
| + suffix = 'KiB' |
| + num = num / 1024; |
| + } |
| + return num.toFixed(2) + ' ' + suffix; |
| + } |
| + return num + ' B'; |
| +} |
| + |
| +D3SymbolTreeMap._NM_SYMBOL_TYPES = 'ABbCDdGgiNpRrSsTtUuVvWw@-?'; |
| +D3SymbolTreeMap._NM_SYMBOL_TYPE_DESCRIPTIONS = { |
| + // Definitions concisely derived from the nm 'man' page |
| + 'A': 'Global absolute (A)', |
| + 'B': 'Global uninitialized data (B)', |
| + 'b': 'Local uninitialized data (b)', |
| + 'C': 'Global uninitialized common (C)', |
| + 'D': 'Global initialized data (D)', |
| + 'd': 'Local initialized data (d)', |
| + 'G': 'Global small initialized data (G)', |
| + 'g': 'Local small initialized data (g)', |
| + 'i': 'Indirect function (i)', |
| + 'N': 'Debugging (N)', |
| + 'p': 'Stack unwind (p)', |
| + 'R': 'Global read-only data (R)', |
| + 'r': 'Local read-only data (r)', |
| + 'S': 'Global small uninitialized data (S)', |
| + 's': 'Local small uninitialized data (s)', |
| + 'T': 'Global code (T)', |
| + 't': 'Local code (t)', |
| + 'U': 'Undefined (U)', |
| + 'u': 'Unique (u)', |
| + 'V': 'Global weak object (V)', |
| + 'v': 'Local weak object (v)', |
| + 'W': 'Global weak symbol (W)', |
| + 'w': 'Local weak symbol (w)', |
| + '@': 'Vtable entry (@)', // non-standard, hack. |
| + '-': 'STABS debugging (-)', |
| + '?': 'Unrecognized (?)', |
| +}; |
| +/** |
| + * Given a symbol type code, look up and return a human-readable description |
| + * of that symbol type. If the symbol type does not match one of the known |
| + * types, the unrecognized description (corresponding to symbol type '?') is |
| + * returned instead of null or undefined. |
| + */ |
| +D3SymbolTreeMap._getSymbolDescription = function(type) { |
| + var result = D3SymbolTreeMap._NM_SYMBOL_TYPE_DESCRIPTIONS[type]; |
| + if (result === undefined) { |
| + result = D3SymbolTreeMap._NM_SYMBOL_TYPE_DESCRIPTIONS['?']; |
| + } |
| + return result; |
| +}; |
| + |
| +// Qualitative 12-value pastel Brewer palette. |
| +D3SymbolTreeMap._colorArray = [ |
| + 'rgb(141,211,199)', |
| + 'rgb(255,255,179)', |
| + 'rgb(190,186,218)', |
| + 'rgb(251,128,114)', |
| + 'rgb(128,177,211)', |
| + 'rgb(253,180,98)', |
| + 'rgb(179,222,105)', |
| + 'rgb(252,205,229)', |
| + 'rgb(217,217,217)', |
| + 'rgb(188,128,189)', |
| + 'rgb(204,235,197)', |
| + 'rgb(255,237,111)']; |
| + |
| +D3SymbolTreeMap._initColorMap = function() { |
| + var map = {}; |
| + var numColors = D3SymbolTreeMap._colorArray.length; |
| + for (var x=0; x<D3SymbolTreeMap._NM_SYMBOL_TYPES.length; x++) { |
| + var key = D3SymbolTreeMap._NM_SYMBOL_TYPES.charAt(x); |
| + var index = x % numColors; |
| + map[key] = d3.rgb(D3SymbolTreeMap._colorArray[index]); |
| + } |
| + D3SymbolTreeMap._colorMap = map; |
| +} |
| +D3SymbolTreeMap._initColorMap(); |
| + |
| +D3SymbolTreeMap.getColorForType = function(type) { |
| + var result = D3SymbolTreeMap._colorMap[type]; |
| + if (result === undefined) return d3.rgb('rgb(255,255,255)'); |
| + return result; |
| +} |
| + |
| +D3SymbolTreeMap.prototype.init = function() { |
| + this.infobox = this._createInfoBox(); |
| + this._mapContainer = d3.select('body').append('div') |
| + .style('position', 'relative') |
| + .style('width', this._mapWidth) |
| + .style('height', this._mapHeight) |
| + .style('padding', 0) |
| + .style('margin', 0) |
| + .style('box-shadow', '5px 5px 5px #888'); |
| + this._createHoverMask(); |
| + this._layout = this._createTreeMapLayout(); |
| + this._setData(tree_data); // TODO: Don't use global 'tree_data' |
| +} |
| + |
| +/** |
| + * Sets the data displayed by the treemap and layint out the map. |
| + */ |
| +D3SymbolTreeMap.prototype._setData = function(data) { |
| + this._treeData = data; |
| + console.time('_crunchStats'); |
| + this._crunchStats(data); |
| + console.timeEnd('_crunchStats'); |
| + this._currentRoot = this._treeData; |
| + this._currentNodes = this._layout.nodes(this._currentRoot); |
| + this._currentMaxDepth = this._maxLevelsToShow; |
| + this._doLayout(); |
| +} |
| + |
| +/** |
| + * Recursively traverses the entire tree starting from the specified node, |
| + * computing statistics and recording metadata as it goes. Call this method |
| + * only once per imported tree. |
| + */ |
| +D3SymbolTreeMap.prototype._crunchStats = function(node) { |
| + var stack = []; |
| + stack.idCounter = 0; |
| + this._crunchStatsHelper(stack, node); |
| +} |
| + |
| +/** |
| + * Invoke the specified visitor function on all data elements currently shown |
| + * in the treemap including any and all of their children, starting at the |
| + * currently-displayed root and descening recursively. The function will be |
| + * passed the datum element representing each node. No traversal guarantees |
| + * are made. |
| + */ |
| +D3SymbolTreeMap.prototype.visitFromDisplayedRoot = function(visitor) { |
| + this._visit(this._currentRoot, visitor); |
| +} |
| + |
| +/** |
| + * Helper function for visit functions. |
| + */ |
| +D3SymbolTreeMap.prototype._visit = function(datum, visitor) { |
| + visitor.call(this, datum); |
| + if (datum.children) for (var i=0; i<datum.children.length; i++) |
|
bulach
2014/04/16 16:32:16
nit: space around = and <
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Done.
|
| + this._visit(datum.children[i], visitor); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._crunchStatsHelper = function(stack, node) { |
| + // Only overwrite the node ID if it isn't already set. |
| + // This allows stats to be crunched multiple times on subsets of data |
| + // without breaking the data-to-ID bindings. New nodes get new IDs. |
| + if (node.id === undefined) node.id = stack.idCounter++; |
| + if (node.children === undefined) { |
| + // Leaf node (symbol); accumulate stats. |
| + for (var i=0; i<stack.length; i++) { |
|
bulach
2014/04/16 16:32:16
nit: space = and <
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Done.
|
| + var ancestor = stack[i]; |
| + if (!ancestor.symbol_stats) ancestor.symbol_stats = {}; |
| + if (ancestor.symbol_stats[node.t] === undefined) { |
| + // New symbol type we haven't seen before, just record. |
| + ancestor.symbol_stats[node.t] = {'count': 1, |
| + 'size': node.value}; |
| + } else { |
| + // Existing symbol type, increment. |
| + ancestor.symbol_stats[node.t].count++; |
| + ancestor.symbol_stats[node.t].size += node.value; |
| + } |
| + } |
| + } else for (var i=0; i<node.children.length; i++) { |
|
bulach
2014/04/16 16:32:16
nit: space = and <
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Done.
|
| + stack.push(node); |
| + this._crunchStatsHelper(stack, node.children[i]); |
| + stack.pop(); |
| + } |
| +} |
| + |
| +D3SymbolTreeMap.prototype._createTreeMapLayout = function() { |
| + var result = d3.layout.treemap() |
| + .padding([this.boxPadding.t, this.boxPadding.r, |
| + this.boxPadding.b, this.boxPadding.l]) |
| + .size([this._mapWidth, this._mapHeight]); |
| + return result; |
| +} |
| + |
| +D3SymbolTreeMap.prototype.resize = function(width, height) { |
| + this._mapWidth = width; |
| + this._mapHeight = height; |
| + this._mapContainer.style('width', width).style('height', height); |
| + this._layout.size([this._mapWidth, this._mapHeight]); |
| + this._currentNodes = this._layout.nodes(this._currentRoot); |
| + this._doLayout(); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._zoomDatum = function(datum) { |
| + if (this._currentRoot === datum) { |
| + return; // already here |
| + } |
| + this._hideHighlight(datum); |
| + this._hideInfoBox(datum); |
| + this._currentRoot = datum; |
| + this._currentNodes = this._layout.nodes(this._currentRoot); |
| + this._currentMaxDepth = this._currentRoot.depth + this._maxLevelsToShow; |
| + console.log('zooming into datum ' + this._currentRoot.n); |
| + this._doLayout(); |
| +} |
| + |
| +D3SymbolTreeMap.prototype.setMaxLevels = function(levelsToShow) { |
| + this._maxLevelsToShow = levelsToShow; |
| + this._currentNodes = this._layout.nodes(this._currentRoot); |
| + this._currentMaxDepth = this._currentRoot.depth + this._maxLevelsToShow; |
| + console.log('setting max levels to show: ' + this._maxLevelsToShow); |
| + this._doLayout(); |
| +} |
| + |
| +/** |
| + * Clone the specified tree, returning an independent copy of the data. |
| + * Only the original attributes expected to exist prior to invoking |
| + * _crunchStatsHelper are retained, with the exception of the 'id' attribute |
| + * (which must be retained for proper transitions). |
| + * If the optional filter parameter is provided, it will be called with 'this' |
| + * set to this treemap instance and passed the 'datum' object as an argument. |
| + * When specified, the copy will retain only the data for which the filter |
| + * function returns true. |
| + */ |
| +D3SymbolTreeMap.prototype._clone = function(datum, filter) { |
| + var trackingStats = false; |
| + if (this.__cloneState === undefined) { |
| + console.time('_clone'); |
| + trackingStats = true; |
| + this.__cloneState = {'accepted': 0, 'rejected': 0, |
| + 'forced': 0, 'pruned': 0}; |
| + } |
| + |
| + // Must go depth-first. All parents of children that are accepted by the |
| + // filter must be preserved! |
| + var copy = {'n': datum.n, 'k': datum.k}; |
| + var childAccepted = false; |
| + if (datum.children !== undefined) { |
| + for (var i=0; i<datum.children.length; i++) { |
| + var copiedChild = this._clone(datum.children[i], filter); |
| + if (copiedChild !== undefined) { |
| + childAccepted = true; // parent must also be accepted. |
| + if (copy.children === undefined) copy.children = []; |
| + copy.children.push(copiedChild); |
| + } |
| + } |
| + } |
| + |
| + // Ignore nodes that don't match the filter, when present. |
| + var accept = false; |
| + if (childAccepted) { |
| + // Parent of an accepted child must also be accepted. |
| + this.__cloneState.forced++; |
| + accept = true; |
| + } else if (filter !== undefined && filter.call(this, datum) !== true) { |
| + this.__cloneState.rejected++; |
| + } else if (datum.children === undefined) { |
| + // Accept leaf nodes that passed the filter |
| + this.__cloneState.accepted++; |
| + accept = true; |
| + } else { |
| + // Non-leaf node. If no children are accepted, prune it. |
| + this.__cloneState.pruned++; |
| + } |
| + |
| + if (accept) { |
| + if (datum.id !== undefined) copy.id = datum.id; |
| + if (datum.lastPathElement !== undefined) { |
| + copy.lastPathElement = datum.lastPathElement; |
| + } |
| + if (datum.t !== undefined) copy.t = datum.t; |
| + if (datum.value !== undefined && datum.children === undefined) { |
| + copy.value = datum.value; |
| + } |
| + } else { |
| + // Discard the copy we were going to return |
| + copy = undefined; |
| + } |
| + |
| + if (trackingStats === true) { |
| + // We are the fist call in the recursive chain. |
| + console.timeEnd('_clone'); |
| + var totalAccepted = this.__cloneState.accepted + |
| + this.__cloneState.forced; |
| + console.log( |
| + totalAccepted + ' nodes retained (' + |
| + this.__cloneState.forced + ' forced by accepted children, ' + |
| + this.__cloneState.accepted + ' accepted on their own merits), ' + |
| + this.__cloneState.rejected + ' nodes (and their children) ' + |
| + 'filtered out,' + |
| + this.__cloneState.pruned + ' nodes pruned because because no ' + |
| + 'children remained.'); |
| + delete this.__cloneState; |
| + } |
| + return copy; |
| +} |
| + |
| +D3SymbolTreeMap.prototype.filter = function(filter) { |
| + // Ensure we have a copy of the original root. |
| + if (this._backupTree === undefined) this._backupTree = this._treeData; |
| + this._mapContainer.selectAll('div').remove(); |
| + this._setData(this._clone(this._backupTree, filter)); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._doLayout = function() { |
| + console.time('_doLayout'); |
| + this._handleInodes(); |
| + this._handleLeaves(); |
| + this._firstTransition = false; |
| + console.timeEnd('_doLayout'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._highlightElement = function(datum, selection) { |
| + this._showHighlight(datum, selection); |
| + //this._showHoverMask(selection); |
|
bulach
2014/04/16 16:32:16
nit: remove? (and 379...)
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Removed the entire hovermask feature, which never
|
| +} |
| + |
| +D3SymbolTreeMap.prototype._unhighlightElement = function(datum, selection) { |
| + this._hideHighlight(datum, selection); |
| + //this._hideHoverMask(selection); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._handleInodes = function() { |
| + console.time('_handleInodes'); |
| + var thisTreeMap = this; |
| + var inodes = this._currentNodes.filter(function(datum){ |
| + return (datum.depth <= thisTreeMap._currentMaxDepth) && |
| + datum.children !== undefined; |
| + }); |
| + var cellsEnter = this._mapContainer.selectAll('div.inode') |
| + .data(inodes, function(datum) { return datum.id; }) |
| + .enter() |
| + .append('div').attr('class', 'inode').attr('id', function(datum){ |
| + return 'node-' + datum.id;}); |
| + |
| + |
| + // Define enter/update/exit for inodes |
| + cellsEnter |
| + .append('div') |
| + .attr('class', 'rect inode_rect_entering') |
| + .style('z-index', function(datum) { return datum.id*2; }) |
| + .style('position', 'absolute') |
| + .style('left', function(datum) { return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum){ return datum.dx; }) |
| + .style('height', function(datum){ return datum.dy; }) |
| + .style('opacity', '0') |
| + .style('border', '1px solid black') |
| + .style('background-image', function(datum) { |
| + return thisTreeMap._makeSymbolBucketBackgroundImage.call( |
| + thisTreeMap, datum); |
| + }) |
| + .style('background-color', function(datum) { |
| + if (datum.t === undefined) return 'rgb(220,220,220)'; |
| + return D3SymbolTreeMap.getColorForType(datum.t).toString(); |
| + }) |
| + .on('mouseover', function(datum){ |
| + thisTreeMap._highlightElement.call( |
| + thisTreeMap, datum, d3.select(this)); |
| + thisTreeMap._showInfoBox.call(thisTreeMap, datum); |
| + }) |
| + .on('mouseout', function(datum){ |
| + thisTreeMap._unhighlightElement.call( |
| + thisTreeMap, datum, d3.select(this)); |
| + thisTreeMap._hideInfoBox.call(thisTreeMap, datum); |
| + }) |
| + .on('mousemove', function(){ |
| + thisTreeMap._moveInfoBox.call(thisTreeMap, event); |
| + }) |
| + .on('dblclick', function(datum){ |
| + if (datum !== thisTreeMap._currentRoot) { |
| + // Zoom into the selection |
| + thisTreeMap._zoomDatum(datum); |
| + } else if (datum.parent) { |
| + console.log('event.shiftKey=' + event.shiftKey); |
| + if (event.shiftKey === true) { |
| + // Back to root |
| + thisTreeMap._zoomDatum(thisTreeMap._treeData); |
| + } else { |
| + // Zoom out of the selection |
| + thisTreeMap._zoomDatum(datum.parent); |
| + } |
| + } |
| + }); |
| + cellsEnter |
| + .append('div') |
| + .attr('class', 'label inode_label_entering') |
| + .style('z-index', function(datum) { return (datum.id*2)+1; }) |
|
bulach
2014/04/16 16:32:16
nit: space
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Done.
|
| + .style('position', 'absolute') |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum) { return datum.dx; }) |
| + .style('height', function(datum) { return thisTreeMap.boxPadding.t; }) |
| + .style('opacity', '0') |
| + .style('pointer-events', 'none') |
| + .style('-webkit-user-select', 'none') |
| + .style('overflow', 'hidden') // required for ellipsis |
| + .style('white-space', 'nowrap') // required for ellipsis |
| + .style('text-overflow', 'ellipsis') |
| + .style('text-align', 'center') |
| + .style('vertical-align', 'top') |
| + .style('visibility', function(datum) { |
| + return (datum.dx < 15 || datum.dy < 15) ? 'hidden' : 'visible'; |
| + }) |
| + .text(function(datum) { |
| + var sizeish = ' [' + D3SymbolTreeMap._byteify(datum.value) + ']' |
| + var text; |
| + if (datum.k === 'b') { // bucket |
| + if (datum === thisTreeMap._currentRoot) { |
| + text = thisTreeMap.pathFor(datum) + ': ' |
| + + D3SymbolTreeMap._getSymbolDescription(datum.t) |
| + } else { |
| + text = D3SymbolTreeMap._getSymbolDescription(datum.t); |
| + } |
| + } else if (datum === thisTreeMap._currentRoot) { |
| + // The top-most level should always show the complete path |
| + text = thisTreeMap.pathFor(datum); |
| + } else { |
| + // Anything that isn't a bucket or a leaf (symbol) or the |
| + // current root should just show its name. |
| + text = datum.n; |
| + } |
| + return text + sizeish; |
| + } |
| + ); |
| + |
| + // Complicated transition logic: |
| + // For nodes that are entering, we want to fade them in in-place AFTER |
| + // any adjusting nodes have resized and moved around. That way, new nodes |
| + // seamlessly appear in the right spot after their containers have resized |
| + // and moved around. |
| + // To do this we do some trickery: |
| + // 1. Define a '_entering' class on the entering elements |
| + // 2. Use this to select only the entering elements and apply the opacity |
| + // transition. |
| + // 3. Use the same transition to drop the '_entering' suffix, so that they |
| + // will correctly update in later zoom/resize/whatever operations. |
| + // 4. The update transition is achieved by selecting the elements without |
| + // the '_entering_' suffix and applying movement and resizing transition |
| + // effects. |
| + this._mapContainer.selectAll('div.inode_rect_entering').transition() |
| + .duration(thisTreeMap._enterDuration).delay( |
| + this._firstTransition ? 0 : thisTreeMap._exitDuration + |
| + thisTreeMap._updateDuration) |
| + .attr('class', 'rect inode_rect') |
| + .style('opacity', '1') |
| + this._mapContainer.selectAll('div.inode_label_entering').transition() |
| + .duration(thisTreeMap._enterDuration).delay( |
| + this._firstTransition ? 0 : thisTreeMap._exitDuration + |
| + thisTreeMap._updateDuration) |
| + .attr('class', 'label inode_label') |
| + .style('opacity', '1') |
| + this._mapContainer.selectAll('div.inode_rect').transition() |
| + .duration(thisTreeMap._updateDuration).delay(thisTreeMap._exitDuration) |
| + .style('opacity', '1') |
| + .style('background-image', function(datum) { |
| + return thisTreeMap._makeSymbolBucketBackgroundImage.call( |
| + thisTreeMap, datum); |
| + }) |
| + .style('left', function(datum) { return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum){ return datum.dx; }) |
| + .style('height', function(datum){ return datum.dy; }); |
| + this._mapContainer.selectAll('div.inode_label').transition() |
| + .duration(thisTreeMap._updateDuration).delay(thisTreeMap._exitDuration) |
| + .style('opacity', '1') |
| + .style('visibility', function(datum) { |
| + return (datum.dx < 15 || datum.dy < 15) ? 'hidden' : 'visible'; |
| + }) |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum) { return datum.dx; }) |
| + .style('height', function(datum) { return thisTreeMap.boxPadding.t; }) |
| + .text(function(datum) { |
| + var sizeish = ' [' + D3SymbolTreeMap._byteify(datum.value) + ']' |
| + var text; |
| + if (datum.k === 'b') { |
| + if (datum === thisTreeMap._currentRoot) { |
| + text = thisTreeMap.pathFor(datum) + ': ' + |
| + D3SymbolTreeMap._getSymbolDescription(datum.t) |
| + } else { |
| + text = D3SymbolTreeMap._getSymbolDescription(datum.t); |
| + } |
| + } else if (datum === thisTreeMap._currentRoot) { |
| + // The top-most level should always show the complete path |
| + text = thisTreeMap.pathFor(datum); |
| + } else { |
| + // Anything that isn't a bucket or a leaf (symbol) or the |
| + // current root should just show its name. |
| + text = datum.n; |
| + } |
| + return text + sizeish; |
| + }); |
| + var exit = this._mapContainer.selectAll('div.inode') |
| + .data(inodes, function(datum) { return 'inode-' + datum.id; }) |
| + .exit(); |
| + exit.selectAll('div.inode_rect').transition().duration( |
| + thisTreeMap._exitDuration).style('opacity', 0); |
| + exit.selectAll('div.inode_label').transition().duration( |
| + thisTreeMap._exitDuration).style('opacity', 0); |
| + exit.transition().delay(thisTreeMap._exitDuration + 1).remove(); |
| + |
| + console.log(inodes.length + ' inodes layed out.'); |
| + console.timeEnd('_handleInodes'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._handleLeaves = function() { |
| + console.time('_handleLeaves'); |
| + var color_fn = d3.scale.category10(); |
| + var thisTreeMap = this; |
| + var leaves = this._currentNodes.filter(function(datum){ |
| + return (datum.depth <= thisTreeMap._currentMaxDepth) && |
| + datum.children === undefined; }); |
| + var cellsEnter = this._mapContainer.selectAll('div.leaf') |
| + .data(leaves, function(datum) { return datum.id; }) |
| + .enter() |
| + .append('div').attr('class', 'leaf').attr('id', function(datum){ |
| + return 'node-' + datum.id; |
| + }); |
| + |
| + // Define enter/update/exit for leaves |
| + cellsEnter |
| + .append('div') |
| + .attr('class', 'rect leaf_rect_entering') |
| + .style('z-index', function(datum) { return datum.id*2; }) |
| + .style('position', 'absolute') |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum){ return datum.dx; }) |
| + .style('height', function(datum){ return datum.dy; }) |
| + .style('opacity', '0') |
| + .style('background-color', function(datum) { |
| + if (datum.t === undefined) return 'rgb(220,220,220)'; |
| + return D3SymbolTreeMap.getColorForType(datum.t) |
| + .darker(0.3).toString(); |
| + }) |
| + .style('border', '1px solid black') |
| + .on('mouseover', function(datum){ |
| + thisTreeMap._highlightElement.call( |
| + thisTreeMap, datum, d3.select(this)); |
| + thisTreeMap._showInfoBox.call(thisTreeMap, datum); |
| + }) |
| + .on('mouseout', function(datum){ |
| + thisTreeMap._unhighlightElement.call( |
| + thisTreeMap, datum, d3.select(this)); |
| + thisTreeMap._hideInfoBox.call(thisTreeMap, datum); |
| + }) |
| + .on('mousemove', function(){ thisTreeMap._moveInfoBox.call( |
| + thisTreeMap, event); |
| + }); |
| + cellsEnter |
| + .append('div') |
| + .attr('class', 'label leaf_label_entering') |
| + .style('z-index', function(datum) { return (datum.id*2)+1; }) |
| + .style('position', 'absolute') |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum) { return datum.dx; }) |
| + .style('height', function(datum) { return datum.dy; }) |
| + .style('opacity', '0') |
| + .style('pointer-events', 'none') |
| + .style('-webkit-user-select', 'none') |
| + .style('overflow', 'hidden') // required for ellipsis |
| + .style('white-space', 'nowrap') // required for ellipsis |
| + .style('text-overflow', 'ellipsis') |
| + .style('text-align', 'center') |
| + .style('vertical-align', 'middle') |
| + .style('visibility', function(datum) { |
| + return (datum.dx < 15 || datum.dy < 15) ? 'hidden' : 'visible'; |
| + }) |
| + .text(function(datum) { return datum.n; }); |
| + |
| + // Complicated transition logic: See note in _handleInodes() |
| + this._mapContainer.selectAll('div.leaf_rect_entering').transition() |
| + .duration(thisTreeMap._enterDuration).delay( |
| + this._firstTransition ? 0 : thisTreeMap._exitDuration + |
| + thisTreeMap._updateDuration) |
| + .attr('class', 'rect leaf_rect') |
| + .style('opacity', '1') |
| + this._mapContainer.selectAll('div.leaf_label_entering').transition() |
| + .duration(thisTreeMap._enterDuration).delay( |
| + this._firstTransition ? 0 : thisTreeMap._exitDuration + |
| + thisTreeMap._updateDuration) |
| + .attr('class', 'label leaf_label') |
| + .style('opacity', '1') |
| + this._mapContainer.selectAll('div.leaf_rect').transition() |
| + .duration(thisTreeMap._updateDuration).delay(thisTreeMap._exitDuration) |
| + .style('opacity', '1') |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum){ return datum.dx; }) |
| + .style('height', function(datum){ return datum.dy; }); |
| + this._mapContainer.selectAll('div.leaf_label').transition() |
| + .duration(thisTreeMap._updateDuration).delay(thisTreeMap._exitDuration) |
| + .style('opacity', '1') |
| + .style('visibility', function(datum) { |
| + return (datum.dx < 15 || datum.dy < 15) ? 'hidden' : 'visible'; |
| + }) |
| + .style('left', function(datum){ return datum.x; }) |
| + .style('top', function(datum){ return datum.y; }) |
| + .style('width', function(datum) { return datum.dx; }) |
| + .style('height', function(datum) { return datum.dy; }); |
| + var exit = this._mapContainer.selectAll('div.leaf') |
| + .data(leaves, function(datum) { return 'leaf-' + datum.id; }) |
| + .exit(); |
| + exit.selectAll('div.leaf_rect').transition() |
| + .duration(thisTreeMap._exitDuration) |
| + .style('opacity', 0); |
| + exit.selectAll('div.leaf_label').transition() |
| + .duration(thisTreeMap._exitDuration) |
| + .style('opacity', 0); |
| + exit.transition().delay(thisTreeMap._exitDuration + 1).remove(); |
| + |
| + console.log(leaves.length + ' leaves layed out.'); |
| + console.timeEnd('_handleLeaves'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._makeSymbolBucketBackgroundImage = function(datum) { |
| + if (datum.t === undefined && datum.depth == this._currentMaxDepth) { |
| + var text = ''; |
| + var lastStop = 0; |
| + for (var x=0; x<D3SymbolTreeMap._NM_SYMBOL_TYPES.length; x++) { |
| + symbol_type = D3SymbolTreeMap._NM_SYMBOL_TYPES.charAt(x); |
| + var stats = datum.symbol_stats[symbol_type]; |
| + if (stats !== undefined) { |
| + if (text.length !== 0) { |
| + text += ', '; |
| + } |
| + var percent = 100 * (stats.size / datum.value); |
| + var nowStop = lastStop + percent; |
| + var tempcolor = D3SymbolTreeMap.getColorForType(symbol_type); |
| + var color = d3.rgb(tempcolor).toString(); |
| + text += color + ' ' + lastStop + '%, ' + color + ' ' + |
| + nowStop + '%'; |
| + lastStop = nowStop; |
| + } |
| + } |
| + return 'linear-gradient(' + (datum.dx > datum.dy ? 'to right' : |
| + 'to bottom') + ', ' + text + ')'; |
| + } else { |
| + return 'none'; |
| + } |
| +} |
| + |
| +D3SymbolTreeMap.prototype.pathFor = function(datum) { |
| + if (datum.__path) return datum.__path; |
| + parts=[]; |
| + node = datum; |
| + while (node) { |
| + if (node.k === 'p') { // path node |
| + if(node.n !== '/') parts.unshift(node.n); |
| + } |
| + node = node.parent; |
| + } |
| + datum.__path = '/' + parts.join('/'); |
| + return datum.__path; |
| +} |
| + |
| +D3SymbolTreeMap.prototype._createHighlight = function(datum, selection) { |
| + var x = parseInt(selection.style('left')); |
| + var y = parseInt(selection.style('top')); |
| + var w = parseInt(selection.style('width')); |
| + var h = parseInt(selection.style('height')); |
| + datum.highlight = this._mapContainer.append('div') |
| + .attr('id', 'h-' + datum.id) |
| + .attr('class', 'highlight') |
| + .style('pointer-events', 'none') |
| + .style('-webkit-user-select', 'none') |
| + .style('z-index', '999999') |
| + .style('position', 'absolute') |
| + .style('top', y-2) |
| + .style('left', x-2) |
| + .style('width', w+4) |
| + .style('height', h+4) |
| + .style('margin', 0) |
| + .style('padding', 0) |
| + .style('border', '4px outset rgba(250,40,200,0.9)') |
| + .style('box-sizing', 'border-box') |
| + .style('opacity', 0.0); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._showHighlight = function(datum, selection) { |
| + if (datum === this._currentRoot) return; |
| + if (datum.highlight === undefined) { |
| + this._createHighlight(datum, selection); |
| + } |
| + datum.highlight.transition().duration(200).style('opacity', 1.0); |
| + /* |
|
bulach
2014/04/16 16:32:16
nit: remove? (ditto for 769-777 block below)
Andrew Hayden (chromium.org)
2014/04/16 17:53:21
Done.
|
| + var ancestors = []; |
| + var node = datum.parent; |
| + var boostedZ = 999999; |
| + while (node && node !== this._currentRoot) { |
| + ancestors.unshift(node); |
| + var container = this._mapContainer.select('#node-' + node.id); |
| + container.select('.label').style('background-color', 'white'); |
| + node = node.parent; |
| + } |
| + this._highlightContainer.selectionParents = ancestors; |
| + */ |
| +} |
| + |
| +D3SymbolTreeMap.prototype._hideHighlight = function(datum, selection) { |
| + if (datum.highlight === undefined) return; |
| + datum.highlight.transition().duration(750) |
| + .style('opacity', 0) |
| + .each('end', function(){ |
| + if (datum.highlight) datum.highlight.remove(); |
| + delete datum.highlight; |
| + }); |
| + /* |
| + while(this._highlightContainer.selectionParents && |
| + this._highlightContainer.selectionParents.length > 0) { |
| + var node = this._highlightContainer.selectionParents.shift(); |
| + var container = this._mapContainer.select('#node-' + node.id); |
| + container.select('.label').style('background-color', null); |
| + } |
| + this._highlightContainer.style('visibility', 'hidden'); |
| + */ |
| +} |
| + |
| +D3SymbolTreeMap.prototype._createHoverMask = function() { |
| + // There are basically two ways to do this: |
| + // 1. SVG with a mask that defines a 'cutout' area. |
| + // 2. A grid of DIVs. |
| + // The SVG approach has potentially nasty z-index problems and would be |
| + // the only SVG in the app and is arguably as complex as the divs approach. |
| + // The DIVs must be resized whenever the treemap changes size. |
| + this._maskContainer = this._mapContainer.append('div') |
| + .attr('id', 'mask') |
| + .style('pointer-events', 'none') |
| + .style('-webkit-user-select', 'none') |
| + .style('z-index', '999999') |
| + .style('position', 'absolute') |
| + .style('top', 0) |
| + .style('left', 0) |
| + .style('width', this._mapWidth) |
| + .style('height', this._mapHeight) |
| + .style('visibility', 'visible'); |
| + |
| + var quadrants=['nw','n','ne','e','se','s','sw','w']; |
| + this._maskContainer.quadrants = {}; |
| + for (var i=0; i<quadrants.length; i++) { |
| + this._maskContainer.quadrants[quadrants[i]] = |
| + this._maskContainer.append('div') |
| + .attr('id', 'mask_' + quadrants[i]) |
| + .style('pointer-events', 'none') |
| + .style('-webkit-user-select', 'none') |
| + .style('background-color', 'white') |
| + .style('opacity', 0.5) |
| + .style('margin', 0) |
| + .style('padding', 0) |
| + .style('position', 'absolute') |
| + .style('visibility', 'hidden') |
| + .style('top', 0) |
| + .style('left', 0) |
| + .style('width', 0) |
| + .style('height', 0); |
| + } |
| +} |
| + |
| +D3SymbolTreeMap.prototype._showHoverMask = function(selection) { |
| + // Resize and reposition each quadrant |
| + var x = parseInt(selection.style('left')); |
| + var y = parseInt(selection.style('top')); |
| + var width = parseInt(selection.style('width')); |
| + var height = parseInt(selection.style('height')); |
| + |
| + var dimension = function(el,x,y,w,h) { |
| + return el.style('top', y) |
| + .style('left', x) |
| + .style('width', w) |
| + .style('height', h) |
| + .style('visibility', 'visible'); |
| + } |
| + var quads = this._maskContainer.quadrants; // for brevity below |
| + dimension(quads['nw'],0,0,x,y); |
| + dimension(quads['n'],x,0,width,y); |
| + dimension(quads['ne'],x+width,0,this._mapWidth - (x+width), y); |
| + dimension(quads['e'],x+width,y,this._mapWidth - (x+width), height); |
| + dimension(quads['se'],x+width,y+height,this._mapWidth - |
| + (x+width), this._mapHeight - (y+height)); |
| + dimension(quads['s'],x,y+height,width,this._mapHeight - (y+height)); |
| + dimension(quads['sw'],0,y+height,x,this._mapHeight - (y+height)); |
| + dimension(quads['w'],0,y,x,height); |
| + this._maskContainer.style('visibility', 'visible'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._hideHoverMask = function() { |
| + this._maskContainer.style('visibility', 'hidden'); |
| + var quadrants=['nw','n','ne','e','se','s','sw','w']; |
| + for (var i=0; i<quadrants.length; i++) { |
| + this._maskContainer |
| + .quadrants[quadrants[i]] |
| + .style('visibility', 'hidden'); |
| + } |
| +} |
| + |
| +D3SymbolTreeMap.prototype._createInfoBox = function() { |
| + return d3.select('body') |
| + .append('div') |
| + .attr('id', 'infobox') |
| + .style('z-index', '2147483647') // (2^31) - 1: Hopefully safe :) |
| + .style('position', 'absolute') |
| + .style('visibility', 'hidden') |
| + .style('background-color', 'rgba(255,255,255, 0.9)') |
| + .style('border', '1px solid black') |
| + .style('padding', '10px') |
| + .style('-webkit-user-select', 'none') |
| + .style('box-shadow', '3px 3px rgba(70,70,70,0.5)') |
| + .style('border-radius', '10px') |
| + .style('white-space', 'nowrap'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._showInfoBox = function(datum) { |
| + this.infobox.text(''); |
| + var numSymbols = 0; |
| + var sizeish = D3SymbolTreeMap._pretty(datum.value) + ' bytes (' + |
| + D3SymbolTreeMap._byteify(datum.value) + ')'; |
| + if (datum.k === 'p' || datum.k === 'b') { // path or bucket |
| + if (datum.symbol_stats) { // can be empty if filters are applied |
| + for (var x=0; x<D3SymbolTreeMap._NM_SYMBOL_TYPES.length; x++) { |
| + symbol_type = D3SymbolTreeMap._NM_SYMBOL_TYPES.charAt(x); |
| + var stats = datum.symbol_stats[symbol_type]; |
| + if (stats !== undefined) numSymbols += stats.count; |
| + } |
| + } |
| + } else if (datum.k === 's') { // symbol |
| + numSymbols = 1; |
| + } |
| + |
| + if (datum.k === 'p' && !datum.lastPathElement) { |
| + this.infobox.append('div').text('Directory: ' + this.pathFor(datum)) |
| + this.infobox.append('div').text('Size: ' + sizeish); |
| + } else { |
| + if (datum.k === 'p') { // path |
| + this.infobox.append('div').text('File: ' + this.pathFor(datum)) |
| + this.infobox.append('div').text('Size: ' + sizeish); |
| + } else if (datum.k === 'b') { // bucket |
| + this.infobox.append('div').text('Symbol Bucket: ' + |
| + D3SymbolTreeMap._getSymbolDescription(datum.t)); |
| + this.infobox.append('div').text('Count: ' + numSymbols); |
| + this.infobox.append('div').text('Size: ' + sizeish); |
| + this.infobox.append('div').text('Location: ' + this.pathFor(datum)) |
| + } else if (datum.k === 's') { // symbol |
| + this.infobox.append('div').text('Symbol: ' + datum.n); |
| + this.infobox.append('div').text('Type: ' + |
| + D3SymbolTreeMap._getSymbolDescription(datum.t)); |
| + this.infobox.append('div').text('Size: ' + sizeish); |
| + this.infobox.append('div').text('Location: ' + this.pathFor(datum)) |
| + } |
| + } |
| + if (datum.k === 'p') { |
| + this.infobox.append('div') |
| + .text('Number of symbols: ' + D3SymbolTreeMap._pretty(numSymbols)); |
| + if (datum.symbol_stats) { // can be empty if filters are applied |
| + var table = this.infobox.append('table') |
| + .attr('border', 1).append('tbody'); |
| + var header = table.append('tr'); |
| + header.append('th').text('Type'); |
| + header.append('th').text('Count'); |
| + header.append('th') |
| + .style('white-space', 'nowrap') |
| + .text('Total Size (Bytes)'); |
| + for (var x=0; x<D3SymbolTreeMap._NM_SYMBOL_TYPES.length; x++) { |
| + symbol_type = D3SymbolTreeMap._NM_SYMBOL_TYPES.charAt(x); |
| + var stats = datum.symbol_stats[symbol_type]; |
| + if (stats !== undefined) { |
| + var tr = table.append('tr'); |
| + tr.append('td') |
| + .style('white-space', 'nowrap') |
| + .text(D3SymbolTreeMap._getSymbolDescription( |
| + symbol_type)); |
| + tr.append('td').text(D3SymbolTreeMap._pretty(stats.count)); |
| + tr.append('td').text(D3SymbolTreeMap._pretty(stats.size)); |
| + } |
| + } |
| + } |
| + } |
| + this.infobox.style('visibility', 'visible'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._hideInfoBox = function(datum) { |
| + this.infobox.style('visibility', 'hidden'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype._moveInfoBox = function(event) { |
| + var element = document.getElementById('infobox'); |
| + var w = element.offsetWidth; |
| + var h = element.offsetHeight; |
| + var offsetLeft = 10; |
| + var offsetTop = 10; |
| + |
| + var rightLimit = window.innerWidth; |
| + var rightEdge = event.pageX + offsetLeft + w; |
| + if (rightEdge > rightLimit) { |
| + // Too close to screen edge, reflect around the cursor |
| + offsetLeft = -1 * (w + offsetLeft); |
| + } |
| + |
| + var bottomLimit = window.innerHeight; |
| + var bottomEdge = event.pageY + offsetTop + h; |
| + if (bottomEdge > bottomLimit) { |
| + // Too close ot screen edge, reflect around the cursor |
| + offsetTop = -1 * (h + offsetTop); |
| + } |
| + |
| + this.infobox.style('top', (event.pageY + offsetTop) + 'px') |
| + .style('left', (event.pageX + offsetLeft) + 'px'); |
| +} |
| + |
| +D3SymbolTreeMap.prototype.biggestSymbols = function(maxRecords) { |
| + var result = undefined; |
| + var smallest = undefined; |
| + var sortFunction = function(a,b) { |
| + var result = b.value - a.value; |
| + if (result !== 0) return result; // sort by size |
| + var pathA = treemap.pathFor(a); // sort by path |
| + var pathB = treemap.pathFor(b); |
| + if (pathA > pathB) return 1; |
| + if (pathB > pathA) return -1; |
| + return a.n - b.n; // sort by symbol name |
| + }; |
| + this.visitFromDisplayedRoot(function(datum) { |
| + if (datum.children) return; // ignore non-leaves |
| + if (!result) { // first element |
| + result = [datum]; |
| + smallest = datum.value; |
| + return; |
| + } |
| + if (result.length < maxRecords) { // filling the array |
| + result.push(datum); |
| + return; |
| + } |
| + if (datum.value > smallest) { // array is already full |
| + result.push(datum); |
| + result.sort(sortFunction); |
| + result.pop(); // get rid of smallest element |
| + smallest = result[maxRecords - 1].value; // new threshold for entry |
| + } |
| + }); |
| + result.sort(sortFunction); |
| + return result; |
| +} |
| + |
| +D3SymbolTreeMap.prototype.biggestPaths = function(maxRecords) { |
| + var result = undefined; |
| + var smallest = undefined; |
| + var sortFunction = function(a,b) { |
| + var result = b.value - a.value; |
| + if (result !== 0) return result; // sort by size |
| + var pathA = treemap.pathFor(a); // sort by path |
| + var pathB = treemap.pathFor(b); |
| + if (pathA > pathB) return 1; |
| + if (pathB > pathA) return -1; |
| + console.log('warning, multiple entries for the same path: ' + pathA); |
| + return 0; // should be impossible |
| + }; |
| + this.visitFromDisplayedRoot(function(datum) { |
| + if (!datum.lastPathElement) return; // ignore non-files |
| + if (!result) { // first element |
| + result = [datum]; |
| + smallest = datum.value; |
| + return; |
| + } |
| + if (result.length < maxRecords) { // filling the array |
| + result.push(datum); |
| + return; |
| + } |
| + if (datum.value > smallest) { // array is already full |
| + result.push(datum); |
| + result.sort(sortFunction); |
| + result.pop(); // get rid of smallest element |
| + smallest = result[maxRecords - 1].value; // new threshold for entry |
| + } |
| + }); |
| + result.sort(sortFunction); |
| + return result; |
| +} |