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} >