diff --git a/local-cli/server/middleware/heapCapture/heapCapture.html b/local-cli/server/middleware/heapCapture/heapCapture.html
index 6cfbe840c..af900934a 100644
--- a/local-cli/server/middleware/heapCapture/heapCapture.html
+++ b/local-cli/server/middleware/heapCapture/heapCapture.html
@@ -3,8 +3,8 @@
JSC Heap Capture
-
-
+
+
diff --git a/local-cli/server/middleware/heapCapture/out/aggrow.js b/local-cli/server/middleware/heapCapture/out/aggrow.js
index d980d073e..e5eec1099 100644
--- a/local-cli/server/middleware/heapCapture/out/aggrow.js
+++ b/local-cli/server/middleware/heapCapture/out/aggrow.js
@@ -16,7 +16,7 @@
// pivot around frames in the middle of a stack by callers / callees
// graphing?
-function stringInterner(){// eslint-disable-line no-unused-vars
+function StringInterner(){// eslint-disable-line no-unused-vars
var strings=[];
var ids={};
return{
@@ -37,20 +37,16 @@ return strings[id];
}
-function stackData(stackIdMap,maxDepth){// eslint-disable-line no-unused-vars
-return{
-maxDepth:maxDepth,
-get:function getStack(id){
-return stackIdMap[id];
-}};
-
-}
-
-function stackRegistry(){// eslint-disable-line no-unused-vars
+function StackRegistry(){// eslint-disable-line no-unused-vars
return{
root:{id:0},
nodeCount:1,
+maxDepth:-1,
+stackIdMap:null,
insert:function insertNode(parent,frameId){
+if(this.stackIdMap!==null){
+throw'stacks already flattened';
+}
var node=parent[frameId];
if(node===undefined){
node={id:this.nodeCount};
@@ -59,7 +55,13 @@ parent[frameId]=node;
}
return node;
},
+get:function getStackArray(id){
+return this.stackIdMap[id];
+},
flatten:function flattenStacks(){
+if(this.stackIdMap!==null){
+return;
+}
var stackFrameCount=0;
function countStacks(tree,depth){
var leaf=true;
@@ -109,13 +111,150 @@ stackIdMap[id]=newStack;
return stackIdMap[id];
}
flattenStacksImpl(this.root,[]);
-
-return new stackData(stackIdMap,maxStackDepth);
+this.root=null;
+this.stackIdMap=stackIdMap;
+this.maxDepth=maxStackDepth;
}};
}
-function aggrow(numRows){// eslint-disable-line no-unused-vars
+function AggrowData(columns){// eslint-disable-line no-unused-vars
+var columnCount=columns.length;
+var columnConverter=columns.map(function(c){
+switch(c.type){
+case'int':// stores raw value
+return function(i){return i;};
+case'string':// stores interned id of string
+return function(s){return c.strings.intern(s);};
+case'stack':// stores id of stack node
+return function(s){return s.id;};
+default:
+throw'unknown AggrowData column type';}
+
+});
+return{
+data:new Int32Array(0),
+columns:columns,
+rowCount:0,
+rowInserter:function rowInserter(numRows){
+console.log(
+'increasing row data from '+(this.data.length*4).toLocaleString()+' B to '+
+(this.data.length*4+numRows*columnCount*4).toLocaleString()+' B');
+
+var newData=new Int32Array(this.data.length+numRows*columnCount);
+newData.set(this.data);
+var currOffset=this.data.length;
+var endOffset=newData.length;
+this.data=newData;
+this.rowCount=newData.length/columnCount;
+return{
+insertRow:function insertRow(){
+if(currOffset>=endOffset){
+throw'tried to insert data off end of added range';
+}
+if(arguments.length!==columnCount){
+throw'expected data for '+columnCount.toString()+' columns, got'+
+arguments.length.toString()+' columns';
+}
+for(var i=0;i>>0).toString();},
+function(rowA,rowB){return data[rowA*columnCount+index]-data[rowB*columnCount+index];});
+
+},
+addStackExpander:function addStackExpander(expanderName,columnName,formatter){
+// TODO: options for caller/callee, pivoting
+var index=columnIndex(columnName,'stack');
+var stacks=columns[index].stacks;
+return this.expander.addCalleeStackExpander(
+expanderName,
+stacks.maxDepth,
+function(row){return stacks.get(data[row*columnCount+index]);},
+formatter);
+
+}};
+
+}
+
+function AggrowExpander(numRows){// eslint-disable-line no-unused-vars
// expander ID definitions
var FIELD_EXPANDER_ID_MIN=0x0000;
var FIELD_EXPANDER_ID_MAX=0x7fff;
@@ -223,8 +362,8 @@ row.state|=NODE_REAGGREGATE_BIT;
function evaluateAggregates(row){
if((row.state&NODE_EXPANDED_BIT)!==0){
var children=row.children;
-for(var _i=0;_i=FIELD_EXPANDER_ID_MIN&&id<=FIELD_EXPANDER_ID_MAX){
if(id-FIELD_EXPANDER_ID_MIN>=state.fieldExpanders.length){
throw'field expander for id '+id.toString()+' does not exist!';
@@ -475,12 +614,12 @@ throw'stack expander for id '+id.toString()+' does not exist!';
}
}
}
-for(var _i9=0;_i9state.aggregators.length){
throw'aggregator id '+id.toString()+' not valid';
}
@@ -524,19 +663,19 @@ state.activeAggregators=ids.slice();
// so someone has to start with root, and it might as well be right away
evaluateAggregate(state.root);
var sorter=noSortOrder;var _loop2=function _loop2(
-_i12){
-var ascending=(ids[_i12]&ACTIVE_AGGREGATOR_ASC_BIT)!==0;
-var id=ids[_i12]&ACTIVE_AGGREGATOR_MASK;
+_i13){
+var ascending=(ids[_i13]&ACTIVE_AGGREGATOR_ASC_BIT)!==0;
+var id=ids[_i13]&ACTIVE_AGGREGATOR_MASK;
var comparer=state.aggregators[id].sorter;
var captureSorter=sorter;
-var captureIndex=_i12;
+var captureIndex=_i13;
sorter=function sorter(a,b){
var c=comparer(a.aggregates[captureIndex],b.aggregates[captureIndex]);
if(c===0){
return captureSorter(a,b);
}
return ascending?-c:c;
-};};for(var _i12=ids.length-1;_i12>=0;_i12--){_loop2(_i12);
+};};for(var _i13=ids.length-1;_i13>=0;_i13--){_loop2(_i13);
}
state.sorter=sorter;
state.root.state|=NODE_REORDER_BIT;
@@ -546,8 +685,8 @@ return state.activeAggregators.slice();
},
getRows:function getRows(top,height){
var result=new Array(height);
-for(var _i13=0;_i130){_loop();
}
}
-function captureRegistry(){
-var strings=stringInterner();
-var stacks=stackRegistry(strings);
-var data=new Int32Array(0);
-
-var idField=0;
-var typeField=1;
-var sizeField=2;
-var traceField=3;
-var pathField=4;
-var reactField=5;
-var valueField=6;
-var moduleField=7;
-var numFields=8;
-
-return{
-strings:strings,
-stacks:stacks,
-data:data,
-register:function registerCapture(captureId,capture){var _this2=this;
+function registerCapture(data,captureId,capture,stacks,strings){
// NB: capture.refs is potentially VERY large, so we try to avoid making
-// copies, even of iteration is a bit more annoying.
+// copies, even if iteration is a bit more annoying.
var rowCount=0;
for(var id in capture.refs){// eslint-disable-line no-unused-vars
rowCount++;
@@ -321,162 +302,80 @@ rowCount++;
for(var _id in capture.markedBlocks){// eslint-disable-line no-unused-vars
rowCount++;
}
-console.log(
-'increasing row data from '+(this.data.length*4).toString()+'B to '+
-(this.data.length*4+rowCount*numFields*4).toString()+'B');
-
-var newData=new Int32Array(this.data.length+rowCount*numFields);
-newData.set(data);
-var dataOffset=this.data.length;
-this.data=null;
-
-registerPathToRoot(capture.refs,this.stacks,this.strings);
-var internedCaptureId=this.strings.intern(captureId);
-var noneString=this.strings.intern('#none');
-var noneStack=this.stacks.insert(this.stacks.root,noneString);
+var inserter=data.rowInserter(rowCount);
+registerPathToRoot(capture.refs,stacks,strings);
+var noneString=strings.intern('#none');
+var noneStack=stacks.insert(stacks.root,noneString);
forEachRef(capture.refs,function(visitor){
+// want to data.append(value, value, value), not IDs
var ref=visitor.getRef();
var id=visitor.id;
-newData[dataOffset+idField]=parseInt(id,16);
-newData[dataOffset+typeField]=_this2.strings.intern(ref.type);
-newData[dataOffset+sizeField]=ref.size;
-newData[dataOffset+traceField]=internedCaptureId;
-if(ref.rootPath===undefined){
-newData[dataOffset+pathField]=noneStack.id;
-}else{
-newData[dataOffset+pathField]=ref.rootPath.id;
-}
-if(ref.reactTree===undefined){
-newData[dataOffset+reactField]=noneStack.id;
-}else{
-newData[dataOffset+reactField]=ref.reactTree.id;
-}
-newData[dataOffset+valueField]=_this2.strings.intern(visitor.getValue());
-if(ref.module){
-newData[dataOffset+moduleField]=_this2.strings.intern(ref.module);
-}else{
-newData[dataOffset+moduleField]=noneString;
-}
-dataOffset+=numFields;
+inserter.insertRow(
+parseInt(id,16),
+ref.type,
+ref.size,
+captureId,
+ref.rootPath===undefined?noneStack:ref.rootPath,
+ref.reactTree===undefined?noneStack:ref.reactTree,
+visitor.getValue(),
+ref.module===undefined?'#none':ref.module);
+
});
for(var _id2 in capture.markedBlocks){
var block=capture.markedBlocks[_id2];
-newData[dataOffset+idField]=parseInt(_id2,16);
-newData[dataOffset+typeField]=this.strings.intern('Marked Block Overhead');
-newData[dataOffset+sizeField]=block.capacity-block.size;
-newData[dataOffset+traceField]=internedCaptureId;
-newData[dataOffset+pathField]=noneStack.id;
-newData[dataOffset+reactField]=noneStack.id;
-newData[dataOffset+valueField]=this.strings.intern(
-'capacity: '+block.capacity+
-', size: '+block.size+
-', granularity: '+block.cellSize);
+inserter.insertRow(
+parseInt(_id2,16),
+'Marked Block Overhead',
+block.capacity-block.size,
+captureId,
+noneStack,
+noneStack,
+'capacity: '+block.capacity+', size: '+block.size+', granularity: '+block.cellSize,
+'#none');
-newData[dataOffset+moduleField]=noneString;
-dataOffset+=numFields;
}
-this.data=newData;
-},
-getAggrow:function getAggrow(){
-var agStrings=this.strings;
-var agStacks=this.stacks.flatten();
-var agData=this.data;
-var agNumRows=agData.length/numFields;
-var ag=new aggrow(agNumRows);
-
-ag.addFieldExpander('Id',
-function getId(row){
-var id=agData[row*numFields+idField];
-if(id<0){
-id+=0x100000000;// data is int32, id is uint32
+inserter.done();
}
-return'0x'+id.toString(16);
-},
-function compareAddress(rowA,rowB){
-return agData[rowA*numFields+idField]-agData[rowB*numFields+idField];
-});
-var typeExpander=ag.addFieldExpander('Type',
-function getType(row){return agStrings.get(agData[row*numFields+typeField]);},
-function compareType(rowA,rowB){
-return agData[rowA*numFields+typeField]-agData[rowB*numFields+typeField];
-});
+if(preLoadedCapture){
+var strings=StringInterner();
+var stacks=new StackRegistry();
+var columns=[
+{name:'id',type:'int'},
+{name:'type',type:'string',strings:strings},
+{name:'size',type:'int'},
+{name:'trace',type:'string',strings:strings},
+{name:'path',type:'stack',stacks:stacks},
+{name:'react',type:'stack',stacks:stacks},
+{name:'value',type:'string',strings:strings},
+{name:'module',type:'string',strings:strings}];
-ag.addFieldExpander('Size',
-function getSize(row){return agData[row*numFields+sizeField].toString();},
-function compareSize(rowA,rowB){
-return agData[rowA*numFields+sizeField]-agData[rowB*numFields+sizeField];
-});
-
-ag.addFieldExpander('Trace',
-function getSize(row){return agStrings.get(agData[row*numFields+traceField]);},
-function compareSize(rowA,rowB){
-return agData[rowA*numFields+traceField]-agData[rowB*numFields+traceField];
-});
-
-var pathExpander=ag.addCalleeStackExpander(
-'Path',
-agStacks.maxDepth,
-function getStack(row){return agStacks.get(agData[row*numFields+pathField]);},
-function getFrame(id){return agStrings.get(id);});
-
-
-var reactExpander=ag.addCalleeStackExpander(
-'React Tree',
-agStacks.maxDepth,
-function getStack(row){return agStacks.get(agData[row*numFields+reactField]);},
-function getFrame(id){return agStrings.get(id);});
-
-
-var valueExpander=ag.addFieldExpander('Value',
-function getValue(row){return agStrings.get(agData[row*numFields+valueField]);},
-function compareValue(rowA,rowB){
-return agData[rowA*numFields+valueField]-agData[rowB*numFields+valueField];
-});
-
-var moduleExpander=ag.addFieldExpander('Module',
-function getModule(row){return agStrings.get(agData[row*numFields+moduleField]);},
-function compareModule(rowA,rowB){
-return agData[rowA*numFields+moduleField]-agData[rowB*numFields+moduleField];
-});
-
-var sizeAggregator=ag.addAggregator('Size',
-function aggregateSize(indices){
-var size=0;
-for(var i=0;i':'...';
headers.push(
React.createElement(DropTarget,{
id:'expander:insert:'+(_i+1).toString(),
-dropFilter:function dropFilter(){return true;},
dropAction:this.props.dropAction},
React.createElement('div',{style:{
@@ -234,7 +224,7 @@ dropAction:React.PropTypes.func.isRequired};var
Table=function(_React$Component6){_inherits(Table,_React$Component6);// eslint-disable-line no-unused-vars
-function Table(props){_classCallCheck(this,Table);var _this7=_possibleConstructorReturn(this,Object.getPrototypeOf(Table).call(this,
+function Table(props){_classCallCheck(this,Table);var _this7=_possibleConstructorReturn(this,(Table.__proto__||Object.getPrototypeOf(Table)).call(this,
props));
_this7.state={
aggrow:props.aggrow,
@@ -547,5 +537,5 @@ document.body.addEventListener('keydown',this.keydown);
{
document.body.removeEventListener('keydown',this.keydown);
}}]);return Table;}(React.Component);
-//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/table.js"],"names":["rowHeight","treeIndent","Draggable","props","id","dragStart","e","dataTransfer","setData","React","cloneElement","children","draggable","onDragStart","Component","propTypes","PropTypes","element","isRequired","string","DropTarget","thisId","dropFilter","dropAction","onDragOver","sourceId","getData","preventDefault","onDrop","func","ExpanderConfiguration","aggrow","expander","toString","width","height","border","margin","getExpanderName","TableConfiguration","state","expanded","ex","expanderText","config","getExpanders","map","renderExpander","display","flexDirection","overflow","borderLeft","setState","object","TableHeader","aggregators","getActiveAggregators","expanders","getActiveExpanders","headers","i","length","name","getAggregatorName","push","s","startsWith","backgroundColor","flexShrink","textAlign","bg","sep","alignItems","borderBottom","Table","viewport","top","cursor","target","Math","floor","scrollTop","clientHeight","ceil","row","newCursor","contract","console","log","expand","_scrollDiv","scrollDiv","getRows","keyCode","_keepCursorInViewport","getHeight","canContract","_contractRow","getRowIndent","indent","canExpand","_expandRow","d","sIndex","parseInt","substr","dIndex","active","dragged","splice","setActiveAggregators","setActiveExpanders","sExpander","substring","flexGrow","scroll","div","position","renderVirtualizedRows","rows","renderRow","child","columns","rowText","aggregates","parent","aggregate","getRowAggregate","marginLeft","getRowLabel","whiteSpace","marginRight","keydown","bind","document","body","addEventListener","removeEventListener"],"mappings":"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","file":"table.js","sourcesContent":["/**\n * Copyright (c) 2016-present, Facebook, Inc.\n * All rights reserved.\n *\n * This source code is licensed under the BSD-style license found in the\n * LICENSE file in the root directory of this source tree. An additional grant\n * of patent rights can be found in the PATENTS file in the same directory.\n */\n\n'use strict';\n/*eslint no-console-disallow: \"off\"*/\n/*global React:true*/\n\n// TODO:\n// selection and arrow keys for navigating\n\nconst rowHeight = 20;\nconst treeIndent = 16;\n\nclass Draggable extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n  }\n\n  render() {\n    const id = this.props.id;\n    function dragStart(e) {\n      e.dataTransfer.setData('text/plain', id);\n    }\n    return React.cloneElement(\n      this.props.children,\n      { draggable: 'true', onDragStart: dragStart }\n    );\n  }\n}\nDraggable.propTypes = {\n  children: React.PropTypes.element.isRequired,\n  id: React.PropTypes.string.isRequired,\n};\n\nclass DropTarget extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n  }\n\n  render() {\n    const thisId = this.props.id;\n    const dropFilter = this.props.dropFilter;\n    const dropAction = this.props.dropAction;\n    return React.cloneElement(\n      this.props.children,\n      {\n        onDragOver: (e) => {\n          const sourceId = e.dataTransfer.getData('text/plain');\n          if (dropFilter(sourceId)) {\n            e.preventDefault();\n          }\n        },\n        onDrop: (e) => {\n          const sourceId = e.dataTransfer.getData('text/plain');\n          if (dropFilter(sourceId)) {\n            e.preventDefault();\n            dropAction(sourceId, thisId);\n          }\n        },\n      }\n    );\n  }\n}\n\nDropTarget.propTypes = {\n  children: React.PropTypes.element.isRequired,\n  id: React.PropTypes.string.isRequired,\n  dropFilter: React.PropTypes.func.isRequired,\n  dropAction: React.PropTypes.func.isRequired,\n};\n\nclass ExpanderConfiguration extends React.Component {\n  constructor(props) {\n    super(props);\n  }\n  render() {\n    const aggrow = this.props.aggrow;\n    const expander = this.props.expander;\n    return (\n      <Draggable id={'expander:add:' + expander.toString()}>\n        <div\n          style={{\n            width: 'auto',\n            height: '26px',\n            border: '1px solid darkGray',\n            margin: '2px',\n          }}>\n          {aggrow.getExpanderName(expander)}\n        </div>\n      </Draggable>\n    );\n  }\n}\n\nclass TableConfiguration extends React.Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      expanded: false,\n    };\n  }\n  renderExpander(ex) {\n    return (<ExpanderConfiguration aggrow={this.props.aggrow} expander={ex} />);\n  }\n  render() {\n    const expanderText = this.state.expanded ? '>>' : '<<';\n    const aggrow = this.props.aggrow;\n    let config = [];\n    if (this.state.expanded) {\n      config = aggrow.getExpanders().map((ex) => this.renderExpander(ex));\n    }\n    return (\n      <div style={{\n          width: this.state.expanded ? '512px' : '26px',\n          height: '100%',\n          display: 'flex',\n          flexDirection: 'column',\n          overflow: 'hidden',\n          borderLeft: '2px solid black',\n        }}>\n        <div style={{\n            width: '100%',\n            height: '26px',\n            border: '1px solid darkGray',\n          }}\n          onClick={ (e) => this.setState({expanded: !this.state.expanded}) }>\n          { expanderText }\n        </div>\n        { config }\n      </div>\n    );\n  }\n}\n\nTableConfiguration.propTypes = {\n  aggrow: React.PropTypes.object.isRequired,\n};\n\nclass TableHeader extends React.Component {\n  constructor(props) {\n    super(props);\n  }\n  render() {\n    const aggrow = this.props.aggrow;\n    const aggregators = aggrow.getActiveAggregators();\n    const expanders = aggrow.getActiveExpanders();\n    const headers = [];\n    for (let i = 0; i < aggregators.length; i++) {\n      const name = aggrow.getAggregatorName(aggregators[i]);\n      headers.push((\n        <DropTarget\n          id={'aggregate:insert:' + i.toString()}\n          dropFilter={(s) => s.startsWith('aggregate')}\n          dropAction={this.props.dropAction}\n        >\n          <div style={{\n            width: '16px',\n            height: 'inherit',\n            backgroundColor: 'darkGray',\n            flexShrink: '0' }}\n          ></div>\n        </DropTarget>));\n      headers.push((<Draggable id={'aggregate:active:' + i.toString()}>\n          <div style={{ width: '128px', textAlign: 'center', flexShrink: '0' }}>{name}</div>\n        </Draggable>));\n    }\n    headers.push((\n      <DropTarget\n        id=\"divider:insert\"\n        dropFilter={(s) => s.startsWith('aggregate') || s.startsWith('expander')}\n        dropAction={this.props.dropAction}\n      >\n        <div style={{\n          width: '16px',\n          height: 'inherit',\n          backgroundColor: 'gold',\n          flexShrink: '0'\n        }}></div>\n      </DropTarget>));\n    for (let i = 0; i < expanders.length; i++) {\n      const name = aggrow.getExpanderName(expanders[i]);\n      const bg = (i % 2 === 0) ? 'white' : 'lightGray';\n      headers.push((<Draggable id={'expander:active:' + i.toString()}>\n          <div style={{\n            width: '128px',\n            textAlign: 'center',\n            backgroundColor: bg,\n            flexShrink: '0'\n          }}>\n            {name}\n          </div>\n        </Draggable>));\n      const sep = i + 1 < expanders.length ? '->' : '...';\n      headers.push((\n        <DropTarget\n          id={'expander:insert:' + (i + 1).toString()}\n          dropFilter={()=>{return true; }}\n          dropAction={this.props.dropAction}\n        >\n          <div style={{\n            height: 'inherit',\n            backgroundColor: 'darkGray',\n            flexShrink: '0'\n          }}>\n            {sep}\n          </div>\n        </DropTarget>)\n      );\n    }\n    return (\n      <div style={{\n        width: '100%',\n        height: '26px',\n        display: 'flex',\n        flexDirection: 'row',\n        alignItems: 'center',\n        borderBottom: '2px solid black',\n      }}>\n        {headers}\n      </div>\n    );\n  }\n}\n\nTableHeader.propTypes = {\n  aggrow: React.PropTypes.object.isRequired,\n  dropAction: React.PropTypes.func.isRequired,\n};\n\nclass Table extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n    this.state = {\n      aggrow: props.aggrow,\n      viewport: { top: 0, height: 100 },\n      cursor: 0,\n    };\n  }\n\n  scroll(e) {\n    const viewport = e.target;\n    const top = Math.floor((viewport.scrollTop - viewport.clientHeight * 1.0) / rowHeight);\n    const height = Math.ceil(viewport.clientHeight * 3.0 / rowHeight);\n    if (top !== this.state.viewport.top || height !== this.state.viewport.height) {\n      this.setState({viewport: {top, height}});\n    }\n  }\n\n  _contractRow(row) {\n    let newCursor = this.state.cursor;\n    if (newCursor > row.top && newCursor < row.top + row.height) { // in contracted section\n      newCursor = row.top;\n    } else if (newCursor >= row.top + row.height) { // below contracted section\n      newCursor -= row.height - 1;\n    }\n    this.state.aggrow.contract(row);\n    this.setState({cursor: newCursor});\n    console.log('-' + row.top);\n  }\n\n  _expandRow(row) {\n    let newCursor = this.state.cursor;\n    this.state.aggrow.expand(row);\n    if (newCursor > row.top) {  // below expanded section\n      newCursor += row.height - 1;\n    }\n    this.setState({cursor: newCursor});\n    console.log('+' + row.top);\n  }\n\n  _scrollDiv: null;\n\n  _keepCursorInViewport() {\n    if (this._scrollDiv) {\n      const cursor = this.state.cursor;\n      const scrollDiv = this._scrollDiv;\n      if (cursor * rowHeight < scrollDiv.scrollTop + scrollDiv.clientHeight * 0.1) {\n        scrollDiv.scrollTop = cursor * rowHeight - scrollDiv.clientHeight * 0.1;\n      } else if ((cursor + 1) * rowHeight > scrollDiv.scrollTop + scrollDiv.clientHeight * 0.9) {\n        scrollDiv.scrollTop = (cursor + 1) * rowHeight - scrollDiv.clientHeight * 0.9;\n      }\n    }\n  }\n\n  keydown(e) {\n    const aggrow = this.state.aggrow;\n    let cursor = this.state.cursor;\n    let row = aggrow.getRows(cursor, 1)[0];\n    switch (e.keyCode) {\n      case 38: // up\n        if (cursor > 0) {\n          this.setState({cursor: cursor - 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 40: // down\n        if (cursor < aggrow.getHeight() - 1) {\n          this.setState({cursor: cursor + 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 37: // left\n        if (aggrow.canContract(row)) {\n          this._contractRow(row);\n        } else if (aggrow.getRowIndent(row) > 0) {\n          const indent = aggrow.getRowIndent(row) - 1;\n          while (aggrow.getRowIndent(row) > indent) {\n            cursor--;\n            row = aggrow.getRows(cursor, 1)[0];\n          }\n          this.setState({cursor: cursor});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 39: // right\n        if (aggrow.canExpand(row)) {\n          this._expandRow(row);\n        } else if (cursor < aggrow.getHeight() - 1) {\n          this.setState({cursor: cursor + 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n    }\n  }\n\n  dropAction(s, d) {\n    const aggrow = this.state.aggrow;\n    console.log('dropped ' + s + ' to ' + d);\n    if (s.startsWith('aggregate:active:')) {\n      const sIndex = parseInt(s.substr(17), 10);\n      let dIndex = -1;\n      const active = aggrow.getActiveAggregators();\n      const dragged = active[sIndex];\n      if (d.startsWith('aggregate:insert:')) {\n        dIndex = parseInt(d.substr(17), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = active.length;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      if (dIndex > sIndex) {\n        dIndex--;\n      }\n      active.splice(sIndex, 1);\n      active.splice(dIndex, 0, dragged);\n      aggrow.setActiveAggregators(active);\n      this.setState({cursor:0});\n    } else if (s.startsWith('expander:active:')) {\n      const sIndex = parseInt(s.substr(16), 10);\n      let dIndex = -1;\n      const active = aggrow.getActiveExpanders();\n      const dragged = active[sIndex];\n      if (d.startsWith('expander:insert:')) {\n        dIndex = parseInt(d.substr(16), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = 0;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      if (dIndex > sIndex) {\n        dIndex--;\n      }\n      active.splice(sIndex, 1);\n      active.splice(dIndex, 0, dragged);\n      aggrow.setActiveExpanders(active);\n      this.setState({cursor:0});\n    } else if (s.startsWith('expander:add:')) {\n      let dIndex = -1;\n      const sExpander = parseInt(s.substring(13), 10);\n      if (d.startsWith('expander:insert:')) {\n        dIndex = parseInt(d.substr(16), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = 0;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      const active = aggrow.getActiveExpanders();\n      active.splice(dIndex, 0, sExpander);\n      aggrow.setActiveExpanders(active);\n      this.setState({cursor:0});\n    }\n  }\n\n  render() {\n    return (\n      <div style={{ width: '100%', height: '100%', display: 'flex', flexDirection: 'row' }}>\n        <div style={{ width: '100%', height: '100%', display: 'flex', flexDirection: 'column', overflow: 'hidden', }}>\n          <TableHeader aggrow={this.state.aggrow} dropAction={(s, d) => this.dropAction(s, d)} />\n          <div\n            style={{\n              width: '100%',\n              flexGrow: '1',\n              overflow: 'scroll'\n            }}\n            onScroll={ (e) => this.scroll(e) }\n            ref={(div) => { this._scrollDiv = div; } }>\n            <div style={{ position: 'relative' }}>\n              { this.renderVirtualizedRows() }\n            </div>\n          </div>\n        </div>\n        <TableConfiguration aggrow={this.state.aggrow} />\n      </div>\n    );\n  }\n\n  renderVirtualizedRows() {\n    const aggrow = this.state.aggrow;\n    const viewport = this.state.viewport;\n    const rows = aggrow.getRows(viewport.top, viewport.height);\n    return (\n      <div style={{\n        position: 'absolute',\n        width: '100%',\n        height: (rowHeight * (aggrow.getHeight() + 20)).toString() + 'px'\n      }}>\n        { rows.map(child => this.renderRow(child)) }\n      </div>\n    );\n  }\n\n  renderRow(row) {\n    if (row === null) {\n      return null;\n    }\n    let bg = 'lightGray';\n    const aggrow = this.state.aggrow;\n    const columns = [];\n    let rowText = '';\n    const indent = 4 + aggrow.getRowIndent(row) * treeIndent;\n    const aggregates = aggrow.getActiveAggregators();\n    if (row.parent !== null && (row.parent.expander % 2 === 0)) {\n      bg = 'white';\n    }\n    if (row.top === this.state.cursor) {\n      bg = 'lightblue';\n    }\n    for (let i = 0; i < aggregates.length; i++) {\n      var aggregate = aggrow.getRowAggregate(row, i);\n      columns.push((\n        <div style={{\n          width: '16px',\n          height: 'inherit',\n          backgroundColor: 'darkGray',\n          flexShrink: '0'\n        }}></div>\n      ));\n      columns.push((\n        <div style={{\n          width: '128px',\n          textAlign: 'right',\n          flexShrink: '0'\n        }}>\n          {aggregate}\n        </div>\n      ));\n    }\n    columns.push((\n      <div style={{\n        width: '16px',\n        height: 'inherit',\n        backgroundColor: 'gold',\n        flexShrink: '0'\n      }}></div>\n    ));\n    if (aggrow.canExpand(row)) {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n            flexShrink: '0',\n            width: '12px',\n            textAlign: 'center',\n            border: '1px solid gray',\n          }}\n          onClick={ () => this._expandRow(row) }\n        >+</div>\n      ));\n    } else if (aggrow.canContract(row)) {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n            flexShrink: '0',\n            width: '12px',\n            textAlign: 'center',\n            border: '1px solid gray',\n          }}\n          onClick={ () => this._contractRow(row) }\n        >-</div>\n      ));\n    } else {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n          }}\n        ></div>\n      ));\n    }\n    rowText += aggrow.getRowLabel(row);\n    columns.push((\n      <div style={{\n        flexShrink: '0',\n        whiteSpace: 'nowrap',\n        marginRight: '20px'\n      }}>\n        {rowText}\n      </div>\n    ));\n    return (\n      <div\n        key={row.top}\n        style={{\n          position: 'absolute',\n          height: (rowHeight - 1).toString() + 'px',\n          top: (rowHeight * row.top).toString() + 'px',\n          display: 'flex',\n          flexDirection: 'row',\n          alignItems: 'center',\n          backgroundColor: bg,\n          borderBottom: '1px solid gray',\n        }}\n        onClick={ () => {\n          this.setState({cursor: row.top});\n        }}>\n        {columns}\n      </div>\n    );\n  }\n\n  componentDidMount() {\n    this.keydown = this.keydown.bind(this);\n    document.body.addEventListener('keydown', this.keydown);\n  }\n\n  componentWillUnmount() {\n    document.body.removeEventListener('keydown', this.keydown);\n  }\n}\n"]}
+//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/table.js"],"names":["rowHeight","treeIndent","Draggable","props","id","React","cloneElement","children","draggable","onDragStart","e","dataTransfer","setData","Component","propTypes","PropTypes","element","isRequired","string","DropTarget","thisId","dropAction","onDragOver","preventDefault","onDrop","sourceId","getData","func","ExpanderConfiguration","aggrow","expander","toString","width","height","border","margin","getExpanderName","TableConfiguration","state","expanded","ex","expanderText","config","getExpanders","map","renderExpander","display","flexDirection","overflow","borderLeft","setState","object","TableHeader","aggregators","getActiveAggregators","expanders","getActiveExpanders","headers","i","length","name","getAggregatorName","push","backgroundColor","flexShrink","textAlign","bg","sep","alignItems","borderBottom","Table","viewport","top","cursor","target","Math","floor","scrollTop","clientHeight","ceil","row","newCursor","contract","console","log","expand","_scrollDiv","scrollDiv","getRows","keyCode","_keepCursorInViewport","getHeight","canContract","_contractRow","getRowIndent","indent","canExpand","_expandRow","s","d","startsWith","sIndex","parseInt","substr","dIndex","active","dragged","splice","setActiveAggregators","setActiveExpanders","sExpander","substring","flexGrow","scroll","div","position","renderVirtualizedRows","rows","renderRow","child","columns","rowText","aggregates","parent","aggregate","getRowAggregate","marginLeft","getRowLabel","whiteSpace","marginRight","keydown","bind","document","body","addEventListener","removeEventListener"],"mappings":"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","file":"table.js","sourcesContent":["/**\n * Copyright (c) 2016-present, Facebook, Inc.\n * All rights reserved.\n *\n * This source code is licensed under the BSD-style license found in the\n * LICENSE file in the root directory of this source tree. An additional grant\n * of patent rights can be found in the PATENTS file in the same directory.\n */\n\n'use strict';\n/*eslint no-console-disallow: \"off\"*/\n/*global React:true*/\n\n// TODO:\n// selection and arrow keys for navigating\n\nconst rowHeight = 20;\nconst treeIndent = 16;\n\nclass Draggable extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n  }\n\n  render() {\n    const id = this.props.id;\n    return React.cloneElement(\n      this.props.children,\n      {\n        draggable: 'true',\n        onDragStart: (e) => {\n          e.dataTransfer.setData('text', id);\n        },\n      }\n    );\n  }\n}\nDraggable.propTypes = {\n  children: React.PropTypes.element.isRequired,\n  id: React.PropTypes.string.isRequired,\n};\n\nclass DropTarget extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n  }\n\n  render() {\n    const thisId = this.props.id;\n    const dropAction = this.props.dropAction;\n    return React.cloneElement(\n      this.props.children,\n      {\n        onDragOver: (e) => e.preventDefault(),\n        onDrop: (e) => {\n          const sourceId = e.dataTransfer.getData('text');\n          e.preventDefault();\n          dropAction(sourceId, thisId);\n        },\n      }\n    );\n  }\n}\n\nDropTarget.propTypes = {\n  children: React.PropTypes.element.isRequired,\n  id: React.PropTypes.string.isRequired,\n  dropAction: React.PropTypes.func.isRequired,\n};\n\nclass ExpanderConfiguration extends React.Component {\n  constructor(props) {\n    super(props);\n  }\n  render() {\n    const aggrow = this.props.aggrow;\n    const expander = this.props.expander;\n    return (\n      <Draggable id={'expander:add:' + expander.toString()}>\n        <div\n          style={{\n            width: 'auto',\n            height: '26px',\n            border: '1px solid darkGray',\n            margin: '2px',\n          }}>\n          {aggrow.getExpanderName(expander)}\n        </div>\n      </Draggable>\n    );\n  }\n}\n\nclass TableConfiguration extends React.Component {\n  constructor(props) {\n    super(props);\n    this.state = {\n      expanded: false,\n    };\n  }\n  renderExpander(ex) {\n    return (<ExpanderConfiguration aggrow={this.props.aggrow} expander={ex} />);\n  }\n  render() {\n    const expanderText = this.state.expanded ? '>>' : '<<';\n    const aggrow = this.props.aggrow;\n    let config = [];\n    if (this.state.expanded) {\n      config = aggrow.getExpanders().map((ex) => this.renderExpander(ex));\n    }\n    return (\n      <div style={{\n          width: this.state.expanded ? '512px' : '26px',\n          height: '100%',\n          display: 'flex',\n          flexDirection: 'column',\n          overflow: 'hidden',\n          borderLeft: '2px solid black',\n        }}>\n        <div style={{\n            width: '100%',\n            height: '26px',\n            border: '1px solid darkGray',\n          }}\n          onClick={ (e) => this.setState({expanded: !this.state.expanded}) }>\n          { expanderText }\n        </div>\n        { config }\n      </div>\n    );\n  }\n}\n\nTableConfiguration.propTypes = {\n  aggrow: React.PropTypes.object.isRequired,\n};\n\nclass TableHeader extends React.Component {\n  constructor(props) {\n    super(props);\n  }\n  render() {\n    const aggrow = this.props.aggrow;\n    const aggregators = aggrow.getActiveAggregators();\n    const expanders = aggrow.getActiveExpanders();\n    const headers = [];\n    for (let i = 0; i < aggregators.length; i++) {\n      const name = aggrow.getAggregatorName(aggregators[i]);\n      headers.push((\n        <DropTarget\n          id={'aggregate:insert:' + i.toString()}\n          dropAction={this.props.dropAction}\n        >\n          <div style={{\n            width: '16px',\n            height: 'inherit',\n            backgroundColor: 'darkGray',\n            flexShrink: '0' }}\n          ></div>\n        </DropTarget>));\n      headers.push((<Draggable id={'aggregate:active:' + i.toString()}>\n          <div style={{ width: '128px', textAlign: 'center', flexShrink: '0' }}>{name}</div>\n        </Draggable>));\n    }\n    headers.push((\n      <DropTarget\n        id=\"divider:insert\"\n        dropAction={this.props.dropAction}\n      >\n        <div style={{\n          width: '16px',\n          height: 'inherit',\n          backgroundColor: 'gold',\n          flexShrink: '0'\n        }}></div>\n      </DropTarget>));\n    for (let i = 0; i < expanders.length; i++) {\n      const name = aggrow.getExpanderName(expanders[i]);\n      const bg = (i % 2 === 0) ? 'white' : 'lightGray';\n      headers.push((<Draggable id={'expander:active:' + i.toString()}>\n          <div style={{\n            width: '128px',\n            textAlign: 'center',\n            backgroundColor: bg,\n            flexShrink: '0'\n          }}>\n            {name}\n          </div>\n        </Draggable>));\n      const sep = i + 1 < expanders.length ? '->' : '...';\n      headers.push((\n        <DropTarget\n          id={'expander:insert:' + (i + 1).toString()}\n          dropAction={this.props.dropAction}\n        >\n          <div style={{\n            height: 'inherit',\n            backgroundColor: 'darkGray',\n            flexShrink: '0'\n          }}>\n            {sep}\n          </div>\n        </DropTarget>)\n      );\n    }\n    return (\n      <div style={{\n        width: '100%',\n        height: '26px',\n        display: 'flex',\n        flexDirection: 'row',\n        alignItems: 'center',\n        borderBottom: '2px solid black',\n      }}>\n        {headers}\n      </div>\n    );\n  }\n}\n\nTableHeader.propTypes = {\n  aggrow: React.PropTypes.object.isRequired,\n  dropAction: React.PropTypes.func.isRequired,\n};\n\nclass Table extends React.Component { // eslint-disable-line no-unused-vars\n  constructor(props) {\n    super(props);\n    this.state = {\n      aggrow: props.aggrow,\n      viewport: { top: 0, height: 100 },\n      cursor: 0,\n    };\n  }\n\n  scroll(e) {\n    const viewport = e.target;\n    const top = Math.floor((viewport.scrollTop - viewport.clientHeight * 1.0) / rowHeight);\n    const height = Math.ceil(viewport.clientHeight * 3.0 / rowHeight);\n    if (top !== this.state.viewport.top || height !== this.state.viewport.height) {\n      this.setState({viewport: {top, height}});\n    }\n  }\n\n  _contractRow(row) {\n    let newCursor = this.state.cursor;\n    if (newCursor > row.top && newCursor < row.top + row.height) { // in contracted section\n      newCursor = row.top;\n    } else if (newCursor >= row.top + row.height) { // below contracted section\n      newCursor -= row.height - 1;\n    }\n    this.state.aggrow.contract(row);\n    this.setState({cursor: newCursor});\n    console.log('-' + row.top);\n  }\n\n  _expandRow(row) {\n    let newCursor = this.state.cursor;\n    this.state.aggrow.expand(row);\n    if (newCursor > row.top) {  // below expanded section\n      newCursor += row.height - 1;\n    }\n    this.setState({cursor: newCursor});\n    console.log('+' + row.top);\n  }\n\n  _scrollDiv: null;\n\n  _keepCursorInViewport() {\n    if (this._scrollDiv) {\n      const cursor = this.state.cursor;\n      const scrollDiv = this._scrollDiv;\n      if (cursor * rowHeight < scrollDiv.scrollTop + scrollDiv.clientHeight * 0.1) {\n        scrollDiv.scrollTop = cursor * rowHeight - scrollDiv.clientHeight * 0.1;\n      } else if ((cursor + 1) * rowHeight > scrollDiv.scrollTop + scrollDiv.clientHeight * 0.9) {\n        scrollDiv.scrollTop = (cursor + 1) * rowHeight - scrollDiv.clientHeight * 0.9;\n      }\n    }\n  }\n\n  keydown(e) {\n    const aggrow = this.state.aggrow;\n    let cursor = this.state.cursor;\n    let row = aggrow.getRows(cursor, 1)[0];\n    switch (e.keyCode) {\n      case 38: // up\n        if (cursor > 0) {\n          this.setState({cursor: cursor - 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 40: // down\n        if (cursor < aggrow.getHeight() - 1) {\n          this.setState({cursor: cursor + 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 37: // left\n        if (aggrow.canContract(row)) {\n          this._contractRow(row);\n        } else if (aggrow.getRowIndent(row) > 0) {\n          const indent = aggrow.getRowIndent(row) - 1;\n          while (aggrow.getRowIndent(row) > indent) {\n            cursor--;\n            row = aggrow.getRows(cursor, 1)[0];\n          }\n          this.setState({cursor: cursor});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n      case 39: // right\n        if (aggrow.canExpand(row)) {\n          this._expandRow(row);\n        } else if (cursor < aggrow.getHeight() - 1) {\n          this.setState({cursor: cursor + 1});\n          this._keepCursorInViewport();\n        }\n        e.preventDefault();\n        break;\n    }\n  }\n\n  dropAction(s, d) {\n    const aggrow = this.state.aggrow;\n    console.log('dropped ' + s + ' to ' + d);\n    if (s.startsWith('aggregate:active:')) {\n      const sIndex = parseInt(s.substr(17), 10);\n      let dIndex = -1;\n      const active = aggrow.getActiveAggregators();\n      const dragged = active[sIndex];\n      if (d.startsWith('aggregate:insert:')) {\n        dIndex = parseInt(d.substr(17), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = active.length;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      if (dIndex > sIndex) {\n        dIndex--;\n      }\n      active.splice(sIndex, 1);\n      active.splice(dIndex, 0, dragged);\n      aggrow.setActiveAggregators(active);\n      this.setState({cursor:0});\n    } else if (s.startsWith('expander:active:')) {\n      const sIndex = parseInt(s.substr(16), 10);\n      let dIndex = -1;\n      const active = aggrow.getActiveExpanders();\n      const dragged = active[sIndex];\n      if (d.startsWith('expander:insert:')) {\n        dIndex = parseInt(d.substr(16), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = 0;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      if (dIndex > sIndex) {\n        dIndex--;\n      }\n      active.splice(sIndex, 1);\n      active.splice(dIndex, 0, dragged);\n      aggrow.setActiveExpanders(active);\n      this.setState({cursor:0});\n    } else if (s.startsWith('expander:add:')) {\n      let dIndex = -1;\n      const sExpander = parseInt(s.substring(13), 10);\n      if (d.startsWith('expander:insert:')) {\n        dIndex = parseInt(d.substr(16), 10);\n      } else if (d === 'divider:insert') {\n        dIndex = 0;\n      } else {\n        throw 'not allowed to drag ' + s + ' to ' + d;\n      }\n      const active = aggrow.getActiveExpanders();\n      active.splice(dIndex, 0, sExpander);\n      aggrow.setActiveExpanders(active);\n      this.setState({cursor:0});\n    }\n  }\n\n  render() {\n    return (\n      <div style={{ width: '100%', height: '100%', display: 'flex', flexDirection: 'row' }}>\n        <div style={{ width: '100%', height: '100%', display: 'flex', flexDirection: 'column', overflow: 'hidden', }}>\n          <TableHeader aggrow={this.state.aggrow} dropAction={(s, d) => this.dropAction(s, d)} />\n          <div\n            style={{\n              width: '100%',\n              flexGrow: '1',\n              overflow: 'scroll'\n            }}\n            onScroll={ (e) => this.scroll(e) }\n            ref={(div) => { this._scrollDiv = div; } }>\n            <div style={{ position: 'relative' }}>\n              { this.renderVirtualizedRows() }\n            </div>\n          </div>\n        </div>\n        <TableConfiguration aggrow={this.state.aggrow} />\n      </div>\n    );\n  }\n\n  renderVirtualizedRows() {\n    const aggrow = this.state.aggrow;\n    const viewport = this.state.viewport;\n    const rows = aggrow.getRows(viewport.top, viewport.height);\n    return (\n      <div style={{\n        position: 'absolute',\n        width: '100%',\n        height: (rowHeight * (aggrow.getHeight() + 20)).toString() + 'px'\n      }}>\n        { rows.map(child => this.renderRow(child)) }\n      </div>\n    );\n  }\n\n  renderRow(row) {\n    if (row === null) {\n      return null;\n    }\n    let bg = 'lightGray';\n    const aggrow = this.state.aggrow;\n    const columns = [];\n    let rowText = '';\n    const indent = 4 + aggrow.getRowIndent(row) * treeIndent;\n    const aggregates = aggrow.getActiveAggregators();\n    if (row.parent !== null && (row.parent.expander % 2 === 0)) {\n      bg = 'white';\n    }\n    if (row.top === this.state.cursor) {\n      bg = 'lightblue';\n    }\n    for (let i = 0; i < aggregates.length; i++) {\n      var aggregate = aggrow.getRowAggregate(row, i);\n      columns.push((\n        <div style={{\n          width: '16px',\n          height: 'inherit',\n          backgroundColor: 'darkGray',\n          flexShrink: '0'\n        }}></div>\n      ));\n      columns.push((\n        <div style={{\n          width: '128px',\n          textAlign: 'right',\n          flexShrink: '0'\n        }}>\n          {aggregate}\n        </div>\n      ));\n    }\n    columns.push((\n      <div style={{\n        width: '16px',\n        height: 'inherit',\n        backgroundColor: 'gold',\n        flexShrink: '0'\n      }}></div>\n    ));\n    if (aggrow.canExpand(row)) {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n            flexShrink: '0',\n            width: '12px',\n            textAlign: 'center',\n            border: '1px solid gray',\n          }}\n          onClick={ () => this._expandRow(row) }\n        >+</div>\n      ));\n    } else if (aggrow.canContract(row)) {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n            flexShrink: '0',\n            width: '12px',\n            textAlign: 'center',\n            border: '1px solid gray',\n          }}\n          onClick={ () => this._contractRow(row) }\n        >-</div>\n      ));\n    } else {\n      columns.push((\n        <div\n          style={{\n            marginLeft: indent.toString() + 'px',\n          }}\n        ></div>\n      ));\n    }\n    rowText += aggrow.getRowLabel(row);\n    columns.push((\n      <div style={{\n        flexShrink: '0',\n        whiteSpace: 'nowrap',\n        marginRight: '20px'\n      }}>\n        {rowText}\n      </div>\n    ));\n    return (\n      <div\n        key={row.top}\n        style={{\n          position: 'absolute',\n          height: (rowHeight - 1).toString() + 'px',\n          top: (rowHeight * row.top).toString() + 'px',\n          display: 'flex',\n          flexDirection: 'row',\n          alignItems: 'center',\n          backgroundColor: bg,\n          borderBottom: '1px solid gray',\n        }}\n        onClick={ () => {\n          this.setState({cursor: row.top});\n        }}>\n        {columns}\n      </div>\n    );\n  }\n\n  componentDidMount() {\n    this.keydown = this.keydown.bind(this);\n    document.body.addEventListener('keydown', this.keydown);\n  }\n\n  componentWillUnmount() {\n    document.body.removeEventListener('keydown', this.keydown);\n  }\n}\n"]}
// @generated
diff --git a/local-cli/server/middleware/heapCapture/src/aggrow.js b/local-cli/server/middleware/heapCapture/src/aggrow.js
index a6c51df55..4be1edeae 100644
--- a/local-cli/server/middleware/heapCapture/src/aggrow.js
+++ b/local-cli/server/middleware/heapCapture/src/aggrow.js
@@ -16,7 +16,7 @@
// pivot around frames in the middle of a stack by callers / callees
// graphing?
-function stringInterner() { // eslint-disable-line no-unused-vars
+function StringInterner() { // eslint-disable-line no-unused-vars
const strings = [];
const ids = {};
return {
@@ -37,20 +37,16 @@ function stringInterner() { // eslint-disable-line no-unused-vars
};
}
-function stackData(stackIdMap, maxDepth) { // eslint-disable-line no-unused-vars
- return {
- maxDepth: maxDepth,
- get: function getStack(id) {
- return stackIdMap[id];
- },
- };
-}
-
-function stackRegistry() { // eslint-disable-line no-unused-vars
+function StackRegistry() { // eslint-disable-line no-unused-vars
return {
root: { id: 0 },
nodeCount: 1,
+ maxDepth: -1,
+ stackIdMap: null,
insert: function insertNode(parent, frameId) {
+ if (this.stackIdMap !== null) {
+ throw 'stacks already flattened';
+ }
let node = parent[frameId];
if (node === undefined) {
node = { id: this.nodeCount };
@@ -59,7 +55,13 @@ function stackRegistry() { // eslint-disable-line no-unused-vars
}
return node;
},
+ get: function getStackArray(id) {
+ return this.stackIdMap[id];
+ },
flatten: function flattenStacks() {
+ if (this.stackIdMap !== null) {
+ return;
+ }
let stackFrameCount = 0;
function countStacks(tree, depth) {
let leaf = true;
@@ -109,13 +111,150 @@ function stackRegistry() { // eslint-disable-line no-unused-vars
return stackIdMap[id];
}
flattenStacksImpl(this.root, []);
-
- return new stackData(stackIdMap, maxStackDepth);
+ this.root = null;
+ this.stackIdMap = stackIdMap;
+ this.maxDepth = maxStackDepth;
},
};
}
-function aggrow(numRows) { // eslint-disable-line no-unused-vars
+function AggrowData(columns) { // eslint-disable-line no-unused-vars
+ const columnCount = columns.length;
+ const columnConverter = columns.map(c => {
+ switch (c.type) {
+ case 'int': // stores raw value
+ return (i) => i;
+ case 'string': // stores interned id of string
+ return (s) => c.strings.intern(s);
+ case 'stack': // stores id of stack node
+ return (s) => s.id;
+ default:
+ throw 'unknown AggrowData column type';
+ }
+ });
+ return {
+ data: new Int32Array(0),
+ columns: columns,
+ rowCount: 0,
+ rowInserter: function rowInserter(numRows) {
+ console.log(
+ 'increasing row data from ' + (this.data.length * 4).toLocaleString() + ' B to ' +
+ (this.data.length * 4 + numRows * columnCount * 4).toLocaleString() + ' B'
+ );
+ const newData = new Int32Array(this.data.length + numRows * columnCount);
+ newData.set(this.data);
+ let currOffset = this.data.length;
+ const endOffset = newData.length;
+ this.data = newData;
+ this.rowCount = newData.length / columnCount;
+ return {
+ insertRow: function insertRow() {
+ if (currOffset >= endOffset) {
+ throw 'tried to insert data off end of added range';
+ }
+ if (arguments.length !== columnCount) {
+ throw 'expected data for ' + columnCount.toString() + ' columns, got' +
+ arguments.length.toString() + ' columns';
+ }
+ for (let i = 0; i < arguments.length; i++) {
+ newData[currOffset + i] = columnConverter[i](arguments[i]);
+ }
+ currOffset += columnCount;
+ },
+ done: function done() {
+ if (currOffset !== endOffset) {
+ throw 'unfilled rows';
+ }
+ },
+ };
+ },
+ };
+}
+
+function Aggrow(aggrowData) {
+ const columns = aggrowData.columns;
+ const columnCount = columns.length;
+ const data = aggrowData.data;
+ function columnIndex(columnName, columnType) {
+ const index = columns.findIndex(c => c.name === columnName && c.type === columnType);
+ if (index < 0) {
+ throw 'did not find data column ' + columnName + ' with type ' + columnType;
+ }
+ return index;
+ }
+ for (let i = 0; i < columns.length; i++) {
+ if (columns[i].type === 'stack') {
+ columns[i].stacks.flatten();
+ }
+ }
+ return {
+ expander: new AggrowExpander(aggrowData.rowCount),
+ addSumAggregator: function addSumAggregator(aggregatorName, columnName) {
+ const index = columnIndex(columnName, 'int');
+ return this.expander.addAggregator(
+ aggregatorName,
+ function aggregateSize(indices) {
+ let size = 0;
+ for (let i = 0; i < indices.length; i++) {
+ const row = indices[i];
+ size += data[row * columnCount + index];
+ }
+ return size;
+ },
+ (value) => value.toLocaleString(),
+ (a, b) => b - a,
+ );
+ },
+ addCountAggregator: function addCountAggregator(aggregatorName) {
+ return this.expander.addAggregator(
+ aggregatorName,
+ function aggregateCount(indices) {
+ return indices.length;
+ },
+ (value) => value.toLocaleString(),
+ (a, b) => b - a,
+ );
+ },
+ addStringExpander: function addStringExpander(expanderName, columnName) {
+ const index = columnIndex(columnName, 'string');
+ const strings = columns[index].strings;
+ return this.expander.addFieldExpander(
+ expanderName,
+ (row) => strings.get(data[row * columnCount + index]),
+ (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],
+ );
+ },
+ addNumberExpander: function addNumberExpander(expanderName, columnName) {
+ const index = columnIndex(columnName, 'int');
+ return this.expander.addFieldExpander(
+ expanderName,
+ (row) => data[row * columnCount + index].toLocaleString(),
+ (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],
+ );
+ },
+ addPointerExpander: function addPointerExpander(expanderName, columnName) {
+ const index = columnIndex(columnName, 'int');
+ return this.expander.addFieldExpander(
+ expanderName,
+ (row) => '0x' + (data[row * columnCount + index] >>> 0).toString(),
+ (rowA, rowB) => data[rowA * columnCount + index] - data[rowB * columnCount + index],
+ );
+ },
+ addStackExpander: function addStackExpander(expanderName, columnName, formatter) {
+ // TODO: options for caller/callee, pivoting
+ const index = columnIndex(columnName, 'stack');
+ const stacks = columns[index].stacks;
+ return this.expander.addCalleeStackExpander(
+ expanderName,
+ stacks.maxDepth,
+ (row) => stacks.get(data[row * columnCount + index]),
+ formatter,
+ );
+ },
+ };
+}
+
+function AggrowExpander(numRows) { // eslint-disable-line no-unused-vars
// expander ID definitions
const FIELD_EXPANDER_ID_MIN = 0x0000;
const FIELD_EXPANDER_ID_MAX = 0x7fff;
diff --git a/local-cli/server/middleware/heapCapture/src/heapCapture.js b/local-cli/server/middleware/heapCapture/src/heapCapture.js
index 7af43b379..4d4099b19 100644
--- a/local-cli/server/middleware/heapCapture/src/heapCapture.js
+++ b/local-cli/server/middleware/heapCapture/src/heapCapture.js
@@ -8,7 +8,7 @@
*/
'use strict';
/*eslint no-console-disallow: "off"*/
-/*global React ReactDOM Table stringInterner stackRegistry aggrow preLoadedCapture:true*/
+/*global React ReactDOM Table StringInterner StackRegistry AggrowData Aggrow preLoadedCapture:true*/
function RefVisitor(refs, id) {
this.refs = refs;
@@ -292,189 +292,88 @@ function registerPathToRoot(refs, registry, strings) {
}
}
-function captureRegistry() {
- const strings = stringInterner();
- const stacks = stackRegistry(strings);
- const data = new Int32Array(0);
-
- const idField = 0;
- const typeField = 1;
- const sizeField = 2;
- const traceField = 3;
- const pathField = 4;
- const reactField = 5;
- const valueField = 6;
- const moduleField = 7;
- const numFields = 8;
-
- return {
- strings: strings,
- stacks: stacks,
- data: data,
- register: function registerCapture(captureId, capture) {
- // NB: capture.refs is potentially VERY large, so we try to avoid making
- // copies, even of iteration is a bit more annoying.
- let rowCount = 0;
- for (const id in capture.refs) { // eslint-disable-line no-unused-vars
- rowCount++;
- }
- for (const id in capture.markedBlocks) { // eslint-disable-line no-unused-vars
- rowCount++;
- }
- console.log(
- 'increasing row data from ' + (this.data.length * 4).toString() + 'B to ' +
- (this.data.length * 4 + rowCount * numFields * 4).toString() + 'B'
- );
- const newData = new Int32Array(this.data.length + rowCount * numFields);
- newData.set(data);
- let dataOffset = this.data.length;
- this.data = null;
-
- registerPathToRoot(capture.refs, this.stacks, this.strings);
- const internedCaptureId = this.strings.intern(captureId);
- const noneString = this.strings.intern('#none');
- const noneStack = this.stacks.insert(this.stacks.root, noneString);
- forEachRef(capture.refs, (visitor) => {
- const ref = visitor.getRef();
- const id = visitor.id;
- newData[dataOffset + idField] = parseInt(id, 16);
- newData[dataOffset + typeField] = this.strings.intern(ref.type);
- newData[dataOffset + sizeField] = ref.size;
- newData[dataOffset + traceField] = internedCaptureId;
- if (ref.rootPath === undefined) {
- newData[dataOffset + pathField] = noneStack.id;
- } else {
- newData[dataOffset + pathField] = ref.rootPath.id;
- }
- if (ref.reactTree === undefined) {
- newData[dataOffset + reactField] = noneStack.id;
- } else {
- newData[dataOffset + reactField] = ref.reactTree.id;
- }
- newData[dataOffset + valueField] = this.strings.intern(visitor.getValue());
- if (ref.module) {
- newData[dataOffset + moduleField] = this.strings.intern(ref.module);
- } else {
- newData[dataOffset + moduleField] = noneString;
- }
- dataOffset += numFields;
- });
- for (const id in capture.markedBlocks) {
- const block = capture.markedBlocks[id];
- newData[dataOffset + idField] = parseInt(id, 16);
- newData[dataOffset + typeField] = this.strings.intern('Marked Block Overhead');
- newData[dataOffset + sizeField] = block.capacity - block.size;
- newData[dataOffset + traceField] = internedCaptureId;
- newData[dataOffset + pathField] = noneStack.id;
- newData[dataOffset + reactField] = noneStack.id;
- newData[dataOffset + valueField] = this.strings.intern(
- 'capacity: ' + block.capacity +
- ', size: ' + block.size +
- ', granularity: ' + block.cellSize
- );
- newData[dataOffset + moduleField] = noneString;
- dataOffset += numFields;
- }
- this.data = newData;
- },
- getAggrow: function getAggrow() {
- const agStrings = this.strings;
- const agStacks = this.stacks.flatten();
- const agData = this.data;
- const agNumRows = agData.length / numFields;
- const ag = new aggrow(agNumRows);
-
- ag.addFieldExpander('Id',
- function getId(row) {
- let id = agData[row * numFields + idField];
- if (id < 0) {
- id += 0x100000000; // data is int32, id is uint32
- }
- return '0x' + id.toString(16);
- },
- function compareAddress(rowA, rowB) {
- return agData[rowA * numFields + idField] - agData[rowB * numFields + idField];
- });
-
- const typeExpander = ag.addFieldExpander('Type',
- function getType(row) { return agStrings.get(agData[row * numFields + typeField]); },
- function compareType(rowA, rowB) {
- return agData[rowA * numFields + typeField] - agData[rowB * numFields + typeField];
- });
-
- ag.addFieldExpander('Size',
- function getSize(row) { return agData[row * numFields + sizeField].toString(); },
- function compareSize(rowA, rowB) {
- return agData[rowA * numFields + sizeField] - agData[rowB * numFields + sizeField];
- });
-
- ag.addFieldExpander('Trace',
- function getSize(row) { return agStrings.get(agData[row * numFields + traceField]); },
- function compareSize(rowA, rowB) {
- return agData[rowA * numFields + traceField] - agData[rowB * numFields + traceField];
- });
-
- const pathExpander = ag.addCalleeStackExpander(
- 'Path',
- agStacks.maxDepth,
- function getStack(row) { return agStacks.get(agData[row * numFields + pathField]); },
- function getFrame(id) { return agStrings.get(id); },
- );
-
- const reactExpander = ag.addCalleeStackExpander(
- 'React Tree',
- agStacks.maxDepth,
- function getStack(row) { return agStacks.get(agData[row * numFields + reactField]); },
- function getFrame(id) { return agStrings.get(id); },
- );
-
- const valueExpander = ag.addFieldExpander('Value',
- function getValue(row) { return agStrings.get(agData[row * numFields + valueField]); },
- function compareValue(rowA, rowB) {
- return agData[rowA * numFields + valueField] - agData[rowB * numFields + valueField];
- });
-
- const moduleExpander = ag.addFieldExpander('Module',
- function getModule(row) { return agStrings.get(agData[row * numFields + moduleField]); },
- function compareModule(rowA, rowB) {
- return agData[rowA * numFields + moduleField] - agData[rowB * numFields + moduleField];
- });
-
- const sizeAggregator = ag.addAggregator('Size',
- function aggregateSize(indices) {
- let size = 0;
- for (let i = 0; i < indices.length; i++) {
- const row = indices[i];
- size += agData[row * numFields + sizeField];
- }
- return size;
- },
- function formatSize(value) { return value.toString(); },
- function sortSize(a, b) { return b - a; } );
-
- const countAggregator = ag.addAggregator('Count',
- function aggregateCount(indices) {
- return indices.length;
- },
- function formatCount(value) { return value.toString(); },
- function sortCount(a, b) { return b - a; } );
-
- ag.setActiveExpanders([
- pathExpander,
- reactExpander,
- moduleExpander,
- typeExpander,
- valueExpander,
- ]);
- ag.setActiveAggregators([sizeAggregator, countAggregator]);
- return ag;
- },
- };
+function registerCapture(data, captureId, capture, stacks, strings) {
+ // NB: capture.refs is potentially VERY large, so we try to avoid making
+ // copies, even if iteration is a bit more annoying.
+ let rowCount = 0;
+ for (const id in capture.refs) { // eslint-disable-line no-unused-vars
+ rowCount++;
+ }
+ for (const id in capture.markedBlocks) { // eslint-disable-line no-unused-vars
+ rowCount++;
+ }
+ const inserter = data.rowInserter(rowCount);
+ registerPathToRoot(capture.refs, stacks, strings);
+ const noneString = strings.intern('#none');
+ const noneStack = stacks.insert(stacks.root, noneString);
+ forEachRef(capture.refs, (visitor) => {
+ // want to data.append(value, value, value), not IDs
+ const ref = visitor.getRef();
+ const id = visitor.id;
+ inserter.insertRow(
+ parseInt(id, 16),
+ ref.type,
+ ref.size,
+ captureId,
+ ref.rootPath === undefined ? noneStack : ref.rootPath,
+ ref.reactTree === undefined ? noneStack : ref.reactTree,
+ visitor.getValue(),
+ ref.module === undefined ? '#none' : ref.module,
+ );
+ });
+ for (const id in capture.markedBlocks) {
+ const block = capture.markedBlocks[id];
+ inserter.insertRow(
+ parseInt(id, 16),
+ 'Marked Block Overhead',
+ block.capacity - block.size,
+ captureId,
+ noneStack,
+ noneStack,
+ 'capacity: ' + block.capacity + ', size: ' + block.size + ', granularity: ' + block.cellSize,
+ '#none',
+ );
+ }
+ inserter.done();
}
if (preLoadedCapture) {
- const r = new captureRegistry();
- r.register('trace', preLoadedCapture);
+ const strings = StringInterner();
+ const stacks = new StackRegistry();
+ const columns = [
+ { name: 'id', type: 'int' },
+ { name: 'type', type: 'string', strings: strings },
+ { name: 'size', type: 'int' },
+ { name: 'trace', type: 'string', strings: strings },
+ { name: 'path', type: 'stack', stacks: stacks },
+ { name: 'react', type: 'stack', stacks: stacks },
+ { name: 'value', type: 'string', strings: strings },
+ { name: 'module', type: 'string', strings: strings },
+ ];
+ const data = new AggrowData(columns);
+ registerCapture(data, 'trace', preLoadedCapture, stacks, strings);
preLoadedCapture = undefined; // let GG clean up the capture
- ReactDOM.render(, document.body);
+ const aggrow = new Aggrow(data);
+ aggrow.addPointerExpander('Id', 'id');
+ const typeExpander = aggrow.addStringExpander('Type', 'type');
+ aggrow.addNumberExpander('Size', 'size');
+ aggrow.addStringExpander('Trace', 'trace');
+ const pathExpander = aggrow.addStackExpander('Path', 'path', strings.get);
+ const reactExpander = aggrow.addStackExpander('React Tree', 'react', strings.get);
+ const valueExpander = aggrow.addStringExpander('Value', 'value');
+ const moduleExpander = aggrow.addStringExpander('Module', 'module');
+ aggrow.expander.setActiveExpanders([
+ pathExpander,
+ reactExpander,
+ moduleExpander,
+ typeExpander,
+ valueExpander,
+ ]);
+ const sizeAggregator = aggrow.addSumAggregator('Size', 'size');
+ const countAggregator = aggrow.addCountAggregator('Count');
+ aggrow.expander.setActiveAggregators([
+ sizeAggregator,
+ countAggregator,
+ ]);
+ ReactDOM.render(, document.body);
}
diff --git a/local-cli/server/middleware/heapCapture/src/table.js b/local-cli/server/middleware/heapCapture/src/table.js
index 9820e1b05..f8c0921e7 100644
--- a/local-cli/server/middleware/heapCapture/src/table.js
+++ b/local-cli/server/middleware/heapCapture/src/table.js
@@ -24,12 +24,14 @@ class Draggable extends React.Component { // eslint-disable-line no-unused-vars
render() {
const id = this.props.id;
- function dragStart(e) {
- e.dataTransfer.setData('text/plain', id);
- }
return React.cloneElement(
this.props.children,
- { draggable: 'true', onDragStart: dragStart }
+ {
+ draggable: 'true',
+ onDragStart: (e) => {
+ e.dataTransfer.setData('text', id);
+ },
+ }
);
}
}
@@ -45,23 +47,15 @@ class DropTarget extends React.Component { // eslint-disable-line no-unused-vars
render() {
const thisId = this.props.id;
- const dropFilter = this.props.dropFilter;
const dropAction = this.props.dropAction;
return React.cloneElement(
this.props.children,
{
- onDragOver: (e) => {
- const sourceId = e.dataTransfer.getData('text/plain');
- if (dropFilter(sourceId)) {
- e.preventDefault();
- }
- },
+ onDragOver: (e) => e.preventDefault(),
onDrop: (e) => {
- const sourceId = e.dataTransfer.getData('text/plain');
- if (dropFilter(sourceId)) {
- e.preventDefault();
- dropAction(sourceId, thisId);
- }
+ const sourceId = e.dataTransfer.getData('text');
+ e.preventDefault();
+ dropAction(sourceId, thisId);
},
}
);
@@ -71,7 +65,6 @@ class DropTarget extends React.Component { // eslint-disable-line no-unused-vars
DropTarget.propTypes = {
children: React.PropTypes.element.isRequired,
id: React.PropTypes.string.isRequired,
- dropFilter: React.PropTypes.func.isRequired,
dropAction: React.PropTypes.func.isRequired,
};
@@ -156,7 +149,6 @@ class TableHeader extends React.Component {
headers.push((
s.startsWith('aggregate')}
dropAction={this.props.dropAction}
>
s.startsWith('aggregate') || s.startsWith('expander')}
dropAction={this.props.dropAction}
>
{return true; }}
dropAction={this.props.dropAction}
>