From ca90eae4554f445c5781eae552b81bf54efb725f Mon Sep 17 00:00:00 2001 From: Jonathan Eiten Date: Mon, 1 Feb 2016 06:17:55 -0500 Subject: [PATCH] rebuilt examples/v0.2 with sprint 16, parts 1 and 2 --- examples/v0.2/index.html | 1321 +-- examples/v0.2/index.js | 20388 ++++++++++++++++++----------------- examples/v0.2/index.min.js | 18 +- 3 files changed, 11170 insertions(+), 10557 deletions(-) diff --git a/examples/v0.2/index.html b/examples/v0.2/index.html index ab8f5ae68..5904d10d4 100644 --- a/examples/v0.2/index.html +++ b/examples/v0.2/index.html @@ -1,4 +1,4 @@ - + fin-hypergrid Demo @@ -49,22 +49,36 @@ border: 1px solid rgba(132, 132, 132, 1); bottom: 3px; } + div#dashboard > label { + font-weight: normal; + } + div#dashboard > button, .toggle-property { + margin: .25em 1em .25em 0; + display: inline-block; + white-space: nowrap; + } + div#dashboard > span.section { + margin-right: 1em; + } + div#dashboard > span.section:after { + content: ':'; + }
-
+
+ - - + - +
@@ -80,715 +94,734 @@
diff --git a/examples/v0.2/index.js b/examples/v0.2/index.js index ea6ddd627..3fc72573c 100644 --- a/examples/v0.2/index.js +++ b/examples/v0.2/index.js @@ -1,5 +1,61 @@ (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }', + 'div.dragon-list > ul { top: 46px; }', + 'div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {', + ' content: \'\\2b24\';', // BLACK LARGE CIRCLE + ' color: #b6b6b6;', + ' font-size: 30px;', + ' margin: 8px 14px 8px 8px; }', + 'li.dragon-pop { opacity:.8; }' + ] +}; + +function addStylesheet(key, referenceElement) { + cssInjector(stylesheets[key], key, referenceElement); +} + +module.exports = addStylesheet; + +},{"css-injector":4}],2:[function(require,module,exports){ +module.exports = { // This file generated by gulp-imagine-64 at 6:03:02 AM on 2/1/2016 "calendar": { type: "image/png", data: "iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAAc0lEQVR4nIXQwQkCMRSE4U9ZLMCT9Xjaq2AfNhfYU5oQLMAOtoN48EWei5iBIRPe/yYQ3qrhf1lFG7iKcEaJxSfukUvMWgdHavt0uWHtg2QwxXnAnJZ2uOLyVZtybzzhgWNmfoFl0/YB87NbzR1cjP9xeQHSDC6mcL1xFQAAAABJRU5ErkJggg==" @@ -34,7 +90,7 @@ module.exports = { // This file generated by gulp-imagine-64 at 1:11:59 PM on 1/ }, }; -},{}],2:[function(require,module,exports){ +},{}],3:[function(require,module,exports){ /* eslint-env browser */ 'use strict'; @@ -59,7 +115,7 @@ images.filter = function(state) { module.exports = images; -},{"./images":1,"object-iterators":18}],3:[function(require,module,exports){ +},{"./images":2,"object-iterators":21}],4:[function(require,module,exports){ 'use strict'; /* eslint-env browser */ @@ -140,7 +196,7 @@ cssInjector.idPrefix = 'injected-stylesheet-'; // Interface module.exports = cssInjector; -},{}],4:[function(require,module,exports){ +},{}],5:[function(require,module,exports){ 'use strict'; /** @namespace extend-me **/ @@ -290,7 +346,7 @@ function initializePrototypeChain() { module.exports = extend; -},{}],5:[function(require,module,exports){ +},{}],6:[function(require,module,exports){ /* eslint-env browser */ // This is the main file, usable as is, such as by /test/index.js. @@ -299,18 +355,14 @@ module.exports = extend; 'use strict'; -var cssInjector = require('css-injector'); +var unstrungify = require('unstrungify'); +var cssInjector = require('./js/css'); var FilterNode = require('./js/FilterNode'); var DefaultFilter = require('./js/FilterLeaf'); var template = require('./js/template'); var operators = require('./js/tree-operators'); -var css; // defined by code inserted by gulpfile between following comments -/* inject:css */ -css = '.filter-tree{font-family:sans-serif;font-size:10pt;line-height:1.5em}.filter-tree label{font-weight:400}.filter-tree input[type=checkbox],.filter-tree input[type=radio]{left:3px;margin-right:3px}.filter-tree ol{margin-top:0}.filter-tree-add,.filter-tree-add-filter,.filter-tree-remove{cursor:pointer}.filter-tree-add,.filter-tree-add-filter{font-style:italic;color:#444;font-size:90%}.filter-tree-add-filter{margin:3px 0 3px 3em;width:120px;display:inline-block}.filter-tree-add-filter:hover,.filter-tree-add:hover{text-decoration:underline}.filter-tree-add-filter.as-menu-header,.filter-tree-add.as-menu-header{background-color:#fff;font-weight:700;font-style:normal}.filter-tree-add-filter.as-menu-header:hover{text-decoration:inherit}.filter-tree-add-filter>div,.filter-tree-add>div,.filter-tree-remove{display:inline-block;width:15px;height:15px;border-radius:8px;background-color:#8c8;font-size:11.5px;font-weight:700;color:#fff;text-align:center;line-height:normal;font-style:normal;font-family:sans-serif;text-shadow:0 0 1.5px grey;margin-right:4px}.filter-tree-add-filter>div:before,.filter-tree-add>div:before{content:\'\\ff0b\'}.filter-tree-remove{background-color:#e88;border:0}.filter-tree-remove:before{content:\'\\2212\'}.filter-tree li::after{font-size:70%;font-style:italic;font-weight:700;color:#080}.filter-tree>ol>li:last-child::after{display:none}.op-or>ol>li::after{margin-left:2.5em;content:\'— OR —\'}.op-and>ol>li::after{margin-left:2.5em;content:\'— AND —\'}.op-nor>ol>li::after{margin-left:2.5em;content:\'— NOR —\'}.filter-tree-default>:enabled{margin:0 .4em;background-color:#ddd;border:0}.filter-tree-default>input[type=text]{width:8em;padding:0 5px}.filter-tree-default>select{border:0}.filter-tree-default>.filter-tree-warning{background-color:#ffc}.filter-tree-default>.filter-tree-error{background-color:#Fcc}.filter-tree .footnotes{font-size:6pt;margin:2px 0 0;line-height:normal;white-space:normal;color:#999}.filter-tree .footnotes>ol{margin:0;padding-left:2em}.filter-tree .footnotes>ol>li{margin:2px 0}.filter-tree .footnotes .field-name,.filter-tree .footnotes .field-value{font-weight:700;color:#777}.filter-tree .footnotes .field-value:after,.filter-tree .footnotes .field-value:before{content:\'\"\'}.filter-tree .footnotes .field-value{font-family:monospace}.filter-tree-chooser{position:absolute;font-size:9pt;outline:0;box-shadow:5px 5px 10px grey}'; -/* endinject */ - var ordinal = 0; var reFilterTreeErrorString = /^filter-tree: /; @@ -347,8 +399,8 @@ var reFilterTreeErrorString = /^filter-tree: /; */ var FilterTree = FilterNode.extend('FilterTree', { - initialize: function(options) { - cssInjector(css, 'filter-tree-base', options && options.cssStylesheetReferenceElement); + preInitialize: function(options) { + cssInjector('filter-tree-base', options && options.cssStylesheetReferenceElement); if (options.editors) { this.editors = options.editors; @@ -376,6 +428,17 @@ var FilterTree = FilterNode.extend('FilterTree', { this.el.addEventListener('click', catchClick.bind(this)); }, + getState: unstrungify, + + getJSON: function() { + var ready = JSON.stringify(this, null, this.JSONspace); + return ready ? ready : ''; + }, + + setJSON: function(json) { + this.setState(JSON.parse(json)); + }, + load: function(state) { if (!state) { var filterEditorNames = Object.keys(this.editors), @@ -389,13 +452,13 @@ var FilterTree = FilterNode.extend('FilterTree', { // Validate `state.operator` if (!(operators[state.operator] || state.operator === undefined && state.children.length === 1)) { - throw this.Error('Expected `operator` property to be one of: ' + Object.keys(operators)); + throw FilterNode.Error('Expected `operator` property to be one of: ' + Object.keys(operators)); } this.operator = state.operator; // Validate `state.children` if (!(state.children instanceof Array && state.children.length)) { - throw this.Error('Expected `children` property to be a non-empty array.'); + throw FilterNode.Error('Expected `children` property to be a non-empty array.'); } this.children = []; var self = this; @@ -487,6 +550,7 @@ var FilterTree = FilterNode.extend('FilterTree', { }, /** + * @param {boolean} [object.rethrow=false] - Catch (do not throw) the error. * @param {boolean} [object.alert=true] - Announce error via window.alert() before returning. * @param {boolean} [object.focus=true] - Place the focus on the offending control and give it error color. * @returns {undefined|string} `undefined` means valid or string containing error message. @@ -496,6 +560,7 @@ var FilterTree = FilterNode.extend('FilterTree', { var focus = options.focus === undefined || options.focus, alert = options.alert === undefined || options.alert, + rethrow = options.rethrow === true, result; try { @@ -504,7 +569,7 @@ var FilterTree = FilterNode.extend('FilterTree', { result = err.message; // Throw when not a filter tree error - if (!reFilterTreeErrorString.test(result)) { + if (rethrow || !reFilterTreeErrorString.test(result)) { throw err; } @@ -519,10 +584,12 @@ var FilterTree = FilterNode.extend('FilterTree', { test: function test(dataRow) { var operator = operators[this.operator], - result = operator.seed; + result = operator.seed, + noChildrenDefined = true; this.children.find(function(child) { if (child) { + noChildrenDefined = false; if (child instanceof DefaultFilter) { result = operator.reduce(result, child.test(dataRow)); } else if (child.children.length) { @@ -534,7 +601,7 @@ var FilterTree = FilterNode.extend('FilterTree', { return false; }); - return operator.negate ? !result : result; + return noChildrenDefined || (operator.negate ? !result : result); }, toJSON: function toJSON() { @@ -548,7 +615,10 @@ var FilterTree = FilterNode.extend('FilterTree', { if (child instanceof DefaultFilter) { state.children.push(child); } else if (child.children.length) { - state.children.push(toJSON.call(child)); + var ready = toJSON.call(child); + if (ready) { + state.children.push(ready); + } } } }); @@ -558,27 +628,29 @@ var FilterTree = FilterNode.extend('FilterTree', { state[key] = metadata[key]; }); - return state; + return state.children.length ? state : undefined; }, - toSQL: function toSQL() { + getSqlWhereClause: function getSqlWhereClause() { var lexeme = operators[this.operator].SQL, - where = lexeme.beg; + where = ''; this.children.forEach(function(child, idx) { var op = idx ? ' ' + lexeme.op + ' ' : ''; if (child) { if (child instanceof DefaultFilter) { - where += op + child.toSQL(); + where += op + child.getSqlWhereClause(); } else if (child.children.length) { - where += op + toSQL.call(child); + where += op + getSqlWhereClause.call(child); } } }); - where += lexeme.end; + if (!where) { + where = 'NULL IS NULL'; + } - return where; + return lexeme.beg + where + lexeme.end; } }); @@ -595,7 +667,7 @@ function throwIfJSON(state) { if (typeof state === 'string') { errMsg += ' See `JSON.parse()`.'; } - throw this.Error(errMsg); + throw FilterNode.Error(errMsg); } } @@ -610,6 +682,10 @@ function catchClick(evt) { // must be called with context handler.call(this, evt); evt.stopPropagation(); } + + if (this.eventHandler) { + this.eventHandler(evt); + } } /** @@ -704,25 +780,25 @@ function detachChooser() { // must be called with context module.exports = FilterTree; -},{"./js/FilterLeaf":6,"./js/FilterNode":7,"./js/template":8,"./js/tree-operators":9,"css-injector":3}],6:[function(require,module,exports){ +},{"./js/FilterLeaf":7,"./js/FilterNode":8,"./js/css":9,"./js/template":11,"./js/tree-operators":12,"unstrungify":26}],7:[function(require,module,exports){ /* eslint-env browser */ /* eslint-disable key-spacing */ 'use strict'; -var regExpLIKE = require('regexp-like').cached; - var FilterNode = require('./FilterNode'); var template = require('./template'); +var operators = require('./leaf-operators'); /** @typedef {object} converter * @property {function} to - Returns input value converted to type. Fails silently. * @property {function} not - Tests input value against type, returning `false if type or `true` if not type. */ -/** @type converter */ +/** @type {converter} */ var numberConverter = { to: Number, not: isNaN }; -/** @type converter */ + +/** @type {converter} */ var dateConverter = { to: function(s) { return new Date(s); }, not: isNaN }; /** @constructor @@ -731,23 +807,19 @@ var dateConverter = { to: function(s) { return new Date(s); }, not: isNaN }; */ var FilterLeaf = FilterNode.extend('FilterLeaf', { - name: 'Column ? Literal', + name: 'column : value', - operators: { - '<' : { test: function(a, b) { return a < b; } }, - '\u2264' : { test: function(a, b) { return a <= b; }, SQL: '<=' }, - '=' : { test: function(a, b) { return a === b; } }, - '\u2265' : { test: function(a, b) { return a >= b; }, SQL: '>=' }, - '>' : { test: function(a, b) { return a > b; } }, - '\u2260' : { test: function(a, b) { return a !== b; }, SQL: '<>' }, - LIKE : { test: function(a, b) { return regExpLIKE(b).test(a); } }, - 'NOT LIKE': { test: function(a, b) { return !regExpLIKE(b).test(a); } } + preInitialize: function() { + this.onChange = cleanUpAndMoveOn.bind(this); }, + operators: operators, + operatorsOptions: operators.options, + destroy: function() { if (this.controls) { for (var key in this.controls) { - this.controls[key].removeEventListener('change', cleanUpAndMoveOn); + this.controls[key].removeEventListener('change', this.onChange); } } }, @@ -756,16 +828,16 @@ var FilterLeaf = FilterNode.extend('FilterLeaf', { var fields = this.parent.nodeFields || this.fields; if (!fields) { - throw this.Error('Terminal node requires a fields list.'); + throw FilterNode.Error('Terminal node requires a fields list.'); } var root = this.el = document.createElement('span'); root.className = 'filter-tree-default'; this.controls = { - column: this.makeElement(root, fields, 'column'), - operator: this.makeElement(root, Object.keys(this.operators), 'operator'), - argument: this.makeElement(root) + column: this.makeElement(root, fields, 'column', true), + operator: this.makeElement(root, this.operatorsOptions, 'operator'), + value: this.makeElement(root) }; root.appendChild(document.createElement('br')); @@ -799,7 +871,7 @@ var FilterLeaf = FilterNode.extend('FilterLeaf', { * * Otherwise, creates a `` element with these options. * @param {null|string} [prompt=''] - Adds an initial `` element to the drop-down with this value, parenthesized, as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses. */ - makeElement: function(container, options, prompt) { + makeElement: function(container, options, prompt, sort) { var el, option, span, tagName = options ? 'select' : 'input'; @@ -816,8 +888,11 @@ var FilterLeaf = FilterNode.extend('FilterLeaf', { container.appendChild(span); } else { - el = addOptions(tagName, options, prompt); - this.el.addEventListener('change', cleanUpAndMoveOn); + el = addOptions(tagName, options, prompt, sort); + if (el.type === 'text' && this.eventHandler) { + this.el.addEventListener('keyup', this.eventHandler); + } + this.el.addEventListener('change', this.onChange); FilterNode.setWarningClass(el); container.appendChild(el); } @@ -887,57 +962,63 @@ var FilterLeaf = FilterNode.extend('FilterLeaf', { * Caught by {@link FilterTree#validate|FilterTree.prototype.validate()}. * * Also performs the following compilation actions: - * * Copies all the `this.controls`'s values from the DOM to similarly named properties of `this`. - * * Pre-sets `this.operation`, `this.converter` and `this.sqlOperator` for efficient access in walks. + * * Copies all `this.controls`' values from the DOM to similarly named properties of `this`. + * * Pre-sets `this.op` and `this.converter` for use in `test`'s tree walk. * * @param {boolean} focus - Move focus to offending control. * @returns {undefined} if valid */ validate: function(focus) { - for (var elementName in this.controls) { + var elementName, fields, field; + + for (elementName in this.controls) { var el = this.controls[elementName], value = controlValue(el).trim(); if (value === '') { - if (focus) { focusOn(el); } + if (focus) { clickIn(el); } throw new FilterNode.Error('Blank ' + elementName + ' control.\nComplete the filter or delete it.'); } else { // Copy each controls's value to property of this object. this[elementName] = value; + } + } - switch (elementName) { - case 'operator': - var operator = this.operators[value]; - this.operation = operator.test; // for efficient access in this.test() - this.sqlOperator = operator.SQL || value; - break; - case 'column': - var fields = this.parent.nodeFields || this.fields, - field = findField(fields, value); - if (field && field.type) { - this.converter = this.converters[field.type]; - } + this.op = this.operators[this.operator]; + + this.converter = undefined; // remains undefined when neither operator nor column is typed + if (this.op.type) { + this.converter = this.converters[this.op.type]; + } else { + for (elementName in this.controls) { + if (/^column/.test(elementName)) { + fields = this.parent.nodeFields || this.fields; + field = findField(fields, this[elementName]); + if (field && field.type) { + this.converter = this.converters[field.type]; + } } } } }, p: function(dataRow) { return dataRow[this.column]; }, - q: function() { return this.argument; }, + q: function() { return this.value; }, test: function(dataRow) { - var p = this.p(dataRow), - q = this.q(dataRow), + var p, q, // untyped versions of args P, Q, // typed versions of p and q - convert = this.converter; + convert; - return ( - convert && - !convert.not(P = convert.to(p)) && - !convert.not(Q = convert.to(q)) - ) - ? this.operation(P, Q) - : this.operation(p, q); + return (p = this.p(dataRow)) === undefined || (q = this.q(dataRow)) === undefined + ? false + : ( + (convert = this.converter) && + !convert.not(P = convert.to(p)) && + !convert.not(Q = convert.to(q)) + ) + ? this.op.test(P, Q) + : this.op.test(p, q); }, toJSON: function(options) { // eslint-disable-line no-unused-vars @@ -954,12 +1035,12 @@ var FilterLeaf = FilterNode.extend('FilterLeaf', { return state; }, - toSQL: function() { - return [ - this.SQL_QUOTED_IDENTIFIER + this.column + this.SQL_QUOTED_IDENTIFIER, - this.sqlOperator, - ' \'' + this.argument.replace(/'/g, '\'\'') + '\'' - ].join(' '); + getSqlWhereClause: function() { + return this.SQL_QUOTED_IDENTIFIER + this.column + this.SQL_QUOTED_IDENTIFIER + ' ' + ( + typeof this.op.sql === 'function' + ? this.op.sql(this.value) + : (this.op.sql || this.operator) + operators.sq(this.value) + ); } }); @@ -1001,9 +1082,13 @@ function cleanUpAndMoveOn(evt) { el.value = ''; // rid of any white space FilterNode.clickIn(el); } + + if (this.eventHandler) { + this.eventHandler(evt); + } } -function focusOn(el) { +function clickIn(el) { setTimeout(function() { el.classList.add('filter-tree-error'); FilterNode.clickIn(el); @@ -1048,7 +1133,7 @@ function controlValue(el) { * @param {null|string} [prompt=''] - Adds an initial `` element to the drop-down with this value in parentheses as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses. * @returns {Element} Either a ` + {{#items}} + + {{/items}} + + */ + }, - style.top = style.right = style.bottom = style.left = 0; + autopopulate: function() { + var behavior = this.grid.behavior; + var point = this.getEditorPoint(); + var colProps = this.grid.getColumnProperties(point.x); + if (!colProps.autopopulateEditor) { + return; + } + var headerCount = this.grid.getHeaderRowCount(); + var rowCount = this.grid.getUnfilteredRowCount() - headerCount; + var column = point.x; + var map = new Map(); + for (var r = 0; r < rowCount; r++) { + var each = behavior.getUnfilteredValue(column, r); + map.set(each, each); + } + var values = map.values; + values.sort(); - style.margin = margins; - style.zIndex = 100; - style.opacity = 1; + if (values.length > 0 && values[0].length > 0) { + values.unshift(''); + } - this.closeTransition = function() { - style.margin = margins; - }; + this.setItems(values); + }, - if (!this._closer) { - this._closer = function(e) { - var key = self.getCharFor(e.keyCode).toLowerCase(); - var keys = self.grid.resolveProperty('editorActivationKeys'); - if (keys.indexOf(key) > -1 || e.keyCode === 27) { - e.preventDefault(); - self.close(); - } - }; - } + //no events are fired while the dropdown is open + //see http://jsfiddle.net/m4tndtu4/6/ - //grid.setFocusable(false); - requestAnimationFrame(function() { - document.addEventListener('keydown', self._closer, false); - requestAnimationFrame(function() { - requestAnimationFrame(function() { - style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION; - style.margin = '15px 35px 35px 15px'; - }); - }); - }); + /** + * @memberOf Choice.prototype + */ + showEditor: function() { + var self = this; + this.input.style.display = 'inline'; setTimeout(function() { - self.overlay.focus(); - }, 100); + self.showDropdown(self.input); + }, 50); + }, + + preShowEditorNotification: function() { + this.autopopulate(); + this.setEditorValue(this.initialValue); }, /** - * @memberOf Overlay.prototype - * @desc close the overlay - * @param {Hypergrid} grid + * @memberOf Choice.prototype + * @param items */ - close: function() { - //grid.setFocusable(true); - this.openNow = false; - document.removeEventListener('keydown', this._closer, false); + setItems: function(items) { + this.items = items; + this.updateView(); + }, + /** + * @memberOf Choice.prototype + * @param input + */ + initializeInput: function(input) { var self = this; + Simple.prototype.initializeInput.apply(this, [input]); + input.onchange = function() { + self.stopEditing(); + }; + } - requestAnimationFrame(function() { - self.closeTransition(); - }); +}); - setTimeout(function() { - self.clear(); - self.overlay.style.zIndex = -1000; - if (self.onClose) { - self.onClose(); - self.onClose = undefined; - } - self.grid.takeFocus(); - }, ANIMATION_TIME); - }, +module.exports = Choice; + +},{"../lib/Mappy":70,"./Simple":40}],37:[function(require,module,exports){ +'use strict'; + +var Simple = require('./Simple'); + +/** + * @constructor + */ +var Color = Simple.extend('Color', { /** - * @memberOf Overlay.prototype - * @desc initialize the overlay surface into the grid - * #### returns: type - * @param {Hypergrid} grid + * my lookup alias + * @type {string} + * @memberOf Color.prototype */ - initializeOverlaySurface: function() { - this.overlay = document.createElement('div'); - this.overlay.setAttribute('tabindex', 0); - this.overlay.addEventListener('wheel', function(evt) { evt.stopPropagation(); }); + alias: 'color', - var style = this.overlay.style; - style.outline = 'none'; - style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)'; - style.position = 'absolute'; + template: function() { + /* + + */ + } - style.margin = 0; - style.overflow = 'hidden'; +}); - //style.display = 'none'; +module.exports = Color; - //style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION; +},{"./Simple":40}],38:[function(require,module,exports){ +/* eslint-env browser */ - style.opacity = 0; - style.zIndex = 10; - this.grid.div.appendChild(this.overlay); - //document.body.appendChild(this.overlay); - }, +'use strict'; + +var Simple = require('./Simple'); +var Formatters = require('../lib/Formatters'); + +function parseDate(input) { + var parts = input.match(/(\d+)/g); + // new Date(year, month [, date [, hours[, minutes[, seconds[, ms]]]]]) + return new window.Date(parts[0], parts[1] - 1, parts[2]); // months are 0-based +} + +/** + * @constructor + */ +var Date = Simple.extend('Date', { /** - * @memberOf Overlay.prototype - * @desc get a human readable description of the key pressed from it's integer representation - * @returns {string} - * @param {Hypergrid} grid - * @param {number} integer - the integer we want the char for + * my lookup alias + * @type {string} + * @memberOf Date.prototype */ - getCharFor: function(integer) { - var charMap = this.grid.getCanvas().getCharMap(); - return charMap[integer][0]; + alias: 'date', + + template: function() { + /* + + */ }, - clear: function() { - this.overlay.innerHTML = ''; + setEditorValue: function(value) { + if (value != null && value.constructor.name === 'Date') { + value = Formatters.date(value); + } + this.getInput().value = value + ''; }, - querySelector: function(selector) { - var elements = this.overlay.querySelector(selector); - return elements; + getEditorValue: function() { + var value = this.getInput().value; + value = parseDate(value); + return value; }, - getAnimationTime: function() { - return ANIMATION_TIME; - } }); -module.exports = TableDialog; +module.exports = Date; -},{"extend-me":4}],30:[function(require,module,exports){ +},{"../lib/Formatters":69,"./Simple":40}],39:[function(require,module,exports){ /* eslint-env browser */ - 'use strict'; -var _ = require('object-iterators'); -var Base = require('extend-me').Base; - -var Column = require('./Column'); -var CellProvider = require('../CellProvider'); - -var noExportProperties = [ - 'columnHeader', - 'columnHeaderColumnSelection', - 'filterProperties', - 'rowHeader', - 'rowHeaderRowSelection', - 'rowNumbersProperties', - 'treeColumnProperties', - 'treeColumnPropertiesColumnSelection', -]; +var CellEditor = require('./CellEditor'); /** * @constructor - * @desc This is the base class for creating behaviors. a behavior can be thought of as a model++. -it contains all code/data that's necessary for easily implementing a virtual data source and it's manipulation/analytics */ -var Behavior = Base.extend('Behavior', { +var Filter = CellEditor.extend('Filter', { /** - * @desc this is the callback for the plugin pattern of nested tags - * @param {Hypergrid} grid - * @memberOf Behavior.prototype + * my lookup alias + * @type {string} + * @memberOf Textfield.prototype */ - initialize: function(grid) { //formerly installOn - grid.setBehavior(this); - this.initializeFeatureChain(grid); + alias: 'filter', - this.getDataModel(); - this.cellProvider = this.createCellProvider(); - this.renderedColumnCount = 30; - this.renderedRowCount = 60; - this.dataUpdates = {}; //for overriding with edit values; - }, + initialize: function() { + var data = document.createElement('div'); + var style = data.style; + style.position = 'absolute'; + style.top = style.bottom = '44px'; + style.right = style.left = '1em'; + style.overflowY = 'scroll'; - /** - * @desc create the feature chain - this is the [chain of responsibility](http://c2.com/cgi/wiki?ChainOfResponsibilityPattern) pattern. - * @param {Hypergrid} grid - * @memberOf Behavior.prototype - */ - initializeFeatureChain: function(grid) { - var self = this; - this.features.forEach(function(FeatureConstructor) { - self.setNextFeature(new FeatureConstructor); - }); + var table = document.createElement('table'); + data.appendChild(table); - this.featureChain.initializeOn(grid); - }, + style = table.style; + style.width = style.height = '100%'; - features: [], // in case implementing class has no features TODO: Will this ever happen? + var tr = document.createElement('tr'); + var td = document.createElement('td'); + table.appendChild(tr); + tr.appendChild(td); + + + this.title = document.createElement('div'); + this.title.innerHTML = 'Filter Editor'; + + this.dialog = document.createElement('div'); + + this.content = td; + this.buttons = document.createElement('div'); - /** - * memento for the user configured visual properties of the table - * @type {object} - * @memberOf Behavior.prototype - */ - tableState: null, + style = this.dialog.style; + style.position = 'absolute'; + style.top = style.left = style.right = style.bottom = 0; + style.whiteSpace = 'nowrap'; - /** - * @type {Hypergrid} - * @memberOf Behavior.prototype - */ - grid: null, + style = this.title.style; + style.position = 'absolute'; + style.top = style.left = style.right = 0; + style.height = '44px'; + style.whiteSpace = 'nowrap'; + style.textAlign = 'center'; + style.padding = '11px'; - /** - * list of default cell editor names - * @type {string[]} - * @memberOf Behavior.prototype - */ - editorTypes: ['choice', 'textfield', 'color', 'slider', 'spinner', 'date'], + style = this.buttons.style; + style.position = 'absolute'; + style.left = style.right = style.bottom = 0; + style.height = '44px'; + style.whiteSpace = 'nowrap'; + style.textAlign = 'center'; + style.padding = '8px'; - /** - * controller chain of command - * @type {object} - * @memberOf Behavior.prototype - */ - featureChain: null, + this.dialog.appendChild(this.title); + this.dialog.appendChild(data); + this.dialog.appendChild(this.buttons); - dataModel: null, - baseModel: null, + this.ok = document.createElement('button'); + this.ok.style.borderRadius = '8px'; + this.ok.style.width = '5.5em'; - scrollPositionX: 0, - scrollPositionY: 0, + this.cancel = document.createElement('button'); + this.cancel.style.marginLeft = '2em'; + this.cancel.style.borderRadius = '8px'; + this.cancel.style.width = '5.5em'; - featureMap: {}, - allColumns: [], - columns: [], + this.delete = document.createElement('button'); + this.delete.style.marginLeft = '2em'; + this.delete.style.borderRadius = '8px'; + this.delete.style.width = '5.5em'; - reset: function() { + this.reset = document.createElement('button'); + this.reset.style.marginLeft = '2em'; + this.reset.style.borderRadius = '8px'; + this.reset.style.width = '5.5em'; - this.cellProvider = this.createCellProvider(); - this.renderedColumnCount = 30; - this.renderedRowCount = 60; - this.dataUpdates = {}; //for overriding with edit values; - this.clearColumns(); - this.clearState(); - this.getDataModel().reset(); - this.createColumns(); - }, + this.ok.innerHTML = 'ok'; + this.cancel.innerHTML = 'cancel'; + this.delete.innerHTML = 'delete'; + this.reset.innerHTML = 'reset'; - clearColumns: function() { - this.columns = []; - this.allColumns = []; - this.columns[-1] = this.newColumn(-1, ''); - this.columns[-2] = this.newColumn(-2, 'Tree'); - this.allColumns[-1] = this.columns[-1]; - this.allColumns[-2] = this.columns[-2]; - }, + this.buttons.appendChild(this.ok); + this.buttons.appendChild(this.reset); + this.buttons.appendChild(this.delete); + this.buttons.appendChild(this.cancel); - getColumn: function(x) { - return this.columns[x]; + var self = this; + this.ok.onclick = function() { + self.okPressed(); + }; + this.cancel.onclick = function() { + self.cancelPressed(); + }; + this.delete.onclick = function() { + self.deletePressed(); + }; + this.reset.onclick = function() { + self.resetPressed(); + }; }, - getColumnId: function(x) { - return this.getColumn(x).label; + tearDown: function() { + this.content.innerHTML = ''; }, - newColumn: function(index, label) { - var properties = this.createColumnProperties(); - this.getPrivateState().columnProperties[index] = properties; - return new Column(this, index, label); + okPressed: function() { + var dialog = this.grid.dialog; + dialog.onOkPressed(); }, - addColumn: function(index, label) { - var column = this.newColumn(index, label); - this.columns.push(column); - this.allColumns.push(column); - return column; + cancelPressed: function() { + var dialog = this.grid.dialog; + dialog.onCancelPressed(); }, - createColumns: function() { - //concrete implementation here + deletePressed: function() { + var dialog = this.grid.dialog; + dialog.onDeletePressed(); }, - createColumnProperties: function() { - var tableState = this.getPrivateState(); - var properties = Object.create(tableState); - - properties.rowNumbersProperties = Object.create(properties, { - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderForegroundSelectionColor; - }, - set: function(value) { - this.columnHeaderForegroundSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderBackgroundSelectionColor; - }, - set: function(value) { - this.columnHeaderBackgroundSelectionColor = value; - } - } - }); - - properties.rowHeader = Object.create(properties, { - font: { - configurable: true, - get: function() { - return this.rowHeaderFont; - }, - set: function(value) { - this.rowHeaderFont = value; - } - }, - color: { - configurable: true, - get: function() { - return this.rowHeaderColor; - }, - set: function(value) { - this.rowHeaderColor = value; - } - }, - backgroundColor: { - configurable: true, - get: function() { - return this.rowHeaderBackgroundColor; - }, - set: function(value) { - this.rowHeaderBackgroundColor = value; - } - }, - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.rowHeaderForegroundSelectionColor; - }, - set: function(value) { - this.rowHeaderForegroundSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.rowHeaderBackgroundSelectionColor; - }, - set: function(value) { - this.rowHeaderBackgroundSelectionColor = value; - } - } - }); - - properties.columnHeader = Object.create(properties, { - format: { - value: 'default' - }, - font: { - configurable: true, - get: function() { - return this.columnHeaderFont; - }, - set: function(value) { - this.columnHeaderFont = value; - } - }, - color: { - configurable: true, - get: function() { - return this.columnHeaderColor; - }, - set: function(value) { - this.columnHeaderColor = value; - } - }, - backgroundColor: { - configurable: true, - get: function() { - return this.columnHeaderBackgroundColor; - }, - set: function(value) { - this.columnHeaderBackgroundColor = value; - } - }, - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderForegroundSelectionColor; - }, - set: function(value) { - this.columnHeaderForegroundSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderBackgroundSelectionColor; - }, - set: function(value) { - this.columnHeaderBackgroundSelectionColor = value; - } - } - }); + resetPressed: function() { + var dialog = this.grid.dialog; + dialog.onResetPressed(); + }, - properties.columnHeaderColumnSelection = Object.create(properties.columnHeader, { - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderForegroundColumnSelectionColor; - }, - set: function(value) { - this.columnHeaderForegroundColumnSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.columnHeaderBackgroundColumnSelectionColor; - }, - set: function(value) { - this.columnHeaderBackgroundColumnSelectionColor = value; - } - } - }); + beginEditAt: function(editorPoint) { + var behavior = this.grid.behavior; + var dialog = this.grid.dialog; - properties.rowHeaderRowSelection = Object.create(properties.rowHeader, { - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.rowHeaderForegroundRowSelectionColor; - }, - set: function(value) { - this.rowHeaderForegroundRowSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.rowHeaderBackgroundRowSelectionColor; - }, - set: function(value) { - this.rowHeaderBackgroundRowSelectionColor = value; - } - } - }); + var columnIndex = editorPoint.x, + title = behavior.getColumnId(columnIndex), + field = behavior.getField(columnIndex), + type = behavior.getColumn(columnIndex).getType(); - properties.filterProperties = Object.create(properties, { - font: { - configurable: true, - get: function() { - return this.filterFont; - }, - set: function(value) { - this.filterFont = value; - } - }, - color: { - configurable: true, - get: function() { - return this.filterColor; - }, - set: function(value) { - this.filterColor = value; - } - }, - backgroundColor: { - configurable: true, - get: function() { - return this.filterBackgroundColor; - }, - set: function(value) { - this.filterBackgroundColor = value; - } - }, - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.filterForegroundSelectionColor; - }, - set: function(value) { - this.filterForegroundSelectionColor = value; - } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.filterBackgroundSelectionColor; - }, - set: function(value) { - this.filterBackgroundSelectionColor = value; - } - }, - cellBorderStyle: { - configurable: true, - get: function() { - return this.filterCellBorderStyle; - }, - set: function(value) { - this.filterCellBorderStyle = value; - } - }, - cellBorderThickness: { - configurable: true, - get: function() { - return this.filterCellBorderThickness; - }, - set: function(value) { - this.filterCellBorderThickness = value; - } - } - }); + var fieldsProvider = function() { + return [{ + name: field, + alias: title, + type: type + }]; + }; + this.title.innerHTML = 'Manage Filters'; + var filter = this.grid.filter; + //var self = this; + if (dialog.isOpen()) { + dialog.close(); + } else { + var self = this; - properties.treeColumnProperties = Object.create(properties, { - font: { - configurable: true, - get: function() { - return this.treeColumnFont; - }, - set: function(value) { - this.treeColumnFont = value; - } - }, - color: { - configurable: true, - get: function() { - return this.treeColumnColor; - }, - set: function(value) { - this.treeColumnColor = value; - } - }, - backgroundColor: { - configurable: true, - get: function() { - return this.treeColumnBackgroundColor; - }, - set: function(value) { - this.treeColumnBackgroundColor = value; + dialog.clear(); + dialog.overlay.appendChild(this.dialog); + + filter.initialize(fieldsProvider); + + dialog.onOkPressed = function() { + if (filter.onOk && filter.onOk()) { // onOK() truthy result means abort; falsy means proceed + return; } - }, - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.treeColumnForegroundSelectionColor; - }, - set: function(value) { - this.treeColumnForegroundSelectionColor = value; + self.tearDown(); + behavior.setComplexFilter(columnIndex, { + //type: filter.alias, + state: filter.getState() + }); + dialog.close(); + behavior.applyAnalytics(); + behavior.changed(); + }; + + dialog.onCancelPressed = function() { + if (filter.onCancel && filter.onCancel()) { + return; } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.treeColumnBackgroundSelectionColor; - }, - set: function(value) { - this.treeColumnBackgroundSelectionColor = value; + self.tearDown(); + dialog.close(); + filter = undefined; + }; + + dialog.onDeletePressed = function() { + if (filter.onDelete && filter.onDelete()) { + return; } - } - }); + self.tearDown(); + behavior.setComplexFilter(columnIndex, undefined); + dialog.close(); + behavior.applyAnalytics(); + behavior.changed(); + }; - properties.treeColumnPropertiesColumnSelection = Object.create(properties.treeColumnProperties, { - foregroundSelectionColor: { - configurable: true, - get: function() { - return this.treeColumnForegroundColumnSelectionColor; - }, - set: function(value) { - this.treeColumnForegroundColumnSelectionColor = value; + dialog.onResetPressed = function() { + if (filter.onReset && filter.onReset()) { + return; } - }, - backgroundSelectionColor: { - configurable: true, - get: function() { - return this.treeColumnBackgroundColumnSelectionColor; - }, - set: function(value) { - this.treeColumnBackgroundColumnSelectionColor = value; + self.tearDown(); + filter.initialize(dialog); + if (filter.onShow) { + filter.onShow(self.content); } - } - }); - - return properties; - }, - - getColumnWidth: function(x) { - var col = this.getColumn(x); - if (!col) { - return this.resolveProperty('defaultColumnWidth'); - } - var width = col.getWidth(); - return width; - }, + }; - setColumnWidth: function(x, width) { - this.getColumn(x).setWidth(width); - this.stateChanged(); - }, + var cellBounds = this.grid._getBoundsOfCell(columnIndex, editorPoint.y); - getDataModel: function() { - if (this.dataModel === null) { - var dataModel = this.getDefaultDataModel(); - this.setDataModel(dataModel); + //hack to accomodate bootstrap margin issues... + var xOffset = + this.grid.div.getBoundingClientRect().left - + this.grid.divCanvas.getBoundingClientRect().left; + cellBounds.x = cellBounds.x - xOffset; + dialog.openFrom(cellBounds); + var previousState = behavior.getComplexFilter(columnIndex); + if (previousState) { + filter.setState(previousState.state); + } + setTimeout(function() { + if (filter.onShow) { + filter.onShow(self.content); + } + }, dialog.getAnimationTime() + 10); } - return this.dataModel; }, - getCellRenderer: function(config, x, y) { - return this.getColumn(x).getCellRenderer(config, y); - }, +}); - setDataModel: function(newDataModel) { - this.dataModel = newDataModel; - }, +module.exports = Filter; - setComplexFilter: function(columnIndex, complexFilter) { - var col = this.getColumn(columnIndex); - if (col) { - col.setComplexFilter(complexFilter); - } - }, +},{"./CellEditor":35}],40:[function(require,module,exports){ +/* eslint-env browser */ - getComplexFilter: function(columnIndex) { - var col = this.getColumn(columnIndex); - if (col) { - return col.getComplexFilter(); - } - return; - }, +'use strict'; - applyFilters: function() { +var CellEditor = require('./CellEditor.js'); - }, +/** + * @constructor + */ +var Simple = CellEditor.extend('Simple', { /** - * @memberOf Behavior.prototype - * @desc utility function to empty an object of its members - * @param {object} obj - the object to empty - * @param {boolean} [exportProps] - * * `undefined` (omitted) - delete *all* properties - * * **falsy** - delete *only* the export properties - * * **truthy** - delete all properties *except* the export properties + * my main input control + * @type {Element} + * @default null + * @memberOf CellEditor.prototype */ - clearObjectProperties: function(obj, exportProps) { - for (var key in obj) { - if ( - obj.hasOwnProperty(key) && ( - exportProps === undefined || - !exportProps && noExportProperties.indexOf(key) >= 0 || - exportProps && noExportProperties.indexOf(key) < 0 - ) - ) { - delete obj[key]; - } - } - }, + input: null, /** - * @memberOf Behavior.prototype - * @desc getter for a [Memento](http://c2.com/cgi/wiki?MementoPattern) Object - * @returns {object} + * my lookup alias + * @type {string} + * @memberOf Simple.prototype */ - getPrivateState: function() { - if (!this.tableState) { - this.tableState = this.getDefaultState(); - } - return this.tableState; - }, + alias: 'simple', - //this is effectively a clone, with certain things removed.... - getState: function() { - var copy = JSON.parse(JSON.stringify(this.getPrivateState())); - this.clearObjectProperties(copy.columnProperties, false); - return copy; - }, /** - * @memberOf Behavior.prototype - * @desc clear all table state + * @memberOf Simple.prototype */ - clearState: function() { - this.tableState = null; + initialize: function() { + this.editorPoint = { + x: 0, + y: 0 + }; }, - /** - * @memberOf Behavior.prototype - * @return {object} Newly created default empty tablestate. - */ - getDefaultState: function() { - var tableProperties = this.getGrid()._getProperties(); - var state = Object.create(tableProperties); - - _(state).extendOwn({ - rowHeights: {}, - cellProperties: {}, - columnProperties: [] - }); - - return state; + specialKeyups: { + //0x08: 'clearStopEditing', // backspace + 0x09: 'stopEditing', // tab + 0x0d: 'stopEditing', // return/enter + 0x1b: 'cancelEditing' // escape }, - /** - * @memberOf Behavior.prototype - * @desc Restore this table to a previous state. - * See the [memento pattern](http://c2.com/cgi/wiki?MementoPattern). - * @param {Object} memento - an encapsulated representation of table state - */ - setState: function(memento) { + keyup: function(e) { + if (e) { + var specialKeyup = this.specialKeyups[e.keyCode]; - //we don't want to clobber the column properties completely - if (!memento.columnIndexes) { - var fields = this.getFields(); - memento.columnIndexes = []; - for (var i = 0; i < fields.length; i++) { - memento.columnIndexes[i] = i; + if (specialKeyup) { + e.preventDefault(); + this[specialKeyup](); + this.grid.repaint(); + this.grid.takeFocus(); } - } - var colProperties = memento.columnProperties; - delete memento.columnProperties; - this.tableState = null; - var state = this.getPrivateState(); - this.createColumns(); - this.setColumnOrder(memento.columnIndexes); - _(state).extendOwn(memento); - this.setAllColumnProperties(colProperties); - memento.columnProperties = colProperties; - //memento.columnProperties = colProperties; - - // this.getDataModel().setState(memento); - // var self = this; - // requestAnimationFrame(function() { - // self.applySorts(); - // self.changed(); - // self.stateChanged(); - // }); - - //just to be close/ it's easier on the eyes - this.setColumnWidth(-1, 24.193359375); - this.getDataModel().applyState(); - }, - - setAllColumnProperties: function(properties) { - properties = properties || []; - for (var i = 0; i < properties.length; i++) { - var current = this.getPrivateState().columnProperties[i]; - this.clearObjectProperties(current, false); - _(current).extendOwn(properties[i]); - } - }, - setColumnOrder: function(indexes) { - if (!indexes) { - this.columns.length = 0; - return; - } - this.columns.length = indexes.length; - for (var i = 0; i < indexes.length; i++) { - this.columns[i] = this.allColumns[indexes[i]]; + this.grid.fireSyntheticEditorKeyUpEvent(this, e); } }, - applySorts: function() { - //if I have sorts, apply them now// - }, - /** - * @memberOf Behavior.prototype - * @desc fetch the value for a property key - * @returns {*} The value of the given property. - * @param {string} key - a property name + * @memberOf Simple.prototype + * @desc the function to override for initialization */ - resolveProperty: function(key) { - return this.grid.resolveProperty(key); + initializeInput: function(input) { + var self = this; + input.addEventListener('keyup', this.keyup.bind(this)); + input.addEventListener('keydown', function(e) { + self.grid.fireSyntheticEditorKeyDownEvent(self, e); + }); + input.addEventListener('keypress', function(e) { + self.grid.fireSyntheticEditorKeyPressEvent(self, e); + }); + input.onblur = function(e) { + self.cancelEditing(); + }; + + // input.addEventListener('focusout', function() { + // self.stopEditing(); + // }); + // input.addEventListener('blur', function() { + // self.stopEditing(); + // }); + + input.style.position = 'absolute'; + input.style.display = 'none'; + input.style.border = 'solid 2px black'; + input.style.outline = 0; + input.style.padding = 0; + input.style.boxShadow = 'white 0px 0px 1px 1px'; }, /** - * @memberOf Behavior.prototype - * @desc A specific cell was clicked; you've been notified. - * @param {Point} cell - point of cell coordinates - * @param {Object} event - all event information + * @memberOf Simple.prototype + * @returns {object} the current editor's value */ - cellClicked: function(cell, event) { - this.getDataModel().cellClicked(cell, event); + getEditorValue: function() { + var value = this.getInput().value; + return value; }, /** - * @memberOf Behavior.prototype - * @desc A specific cell was le double-clicked; you've been notified. - * @param {Point} cell - point of cell coordinates - * @param {Object} event - all event information + * @memberOf Simple.prototype + * @desc save the new value into the behavior(model) */ - cellDoubleClicked: function(cell, event) { + setEditorValue: function(value) { + this.getInput().value = value; + }, + clearStopEditing: function() { + this.setEditorValue(''); + this.stopEditing(); }, - /** - * @memberOf Behavior.prototype - * @desc add nextFeature to me If I don't have a next node, otherwise pass it along - * @param {Feature} - */ - setNextFeature: function(nextFeature) { - this.featureMap[nextFeature.alias] = nextFeature; - if (this.featureChain) { - this.featureChain.setNext(nextFeature); - } else { - this.featureChain = nextFeature; + cancelEditing: function() { + if (!this.isEditing) { + return; } + this.getInput().value = null; + this.isEditing = false; + this.hideEditor(); }, - lookupFeature: function(key) { - return this.featureMap[key]; + /** + * @memberOf Simple.prototype + * @desc display the editor + */ + showEditor: function() { + this.getInput().style.display = 'inline'; }, /** - * @memberOf Behavior.prototype - * @desc getter for the cell provider - * @return {CellProvider} + * @memberOf Simple.prototype + * @desc hide the editor */ - getCellProvider: function() { - return this.cellProvider; + hideEditor: function() { + this.getInput().style.display = 'none'; }, /** - * @memberOf Behavior.prototype - * @desc setter for the hypergrid - * @param {Hypergrid} grid + * @summary Request focus for my input control. + * @desc See GRID-95 "Scrollbar moves inward" for issue and work-around explanation. + * @memberOf Simple.prototype */ - setGrid: function(finGrid) { - this.grid = finGrid; - this.getDataModel().setGrid(finGrid); - this.clearColumns(); + takeFocus: function() { + var self = this; + setTimeout(function() { + var transformWas = self.input.style.transform; + self.input.style.transform = 'translate(0,0)'; // work-around: move to upper left + + self.input.focus(); + self.selectAll(); + + self.input.style.transform = transformWas; + }); }, /** - * @memberOf Behavior.prototype - * @returns: {Hypergrid} The hypergrid to which this behavior is attached. - * @param {type} varname - descripton + * @memberOf Simple.prototype + * @desc select everything */ - getGrid: function() { - return this.grid; + selectAll: function() { + }, /** - * @memberOf Behavior.prototype - * @desc You can override this function and substitute your own cell provider. - * @return {CellProvider} + * @memberOf Simple.prototype + * @desc how much should I offset my bounds from 0,0 */ - createCellProvider: function() { - return new CellProvider(); + originOffset: function() { + return [0, 0]; }, /** - * @memberOf Behavior.prototype - * @desc First check to see if something was overridden. - * @return {*} The value at `x,y` for the top left section of the hypergrid. - * @param {number} x - x coordinate - * @param {number} y - y coordinate + * @memberOf Simple.prototype + * @desc set the bounds of my input control + * @param {rectangle} rectangle - the bounds to move to */ - getValue: function(x, y) { - var column = this.getColumn(x); - if (!column) { - return undefined; - } - return column.getValue(y); + setBounds: function(cellBounds) { + var originOffset = this.originOffset(); + var translation = 'translate(' + + (cellBounds.x - 1 + originOffset[0]) + 'px,' + + (cellBounds.y - 1 + originOffset[1]) + 'px)'; + + var input = this.getInput(); + + input.style.boxSizing = 'border-box'; + + input.style.webkitTransform = translation; + input.style.MozTransform = translation; + input.style.msTransform = translation; + input.style.OTransform = translation; + + // TODO: Obviously this was changed at some point from left,top to trnasform:translation. Wondering why this was necessary...? + + // input.style.left = cellBounds.x + originOffset[0] + 'px'; + // input.style.top = cellBounds.y + originOffset[1] + 'px'; + + input.style.width = (cellBounds.width + 2) + 'px'; + input.style.height = (cellBounds.height + 2) + 'px'; + //var xOffset = this.grid.canvas.getBoundingClientRect().left; }, - getUnfilteredValue: function(x, y) { - var column = this.getColumn(x); - if (!column) { - return undefined; + saveEditorValue: function() { + var point = this.getEditorPoint(); + var value = this.getEditorValue(); + if (value === this.initialValue) { + return; //data didn't change do nothing + } + if (parseFloat(this.initialValue) === this.initialValue) { // I'm a number + value = parseFloat(value); + } + var continued = this.grid.fireBeforeCellEdit(point, this.initialValue, value, this); + if (!continued) { + return; } - return column.getUnfilteredValue(y); + this.grid.behavior.setValue(point.x, point.y, value); + this.grid.fireAfterCellEdit(point, this.initialValue, value, this); }, /** - * @memberOf Behavior.prototype - * @desc update the data at point x, y with value - * @return The data. - * @param {number} x - x coordinate - * @param {number} y - y coordinate - * @param {Object} value - the value to use + * @memberOf CellEditor.prototype + * @desc move the editor to the current editor point */ - setValue: function(x, y, value) { - var column = this.getColumn(x); - if (!column) { + _moveEditor: function() { + var editorPoint = this.getEditorPoint(); + var cellBounds = this.grid._getBoundsOfCell(editorPoint.x, editorPoint.y); + + //hack to accommodate bootstrap margin issues... + var xOffset = + this.grid.div.getBoundingClientRect().left - + this.grid.divCanvas.getBoundingClientRect().left; + cellBounds.x = cellBounds.x - xOffset; + + this.setBounds(cellBounds); + }, + + moveEditor: function() { + this._moveEditor(); + this.takeFocus(); + }, + + beginEditAt: function(point) { + + if (!this.isAdded) { + this.isAdded = true; + this.attachEditor(); + } + + this.setEditorPoint(point); + var value = this.grid.behavior.getValue(point.x, point.y); + if (value.constructor.name === 'Array') { + value = value[1]; //it's a nested object + } + var proceed = this.grid.fireRequestCellEdit(point, value); + if (!proceed) { + //we were cancelled return; } - return column.setValue(y, value); + this.initialValue = value; + this.isEditing = true; + this.setCheckEditorPositionFlag(); + this.checkEditor(); }, - getDataValue: function(x, y) { - return this.getDataModel().getValue(x, y); + checkEditor: function() { + if (!this.checkEditorPositionFlag) { + return; + } else { + this.checkEditorPositionFlag = false; + } + if (!this.isEditing) { + return; + } + var editorPoint = this.getEditorPoint(); + if (this.grid.isDataVisible(editorPoint.x, editorPoint.y)) { + this.preShowEditorNotification(); + this.attachEditor(); + this.moveEditor(); + this.showEditor(); + } else { + this.hideEditor(); + } }, - setDataValue: function(x, y, value) { - this.getDataModel().setValue(x, y, value); + attachEditor: function() { + var input = this.getInput(), + div = this.grid.div, + referenceNode = div.querySelectorAll('.finbar-horizontal, .finbar-vertical'); + + div.insertBefore(input, referenceNode.length ? referenceNode[0] : null); }, - /** - * @memberOf Behavior.prototype - * @desc First checks to see if something was overridden. - * @return {*} The value at x,y for the top left section of the hypergrid. - * @param {number} x - x coordinate - * @param {number} y - y coordinate - */ - getCellProperties: function(x, y) { - var col = this.getColumn(x); - return col.getCellProperties(y); + + preShowEditorNotification: function() { + this.setEditorValue(this.initialValue); }, - /** - * @memberOf Behavior.prototype - * @desc update the data at point x, y with value - * @param {number} x - x coordinate - * @param {number} y - y coordinate - * @param {Object} value - the value to use - */ - setCellProperties: function(x, y, value) { - var col = this.getColumn(x); - if (col) { - col.setCellProperties(y, value); + getInput: function() { + if (!this.input) { + this.input = this.getDefaultInput(); } + return this.input; }, - /** - * @memberOf Behavior.prototype - * @return {number} The number of rows in the hypergrid. - */ - getRowCount: function() { - return this.getDataModel().getRowCount(); + + getDefaultInput: function() { + var div = document.createElement('DIV'); + div.innerHTML = this.getHTML(); + var input = div.firstChild; + this.initializeInput(input); + return input; }, - getUnfilteredRowCount: function() { - return this.getDataModel().getUnfilteredRowCount(); + updateView: function() { + var oldGuy = this.getInput(); + var parent = oldGuy.parentNode; + var newGuy = this.getDefaultInput(); + this.input = newGuy; + parent.replaceChild(newGuy, oldGuy); }, + + showDropdown: function(element) { + var event; + event = document.createEvent('MouseEvents'); + event.initMouseEvent('mousedown', true, true, window); + element.dispatchEvent(event); + } +}); + +module.exports = Simple; + +},{"./CellEditor.js":35}],41:[function(require,module,exports){ +'use strict'; + +var Simple = require('./Simple'); + +/** + * @constructor + */ +var Slider = Simple.extend('Slider', { + /** - * @memberOf Behavior.prototype - * @return {number} The height in pixels of the fixed rows area of the hypergrid. + * my lookup alias + * @type {string} + * @memberOf Slider.prototype */ - getFixedRowsHeight: function() { - var count = this.getFixedRowCount(); - var total = 0; - for (var i = 0; i < count; i++) { - total = total + this.getRowHeight(i); - } - //var footerHeight = this.getDefaultRowHeight(); - //total = total + (footerHeight * this.getFooterRowCount()); - return total; - }, + alias: 'slider', + + template: function() { + /* + + */ + } + +}); + +module.exports = Slider; + +},{"./Simple":40}],42:[function(require,module,exports){ +'use strict'; + +var Simple = require('./Simple'); + +/** + * @constructor + */ +var Spinner = Simple.extend('Spinner', { + + /** + * my lookup alias + * @type {string} + * @memberOf Spinner.prototype + */ + alias: 'spinner', + + template: function() { + /* + + */ + } + +}); + +module.exports = Spinner; + +},{"./Simple":40}],43:[function(require,module,exports){ +'use strict'; - /** - * @memberOf Behavior.prototype - * @return {number} The height in pixels of a specific row in the hypergrid. - * @param {number} rowNum - row index of interest - */ - getRowHeight: function(rowNum) { - var tableState = this.getPrivateState(); - if (tableState.rowHeights) { - var override = tableState.rowHeights[rowNum]; - if (override) { - return override; - } - } - return this.getDefaultRowHeight(); - }, +var Simple = require('./Simple'); + +/** + * @constructor + */ +var Textfield = Simple.extend('Textfield', { /** - * @memberOf Behavior.prototype - * @desc The value is lazily initialized and comes from the properties mechanism for '`defaultRowHeight`', which should be ~20px. - * @returns {number} The row height in pixels. + * my lookup alias + * @type {string} + * @memberOf Textfield.prototype */ - getDefaultRowHeight: function() { - if (!this.defaultRowHeight) { - this.defaultRowHeight = this.resolveProperty('defaultRowHeight'); - } - return this.defaultRowHeight; + alias: 'textfield', + + template: function() { + /* + + */ }, - /** - * @memberOf Behavior.prototype - * @desc set the pixel height of a specific row - * @param {number} rowNum - the row index of interest - * @param {number} height - pixel height - */ - setRowHeight: function(rowNum, height) { - var tableState = this.getPrivateState(); - tableState.rowHeights[rowNum] = Math.max(5, height); - this.stateChanged(); + selectAll: function() { + this.input.setSelectionRange(0, this.input.value.length); }, - /** - * @memberOf Behavior.prototype - * @desc This will allow 'floating' fixed rows. - * @return {number} The maximum height of the fixed rows area in the hypergrid. - */ - getFixedRowsMaxHeight: function() { - return this.getFixedRowsHeight(); + specialKeyups: { + 0x09: 'stopEditing', // tab + 0x0d: 'stopEditing', // return/enter + 0x1b: 'cancelEditing' // escape }, - /** - * @memberOf Behavior.prototype - * @return {number} The width of the fixed column area in the hypergrid. - */ - getFixedColumnsWidth: function() { - var count = this.getFixedColumnCount(); - var total = 0; - if (this.getGrid().isShowRowNumbers()) { - total = this.getColumnWidth(-1); - } - for (var i = 0; i < count; i++) { - total = total + this.getColumnWidth(i); + keyup: function(e) { + if (e) { + Simple.prototype.keyup.call(this, e); + + if (this.grid.isFilterRow(this.getEditorPoint().y)) { + setTimeout(keyup.bind(this)); + } } - return total; - }, + } +}); - /** - * @memberOf Behavior.prototype - * @desc This exists to support "floating" columns. - * @return {number} The total width of the fixed columns area. - */ - getFixedColumnsMaxWidth: function() { - var width = this.getFixedColumnsWidth(); - return width; +function keyup() { + this.saveEditorValue(); + this._moveEditor(); +} + +module.exports = Textfield; + +},{"./Simple":40}],44:[function(require,module,exports){ +'use strict'; + +module.exports = { + CellEditor: require('./CellEditor'), // abstract base class + Textfield: require('./Textfield'), + Choice: require('./Choice'), + //Combo: require('./Combo'), + Color: require('./Color'), + Date: require('./Date'), + Simple: require('./Simple'), + Slider: require('./Slider'), + Spinner: require('./Spinner'), + Filter: require('./Filter') +}; + +},{"./CellEditor":35,"./Choice":36,"./Color":37,"./Date":38,"./Filter":39,"./Simple":40,"./Slider":41,"./Spinner":42,"./Textfield":43}],45:[function(require,module,exports){ +'use strict'; + +var Base = require('../lib/Base'); + +var A = 'A'.charCodeAt(0); + +/** + * @constructor + */ +var DataModel = Base.extend('DataModel', { + + next: null, + + grid: null, + + setGrid: function(newGrid) { + this.grid = newGrid; }, - /** - * @memberOf Behavior.prototype - * @desc Set the scroll position in vertical dimension and notify listeners. - * @param {number} y - the new y value - */ - _setScrollPositionY: function(y) { - this.setScrollPositionY(y); - this.changed(); + /** @deprecated Use `.grid` property instead. */ + getGrid: function() { + return this.deprecated('grid', { since: '0.2' }); }, - /** - * @memberOf Behavior.prototype - * @desc Set the scroll position in horizontal dimension and notify listeners. - * @param {number} x - the new x value - */ - _setScrollPositionX: function(x) { - this.setScrollPositionX(x); - this.changed(); + /** @deprecated Use `.grid.behavior` property instead. */ + getBehavior: function() { + return this.deprecated('grid.behavior', { since: '0.2' }); }, - /** - * @memberOf Behavior.prototype - * @desc Set the number of columns just rendered, including partially rendered columns. - * @param {number} count - how many columns were just rendered - */ - setRenderedColumnCount: function(count) { - this.renderedColumnCount = count; + changed: function() { + this.grid.behavior.changed(); }, - /** - * @memberOf Behavior.prototype - * @desc Set the number of rows just rendered, including partially rendered rows. - * @param {number} count - how many rows were just rendered - */ - setRenderedRowCount: function(count) { - this.renderedRowCount = count; + getPrivateState: function() { + return this.grid.getPrivateState(); }, + applyState: function() { - /** - * @memberOf Behavior.prototype - * @desc The fixed row area has been clicked, massage the details and call the real function. - * @param {Hypergrid} grid - * @param {Object} mouse - event details - */ - _fixedRowClicked: function(grid, mouse) { - var x = this.translateColumnIndex(this.getScrollPositionX() + mouse.gridCell.x - this.getFixedColumnCount()); - var translatedPoint = this.grid.newPoint(x, mouse.gridCell.y); - mouse.gridCell = translatedPoint; - this.fixedRowClicked(grid, mouse); }, - /** - * @memberOf Behavior.prototype - * @desc The fixed column area has been clicked, massage the details and call the real function. - * @param {Hypergrid} grid - * @param {Object} mouse - event details - */ - _fixedColumnClicked: function(grid, mouse) { - var translatedPoint = this.grid.newPoint(mouse.gridCell.x, this.getScrollPositionY() + mouse.gridCell.y - this.getFixedRowCount()); - mouse.gridCell = translatedPoint; - this.fixedColumnClicked(grid, mouse); + alphaFor: function(i) { + // Name the column headers in A, .., AA, AB, AC, .., AZ format + // quotient/remainder + //var quo = Math.floor(col/27); + var quo = Math.floor(i / 26); + var rem = i % 26; + var code = ''; + if (quo > 0) { + code += this.alpha(quo - 1); + } + code += this.alpha(rem); + return code; }, - moveSingleSelect: function(grid, x, y) { - if (this.featureChain) { - this.featureChain.moveSingleSelect(grid, x, y); - this.setCursor(grid); - } + alpha: function(i) { + return String.fromCharCode(A + i); }, - /** - * @memberOf Behavior.prototype - * @desc delegate setting the cursor up the feature chain of responsibility - * @param {Hypergrid} grid - */ - setCursor: function(grid) { - grid.updateCursor(); - this.featureChain.setCursor(grid); + getCellEditorAt: function(x, y) { }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling mouse move to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onMouseMove: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleMouseMove(grid, event); - this.setCursor(grid); - } +}); + +module.exports = DataModel; + +},{"../lib/Base":66}],46:[function(require,module,exports){ +'use strict'; + +//var analytics = require('hyper-analytics'); +//var analytics = require('../local_node_modules/hyper-analytics'); +var analytics = require('../local_node_modules/finanalytics'); +var DataModel = require('./DataModel'); +var images = require('../../images'); + +var UPWARDS_BLACK_ARROW = '\u25b2', // aka '▲' + DOWNWARDS_BLACK_ARROW = '\u25bc'; // aka '▼' + +var nullDataSource = { + isNullObject: function() { + return true; + }, + getFields: function() { + return []; + }, + getHeaders: function() { + return []; + }, + getColumnCount: function() { + return 0; + }, + getRowCount: function() { + return 0; + }, + getAggregateTotals: function() { + return []; + }, + hasAggregates: function() { + return false; + }, + hasGroups: function() { + return false; }, + getRow: function() { + return null; + } +}; + +/** + * @name dataModels.JSON + * @constructor + */ +var JSON = DataModel.extend('dataModels.JSON', { + + //null object pattern for the source object + source: nullDataSource, - /** - * @memberOf Behavior.prototype - * @desc delegate handling tap to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onTap: function(grid, event) { + preglobalfilter: nullDataSource, + prefilter: nullDataSource, - if (this.featureChain) { - this.featureChain.handleTap(grid, event); - this.setCursor(grid); - } - }, + presorter: nullDataSource, + analytics: nullDataSource, + postglobalfilter: nullDataSource, + postfilter: nullDataSource, + postsorter: nullDataSource, - /** - * @memberOf Behavior.prototype - * @desc delegate handling tap to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onContextMenu: function(grid, event) { - var proceed = grid.fireSyntheticContextMenuEvent(event); - if (proceed && this.featureChain) { - this.featureChain.handleContextMenu(grid, event); - this.setCursor(grid); - } - }, + topTotals: [], + bottomTotals: [], - /** - * @memberOf Behavior.prototype - * @desc delegate handling wheel moved to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onWheelMoved: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleWheelMoved(grid, event); - this.setCursor(grid); - } + initialize: function() { + this.selectedData = []; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling mouse up to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onMouseUp: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleMouseUp(grid, event); - this.setCursor(grid); - } + clearSelectedData: function() { + this.selectedData.length = 0; }, /** - * @memberOf Behavior.prototype - * @desc delegate handling mouse drag to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf dataModels.JSON.prototype + * @returns {boolean} */ - onMouseDrag: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleMouseDrag(grid, event); - this.setCursor(grid); - } + hasAggregates: function() { + return this.analytics.hasAggregates(); }, /** - * @memberOf Behavior.prototype - * @desc delegate handling key down to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf dataModels.JSON.prototype + * @returns {boolean} */ - onKeyDown: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleKeyDown(grid, event); - this.setCursor(grid); - } + hasGroups: function() { + return this.analytics.hasGroups(); }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling key up to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onKeyUp: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleKeyUp(grid, event); - this.setCursor(grid); - } + getDataSource: function() { + return this.postsorter; //this.hasAggregates() ? this.analytics : this.presorter; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling double click to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onDoubleClick: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleDoubleClick(grid, event); - this.setCursor(grid); - } + getFilterSource: function() { + return this.postfilter; //this.hasAggregates() ? this.postfilter : this.prefilter; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling hold pulse to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - onHoldPulse: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleHoldPulse(grid, event); - this.setCursor(grid); - } + getGlobalFilterSource: function() { + return this.postglobalfilter; //this.hasAggregates() ? this.postfilter : this.prefilter; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling double click to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - toggleColumnPicker: function() { - var dialog = this.grid.dialog; - var self = this; - if (dialog.isOpen()) { - dialog.close(); - } else { - this.buildColumnPicker(dialog.overlay); - dialog.onClose = function() { - self.updateFromColumnPicker(dialog.overlay); - }; - dialog.open(); - } + getSortingSource: function() { + return this.postsorter; //this.hasAggregates() ? this.postsorter : this.presorter; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling mouse down to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDown: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleMouseDown(grid, event); - this.setCursor(grid); - } + getData: function() { + return this.source.data; }, - /** - * @memberOf Behavior.prototype - * @desc delegate handling mouse exit to the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseExit: function(grid, event) { - if (this.featureChain) { - this.featureChain.handleMouseExit(grid, event); - this.setCursor(grid); + getFilteredData: function() { + var ds = this.getDataSource(); + var count = ds.getRowCount(); + var result = new Array(count); + for (var y = 0; y < count; y++) { + result[y] = ds.getRow(y); } + return result; }, /** - * @memberOf Behavior.prototype - * @desc this function is replaced by the grid on initialization and serves as the callback - */ - changed: function() {}, - - /** - * @memberOf Behavior.prototype - * @desc this function is replaced by the grid on initialization and serves as the callback - */ - shapeChanged: function() {}, - - /** - * @memberOf Behavior.prototype - * @return {boolean} Can re-order columns. + * @memberOf dataModels.JSON.prototype + * @param {number} x + * @param {number} y + * @returns {*} */ - isColumnReorderable: function() { - return true; + getValue: function(x, y) { + var hasHierarchyColumn = this.hasHierarchyColumn(); + var headerRowCount = this.grid.getHeaderRowCount(); + var value; + if (hasHierarchyColumn) { + if (x === -2) { + x = 0; + } + } else if (this.hasAggregates()) { + x += 1; + } + if (y < headerRowCount) { + value = this.getHeaderRowValue(x, y); + return value; + } + // if (hasHierarchyColumn) { + // y += 1; + // } + value = this.getDataSource().getValue(x, y - headerRowCount); + return value; }, /** - * @memberOf Behavior.prototype - * @return {Object} The properties for a specific column. These are used if no cell properties are specified. - * @param {index} columnIndex - the column index of interest + * @memberOf dataModels.JSON.prototype + * @param {number} x + * @param {number} y - negative values refer to _bottom totals_ rows + * @returns {*} */ - getColumnProperties: function(columnIndex) { - var col = this.columns[columnIndex]; - if (!col) { - return { - isNull: true - }; - } - var properties = col.getProperties(); //TODO: returns `null` on Hypergrid.reset(); - if (!properties) { - return { - isNull: true - }; - } - return properties; - }, - setColumnProperties: function(columnIndex, properties) { - var columnProperties = this.allColumns[columnIndex].getProperties(); - _(columnProperties).extendOwn(properties); - this.changed(); + getHeaderRowValue: function(x, y) { + var value; + if (y === undefined) { + value = this.getHeaders()[Math.max(x, 0)]; + } else if (y < 0) { // bottom totals rows + var bottomTotals = this.getBottomTotals(); + value = bottomTotals[bottomTotals.length + y][x]; + } else { + var isFilterRow = this.grid.isShowFilterRow(), + isHeaderRow = this.grid.isShowHeaderRow(), + topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0); + if (y >= topTotalsOffset) { // top totals rows + value = this.getTopTotals()[y - topTotalsOffset][x]; + } else if (isHeaderRow && y === 0) { + value = this.getHeaders()[x]; + var sortString = this.getSortImageForColumn(x); + if (sortString) { value = sortString + value; } + } else { // must be filter row + value = this.getFilter(x); + var icon = images.filter(value.length); + return [null, value, icon]; + } + } + return value; }, /** - * @memberOf Behavior.prototype - * @return {string} The field at `colIndex`. - * @param {number} colIndex - the column index of interest + * @memberOf dataModels.JSON.prototype + * @param {number} x + * @param {number} y + * @param value */ - getField: function(colIndex) { - if (colIndex === -1) { - return 'tree'; + setValue: function(x, y, value) { + var hasHierarchyColumn = this.hasHierarchyColumn(); + var headerRowCount = this.grid.getHeaderRowCount(); + if (hasHierarchyColumn) { + if (x === -2) { + x = 0; + } + } else if (this.hasAggregates()) { + x += 1; + } + if (y < headerRowCount) { + this.setHeaderRowValue(x, y, value); + } else { + this.getDataSource().setValue(x, y - headerRowCount, value); } - var col = this.getColumn(colIndex); - return col.getField(); + this.changed(); }, + /** - * @memberOf Behavior.prototype - * @return {string} The column heading at `colIndex'. - * @param {number} colIndex - the column index of interest + * @memberOf dataModels.JSON.prototype + * @param {number} x + * @param {number} y + * @param value + * @returns {*} */ - getHeader: function(colIndex) { - if (colIndex === -1) { - return 'Tree'; + setHeaderRowValue: function(x, y, value) { + if (value === undefined) { + return this._setHeader(x, y); // y is really the value + } + var isFilterRow = this.grid.isShowFilterRow(); + var isHeaderRow = this.grid.isShowHeaderRow(); + var isBoth = isFilterRow && isHeaderRow; + var topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0); + if (y >= topTotalsOffset) { + this.getTopTotals()[y - topTotalsOffset][x] = value; + } else if (x === -1) { + return; // can't change the row numbers + } else if (isBoth) { + if (y === 0) { + return this._setHeader(x, value); + } else { + this.setFilter(x, value); + } + } else if (isFilterRow) { + this.setFilter(x, value); + } else { + return this._setHeader(x, value); } - var col = this.getColumn(colIndex); - return col.getHeader(); + return ''; }, + /** - * @memberOf Behavior.prototype - * @desc this is called by the column editor post closing; rebuild the column order indexes - * @param {Array} list - list of column objects from the column editor + * @memberOf dataModels.JSON.prototype + * @param {number} colIndex + * @returns {*} */ - setColumnDescriptors: function(lists) { - //assumes there is one row.... - var visible = lists.visible; - var tableState = this.getPrivateState(); - - var columnCount = visible.length; - var indexes = []; - var i; - for (i = 0; i < columnCount; i++) { - indexes.push(visible[i].id); + getColumnProperties: function(colIndex) { + //access directly because we want it ordered + var column = this.grid.behavior.allColumns[colIndex]; + if (column) { + return column.getProperties(); } - tableState.columnIndexes = indexes; - this.changed(); + return undefined; }, /** - * @memberOf Behavior.prototype - * @return {string[]} All the currently hidden column header labels. + * @memberOf dataModels.JSON.prototype + * @param {number} colIndex + * @returns {string} The text to filter on for this column. */ - getHiddenColumnDescriptors: function() { - var tableState = this.getPrivateState(); - var indexes = tableState.columnIndexes; - var labels = []; - var columnCount = this.getColumnCount(); - for (var i = 0; i < columnCount; i++) { - if (indexes.indexOf(i) === -1) { - labels.push({ - id: i, - label: this.getHeader(i), - field: this.getField(i) - }); - } + getFilter: function(colIndex) { + var filter, columnProperties; + + if ((columnProperties = this.getColumnProperties(colIndex))) { + filter = columnProperties.filter; } - return labels; + + return filter || ''; }, + /** @typedef {function} rowFilterFunction + * @param {function|*} data - Data to test (or function to call to get data to test) to see if it qualifies for the result set. + * @returns {boolean} Row qualifies for the result set (passes through filter). + */ /** - * @memberOf Behavior.prototype - * @desc hide columns that are specified by their indexes - * @param {Array} arrayOfIndexes - an array of column indexes to hide + * @param {number} colIndex + * @returns {undefined|rowFilterFunction} row filtering function */ - hideColumns: function(arrayOfIndexes) { - var tableState = this.getPrivateState(); - var order = tableState.columnIndexes; - for (var i = 0; i < arrayOfIndexes.length; i++) { - var each = arrayOfIndexes[i]; - if (order.indexOf(each) !== -1) { - order.splice(order.indexOf(each), 1); - } + getComplexFilter: function(colIndex) { + var rowFilter, columnProperties, filter, filterObject, newFilter; + + if ( + (columnProperties = this.getColumnProperties(colIndex)) && + (filter = columnProperties.complexFilter) && + (filterObject = this.grid.filter) && + (newFilter = filterObject.create(filter.state)) + ) { + rowFilter = function(data) { + var transformed = valueOrFunctionExecute(data); + return newFilter(transformed); + }; } + + return rowFilter; }, /** - * @memberOf Behavior.prototype - * @return {integer} The number of fixed columns. + * @memberOf dataModels.JSON.prototype + * @param {number} colIndex + * @param value */ - getFixedColumnCount: function() { - var tableState = this.getPrivateState(); - return tableState.fixedColumnCount || 0; + setFilter: function(colIndex, value) { + var columnProperties = this.getColumnProperties(colIndex); + columnProperties.filter = value; + this.applyAnalytics(); }, /** - * @memberOf Behavior.prototype - * @desc set the number of fixed columns - * @param {number} n - the integer count of how many columns to be fixed + * @memberOf dataModels.JSON.prototype + * @returns {number} */ - setFixedColumnCount: function(n) { - var tableState = this.getPrivateState(); - tableState.fixedColumnCount = n; + getColumnCount: function() { + var showTree = this.grid.resolveProperty('showTreeColumn') === true; + var hasAggregates = this.hasAggregates(); + var offset = (hasAggregates && !showTree) ? -1 : 0; + return this.analytics.getColumnCount() + offset; }, /** - * @memberOf Behavior.prototype - * @return {integer} The number of fixed rows. + * @memberOf dataModels.JSON.prototype + * @returns {number} */ - getFixedRowCount: function() { - if (!this.tableState) { - return 0; - } - var headers = this.getGrid().getHeaderRowCount(); - var usersSize = this.tableState.fixedRowCount || 0; - return headers + usersSize; + getRowCount: function() { + var count = this.getDataSource().getRowCount(); + count += this.grid.getHeaderRowCount(); + return count; }, /** - * @memberOf Behavior.prototype - * @desc Set the number of fixed rows, which includes (top to bottom order): - * 1. The header rows - * 1. The header labels row (optional) - * 2. The filter row (optional) - * 3. The top total rows (0 or more) - * 2. The non-scrolling rows (externally called "the fixed rows") - * - * @returns {number} Sum of the above or 0 if none of the above are in use. - * - * @param {number} n - The number of rows. + * @memberOf dataModels.JSON.prototype + * @returns {string[]} */ - setFixedRowCount: function(n) { - this.tableState.fixedRowCount = n; + getHeaders: function() { + return this.analytics.getHeaders(); }, /** - * @memberOf Behavior.prototype - * @return {number} The number of header rows. - * A portion of the number returned by {@link Behavior#getFixedRowCount()|getFixedRowCount()}. - * (The remaining _fixed rows_ are the _top totals_ rows.) + * @memberOf dataModels.JSON.prototype + * @param {string[]} headers */ - getHeaderRowCount: function() { - var grid = this.getGrid(); - var header = grid.isShowHeaderRow() ? 1 : 0; - var filter = grid.isShowFilterRow() ? 1 : 0; - var totals = this.getTopTotals().length; - return header + filter + totals; + setHeaders: function(headers) { + this.getDataSource().setHeaders(headers); }, /** - * @memberOf Behavior.prototype - * @return {number} The number of footer rows, consisting entirely of 0 or more _bottom totals_ rows. + * @memberOf dataModels.JSON.prototype + * @param {string[]} fields */ - getFooterRowCount: function() { - return this.getBottomTotals().length; + setFields: function(fields) { + this.getDataSource().setFields(fields); }, - getTopTotals: function() { - return this.getDataModel().getTopTotals(); - }, /** - * @memberOf Behavior.prototype - * @summary Set the number of header rows. - * @param {number} n - The number of _fixed rows_ to reserve as header rows. - * (The remaining _fixed rows_ are the _top totals_ rows.) + * @memberOf dataModels.JSON.prototype + * @returns {string[]} */ - setHeaderRowCount: function(n) { - this.tableState.headerRowCount = n; + getFields: function() { + return this.getDataSource().getFields(); }, /** - * @memberOf Behavior.prototype - * @return {number} The number of fixed rows. + * @memberOf dataModels.JSON.prototype + * @param {object[]} dataRows */ - getHeaderColumnCount: function() { - var grid = this.getGrid(); - var count = grid.resolveProperty('headerColumnCount'); - return count; + setData: function(dataRows) { + this.source = new analytics.JSDataSource(dataRows); + //this.preglobalfilter = new analytics.DataSourceGlobalFilter(this.source); + //this.prefilter = new analytics.DataSourceFilter(this.preglobalfilter); + //this.presorter = new analytics.DataSourceSorterComposite(this.prefilter); + + this.analytics = new analytics.DataSourceAggregator(this.source); + + this.postglobalfilter = new analytics.DataSourceGlobalFilter(this.analytics); + this.postfilter = new analytics.DataSourceFilter(this.postglobalfilter); + this.postsorter = new analytics.DataSourceSorterComposite(this.postfilter); + + this.applyAnalytics(); + }, /** - * @memberOf Behavior.prototype - * @param {number} The number of fixed rows. + * @memberOf dataModels.JSON.prototype + * @param {Array} totalRows */ - setHeaderColumnCount: function(numberOfHeaderColumns) { - this.tableState.headerColumnCount = numberOfHeaderColumns; + setTopTotals: function(totalRows) { + this.topTotals = totalRows; }, + /** - * @memberOf Behavior.prototype - * @desc build and open the editor within the container div argument - * @return {boolean} `false` prevents editor from opening - * @param {HTMLDivElement} div - the containing div element + * @memberOf dataModels.JSON.prototype + * @returns {Array} */ - buildColumnPicker: function(div) { - var container = document.createElement('div'); - - var hidden = document.createElement('fin-hypergrid-dnd-list'); - var visible = document.createElement('fin-hypergrid-dnd-list'); - - container.appendChild(hidden); - container.appendChild(visible); - - this.beColumnStyle(hidden.style); - hidden.title = 'hidden columns'; - hidden.list = this.getHiddenColumnDescriptors(); + getTopTotals: function() { + if (!this.hasAggregates()) { + return this.topTotals; + } + return this.getDataSource().getGrandTotals(); + }, - this.beColumnStyle(visible.style); - visible.style.left = '50%'; - visible.title = 'visible columns'; - visible.list = this.getColumnDescriptors(); + /** + * @memberOf dataModels.JSON.prototype + * @param {Array} totalRows + */ + setBottomTotals: function(totalRows) { + this.bottomTotals = totalRows; + }, - div.lists = { - hidden: hidden.list, - visible: visible.list - }; - div.appendChild(container); - return true; + /** + * @memberOf dataModels.JSON.prototype + * @returns {Array} + */ + getBottomTotals: function() { + if (!this.hasAggregates()) { + return this.bottomTotals; + } + return this.getDataSource().getGrandTotals(); }, /** - * @memberOf Behavior.prototype - * @desc the editor is requesting close; deal with the edits - * @return `true` - * @param {HTMLDivElement} div - the containing div element + * @memberOf dataModels.JSON.prototype + * @param groups */ - updateFromColumnPicker: function(div) { - var lists = div.lists; - this.setColumnDescriptors(lists); - return true; + setGroups: function(groups) { + this.analytics.setGroupBys(groups); + this.applyAnalytics(); + this.grid.fireSyntheticGroupsChangedEvent(this.getGroups()); }, /** - * @memberOf Behavior.prototype - * @desc a dnd column has just been dropped, we've been notified + * @memberOf dataModels.JSON.prototype + * @returns {object[]} */ - endDragColumnNotification: function() {}, + getGroups: function() { + var headers = this.getHeaders().slice(0); + var fields = this.getFields().slice(0); + var groupBys = this.analytics.groupBys; + var groups = []; + for (var i = 0; i < groupBys.length; i++) { + var field = headers[groupBys[i]]; + groups.push({ + id: groupBys[i], + label: field, + field: fields + }); + } + return groups; + }, /** - * @memberOf Behavior.prototype - * @desc bind column editor appropriate css values to arg style - * @param {HTMLStyleElement} style - the style object to enhance + * @memberOf dataModels.JSON.prototype + * @returns {object[]} */ - beColumnStyle: function(style) { - style.top = '5%'; - style.position = 'absolute'; - style.width = '50%'; - style.height = '100%'; - style.whiteSpace = 'nowrap'; + getAvailableGroups: function() { + var headers = this.source.getHeaders().slice(0); + var groupBys = this.analytics.groupBys; + var groups = []; + for (var i = 0; i < headers.length; i++) { + if (groupBys.indexOf(i) === -1) { + var field = headers[i]; + groups.push({ + id: i, + label: field, + field: field + }); + } + } + return groups; }, /** - * @memberOf Behavior.prototype - * @return {null} the cursor at a specific x,y coordinate - * @param {number} x - the x coordinate - * @param {number} y - the y coordinate + * @memberOf dataModels.JSON.prototype + * @returns {object[]} */ - getCursorAt: function(x, y) { - return null; + getVisibleColumns: function() { + var items = this.grid.behavior.columns; + items = items.filter(function(each) { + return each.label !== 'Tree'; + }); + return items; }, /** - * @memberOf Behavior.prototype - * @return {number} The total number of columns. + * @memberOf dataModels.JSON.prototype + * @returns {object[]} */ - getColumnCount: function() { - return this.columns.length; + getHiddenColumns: function() { + var visible = this.grid.behavior.columns; + var all = this.grid.behavior.allColumns; + var hidden = []; + for (var i = 0; i < all.length; i++) { + if (visible.indexOf(all[i]) === -1) { + hidden.push(all[i]); + } + } + hidden.sort(function(a, b) { + return a.label < b.label; + }); + return hidden; }, /** - * @memberOf Behavior.prototype - * @return {string} The column alignment at column `x`: `'left'`, `'center'` , or `'right'` - * @param {number} x - The column index of interest. + * @memberOf dataModels.JSON.prototype + * @param aggregations */ - getColumnAlignment: function(x) { - return 'center'; + setAggregates: function(aggregations) { + this.quietlySetAggregates(aggregations); + this.applyAnalytics(); }, /** - * @memberOf Behavior.prototype - * @desc Quietly set the horizontal scroll position. - * @param {number} x - The new position in pixels. + * @memberOf dataModels.JSON.prototype + * @param aggregations */ - setScrollPositionX: function(x) { - this.scrollPositionX = x; + quietlySetAggregates: function(aggregations) { + this.analytics.setAggregates(aggregations); }, - getScrollPositionX: function() { - return this.scrollPositionX; + /** + * @memberOf dataModels.JSON.prototype + * @returns {boolean} + */ + hasHierarchyColumn: function() { + var showTree = this.grid.resolveProperty('showTreeColumn') === true; + return this.hasAggregates() && this.hasGroups() && showTree; }, /** - * @memberOf Behavior.prototype - * @desc Quietly set the vertical scroll position. - * @param {number} y - The new position in pixels. + * @memberOf dataModels.JSON.prototype */ - setScrollPositionY: function(y) { - this.scrollPositionY = y; + applyAnalytics: function(dontApplyGroupBysAndAggregations) { + selectedDataRowsBackingSelectedGridRows.call(this); + + if (!dontApplyGroupBysAndAggregations) { + applyGroupBysAndAggregations.call(this); + } + applyFilters.call(this); + applySorts.call(this); + + reselectGridRowsBackedBySelectedDataRows.call(this); }, - getScrollPositionY: function() { - return this.scrollPositionY; + createFormattedFilter: function(formatter, filter) { + return function(value) { + var formattedValue = formatter(value); + return filter(formattedValue); + }; }, /** - * @memberOf Behavior.prototype - * @return {cellEditor} The cell editor for the cell at cell coordinates `x,y` - * @param {number} x - The horizontal cell coordinate. - * @param {number} y - The vertical cell coordinate. + * @memberOf dataModels.JSON.prototype + * @param {number} colIndex + * @param keys */ - _getCellEditorAt: function(x, y) { - var editor = this.getColumn(x).getCellEditorAt(x, y); - if (editor) { - return editor; + toggleSort: function(colIndex, keys) { + this.incrementSortState(colIndex, keys); + this.applyAnalytics(); + }, + + /** + * @memberOf dataModels.JSON.prototype + * @param {number} colIndex + * @param {string[]} keys + */ + incrementSortState: function(colIndex, keys) { + colIndex++; //hack to get around 0 index + var state = this.getPrivateState(); + var hasCTRL = keys.indexOf('CTRL') > -1; + state.sorts = state.sorts || []; + var already = state.sorts.indexOf(colIndex); + if (already === -1) { + already = state.sorts.indexOf(-1 * colIndex); + } + if (already > -1) { + if (state.sorts[already] > 0) { + state.sorts[already] = -1 * state.sorts[already]; + } else { + state.sorts.splice(already, 1); + } + } else if (hasCTRL || state.sorts.length === 0) { + state.sorts.unshift(colIndex); + } else { + state.sorts.length = 0; + state.sorts.unshift(colIndex); + } + if (state.sorts.length > 3) { + state.sorts.length = 3; } - var grid = this.getGrid(); - var column = this.getColumn(x); - var type = grid.isFilterRow(y) ? column.getFilterType() : column.getType(); - editor = grid.resolveCellEditor(type); - return editor; }, - getCellEditorAt: function(x, y) { - var grid = this.getGrid(); - if (grid.isFilterRow(y)) { - return grid.cellEditors.textfield; + /** + * @memberOf dataModels.JSON.prototype + * @param index + * @param returnAsString + * @returns {*} + */ + getSortImageForColumn: function(index) { + index++; + var up = true; + var sorts = this.getPrivateState().sorts; + if (!sorts) { + return null; } - var editor = this.getDataModel().getCellEditorAt(x, y); - return editor; + var position = sorts.indexOf(index); + if (position < 0) { + position = sorts.indexOf(-1 * index); + up = false; + } + if (position < 0) { + return null; + } + var rank = sorts.length - position; + var arrow = up ? UPWARDS_BLACK_ARROW : DOWNWARDS_BLACK_ARROW; + return rank + arrow + ' '; }, /** - * @memberOf Behavior.prototype - * @param {number} x - The column index. - * @param {string[]} keys + * @memberOf dataModels.JSON.prototype + * @param cell + * @param event */ - toggleSort: function(x, keys) { - this.getColumn(x).toggleSort(keys); + cellClicked: function(cell, event) { + if (!this.hasAggregates()) { + return; + } + if (event.gridCell.x !== 0) { + return; // this wasn't a click on the hierarchy column + } + var headerRowCount = this.grid.getHeaderRowCount(); + var y = event.gridCell.y - headerRowCount; + this.getDataSource().click(y); + this.applyAnalytics(true); + this.changed(); }, /** - * @memberOf Behavior.prototype - * @return {boolean} `true` if we should highlight on hover - * @param {boolean} isColumnHovered - the column is hovered or not - * @param {boolean} isRowHovered - the row is hovered or not + * @memberOf dataModels.JSON.prototype + * @param {number} y + * @returns {object} */ - highlightCellOnHover: function(isColumnHovered, isRowHovered) { - return isColumnHovered && isRowHovered; + getRow: function(y) { + var headerRowCount = this.grid.getHeaderRowCount(); + if (y < headerRowCount && !this.hasAggregates()) { + var topTotals = this.getTopTotals(); + return topTotals[y - (headerRowCount - topTotals.length)]; + } + return this.getDataSource().getRow(y - headerRowCount); }, /** - * @memberOf Behavior.prototype - * @desc this function is a hook and is called just before the painting of a cell occurs - * @param {window.fin.rectangular.Point} cell + * @memberOf dataModels.JSON.prototype + * @param {number} y + * @returns {object} */ - cellPropertiesPrePaintNotification: function(cellProperties) { - var row = this.getRow(cellProperties.y); - var columnId = this.getHeader(cellProperties.x); - cellProperties.row = row; - cellProperties.columnId = columnId; + buildRow: function(y) { + var colCount = this.getColumnCount(); + var fields = [].concat(this.getFields()); + var result = {}; + if (this.hasAggregates()) { + result.tree = this.getValue(-2, y); + fields.shift(); + } + for (var i = 0; i < colCount; i++) { + result[fields[i]] = this.getValue(i, y); + } + return result; }, /** - * @memberOf Behavior.prototype - * @desc this function is a hook and is called just before the painting of a fixed row cell occurs - * @param {window.fin.rectangular.Point} cell + * @memberOf dataModels.JSON.prototype + * @param {number} y + * @returns {object} */ - cellFixedRowPrePaintNotification: function(cell) { - + getComputedRow: function(y) { + var rcf = this.getRowContextFunction([y]); + var fields = this.getFields(); + var row = {}; + for (var i = 0; i < fields.length; i++) { + var field = fields[i]; + row[field] = rcf(field)[0]; + } + return row; }, /** - * @memberOf Behavior.prototype - * @desc this function is a hook and is called just before the painting of a fixed column cell occurs - * @param {window.fin.rectangular.Point} cell + * @memberOf dataModels.JSON.prototype + * @param {string} fieldName + * @param {number} y + * @returns {*} */ - cellFixedColumnPrePaintNotification: function(cell) { - + getValueByField: function(fieldName, y) { + var index = this.getFields().indexOf(fieldName); + if (this.hasAggregates()) { + y += 1; + } + return this.getDataSource().getValue(index, y); }, /** - * @memberOf Behavior.prototype - * @desc this function is a hook and is called just before the painting of a top left cell occurs - * @param {window.fin.rectangular.Point} cell + * @memberOf dataModels.JSON.prototype + * @param {sring} string */ - cellTopLeftPrePaintNotification: function(cell) { - + setGlobalFilter: function(string) { + var globalFilterSource = this.getGlobalFilterSource(); + if (!string || string.length === 0) { + globalFilterSource.clear(); + } else { + globalFilterSource.set(textMatchFilter(string)); + } + this.applyAnalytics(); }, /** - * @memberOf Behavior.prototype - * @desc this function enhance the double click event just before it's broadcast to listeners - * @param {Object} event - event to enhance + * @memberOf dataModels.JSON.prototype + * @param {object} config + * @param {number} x + * @param {number} y + * @param {number} untranslatedX + * @param {number} untranslatedY + * @returns {object} */ - enhanceDoubleClickEvent: function(event) {}, + getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) { + var renderer; + var provider = this.grid.getCellProvider(); - /** - * @memberOf Behavior.prototype - * @desc swap src and tar columns - * @param {number} src - column index - * @param {number} tar - column index - */ - swapColumns: function(source, target) { - var columns = this.columns; - var tmp = columns[source]; - columns[source] = columns[target]; - columns[target] = tmp; - this.changed(); - }, + config.x = x; + config.y = y; + config.untranslatedX = untranslatedX; + config.untranslatedY = untranslatedY; - getColumnEdge: function(c, renderer) { - return this.getDataModel().getColumnEdge(c, renderer); + renderer = provider.getCell(config); + renderer.config = config; + + return renderer; }, /** - * @memberOf Behavior.prototype - * @param {number} x - column index - * @param {number} y - totals row index local to the totals area - * @param value - * @param {boolean} [atBottom=false] - this value is in the "bottom" totals area + * @memberOf dataModels.JSON.prototype */ - setTotalsValue: function(x, y, value, atBottom) { - this.getGrid().setTotalsValueNotification(x, y, value, !!atBottom); + applyState: function() { + this.applyAnalytics(); }, /** - * @memberOf Behavior.prototype - * @return {object} The object at y index. - * @param {number} y - the row index of interest + * @memberOf dataModels.JSON.prototype */ - getRow: function(y) { - return this.getDataModel().getRow(y); + reset: function() { + this.setData([]); }, - convertViewPointToDataPoint: function(viewPoint) { - var newX = this.getColumn(viewPoint.x).index; - var newPoint = this.getGrid().newPoint(newX, viewPoint.y); - return newPoint; + getUnfilteredValue: function(x, y) { + return this.source.getValue(x, y); }, - setGroups: function(arrayOfColumnIndexes) { - this.getDataModel().setGroups(arrayOfColumnIndexes); - this.createColumns(); - this.changed(); + getUnfilteredRowCount: function() { + return this.source.getRowCount(); }, - setAggregates: function(mapOfKeysToFunctions) { - var self = this; - this.getDataModel().setAggregates(mapOfKeysToFunctions); - this.createColumns(); - setTimeout(function() { - self.changed(); - }, 100); - }, +}); - hasHierarchyColumn: function() { - return false; - }, +function valueOrFunctionExecute(valueOrFunction) { + return typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction; +} - getRowContextFunction: function(selectedRows) { - return function() { - return null; - }; - }, +function textMatchFilter(string) { + string = string.toLowerCase(); + return function(each) { + each = valueOrFunctionExecute(each); + return (each + '').toLowerCase().indexOf(string) > -1; + }; +} - getSelectionMatrixFunction: function(selectedRows) { - return function() { - return null; - }; - }, +// LOCAL METHODS -- to be called with `.call(this` - getFieldName: function(index) { - return this.getFields()[index]; - }, +/** + * Accumulate actual data row objects backing current grid row selections. + * This call should be paired with a subsequent call to `reselectGridRowsBackedBySelectedDataRows`. + * @private + * @memberOf dataModels.JSON.prototype + */ +function selectedDataRowsBackingSelectedGridRows() { + //// STEP 1: Accumulate actual data row objects backing current grid row selections. + if (this.grid.selectionModel.hasRowSelections()) { // any current grid row selections? + var filteredData = this.getFilteredData(), + selectedGridRows = this.grid.getSelectedRows(), + selectedData = this.selectedData; + + // 1.a. Remove any filtered data rows from the recently selected list. + selectedData.forEach(function(dataRow, idx) { + if (filteredData.indexOf(dataRow) > 0) { + delete selectedData[idx]; + } + }); - getColumnIndex: function(fieldName) { - return this.getFields().indexOf(fieldName); - }, + // 1.b. Accumulate the data rows backing any currently selected grid rows in `this.selectedData`. + selectedGridRows.forEach(function(selectedRowIndex) { + var dataRow = filteredData[selectedRowIndex]; + if (selectedData.indexOf(dataRow) < 0) { + selectedData.push(dataRow); + } + }); + } +} - getComputedRow: function(y) { - return this.getDataModel().getComputedRow(y); - }, +/** + * Re-establish grid row selections based on actual data row objects accumulated by `selectedDataRowsBackingSelectedGridRows` which should be called first. + * @private + * @memberOf dataModels.JSON.prototype + */ +function reselectGridRowsBackedBySelectedDataRows() { + if (this.selectedData.length) { // any data row objects added from previous grid row selections? + var selectionModel = this.grid.selectionModel, + offset = this.grid.getHeaderRowCount(), + filteredData = this.getFilteredData(); - autosizeAllColumns: function() { - this.checkColumnAutosizing(true); - this.changed(); - }, + selectionModel.clearRowSelection(); - checkColumnAutosizing: function(force) { - force = force === true; - this.autoSizeRowNumberColumn(); - this.allColumns[-2].checkColumnAutosizing(force); - this.allColumns.forEach(function(column) { - column.checkColumnAutosizing(force); + this.selectedData.forEach(function(dataRow) { + var index = filteredData.indexOf(dataRow); + if (index >= 0) { + selectionModel.selectRow(offset + index); + } }); - }, + } +} - autoSizeRowNumberColumn: function() { - if (this.getGrid().isRowNumberAutosizing()) { - this.allColumns[-1].checkColumnAutosizing(true); +/** + * @private + * @memberOf dataModels.JSON.prototype + */ +function applyGroupBysAndAggregations() { + if (this.analytics.aggregates.length === 0) { + this.quietlySetAggregates({}); + } + this.analytics.apply(); +} + +/** + * @private + * @memberOf dataModels.JSON.prototype + */ +function applyFilters() { + var visibleColumns = this.getVisibleColumns(); + this.getGlobalFilterSource().apply(visibleColumns); + var details = []; + var filterSource = this.getFilterSource(); + var groupOffset = 0; //this.hasHierarchyColumn() ? 0 : 1; + + // apply column filters + filterSource.clearAll(); + + visibleColumns.forEach(function(column) { + var columnIndex = column.index, + filterText = this.getFilter(columnIndex), + formatterType = column.getProperties().format, + formatter = this.grid.getFormatter(formatterType), + complexFilter = this.getComplexFilter(columnIndex), + filter = complexFilter || filterText.length > 0 && textMatchFilter(filterText); + + if (filter) { + filterSource.add(columnIndex - groupOffset, this.createFormattedFilter(formatter, filter)); + details.push({ + column: column.label, + format: complexFilter ? 'complex' : formatterType + }); } - }, + }.bind(this)); - setGlobalFilter: function(string) { - this.getDataModel().setGlobalFilter(string); - }, + filterSource.applyAll(); - getSelectedRows: function() { - return this.getGrid().getSelectionModel().getSelectedRows(); - }, + this.grid.fireSyntheticFilterAppliedEvent({ + details: details + }); +} - getSelectedColumns: function() { - return this.getGrid().getSelectionModel().getSelectedColumns(); - }, +/** + * @private + * @memberOf dataModels.JSON.prototype + */ +function applySorts() { + var sortingSource = this.getSortingSource(); + var sorts = this.getPrivateState().sorts; + var groupOffset = this.hasAggregates() ? 1 : 0; + if (!sorts || sorts.length === 0) { + sortingSource.clearSorts(); + } else { + for (var i = 0; i < sorts.length; i++) { + var colIndex = Math.abs(sorts[i]) - 1; + var type = sorts[i] < 0 ? -1 : 1; + sortingSource.sortOn(colIndex - groupOffset, type); + } + } + sortingSource.applySorts(); +} - getSelections: function() { - return this.getGrid().getSelectionModel().getSelections(); - }, +module.exports = JSON; - getData: function() { - return this.getDataModel().getData(); - }, +},{"../../images":3,"../local_node_modules/finanalytics":90,"./DataModel":45}],47:[function(require,module,exports){ +/* eslint-env browser */ - getFilteredData: function() { - return this.getDataModel().getFilteredData(); - }, -}); +'use strict'; -module.exports = Behavior; +var LRUCache = require('lru-cache'); + +var renderCellError = require('./lib/renderCellError'); + +/** + * This module lists the properties that can be set on a {@link Hypergrid} along with their default values. + * Edit this file to override the defaults. + * @module defaults + */ -},{"../CellProvider":23,"./Column":31,"extend-me":4,"object-iterators":18}],31:[function(require,module,exports){ -/* eslint-env browser */ +module.exports = { -'use strict'; + //these are for the theme -var _ = require('object-iterators'); -function Column(behavior, index, label) { - this.behavior = behavior; - this.dataModel = behavior.getDataModel(); - this.index = index; - this.label = label; -} + /** + * The font for data cells. + * @default '13px Tahoma, Geneva, sans-serif' + * @type {cssFont} + * @instance + */ + font: '13px Tahoma, Geneva, sans-serif', -Column.prototype = { - constructor: Column.prototype.constructor, + /** + * Font color for data cells. + * @default 'rgb(25, 25, 25)' + * @type {string} + * @instance + */ + color: 'rgb(25, 25, 25)', - getUnfilteredValue: function(y) { - return this.dataModel.getUnfilteredValue(this.index, y); - }, + /** + * Background color for data cells. + * @default 'rgb(241, 241, 241)' + * @type {string} + * @instance + */ + backgroundColor: 'rgb(241, 241, 241)', - getValue: function(y) { - return this.dataModel.getValue(this.index, y); - }, + /** + * Font color for selected cell(s). + * @default 'rgb(25, 25, 25)' + * @type {string} + * @instance + */ + foregroundSelectionColor: 'black', - setValue: function(y, value) { - return this.dataModel.setValue(this.index, y, value); - }, + /** + * Background color for selected cell(s). + * @default 'rgba(147, 185, 255, 0.45)' + * @type {string} + * @instance + */ + backgroundSelectionColor: 'rgba(147, 185, 255, 0.45)', - getWidth: function() { - var properties = this.getProperties(); - if (properties) { - var override = properties.width; - if (override) { - return override; - } - } - return this.behavior.resolveProperty('defaultColumnWidth'); - }, - setWidth: function(width) { - this.getProperties().width = Math.max(5, width); - }, + /********** SECTION: COLUMN HEADER COLORS **********/ - getCellRenderer: function(config, y) { - return this.dataModel.getCellRenderer(config, this.index, y); - }, + // IMPORTANT CAVEAT: The code is inconsistent regarding the terminology. Is the "column header" section _the row_ of cells at the top (that act as headers for each column) or is it _the column_ of cells (that act as headers for each row)? Oh my. - getCellProperties: function(y) { - return this.behavior.getPrivateState().cellProperties[this.index + ',' + y]; - }, + /** + * @default '12px Tahoma, Geneva, sans-serif' + * @type {cssFont} + * @instance + */ + columnHeaderFont: '12px Tahoma, Geneva, sans-serif', - setCellProperties: function(y, value) { - this.behavior.getPrivateState().cellProperties[this.index + ',' + y] = value; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + columnHeaderColor: 'rgb(25, 25, 25)', - setComplexFilter: function(data) { - this.getProperties().complexFilter = data; - }, + /** + * @default 'rgb(223, 227, 232)' + * @type {cssColor} + * @instance + */ + columnHeaderBackgroundColor: 'rgb(223, 227, 232)', - getComplexFilter: function() { - return this.getProperties().complexFilter; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + columnHeaderForegroundSelectionColor: 'rgb(25, 25, 25)', - checkColumnAutosizing: function(force) { - var properties = this.getProperties(); - var a, b, d; - if (properties) { - a = properties.width; - b = properties.preferredWidth || properties.width; - d = properties.columnAutosized && !force; - if (a !== b || !d) { - properties.width = !d ? b : Math.max(a, b); - properties.columnAutosized = !isNaN(properties.width); - } - } - }, + /** + * @default 'rgba(255, 220, 97, 0.45)' + * @type {cssColor} + * @instance + */ + columnHeaderBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)', - getCellType: function(y) { - var value = this.getValue(y); - var type = this.typeOf(value); - return type; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + columnHeaderForegroundColumnSelectionColor: 'rgb(25, 25, 25)', - getFilterType: function() { - // var props = this.getProperties(); - // var type = props.filterType; - // if (!type) { - // type = this.getType(); - // if (type !== 'unkknown') { - // props.type = type; - // } - // } - // return type; - return 'filter'; - }, + /** + * @default 'rgb(255, 180, 0)' + * @type {cssColor} + * @instance + */ + columnHeaderBackgroundColumnSelectionColor: 'rgb(255, 180, 0)', - getType: function() { - var props = this.getProperties(); - var type = props.type; - if (!type) { - type = this.computeColumnType(); - if (type !== 'unkknown') { - props.type = type; - } - } - return type; - }, - computeColumnType: function() { - var headerRowCount = this.behavior.getHeaderRowCount(); - var height = this.behavior.getRowCount(); - var value = this.getValue(headerRowCount); - var eachType = this.typeOf(value); - if (!eachType) { - return 'unknown'; - } - var type = this.typeOf(value); - var isNumber = ((typeof value) === 'number'); - for (var y = headerRowCount; y < height; y++) { - value = this.getValue(y); - eachType = this.typeOf(value); - if (type !== eachType) { - if (isNumber && (typeof value === 'number')) { - type = 'float'; - } else { - return 'mixed'; - } - } - } - return type; - }, + /********** SECTION: ROW HEADER COLORS **********/ - typeOf: function(something) { - var typeOf = typeof something; - switch (typeOf) { - case 'object': - return something.constructor.name.toLowerCase(); - case 'number': - return parseInt(something) === something ? 'int' : 'float'; - default: - return typeOf; - } - }, + /** + * @default '12px Tahoma, Geneva, sans-serif' + * @type {cssFont} + * @instance + */ + rowHeaderFont: '12px Tahoma, Geneva, sans-serif', - getProperties: function() { - return this.behavior.getPrivateState().columnProperties[this.index]; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + rowHeaderColor: 'rgb(25, 25, 25)', - setProperties: function(properties) { - var current = this.behavior.getPrivateState().columnProperties[this.index]; - this.clearObjectProperties(current, false); - _(current).extendOwn(properties); - }, + /** + * @default 'rgb(223, 227, 232)' + * @type {cssColor} + * @instance + */ + rowHeaderBackgroundColor: 'rgb(223, 227, 232)', - toggleSort: function(keys) { - this.dataModel.toggleSort(this.index, keys); - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + rowHeaderForegroundSelectionColor: 'rgb(25, 25, 25)', - getCellEditorAt: function(x, y) { - return this.dataModel.getCellEditorAt(this.index, y); - }, + /** + * @default 'rgba(255, 220, 97, 0.45)' + * @type {cssColor} + * @instance + */ + rowHeaderBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)', - getHeader: function() { - return this.label; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + rowHeaderForegroundRowSelectionColor: 'rgb(25, 25, 25)', - getField: function() { - return this.dataModel.getFields()[this.index]; - } -}; + /** + * @default 'rgb(255, 180, 0)' + * @type {cssColor} + * @instance + */ + rowHeaderBackgroundRowSelectionColor: 'rgb(255, 180, 0)', -module.exports = Column; -},{"object-iterators":18}],32:[function(require,module,exports){ -'use strict'; + /********** SECTION: FILTER ROW COLORS **********/ -function DataModelDecorator(grid, component) { - this.setComponent(component); - this.setGrid(grid); -} + /** + * @default '12px Tahoma, Geneva, sans-serif' + * @type {cssFont} + * @instance + */ + filterFont: '12px Tahoma, Geneva, sans-serif', -DataModelDecorator.prototype = { - constructor: DataModelDecorator.prototype.constructor, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + filterColor: 'rgb(25, 25, 25)', - component: null, - grid: null, + /** + * @default 'white' + * @type {cssColor} + * @instance + */ + filterBackgroundColor: 'white', - getGrid: function() { - return this.grid; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + filterForegroundSelectionColor: 'rgb(25, 25, 25)', - setGrid: function(newGrid) { - this.grid = newGrid; - this.getComponent().setGrid(newGrid); - }, + /** + * @default 'rgb(255, 220, 97)' + * @type {cssColor} + * @instance + */ + filterBackgroundSelectionColor: 'rgb(255, 220, 97)', - getBehavior: function() { - return this.getGrid().getBehavior(); - }, + /** + * @default 'rgba(0,0,0,0.8)' + * @type {cssColor} + * @instance + */ + filterCellBorderStyle: 'rgba(0,0,0,0.8)', - changed: function() { - this.getBehavior().changed(); - }, + /** + * @default 0.4 + * @type {number} + * @instance + */ + filterCellBorderThickness: 0.4, - getPrivateState: function() { - return this.getGrid().getPrivateState(); - }, - applyState: function() { + /********** SECTION: TREE COLUMN COLORS **********/ + // The "tree column" contains the hierarchical drill-down controls. - }, + /** + * @default '12px Tahoma, Geneva, sans-serif' + * @type {cssFont} + * @instance + */ + treeColumnFont: '12px Tahoma, Geneva, sans-serif', - setComponent: function(newComponent) { - this.component = newComponent; - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + treeColumnColor: 'rgb(25, 25, 25)', - getComponent: function() { - return this.component; - }, + /** + * @default 'rgb(223, 227, 232)' + * @type {cssColor} + * @instance + */ + treeColumnBackgroundColor: 'rgb(223, 227, 232)', - setGlobalFilter: function(string) { - return this.getComponent().setGlobalFilter(string); - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + treeColumnForegroundSelectionColor: 'rgb(25, 25, 25)', - getData: function() { - return this.getComponent().getData(); - }, + /** + * @default 'rgba(255, 220, 97, 0.45)' + * @type {cssColor} + * @instance + */ + treeColumnBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)', - getFilteredData: function() { - return this.getComponent().getFilteredData(); - }, + /** + * @default 'rgb(25, 25, 25)' + * @type {cssColor} + * @instance + */ + treeColumnForegroundColumnSelectionColor: 'rgb(25, 25, 25)', - getValue: function(x, y) { - return this.getComponent().getValue(x, y); - }, + /** + * @default 'rgb(255, 180, 0)' + * @type {cssColor} + * @instance + */ + treeColumnBackgroundColumnSelectionColor: 'rgb(255, 180, 0)', - getUnfilteredValue: function(x, y) { - return this.getComponent().getUnfilteredValue(x, y); - }, + /** + * @default 'rgb(201, 201, 201)' + * @type {cssColor} + * @instance + */ + backgroundColor2: 'rgb(201, 201, 201)', - setValue: function(x, y, value) { - this.getComponent().setValue(x, y, value); - }, + /** + * @default 0 + * @type {number} + * @instance + */ + voffset: 0, - getColumnCount: function() { - return this.getComponent().getColumnCount(); - }, + /** + * @default 'visible' + * @type {string} + * @instance + */ + scrollbarHoverOver: 'visible', - applyFilters: function() { - return this.getComponent().applyFilters(); - }, + /** + * @default 'hidden' + * @type {string} + * @instance + */ + scrollbarHoverOff: 'hidden', - getRowCount: function() { - return this.getComponent().getRowCount(); - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + scrollingEnabled: true, - getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) { - return this.getComponent().getCellRenderer(config, x, y, untranslatedX, untranslatedY); - }, + /** + * @default '' + * @type {string} + * @instance + */ + vScrollbarClassPrefix: '', - getRowHeight: function(y) { - return this.getComponent().getRowHeight(y); - }, + /** + * @default '' + * @type {string} + * @instance + */ + hScrollbarClassPrefix: '', - getColumnEdge: function(x, renderer) { - return this.getComponent().getColumnEdge(x, renderer); - }, + //these used to be in the constants element - getColumnWidth: function(x) { - return this.getComponent().getColumnWidth(x); - }, + /** + * @default 'center' + * @type {string} + * @instance + */ + fixedRowAlign: 'center', - setColumnWidth: function(x, width) { - this.getComponent().setColumnWidth(x, width); - }, + /** + * @default 'center' + * @type {string} + * @instance + */ + fixedColAlign: 'center', - toggleSort: function(x, keys) { - this.getComponent().toggleSort(x, keys); - }, + /** + * @default 5 + * @type {number} + * @instance + */ + cellPadding: 5, - getColumnProperties: function(columnIndex) { - return this.getComponent().getColumnProperties(columnIndex); - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + gridLinesH: true, - setColumnProperties: function(columnIndex, properties) { - this.getComponent().setColumnProperties(columnIndex, properties); - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + gridLinesV: true, - getHeaders: function() { - return this.getComponent().getHeaders(); - }, + /** + * @default 'rgb(199, 199 199)' + * @type {cssColor} + * @instance + */ + lineColor: 'rgb(199, 199, 199)', - getFields: function() { - return this.getComponent().getFields(); - }, + /** + * @default 0.4 + * @type {number} + * @instance + */ + lineWidth: 0.4, - setFields: function(fields) { - this.getComponent().setFields(fields); - }, - getCellProperties: function(x, y) { - return this.getComponent().getCellProperties(x, y); - }, + /** + * @default 15 + * @type {number} + * @instance + */ + defaultRowHeight: 15, - setCellProperties: function(x, y, value) { - this.getComponent().setCellProperties(x, y, value); - }, + /** + * @default 100 + * @type {number} + * @instance + */ + defaultColumnWidth: 100, - getRow: function(y) { - return this.getComponent().getRow(y); - }, + //for immediate painting, set these values to 0, true respectively - getTopTotals: function() { - return this.getComponent().getTopTotals(); - }, + /** + * @default 60 + * @type {number} + * @instance + */ + repaintIntervalRate: 60, + + /** + * @default `false` + * @type {boolean} + * @instance + */ + repaintImmediately: false, - setTopTotals: function(totalRows) { - this.getComponent().setTopTotals(totalRows); - }, + //enable or disable double buffering - getBottomTotals: function() { - return this.getComponent().getBottomTotals(); - }, + /** + * @default `false` + * @type {boolean} + * @instance + */ + useBitBlit: false, - setBottomTotals: function(totalRows) { - this.getComponent().setBottomTotals(totalRows); - }, - setData: function(y) { - return this.getComponent().setData(y); - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + useHiDPI: true, - hasHierarchyColumn: function() { - return this.getComponent().hasHierarchyColumn(); - }, + /** + * @default ['alt', 'esc'] + * @type {string} + * @instance + */ + editorActivationKeys: ['alt', 'esc'], - setHeaders: function(headerLabels) { - return this.getComponent().setHeaders(headerLabels); - }, + /** + * @default `false` + * @type {boolean} + * @instance + */ + readOnly: false, - cellClicked: function(cell, event) { - return this.getComponent().cellClicked(cell, event); - }, + // inherited by cell renderers - getAvailableGroups: function() { - return this.getComponent().getAvailableGroups(); - }, + /** + * @default getTextWidth + * @type {function} + * @instance + */ + getTextWidth: getTextWidth, - getGroups: function() { - return this.getComponent().getGroups(); - }, + /** + * @default getTextHeight + * @type {function} + * @instance + */ + getTextHeight: getTextHeight, - setGroups: function(groups) { - this.getComponent().setGroups(groups); - }, - getHiddenColumns: function() { - return this.getComponent().getHiddenColumns(); - }, + /** + * @default 0 + * @type {number} + * @instance + */ + fixedColumnCount: 0, - getVisibleColumns: function() { - return this.getComponent().getVisibleColumns(); - }, + /** + * @default 0 + * @type {number} + * @instance + */ + fixedRowCount: 0, - setAggregates: function(aggregates) { - return this.getComponent().setAggregates(aggregates); - }, + /** + * @default 0 + * @type {number} + * @instance + */ + headerColumnCount: 0, - reset: function() { - this.getComponent().reset(); - }, - getCellEditorAt: function(x, y) { - return this.getComponent().getCellEditorAt(x, y); - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + showRowNumbers: true, - getUnfilteredRowCount: function() { - return this.getComponent().getUnfilteredRowCount(); - } -}; + /** + * @default `true` + * @type {boolean} + * @instance + */ + showTreeColumn: true, -module.exports = DataModelDecorator; + /** + * @default `true` + * @type {boolean} + * @instance + */ + showHeaderRow: true, -},{}],33:[function(require,module,exports){ -'use strict'; + /** + * @default `true` + * @type {boolean} + * @instance + */ + showFilterRow: true, -var ListDragon = require('list-dragon'); -var Behavior = require('./Behavior'); -var DataModelDecorator = require('./DataModelDecorator'); -var DataModelJSON = require('../dataModels/JSON'); -var features = require('../features/index'); -var addStylesheet = require('../stylesheets'); -//var aggregations = require('hyper-analytics').util.aggregations; -//var aggregations = require('../local_node_modules/hyper-analytics').util.aggregations; -var aggregations = require('../local_node_modules/finanalytics').aggregations; + /** Clicking in a cell "selects" it; it is added to the select region and repainted with "cell selection" colors. + * @default `true` + * @type {boolean} + * @instance + */ + cellSelection: true, -/** - * @name behaviors.JSON - * @desc > Same parameters as {@link behaviors.JSON#initialize|initialize}, which is called by this constructor. - * @constructor - */ -var JSON = Behavior.extend('behaviors.JSON', { + /** Clicking in a row header (leftmost column) "selects" the row; the entire row is added to the select region and repainted with "row selection" colors. + * @default `true` + * @type {boolean} + * @instance + */ + columnSelection: true, + + /** Clicking in a column header (top row) "selects" the column; the entire column is added to the select region and repainted with "column selection" colors. + * @default `true` + * @type {boolean} + * @instance + */ + rowSelection: true, /** - * @summary Constructor logic, called _after_{@link Behavior#initialize|Behavior.initialize()}. - * @desc This method will be called upon instantiation of this class or of any class that extends from this class. - * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most "senior" class through that of the class of the new instance. - * - * @param grid - the hypergrid - * @param {object[]} dataRows - array of uniform data objects - * @memberOf behaviors.JSON.prototype + * @default `true` + * @type {boolean} + * @instance */ - initialize: function(grid, dataRows) { - this.setData(dataRows); - }, + singleRowSelectionMode: true, - features: [ - features.CellSelection, - features.KeyPaging, - features.ColumnPicker, - features.ColumnResizing, - features.RowResizing, - features.Filters, - features.RowSelection, - features.ColumnSelection, - features.ColumnMoving, - features.ColumnSorting, - features.CellEditing, - features.CellClick, - features.OnHover - ], + /** + * @default 'rgba(0, 0, 0, 0.2)' + * @type {cssColor} + * @instance + */ + selectionRegionOverlayColor: 'rgba(0, 0, 0, 0.2)', - aggregations: aggregations, + /** + * @default 'black' + * @type {string} + * @instance + */ + selectionRegionOutlineColor: 'black', - createColumns: function() { - var dataModel = this.getDataModel(); - var columnCount = dataModel.getColumnCount(); - var headers = dataModel.getHeaders(); - var fields = dataModel.getFields(); - this.clearColumns(); - for (var i = 0; i < columnCount; i++) { - var header = headers[i]; - var column = this.addColumn(i, header); - var properties = column.getProperties(); - properties.field = fields[i]; - properties.header = header; - properties.complexFilter = null; - } - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + columnAutosizing: true, - getDefaultDataModel: function() { - var model = new DataModelJSON(); - var wrapper = new DataModelDecorator(this.getGrid(), model); - wrapper.setComponent(model); - return wrapper; - }, + /** + * @default `true` + * @type {boolean} + * @instance + */ + rowNumberAutosizing: true, - applyFilters: function() { - this.dataModel.applyFilters(); - }, + /** + * @default `false` + * @type {boolean} + * @instance + */ + headerTextWrapping: false, /** - * @memberOf behaviors.JSON.prototype - * @description Set the header labels. - * @param {string[]} headerLabels - The header labels. + * @default `false` + * @type {boolean} + * @instance */ - setHeaders: function(headerLabels) { - this.getDataModel().setHeaders(headerLabels); - }, + rowResize: false, /** - * @memberOf behaviors.JSON.prototype - * @desc * @returns {string[]} The header labels. + * @default `true` + * @type {boolean} + * @instance */ - getHeaders: function() { - return this.getDataModel().getHeaders(); - }, + editable: true, /** - * @memberOf behaviors.JSON.prototype - * @description Set the fields array. - * @param {string[]} fieldNames - The field names. + * @default `true` + * @type {boolean} + * @instance */ - setFields: function(fieldNames) { - //were defining the columns based on field names.... - //we must rebuild the column definitions - this.getDataModel().setFields(fieldNames); - this.createColumns(); - }, + editOnDoubleClick: true, + + /** + * @default 325 + * @type {number} + * @instance + */ + doubleClickDelay: 325, /** - * @memberOf behaviors.JSON.prototype - * @description Get the field names. - * @returns {string[]} + * Grid-level property. + * When user presses a printable character key _or_ BACKSPACE _or_ DELETE: + * 1. Activate cell editor on current cell (i.e., origin of most recent selection). + * 2. If cell editor is a text editor: + * 1. Replace current value with the character the user typed; or + * 2. Clear it on BACKSPACE, DELETE, or other invalid character (_e.g._ when user types a letter but the cell editor only accepts digits). + * + * > In invoked, user has the option to back out by pressing the ESCAPE key. + * + * @default `true` + * @type {boolean} + * @instance */ - getFields: function() { - return this.getDataModel().getFields(); - }, + editOnKeydown: true, /** - * @memberOf behaviors.JSON.prototype - * @description Set the data field. - * @param {object[]} objects - An array of uniform objects, each being a row in the grid. + * @default renderCellError + * @type {function} */ - setData: function(dataRows) { - this.getDataModel().setData(dataRows); - this.createColumns(); - var self = this; - if (this.getGrid().isColumnAutosizing()) { - setTimeout(function() { - self.autosizeAllColumns(); - }, 100); - self.changed(); - } else { - setTimeout(function() { - self.allColumns[-1].checkColumnAutosizing(true); - self.changed(); - }); - } - }, + renderCellError: renderCellError, /** - * @summary Set the top totals. - * @memberOf behaviors.JSON.prototype - * @param {Array} totalRows - array of rows (arrays) of totals + * @default `false` + * @type {boolean} */ - setTopTotals: function(totalRows) { - this.getDataModel().setTopTotals(totalRows); - }, + checkboxOnlyRowSelections: false, - /** - * @summary Get the top totals. - * @memberOf behaviors.JSON.prototype - * @returns {Array} + /** Name of a formatters for cell text. + * @see /src/Formatters.js */ - getTopTotals: function() { - return this.getDataModel().getTopTotals(); - }, + format: 'default', - /** - * @summary Set the bottom totals. - * @memberOf behaviors.JSON.prototype - * @param {Array} totalRows - array of rows (arrays) of totals + /********** HOVER COLORS **********/ + + /** @typedef hoverColors + * @property {boolean} [enable=false] - `false` means not hilite on hover + * @property {cssColor} backgroundColor - cell, row, or colummn background color. Alpha channel will be respected and if given will be painted over the cells predetermined color. + * @property {cssColor} [header.backgroundColor=backgroundColor] - for columns and rows, this is the background color of the column or row "handle" (header rows or columns, respectively). (Not used for cells.) */ - setBottomTotals: function(totalRows) { - this.getDataModel().setBottomTotals(totalRows); - }, - /** - * @summary Get the bottom totals. - * @memberOf behaviors.JSON.prototype - * @returns {Array} + /** On mouse hover, whether to repaint the cell background and how. + * @type {hoverColors} + * @default '{ enabled: true, background: rgba(160, 160, 40, 0.30) }' */ - getBottomTotals: function() { - return this.getDataModel().getBottomTotals(); + hoverCellHighlight: { + enabled: true, + backgroundColor: 'rgba(160, 160, 40, 0.45)' }, - /** - * @memberOf behaviors.JSON.prototype - * @description Build the fields and headers from the supplied column definitions. - * ```javascript - * myJsonBehavior.setColumns([ - * { title: 'Stock Name', field: 'short_description' }, - * { title: 'Status', field: 'trading_phase' }, - * { title: 'Reference Price', field: 'reference_price' } - * ]); - * ``` - * @param {Array} columnDefinitions - an array of objects with fields 'title', and 'field' + /** On mouse hover, whether to repaint the row background and how. + * @type {hoverColors} + * @default '{ enabled: true, background: rgba(100, 100, 25, 0.15) }' */ - setColumns: function(columnDefinitions) { - this.getDataModel().setColumns(columnDefinitions); + hoverRowHighlight: { + enabled: true, + backgroundColor: 'rgba(100, 100, 25, 0.30)' + }, - /** - * @memberOf behaviors.JSON.prototype - * @description Enhance the double-click event just before it's broadcast to listeners. - * @param {Point} event + /** On mouse hover, whether to repaint the column background and how. + * @type {hoverColors} + * @default '{ enabled: true, background: rgba(60, 60, 15, 0.15) }' */ - enhanceDoubleClickEvent: function(event) { - event.row = this.getRow(event.gridCell.y); + hoverColumnHighlight: { + enabled: true, + backgroundColor: 'rgba(60, 60, 15, 0.15)' }, - setDataProvider: function(dataProvider) { - this.getDataModel().setDataProvider(dataProvider); - }, - hasHierarchyColumn: function() { - return this.getDataModel().hasHierarchyColumn(); - }, + /** Display cell font with under-score line drawn over it. + * > Implementation of links right now is not automatic; you must attach a 'fin-click' listener to the hypergrid object, etc. + * @type {boolean} + * @default `false` + */ + link: false, - getColumnAlignment: function(x) { - if (x === 0 && this.hasHierarchyColumn()) { - return 'left'; - } else { - return 'center'; - } - }, + /** Display cell font with strike-through line drawn over it. + * @type {boolean} + * @default `false` + */ + strikeThrough: false, - getRowSelectionMatrix: function(selectedRows) { - return this.getDataModel().getRowSelectionMatrix(selectedRows); - }, +}; - getColumnSelectionMatrix: function(selectedColumns) { - return this.getDataModel().getColumnSelectionMatrix(selectedColumns); - }, +/** @typedef {string} cssColor + * @see https://developer.mozilla.org/en-US/docs/Web/CSS/color_value + */ +/** @typedef {string} cssFont + * @see https://developer.mozilla.org/en-US/docs/Web/CSS/font + */ - getSelectionMatrix: function(selections) { - return this.getDataModel().getSelectionMatrix(selections); - }, +var textWidthCache = new LRUCache(2000); - getRowSelection: function() { - var selectedRows = this.getSelectedRows(); - return this.getDataModel().getRowSelection(selectedRows); - }, +function getTextWidth(gc, string) { + if (string === null || string === undefined) { + return 0; + } + string = string + ''; + if (string.length === 0) { + return 0; + } + var key = gc.font + string; + var width = textWidthCache.get(key); + if (!width) { + width = gc.measureText(string).width; + textWidthCache.set(key, width); + } + return width; +} - getColumnSelection: function() { - var selectedColumns = this.getSelectedColumns(); - return this.getDataModel().getColumnSelection(selectedColumns); - }, +var fontData = {}; - getSelection: function() { - var selections = this.getSelections(); - return this.getDataModel().getSelection(selections); - }, +function getTextHeight(font) { + var result = fontData[font]; - buildColumnPicker: function(div) { - if (!this.isColumnReorderable()) { - return false; - } + if (!result) { + result = {}; - var listOptions = { - cssStylesheetReferenceElement: div - }; + var text = document.createElement('span'); + text.textContent = 'Hg'; + text.style.font = font; - var groups = { models: this.getGroups(), title: 'Groups' }, - availableGroups = { models: this.getAvailableGroups(), title: 'Available Groups' }, - hiddenColumns = { models: this.getHiddenColumns(), title: 'Hidden Columns' }, - visibleColumns = { models: this.getVisibleColumns(), title: 'Visible Columns'}, - groupLists = new ListDragon([groups, availableGroups], listOptions), - columnLists = new ListDragon([hiddenColumns, visibleColumns], listOptions), - listSets = [groupLists, columnLists]; + var block = document.createElement('div'); + block.style.display = 'inline-block'; + block.style.width = '1px'; + block.style.height = '0px'; - addStylesheet('list-dragon', div); + var div = document.createElement('div'); + div.appendChild(text); + div.appendChild(block); - listSets.forEach(function(listSet) { - listSet.modelLists.forEach(function(list) { - div.appendChild(list.container); - }); - }); + div.style.position = 'absolute'; + document.body.appendChild(div); - //attach for later retrieval - div.lists = { - group: groups.models, - availableGroups: availableGroups.models, - hidden: hiddenColumns.models, - visible: visibleColumns.models - }; + try { - return true; - }, - getGroups: function() { - return this.getDataModel().getGroups(); - }, - getAvailableGroups: function() { - return this.getDataModel().getAvailableGroups(); - }, - getHiddenColumns: function() { - return this.getDataModel().getHiddenColumns(); - }, - getVisibleColumns: function() { - return this.getDataModel().getVisibleColumns(); - }, - setColumnDescriptors: function(lists) { - //assumes there is one row.... - var tree = this.columns[0]; - this.columns.length = 0; - if (tree && tree.label === 'Tree') { - this.columns.push(tree); + block.style.verticalAlign = 'baseline'; + + var blockRect = block.getBoundingClientRect(); + var textRect = text.getBoundingClientRect(); + + result.ascent = blockRect.top - textRect.top; + + block.style.verticalAlign = 'bottom'; + result.height = blockRect.top - textRect.top; + + result.descent = result.height - result.ascent; + + } finally { + document.body.removeChild(div); } - for (var i = 0; i < lists.visible.length; i++) { - this.columns.push(lists.visible[i]); + if (result.height !== 0) { + fontData[font] = result; } + } - var groupBys = lists.group.map(function(e) { - return e.id; - }); - this.getDataModel().setGroups(groupBys); + return result; +} - this.changed(); - }, +},{"./lib/renderCellError":75,"lru-cache":19}],48:[function(require,module,exports){ +/* eslint-env browser */ - getSelectedRows: function() { - var offset = -this.getGrid().getHeaderRowCount(); - var selections = this.getGrid().getSelectionModel().getSelectedRows(); - var result = selections.map(function(each) { - return each + offset; - }); - return result; - }, +'use strict'; + +require('object-iterators'); // Installs the Array.find polyfill, as needed + +var Hypergrid = require('./Hypergrid'); + +Hypergrid.images = require('../images'); +Hypergrid.behaviors = require('./behaviors/index'); +Hypergrid.cellEditors = require('./cellEditors/index'); +Hypergrid.features = require('./features/index'); + +(window.fin = window.fin || {}).Hypergrid = Hypergrid; + +},{"../images":3,"./Hypergrid":27,"./behaviors/index":34,"./cellEditors/index":44,"./features/index":65,"object-iterators":21}],49:[function(require,module,exports){ +'use strict'; + +var Feature = require('./Feature.js'); + +/** + * @constructor + */ +var CellClick = Feature.extend('CellClick', { - getSelectedColumns: function() { - return this.getGrid().getSelectionModel().getSelectedColumns(); - }, + alias: 'CellClick', - getSelections: function() { - return this.getGrid().getSelectionModel().getSelections(); + /** + * @memberOf CellClick.prototype + * @desc Handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleTap: function(grid, event) { + if ( + event.gridCell.y >= grid.behavior.getHeaderRowCount() && + event.gridCell.x >= grid.behavior.getHeaderColumnCount() + ) { + grid.cellClicked(event); + } else if (this.next) { + this.next.handleTap(grid, event); + } } - }); -module.exports = JSON; +module.exports = CellClick; -},{"../dataModels/JSON":47,"../features/index":66,"../local_node_modules/finanalytics":81,"../stylesheets":84,"./Behavior":30,"./DataModelDecorator":32,"list-dragon":15}],34:[function(require,module,exports){ +},{"./Feature.js":58}],50:[function(require,module,exports){ 'use strict'; -var Behavior = require('./Behavior'); - -var noop = function() {}, - n00p = function() { return 0; }; +var Feature = require('./Feature.js'); /** * @constructor */ -var Null = Behavior.extend('Null', { +var CellEditing = Feature.extend('CellEditing', { - //initalize: function(grid, component) {}, + alias: 'CellEditing', - setScrollPositionY: noop, - setScrollPositionX: noop, - getColumnCount: n00p, - getFixedColumnCount: n00p, - getFixedColumnsWidth: n00p, - getFixedColumnsMaxWidth: n00p, - setRenderedWidth: n00p, - getRowCount: n00p, - getFixedRowCount: n00p, - getFixedRowsHeight: n00p, - getFixedRowsMaxHeight: n00p, - setRenderedHeight: n00p, - getCellProvider: noop, - click: noop, - doubleClick: noop -}); + /** + * @memberOf CellEditing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleDoubleClick: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleDoubleClick(grid, event); + } + }, -module.exports = Null; + handleTap: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleTap(grid, event); + } + }, -},{"./Behavior":30}],35:[function(require,module,exports){ -'use strict'; + /** + * @memberOf CellEditing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleHoldPulse: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleHoldPulse(grid, event); + } + }, -module.exports = { - Behavior: require('./Behavior'), // abstract base class - JSON: require('./JSON'), - Null: require('./Null') -}; -},{"./Behavior":30,"./JSON":33,"./Null":34}],36:[function(require,module,exports){ -/* eslint-env browser */ + checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) { + var headerRowCount = grid.behavior.getHeaderRowCount(); + var headerColumnCount = grid.behavior.getHeaderColumnCount(); + var gridCell = event.gridCell; + var isFilterRow = grid.isFilterRow(gridCell.y); + var activateEditor = isDoubleClickEditorActivation && gridCell.x >= headerColumnCount && (isFilterRow || gridCell.y >= headerRowCount); + return activateEditor; + } + +}); + +module.exports = CellEditing; +},{"./Feature.js":58}],51:[function(require,module,exports){ 'use strict'; -var mustache = require('mustache'); -var Base = require('extend-me').Base; +var Feature = require('./Feature.js'); /** * @constructor */ -var CellEditor = Base.extend('CellEditor', { - - alias: 'base', +var CellSelection = Feature.extend('CellSelection', { - /** - * am I currently editing (i.e., between calls to `beginEditAt` and either `stopEditing` or `cancelEditing`) - * @type {boolean} - * @default false - * @memberOf CellEditor.prototype - */ - isEditing: false, + alias: 'CellSelection', /** - * the point that I am editing at right now - * @type {Point} - * @default null - * @memberOf CellEditor.prototype + * The pixel location of the mouse pointer during a drag operation. + * @type {window.fin.rectangular.Point} + * @memberOf CellSelection.prototype */ - editorPoint: { - x: -1, - y: -1 - }, + currentDrag: null, /** - * if true, check that the editor is in the right location - * @type {boolean} - * @default false - * @memberOf CellEditor.prototype + * the cell coordinates of the where the mouse pointer is during a drag operation + * @type {Object} + * @memberOf CellSelection.prototype */ - checkEditorPositionFlag: false, + lastDragCell: null, /** - * my instance of hypergrid - * @type {Hypergrid} - * @default null - * @memberOf CellEditor.prototype + * a millisecond value representing the previous time an autoscroll started + * @type {number} + * @default 0 + * @memberOf CellSelection.prototype */ - grid: null, + sbLastAuto: 0, /** - * the value before editing - * @type {type} - * @default null - * @memberOf CellEditor.prototype + * a millisecond value representing the time the current autoscroll started + * @type {number} + * @default 0 + * @memberOf CellSelection.prototype */ - initialValue: null, + sbAutoStart: 0, /** - * @memberOf CellEditor.prototype - * @desc return the behavior (model) - * @returns {Behavior} The behavior (model). + * @memberOf CellSelection.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - getBehavior: function() { - return this.grid.getBehavior(); + handleMouseUp: function(grid, event) { + if (this.dragging) { + this.dragging = false; + } + if (this.next) { + this.next.handleMouseUp(grid, event); + } }, /** - * @memberOf CellEditor.prototype - * @desc This function is a callback from the fin-hypergrid. It is called after each paint of the canvas. + * @memberOf CellSelection.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - gridRenderedNotification: function() { - this.checkEditor(); + handleMouseDown: function(grid, event) { + var isRightClick = event.primitiveEvent.detail.isRightClick; + var cell = event.gridCell; + var viewCell = event.viewPoint; + var dx = cell.x; + var dy = cell.y; + var headerRowCount = grid.behavior.getHeaderRowCount(); + var headerColumnCount = grid.behavior.getHeaderColumnCount(); + var columnCount = grid.behavior.getColumnCount(); + var isOutside = viewCell.x >= columnCount; + + var isHeader = dy < headerRowCount || dx < headerColumnCount; + + if (!grid.isCellSelection() || isRightClick || isHeader || isOutside) { + if (this.next) { + this.next.handleMouseDown(grid, event); + } + } else { + var numFixedColumns = grid.getFixedColumnCount(); + var numFixedRows = grid.getFixedRowCount(); + + //if we are in the fixed area do not apply the scroll values + //check both x and y values independently + if (viewCell.x < numFixedColumns) { + dx = viewCell.x; + } + + if (viewCell.y < numFixedRows) { + dy = viewCell.y; + } + + var dCell = grid.newPoint(dx, dy); + + var primEvent = event.primitiveEvent; + var keys = primEvent.detail.keys; + this.dragging = true; + this.extendSelection(grid, dCell, keys); + } }, /** - * @memberOf CellEditor.prototype - * @desc scroll values have changed, we've been notified + * @memberOf CellSelection.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - scrollValueChangedNotification: function() { - this.setCheckEditorPositionFlag(); - }, + handleMouseDrag: function(grid, event) { + var isRightClick = event.primitiveEvent.detail.isRightClick; + + if (!grid.isCellSelection() || isRightClick || !this.dragging) { + if (this.next) { + this.next.handleMouseDrag(grid, event); + } + } else { + + var numFixedColumns = grid.getFixedColumnCount(); + var numFixedRows = grid.getFixedRowCount(); + + var cell = event.gridCell; + var viewCell = event.viewPoint; + var dx = cell.x; + var dy = cell.y; + + //if we are in the fixed area do not apply the scroll values + //check both x and y values independently + if (viewCell.x < numFixedColumns) { + dx = viewCell.x; + } + + if (viewCell.y < numFixedRows) { + dy = viewCell.y; + } + + var dCell = grid.newPoint(dx, dy); + + var primEvent = event.primitiveEvent; + this.currentDrag = primEvent.detail.mouse; + this.lastDragCell = dCell; - /** - * @memberOf CellEditor.prototype - * @desc turn on checkEditorPositionFlag boolean field - */ - setCheckEditorPositionFlag: function() { - this.checkEditorPositionFlag = true; + this.checkDragScroll(grid, this.currentDrag); + this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys); + } }, /** - * @memberOf CellEditor.prototype - * @desc begin editing at location point - * @param {Point} point - the location to start editing at + * @memberOf CellSelection.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - beginEditAt: function(point) { - this.editorPoint = point; + handleKeyDown: function(grid, event) { + var command = 'handle' + event.detail.char; + if (this[command]) { + this[command].call(this, grid, event.detail); + } }, /** - * @memberOf CellEditor.prototype - * @desc put value into our editor - * @param {object} value - whatever value we want to edit + * @memberOf CellSelection.prototype + * @desc Handle a mousedrag selection. + * @param {Hypergrid} grid + * @param {Object} mouse - the event details + * @param {Array} keys - array of the keys that are currently pressed down */ - setEditorValue: function(value) {}, + handleMouseDragCellSelection: function(grid, gridCell, keys) { - /** - * @memberOf CellEditor.prototype - * @desc returns the point at which we are currently editing - * @returns {Point} - */ - getEditorPoint: function() { - return this.editorPoint; - }, + var headerRowCount = grid.behavior.getHeaderRowCount(); + var headerColumnCount = grid.behavior.getHeaderColumnCount(); + var x = gridCell.x; + var y = gridCell.y; + x = Math.max(headerColumnCount, x); + y = Math.max(headerRowCount, y); - /** - * @memberOf CellEditor.prototype - * @desc set the current editor location - * @param {Point} point - the data location of the current editor - */ - setEditorPoint: function(point) { - this.editorPoint = point; - this.modelPoint = this.getGrid().convertViewPointToDataPoint(point); - }, + var previousDragExtent = grid.getDragExtent(); + var mouseDown = grid.getMouseDown(); - /** - * @memberOf CellEditor.prototype - * @desc display the editor - */ - showEditor: function() {}, + //var scrollingNow = grid.isScrollingNow(); - /** - * @memberOf CellEditor.prototype - * @desc hide the editor - */ - hideEditor: function() {}, + var newX = x - mouseDown.x; + var newY = y - mouseDown.y; - /** - * @memberOf CellEditor.prototype - * @desc stop editing - */ - stopEditing: function() { - if (!this.isEditing) { - return; - } - var proceed = this.getGrid().fireSyntheticEditorDataChangeEvent(this, this.initialValue, this.getEditorValue, this); - if (!proceed) { + if (previousDragExtent.x === newX && previousDragExtent.y === newY) { return; } - this.saveEditorValue(); - this.isEditing = false; - this.hideEditor(); - }, - cancelEditing: function() { - if (!this.isEditing) { - return; - } - this.isEditing = false; - this.hideEditor(); - }, + grid.clearMostRecentSelection(); - /** - * @memberOf CellEditor.prototype - * @desc save the new value into the behavior(model) - */ - saveEditorValue: function() {}, + grid.select(mouseDown.x, mouseDown.y, newX, newY); + grid.setDragExtent(grid.newPoint(newX, newY)); - /** - * @memberOf CellEditor.prototype - * @desc return the current editor's value - */ - getEditorValue: function() {}, + grid.repaint(); + }, /** - * @memberOf CellEditor.prototype - * @desc request focus + * @memberOf CellSelection.prototype + * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above) + * @param {Hypergrid} grid + * @param {Object} mouse - the event details */ - takeFocus: function() {}, - + checkDragScroll: function(grid, mouse) { + if (!grid.resolveProperty('scrollingEnabled')) { + return; + } + var b = grid.getDataBounds(); + var inside = b.contains(mouse); + if (inside) { + if (grid.isScrollingNow()) { + grid.setScrollingNow(false); + } + } else if (!grid.isScrollingNow()) { + grid.setScrollingNow(true); + this.scrollDrag(grid); + } + }, /** - * @memberOf CellEditor.prototype - * @desc check that the editor is in the correct location, and is showing/hidden appropriately + * @memberOf CellSelection.prototype + * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly + * @param {Hypergrid} grid */ - checkEditor: function() { - }, + scrollDrag: function(grid) { - getGrid: function() { - return this.grid; - }, + if (!grid.isScrollingNow()) { + return; + } - template: function() { - /* + var dragStartedInHeaderArea = grid.isMouseDownInHeaderArea(); + var lastDragCell = this.lastDragCell; + var b = grid.getDataBounds(); + var xOffset = 0; + var yOffset = 0; - */ - }, + var numFixedColumns = grid.getFixedColumnCount(); + var numFixedRows = grid.getFixedRowCount(); - getHTML: function() { - var string = this.template.toString().split('\n'); - string.shift(); - string.shift(); - string.length = string.length - 2; - string = string.join('\n').trim(); - return mustache.render(string, this); - }, + var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns; + var dragEndInFixedAreaY = lastDragCell.y < numFixedRows; -}); + if (!dragStartedInHeaderArea) { + if (this.currentDrag.x < b.origin.x) { + xOffset = -1; + } + if (this.currentDrag.y < b.origin.y) { + yOffset = -1; + } + } + if (this.currentDrag.x > b.origin.x + b.extent.x) { + xOffset = 1; + } + if (this.currentDrag.y > b.origin.y + b.extent.y) { + yOffset = 1; + } -module.exports = CellEditor; + var dragCellOffsetX = xOffset; + var dragCellOffsetY = yOffset; -},{"extend-me":4,"mustache":17}],37:[function(require,module,exports){ -'use strict'; + if (dragEndInFixedAreaX) { + dragCellOffsetX = 0; + } -var Simple = require('./Simple'); -var Map = require('../Mappy'); + if (dragEndInFixedAreaY) { + dragCellOffsetY = 0; + } -/** - * @constructor - */ -var Choice = Simple.extend('Choice', { + this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY); + grid.scrollBy(xOffset, yOffset); + this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection + grid.repaint(); + setTimeout(this.scrollDrag.bind(this, grid), 25); + }, /** - * my lookup alias - * @type {string} - * @memberOf Choice.prototype + * @memberOf CellSelection.prototype + * @desc extend a selection or create one if there isnt yet + * @param {Hypergrid} grid + * @param {Object} gridCell - the event details + * @param {Array} keys - array of the keys that are currently pressed down */ - alias: 'choice', + extendSelection: function(grid, gridCell, keys) { + var hasCTRL = keys.indexOf('CTRL') >= 0; + var hasSHIFT = keys.indexOf('SHIFT') >= 0; + // var scrollTop = grid.getVScrollValue(); + // var scrollLeft = grid.getHScrollValue(); - /** - * the list of items to pick from - * @type {Array} - * @memberOf Choice.prototype - */ - items: ['a', 'b', 'c'], + // var numFixedColumns = 0;//grid.getFixedColumnCount(); + // var numFixedRows = 0;//grid.getFixedRowCount(); - template: function() { - /* - - */ - }, + var mousePoint = grid.getMouseDown(); + var x = gridCell.x; // - numFixedColumns + scrollLeft; + var y = gridCell.y; // - numFixedRows + scrollTop; - autopopulate: function() { - var grid = this.getGrid(); - var behavior = grid.getBehavior(); - var point = this.getEditorPoint(); - var colProps = grid.getColumnProperties(point.x); - if (!colProps.autopopulateEditor) { + //were outside of the grid do nothing + if (x < 0 || y < 0) { return; } - var headerCount = grid.getHeaderRowCount(); - var rowCount = grid.getUnfilteredRowCount() - headerCount; - var column = point.x; - var map = new Map(); - for (var r = 0; r < rowCount; r++) { - var each = behavior.getUnfilteredValue(column, r); - map.set(each, each); + + //we have repeated a click in the same spot deslect the value from last time + if ( + hasCTRL && + x === mousePoint.x && + y === mousePoint.y + ) { + grid.clearMostRecentSelection(); + grid.popMouseDown(); + grid.repaint(); + return; } - var values = map.values; - values.sort(); - if (values.length > 0 && values[0].length > 0) { - values.unshift(''); + if (!hasCTRL && !hasSHIFT) { + grid.clearSelections(); + } + + if (hasSHIFT) { + grid.clearMostRecentSelection(); + grid.select(mousePoint.x, mousePoint.y, x - mousePoint.x + 1, y - mousePoint.y + 1); + grid.setDragExtent(grid.newPoint(x - mousePoint.x + 1, y - mousePoint.y)); + } else { + grid.select(x, y, 0, 0); + grid.setMouseDown(grid.newPoint(x, y)); + grid.setDragExtent(grid.newPoint(0, 0)); } - - this.setItems(values); + grid.repaint(); }, - //no events are fired while the dropdown is open - //see http://jsfiddle.net/m4tndtu4/6/ /** - * @memberOf Choice.prototype + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid */ - showEditor: function() { - var self = this; - this.input.style.display = 'inline'; - setTimeout(function() { - self.showDropdown(self.input); - }, 50); + handleDOWNSHIFT: function(grid) { + this.moveShiftSelect(grid, 0, 1); }, - preShowEditorNotification: function() { - this.autopopulate(); - this.setEditorValue(this.initialValue); + /** + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleUPSHIFT: function(grid) { + this.moveShiftSelect(grid, 0, -1); }, /** - * @memberOf Choice.prototype - * @param items + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - setItems: function(items) { - this.items = items; - this.updateView(); + handleLEFTSHIFT: function(grid) { + this.moveShiftSelect(grid, -1, 0); }, /** - * @memberOf Choice.prototype - * @param input + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - initializeInput: function(input) { - var self = this; - Simple.prototype.initializeInput.apply(this, [input]); - input.onchange = function() { - self.stopEditing(); - }; - } - -}); - -module.exports = Choice; - -},{"../Mappy":26,"./Simple":41}],38:[function(require,module,exports){ -'use strict'; - -var Simple = require('./Simple'); - -/** - * @constructor - */ -var Color = Simple.extend('Color', { + handleRIGHTSHIFT: function(grid) { + this.moveShiftSelect(grid, 1, 0); + }, /** - * my lookup alias - * @type {string} - * @memberOf Color.prototype + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - alias: 'color', - - template: function() { - /* - - */ - } - -}); - -module.exports = Color; - -},{"./Simple":41}],39:[function(require,module,exports){ -/* eslint-env browser */ - -'use strict'; - -var Simple = require('./Simple'); - -var parseDate = function(input) { - var parts = input.match(/(\d+)/g); - // new Date(year, month [, date [, hours[, minutes[, seconds[, ms]]]]]) - return new window.Date(parts[0], parts[1] - 1, parts[2]); // months are 0-based -}; + handleDOWN: function(grid, event) { + //keep the browser viewport from auto scrolling on key event + event.primitiveEvent.preventDefault(); -var leadingZeroIfNecessary = function(number) { - return number < 10 ? '0' + number : number + ''; -}; -/** - * @constructor - */ -var Date = Simple.extend('Date', { + var count = this.getAutoScrollAcceleration(); + this.moveSingleSelect(grid, 0, count); + }, /** - * my lookup alias - * @type {string} - * @memberOf Date.prototype + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - alias: 'date', + handleUP: function(grid, event) { + //keep the browser viewport from auto scrolling on key event + event.primitiveEvent.preventDefault(); - template: function() { - /* - - */ + var count = this.getAutoScrollAcceleration(); + this.moveSingleSelect(grid, 0, -count); }, - setEditorValue: function(value) { - if (value && value.constructor.name === 'Date') { - value = value.getFullYear() + '-' + leadingZeroIfNecessary(value.getMonth() + 1) + '-' + leadingZeroIfNecessary(value.getDay()); - } - this.getInput().value = value + ''; + /** + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleLEFT: function(grid) { + this.moveSingleSelect(grid, -1, 0); }, - getEditorValue: function() { - var value = this.getInput().value; - value = parseDate(value); - return value; + /** + * @memberOf CellSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleRIGHT: function(grid) { + this.moveSingleSelect(grid, 1, 0); }, -}); - -module.exports = Date; - -},{"./Simple":41}],40:[function(require,module,exports){ -/* eslint-env browser */ -'use strict'; - -var CellEditor = require('./CellEditor'); - -/** - * @constructor - */ -var Filter = CellEditor.extend('Filter', { - /** - * my lookup alias - * @type {string} - * @memberOf Textfield.prototype + * @memberOf CellSelection.prototype + * @desc If we are holding down the same navigation key, accelerate the increment we scroll + * #### returns: integer */ - alias: 'filter', - - initialize: function() { - var data = document.createElement('div'); - var style = data.style; - style.position = 'absolute'; - style.top = style.bottom = '44px'; - style.right = style.left = '1em'; - style.overflowY = 'scroll'; - - var table = document.createElement('table'); - data.appendChild(table); - - style = table.style; - style.width = style.height = '100%'; - - var tr = document.createElement('tr'); - var td = document.createElement('td'); - table.appendChild(tr); - tr.appendChild(td); - - - this.title = document.createElement('div'); - this.title.innerHTML = 'Filter Editor'; - - this.dialog = document.createElement('div'); - - this.content = td; - this.buttons = document.createElement('div'); + getAutoScrollAcceleration: function() { + var count = 1; + var elapsed = this.getAutoScrollDuration() / 2000; + count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed)); + return count; + }, - style = this.dialog.style; - style.position = 'absolute'; - style.top = style.left = style.right = style.bottom = 0; - style.whiteSpace = 'nowrap'; + /** + * @memberOf CellSelection.prototype + * @desc set the start time to right now when we initiate an auto scroll + */ + setAutoScrollStartTime: function() { + this.sbAutoStart = Date.now(); + }, - style = this.title.style; - style.position = 'absolute'; - style.top = style.left = style.right = 0; - style.height = '44px'; - style.whiteSpace = 'nowrap'; - style.textAlign = 'center'; - style.padding = '11px'; + /** + * @memberOf CellSelection.prototype + * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time + */ + pingAutoScroll: function() { + var now = Date.now(); + if (now - this.sbLastAuto > 500) { + this.setAutoScrollStartTime(); + } + this.sbLastAuto = Date.now(); + }, - style = this.buttons.style; - style.position = 'absolute'; - style.left = style.right = style.bottom = 0; - style.height = '44px'; - style.whiteSpace = 'nowrap'; - style.textAlign = 'center'; - style.padding = '8px'; + /** + * @memberOf CellSelection.prototype + * @desc answer how long we have been auto scrolling + * #### returns: integer + */ + getAutoScrollDuration: function() { + if (Date.now() - this.sbLastAuto > 500) { + return 0; + } + return Date.now() - this.sbAutoStart; + }, - this.dialog.appendChild(this.title); - this.dialog.appendChild(data); - this.dialog.appendChild(this.buttons); + /** + * @memberOf CellSelection.prototype + * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary. + * @param {Hypergrid} grid + * @param {number} offsetX - x coordinate to start at + * @param {number} offsetY - y coordinate to start at + */ + moveShiftSelect: function(grid, offsetX, offsetY) { - this.ok = document.createElement('button'); - this.ok.style.borderRadius = '8px'; - this.ok.style.width = '5.5em'; + var maxColumns = grid.getColumnCount() - 1; + var maxRows = grid.getRowCount() - 1; - this.cancel = document.createElement('button'); - this.cancel.style.marginLeft = '2em'; - this.cancel.style.borderRadius = '8px'; - this.cancel.style.width = '5.5em'; + var maxViewableColumns = grid.getVisibleColumns() - 1; + var maxViewableRows = grid.getVisibleRows() - 1; - this.delete = document.createElement('button'); - this.delete.style.marginLeft = '2em'; - this.delete.style.borderRadius = '8px'; - this.delete.style.width = '5.5em'; + if (!grid.resolveProperty('scrollingEnabled')) { + maxColumns = Math.min(maxColumns, maxViewableColumns); + maxRows = Math.min(maxRows, maxViewableRows); + } - this.reset = document.createElement('button'); - this.reset.style.marginLeft = '2em'; - this.reset.style.borderRadius = '8px'; - this.reset.style.width = '5.5em'; + var origin = grid.getMouseDown(); + var extent = grid.getDragExtent(); - this.ok.innerHTML = 'ok'; - this.cancel.innerHTML = 'cancel'; - this.delete.innerHTML = 'delete'; - this.reset.innerHTML = 'reset'; + var newX = extent.x + offsetX; + var newY = extent.y + offsetY; - this.buttons.appendChild(this.ok); - this.buttons.appendChild(this.reset); - this.buttons.appendChild(this.delete); - this.buttons.appendChild(this.cancel); + newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX)); + newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY)); - var self = this; - this.ok.onclick = function() { - self.okPressed(); - }; - this.cancel.onclick = function() { - self.cancelPressed(); - }; - this.delete.onclick = function() { - self.deletePressed(); - }; - this.reset.onclick = function() { - self.resetPressed(); - }; - }, + grid.clearMostRecentSelection(); + grid.select(origin.x, origin.y, newX, newY); - tearDown: function() { - this.content.innerHTML = ''; - }, + grid.setDragExtent(grid.newPoint(newX, newY)); - okPressed: function() { - var dialog = this.getGrid().dialog; - dialog.onOkPressed(); - }, + if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) { + this.pingAutoScroll(); + } + if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) { + this.pingAutoScroll(); + } - cancelPressed: function() { - var dialog = this.getGrid().dialog; - dialog.onCancelPressed(); - }, + grid.repaint(); - deletePressed: function() { - var dialog = this.getGrid().dialog; - dialog.onDeletePressed(); }, - resetPressed: function() { - var dialog = this.getGrid().dialog; - dialog.onResetPressed(); - }, + /** + * @memberOf CellSelection.prototype + * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent. + * @param {Hypergrid} grid + * @param {number} offsetX - x coordinate to start at + * @param {number} offsetY - y coordinate to start at + */ + moveSingleSelect: function(grid, offsetX, offsetY) { - beginEditAt: function(editorPoint) { - var grid = this.getGrid(); - var behavior = grid.getBehavior(); - var dialog = grid.dialog; + var maxColumns = grid.getColumnCount() - 1; + var maxRows = grid.getRowCount() - 1; - var columnIndex = editorPoint.x, - title = behavior.getColumnId(columnIndex), - field = behavior.getField(columnIndex), - type = behavior.getColumn(columnIndex).getType(); + var maxViewableColumns = grid.getVisibleColumnsCount() - 1; + var maxViewableRows = grid.getVisibleRowsCount() - 1; - var fieldsProvider = function() { - return [{ - name: field, - alias: title, - type: type - }]; - }; - this.title.innerHTML = 'filter for ' + title + ' column'; - var filter = grid.getFilterFor(columnIndex); - //var self = this; - if (dialog.isOpen()) { - dialog.close(); - } else { - var self = this; + var minRows = grid.getHeaderRowCount(); + var minCols = grid.getHeaderColumnCount(); - dialog.clear(); - dialog.overlay.appendChild(this.dialog); + if (!grid.resolveProperty('scrollingEnabled')) { + maxColumns = Math.min(maxColumns, maxViewableColumns); + maxRows = Math.min(maxRows, maxViewableRows); + } - filter.initialize(fieldsProvider); + var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); - dialog.onOkPressed = function() { - if (filter.onOk && filter.onOk()) { // onOK() truthy result means abort; falsy means proceed - return; - } - self.tearDown(); - behavior.setComplexFilter(columnIndex, { - type: filter.alias, - state: filter.getState() - }); - dialog.close(); - behavior.applyFilters(); - behavior.changed(); - }; + var newX = mouseCorner.x + offsetX; + var newY = mouseCorner.y + offsetY; - dialog.onCancelPressed = function() { - if (filter.onCancel && filter.onCancel()) { - return; - } - self.tearDown(); - dialog.close(); - filter = undefined; - }; + newX = Math.min(maxColumns, Math.max(minCols, newX)); + newY = Math.min(maxRows, Math.max(minRows, newY)); - dialog.onDeletePressed = function() { - if (filter.onDelete && filter.onDelete()) { - return; - } - self.tearDown(); - behavior.setComplexFilter(columnIndex, undefined); - dialog.close(); - behavior.applyFilters(); - behavior.changed(); - }; + grid.clearSelections(); + grid.select(newX, newY, 0, 0); + grid.setMouseDown(grid.newPoint(newX, newY)); + grid.setDragExtent(grid.newPoint(0, 0)); - dialog.onResetPressed = function() { - if (filter.onReset && filter.onReset()) { - return; - } - self.tearDown(); - filter.initialize(dialog); - if (filter.onShow) { - filter.onShow(self.content); - } - }; + if (grid.insureModelColIsVisible(newX, offsetX)) { + this.pingAutoScroll(); + } + if (grid.insureModelRowIsVisible(newY, offsetY)) { + this.pingAutoScroll(); + } - var cellBounds = grid._getBoundsOfCell(columnIndex, editorPoint.y); + grid.repaint(); - //hack to accomodate bootstrap margin issues... - var xOffset = grid.div.getBoundingClientRect().left - grid.divCanvas.getBoundingClientRect().left; - cellBounds.x = cellBounds.x - xOffset; - dialog.openFrom(cellBounds); - var previousState = behavior.getComplexFilter(columnIndex); - if (previousState) { - filter.setState(previousState.state); - } - setTimeout(function() { - if (filter.onShow) { - filter.onShow(self.content); - } - }, dialog.getAnimationTime() + 10); - } - }, + } }); -module.exports = Filter; - -},{"./CellEditor":36}],41:[function(require,module,exports){ -/* eslint-env browser */ +module.exports = CellSelection; +},{"./Feature.js":58}],52:[function(require,module,exports){ 'use strict'; -var CellEditor = require('./CellEditor.js'); +var Feature = require('./Feature.js'); /** * @constructor */ -var Simple = CellEditor.extend('Simple', { +var ColumnAutosizing = Feature.extend('ColumnAutosizing', { - /** - * my main input control - * @type {Element} - * @default null - * @memberOf CellEditor.prototype - */ - input: null, + alias: 'ColumnAutosizing', /** - * my lookup alias - * @type {string} - * @memberOf Simple.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + * @memberOf ColumnAutosizing.prototype */ - alias: 'simple', + handleDoubleClick: function(grid, event) { + var headerRowCount = grid.getHeaderRowCount(); + //var headerColCount = grid.getHeaderColumnCount(); + var gridCell = event.gridCell; + if (gridCell.y <= headerRowCount) { + grid.autosizeColumn(gridCell.x); + } else if (this.next) { + this.next.handleDoubleClick(grid, event); + } + } - /** - * @memberOf Simple.prototype - */ - initialize: function() { - this.editorPoint = { - x: 0, - y: 0 - }; - }, +}); + +module.exports = ColumnAutosizing; + +},{"./Feature.js":58}],53:[function(require,module,exports){ +/* eslint-env browser */ +/* global requestAnimationFrame */ + +'use strict'; + +// This feature is responsible for column drag and drop reordering. +// This object is a mess and desperately needs a complete rewrite..... + +var Feature = require('./Feature.js'); + +var columnAnimationTime = 150; +var dragger; +var draggerCTX; +var floatColumn; +var floatColumnCTX; + +/** + * @constructor + */ +var ColumnMoving = Feature.extend('ColumnMoving', { + + alias: 'ColumnMoving', /** - * @memberOf Simple.prototype - * @desc the function to override for initialization + * queue up the animations that need to play so they are done synchronously + * @type {Array} + * @memberOf CellMoving.prototype */ - initializeInput: function(input) { - var self = this; - input.addEventListener('keyup', function(e) { - if (e && (e.keyCode === 13 || e.keyCode === 27 || e.keyCode === 8)) { - e.preventDefault(); - if (e.keyCode === 8) { - self.clearStopEditing(); - } else if (e.keyCode === 27) { - self.cancelEditing(); - } else { - self.stopEditing(); - } - self.getGrid().repaint(); - self.getGrid().takeFocus(); - } - self.getGrid().fireSyntheticEditorKeyUpEvent(self, e); - }); - input.addEventListener('keydown', function(e) { - self.getGrid().fireSyntheticEditorKeyDownEvent(self, e); - }); - input.addEventListener('keypress', function(e) { - self.getGrid().fireSyntheticEditorKeyPressEvent(self, e); - }); - input.onblur = function(e) { - self.cancelEditing(); - }; - - // input.addEventListener('focusout', function() { - // self.stopEditing(); - // }); - // input.addEventListener('blur', function() { - // self.stopEditing(); - // }); - input.style.position = 'absolute'; - input.style.display = 'none'; - input.style.border = 'solid 2px black'; - input.style.outline = 0; - input.style.padding = 0; - input.style.zIndex = 1000; - //input.style.fontSize = '8pt'; - input.style.boxShadow = 'white 0px 0px 1px 1px'; - }, + floaterAnimationQueue: [], /** - * @memberOf Simple.prototype - * @returns {object} the current editor's value + * am I currently auto scrolling right + * @type {boolean} + * @memberOf CellMoving.prototype */ - getEditorValue: function() { - var value = this.getInput().value; - return value; - }, + columnDragAutoScrollingRight: false, /** - * @memberOf Simple.prototype - * @desc save the new value into the behavior(model) + * am I currently auto scrolling left + * @type {boolean} + * @memberOf CellMoving.prototype */ - setEditorValue: function(value) { - this.getInput().value = value + ''; - }, - - clearStopEditing: function() { - this.setEditorValue(''); - this.stopEditing(); - }, - - cancelEditing: function() { - if (!this.isEditing) { - return; - } - this.getInput().value = null; - this.isEditing = false; - this.hideEditor(); - }, + columnDragAutoScrollingLeft: false, /** - * @memberOf Simple.prototype - * @desc display the editor + * is the drag mechanism currently enabled ("armed") + * @type {boolean} + * @memberOf CellMoving.prototype */ - showEditor: function() { - this.getInput().style.display = 'inline'; - }, + dragArmed: false, /** - * @memberOf Simple.prototype - * @desc hide the editor + * am I dragging right now + * @type {boolean} + * @memberOf CellMoving.prototype */ - hideEditor: function() { - this.getInput().style.display = 'none'; - }, + dragging: false, /** - * @memberOf Simple.prototype - * @desc request focus for my input control + * the column index of the currently dragged column + * @type {number} + * @memberOf CellMoving.prototype */ - takeFocus: function() { - var self = this; - setTimeout(function() { - self.input.focus(); - self.selectAll(); - }, 300); - }, + dragCol: -1, /** - * @memberOf Simple.prototype - * @desc select everything + * an offset to position the dragged item from the cursor + * @type {number} + * @memberOf CellMoving.prototype */ - selectAll: function() { - - }, + dragOffset: 0, /** - * @memberOf Simple.prototype - * @desc how much should I offset my bounds from 0,0 + * @memberOf CellMoving.prototype + * @desc give me an opportunity to initialize stuff on the grid + * @param {Hypergrid} grid */ - originOffset: function() { - return [0, 0]; + initializeOn: function(grid) { + this.isFloatingNow = false; + this.initializeAnimationSupport(grid); + if (this.next) { + this.next.initializeOn(grid); + } }, /** - * @memberOf Simple.prototype - * @desc set the bounds of my input control - * @param {rectangle} rectangle - the bounds to move to + * @memberOf CellMoving.prototype + * @desc initialize animation support on the grid + * @param {Hypergrid} grid */ - setBounds: function(cellBounds) { - var originOffset = this.originOffset(); - var translation = 'translate(' - + (cellBounds.x - 1 + originOffset[0]) + 'px,' - + (cellBounds.y - 1 + originOffset[1]) + 'px)'; - - var input = this.getInput(); + initializeAnimationSupport: function(grid) { + if (!dragger) { + dragger = document.createElement('canvas'); + dragger.setAttribute('width', '0px'); + dragger.setAttribute('height', '0px'); - input.style.boxSizing = 'border-box'; + document.body.appendChild(dragger); + draggerCTX = dragger.getContext('2d'); + } + if (!floatColumn) { + floatColumn = document.createElement('canvas'); + floatColumn.setAttribute('width', '0px'); + floatColumn.setAttribute('height', '0px'); - input.style.webkitTransform = translation; - input.style.MozTransform = translation; - input.style.msTransform = translation; - input.style.OTransform = translation; + document.body.appendChild(floatColumn); + floatColumnCTX = floatColumn.getContext('2d'); + } - // input.style.left = cellBounds.x + originOffset[0] + 'px'; - // input.style.top = cellBounds.y + originOffset[1] + 'px'; + }, - input.style.width = (cellBounds.width + 2) + 'px'; - input.style.height = (cellBounds.height + 2) + 'px'; - //var xOffset = this.grid.canvas.getBoundingClientRect().left; + getCanDragCursorName: function() { + return '-webkit-grab'; }, - saveEditorValue: function() { - var point = this.getEditorPoint(); - var value = this.getEditorValue(); - if (value === this.initialValue) { - return; //data didn't change do nothing - } - if (parseFloat(this.initialValue) === this.initialValue) { // I'm a number - value = parseFloat(value); - } - var continued = this.getGrid().fireBeforeCellEdit(point, this.initialValue, value, this); - if (!continued) { - return; - } - this.getBehavior().setValue(point.x, point.y, value); - this.getGrid().fireAfterCellEdit(point, this.initialValue, value, this); + getDraggingCursorName: function() { + return '-webkit-grabbing'; }, /** - * @memberOf CellEditor.prototype - * @desc move the editor to the current editor point + * @memberOf CellMoving.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - _moveEditor: function() { - var grid = this.getGrid(); - var editorPoint = this.getEditorPoint(); - var cellBounds = grid._getBoundsOfCell(editorPoint.x, editorPoint.y); - - //hack to accomodate bootstrap margin issues... - var xOffset = grid.div.getBoundingClientRect().left - grid.divCanvas.getBoundingClientRect().left; - cellBounds.x = cellBounds.x - xOffset; - - this.setBounds(cellBounds); - }, + handleMouseDrag: function(grid, event) { - moveEditor: function() { - this._moveEditor(); - this.takeFocus(); - }, + var gridCell = event.gridCell; + var x; + //var y; - beginEditAt: function(point) { + var distance = Math.abs(event.primitiveEvent.detail.dragstart.x - event.primitiveEvent.detail.mouse.x); - if (!this.isAdded) { - this.isAdded = true; - this.grid.div.appendChild(this.getInput()); + if (distance < 10) { + if (this.next) { + this.next.handleMouseDrag(grid, event); + } + return; } - this.setEditorPoint(point); - var model = this.getBehavior(); - var value = model.getValue(point.x, point.y); - if (value.constructor.name === 'Array') { - value = value[1]; //it's a nested object + if (this.isHeaderRow(grid, event) && this.dragArmed && !this.dragging) { + this.dragging = true; + this.dragCol = gridCell.x; + this.dragOffset = event.mousePoint.x; + this.detachChain(); + x = event.primitiveEvent.detail.mouse.x - this.dragOffset; + //y = event.primitiveEvent.detail.mouse.y; + this.createDragColumn(grid, x, this.dragCol); + } else if (this.next) { + this.next.handleMouseDrag(grid, event); } - var proceed = this.grid.fireRequestCellEdit(point, value); - if (!proceed) { - //we were cancelled - return; + + if (this.dragging) { + x = event.primitiveEvent.detail.mouse.x - this.dragOffset; + //y = event.primitiveEvent.detail.mouse.y; + this.dragColumn(grid, x); } - this.initialValue = value; - this.isEditing = true; - this.setCheckEditorPositionFlag(); - this.checkEditor(); }, - checkEditor: function() { - if (!this.checkEditorPositionFlag) { - return; - } else { - this.checkEditorPositionFlag = false; - } - if (!this.isEditing) { - return; - } - var editorPoint = this.getEditorPoint(); - if (this.grid.isDataVisible(editorPoint.x, editorPoint.y)) { - this.preShowEditorNotification(); - this.attachEditor(); - this.moveEditor(); - this.showEditor(); - } else { - this.hideEditor(); + /** + * @memberOf CellMoving.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleMouseDown: function(grid, event) { + if (grid.behavior.isColumnReorderable()) { + if (this.isHeaderRow(grid, event) && event.gridCell.x !== -1) { + this.dragArmed = true; + this.cursor = this.getDraggingCursorName(); + grid.clearSelections(); + } + } + if (this.next) { + this.next.handleMouseDown(grid, event); } }, - attachEditor: function() { - var input = this.getInput(); - this.grid.div.appendChild(input); - }, - - preShowEditorNotification: function() { - this.setEditorValue(this.initialValue); - }, - - getInput: function() { - if (!this.input) { - this.input = this.getDefaultInput(); + /** + * @memberOf CellMoving.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleMouseUp: function(grid, event) { + //var col = event.gridCell.x; + if (this.dragging) { + this.cursor = null; + //delay here to give other events a chance to be dropped + var self = this; + this.endDragColumn(grid); + setTimeout(function() { + self.attachChain(); + }, 200); } - return this.input; - }, + this.dragCol = -1; + this.dragging = false; + this.dragArmed = false; + this.cursor = null; + grid.repaint(); - getDefaultInput: function() { - var div = document.createElement('DIV'); - div.innerHTML = this.getHTML(); - var input = div.firstChild; - this.initializeInput(input); - return input; - }, + if (this.next) { + this.next.handleMouseUp(grid, event); + } - updateView: function() { - var oldGuy = this.getInput(); - var parent = oldGuy.parentNode; - var newGuy = this.getDefaultInput(); - this.input = newGuy; - parent.replaceChild(newGuy, oldGuy); }, - showDropdown: function(element) { - var event; - event = document.createEvent('MouseEvents'); - event.initMouseEvent('mousedown', true, true, window); - element.dispatchEvent(event); - } -}); - -module.exports = Simple; + /** + * @memberOf CellMoving.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleMouseMove: function(grid, event) { -},{"./CellEditor.js":36}],42:[function(require,module,exports){ -'use strict'; + if (!this.dragging && event.mousePoint.y < 5 && event.viewPoint.y === 0) { + this.cursor = this.getCanDragCursorName(); + } else { + this.cursor = null; + } -var Simple = require('./Simple'); + if (this.next) { + this.next.handleMouseMove(grid, event); + } -/** - * @constructor - */ -var Slider = Simple.extend('Slider', { + if (this.isHeaderRow(grid, event) && this.dragging) { + this.cursor = this.getDraggingCursorName(); //move'; + } + }, /** - * my lookup alias - * @type {string} - * @memberOf Slider.prototype + * @memberOf CellMoving.prototype + * @desc this is the main event handler that manages the dragging of the column + * @param {Hypergrid} grid + * @param {boolean} draggedToTheRight - are we moving to the right */ - alias: 'slider', + floatColumnTo: function(grid, draggedToTheRight) { + this.floatingNow = true; - template: function() { - /* - - */ - } + var renderer = grid.getRenderer(); + var colEdges = renderer.getColumnEdges(); + var scrollLeft = grid.getHScrollValue(); + var floaterIndex = grid.renderOverridesCache.floater.columnIndex; + var draggerIndex = grid.renderOverridesCache.dragger.columnIndex; + var hdpiratio = grid.renderOverridesCache.dragger.hdpiratio; -}); + var draggerStartX; + var floaterStartX; + var fixedColumnCount = grid.getFixedColumnCount(); + var draggerWidth = grid.getColumnWidth(draggerIndex); + var floaterWidth = grid.getColumnWidth(floaterIndex); -module.exports = Slider; + var max = grid.getVisibleColumnsCount(); -},{"./Simple":41}],43:[function(require,module,exports){ -'use strict'; + var doffset = 0; + var foffset = 0; -var Simple = require('./Simple'); + if (draggerIndex >= fixedColumnCount) { + doffset = scrollLeft; + } + if (floaterIndex >= fixedColumnCount) { + foffset = scrollLeft; + } -/** - * @constructor - */ -var Spinner = Simple.extend('Spinner', { + if (draggedToTheRight) { + draggerStartX = colEdges[Math.min(max, draggerIndex - doffset)]; + floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)]; - /** - * my lookup alias - * @type {string} - * @memberOf Spinner.prototype - */ - alias: 'spinner', + grid.renderOverridesCache.dragger.startX = (draggerStartX + floaterWidth) * hdpiratio; + grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio; - template: function() { - /* - - */ - } + } else { + floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)]; + draggerStartX = floaterStartX + draggerWidth; -}); + grid.renderOverridesCache.dragger.startX = floaterStartX * hdpiratio; + grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio; + } + grid.swapColumns(draggerIndex, floaterIndex); + grid.renderOverridesCache.dragger.columnIndex = floaterIndex; + grid.renderOverridesCache.floater.columnIndex = draggerIndex; -module.exports = Spinner; -},{"./Simple":41}],44:[function(require,module,exports){ -'use strict'; + this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(grid, floaterStartX, draggerStartX)); -var Simple = require('./Simple'); + this.doFloaterAnimation(grid); -/** - * @constructor - */ -var Textfield = Simple.extend('Textfield', { + }, /** - * my lookup alias - * @type {string} - * @memberOf Textfield.prototype + * @memberOf CellMoving.prototype + * @desc manifest the column drag and drop animation + * @param {Hypergrid} grid + * @param {number} floaterStartX - the x start coordinate of the column underneath that floats behind the dragged column + * @param {number} draggerStartX - the x start coordinate of the dragged column */ - alias: 'textfield', + doColumnMoveAnimation: function(grid, floaterStartX, draggerStartX) { + var self = this; + return function() { + var d = floatColumn; + d.style.display = 'inline'; + self.setCrossBrowserProperty(d, 'transform', 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'); - template: function() { - /* - - */ - }, + //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'; + //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'; - selectAll: function() { - this.input.setSelectionRange(0, this.input.value.length); - }, + requestAnimationFrame(function() { + self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease'); + self.setCrossBrowserProperty(d, 'transform', 'translate(' + draggerStartX + 'px, ' + -2 + 'px)'); + }); + grid.repaint(); + //need to change this to key frames - initializeInput: function(input) { - var self = this; - input.addEventListener('keyup', function(e) { - if (e && (e.keyCode === 13 || e.keyCode === 27)) { - e.preventDefault(); - if (e.keyCode === 27) { - self.cancelEditing(); - } else { - self.stopEditing(); - } - self.getGrid().repaint(); - self.getGrid().takeFocus(); - } - if (self.getGrid().isFilterRow(self.getEditorPoint().y)) { - setTimeout(function() { - self.saveEditorValue(); - self._moveEditor(); + setTimeout(function() { + self.setCrossBrowserProperty(d, 'transition', ''); + grid.renderOverridesCache.floater = null; + grid.repaint(); + self.doFloaterAnimation(grid); + requestAnimationFrame(function() { + d.style.display = 'none'; + self.isFloatingNow = false; }); - } - self.getGrid().fireSyntheticEditorKeyUpEvent(self, e); - }); - input.addEventListener('keydown', function(e) { - self.getGrid().fireSyntheticEditorKeyDownEvent(self, e); - }); - input.addEventListener('keypress', function(e) { - self.getGrid().fireSyntheticEditorKeyPressEvent(self, e); - }); - input.onblur = function(e) { - self.cancelEditing(); + }, columnAnimationTime + 50); }; - // input.addEventListener('focusout', function() { - // self.stopEditing(); - // }); - // input.addEventListener('blur', function() { - // self.stopEditing(); - // }); - input.style.position = 'absolute'; - input.style.display = 'none'; - input.style.border = 'solid 2px black'; - input.style.outline = 0; - input.style.padding = 0; - input.style.zIndex = 1000; - //input.style.fontSize = '8pt'; - input.style.boxShadow = 'white 0px 0px 1px 1px'; }, -}); -module.exports = Textfield; + /** + * @memberOf CellMoving.prototype + * @desc manifest the floater animation + * @param {Hypergrid} grid + */ + doFloaterAnimation: function(grid) { + if (this.floaterAnimationQueue.length === 0) { + this.floatingNow = false; + grid.repaint(); + return; + } + var animation = this.floaterAnimationQueue.pop(); + animation(); + }, + + /** + * @memberOf CellMoving.prototype + * @desc create the float column at columnIndex underneath the dragged column + * @param {Hypergrid} grid + * @param {number} columnIndex - the index of the column that will be floating + */ + createFloatColumn: function(grid, columnIndex) { -},{"./Simple":41}],45:[function(require,module,exports){ -'use strict'; + var fixedColumnCount = grid.getFixedColumnCount(); + var scrollLeft = grid.getHScrollValue(); -module.exports = { - CellEditor: require('./CellEditor'), // abstract base class - Textfield: require('./Textfield'), - Choice: require('./Choice'), - //Combo: require('./Combo'), - Color: require('./Color'), - Date: require('./Date'), - Slider: require('./Slider'), - Spinner: require('./Spinner'), - Filter: require('./Filter') -}; + if (columnIndex < fixedColumnCount) { + scrollLeft = 0; + } -},{"./CellEditor":36,"./Choice":37,"./Color":38,"./Date":39,"./Filter":40,"./Slider":42,"./Spinner":43,"./Textfield":44}],46:[function(require,module,exports){ -'use strict'; + var renderer = grid.getRenderer(); + var columnEdges = renderer.getColumnEdges(); -var Base = require('extend-me').Base; + var columnWidth = grid.getColumnWidth(columnIndex); + var colHeight = grid.div.clientHeight; + var d = floatColumn; + var style = d.style; + var location = grid.div.getBoundingClientRect(); -var A = 'A'.charCodeAt(0); + style.top = (location.top - 2) + 'px'; + style.left = location.left + 'px'; + style.position = 'fixed'; -/** - * @constructor - */ -var DataModel = Base.extend('DataModel', { + var hdpiRatio = grid.getHiDPI(floatColumnCTX); - next: null, + d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px'); + d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px'); + style.boxShadow = '0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)'; + style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px'; + style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px'; + style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor'); + style.backgroundColor = renderer.resolveProperty('backgroundColor'); - grid: null, + var startX = columnEdges[columnIndex - scrollLeft]; + startX = startX * hdpiRatio; - setGrid: function(newGrid) { - this.grid = newGrid; - }, + floatColumnCTX.scale(hdpiRatio, hdpiRatio); - getGrid: function() { - return this.grid; - }, + grid.renderOverridesCache.floater = { + columnIndex: columnIndex, + ctx: floatColumnCTX, + startX: startX, + width: columnWidth, + height: colHeight, + hdpiratio: hdpiRatio + }; - getBehavior: function() { - return this.getGrid().getBehavior(); + style.zIndex = '4'; + this.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -2 + 'px)'); + style.cursor = this.getDraggingCursorName(); + grid.repaint(); }, - changed: function() { - this.getBehavior().changed(); + /** + * @memberOf CellMoving.prototype + * @desc utility function for setting cross browser css properties + * @param {HTMLElement} element - descripton + * @param {string} property - the property + * @param {string} value - the value to assign + */ + setCrossBrowserProperty: function(element, property, value) { + var uProperty = property[0].toUpperCase() + property.substr(1); + this.setProp(element, 'webkit' + uProperty, value); + this.setProp(element, 'Moz' + uProperty, value); + this.setProp(element, 'ms' + uProperty, value); + this.setProp(element, 'O' + uProperty, value); + this.setProp(element, property, value); }, - getPrivateState: function() { - return this.getGrid().getPrivateState(); + /** + * @memberOf CellMoving.prototype + * @desc utility function for setting properties on HTMLElements + * @param {HTMLElement} element - descripton + * @param {string} property - the property + * @param {string} value - the value to assign + */ + setProp: function(element, property, value) { + if (property in element.style) { + element.style[property] = value; + } }, - applyState: function() { + /** + * @memberOf CellMoving.prototype + * @desc create the dragged column at columnIndex above the floated column + * @param {Hypergrid} grid + * @param {number} x - the start position + * @param {number} columnIndex - the index of the column that will be floating + */ + createDragColumn: function(grid, x, columnIndex) { - }, + var fixedColumnCount = grid.getFixedColumnCount(); + var scrollLeft = grid.getHScrollValue(); - alphaFor: function(i) { - // Name the column headers in A, .., AA, AB, AC, .., AZ format - // quotient/remainder - //var quo = Math.floor(col/27); - var quo = Math.floor(i / 26); - var rem = i % 26; - var code = ''; - if (quo > 0) { - code += this.alpha(quo - 1); + if (columnIndex < fixedColumnCount) { + scrollLeft = 0; } - code += this.alpha(rem); - return code; - }, - alpha: function(i) { - return String.fromCharCode(A + i); - }, + var renderer = grid.getRenderer(); + var columnEdges = renderer.getColumnEdges(); + var hdpiRatio = grid.getHiDPI(draggerCTX); + var columnWidth = grid.getColumnWidth(columnIndex); + var colHeight = grid.div.clientHeight; + var d = dragger; + var location = grid.div.getBoundingClientRect(); + var style = d.style; - getCellEditorAt: function(x, y) { - }, + style.top = location.top + 'px'; + style.left = location.left + 'px'; + style.position = 'fixed'; + style.opacity = 0.85; + style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)'; + //style.zIndex = 100; + style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor'); + style.backgroundColor = grid.renderer.resolveProperty('backgroundColor'); -}); + d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px'); + d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px'); -module.exports = DataModel; + style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px'; + style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px'; -},{"extend-me":4}],47:[function(require,module,exports){ -'use strict'; + var startX = columnEdges[columnIndex - scrollLeft]; + startX = startX * hdpiRatio; -//var analytics = require('hyper-analytics'); -//var analytics = require('../local_node_modules/hyper-analytics'); -var analytics = require('../local_node_modules/finanalytics'); -var DataModel = require('./DataModel'); -var images = require('../../images'); + draggerCTX.scale(hdpiRatio, hdpiRatio); -var UPWARDS_BLACK_ARROW = '\u25b2', // aka '▲' - DOWNWARDS_BLACK_ARROW = '\u25bc'; // aka '▼' + grid.renderOverridesCache.dragger = { + columnIndex: columnIndex, + ctx: draggerCTX, + startX: startX, + width: columnWidth, + height: colHeight, + hdpiratio: hdpiRatio + }; -var nullDataSource = { - isNullObject: function() { - return true; - }, - getFields: function() { - return []; - }, - getHeaders: function() { - return []; - }, - getColumnCount: function() { - return 0; - }, - getRowCount: function() { - return 0; - }, - getAggregateTotals: function() { - return []; - }, - hasAggregates: function() { - return false; - }, - hasGroups: function() { - return false; + this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, -5px)'); + style.zIndex = '5'; + style.cursor = this.getDraggingCursorName(); + grid.repaint(); }, - getRow: function() { - return null; - } -}; -/** - * @name dataModels.JSON - * @constructor - */ -var JSON = DataModel.extend('dataModels.JSON', { + /** + * @memberOf CellMoving.prototype + * @desc this function is the main dragging logic + * @param {Hypergrid} grid + * @param {number} x - the start position + */ + dragColumn: function(grid, x) { - //null object pattern for the source object - source: nullDataSource, + //TODO: this function is overly complex, refactor this in to something more reasonable + var self = this; + //var renderer = grid.getRenderer(); + //var columnEdges = renderer.getColumnEdges(); - preglobalfilter: nullDataSource, - prefilter: nullDataSource, + var autoScrollingNow = this.columnDragAutoScrollingRight || this.columnDragAutoScrollingLeft; - presorter: nullDataSource, - analytics: nullDataSource, - postglobalfilter: nullDataSource, - postfilter: nullDataSource, - postsorter: nullDataSource, + var hdpiRatio = grid.getHiDPI(draggerCTX); - topTotals: [], - bottomTotals: [], + var dragColumnIndex = grid.renderOverridesCache.dragger.columnIndex; + var columnWidth = grid.renderOverridesCache.dragger.width; - /** - * @memberOf dataModels.JSON.prototype - * @returns {boolean} - */ - hasAggregates: function() { - return this.analytics.hasAggregates(); - }, + var minX = 0; //grid.getFixedColumnsWidth(); + var maxX = grid.renderer.getFinalVisableColumnBoundary() - columnWidth; + x = Math.min(x, maxX + 15); + x = Math.max(minX - 15, x); - /** - * @memberOf dataModels.JSON.prototype - * @returns {boolean} - */ - hasGroups: function() { - return this.analytics.hasGroups(); - }, + //am I at my lower bound + var atMin = x < minX && dragColumnIndex !== 0; - getDataSource: function() { - return this.postsorter; //this.hasAggregates() ? this.analytics : this.presorter; - }, + //am I at my upper bound + var atMax = x > maxX; - getFilterSource: function() { - return this.postfilter; //this.hasAggregates() ? this.postfilter : this.prefilter; - }, + var d = dragger; - getGlobalFilterSource: function() { - return this.postglobalfilter; //this.hasAggregates() ? this.postfilter : this.prefilter; - }, + this.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + 0 + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease'); - getSortingSource: function() { - return this.postsorter; //this.hasAggregates() ? this.postsorter : this.presorter; - }, + this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, ' + -10 + 'px)'); + requestAnimationFrame(function() { + d.style.display = 'inline'; + }); - getData: function() { - return this.source.data; - }, + var overCol = grid.renderer.getColumnFromPixelX(x + (d.width / 2 / hdpiRatio)); - getFilteredData: function() { - var ds = this.getDataSource(); - var count = ds.getRowCount(); - var result = new Array(count); - for (var y = 0; y < count; y++) { - result[y] = ds.getRow(y); + if (atMin) { + overCol = 0; } - return result; - }, - /** - * @memberOf dataModels.JSON.prototype - * @param {number} x - * @param {number} y - * @returns {*} - */ - getValue: function(x, y) { - var hasHierarchyColumn = this.hasHierarchyColumn(); - var grid = this.getGrid(); - var headerRowCount = grid.getHeaderRowCount(); - var value; - if (hasHierarchyColumn) { - if (x === -2) { - x = 0; - } - } else if (this.hasAggregates()) { - x += 1; - } - if (y < headerRowCount) { - value = this.getHeaderRowValue(x, y); - return value; + if (atMax) { + overCol = grid.getColumnCount() - 1; } - // if (hasHierarchyColumn) { - // y += 1; - // } - value = this.getDataSource().getValue(x, y - headerRowCount); - return value; - }, - /** - * @memberOf dataModels.JSON.prototype - * @param {number} x - * @param {number} y - negative values refer to _bottom totals_ rows - * @returns {*} - */ - getHeaderRowValue: function(x, y) { - var value; - if (y === undefined) { - value = this.getHeaders()[Math.max(x, 0)]; - } else if (y < 0) { // bottom totals rows - var bottomTotals = this.getBottomTotals(); - value = bottomTotals[bottomTotals.length + y][x]; + var doAFloat = dragColumnIndex > overCol; + doAFloat = doAFloat || (overCol - dragColumnIndex >= 1); + + if (doAFloat && !atMax && !autoScrollingNow) { + var draggedToTheRight = dragColumnIndex < overCol; + // if (draggedToTheRight) { + // overCol = overCol - 1; + // } + if (this.isFloatingNow) { + return; + } + + this.isFloatingNow = true; + this.createFloatColumn(grid, overCol); + this.floatColumnTo(grid, draggedToTheRight); } else { - var grid = this.getGrid(), - isFilterRow = grid.isShowFilterRow(), - isHeaderRow = grid.isShowHeaderRow(), - topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0); - if (y >= topTotalsOffset) { // top totals rows - value = this.getTopTotals()[y - topTotalsOffset][x]; - } else if (isHeaderRow && y === 0) { - value = this.getHeaders()[x]; - var sortString = this.getSortImageForColumn(x); - if (sortString) { value = sortString + value; } - } else { // must be filter row - value = this.getFilter(x); - var icon = images.filter(value.length); - return [null, value, icon]; + + if (x < minX - 10) { + this.checkAutoScrollToLeft(grid, x); + } + if (x > minX - 10) { + this.columnDragAutoScrollingLeft = false; + } + //lets check for autoscroll to right if were up against it + if (atMax || x > maxX + 10) { + this.checkAutoScrollToRight(grid, x); + return; + } + if (x < maxX + 10) { + this.columnDragAutoScrollingRight = false; } } - return value; }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} x - * @param {number} y - * @param value + * @memberOf CellMoving.prototype + * @desc autoscroll to the right if necessary + * @param {Hypergrid} grid + * @param {number} x - the start position */ - setValue: function(x, y, value) { - var hasHierarchyColumn = this.hasHierarchyColumn(); - var grid = this.getGrid(); - var headerRowCount = grid.getHeaderRowCount(); - if (hasHierarchyColumn) { - if (x === -2) { - x = 0; - } - } else if (this.hasAggregates()) { - x += 1; - } - if (y < headerRowCount) { - this.setHeaderRowValue(x, y, value); - } else { - this.getDataSource().setValue(x, y - headerRowCount, value); + checkAutoScrollToRight: function(grid, x) { + if (this.columnDragAutoScrollingRight) { + return; } - this.changed(); + this.columnDragAutoScrollingRight = true; + this._checkAutoScrollToRight(grid, x); }, - /** - * @memberOf dataModels.JSON.prototype - * @param {number} x - * @param {number} y - * @param value - * @returns {*} - */ - setHeaderRowValue: function(x, y, value) { - if (value === undefined) { - return this._setHeader(x, y); // y is really the value + _checkAutoScrollToRight: function(grid, x) { + if (!this.columnDragAutoScrollingRight) { + return; } - var grid = this.getGrid(); - var isFilterRow = grid.isShowFilterRow(); - var isHeaderRow = grid.isShowHeaderRow(); - var isBoth = isFilterRow && isHeaderRow; - var topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0); - if (y >= topTotalsOffset) { - this.getTopTotals()[y - topTotalsOffset][x] = value; - } else if (x === -1) { - return; // can't change the row numbers - } else if (isBoth) { - if (y === 0) { - return this._setHeader(x, value); - } else { - this.setFilter(x, value); - } - } else if (isFilterRow) { - this.setFilter(x, value); - } else { - return this._setHeader(x, value); + var scrollLeft = grid.getHScrollValue(); + if (!grid.dragging || scrollLeft > (grid.sbHScrollConfig.rangeStop - 2)) { + return; } - return ''; - }, + var draggedIndex = grid.renderOverridesCache.dragger.columnIndex; + grid.scrollBy(1, 0); + var newIndex = draggedIndex + 1; + console.log(newIndex, draggedIndex); + grid.swapColumns(newIndex, draggedIndex); + grid.renderOverridesCache.dragger.columnIndex = newIndex; - /** - * @memberOf dataModels.JSON.prototype - * @param {number} colIndex - * @returns {*} - */ - getColumnProperties: function(colIndex) { - //access directly because we want it ordered - var column = this.getBehavior().allColumns[colIndex]; - if (column) { - return column.getProperties(); - } - return undefined; + setTimeout(this._checkAutoScrollToRight.bind(this, grid, x), 250); }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} colIndex - * @returns {*} + * @memberOf CellMoving.prototype + * @desc autoscroll to the left if necessary + * @param {Hypergrid} grid + * @param {number} x - the start position */ - getFilter: function(colIndex) { - var columnProperties = this.getColumnProperties(colIndex); - if (!columnProperties) { - return ''; + checkAutoScrollToLeft: function(grid, x) { + if (this.columnDragAutoScrollingLeft) { + return; } - return columnProperties.filter || ''; + this.columnDragAutoScrollingLeft = true; + this._checkAutoScrollToLeft(grid, x); }, - getComplexFilter: function(colIndex) { - var columnProperties = this.getColumnProperties(colIndex); - if (!columnProperties) { - return ''; + _checkAutoScrollToLeft: function(grid, x) { + if (!this.columnDragAutoScrollingLeft) { + return; } - var filter = columnProperties.complexFilter; - if (filter) { - var filterObject = this.getGrid().resolveFilter(filter.type); - var newFilter = filterObject.create(filter.state); - return function(data) { - var transformed = valueOrFunctionExecute(data); - return newFilter(transformed); - }; - } else { + + var scrollLeft = grid.getHScrollValue(); + if (!grid.dragging || scrollLeft < 1) { return; } + var draggedIndex = grid.renderOverridesCache.dragger.columnIndex; + grid.swapColumns(draggedIndex + scrollLeft, draggedIndex + scrollLeft - 1); + grid.scrollBy(-1, 0); + setTimeout(this._checkAutoScrollToLeft.bind(this, grid, x), 250); }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} colIndex - * @param value + * @memberOf CellMoving.prototype + * @desc a column drag has completed, update data and cleanup + * @param {Hypergrid} grid */ - setFilter: function(colIndex, value) { - var columnProperties = this.getColumnProperties(colIndex); - columnProperties.filter = value; - this.applyAnalytics(); - }, + endDragColumn: function(grid) { + + var fixedColumnCount = grid.getFixedColumnCount(); + var scrollLeft = grid.getHScrollValue(); + + var columnIndex = grid.renderOverridesCache.dragger.columnIndex; + + if (columnIndex < fixedColumnCount) { + scrollLeft = 0; + } + + var renderer = grid.getRenderer(); + var columnEdges = renderer.getColumnEdges(); + var self = this; + var startX = columnEdges[columnIndex - scrollLeft]; + var d = dragger; + + self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease'); + self.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -1 + 'px)'); + d.style.boxShadow = '0px 0px 0px #888888'; + + setTimeout(function() { + grid.renderOverridesCache.dragger = null; + grid.repaint(); + requestAnimationFrame(function() { + d.style.display = 'none'; + grid.endDragColumnNotification(); + }); + }, columnAnimationTime + 50); - /** - * @memberOf dataModels.JSON.prototype - * @returns {number} - */ - getColumnCount: function() { - var showTree = this.getGrid().resolveProperty('showTreeColumn') === true; - var hasAggregates = this.hasAggregates(); - var offset = (hasAggregates && !showTree) ? -1 : 0; - return this.analytics.getColumnCount() + offset; }, /** - * @memberOf dataModels.JSON.prototype - * @returns {number} + * @memberOf CellMoving.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - getRowCount: function() { - var grid = this.getGrid(); - var count = this.getDataSource().getRowCount(); - count += grid.getHeaderRowCount(); - return count; - }, + isHeaderRow: function(grid, event) { + var gridCell = event.viewPoint; + var isFixed = gridCell.y === 0; + return isFixed; + } - /** - * @memberOf dataModels.JSON.prototype - * @returns {string[]} - */ - getHeaders: function() { - return this.analytics.getHeaders(); - }, +}); - /** - * @memberOf dataModels.JSON.prototype - * @param {string[]} headers - */ - setHeaders: function(headers) { - this.getDataSource().setHeaders(headers); - }, +module.exports = ColumnMoving; - /** - * @memberOf dataModels.JSON.prototype - * @param {string[]} fields - */ - setFields: function(fields) { - this.getDataSource().setFields(fields); - }, +},{"./Feature.js":58}],54:[function(require,module,exports){ +/* eslint-env browser */ +/* global requestAnimationFrame */ + +'use strict'; + +var Feature = require('./Feature.js'); + +/** + * @constructor + */ +var ColumnPicker = Feature.extend('ColumnPicker', { + + alias: 'ColumnPicker', /** - * @memberOf dataModels.JSON.prototype - * @returns {string[]} + * @memberOf ColumnPicker.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - getFields: function() { - return this.getDataSource().getFields(); + handleKeyUp: function(grid, event) { + var key = event.detail.char.toLowerCase(); + var keys = grid.resolveProperty('editorActivationKeys'); + if (keys.indexOf(key) > -1) { + grid.toggleColumnPicker(); + } }, - /** - * @memberOf dataModels.JSON.prototype - * @param {object[]} dataRows - */ - setData: function(dataRows) { - this.source = new analytics.JSDataSource(dataRows); - //this.preglobalfilter = new analytics.DataSourceGlobalFilter(this.source); - //this.prefilter = new analytics.DataSourceFilter(this.preglobalfilter); - //this.presorter = new analytics.DataSourceSorterComposite(this.prefilter); +}); - this.analytics = new analytics.DataSourceAggregator(this.source); +module.exports = ColumnPicker; - this.postglobalfilter = new analytics.DataSourceGlobalFilter(this.analytics); - this.postfilter = new analytics.DataSourceFilter(this.postglobalfilter); - this.postsorter = new analytics.DataSourceSorterComposite(this.postfilter); +},{"./Feature.js":58}],55:[function(require,module,exports){ +'use strict'; - this.applyAnalytics(); +var Feature = require('./Feature.js'); - }, +/** + * @constructor + */ +var ColumnResizing = Feature.extend('ColumnResizing', { - /** - * @memberOf dataModels.JSON.prototype - * @param {Array} totalRows - */ - setTopTotals: function(totalRows) { - this.topTotals = totalRows; - }, + alias: 'ColumnResizing', /** - * @memberOf dataModels.JSON.prototype - * @returns {Array} + * the index of the column wall were currently dragging + * @type {number} + * @default -2 + * @memberOf ColumnResizing.prototype */ - getTopTotals: function() { - if (!this.hasAggregates()) { - return this.topTotals; - } - return this.getDataSource().getGrandTotals(); - }, + dragIndex: -2, /** - * @memberOf dataModels.JSON.prototype - * @param {Array} totalRows + * the pixel location of the where the drag was initiated + * @type {number} + * @default -1 + * @memberOf ColumnResizing.prototype */ - setBottomTotals: function(totalRows) { - this.bottomTotals = totalRows; - }, + dragStart: -1, /** - * @memberOf dataModels.JSON.prototype - * @returns {Array} + * the starting width/height of the row/column we are dragging + * @type {number} + * @default -1 + * @memberOf ColumnResizing.prototype */ - getBottomTotals: function() { - if (!this.hasAggregates()) { - return this.bottomTotals; - } - return this.getDataSource().getGrandTotals(); - }, + dragIndexStartingSize: -1, /** - * @memberOf dataModels.JSON.prototype - * @param groups + * @memberOf ColumnResizing.prototype + * @desc get the mouse x,y coordinate + * @returns {number} + * @param {MouseEvent} event - the mouse event to query */ - setGroups: function(groups) { - this.analytics.setGroupBys(groups); - this.applyAnalytics(); - this.getGrid().fireSyntheticGroupsChangedEvent(this.getGroups()); + getMouseValue: function(event) { + return event.primitiveEvent.detail.mouse.x; }, /** - * @memberOf dataModels.JSON.prototype - * @returns {object[]} + * @memberOf ColumnResizing.prototype + * @desc get the grid cell x,y coordinate + * @returns {number} + * @param {window.fin.rectangular.Point} gridCell */ - getGroups: function() { - var headers = this.getHeaders().slice(0); - var fields = this.getFields().slice(0); - var groupBys = this.analytics.groupBys; - var groups = []; - for (var i = 0; i < groupBys.length; i++) { - var field = headers[groupBys[i]]; - groups.push({ - id: groupBys[i], - label: field, - field: fields - }); - } - return groups; + getGridCellValue: function(gridCell) { + return gridCell.y; }, /** - * @memberOf dataModels.JSON.prototype - * @returns {object[]} + * @memberOf ColumnResizing.prototype + * @desc return the grids x,y scroll value + * @returns {number} + * @param {Hypergrid} grid */ - getAvailableGroups: function() { - var headers = this.source.getHeaders().slice(0); - var groupBys = this.analytics.groupBys; - var groups = []; - for (var i = 0; i < headers.length; i++) { - if (groupBys.indexOf(i) === -1) { - var field = headers[i]; - groups.push({ - id: i, - label: field, - field: field - }); - } - } - return groups; + getScrollValue: function(grid) { + return grid.getHScrollValue(); }, /** - * @memberOf dataModels.JSON.prototype - * @returns {object[]} + * @memberOf ColumnResizing.prototype + * @desc return the width/height of the row/column of interest + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest */ - getVisibleColumns: function() { - var items = this.getBehavior().columns; - items = items.filter(function(each) { - return each.label !== 'Tree'; - }); - return items; + getAreaSize: function(grid, index) { + return grid.getColumnWidth(index); }, /** - * @memberOf dataModels.JSON.prototype - * @returns {object[]} + * @memberOf ColumnResizing.prototype + * @desc set the width/height of the row/column at index + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest + * @param {number} value - the width/height to set to */ - getHiddenColumns: function() { - var visible = this.getBehavior().columns; - var all = this.getBehavior().allColumns; - var hidden = []; - for (var i = 0; i < all.length; i++) { - if (visible.indexOf(all[i]) === -1) { - hidden.push(all[i]); - } - } - hidden.sort(function(a, b) { - return a.label < b.label; - }); - return hidden; + setAreaSize: function(grid, index, value) { + grid.setColumnWidth(index, value); }, /** - * @memberOf dataModels.JSON.prototype - * @param aggregations + * @memberOf ColumnResizing.prototype + * @desc return the recently rendered area's width/height + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest */ - setAggregates: function(aggregations) { - this.quietlySetAggregates(aggregations); - this.applyAnalytics(); + getPreviousAbsoluteSize: function(grid, index) { + return grid.getRenderedWidth(index); }, /** - * @memberOf dataModels.JSON.prototype - * @param aggregations + * @memberOf ColumnResizing.prototype + * @desc returns the index of which divider I'm over + * @returns {number} + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - quietlySetAggregates: function(aggregations) { - this.analytics.setAggregates(aggregations); + overAreaDivider: function(grid, event) { + return grid.overColumnDivider(event); }, /** - * @memberOf dataModels.JSON.prototype + * @memberOf ColumnResizing.prototype + * @desc am I over the column/row area * @returns {boolean} + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - hasHierarchyColumn: function() { - var showTree = this.getGrid().resolveProperty('showTreeColumn') === true; - return this.hasAggregates() && this.hasGroups() && showTree; + isFirstFixedOtherArea: function(grid, event) { + return this.isFirstFixedRow(grid, event); }, /** - * @memberOf dataModels.JSON.prototype + * @memberOf ColumnResizing.prototype + * @desc return the cursor name + * @returns {string} */ - applyAnalytics: function() { - this.applyGroupBysAndAggregations(); - this.applyFilters(); - this.applySorts(); + getCursorName: function() { + return 'col-resize'; }, /** - * @memberOf dataModels.JSON.prototype + * @memberOf ColumnResizing.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - applyGroupBysAndAggregations: function() { - if (this.analytics.aggregates.length === 0) { - this.quietlySetAggregates({}); + handleMouseDrag: function(grid, event) { + if (this.dragIndex > -2) { + //var fixedAreaCount = this.getFixedAreaCount(grid); + //var offset = this.getFixedAreaSize(grid, fixedAreaCount + areaIndex); + var mouse = this.getMouseValue(event); + var scrollValue = this.getScrollValue(grid); + if (this.dragIndex < this.getFixedAreaCount(grid)) { + scrollValue = 0; + } + var previous = this.getPreviousAbsoluteSize(grid, this.dragIndex - scrollValue); + var distance = mouse - previous; + this.setAreaSize(grid, this.dragIndex, distance); + } else if (this.next) { + this.next.handleMouseDrag(grid, event); } - this.analytics.apply(); }, /** - * @memberOf dataModels.JSON.prototype + * @memberOf ColumnResizing.prototype + * @desc get the width/height of a specific row/column + * @param {Hypergrid} grid + * @param {number} areaIndex - the row/column index of interest */ - applyFilters: function() { - var grid = this.getGrid(); - var visibleColumns = this.getVisibleColumns(); - this.getGlobalFilterSource().apply(visibleColumns); - var details = []; - var filterSource = this.getFilterSource(); - var groupOffset = 0; //this.hasHierarchyColumn() ? 0 : 1; - - // apply column filters - filterSource.clearAll(); - - visibleColumns.forEach(function(column) { - var columnIndex = column.index, - filterText = this.getFilter(columnIndex), - formatterType = column.getProperties().format, - formatter = grid.getFormatter(formatterType), - complexFilter = this.getComplexFilter(columnIndex), - filter = complexFilter || filterText.length > 0 && textMatchFilter(filterText); - - if (filter) { - filterSource.add(columnIndex - groupOffset, this.createFormattedFilter(formatter, filter)); - details.push({ - column: column.label, - format: complexFilter ? 'complex' : formatterType - }); - } - }.bind(this)); - - filterSource.applyAll(); - - grid.fireSyntheticFilterAppliedEvent({ - details: details - }); + getSize: function(grid, areaIndex) { + return this.getAreaSize(grid, areaIndex); }, - createFormattedFilter: function(formatter, filter) { - return function(value) { - var formattedValue = formatter(value); - return filter(formattedValue); - }; + /** + * @memberOf ColumnResizing.prototype + * @desc return the fixed area rows/columns count + * @returns {number} + * @param {Hypergrid} grid + */ + getOtherFixedAreaCount: function(grid) { + return grid.getFixedRowCount(); }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} colIndex - * @param keys + * @memberOf ColumnResizing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - toggleSort: function(colIndex, keys) { - this.incrementSortState(colIndex, keys); - this.applyAnalytics(); + handleMouseDown: function(grid, event) { + var isEnabled = this.isEnabled(grid); + var overArea = this.overAreaDivider(grid, event); + if (isEnabled && overArea > -1 && this.isFirstFixedOtherArea(grid, event)) { + var scrollValue = this.getScrollValue(grid); + if (overArea < this.getFixedAreaCount(grid)) { + scrollValue = 0; + } + this.dragIndex = overArea - 1 + scrollValue; + this.dragStart = this.getMouseValue(event); + this.dragIndexStartingSize = 0; + this.detachChain(); + } else if (this.next) { + this.next.handleMouseDown(grid, event); + } }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} colIndex - * @param {string[]} keys + * @memberOf ColumnResizing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - incrementSortState: function(colIndex, keys) { - colIndex++; //hack to get around 0 index - var state = this.getPrivateState(); - var hasCTRL = keys.indexOf('CTRL') > -1; - state.sorts = state.sorts || []; - var already = state.sorts.indexOf(colIndex); - if (already === -1) { - already = state.sorts.indexOf(-1 * colIndex); + handleMouseUp: function(grid, event) { + var isEnabled = this.isEnabled(grid); + if (isEnabled && this.dragIndex > -2) { + this.cursor = null; + this.dragIndex = -2; + + event.primitiveEvent.stopPropagation(); + //delay here to give other events a chance to be dropped + var self = this; + grid.synchronizeScrollingBoundries(); + setTimeout(function() { + self.attachChain(); + }, 200); + } else if (this.next) { + this.next.handleMouseUp(grid, event); } - if (already > -1) { - if (state.sorts[already] > 0) { - state.sorts[already] = -1 * state.sorts[already]; - } else { - state.sorts.splice(already, 1); - } - } else if (hasCTRL || state.sorts.length === 0) { - state.sorts.unshift(colIndex); - } else { - state.sorts.length = 0; - state.sorts.unshift(colIndex); + }, + + /** + * @memberOf ColumnResizing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleMouseMove: function(grid, event) { + if (this.dragIndex > -2) { + return; } - if (state.sorts.length > 3) { - state.sorts.length = 3; + this.cursor = null; + if (this.next) { + this.next.handleMouseMove(grid, event); } + this.checkForAreaResizeCursorChange(grid, event); }, /** - * @memberOf dataModels.JSON.prototype + * @memberOf ColumnResizing.prototype + * @desc fill this in + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - applySorts: function() { - var sortingSource = this.getSortingSource(); - var sorts = this.getPrivateState().sorts; - var groupOffset = this.hasAggregates() ? 1 : 0; - if (!sorts || sorts.length === 0) { - sortingSource.clearSorts(); + checkForAreaResizeCursorChange: function(grid, event) { + var isEnabled = this.isEnabled(grid); + if (isEnabled && this.overAreaDivider(grid, event) > -1 && this.isFirstFixedOtherArea(grid, event)) { + this.cursor = this.getCursorName(); } else { - for (var i = 0; i < sorts.length; i++) { - var colIndex = Math.abs(sorts[i]) - 1; - var type = sorts[i] < 0 ? -1 : 1; - sortingSource.sortOn(colIndex - groupOffset, type); - } + this.cursor = null; } - sortingSource.applySorts(); + }, /** - * @memberOf dataModels.JSON.prototype - * @param index - * @param returnAsString - * @returns {*} + * @param {Hypergrid} grid + * @returns {number} + * @default -2 + * @memberOf ColumnResizing.prototype */ - getSortImageForColumn: function(index) { - index++; - var up = true; - var sorts = this.getPrivateState().sorts; - if (!sorts) { - return null; - } - var position = sorts.indexOf(index); - if (position < 0) { - position = sorts.indexOf(-1 * index); - up = false; - } - if (position < 0) { - return null; - } - var rank = sorts.length - position; - var arrow = up ? UPWARDS_BLACK_ARROW : DOWNWARDS_BLACK_ARROW; - return rank + arrow + ' '; + getFixedAreaCount: function(grid) { + var count = grid.getFixedColumnCount() + (grid.isShowRowNumbers() ? 1 : 0) + (grid.hasHierarchyColumn() ? 1 : 0); + return count; }, /** - * @memberOf dataModels.JSON.prototype - * @param cell + * @param {Hypergrid} grid * @param event + * @default -2 + * @memberOf ColumnResizing.prototype */ - cellClicked: function(cell, event) { - if (!this.hasAggregates()) { - return; - } - if (event.gridCell.x !== 0) { - return; // this wasn't a click on the hierarchy column - } - var grid = this.getGrid(); + handleDoubleClick: function(grid, event) { + var isEnabled = this.isEnabled(grid); + var hasCursor = this.overAreaDivider(grid, event) > -1; //this.cursor !== null; var headerRowCount = grid.getHeaderRowCount(); - var y = event.gridCell.y - headerRowCount; - this.getDataSource().click(y); - this.applyFilters(); - this.applySorts(); - this.changed(); + //var headerColCount = grid.getHeaderColumnCount(); + var gridCell = event.gridCell; + if (isEnabled && hasCursor && (gridCell.y <= headerRowCount)) { + grid.autosizeColumn(gridCell.x - 1); + } else if (this.next) { + this.next.handleDoubleClick(grid, event); + } }, /** - * @memberOf dataModels.JSON.prototype - * @param {number} y - * @returns {object} + * @param {Hypergrid} grid + * @returns {boolean} + * @default -2 + * @memberOf ColumnResizing.prototype + */ + isEnabled: function(grid) { + return true; + } + +}); + +module.exports = ColumnResizing; + +},{"./Feature.js":58}],56:[function(require,module,exports){ +'use strict'; + +var Feature = require('./Feature.js'); + +/** + * Extra msecs to avoid race condition with fincanvas's double click timer. + * @type {number} + * @defaultvalue 50 + * NOTE: 50 msecs seems to work well. 10 and even 25 proved insufficient in Chrome. + * @private + */ +var RACE_TIME = 50; + +/** + * @constructor + */ +var ColumnSelection = Feature.extend('ColumnSelection', { + + alias: 'ColumnSelection', + + /** + * The pixel location of the mouse pointer during a drag operation. + * @type {window.fin.rectangular.Point} + * @default null + * @memberOf ColumnSelection.prototype + */ + currentDrag: null, + + /** + * The cell coordinates of the where the mouse pointer is during a drag operation. + * @type {Object} + * @default null + * @memberOf ColumnSelection.prototype */ - getRow: function(y) { - var grid = this.getGrid(); - var headerRowCount = grid.getHeaderRowCount(); - if (y < headerRowCount && !this.hasAggregates()) { - var topTotals = this.getTopTotals(); - return topTotals[y - (headerRowCount - topTotals.length)]; - } - return this.getDataSource().getRow(y - headerRowCount); - }, + lastDragCell: null, /** - * @memberOf dataModels.JSON.prototype - * @param {number} y - * @returns {object} + * a millisecond value representing the previous time an autoscroll started + * @type {number} + * @default 0 + * @memberOf ColumnSelection.prototype */ - buildRow: function(y) { - var colCount = this.getColumnCount(); - var fields = [].concat(this.getFields()); - var result = {}; - if (this.hasAggregates()) { - result.tree = this.getValue(-2, y); - fields.shift(); - } - for (var i = 0; i < colCount; i++) { - result[fields[i]] = this.getValue(i, y); - } - return result; - }, + sbLastAuto: 0, /** - * @memberOf dataModels.JSON.prototype - * @param {number} y - * @returns {object} + * a millisecond value representing the time the current autoscroll started + * @type {number} + * @default 0 + * @memberOf ColumnSelection.prototype */ - getComputedRow: function(y) { - var rcf = this.getRowContextFunction([y]); - var fields = this.getFields(); - var row = {}; - for (var i = 0; i < fields.length; i++) { - var field = fields[i]; - row[field] = rcf(field)[0]; - } - return row; - }, + sbAutoStart: 0, + /** - * @memberOf dataModels.JSON.prototype - * @param {string} fieldName - * @param {number} y - * @returns {*} + * @memberOf ColumnSelection.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - getValueByField: function(fieldName, y) { - var index = this.getFields().indexOf(fieldName); - if (this.hasAggregates()) { - y += 1; + handleMouseUp: function(grid, event) { + if (this.dragging) { + this.dragging = false; + } + if (this.next) { + this.next.handleMouseUp(grid, event); } - return this.getDataSource().getValue(index, y); }, - /** - * @memberOf dataModels.JSON.prototype - * @param {sring} string - */ - setGlobalFilter: function(string) { - var globalFilterSource = this.getGlobalFilterSource(); - if (!string || string.length === 0) { - globalFilterSource.clear(); - } else { - globalFilterSource.set(textMatchFilter(string)); + handleDoubleClick: function(grid, event) { + if (this.doubleClickTimer) { + clearTimeout(this.doubleClickTimer); // prevent mouseDown from continuing + this.doubleClickTimer = undefined; + } + if (this.next) { + this.next.handleDoubleClick(grid, event); } - this.applyAnalytics(); }, /** - * @memberOf dataModels.JSON.prototype - * @param {object} config - * @param {number} x - * @param {number} y - * @param {number} untranslatedX - * @param {number} untranslatedY - * @returns {object} + * @memberOf ColumnSelection.prototype + * @desc * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) { - var renderer; - var provider = this.getGrid().getCellProvider(); - - config.x = x; - config.y = y; - config.untranslatedX = untranslatedX; - config.untranslatedY = untranslatedY; + handleMouseDown: function(grid, event) { + if (this.doubleClickTimer) { + return; + } - renderer = provider.getCell(config); - renderer.config = config; + if ((!grid.isColumnSelection() || event.mousePoint.y < 5) && this.next) { + this.next.handleMouseDown(grid, event); + return; + } - return renderer; - }, + var isRightClick = event.primitiveEvent.detail.isRightClick; + var cell = event.gridCell; + var viewCell = event.viewPoint; + var dx = cell.x; + var dy = cell.y; - /** - * @memberOf dataModels.JSON.prototype - */ - applyState: function() { - this.applyAnalytics(); - }, + var isHeader = grid.isShowHeaderRow() && dy === 0 && dx !== -1; - /** - * @memberOf dataModels.JSON.prototype - */ - reset: function() { - this.setData([]); - }, + if (isRightClick || !isHeader) { + if (this.next) { + this.next.handleMouseDown(grid, event); + } + } else { + // HOLD OFF WHILE WAITING FOR DOUBLE-CLICK + this.doubleClickTimer = setTimeout(function() { + this.doubleClickTimer = undefined; + var numFixedColumns = grid.getFixedColumnCount(); + + //if we are in the fixed area do not apply the scroll values + //check both x and y values independently + if (viewCell.x < numFixedColumns) { + dx = viewCell.x; + } - getUnfilteredValue: function(x, y) { - return this.source.getValue(x, y); - }, + var dCell = grid.newPoint(dx, 0); - getUnfilteredRowCount: function() { - return this.source.getRowCount(); + var primEvent = event.primitiveEvent; + var keys = primEvent.detail.keys; + this.dragging = true; + this.extendSelection(grid, dCell, keys); + }.bind(this), grid.resolveProperty('doubleClickDelay') + RACE_TIME); + } }, -}); - -function valueOrFunctionExecute(valueOrFunction) { - return typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction; -} + /** + * @memberOf ColumnSelection.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleMouseDrag: function(grid, event) { -function textMatchFilter(string) { - string = string.toLowerCase(); - return function(each) { - each = valueOrFunctionExecute(each); - return (each + '').toLowerCase().indexOf(string) > -1; - }; -} + if ((!grid.isColumnSelection() || this.isColumnDragging(grid)) && this.next) { + this.next.handleMouseDrag(grid, event); + return; + } -module.exports = JSON; + var isRightClick = event.primitiveEvent.detail.isRightClick; -},{"../../images":2,"../local_node_modules/finanalytics":81,"./DataModel":46}],48:[function(require,module,exports){ -/* eslint-env browser */ + if (isRightClick || !this.dragging) { + if (this.next) { + this.next.handleMouseDrag(grid, event); + } + } else { -'use strict'; + var numFixedColumns = grid.getFixedColumnCount(); -var LRUCache = require('lru-cache'); + var cell = event.gridCell; + var viewCell = event.viewPoint; + var dx = cell.x; + var dy = cell.y; -var renderCellError = require('./renderCellError'); + //if we are in the fixed area do not apply the scroll values + //check both x and y values independently + if (viewCell.x < numFixedColumns) { + dx = viewCell.x; + } -/** - * This module lists the properties that can be set on a {@link Hypergrid} along with their default values. - * Edit this file to override the defaults. - * @module defaults - */ + var dCell = grid.newPoint(dx, dy); -module.exports = { + var primEvent = event.primitiveEvent; + this.currentDrag = primEvent.detail.mouse; + this.lastDragCell = dCell; - //these are for the theme + this.checkDragScroll(grid, this.currentDrag); + this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys); + } + }, /** - * The font for data cells. - * @default '13px Tahoma, Geneva, sans-serif' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - font: '13px Tahoma, Geneva, sans-serif', + handleKeyDown: function(grid, event) { + if (grid.getLastSelectionType() !== 'column') { + if (this.next) { + this.next.handleKeyDown(grid, event); + } + return; + } + var command = 'handle' + event.detail.char; + if (this[command]) { + this[command].call(this, grid, event.detail); + } + }, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc Handle a mousedrag selection + * @param {Hypergrid} grid + * @param {Object} mouse - the event details + * @param {Array} keys - array of the keys that are currently pressed down */ - color: 'rgb(25, 25, 25)', + handleMouseDragCellSelection: function(grid, gridCell, keys) { + var x = gridCell.x; + // var previousDragExtent = grid.getDragExtent(); + var mouseDown = grid.getMouseDown(); + + var newX = x - mouseDown.x; + //var newY = y - mouseDown.y; + + // if (previousDragExtent.x === newX && previousDragExtent.y === newY) { + // return; + // } + + grid.clearMostRecentColumnSelection(); + + grid.selectColumn(mouseDown.x, x); + grid.setDragExtent(grid.newPoint(newX, 0)); + + grid.repaint(); + }, /** - * @default 'rgb(241, 241, 241)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above) + * @param {Hypergrid} grid + * @param {Object} mouse - the event details */ - backgroundColor: 'rgb(241, 241, 241)', + checkDragScroll: function(grid, mouse) { + if (!grid.resolveProperty('scrollingEnabled')) { + return; + } + var b = grid.getDataBounds(); + var inside = b.contains(mouse); + if (inside) { + if (grid.isScrollingNow()) { + grid.setScrollingNow(false); + } + } else if (!grid.isScrollingNow()) { + grid.setScrollingNow(true); + this.scrollDrag(grid); + } + }, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly + * @param {Hypergrid} grid */ - foregroundSelectionColor: 'rgb(25, 25, 25)', + scrollDrag: function(grid) { - /** - * @default 'rgb(183, 219, 255)' - * @type {string} - * @instance - */ - backgroundSelectionColor: 'rgb(183, 219, 255)', + if (!grid.isScrollingNow()) { + return; + } + var lastDragCell = this.lastDragCell; + var b = grid.getDataBounds(); + var xOffset = 0; + var yOffset = 0; - /** - * @default '12px Tahoma, Geneva, sans-serif' - * @type {string} - * @instance - */ - columnHeaderFont: '12px Tahoma, Geneva, sans-serif', + var numFixedColumns = grid.getFixedColumnCount(); + var numFixedRows = grid.getFixedRowCount(); - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - columnHeaderColor: 'rgb(25, 25, 25)', + var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns; + var dragEndInFixedAreaY = lastDragCell.y < numFixedRows; - /** - * @default 'rgb(223, 227, 232)' - * @type {string} - * @instance - */ - columnHeaderBackgroundColor: 'rgb(223, 227, 232)', + if (this.currentDrag.x < b.origin.x) { + xOffset = -1; + } - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - columnHeaderForegroundSelectionColor: 'rgb(25, 25, 25)', + if (this.currentDrag.x > b.origin.x + b.extent.x) { + xOffset = 1; + } - /** - * @default 'rgb(255, 220, 97)' - * @type {string} - * @instance - */ - columnHeaderBackgroundSelectionColor: 'rgb(255, 220, 97)', + var dragCellOffsetX = xOffset; + var dragCellOffsetY = yOffset; - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - columnHeaderForegroundColumnSelectionColor: 'rgb(25, 25, 25)', + if (dragEndInFixedAreaX) { + dragCellOffsetX = 0; + } - /** - * @default 'rgb(255, 180, 0)' - * @type {string} - * @instance - */ - columnHeaderBackgroundColumnSelectionColor: 'rgb(255, 180, 0)', + if (dragEndInFixedAreaY) { + dragCellOffsetY = 0; + } + this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY); + grid.scrollBy(xOffset, yOffset); + this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection + grid.repaint(); + setTimeout(this.scrollDrag.bind(this, grid), 25); + }, /** - * @default '12px Tahoma, Geneva, sans-serif' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc extend a selection or create one if there isnt yet + * @param {Hypergrid} grid + * @param {Object} gridCell - the event details + * @param {Array} keys - array of the keys that are currently pressed down */ - rowHeaderFont: '12px Tahoma, Geneva, sans-serif', + extendSelection: function(grid, gridCell, keys) { + grid.stopEditing(); + //var hasCTRL = keys.indexOf('CTRL') !== -1; + var hasSHIFT = keys.indexOf('SHIFT') !== -1; - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - rowHeaderColor: 'rgb(25, 25, 25)', + // var scrollTop = grid.getVScrollValue(); + // var scrollLeft = grid.getHScrollValue(); - /** - * @default 'rgb(223, 227, 232)' - * @type {string} - * @instance - */ - rowHeaderBackgroundColor: 'rgb(223, 227, 232)', + // var numFixedColumns = 0;//grid.getFixedColumnCount(); + // var numFixedRows = 0;//grid.getFixedRowCount(); - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - rowHeaderForegroundSelectionColor: 'rgb(25, 25, 25)', + var mousePoint = grid.getMouseDown(); + var x = gridCell.x; // - numFixedColumns + scrollLeft; + var y = gridCell.y; // - numFixedRows + scrollTop; - /** - * @default 'rgb(255, 220, 97)' - * @type {string} - * @instance - */ - rowHeaderBackgroundSelectionColor: 'rgb(255, 220, 97)', + //were outside of the grid do nothing + if (x < 0 || y < 0) { + return; + } - /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance - */ - rowHeaderForegroundRowSelectionColor: 'rgb(25, 25, 25)', + //we have repeated a click in the same spot deslect the value from last time + // if (mousePoint && x === mousePoint.x && y === mousePoint.y) { + // grid.clearSelections(); + // grid.popMouseDown(); + // grid.repaint(); + // return; + // } - /** - * @default 'rgb(255, 180, 0)' - * @type {string} - * @instance - */ - rowHeaderBackgroundRowSelectionColor: 'rgb(255, 180, 0)', + // if (!hasCTRL && !hasSHIFT) { + // grid.clearSelections(); + // } + if (hasSHIFT) { + grid.clearMostRecentColumnSelection(); + grid.selectColumn(x, mousePoint.x); + grid.setDragExtent(grid.newPoint(x - mousePoint.x, 0)); + } else { + grid.toggleSelectColumn(x, keys); + grid.setMouseDown(grid.newPoint(x, y)); + grid.setDragExtent(grid.newPoint(0, 0)); + } + grid.repaint(); + }, - /** - * @default '12px Tahoma, Geneva, sans-serif' - * @type {string} - * @instance - */ - filterFont: '12px Tahoma, Geneva, sans-serif', /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid */ - filterColor: 'rgb(25, 25, 25)', + handleDOWNSHIFT: function(grid) {}, /** - * @default 'white' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - filterBackgroundColor: 'white', + handleUPSHIFT: function(grid) {}, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - filterForegroundSelectionColor: 'rgb(25, 25, 25)', + handleLEFTSHIFT: function(grid) { + this.moveShiftSelect(grid, -1); + }, /** - * @default 'rgb(255, 220, 97)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - filterBackgroundSelectionColor: 'rgb(255, 220, 97)', + handleRIGHTSHIFT: function(grid) { + this.moveShiftSelect(grid, 1); + }, /** - * @default 'rgba(0,0,0,0.8)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - filterCellBorderStyle: 'rgba(0,0,0,0.8)', + handleDOWN: function(grid) { - /** - * @default 0.4 - * @type {number} - * @instance - */ - filterCellBorderThickness: 0.4, + // var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); + // var maxRows = grid.getRowCount() - 1; + + // var newX = mouseCorner.x; + // var newY = grid.getHeaderRowCount() + grid.getVScrollValue(); + + // newY = Math.min(maxRows, newY); + + // grid.clearSelections(); + // grid.select(newX, newY, 0, 0); + // grid.setMouseDown(new grid.rectangular.Point(newX, newY)); + // grid.setDragExtent(new grid.rectangular.Point(0, 0)); + // grid.repaint(); + }, /** - * @default '12px Tahoma, Geneva, sans-serif' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - treeColumnFont: '12px Tahoma, Geneva, sans-serif', + handleUP: function(grid) {}, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - treeColumnColor: 'rgb(25, 25, 25)', + handleLEFT: function(grid) { + this.moveSingleSelect(grid, -1); + }, /** - * @default 'rgb(223, 227, 232)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc handle this event + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - treeColumnBackgroundColor: 'rgb(223, 227, 232)', + handleRIGHT: function(grid) { + this.moveSingleSelect(grid, 1); + }, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc If we are holding down the same navigation key, accelerate the increment we scroll + * #### returns: integer */ - treeColumnForegroundSelectionColor: 'rgb(25, 25, 25)', + getAutoScrollAcceleration: function() { + var count = 1; + var elapsed = this.getAutoScrollDuration() / 2000; + count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed)); + return count; + }, /** - * @default 'rgb(255, 220, 97)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc set the start time to right now when we initiate an auto scroll */ - treeColumnBackgroundSelectionColor: 'rgb(255, 220, 97)', + setAutoScrollStartTime: function() { + this.sbAutoStart = Date.now(); + }, /** - * @default 'rgb(25, 25, 25)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time */ - treeColumnForegroundColumnSelectionColor: 'rgb(25, 25, 25)', + pingAutoScroll: function() { + var now = Date.now(); + if (now - this.sbLastAuto > 500) { + this.setAutoScrollStartTime(); + } + this.sbLastAuto = Date.now(); + }, /** - * @default 'rgb(255, 180, 0)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc answer how long we have been auto scrolling + * #### returns: integer */ - treeColumnBackgroundColumnSelectionColor: 'rgb(255, 180, 0)', + getAutoScrollDuration: function() { + if (Date.now() - this.sbLastAuto > 500) { + return 0; + } + return Date.now() - this.sbAutoStart; + }, /** - * @default 'rgb(201, 201, 201)' - * @type {string} - * @instance + * @memberOf ColumnSelection.prototype + * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary. + * @param {Hypergrid} grid + * @param {number} offsetX - x coordinate to start at + * @param {number} offsetY - y coordinate to start at */ - backgroundColor2: 'rgb(201, 201, 201)', + moveShiftSelect: function(grid, offsetX) { - /** - * @default 0 - * @type {number} - * @instance - */ - voffset: 0, + var maxColumns = grid.getColumnCount() - 1; - /** - * @default 'visible' - * @type {string} - * @instance - */ - scrollbarHoverOver: 'visible', + var maxViewableColumns = grid.getVisibleColumns() - 1; - /** - * @default 'hidden' - * @type {string} - * @instance - */ - scrollbarHoverOff: 'hidden', + if (!grid.resolveProperty('scrollingEnabled')) { + maxColumns = Math.min(maxColumns, maxViewableColumns); + } - /** - * @default true - * @type {boolean} - * @instance - */ - scrollingEnabled: true, + var origin = grid.getMouseDown(); + var extent = grid.getDragExtent(); - /** - * @default '' - * @type {string} - * @instance - */ - vScrollbarClassPrefix: '', + var newX = extent.x + offsetX; + //var newY = grid.getRowCount(); - /** - * @default '' - * @type {string} - * @instance - */ - hScrollbarClassPrefix: '', + newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX)); - //these used to be in the constants element + grid.clearMostRecentColumnSelection(); + grid.selectColumn(origin.x, origin.x + newX); - /** - * @default 'center' - * @type {string} - * @instance - */ - fixedRowAlign: 'center', + grid.setDragExtent(grid.newPoint(newX, 0)); - /** - * @default 'center' - * @type {string} - * @instance - */ - fixedColAlign: 'center', + if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) { + this.pingAutoScroll(); + } - /** - * @default 5 - * @type {number} - * @instance - */ - cellPadding: 5, + grid.repaint(); - /** - * @default true - * @type {boolean} - * @instance - */ - gridLinesH: true, + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf ColumnSelection.prototype + * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent. + * @param {Hypergrid} grid + * @param {number} offsetX - x coordinate to start at + * @param {number} offsetY - y coordinate to start at */ - gridLinesV: true, + moveSingleSelect: function(grid, offsetX) { - /** - * @default 'rgb(199, 199 199)' - * @type {string} - * @instance - */ - lineColor: 'rgb(199, 199, 199)', + var maxColumns = grid.getColumnCount() - 1; - /** - * @default 0.4 - * @type {number} - * @instance - */ - lineWidth: 0.4, + var maxViewableColumns = grid.getVisibleColumnsCount() - 1; + if (!grid.resolveProperty('scrollingEnabled')) { + maxColumns = Math.min(maxColumns, maxViewableColumns); + } + + var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); + + var newX = mouseCorner.x + offsetX; + //var newY = grid.getRowCount(); + + newX = Math.min(maxColumns, Math.max(0, newX)); + + grid.clearSelections(); + grid.selectColumn(newX); + grid.setMouseDown(grid.newPoint(newX, 0)); + grid.setDragExtent(grid.newPoint(0, 0)); + + if (grid.insureModelColIsVisible(newX, offsetX)) { + this.pingAutoScroll(); + } + + grid.repaint(); + + }, + + isColumnDragging: function(grid) { + var dragger = grid.lookupFeature('ColumnMoving'); + if (!dragger) { + return false; + } + var isActivated = dragger.dragging && !this.dragging; + return isActivated; + } + +}); + +module.exports = ColumnSelection; + +},{"./Feature.js":58}],57:[function(require,module,exports){ +'use strict'; + +var Feature = require('./Feature.js'); + +/** + * @constructor + */ +var ColumnSorting = Feature.extend('ColumnSorting', { + + alias: 'ColumnSorting', /** - * @default 15 - * @type {number} - * @instance + * @memberOf ColumnSorting.prototype + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - defaultRowHeight: 15, + + handleDoubleClick: function(grid, event) { + var gridCell = event.gridCell; + if (grid.isShowHeaderRow() && gridCell.y === 0 && gridCell.x !== -1) { + var keys = event.primitiveEvent.detail.keys; + grid.toggleSort(gridCell.x, keys); + } else if (this.next) { + this.next.handleDoubleClick(grid, event); + } + }, /** - * @default 100 - * @type {number} - * @instance + * @memberOf ColumnSorting.prototype + * @desc * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - defaultColumnWidth: 100, + handleMouseMove: function(grid, event) { + var y = event.gridCell.y; + if (this.isFixedRow(grid, event) && y < 1) { + this.cursor = 'pointer'; + } else { + this.cursor = null; + } + if (this.next) { + this.next.handleMouseMove(grid, event); + } + } - //for immediate painting, set these values to 0, true respectively +}); - /** - * @default 60 - * @type {number} - * @instance - */ - repaintIntervalRate: 60, +module.exports = ColumnSorting; - /** - * - * @instance - */ - repaintImmediately: false, +},{"./Feature.js":58}],58:[function(require,module,exports){ +'use strict'; - //enable or disable double buffering +var Base = require('../lib/Base'); + +/** + * @constructor + * @desc instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid. + */ +var Feature = Base.extend('Feature', { /** - * @default false - * @type {boolean} - * @instance + * the next feature to be given a chance to handle incoming events + * @type {Feature} + * @default null + * @memberOf Feature.prototype */ - useBitBlit: false, - + next: null, /** - * @default true - * @type {boolean} - * @instance + * a temporary holding field for my next feature when I'm in a disconnected state + * @type {Feature} + * @default null + * @memberOf Feature.prototype */ - useHiDPI: true, + detached: null, /** - * @default ['alt', 'esc'] + * the cursor I want to be displayed * @type {string} - * @instance + * @default null + * @memberOf Feature.prototype */ - editorActivationKeys: ['alt', 'esc'], + cursor: null, /** - * @default false - * @type {boolean} - * @instance + * the cell location where the cursor is currently + * @type {Point} + * @default null + * @memberOf Feature.prototype */ - readOnly: false, - - //inhertied by cell renderers + currentHoverCell: null, /** - * @default getTextWidth - * @type {function} - * @instance + * @memberOf Feature.prototype + * @desc set my next field, or if it's populated delegate to the feature in my next field + * @param {Feature} nextFeature - this is how we build the chain of responsibility */ - getTextWidth: getTextWidth, + setNext: function(nextFeature) { + if (this.next) { + this.next.setNext(nextFeature); + } else { + this.next = nextFeature; + this.detached = nextFeature; + } + }, /** - * @default getTextHeight - * @type {function} - * @instance + * @memberOf Feature.prototype + * @desc disconnect my child */ - getTextHeight: getTextHeight, - + detachChain: function() { + this.next = null; + }, /** - * @default 0 - * @type {number} - * @instance + * @memberOf Feature.prototype + * @desc reattach my child from the detached reference */ - fixedColumnCount: 0, + attachChain: function() { + this.next = this.detached; + }, /** - * @default 0 - * @type {number} - * @instance + * @memberOf Feature.prototype + * @desc handle mouse move down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - fixedRowCount: 0, + handleMouseMove: function(grid, event) { + if (this.next) { + this.next.handleMouseMove(grid, event); + } + }, /** - * @default 0 - * @type {number} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - headerColumnCount: 0, - + handleMouseExit: function(grid, event) { + if (this.next) { + this.next.handleMouseExit(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - showRowNumbers: true, + handleMouseEnter: function(grid, event) { + if (this.next) { + this.next.handleMouseEnter(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - showTreeColumn: true, + handleMouseDown: function(grid, event) { + if (this.next) { + this.next.handleMouseDown(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - showHeaderRow: true, + handleMouseUp: function(grid, event) { + if (this.next) { + this.next.handleMouseUp(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - showFilterRow: true, + handleKeyDown: function(grid, event) { + if (this.next) { + this.next.handleKeyDown(grid, event); + } + }, + /** + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleKeyUp: function(grid, event) { + if (this.next) { + this.next.handleKeyUp(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - cellSelection: true, + handleWheelMoved: function(grid, event) { + if (this.next) { + this.next.handleWheelMoved(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - columnSelection: true, + handleDoubleClick: function(grid, event) { + if (this.next) { + this.next.handleDoubleClick(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - rowSelection: true, + handleHoldPulse: function(grid, event) { + if (this.next) { + this.next.handleHoldPulse(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - singleRowSelectionMode: true, + handleTap: function(grid, event) { + if (this.next) { + this.next.handleTap(grid, event); + } + }, /** - * @default 'rgba(0, 0, 0, 0.2)' - * @type {string} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - selectionRegionOverlayColor: 'rgba(0, 0, 0, 0.2)', + handleMouseDrag: function(grid, event) { + if (this.next) { + this.next.handleMouseDrag(grid, event); + } + }, /** - * @default 'black' - * @type {string} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - selectionRegionOutlineColor: 'black', + handleContextMenu: function(grid, event) { + if (this.next) { + this.next.handleContextMenu(grid, event); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc toggle the column picker */ - columnAutosizing: true, + + moveSingleSelect: function(grid, x, y) { + if (this.next) { + this.next.moveSingleSelect(grid, x, y); + } + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - rowNumberAutosizing: true, + isFixedRow: function(grid, event) { + var gridCell = event.viewPoint; + var isFixed = gridCell.y < grid.getFixedRowCount(); + return isFixed; + }, /** - * @default false - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - headerTextWrapping: false, + isFirstFixedRow: function(grid, event) { + var gridCell = event.viewPoint; + var isFixed = gridCell.y < 1; + return isFixed; + }, /** - * @default false - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - rowResize: false, + isFixedColumn: function(grid, event) { + var gridCell = event.viewPoint; + var isFixed = gridCell.x < grid.getFixedColumnCount(); + return isFixed; + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - editable: true, + isFirstFixedColumn: function(grid, event) { + var gridCell = event.viewPoint; + var edge = grid.isShowRowNumbers() ? 0 : 1; + var isFixed = gridCell.x < edge; + return isFixed; + }, /** - * @default true - * @type {boolean} - * @instance + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - editOnDoubleClick: true, + isTopLeft: function(grid, event) { + var isTopLeft = this.isFixedRow(grid, event) && this.isFixedColumn(grid, event); + return isTopLeft; + }, /** - * @default renderCellError - * @type {function} + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - renderCellError: renderCellError, + setCursor: function(grid) { + if (this.next) { + this.next.setCursor(grid); + } + if (this.cursor) { + grid.beCursor(this.cursor); + } + }, /** - * @default false - * @type {boolean} + * @memberOf Feature.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details */ - checkboxOnlyRowSelections: false, - - format: 'default', - - hoverCellHighlight: true, - hoverRowHighlight: true, - hoverColumnHighlight: true, - - hoverCellColor: 'lightgray', - hoverRowColor: 'gray', - hoverColumnColor: 'gray', - - link: false, - strikeThrough: false, - -}; - -var textWidthCache = new LRUCache(2000); - -function getTextWidth(gc, string) { - if (string === null || string === undefined) { - return 0; - } - string = string + ''; - if (string.length === 0) { - return 0; - } - var key = gc.font + string; - var width = textWidthCache.get(key); - if (!width) { - width = gc.measureText(string).width; - textWidthCache.set(key, width); + initializeOn: function(grid) { + if (this.next) { + this.next.initializeOn(grid); + } } - return width; -} - -var fontData = {}; - -function getTextHeight(font) { - var result = fontData[font]; - - if (!result) { - result = {}; - - var text = document.createElement('span'); - text.textContent = 'Hg'; - text.style.font = font; - - var block = document.createElement('div'); - block.style.display = 'inline-block'; - block.style.width = '1px'; - block.style.height = '0px'; - - var div = document.createElement('div'); - div.appendChild(text); - div.appendChild(block); - div.style.position = 'absolute'; - document.body.appendChild(div); +}); - try { +module.exports = Feature; - block.style.verticalAlign = 'baseline'; +},{"../lib/Base":66}],59:[function(require,module,exports){ +'use strict'; - var blockRect = block.getBoundingClientRect(); - var textRect = text.getBoundingClientRect(); +var Feature = require('./Feature.js'); - result.ascent = blockRect.top - textRect.top; +/** + * @constructor + */ +var Filters = Feature.extend('Filters', { - block.style.verticalAlign = 'bottom'; - result.height = blockRect.top - textRect.top; + alias: 'Filters', - result.descent = result.height - result.ascent; + handleDoubleClick: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleDoubleClick(grid, event); + } + }, - } finally { - document.body.removeChild(div); + handleTap: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleTap(grid, event); } - if (result.height !== 0) { - fontData[font] = result; + }, + + /** + * @memberOf CellEditing.prototype + * @desc handle this event down the feature chain of responsibility + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + handleHoldPulse: function(grid, event) { + var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); + if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { + grid._activateEditor(event); + } else if (this.next) { + this.next.handleHoldPulse(grid, event); } + }, + + checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) { + var isFilterRow = grid.isFilterRow(event.gridCell.y); + var activateEditor = isDoubleClickEditorActivation && isFilterRow; + return activateEditor; } - return result; -} +}); -},{"./renderCellError":83,"lru-cache":16}],49:[function(require,module,exports){ -/* eslint-env browser */ +module.exports = Filters; +},{"./Feature.js":58}],60:[function(require,module,exports){ 'use strict'; -require('object-iterators'); // Install the Array.find polyfill, as needed +var Feature = require('./Feature.js'); -var Hypergrid = require('./Hypergrid'); +var commands = { + PAGEDOWN: function(grid) { grid.pageDown(); }, + PAGEUP: function(grid) { grid.pageUp(); }, + PAGELEFT: function(grid) { grid.pageLeft(); }, + PAGERIGHT: function(grid) { grid.pageRight(); } +}; -Hypergrid.images = require('../images'); -Hypergrid.behaviors = require('./behaviors/index'); -Hypergrid.cellEditors = require('./cellEditors/index'); -Hypergrid.features = require('./features/index'); +/** + * @constructor + */ +var KeyPaging = Feature.extend('KeyPaging', { + + alias: 'KeyPaging', + + /** + * @desc Handle this event down the feature chain of responsibility. + * @param {Hypergrid} grid + * @param {Object} event - the event details + * @memberOf KeyPaging.prototype + */ + handleKeyDown: function(grid, event) { + var detail = event.detail.char; + var func = commands[detail]; + if (func) { + func(grid); + } else if (this.next) { + this.next.handleKeyDown(grid, event); + } + } -window.fin = { - Hypergrid: Hypergrid, - FilterTree: require('filter-tree') -}; +}); -},{"../images":2,"./Hypergrid":25,"./behaviors/index":35,"./cellEditors/index":45,"./features/index":66,"filter-tree":5,"object-iterators":18}],50:[function(require,module,exports){ +module.exports = KeyPaging; + +},{"./Feature.js":58}],61:[function(require,module,exports){ 'use strict'; var Feature = require('./Feature.js'); @@ -19545,98 +20062,212 @@ var Feature = require('./Feature.js'); /** * @constructor */ -var CellClick = Feature.extend('CellClick', { +var OnHover = Feature.extend('OnHover', { - alias: 'CellClick', + alias: 'OnHover', /** - * @memberOf CellClick.prototype - * @desc Handle this event down the feature chain of responsibility + * @desc Hhandle this event down the feature chain of responsibility. * @param {Hypergrid} grid * @param {Object} event - the event details + * @memberOf OnHover.prototype */ - handleTap: function(grid, event) { - var gridCell = event.gridCell; - var behavior = grid.getBehavior(); - var headerRowCount = behavior.getHeaderRowCount(); - var headerColumnCount = behavior.getHeaderColumnCount(); - if ((gridCell.y >= headerRowCount) && - (gridCell.x >= headerColumnCount)) { - grid.cellClicked(event); - } else if (this.next) { - this.next.handleTap(grid, event); + handleMouseMove: function(grid, event) { + var currentHoverCell = grid.getHoverCell(); + if (!event.gridCell.equals(currentHoverCell)) { + if (currentHoverCell) { + this.handleMouseExit(grid, currentHoverCell); + } + this.handleMouseEnter(grid, event); + grid.setHoverCell(event.gridCell); + } else { + if (this.next) { + this.next.handleMouseMove(grid, event); + } } } + }); -module.exports = CellClick; +module.exports = OnHover; -},{"./Feature.js":59}],51:[function(require,module,exports){ +},{"./Feature.js":58}],62:[function(require,module,exports){ 'use strict'; -var Feature = require('./Feature.js'); +var ColumnResizing = require('./ColumnResizing'); /** * @constructor */ -var CellEditing = Feature.extend('CellEditing', { +var RowResizing = ColumnResizing.extend('RowResizing', { - alias: 'CellEditing', + alias: 'RowResizing', /** - * @memberOf CellEditing.prototype - * @desc handle this event down the feature chain of responsibility + * the index of the row/column we are dragging + * @type {number} + * @default -1 + * @memberOf RowResizing.prototype + */ + dragArea: -1, + + /** + * the pixel location of the where the drag was initiated + * @type {number} + * @default -1 + * @memberOf RowResizing.prototype + */ + dragStart: -1, + + /** + * the starting width/height of the row/column we are dragging + * @type {number} + * @default -1 + * @memberOf RowResizing.prototype + */ + dragAreaStartingSize: -1, + + /** + * @memberOf RowResizing.prototype + * @desc get the mouse x,y coordinate + * @returns {number} + * @param {MouseEvent} event - the mouse event to query + */ + getMouseValue: function(event) { + return event.primitiveEvent.detail.mouse.y; + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc get the grid cell x,y coordinate + * @returns {number} + * @param {Point} gridCell + */ + getGridCellValue: function(gridCell) { + return gridCell.x; + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc return the grids x,y scroll value + * @returns {number} * @param {Hypergrid} grid - * @param {Object} event - the event details */ - handleDoubleClick: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleDoubleClick(grid, event); - } + getScrollValue: function(grid) { + return grid.getVScrollValue(); }, - handleTap: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleTap(grid, event); - } + /** + * @function + * @memberOf RowResizing.prototype + * @desc return the width/height of the row/column of interest + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest + */ + getAreaSize: function(grid, index) { + return grid.getRowHeight(index); }, /** - * @memberOf CellEditing.prototype - * @desc handle this event down the feature chain of responsibility + * @function + * @memberOf RowResizing.prototype + * @desc set the width/height of the row/column at index + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest + * @param {number} value - the width/height to set to + */ + setAreaSize: function(grid, index, value) { + grid.setRowHeight(index, value); + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc returns the index of which divider I'm over + * @returns {number} * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleHoldPulse: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleHoldPulse(grid, event); - } + overAreaDivider: function(grid, event) { + return grid.overRowDivider(event); }, - checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) { - var behavior = grid.getBehavior(); - var headerRowCount = behavior.getHeaderRowCount(); - var headerColumnCount = behavior.getHeaderColumnCount(); - var gridCell = event.gridCell; - var isFilterRow = grid.isFilterRow(gridCell.y); - var activateEditor = isDoubleClickEditorActivation && gridCell.x >= headerColumnCount && (isFilterRow || gridCell.y >= headerRowCount); - return activateEditor; + /** + * @function + * @memberOf RowResizing.prototype + * @desc am I over the column/row area + * @returns {boolean} + * @param {Hypergrid} grid + * @param {Object} event - the event details + */ + isFirstFixedOtherArea: function(grid, event) { + return this.isFirstFixedColumn(grid, event); + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc return the cursor name + * @returns {string} + */ + getCursorName: function() { + return 'row-resize'; + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc return the recently rendered area's width/height + * @returns {number} + * @param {Hypergrid} grid + * @param {number} index - the row/column index of interest + */ + getPreviousAbsoluteSize: function(grid, index) { + return grid.getRenderedHeight(index); + }, + + /** + * @function + * @memberOf RowResizing.prototype + * @desc return the fixed area rows/columns count + * @returns {number} + * @param {Hypergrid} grid + */ + getOtherFixedAreaCount: function(grid) { + return grid.getFixedColumnCount(); + }, + + /** + * + * @param {Hypergrid} grid + * @returns {number} + * @default -2 + * @memberOf ColumnResizing.prototype + */ + getFixedAreaCount: function(grid) { + return grid.getFixedRowCount() + grid.getHeaderRowCount(); + }, + + /** + * + * @param {Hypergrid} grid + * @returns {boolean} + * @default -2 + * @memberOf ColumnResizing.prototype + */ + isEnabled: function(grid) { + return grid.isRowResizeable(); } }); -module.exports = CellEditing; +module.exports = RowResizing; -},{"./Feature.js":59}],52:[function(require,module,exports){ +},{"./ColumnResizing":55}],63:[function(require,module,exports){ 'use strict'; var Feature = require('./Feature.js'); @@ -19644,21 +20275,23 @@ var Feature = require('./Feature.js'); /** * @constructor */ -var CellSelection = Feature.extend('CellSelection', { +var RowSelection = Feature.extend('RowSelection', { - alias: 'CellSelection', + alias: 'RowSelection', /** * The pixel location of the mouse pointer during a drag operation. - * @type {window.fin.rectangular.Point} - * @memberOf CellSelection.prototype + * @type {Point} + * @default null + * @memberOf RowSelection.prototype */ currentDrag: null, /** - * the cell coordinates of the where the mouse pointer is during a drag operation + * The cell coordinates of the where the mouse pointer is during a drag operation. * @type {Object} - * @memberOf CellSelection.prototype + * @default null + * @memberOf RowSelection.prototype */ lastDragCell: null, @@ -19666,7 +20299,7 @@ var CellSelection = Feature.extend('CellSelection', { * a millisecond value representing the previous time an autoscroll started * @type {number} * @default 0 - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype */ sbLastAuto: 0, @@ -19674,107 +20307,103 @@ var CellSelection = Feature.extend('CellSelection', { * a millisecond value representing the time the current autoscroll started * @type {number} * @default 0 - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype */ sbAutoStart: 0, + dragArmed: false, + /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc Handle this event down the feature chain of responsibility. * @param {Hypergrid} grid * @param {Object} event - the event details */ handleMouseUp: function(grid, event) { - if (this.dragging) { + if (this.dragArmed) { + this.dragArmed = false; + //global row selection + if (event.gridCell.x === -1 && event.gridCell.y === 0) { + grid.toggleSelectAllRows(); + } + grid.fireSyntheticRowSelectionChangedEvent(); + } else if (this.dragging) { this.dragging = false; - } - if (this.next) { + grid.fireSyntheticRowSelectionChangedEvent(); + } else if (this.next) { this.next.handleMouseUp(grid, event); } }, /** - * @memberOf CellSelection.prototype - * @desc Handle this event down the feature chain of responsibility. + * @memberOf RowSelection.prototype + * @desc * @desc Handle this event down the feature chain of responsibility. * @param {Hypergrid} grid * @param {Object} event - the event details */ handleMouseDown: function(grid, event) { + var isRightClick = event.primitiveEvent.detail.isRightClick; - var behavior = grid.getBehavior(); var cell = event.gridCell; var viewCell = event.viewPoint; var dx = cell.x; var dy = cell.y; - var headerRowCount = behavior.getHeaderRowCount(); - var headerColumnCount = behavior.getHeaderColumnCount(); - var columnCount = behavior.getColumnCount(); - var isOutside = viewCell.x >= columnCount; - var isHeader = dy < headerRowCount || dx < headerColumnCount; - if (!grid.isCellSelection() || isRightClick || isHeader || isOutside) { + var isHeader = grid.isShowRowNumbers() && dx < 0; + + if (!grid.isRowSelection() || isRightClick || !isHeader) { if (this.next) { this.next.handleMouseDown(grid, event); } } else { - var numFixedColumns = grid.getFixedColumnCount(); var numFixedRows = grid.getFixedRowCount(); //if we are in the fixed area do not apply the scroll values //check both x and y values independently - if (viewCell.x < numFixedColumns) { - dx = viewCell.x; - } - if (viewCell.y < numFixedRows) { dy = viewCell.y; } - var dCell = grid.newPoint(dx, dy); + var dCell = grid.newPoint(0, dy); var primEvent = event.primitiveEvent; var keys = primEvent.detail.keys; - this.dragging = true; + this.dragArmed = true; this.extendSelection(grid, dCell, keys); } }, /** - * @memberOf CellSelection.prototype - * @desc Handle this event down the feature chain of responsibility. + * @memberOf RowSelection.prototype + * @desc handle this event down the feature chain of responsibility * @param {Hypergrid} grid * @param {Object} event - the event details */ handleMouseDrag: function(grid, event) { var isRightClick = event.primitiveEvent.detail.isRightClick; - if (!grid.isCellSelection() || isRightClick || !this.dragging) { + if (!this.dragArmed || !grid.isRowSelection() || isRightClick) { if (this.next) { this.next.handleMouseDrag(grid, event); } } else { - - var numFixedColumns = grid.getFixedColumnCount(); + this.dragging = true; var numFixedRows = grid.getFixedRowCount(); var cell = event.gridCell; var viewCell = event.viewPoint; - var dx = cell.x; + //var dx = cell.x; var dy = cell.y; //if we are in the fixed area do not apply the scroll values //check both x and y values independently - if (viewCell.x < numFixedColumns) { - dx = viewCell.x; - } - if (viewCell.y < numFixedRows) { dy = viewCell.y; } - var dCell = grid.newPoint(dx, dy); + var dCell = grid.newPoint(0, dy); var primEvent = event.primitiveEvent; this.currentDrag = primEvent.detail.mouse; @@ -19786,12 +20415,18 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype - * @desc Handle this event down the feature chain of responsibility. + * @memberOf RowSelection.prototype + * @desc handle this event down the feature chain of responsibility * @param {Hypergrid} grid * @param {Object} event - the event details */ handleKeyDown: function(grid, event) { + if (grid.getLastSelectionType() !== 'row') { + if (this.next) { + this.next.handleKeyDown(grid, event); + } + return; + } var command = 'handle' + event.detail.char; if (this[command]) { this[command].call(this, grid, event.detail); @@ -19799,44 +20434,34 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype - * @desc Handle a mousedrag selection. + * @memberOf RowSelection.prototype + * @desc Handle a mousedrag selection * @param {Hypergrid} grid * @param {Object} mouse - the event details * @param {Array} keys - array of the keys that are currently pressed down */ handleMouseDragCellSelection: function(grid, gridCell, keys) { - - var behavior = grid.getBehavior(); - var headerRowCount = behavior.getHeaderRowCount(); - var headerColumnCount = behavior.getHeaderColumnCount(); - var x = gridCell.x; var y = gridCell.y; - x = Math.max(headerColumnCount, x); - y = Math.max(headerRowCount, y); - - var previousDragExtent = grid.getDragExtent(); + // var previousDragExtent = grid.getDragExtent(); var mouseDown = grid.getMouseDown(); - //var scrollingNow = grid.isScrollingNow(); - - var newX = x - mouseDown.x; var newY = y - mouseDown.y; + //var newY = y - mouseDown.y; - if (previousDragExtent.x === newX && previousDragExtent.y === newY) { - return; - } + // if (previousDragExtent.x === newX && previousDragExtent.y === newY) { + // return; + // } - grid.clearMostRecentSelection(); + grid.clearMostRecentRowSelection(); - grid.select(mouseDown.x, mouseDown.y, newX, newY); - grid.setDragExtent(grid.newPoint(newX, newY)); + grid.selectRow(mouseDown.y, y); + grid.setDragExtent(grid.newPoint(0, newY)); grid.repaint(); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above) * @param {Hypergrid} grid * @param {Object} mouse - the event details @@ -19858,17 +20483,15 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly * @param {Hypergrid} grid */ scrollDrag: function(grid) { - if (!grid.isScrollingNow()) { return; } - var dragStartedInHeaderArea = grid.isMouseDownInHeaderArea(); var lastDragCell = this.lastDragCell; var b = grid.getDataBounds(); var xOffset = 0; @@ -19880,17 +20503,10 @@ var CellSelection = Feature.extend('CellSelection', { var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns; var dragEndInFixedAreaY = lastDragCell.y < numFixedRows; - if (!dragStartedInHeaderArea) { - if (this.currentDrag.x < b.origin.x) { - xOffset = -1; - } - if (this.currentDrag.y < b.origin.y) { - yOffset = -1; - } - } - if (this.currentDrag.x > b.origin.x + b.extent.x) { - xOffset = 1; + if (this.currentDrag.y < b.origin.y) { + yOffset = -1; } + if (this.currentDrag.y > b.origin.y + b.extent.y) { yOffset = 1; } @@ -19914,20 +20530,16 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc extend a selection or create one if there isnt yet * @param {Hypergrid} grid * @param {Object} gridCell - the event details * @param {Array} keys - array of the keys that are currently pressed down */ extendSelection: function(grid, gridCell, keys) { - var hasCTRL = keys.indexOf('CTRL') !== -1; + grid.stopEditing(); + //var hasCTRL = keys.indexOf('CTRL') !== -1; var hasSHIFT = keys.indexOf('SHIFT') !== -1; - // var scrollTop = grid.getVScrollValue(); - // var scrollLeft = grid.getHScrollValue(); - - // var numFixedColumns = 0;//grid.getFixedColumnCount(); - // var numFixedRows = 0;//grid.getFixedRowCount(); var mousePoint = grid.getMouseDown(); var x = gridCell.x; // - numFixedColumns + scrollLeft; @@ -19938,24 +20550,12 @@ var CellSelection = Feature.extend('CellSelection', { return; } - //we have repeated a click in the same spot deslect the value from last time - if (x === mousePoint.x && y === mousePoint.y) { - grid.clearMostRecentSelection(); - grid.popMouseDown(); - grid.repaint(); - return; - } - - if (!hasCTRL && !hasSHIFT) { - grid.clearSelections(); - } - if (hasSHIFT) { - grid.clearMostRecentSelection(); - grid.select(mousePoint.x, mousePoint.y, x - mousePoint.x + 1, y - mousePoint.y + 1); - grid.setDragExtent(grid.newPoint(x - mousePoint.x + 1, y - mousePoint.y)); + grid.clearMostRecentRowSelection(); + grid.selectRow(y, mousePoint.y); + grid.setDragExtent(grid.newPoint(0, y - mousePoint.y)); } else { - grid.select(x, y, 0, 0); + grid.toggleSelectRow(y, keys); grid.setMouseDown(grid.newPoint(x, y)); grid.setDragExtent(grid.newPoint(0, 0)); } @@ -19964,94 +20564,94 @@ var CellSelection = Feature.extend('CellSelection', { /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid */ handleDOWNSHIFT: function(grid) { - this.moveShiftSelect(grid, 0, 1); + this.moveShiftSelect(grid, 1); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ handleUPSHIFT: function(grid) { - this.moveShiftSelect(grid, 0, -1); + this.moveShiftSelect(grid, -1); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleLEFTSHIFT: function(grid) { - this.moveShiftSelect(grid, -1, 0); - }, + handleLEFTSHIFT: function(grid) {}, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleRIGHTSHIFT: function(grid) { - this.moveShiftSelect(grid, 1, 0); - }, + handleRIGHTSHIFT: function(grid) {}, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleDOWN: function(grid, event) { - //keep the browser viewport from auto scrolling on key event - event.primitiveEvent.preventDefault(); - - var count = this.getAutoScrollAcceleration(); - this.moveSingleSelect(grid, 0, count); + handleDOWN: function(grid) { + this.moveSingleSelect(grid, 1); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleUP: function(grid, event) { - //keep the browser viewport from auto scrolling on key event - event.primitiveEvent.preventDefault(); - - var count = this.getAutoScrollAcceleration(); - this.moveSingleSelect(grid, 0, -count); + handleUP: function(grid) { + this.moveSingleSelect(grid, -1); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ - handleLEFT: function(grid) { - this.moveSingleSelect(grid, -1, 0); - }, + handleLEFT: function(grid) {}, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc handle this event * @param {Hypergrid} grid * @param {Object} event - the event details */ handleRIGHT: function(grid) { - this.moveSingleSelect(grid, 1, 0); + + var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); + var maxColumns = grid.getColumnCount() - 1; + + var newX = grid.getHeaderColumnCount() + grid.getHScrollValue(); + var newY = mouseCorner.y; + + newX = Math.min(maxColumns, newX); + + grid.clearSelections(); + grid.select(newX, newY, 0, 0); + grid.setMouseDown(grid.newPoint(newX, newY)); + grid.setDragExtent(grid.newPoint(0, 0)); + + grid.repaint(); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc If we are holding down the same navigation key, accelerate the increment we scroll * #### returns: integer */ @@ -20063,7 +20663,7 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc set the start time to right now when we initiate an auto scroll */ setAutoScrollStartTime: function() { @@ -20071,7 +20671,7 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time */ pingAutoScroll: function() { @@ -20083,7 +20683,7 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc answer how long we have been auto scrolling * #### returns: integer */ @@ -20095,102 +20695,91 @@ var CellSelection = Feature.extend('CellSelection', { }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary. * @param {Hypergrid} grid * @param {number} offsetX - x coordinate to start at * @param {number} offsetY - y coordinate to start at */ - moveShiftSelect: function(grid, offsetX, offsetY) { + moveShiftSelect: function(grid, offsetY) { - var maxColumns = grid.getColumnCount() - 1; var maxRows = grid.getRowCount() - 1; - var maxViewableColumns = grid.getVisibleColumns() - 1; var maxViewableRows = grid.getVisibleRows() - 1; if (!grid.resolveProperty('scrollingEnabled')) { - maxColumns = Math.min(maxColumns, maxViewableColumns); maxRows = Math.min(maxRows, maxViewableRows); } var origin = grid.getMouseDown(); var extent = grid.getDragExtent(); - var newX = extent.x + offsetX; var newY = extent.y + offsetY; + //var newY = grid.getRowCount(); - newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX)); newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY)); - grid.clearMostRecentSelection(); - grid.select(origin.x, origin.y, newX, newY); + grid.clearMostRecentRowSelection(); + grid.selectRow(origin.y, origin.y + newY); - grid.setDragExtent(grid.newPoint(newX, newY)); + grid.setDragExtent(grid.newPoint(0, newY)); - if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) { - this.pingAutoScroll(); - } if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) { this.pingAutoScroll(); } + grid.fireSyntheticRowSelectionChangedEvent(); grid.repaint(); }, /** - * @memberOf CellSelection.prototype + * @memberOf RowSelection.prototype * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent. * @param {Hypergrid} grid * @param {number} offsetX - x coordinate to start at * @param {number} offsetY - y coordinate to start at */ - moveSingleSelect: function(grid, offsetX, offsetY) { + moveSingleSelect: function(grid, offsetY) { - var maxColumns = grid.getColumnCount() - 1; var maxRows = grid.getRowCount() - 1; - var maxViewableColumns = grid.getVisibleColumnsCount() - 1; var maxViewableRows = grid.getVisibleRowsCount() - 1; - var minRows = grid.getHeaderRowCount(); - var minCols = grid.getHeaderColumnCount(); - if (!grid.resolveProperty('scrollingEnabled')) { - maxColumns = Math.min(maxColumns, maxViewableColumns); maxRows = Math.min(maxRows, maxViewableRows); } var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); - var newX = mouseCorner.x + offsetX; var newY = mouseCorner.y + offsetY; + //var newY = grid.getRowCount(); - newX = Math.min(maxColumns, Math.max(minCols, newX)); - newY = Math.min(maxRows, Math.max(minRows, newY)); + newY = Math.min(maxRows, Math.max(0, newY)); grid.clearSelections(); - grid.select(newX, newY, 0, 0); - grid.setMouseDown(grid.newPoint(newX, newY)); + grid.selectRow(newY); + grid.setMouseDown(grid.newPoint(0, newY)); grid.setDragExtent(grid.newPoint(0, 0)); - if (grid.insureModelColIsVisible(newX, offsetX)) { - this.pingAutoScroll(); - } if (grid.insureModelRowIsVisible(newY, offsetY)) { this.pingAutoScroll(); } + grid.fireSyntheticRowSelectionChangedEvent(); grid.repaint(); + }, + + isSingleRowSelection: function() { + return true; } }); -module.exports = CellSelection; +module.exports = RowSelection; -},{"./Feature.js":59}],53:[function(require,module,exports){ +},{"./Feature.js":58}],64:[function(require,module,exports){ 'use strict'; var Feature = require('./Feature.js'); @@ -20198,2855 +20787,2964 @@ var Feature = require('./Feature.js'); /** * @constructor */ -var ColumnAutosizing = Feature.extend('ColumnAutosizing', { +var ThumbwheelScrolling = Feature.extend('ThumbwheelScrolling', { - alias: 'ColumnAutosizing', + alias: 'ThumbwheelScrolling', /** + * @memberOf ThumbwheelScrolling.prototype * @desc handle this event down the feature chain of responsibility * @param {Hypergrid} grid * @param {Object} event - the event details - * @memberOf ColumnAutosizing.prototype */ - handleDoubleClick: function(grid, event) { - var headerRowCount = grid.getHeaderRowCount(); - //var headerColCount = grid.getHeaderColumnCount(); - var gridCell = event.gridCell; - if (gridCell.y <= headerRowCount) { - grid.autosizeColumn(gridCell.x); - } else if (this.next) { - this.next.handleDoubleClick(grid, event); + handleWheelMoved: function(grid, e) { + if (!grid.resolveProperty('scrollingEnabled')) { + return; + } + var primEvent = e.primitiveEvent; + var deltaY = primEvent.wheelDeltaY || -primEvent.deltaY; + var deltaX = primEvent.wheelDeltaX || -primEvent.deltaX; + if (deltaY > 0) { + grid.scrollBy(0, -1); + } else if (deltaY < -0) { + grid.scrollBy(0, 1); + } else if (deltaX > 0) { + grid.scrollBy(-1, 0); + } else if (deltaX < -0) { + grid.scrollBy(1, 0); } } }); -module.exports = ColumnAutosizing; -},{"./Feature.js":59}],54:[function(require,module,exports){ -/* eslint-env browser */ -/* global requestAnimationFrame */ +module.exports = ThumbwheelScrolling; +},{"./Feature.js":58}],65:[function(require,module,exports){ 'use strict'; -// This feature is responsible for column drag and drop reordering. -// This object is a mess and desperately needs a complete rewrite..... - -var Feature = require('./Feature.js'); - -var columnAnimationTime = 150; -var dragger; -var draggerCTX; -var floatColumn; -var floatColumnCTX; +module.exports = { + Feature: require('./Feature'), // abstract base class + CellClick: require('./CellClick'), + CellEditing: require('./CellEditing'), + CellSelection: require('./CellSelection'), + ColumnAutosizing: require('./ColumnAutosizing'), + ColumnMoving: require('./ColumnMoving'), + ColumnResizing: require('./ColumnResizing'), + ColumnSelection: require('./ColumnSelection'), + ColumnSorting: require('./ColumnSorting'), + Filters: require('./Filters'), + KeyPaging: require('./KeyPaging'), + OnHover: require('./OnHover'), + ColumnPicker: require('./ColumnPicker'), + RowResizing: require('./RowResizing'), + RowSelection: require('./RowSelection'), + ThumbwheelScrolling: require('./ThumbwheelScrolling') +}; -/** - * @constructor - */ -var ColumnMoving = Feature.extend('ColumnMoving', { +},{"./CellClick":49,"./CellEditing":50,"./CellSelection":51,"./ColumnAutosizing":52,"./ColumnMoving":53,"./ColumnPicker":54,"./ColumnResizing":55,"./ColumnSelection":56,"./ColumnSorting":57,"./Feature":58,"./Filters":59,"./KeyPaging":60,"./OnHover":61,"./RowResizing":62,"./RowSelection":63,"./ThumbwheelScrolling":64}],66:[function(require,module,exports){ +'use strict'; - alias: 'ColumnMoving', +var deprecated = require('./deprecated'); +var Base = require('extend-me').Base; - /** - * queue up the animations that need to play so they are done synchronously - * @type {Array} - * @memberOf CellMoving.prototype - */ - floaterAnimationQueue: [], +Base.prototype.deprecated = deprecated; - /** - * am I currently auto scrolling right - * @type {boolean} - * @memberOf CellMoving.prototype - */ - columnDragAutoScrollingRight: false, +module.exports = Base; - /** - * am I currently auto scrolling left - * @type {boolean} - * @memberOf CellMoving.prototype - */ - columnDragAutoScrollingLeft: false, +},{"./deprecated":74,"extend-me":5}],67:[function(require,module,exports){ +'use strict'; - /** - * is the drag mechanism currently enabled ("armed") - * @type {boolean} - * @memberOf CellMoving.prototype - */ - dragArmed: false, +var Base = require('./Base'); - /** - * am I dragging right now - * @type {boolean} - * @memberOf CellMoving.prototype - */ - dragging: false, +/** @constructor + * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid. + * + * See {@link CellProvider#initialize|initialize} which is called by the constructor. + */ +var CellProvider = Base.extend('CellProvider', { /** - * the column index of the currently dragged column - * @type {number} - * @memberOf CellMoving.prototype + * @summary Constructor logic + * @desc This method will be called upon instantiation of this class or of any class that extends from this class. + * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most "senior" class through that of the class of the new instance. + * @memberOf CellProvider.prototype */ - dragCol: -1, + initialize: function() { + this.cellCache = {}; + this.initializeCells(); + }, /** - * an offset to position the dragged item from the cursor - * @type {number} - * @memberOf CellMoving.prototype + * @desc replace this function in on your instance of cellProvider + * @returns cell + * @param {object} config - an object with everything you might need for renderering a cell + * @memberOf CellProvider.prototype */ - dragOffset: 0, + getCell: function(config) { + var cell = this.cellCache.simpleCellRenderer; + cell.config = config; + return cell; + }, /** - * @memberOf CellMoving.prototype - * @desc give me an opportunity to initialize stuff on the grid - * @param {Hypergrid} grid + * @desc replace this function in on your instance of cellProvider + * @returns cell + * @param {object} config - an object with everything you might need for renderering a cell + * @memberOf CellProvider.prototype */ - initializeOn: function(grid) { - this.isFloatingNow = false; - this.initializeAnimationSupport(grid); - if (this.next) { - this.next.initializeOn(grid); - } + getColumnHeaderCell: function(config) { + var cell = this.cellCache.simpleCellRenderer; + cell.config = config; + return cell; }, /** - * @memberOf CellMoving.prototype - * @desc initialize animation support on the grid - * @param {Hypergrid} grid + * @desc replace this function in on your instance of cellProvider + * @returns cell + * @param {object} config - an object with everything you might need for renderering a cell + * @memberOf CellProvider.prototype */ - initializeAnimationSupport: function(grid) { - if (!dragger) { - dragger = document.createElement('canvas'); - dragger.setAttribute('width', '0px'); - dragger.setAttribute('height', '0px'); + getRowHeaderCell: function(config) { + var cell = this.cellCache.simpleCellRenderer; + cell.config = config; + return cell; + }, - document.body.appendChild(dragger); - draggerCTX = dragger.getContext('2d'); + paintButton: function(gc, config) { + var val = config.value; + var c = config.x; + var r = config.y; + var bounds = config.bounds; + var x = bounds.x + 2; + var y = bounds.y + 2; + var width = bounds.width - 3; + var height = bounds.height - 3; + var radius = height / 2; + var arcGradient = gc.createLinearGradient(x, y, x, y + height); + if (config.mouseDown) { + arcGradient.addColorStop(0, '#B5CBED'); + arcGradient.addColorStop(1, '#4d74ea'); + } else { + arcGradient.addColorStop(0, '#ffffff'); + arcGradient.addColorStop(1, '#aaaaaa'); } - if (!floatColumn) { - floatColumn = document.createElement('canvas'); - floatColumn.setAttribute('width', '0px'); - floatColumn.setAttribute('height', '0px'); + gc.fillStyle = arcGradient; + gc.strokeStyle = '#000000'; + roundRect(gc, x, y, width, height, radius, arcGradient, true); - document.body.appendChild(floatColumn); - floatColumnCTX = floatColumn.getContext('2d'); + var ox = (width - config.getTextWidth(gc, val)) / 2; + var oy = (height - config.getTextHeight(gc.font).descent) / 2; + + if (gc.textBaseline !== 'middle') { + gc.textBaseline = 'middle'; } - }, + gc.fillStyle = '#000000'; - getCanDragCursorName: function() { - return '-webkit-grab'; - }, + config.backgroundColor = 'rgba(0,0,0,0)'; + gc.fillText(val, x + ox, y + oy); - getDraggingCursorName: function() { - return '-webkit-grabbing'; + //identify that we are a button + config.buttonCells[c + ',' + r] = true; }, /** - * @memberOf CellMoving.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @summary The default cell rendering function for rendering a vanilla cell. + * @desc Great care has been taken in crafting this function as it needs to perform extremely fast. Reads on the gc object are expensive but not quite as expensive as writes to it. We do our best to avoid writes, then avoid reads. Clipping bounds are not set here as this is also an expensive operation. Instead, we truncate overflowing text and content by filling a rectangle with background color column by column instead of cell by cell. This column by column fill happens higher up on the stack in a calling function from fin-hypergrid-renderer. Take note we do not do cell by cell border renderering as that is expensive. Instead we render many fewer gridlines after all cells are rendered. + * @param {CanvasGraphicsContext} gc + * @param {number} config.bounds.x - the x screen coordinate of my origin + * @param {number} config.bounds.y - the y screen coordinate of my origin + * @param {number} config.bounds.width - the width I'm allowed to draw within + * @param {number} config.bounds.height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype */ - handleMouseDrag: function(grid, event) { + defaultCellPaint: function(gc, config) { + var val = config.value, + x = config.bounds.x, + y = config.bounds.y, + width = config.bounds.width, + height = config.bounds.height, + wrapHeaders = config.headerTextWrapping, + leftPadding = 2, //TODO: fix this + isHeader = config.y === 0; - var gridCell = event.gridCell; - var x; - //var y; + var leftIcon, rightIcon, centerIcon, ixoffset, iyoffset; - var distance = Math.abs(event.primitiveEvent.detail.dragstart.x - event.primitiveEvent.detail.mouse.x); + // setting gc properties are expensive, let's not do it needlessly - if (distance < 10) { - if (this.next) { - this.next.handleMouseDrag(grid, event); + if (val && val.constructor === Array) { + leftIcon = val[0]; + rightIcon = val[2]; + val = val[1]; + if (val && typeof val === 'object') { + if (val.constructor.name === 'HTMLImageElement') { // must be an image + centerIcon = val; + val = null; + } + } + if (leftIcon && leftIcon.nodeName !== 'IMG') { + leftIcon = null; + } + if (rightIcon && rightIcon.nodeName !== 'IMG') { + rightIcon = null; + } + if (centerIcon && centerIcon.nodeName !== 'IMG') { + centerIcon = null; + } + } + + val = valueOrFunctionExecute(config, val); + + val = config.formatter(val); + + if (gc.font !== config.font) { + gc.font = config.font; + } + if (gc.textAlign !== 'left') { + gc.textAlign = 'left'; + } + if (gc.textBaseline !== 'middle') { + gc.textBaseline = 'middle'; + } + + // fill background only if our bgColor is populated or we are a selected cell + var backgroundColor, hover, hoverColor, selectColor; + if (config.isCellHovered && config.hoverCellHighlight.enabled) { + hoverColor = config.hoverCellHighlight.backgroundColor; + } else if (config.isRowHovered && (hover = config.hoverRowHighlight).enabled) { + hoverColor = config.isGridColumn || !hover.header || hover.header.backgroundColor === undefined ? hover.backgroundColor : hover.header.backgroundColor; + } else if (config.isColumnHovered && (hover = config.hoverColumnHighlight).enabled) { + hoverColor = config.isGridRow || !hover.header || hover.header.backgroundColor === undefined ? hover.backgroundColor : hover.header.backgroundColor; + } + if (alpha(hoverColor) < 1) { + if (config.isSelected) { + selectColor = valueOrFunctionExecute(config, config.backgroundSelectionColor); + } + if (alpha(selectColor) < 1) { + backgroundColor = valueOrFunctionExecute(config, config.backgroundColor); + if (alpha(backgroundColor) > 0) { + gc.fillStyle = backgroundColor; + gc.fillRect(x, y, width, height); + } } - return; + if (selectColor !== undefined) { + gc.fillStyle = selectColor; + gc.fillRect(x, y, width, height); + } + } + if (hoverColor !== undefined) { + gc.fillStyle = hoverColor; + gc.fillRect(x, y, width, height); } - if (this.isHeaderRow(grid, event) && this.dragArmed && !this.dragging) { - this.dragging = true; - this.dragCol = gridCell.x; - this.dragOffset = event.mousePoint.x; - this.detachChain(); - x = event.primitiveEvent.detail.mouse.x - this.dragOffset; - //y = event.primitiveEvent.detail.mouse.y; - this.createDragColumn(grid, x, this.dragCol); - } else if (this.next) { - this.next.handleMouseDrag(grid, event); + // draw text + var theColor = valueOrFunctionExecute(config, config.isSelected ? config.foregroundSelectionColor : config.color); + if (gc.fillStyle !== theColor) { + gc.fillStyle = theColor; + gc.strokeStyle = theColor; } - if (this.dragging) { - x = event.primitiveEvent.detail.mouse.x - this.dragOffset; - //y = event.primitiveEvent.detail.mouse.y; - this.dragColumn(grid, x); + if (isHeader && wrapHeaders) { + this.renderMultiLineText(gc, x, y, height, width, config, val); + } else { + this.renderSingleLineText(gc, x, y, height, width, config, val); } - }, - /** - * @memberOf CellMoving.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDown: function(grid, event) { - if (grid.getBehavior().isColumnReorderable()) { - if (this.isHeaderRow(grid, event) && event.gridCell.x !== -1) { - this.dragArmed = true; - this.cursor = this.getDraggingCursorName(); - grid.clearSelections(); + var iconWidth = 0; + if (leftIcon) { + iyoffset = Math.round((height - leftIcon.height) / 2); + gc.drawImage(leftIcon, x + leftPadding, y + iyoffset); + iconWidth = Math.max(leftIcon.width + 2); + } + if (rightIcon && width > 1.75 * height) { + iyoffset = Math.round((height - rightIcon.height) / 2); + var rightX = x + width - rightIcon.width; + if (backgroundColor !== undefined) { + gc.fillStyle = backgroundColor; + gc.fillRect(rightX, y, rightIcon.width, height); + } else { + gc.clearRect(rightX, y, rightIcon.width, height); } + gc.drawImage(rightIcon, rightX, y + iyoffset); + iconWidth = Math.max(rightIcon.width + 2); } - if (this.next) { - this.next.handleMouseDown(grid, event); + if (centerIcon) { + iyoffset = Math.round((height - centerIcon.height) / 2); + ixoffset = Math.round((width - centerIcon.width) / 2); + gc.drawImage(centerIcon, x + width - ixoffset - centerIcon.width, y + iyoffset); + iconWidth = Math.max(centerIcon.width + 2); } - }, + if (config.cellBorderThickness) { + gc.beginPath(); + gc.rect(x, y, width, height); + gc.lineWidth = config.cellBorderThickness; + gc.strokeStyle = config.cellBorderStyle; - /** - * @memberOf CellMoving.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseUp: function(grid, event) { - //var col = event.gridCell.x; - if (this.dragging) { - this.cursor = null; - //delay here to give other events a chance to be dropped - var self = this; - this.endDragColumn(grid); - setTimeout(function() { - self.attachChain(); - }, 200); - } - this.dragCol = -1; - this.dragging = false; - this.dragArmed = false; - this.cursor = null; - grid.repaint(); + // animate the dashed line a bit here for fun - if (this.next) { - this.next.handleMouseUp(grid, event); + gc.stroke(); + gc.closePath(); } - + config.minWidth = config.minWidth + 2 * (iconWidth); }, - /** - * @memberOf CellMoving.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseMove: function(grid, event) { - - if (!this.dragging && event.mousePoint.y < 5 && event.viewPoint.y === 0) { - this.cursor = this.getCanDragCursorName(); - } else { - this.cursor = null; + renderMultiLineText: function(gc, x, y, height, width, config, val) { + var lines = fitText(gc, config, val, width); + if (lines.length === 1) { + return this.renderSingleLineText(gc, x, y, height, width, config, squeeze(val)); } - if (this.next) { - this.next.handleMouseMove(grid, event); - } + var colHEdgeOffset = config.cellPadding, + halignOffset = 0, + valignOffset = config.voffset, + halign = config.halign, + textHeight = config.getTextHeight(config.font).height; - if (this.isHeaderRow(grid, event) && this.dragging) { - this.cursor = this.getDraggingCursorName(); //move'; + switch (halign) { + case 'right': + halignOffset = width - colHEdgeOffset; + break; + case 'center': + halignOffset = width / 2; + break; + case 'left': + halignOffset = colHEdgeOffset; + break; } - }, - /** - * @memberOf CellMoving.prototype - * @desc this is the main event handler that manages the dragging of the column - * @param {Hypergrid} grid - * @param {boolean} draggedToTheRight - are we moving to the right - */ - floatColumnTo: function(grid, draggedToTheRight) { - this.floatingNow = true; + var hMin = 0, vMin = Math.ceil(textHeight / 2); - var renderer = grid.getRenderer(); - var colEdges = renderer.getColumnEdges(); - //var behavior = grid.getBehavior(); - var scrollLeft = grid.getHScrollValue(); - var floaterIndex = grid.renderOverridesCache.floater.columnIndex; - var draggerIndex = grid.renderOverridesCache.dragger.columnIndex; - var hdpiratio = grid.renderOverridesCache.dragger.hdpiratio; + valignOffset += Math.ceil((height - (lines.length - 1) * textHeight) / 2); - var draggerStartX; - var floaterStartX; - var fixedColumnCount = grid.getFixedColumnCount(); - var draggerWidth = grid.getColumnWidth(draggerIndex); - var floaterWidth = grid.getColumnWidth(floaterIndex); + halignOffset = Math.max(hMin, halignOffset); + valignOffset = Math.max(vMin, valignOffset); - var max = grid.getVisibleColumnsCount(); + gc.save(); // define a clipping region for cell + gc.rect(x, y, width, height); + gc.clip(); - var doffset = 0; - var foffset = 0; + gc.textAlign = halign; - if (draggerIndex >= fixedColumnCount) { - doffset = scrollLeft; - } - if (floaterIndex >= fixedColumnCount) { - foffset = scrollLeft; + for (var i = 0; i < lines.length; i++) { + gc.fillText(lines[i], x + halignOffset, y + valignOffset + (i * textHeight)); } - if (draggedToTheRight) { - draggerStartX = colEdges[Math.min(max, draggerIndex - doffset)]; - floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)]; + gc.restore(); // discard clipping region + }, - grid.renderOverridesCache.dragger.startX = (draggerStartX + floaterWidth) * hdpiratio; - grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio; + renderSingleLineText: function(gc, x, y, height, width, config, val) { + var colHEdgeOffset = config.cellPadding, + halignOffset = 0, + valignOffset = config.voffset, + halign = config.halign, + isCellHovered = config.isCellHovered, + isLink = config.link; - } else { - floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)]; - draggerStartX = floaterStartX + draggerWidth; + var fontMetrics = config.getTextHeight(config.font); + var textWidth = config.getTextWidth(gc, val); - grid.renderOverridesCache.dragger.startX = floaterStartX * hdpiratio; - grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio; + //we must set this in order to compute the minimum width + //for column autosizing purposes + config.minWidth = textWidth + (2 * colHEdgeOffset); + + switch (halign) { + case 'right': + //textWidth = config.getTextWidth(gc, config.value); + halignOffset = width - colHEdgeOffset - textWidth; + break; + case 'center': + //textWidth = config.getTextWidth(gc, config.value); + halignOffset = (width - textWidth) / 2; + break; + case 'left': + halignOffset = colHEdgeOffset; + break; } - grid.swapColumns(draggerIndex, floaterIndex); - grid.renderOverridesCache.dragger.columnIndex = floaterIndex; - grid.renderOverridesCache.floater.columnIndex = draggerIndex; + halignOffset = Math.max(0, halignOffset); + valignOffset = valignOffset + Math.ceil(height / 2); - this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(grid, floaterStartX, draggerStartX)); + if (val !== null) { + gc.fillText(val, x + halignOffset, y + valignOffset); + } - this.doFloaterAnimation(grid); + if (isCellHovered) { + gc.beginPath(); + if (isLink) { + underline(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1); + gc.stroke(); + } + gc.closePath(); + } + if (config.strikeThrough === true) { + gc.beginPath(); + strikeThrough(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1); + gc.stroke(); + gc.closePath(); + } + }, + /** + * @param {CanvasGraphicsContext} gc + * @param {number} x - the x screen coordinate of my origin + * @param {number} y - the y screen coordinate of my origin + * @param {number} width - the width I'm allowed to draw within + * @param {number} height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype + * @desc Emerson's paint function for a slider button. currently the user cannot interact with it + */ + paintSlider: function(gc, x, y, width, height) { + // gc.strokeStyle = 'white'; + // var val = this.config.value; + // var radius = height / 2; + // var offset = width * val; + // var bgColor = this.config.isSelected ? this.config.bgSelColor : '#333333'; + // var btnGradient = gc.createLinearGradient(x, y, x, y + height); + // btnGradient.addColorStop(0, bgColor); + // btnGradient.addColorStop(1, '#666666'); + // var arcGradient = gc.createLinearGradient(x, y, x, y + height); + // arcGradient.addColorStop(0, '#aaaaaa'); + // arcGradient.addColorStop(1, '#777777'); + // gc.fillStyle = btnGradient; + // roundRect(gc, x, y, width, height, radius, btnGradient); + // if (val < 1.0) { + // gc.fillStyle = arcGradient; + // } else { + // gc.fillStyle = '#eeeeee'; + // } + // gc.beginPath(); + // gc.arc(x + Math.max(offset - radius, radius), y + radius, radius, 0, 2 * Math.PI); + // gc.fill(); + // gc.closePath(); + // this.config.minWidth = 100; }, /** - * @memberOf CellMoving.prototype - * @desc manifest the column drag and drop animation - * @param {Hypergrid} grid - * @param {number} floaterStartX - the x start coordinate of the column underneath that floats behind the dragged column - * @param {number} draggerStartX - the x start coordinate of the dragged column + * @desc A simple implementation of a sparkline, because it's a barchart we've changed the name ;). + * @param {CanvasGraphicsContext} gc + * @param {number} x - the x screen coordinate of my origin + * @param {number} y - the y screen coordinate of my origin + * @param {number} width - the width I'm allowed to draw within + * @param {number} height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype */ - doColumnMoveAnimation: function(grid, floaterStartX, draggerStartX) { - var self = this; - return function() { - var d = floatColumn; - d.style.display = 'inline'; - self.setCrossBrowserProperty(d, 'transform', 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'); - - //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'; - //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)'; - - requestAnimationFrame(function() { - self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease'); - self.setCrossBrowserProperty(d, 'transform', 'translate(' + draggerStartX + 'px, ' + -2 + 'px)'); - }); - grid.repaint(); - //need to change this to key frames + paintSparkbar: function(gc, x, y, width, height) { + gc.beginPath(); + var val = this.config.value; + if (!val || !val.length) { + return; + } + var count = val.length; + var eWidth = width / count; + var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor; + if (this.config.bgColor || this.config.isSelected) { + gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor; + gc.fillRect(x, y, width, height); + } + gc.fillStyle = fgColor; + for (var i = 0; i < val.length; i++) { + var barheight = val[i] / 110 * height; + gc.fillRect(x + 5, y + height - barheight, eWidth * 0.6666, barheight); + x = x + eWidth; + } + gc.closePath(); + this.config.minWidth = count * 10; - setTimeout(function() { - self.setCrossBrowserProperty(d, 'transition', ''); - grid.renderOverridesCache.floater = null; - grid.repaint(); - self.doFloaterAnimation(grid); - requestAnimationFrame(function() { - d.style.display = 'none'; - self.isFloatingNow = false; - }); - }, columnAnimationTime + 50); - }; }, /** - * @memberOf CellMoving.prototype - * @desc manifest the floater animation - * @param {Hypergrid} grid + * @desc A simple implementation of a sparkline. see [Edward Tufte sparkline](http://www.edwardtufte.com/bboard/q-and-a-fetch-msg?msg_id=0001OR) + * @param {CanvasGraphicsContext} gc + * @param {number} x - the x screen coordinate of my origin + * @param {number} y - the y screen coordinate of my origin + * @param {number} width - the width I'm allowed to draw within + * @param {number} height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype */ - doFloaterAnimation: function(grid) { - if (this.floaterAnimationQueue.length === 0) { - this.floatingNow = false; - grid.repaint(); + paintSparkline: function(gc, x, y, width, height) { + gc.beginPath(); + var val = this.config.value; + if (!val || !val.length) { return; } - var animation = this.floaterAnimationQueue.pop(); - animation(); + var count = val.length; + var eWidth = width / count; + + var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor; + if (this.config.bgColor || this.config.isSelected) { + gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor; + gc.fillRect(x, y, width, height); + } + gc.strokeStyle = fgColor; + gc.fillStyle = fgColor; + gc.beginPath(); + var prev; + for (var i = 0; i < val.length; i++) { + var barheight = val[i] / 110 * height; + if (!prev) { + prev = barheight; + } + gc.lineTo(x + 5, y + height - barheight); + gc.arc(x + 5, y + height - barheight, 1, 0, 2 * Math.PI, false); + x = x + eWidth; + } + this.config.minWidth = count * 10; + gc.stroke(); + gc.closePath(); }, /** - * @memberOf CellMoving.prototype - * @desc create the float column at columnIndex underneath the dragged column - * @param {Hypergrid} grid - * @param {number} columnIndex - the index of the column that will be floating + * @desc A simple implementation of a tree cell renderer for use mainly with the qtree. + * @param {CanvasGraphicsContext} gc + * @param {number} x - the x screen coordinate of my origin + * @param {number} y - the y screen coordinate of my origin + * @param {number} width - the width I'm allowed to draw within + * @param {number} height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype */ - createFloatColumn: function(grid, columnIndex) { - - var fixedColumnCount = grid.getFixedColumnCount(); - var scrollLeft = grid.getHScrollValue(); + treeCellRenderer: function(gc, x, y, width, height) { + var val = this.config.value.data; + var indent = this.config.value.indent; + var icon = this.config.value.icon; - if (columnIndex < fixedColumnCount) { - scrollLeft = 0; + //fill background only if our bgColor is populated or we are a selected cell + if (this.config.bgColor || this.config.isSelected) { + gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor; + gc.fillRect(x, y, width, height); } - var renderer = grid.getRenderer(); - var columnEdges = renderer.getColumnEdges(); - - var columnWidth = grid.getColumnWidth(columnIndex); - var colHeight = grid.div.clientHeight; - var d = floatColumn; - var style = d.style; - var location = grid.div.getBoundingClientRect(); - - style.top = (location.top - 2) + 'px'; - style.left = location.left + 'px'; - style.position = 'fixed'; - - var hdpiRatio = grid.getHiDPI(floatColumnCTX); - - d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px'); - d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px'); - style.boxShadow = '0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)'; - style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px'; - style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px'; - style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor'); - style.backgroundColor = renderer.resolveProperty('backgroundColor'); - - var startX = columnEdges[columnIndex - scrollLeft]; - startX = startX * hdpiRatio; - - floatColumnCTX.scale(hdpiRatio, hdpiRatio); + if (!val || !val.length) { + return; + } + var valignOffset = Math.ceil(height / 2); - grid.renderOverridesCache.floater = { - columnIndex: columnIndex, - ctx: floatColumnCTX, - startX: startX, - width: columnWidth, - height: colHeight, - hdpiratio: hdpiRatio - }; + gc.fillStyle = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor; + gc.fillText(icon + val, x + indent, y + valignOffset); - style.zIndex = '4'; - this.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -2 + 'px)'); - style.cursor = this.getDraggingCursorName(); - grid.repaint(); + var textWidth = this.config.getTextWidth(gc, icon + val); + var minWidth = x + indent + textWidth + 10; + this.config.minWidth = minWidth; }, /** - * @memberOf CellMoving.prototype - * @desc utility function for setting cross browser css properties - * @param {HTMLElement} element - descripton - * @param {string} property - the property - * @param {string} value - the value to assign + * @desc An empty implementation of a cell renderer, see [the null object pattern](http://c2.com/cgi/wiki?NullObject). + * @param {CanvasGraphicsContext} gc + * @param {number} x - the x screen coordinate of my origin + * @param {number} y - the y screen coordinate of my origin + * @param {number} width - the width I'm allowed to draw within + * @param {number} height - the height I'm allowed to draw within + * @memberOf CellProvider.prototype */ - setCrossBrowserProperty: function(element, property, value) { - var uProperty = property[0].toUpperCase() + property.substr(1); - this.setProp(element, 'webkit' + uProperty, value); - this.setProp(element, 'Moz' + uProperty, value); - this.setProp(element, 'ms' + uProperty, value); - this.setProp(element, 'O' + uProperty, value); - this.setProp(element, property, value); - }, + emptyCellRenderer: function(gc, x, y, width, height) {}, /** - * @memberOf CellMoving.prototype - * @desc utility function for setting properties on HTMLElements - * @param {HTMLElement} element - descripton - * @param {string} property - the property - * @param {string} value - the value to assign + * @memberOf CellProvider.prototype + * @private */ - setProp: function(element, property, value) { - if (property in element.style) { - element.style[property] = value; - } - }, + initializeCells: function() { + var self = this; + this.cellCache.simpleCellRenderer = { + paint: this.defaultCellPaint, + renderSingleLineText: this.renderSingleLineText, + renderMultiLineText: this.renderMultiLineText + }; + this.cellCache.sliderCellRenderer = { + paint: this.paintSlider + }; + this.cellCache.sparkbarCellRenderer = { + paint: this.paintSparkbar + }; + this.cellCache.sparklineCellRenderer = { + paint: this.paintSparkline + }; + this.cellCache.treeCellRenderer = { + paint: this.treeCellRenderer + }; + this.cellCache.emptyCellRenderer = { + paint: this.emptyCellRenderer + }; + this.cellCache.buttonRenderer = { + paint: this.paintButton, + //defaultCellPaint: this.defaultCellPaint + }; + this.cellCache.linkCellRenderer = { + paint: function(gc, x, y, width, height) { + self.config = this.config; + self.defaultCellPaint(gc, x, y, width, height, true); + } + }; + } +}); - /** - * @memberOf CellMoving.prototype - * @desc create the dragged column at columnIndex above the floated column - * @param {Hypergrid} grid - * @param {number} x - the start position - * @param {number} columnIndex - the index of the column that will be floating - */ - createDragColumn: function(grid, x, columnIndex) { +function valueOrFunctionExecute(config, valueOrFunction) { + var isFunction = (((typeof valueOrFunction)[0]) === 'f'); + var result = isFunction ? valueOrFunction(config) : valueOrFunction; + if (!result && result !== 0) { + return ''; + } + return result; +} - var fixedColumnCount = grid.getFixedColumnCount(); - var scrollLeft = grid.getHScrollValue(); +function underline(config, gc, text, x, y, thickness) { + var width = config.getTextWidth(gc, text); - if (columnIndex < fixedColumnCount) { - scrollLeft = 0; - } + switch (gc.textAlign) { + case 'center': + x -= (width / 2); + break; + case 'right': + x -= width; + break; + } - var renderer = grid.getRenderer(); - var columnEdges = renderer.getColumnEdges(); - var hdpiRatio = grid.getHiDPI(draggerCTX); - var columnWidth = grid.getColumnWidth(columnIndex); - var colHeight = grid.div.clientHeight; - var d = dragger; - var location = grid.div.getBoundingClientRect(); - var style = d.style; + //gc.beginPath(); + gc.lineWidth = thickness; + gc.moveTo(x + 0.5, y + 0.5); + gc.lineTo(x + width + 0.5, y + 0.5); +} - style.top = location.top + 'px'; - style.left = location.left + 'px'; - style.position = 'fixed'; - style.opacity = 0.85; - style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)'; - //style.zIndex = 100; - style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor'); - style.backgroundColor = grid.renderer.resolveProperty('backgroundColor'); +function strikeThrough(config, gc, text, x, y, thickness) { + var fontMetrics = config.getTextHeight(config.font); + var width = config.getTextWidth(gc, text); + y = y - (fontMetrics.height * 0.4); - d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px'); - d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px'); + switch (gc.textAlign) { + case 'center': + x -= (width / 2); + break; + case 'right': + x -= width; + break; + } - style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px'; - style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px'; + //gc.beginPath(); + gc.lineWidth = thickness; + gc.moveTo(x + 0.5, y + 0.5); + gc.lineTo(x + width + 0.5, y + 0.5); +} - var startX = columnEdges[columnIndex - scrollLeft]; - startX = startX * hdpiRatio; +function findLines(gc, config, words, width) { - draggerCTX.scale(hdpiRatio, hdpiRatio); + if (words.length === 1) { + return words; + } - grid.renderOverridesCache.dragger = { - columnIndex: columnIndex, - ctx: draggerCTX, - startX: startX, - width: columnWidth, - height: colHeight, - hdpiratio: hdpiRatio - }; + // starting with just the first word… + var stillFits, line = [words.shift()]; + while ( + // so lone as line still fits within current column… + (stillFits = config.getTextWidth(gc, line.join(' ')) < width) + // …AND there are more words available… + && words.length + ) { + // …add another word to end of line and retest + line.push(words.shift()); + } - this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, -5px)'); - style.zIndex = '5'; - style.cursor = this.getDraggingCursorName(); - grid.repaint(); - }, + if ( + !stillFits // if line is now too long… + && line.length > 1 // …AND is multiple words… + ) { + words.unshift(line.pop()); // …back off by (i.e., remove) one word + } - /** - * @memberOf CellMoving.prototype - * @desc this function is the main dragging logic - * @param {Hypergrid} grid - * @param {number} x - the start position - */ - dragColumn: function(grid, x) { + line = [line.join(' ')]; - //TODO: this function is overly complex, refactor this in to something more reasonable - var self = this; - //var renderer = grid.getRenderer(); - //var columnEdges = renderer.getColumnEdges(); + if (words.length) { // if there's anything left… + line = line.concat(findLines(gc, config, words, width)); // …break it up as well + } - var autoScrollingNow = this.columnDragAutoScrollingRight || this.columnDragAutoScrollingLeft; + return line; +} - var hdpiRatio = grid.getHiDPI(draggerCTX); +function fitText(gc, config, string, width) { + return findLines(gc, config, squeeze(string).split(' '), width); +} - var dragColumnIndex = grid.renderOverridesCache.dragger.columnIndex; - var columnWidth = grid.renderOverridesCache.dragger.width; +// trim string; then reduce all runs of multiple spaces to a single space +function squeeze(string) { + return string.toString().trim().replace(/\s\s+/g, ' '); +} - var minX = 0; //grid.getFixedColumnsWidth(); - var maxX = grid.renderer.getFinalVisableColumnBoundary() - columnWidth; - x = Math.min(x, maxX + 15); - x = Math.max(minX - 15, x); +function roundRect(gc, x, y, width, height, radius, fill, stroke) { - //am I at my lower bound - var atMin = x < minX && dragColumnIndex !== 0; + if (!stroke) { + stroke = true; + } + if (!radius) { + radius = 5; + } + gc.beginPath(); + gc.moveTo(x + radius, y); + gc.lineTo(x + width - radius, y); + gc.quadraticCurveTo(x + width, y, x + width, y + radius); + gc.lineTo(x + width, y + height - radius); + gc.quadraticCurveTo(x + width, y + height, x + width - radius, y + height); + gc.lineTo(x + radius, y + height); + gc.quadraticCurveTo(x, y + height, x, y + height - radius); + gc.lineTo(x, y + radius); + gc.quadraticCurveTo(x, y, x + radius, y); + gc.closePath(); + if (stroke) { + gc.stroke(); + } + if (fill) { + gc.fill(); + } + gc.closePath(); +} - //am I at my upper bound - var atMax = x > maxX; +function alpha(cssColorSpec) { + if (cssColorSpec === undefined) { + // undefined so not visible; treat as transparent + return 0; + } - var d = dragger; + var matches = cssColorSpec.match(alpha.regex); - this.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + 0 + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease'); + if (matches === null) { + // an opaque color (a color spec with no alpha channel) + return 1; + } - this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, ' + -10 + 'px)'); - requestAnimationFrame(function() { - d.style.display = 'inline'; - }); + var A = matches[4]; - var overCol = grid.renderer.getColumnFromPixelX(x + (d.width / 2 / hdpiRatio)); + if (A === undefined) { + // cssColorSpec must have been 'transparent' + return 0; + } - if (atMin) { - overCol = 0; - } + return Number(A); +} - if (atMax) { - overCol = grid.getColumnCount() - 1; - } +alpha.regex = /^(transparent|((RGB|HSL)A\(.*,\s*([\d\.]+)\)))$/i; - var doAFloat = dragColumnIndex > overCol; - doAFloat = doAFloat || (overCol - dragColumnIndex >= 1); +module.exports = CellProvider; - if (doAFloat && !atMax && !autoScrollingNow) { - var draggedToTheRight = dragColumnIndex < overCol; - // if (draggedToTheRight) { - // overCol = overCol - 1; - // } - if (this.isFloatingNow) { - return; - } +},{"./Base":66}],68:[function(require,module,exports){ +'use strict'; - this.isFloatingNow = true; - this.createFloatColumn(grid, overCol); - this.floatColumnTo(grid, draggedToTheRight); - } else { +var FilterTree = require('filter-tree'); - if (x < minX - 10) { - this.checkAutoScrollToLeft(grid, x); - } - if (x > minX - 10) { - this.columnDragAutoScrollingLeft = false; - } - //lets check for autoscroll to right if were up against it - if (atMax || x > maxX + 10) { - this.checkAutoScrollToRight(grid, x); - return; - } - if (x < maxX + 10) { - this.columnDragAutoScrollingRight = false; - } - } - }, +/** @typedef {function} fieldsProviderFunc + * @returns {fieldOption[]} see jsdoc typedef in filter-tree/js/FillerLeaf.js + */ - /** - * @memberOf CellMoving.prototype - * @desc autoscroll to the right if necessary - * @param {Hypergrid} grid - * @param {number} x - the start position - */ - checkAutoScrollToRight: function(grid, x) { - if (this.columnDragAutoScrollingRight) { - return; - } - this.columnDragAutoScrollingRight = true; - this._checkAutoScrollToRight(grid, x); - }, +var validateQuietlyOptions = { + alert: false, + focus: false +}; - _checkAutoScrollToRight: function(grid, x) { - if (!this.columnDragAutoScrollingRight) { - return; - } - var scrollLeft = grid.getHScrollValue(); - if (!grid.dragging || scrollLeft > (grid.sbHScrollConfig.rangeStop - 2)) { - return; - } - var draggedIndex = grid.renderOverridesCache.dragger.columnIndex; - grid.scrollBy(1, 0); - var newIndex = draggedIndex + 1; - console.log(newIndex, draggedIndex); - grid.swapColumns(newIndex, draggedIndex); - grid.renderOverridesCache.dragger.columnIndex = newIndex; +/** @constructor + */ +function CustomFilter() { + this.getDisplayString = function() { // TODO: What the heck is this? (not in use) + return '< ' + this.value; + }; - setTimeout(this._checkAutoScrollToRight.bind(this, grid, x), 250); - }, + // Following methods service dialog during which `this.filterTree` is valid + this.initialize = function(fieldsProvider) { + /** @type {fieldsProviderFunc} + */ + this.fieldsProvider = fieldsProvider; + this.filterTree = new FilterTree({ + fields: fieldsProvider() + }); + delete this.filter; // forces this.create to recreate the filter function + }; - /** - * @memberOf CellMoving.prototype - * @desc autoscroll to the left if necessary - * @param {Hypergrid} grid - * @param {number} x - the start position - */ - checkAutoScrollToLeft: function(grid, x) { - if (this.columnDragAutoScrollingLeft) { - return; - } - this.columnDragAutoScrollingLeft = true; - this._checkAutoScrollToLeft(grid, x); - }, + this.onShow = function(container) { + container.appendChild(this.filterTree.el); + }; - _checkAutoScrollToLeft: function(grid, x) { - if (!this.columnDragAutoScrollingLeft) { - return; - } + this.onOk = function() { + return this.filterTree.validate(); + }; - var scrollLeft = grid.getHScrollValue(); - if (!grid.dragging || scrollLeft < 1) { - return; - } - var draggedIndex = grid.renderOverridesCache.dragger.columnIndex; - grid.swapColumns(draggedIndex + scrollLeft, draggedIndex + scrollLeft - 1); - grid.scrollBy(-1, 0); - setTimeout(this._checkAutoScrollToLeft.bind(this, grid, x), 250); - }, + this.getState = function() { + var state = JSON.parse(JSON.stringify(this.filterTree)); // calls toJSON functions as needed + delete this.filterTree; + return state; + }; - /** - * @memberOf CellMoving.prototype - * @desc a column drag has completed, update data and cleanup - * @param {Hypergrid} grid - */ - endDragColumn: function(grid) { + this.onReset = function() { - var fixedColumnCount = grid.getFixedColumnCount(); - var scrollLeft = grid.getHScrollValue(); + }; - var columnIndex = grid.renderOverridesCache.dragger.columnIndex; + this.onDelete = function() { + delete this.filter; + delete this.filterTree; + }; - if (columnIndex < fixedColumnCount) { - scrollLeft = 0; + this.onCancel = function() { + delete this.filterTree; + }; + + // Following methods called with `state` are independent of dialog; `this.filterTree` is undefined + this.create = function(state) { + if (!this.filter) { + var filterTree = this.setState(state); + var dataRow = {}; + var fieldOption = this.fieldsProvider()[0], + fieldName = fieldOption.name || fieldOption; + + if (!filterTree.validate(validateQuietlyOptions)) { + // `validate()` returned `undefined` meaning valid (returns error string when invalid) + this.filter = function(data) { + dataRow[fieldName] = data; + return filterTree.test(dataRow); + }; + } } + return this.filter; + }; - var renderer = grid.getRenderer(); - var columnEdges = renderer.getColumnEdges(); - var self = this; - var startX = columnEdges[columnIndex - scrollLeft]; - var d = dragger; + this.setState = function(state) { + return ( + this.filterTree = new FilterTree({ + state: state, + fields: this.fieldsProvider() + }) + ); + }; +} - self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease'); - self.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -1 + 'px)'); - d.style.boxShadow = '0px 0px 0px #888888'; +module.exports = CustomFilter; - setTimeout(function() { - grid.renderOverridesCache.dragger = null; - grid.repaint(); - requestAnimationFrame(function() { - d.style.display = 'none'; - grid.endDragColumnNotification(); - }); - }, columnAnimationTime + 50); +},{"filter-tree":6}],69:[function(require,module,exports){ +'use strict'; + +function nn(number) { + return (number < 10 ? '-0' : '-') + number; +} +module.exports = { + date: function(value) { + return value instanceof Date + ? value.getFullYear() + nn(value.getMonth() + 1) + nn(value.getDate()) + : value + ''; }, + default: function(value) { + return value + ''; + } +}; - /** - * @memberOf CellMoving.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isHeaderRow: function(grid, event) { - var gridCell = event.viewPoint; - var isFixed = gridCell.y === 0; - return isFixed; +},{}],70:[function(require,module,exports){ +'use strict'; + +module.exports = (function() { + + var oidPrefix = '.~.#%_'; //this should be something we never will see at the begining of a string + var counter = 0; + + var hash = function(key) { + var typeOf = typeof key; + switch (typeOf) { + case 'number': + return oidPrefix + typeOf + '_' + key; + case 'string': + return oidPrefix + typeOf + '_' + key; + case 'boolean': + return oidPrefix + typeOf + '_' + key; + case 'symbol': + return oidPrefix + typeOf + '_' + key; + case 'undefined': + return oidPrefix + 'undefined'; + case 'object': + case 'function': + /*eslint-disable */ + if (!key.___finhash) { + key.___finhash = oidPrefix + counter++; + } + return key.___finhash; + /*eslint-enable */ + } + }; + + // Object.is polyfill, courtesy of @WebReflection + var is = Object.is || + function(a, b) { + return a === b ? a !== 0 || 1 / a == 1 / b : a != a && b != b; // eslint-disable-line + }; + + // More reliable indexOf, courtesy of @WebReflection + var betterIndexOf = function(arr, value) { + if (value != value || value === 0) { // eslint-disable-line + for (var i = arr.length; i-- && !is(arr[i], value);) { // eslint-disable-line + } + } else { + i = [].indexOf.call(arr, value); + } + return i; + }; + + function Mappy() { + this.keys = []; + this.data = {}; + this.values = []; } -}); + Mappy.prototype.set = function(key, value) { + var hashCode = hash(key); + if (this.data[hashCode] === undefined) { + this.keys.push(key); + this.values.push(value); + } + this.data[hashCode] = value; + }; -module.exports = ColumnMoving; + Mappy.prototype.get = function(key) { + var hashCode = hash(key); + return this.data[hashCode]; + }; -},{"./Feature.js":59}],55:[function(require,module,exports){ -/* eslint-env browser */ -/* global requestAnimationFrame */ + Mappy.prototype.getIfAbsent = function(key, ifAbsentFunc) { + var value = this.get(key); + if (value === undefined) { + value = ifAbsentFunc(key, this); + } + return value; + }; -'use strict'; + Mappy.prototype.size = function() { + return this.keys.length; + }; -var Feature = require('./Feature.js'); + Mappy.prototype.clear = function() { + this.keys.length = 0; + this.data = {}; + }; -/** - * @constructor - */ -var ColumnPicker = Feature.extend('ColumnPicker', { + Mappy.prototype.delete = function(key) { + var hashCode = hash(key); + if (this.data[hashCode] === undefined) { + return; + } + var index = betterIndexOf(this.keys, key); + this.keys.splice(index, 1); + this.values.splice(index, 1); + delete this.data[hashCode]; + }; - alias: 'ColumnPicker', + Mappy.prototype.forEach = function(func) { + var keys = this.keys; + for (var i = 0; i < keys.length; i++) { + var key = keys[i]; + var value = this.get(key); + func(value, key, this); + } + }; - /** - * @memberOf ColumnPicker.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleKeyUp: function(grid, event) { - var key = event.detail.char.toLowerCase(); - var keys = grid.resolveProperty('editorActivationKeys'); - if (keys.indexOf(key) > -1) { - grid.toggleColumnPicker(); + Mappy.prototype.map = function(func) { + var keys = this.keys; + var newMap = new Mappy(); + for (var i = 0; i < keys.length; i++) { + var key = keys[i]; + var value = this.get(key); + var transformed = func(value, key, this); + newMap.set(key, transformed); } - }, + return newMap; + }; -}); + Mappy.prototype.copy = function() { + var keys = this.keys; + var newMap = new Mappy(); + for (var i = 0; i < keys.length; i++) { + var key = keys[i]; + var value = this.get(key); + newMap.set(key, value); + } + return newMap; + }; -module.exports = ColumnPicker; + return Mappy; + +})(); + +},{}],71:[function(require,module,exports){ +/* eslint-env browser */ +/* global requestAnimationFrame */ -},{"./Feature.js":59}],56:[function(require,module,exports){ 'use strict'; -var Feature = require('./Feature.js'); +var _ = require('object-iterators'); +var Base = require('./Base'); + +var images = require('../../images/index'); + +/** @typedef {object} CanvasRenderingContext2D + * @see [CanvasRenderingContext2D](https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D) + */ /** * @constructor + * @desc fin-hypergrid-renderer is the canvas enabled top level sub component that handles the renderering of the Grid. + * + * It relies on two other external subprojects + * + * 1. fin-canvas: a wrapper to provide a simpler interface to the HTML5 canvas component + * 2. rectangular: a small npm module providing Point and Rectangle objects + * + * The fin-hypergrid-renderer is in a unique position to provide critical functionality to the fin-hypergrid in a hightly performant manner. + * Because it MUST iterate over all the visible cells it can store various bits of information that can be encapsulated as a service for consumption by the fin-hypergrid component. + * + * Instances of this object have basically four main functions. + * + * 1. render fixed row headers + * 2. render fixed col headers + * 3. render main data cells + * 4. render grid lines + * + * Same parameters as {@link Renderer#initialize|initialize}, which is called by this constructor. + * */ -var ColumnResizing = Feature.extend('ColumnResizing', { - - alias: 'ColumnResizing', - - /** - * the index of the column wall were currently dragging - * @type {number} - * @default -2 - * @memberOf ColumnResizing.prototype - */ - dragIndex: -2, +var Renderer = Base.extend('Renderer', { - /** - * the pixel location of the where the drag was initiated - * @type {number} - * @default -1 - * @memberOf ColumnResizing.prototype - */ - dragStart: -1, + //the shared single item "pooled" cell object for drawing each cell + cell: { + x: 0, + y: 0, + width: 0, + height: 0 + }, - /** - * the starting width/height of the row/column we are dragging - * @type {number} - * @default -1 - * @memberOf ColumnResizing.prototype - */ - dragIndexStartingSize: -1, + scrollHeight: 0, - /** - * @memberOf ColumnResizing.prototype - * @desc get the mouse x,y coordinate - * @returns {number} - * @param {MouseEvent} event - the mouse event to query - */ - getMouseValue: function(event) { - return event.primitiveEvent.detail.mouse.x; - }, + viewHeight: 0, - /** - * @memberOf ColumnResizing.prototype - * @desc get the grid cell x,y coordinate - * @returns {number} - * @param {window.fin.rectangular.Point} gridCell - */ - getGridCellValue: function(gridCell) { - return gridCell.y; + reset: function() { + this.bounds = { + width: 0, + height: 0 + }; + this.columnEdges = []; + this.columnEdgesIndexMap = []; + this.renderedColumnMinWidths = []; + this.rowEdges = []; + this.rowEdgesIndexMap = []; + this.visibleColumns = []; + this.visibleRows = []; + this.insertionBounds = []; }, /** - * @memberOf ColumnResizing.prototype - * @desc return the grids x,y scroll value - * @returns {number} - * @param {Hypergrid} grid + * @summary Constructor logic + * @desc This method will be called upon instantiation of this class or of any class that extends from this class. + * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most "senior" class through that of the class of the new instance. + * @memberOf Renderer.prototype */ - getScrollValue: function(grid) { - return grid.getHScrollValue(); + initialize: function(grid) { + this.grid = grid; + this.reset(); }, - /** - * @memberOf ColumnResizing.prototype - * @desc return the width/height of the row/column of interest - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest - */ - getAreaSize: function(grid, index) { - return grid.getColumnWidth(index); - }, + //this function computes the grid coordinates used for extremely fast iteration over + //painting the grid cells. this function is very fast, for thousand rows X 100 columns + //on a modest machine taking usually 0ms and no more that 3 ms. + computeCellsBounds: function() { - /** - * @memberOf ColumnResizing.prototype - * @desc set the width/height of the row/column at index - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest - * @param {number} value - the width/height to set to - */ - setAreaSize: function(grid, index, value) { - grid.setColumnWidth(index, value); - }, + //var startTime = Date.now(); - /** - * @memberOf ColumnResizing.prototype - * @desc return the recently rendered area's width/height - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest - */ - getPreviousAbsoluteSize: function(grid, index) { - return grid.getRenderedWidth(index); - }, + var scrollTop = this.getScrollTop(), + scrollLeft = this.getScrollLeft(), - /** - * @memberOf ColumnResizing.prototype - * @desc returns the index of which divider I'm over - * @returns {number} - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - overAreaDivider: function(grid, event) { - return grid.overColumnDivider(event); - }, + numColumns = this.getColumnCount(), + numFixedColumns = this.getFixedColumnCount(), - /** - * @memberOf ColumnResizing.prototype - * @desc am I over the column/row area - * @returns {boolean} - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isFirstFixedOtherArea: function(grid, event) { - return this.isFirstFixedRow(grid, event); - }, + numRows = this.getRowCount(), + numFixedRows = this.getFixedRowCount(), - /** - * @memberOf ColumnResizing.prototype - * @desc return the cursor name - * @returns {string} - */ - getCursorName: function() { - return 'col-resize'; - }, + bounds = this.getBounds(), + grid = this.grid, + numberOfBottomTotalsRows = grid.behavior.getDataModel().getBottomTotals().length, + viewWidth = bounds.width || grid.canvas.width, // if 0, we must be in bootstrap + viewHeight = bounds.height - numberOfBottomTotalsRows * grid.behavior.getDefaultRowHeight(), - /** - * @memberOf ColumnResizing.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDrag: function(grid, event) { - if (this.dragIndex > -2) { - //var fixedAreaCount = this.getFixedAreaCount(grid); - //var offset = this.getFixedAreaSize(grid, fixedAreaCount + areaIndex); - var mouse = this.getMouseValue(event); - var scrollValue = this.getScrollValue(grid); - if (this.dragIndex < this.getFixedAreaCount(grid)) { - scrollValue = 0; - } - var previous = this.getPreviousAbsoluteSize(grid, this.dragIndex - scrollValue); - var distance = mouse - previous; - this.setAreaSize(grid, this.dragIndex, distance); - } else if (this.next) { - this.next.handleMouseDrag(grid, event); - } - }, + insertionBoundsCursor = 0, + previousInsertionBoundsCursorValue = 0, - /** - * @memberOf ColumnResizing.prototype - * @desc get the width/height of a specific row/column - * @param {Hypergrid} grid - * @param {number} areaIndex - the row/column index of interest - */ - getSize: function(grid, areaIndex) { - return this.getAreaSize(grid, areaIndex); - }, + start = 0, + x = 0, y = 0, + c, r, + vx, vy, + width, height, + firstVX, lastVX, + firstVY, lastVY; - /** - * @memberOf ColumnResizing.prototype - * @desc return the fixed area rows/columns count - * @returns {number} - * @param {Hypergrid} grid - */ - getOtherFixedAreaCount: function(grid) { - return grid.getFixedRowCount(); - }, + this.getColumnEdges().length = 0; + this.rowEdges.length = 0; - /** - * @memberOf ColumnResizing.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDown: function(grid, event) { - var isEnabled = this.isEnabled(grid); - var overArea = this.overAreaDivider(grid, event); - if (isEnabled && overArea > -1 && this.isFirstFixedOtherArea(grid, event)) { - var scrollValue = this.getScrollValue(grid); - if (overArea < this.getFixedAreaCount(grid)) { - scrollValue = 0; - } - this.dragIndex = overArea - 1 + scrollValue; - this.dragStart = this.getMouseValue(event); - this.dragIndexStartingSize = 0; - this.detachChain(); - } else if (this.next) { - this.next.handleMouseDown(grid, event); - } - }, + this.columnEdges[0] = 0; + this.rowEdges[0] = 0; + this.scrollHeight = 0; - /** - * @memberOf ColumnResizing.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseUp: function(grid, event) { - var isEnabled = this.isEnabled(grid); - if (isEnabled && this.dragIndex > -2) { - this.cursor = null; - this.dragIndex = -2; + this.visibleColumns.length = 0; + this.visibleRows.length = 0; + this.columnEdgesIndexMap = []; + this.rowEdgesIndexMap = []; - event.primitiveEvent.stopPropagation(); - //delay here to give other events a chance to be dropped - var self = this; - grid.synchronizeScrollingBoundries(); - setTimeout(function() { - self.attachChain(); - }, 200); - } else if (this.next) { - this.next.handleMouseUp(grid, event); + this.insertionBounds = []; + + if (this.grid.isShowRowNumbers()) { + start--; + this.columnEdges[-1] = -1; } - }, - /** - * @memberOf ColumnResizing.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseMove: function(grid, event) { - if (this.dragIndex > -2) { - return; + for (c = start; c < numColumns; c++) { + vx = c; + if (c >= numFixedColumns) { + vx = vx + scrollLeft; + if (firstVX === undefined) { + firstVX = vx; + } + lastVX = vx; + } + if (x > viewWidth || numColumns <= vx) { + break; + } + width = grid.getColumnWidth(vx); + x = x + width; + this.columnEdges[c + 1] = Math.round(x); + this.visibleColumns[c] = vx; + this.columnEdgesIndexMap[vx] = c; + + insertionBoundsCursor = insertionBoundsCursor + Math.round(width / 2) + previousInsertionBoundsCursorValue; + this.insertionBounds.push(insertionBoundsCursor); + previousInsertionBoundsCursorValue = Math.round(width / 2); } - this.cursor = null; - if (this.next) { - this.next.handleMouseMove(grid, event); + + for (r = 0; r < numRows; r++) { + vy = r; + if (r >= numFixedRows) { + vy = vy + scrollTop; + if (firstVY === undefined) { + firstVY = vy; + } + lastVY = vy; + } + if (y > viewHeight || numRows <= vy) { + break; + } + height = grid.getRowHeight(vy); + y = y + height; + this.rowEdges[r + 1] = Math.round(y); + this.visibleRows[r] = vy; + this.rowEdgesIndexMap[vy] = r; } - this.checkForAreaResizeCursorChange(grid, event); + this.viewHeight = viewHeight; + this.dataWindow = this.grid.newRectangle(firstVX, firstVY, lastVX - firstVX, lastVY - firstVY); }, /** - * @memberOf ColumnResizing.prototype - * @desc fill this in - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {Object} a property value at a key, delegates to the grid */ - checkForAreaResizeCursorChange: function(grid, event) { - var isEnabled = this.isEnabled(grid); - if (isEnabled && this.overAreaDivider(grid, event) > -1 && this.isFirstFixedOtherArea(grid, event)) { - this.cursor = this.getCursorName(); - } else { - this.cursor = null; - } - + resolveProperty: function(key) { + return this.grid.resolveProperty(key); }, - /** - * @param {Hypergrid} grid - * @returns {number} - * @default -2 - * @memberOf ColumnResizing.prototype + /** @deprecated Use `.grid` property instead. + * @memberOf Renderer.prototype + * @returns {Hypergrid} grid */ - getFixedAreaCount: function(grid) { - var count = grid.getFixedColumnCount() + (grid.isShowRowNumbers() ? 1 : 0) + (grid.hasHierarchyColumn() ? 1 : 0); - return count; + getGrid: function() { + return this.deprecated('grid', { since: '0.2' }); }, /** - * @param {Hypergrid} grid - * @param event - * @default -2 - * @memberOf ColumnResizing.prototype + * @memberOf Renderer.prototype + * @summary Notify the fin-hypergrid everytime we've repainted. + * @desc This is the entry point from fin-canvas. + * @param {CanvasRenderingContext2D} gc */ - handleDoubleClick: function(grid, event) { - var isEnabled = this.isEnabled(grid); - var hasCursor = this.overAreaDivider(grid, event) > -1; //this.cursor !== null; - var headerRowCount = grid.getHeaderRowCount(); - //var headerColCount = grid.getHeaderColumnCount(); - var gridCell = event.gridCell; - if (isEnabled && hasCursor && (gridCell.y <= headerRowCount)) { - grid.autosizeColumn(gridCell.x - 1); - } else if (this.next) { - this.next.handleDoubleClick(grid, event); + _paint: function(gc) { + if (this.grid) { + this.renderGrid(gc); + this.grid.gridRenderedNotification(); } }, /** - * @param {Hypergrid} grid - * @returns {boolean} - * @default -2 - * @memberOf ColumnResizing.prototype + * @memberOf Renderer.prototype + * @returns {number} Answer how many rows we rendered */ - isEnabled: function(grid) { - return true; - } - -}); - -module.exports = ColumnResizing; - -},{"./Feature.js":59}],57:[function(require,module,exports){ -'use strict'; - -var Feature = require('./Feature.js'); - -/** - * @constructor - */ -var ColumnSelection = Feature.extend('ColumnSelection', { + getVisibleRowsCount: function() { + return this.visibleRows.length - 1; + }, - alias: 'ColumnSelection', + getVisibleScrollHeight: function() { + return this.viewHeight - this.grid.getFixedRowsHeight(); + }, /** - * The pixel location of the mouse pointer during a drag operation. - * @type {window.fin.rectangular.Point} - * @default null - * @memberOf ColumnSelection.prototype + * @memberOf Renderer.prototype + * @returns {number[]} Rows we just rendered. */ - currentDrag: null, + getVisibleRows: function() { + return this.visibleRows; + }, /** - * The cell coordinates of the where the mouse pointer is during a drag operation. - * @type {Object} - * @default null - * @memberOf ColumnSelection.prototype + * @memberOf Renderer.prototype + * @returns {number} Numer of columns we just rendered. */ - lastDragCell: null, + getVisibleColumnsCount: function() { + return this.visibleColumns.length - 1; + }, /** - * a millisecond value representing the previous time an autoscroll started - * @type {number} - * @default 0 - * @memberOf ColumnSelection.prototype + * @memberOf Renderer.prototype + * @returns {number} Columns we just rendered. */ - sbLastAuto: 0, + getVisibleColumns: function() { + return this.visibleColumns; + }, /** - * a millisecond value representing the time the current autoscroll started - * @type {number} - * @default 0 - * @memberOf ColumnSelection.prototype + * @memberOf Renderer.prototype + * @returns {number} The column index whne the mouseEvent coordinates are over a column divider. */ - sbAutoStart: 0, + overColumnDivider: function(x) { + x = Math.round(x); + var edges = this.getColumnEdges(); + var whichCol = edges.indexOf(x - 1); + if (whichCol < 0) { + whichCol = edges.indexOf(x); + } + if (whichCol < 0) { + whichCol = edges.indexOf(x - 2); + } + if (whichCol < 0) { + whichCol = edges.indexOf(x + 1); + } + if (whichCol < 0) { + whichCol = edges.indexOf(x - 3); + } + return whichCol; + }, /** - * @memberOf ColumnSelection.prototype - * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {number} The row index when the mouseEvent coordinates are over a row divider. */ - handleMouseUp: function(grid, event) { - if (this.dragging) { - this.dragging = false; + overRowDivider: function(y) { + y = Math.round(y); + var which = this.rowEdges.indexOf(y + 1); + if (which < 0) { + which = this.rowEdges.indexOf(y); } - if (this.next) { - this.next.handleMouseUp(grid, event); - return; + if (which < 0) { + which = this.rowEdges.indexOf(y - 1); } + return which; }, /** - * @memberOf ColumnSelection.prototype - * @desc * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @param {Point} cell + * @returns {Rectangle} Bounding rect of the given `cell`. */ - handleMouseDown: function(grid, event) { + getBoundsOfCell: function(cell) { + return this._getBoundsOfCell(cell.x, cell.y); + }, - if ((!grid.isColumnSelection() || event.mousePoint.y < 5) && this.next) { - this.next.handleMouseDown(grid, event); - return; - } + /** + * @memberOf Renderer.prototype + * @param {number} c - The horizontal coordinate. + * @param {number} r - The vertical coordinate. + * @returns {Rectangle} Bounding rect of cell with the given coordinates. + */ + _getBoundsOfCell: function(c, r) { + var xOutside = false, + yOutside = false, + cell = this.cell; - var isRightClick = event.primitiveEvent.detail.isRightClick; - var cell = event.gridCell; - var viewCell = event.viewPoint; - var dx = cell.x; - var dy = cell.y; + var y, x = this.columnEdgesIndexMap[c]; + if (x === undefined) { + x = this.columnEdgesIndexMap[c - 1]; + xOutside = true; + } - var isHeader = grid.isShowHeaderRow() && dy === 0 && dx !== -1; + var oy, ox = this.columnEdges[x], + cy, cx = this.columnEdges[x + 1], + ey, ex = cx - ox; - if (isRightClick || !isHeader) { - if (this.next) { - this.next.handleMouseDown(grid, event); - } - } else { + cell.x = xOutside ? cx : ox; + cell.width = xOutside ? 0 : ex; - var numFixedColumns = grid.getFixedColumnCount(); + if (r < 0) { // bottom totals rows + var behavior = this.grid.behavior, + bounds = this.getBounds(); - //if we are in the fixed area do not apply the scroll values - //check both x and y values independently - if (viewCell.x < numFixedColumns) { - dx = viewCell.x; + ey = behavior.getDefaultRowHeight(); + oy = bounds.height + r * ey; + cy = oy + ey; + } else { + y = this.rowEdgesIndexMap[r]; + if (y === undefined) { + y = this.rowEdgesIndexMap[r - 1]; + yOutside = true; } - var dCell = grid.newPoint(dx, 0); - - var primEvent = event.primitiveEvent; - var keys = primEvent.detail.keys; - this.dragging = true; - this.extendSelection(grid, dCell, keys); + oy = this.rowEdges[y]; + cy = this.rowEdges[y + 1]; + ey = cy - oy; } + + cell.y = yOutside ? cy : oy; + cell.height = yOutside ? 0 : ey; + + return cell; }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @desc answer the column index under the coordinate at pixelX + * @param {number} pixelX - The horizontal coordinate. + * @returns {number} The column index under the coordinate at pixelX. */ - handleMouseDrag: function(grid, event) { + getColumnFromPixelX: function(pixelX) { + var width = 0, + fixedColumnCount = this.getFixedColumnCount(), + scrollLeft = this.grid.getHScrollValue(), + edges = this.getColumnEdges(); - if ((!grid.isColumnSelection() || this.isColumnDragging(grid)) && this.next) { - this.next.handleMouseDrag(grid, event); - return; + for (var c = 1; c < edges.length - 1; c++) { + width = edges[c] - (edges[c] - edges[c - 1]) / 2; + if (pixelX < width) { + if (c > fixedColumnCount) { + c = c + scrollLeft; + } + return c - 1; + } + } + if (c > fixedColumnCount) { + c = c + scrollLeft; } + return c - 1; + }, - var isRightClick = event.primitiveEvent.detail.isRightClick; - if (isRightClick || !this.dragging) { - if (this.next) { - this.next.handleMouseDrag(grid, event); - } - } else { + /** + * @memberOf Renderer.prototype + * @desc Answer specific data cell coordinates given mouse coordinates in pixels. + * @param {Point} point + * @returns {Point} Cell coordinates + */ + getGridCellFromMousePoint: function(point) { - var numFixedColumns = grid.getFixedColumnCount(); + var behavior = this.grid.behavior; + var width = 0; + var height = 0; + var x, y, c, r; + var previous = 0; + var columnEdges = this.getColumnEdges(); + var fixedColumnCount = this.getFixedColumnCount(); // + gridSize; + var fixedRowCount = this.getFixedRowCount(); - var cell = event.gridCell; - var viewCell = event.viewPoint; - var dx = cell.x; - var dy = cell.y; + // var fixedColumnCount = this.getFixedColumnCount(); + // var fixedRowCount = this.getFixedRowCount(); + var scrollX = this.getScrollLeft(); + var scrollY = this.getScrollTop(); - //if we are in the fixed area do not apply the scroll values - //check both x and y values independently - if (viewCell.x < numFixedColumns) { - dx = viewCell.x; + for (c = 0; c < columnEdges.length; c++) { + width = columnEdges[c]; + if (point.x < width) { + x = Math.max(0, point.x - previous - 2); + break; + } + previous = width; + } + c--; + previous = 0; + for (r = 0; r < this.rowEdges.length; r++) { + height = this.rowEdges[r]; + if (point.y < height) { + y = Math.max(0, point.y - previous - 2); + break; } + previous = height; + } + r--; + if (point.x < 0) { + c = -1; + } + if (point.y < 0) { + r = -1; + } - var dCell = grid.newPoint(dx, dy); + var viewPoint = this.grid.newPoint(c, r); - var primEvent = event.primitiveEvent; - this.currentDrag = primEvent.detail.mouse; - this.lastDragCell = dCell; + //compensate if we are scrolled + if (c >= fixedColumnCount) { + c = c + scrollX; + } + if (r >= fixedRowCount) { + r = r + scrollY; + } - this.checkDragScroll(grid, this.currentDrag); - this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys); + var translatedIndex = -1; + + var column = behavior.getColumn(c); + if (column) { + translatedIndex = column.index; } + + return { + gridCell: this.grid.newPoint(c, r), + mousePoint: this.grid.newPoint(x, y), + viewPoint: viewPoint, + dataCell: this.grid.newPoint(translatedIndex, r), + }; }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @summary Determines if a column is visible. + * @param {number} colIndex - the column index* + * @returns {boolean} The given column is fully visible. */ - handleKeyDown: function(grid, event) { - if (grid.getLastSelectionType() !== 'column') { - if (this.next) { - this.next.handleKeyDown(grid, event); - } - return; - } - var command = 'handle' + event.detail.char; - if (this[command]) { - this[command].call(this, grid, event.detail); - } + isColumnVisible: function(colIndex) { + var isVisible = this.visibleColumns.indexOf(colIndex) !== -1; + return isVisible; }, /** - * @memberOf ColumnSelection.prototype - * @desc Handle a mousedrag selection - * @param {Hypergrid} grid - * @param {Object} mouse - the event details - * @param {Array} keys - array of the keys that are currently pressed down + * @memberOf Renderer.prototype + * @returns {number} The width x coordinate of the last rendered column */ - handleMouseDragCellSelection: function(grid, gridCell, keys) { - - //var behavior = grid.getBehavior(); - var x = gridCell.x; - // var previousDragExtent = grid.getDragExtent(); - var mouseDown = grid.getMouseDown(); - - var newX = x - mouseDown.x; - //var newY = y - mouseDown.y; - - // if (previousDragExtent.x === newX && previousDragExtent.y === newY) { - // return; - // } - - grid.clearMostRecentColumnSelection(); - - grid.selectColumn(mouseDown.x, x); - grid.setDragExtent(grid.newPoint(newX, 0)); + getFinalVisableColumnBoundary: function() { + var isMaxX = this.isLastColumnVisible(); + var chop = isMaxX ? 2 : 1; + var colWall = this.getColumnEdges()[this.getColumnEdges().length - chop]; + var result = Math.min(colWall, this.getBounds().width - 200); + return result; + }, - grid.repaint(); + /** + * @memberOf Renderer.prototype + * @summary Determines visibility of a row. + * @param {number} rowIndex - the row index + * @returns {boolean} The given row is fully visible. + */ + isRowVisible: function(rowIndex) { + var isVisible = this.visibleRows.indexOf(rowIndex) !== -1; + return isVisible; }, /** - * @memberOf ColumnSelection.prototype - * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above) - * @param {Hypergrid} grid - * @param {Object} mouse - the event details + * @memberOf Renderer.prototype + * @summary Determines if a cell is selected. + * @param {number} x - the x cell coordinate + * @param {number} y - the y cell coordinate* + * @returns {boolean} The given cell is fully visible. */ - checkDragScroll: function(grid, mouse) { - if (!grid.resolveProperty('scrollingEnabled')) { - return; - } - var b = grid.getDataBounds(); - var inside = b.contains(mouse); - if (inside) { - if (grid.isScrollingNow()) { - grid.setScrollingNow(false); - } - } else if (!grid.isScrollingNow()) { - grid.setScrollingNow(true); - this.scrollDrag(grid); - } + isSelected: function(x, y) { + return this.grid.isSelected(x, y); }, /** - * @memberOf ColumnSelection.prototype - * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly - * @param {Hypergrid} grid + * @memberOf Renderer.prototype + * @desc This is the main forking of the renderering task. + * @param {CanvasRenderingContext2D} gc */ - scrollDrag: function(grid) { + renderGrid: function(gc) { + gc.beginPath(); + + this.paintCells(gc); + this.paintGridlines(gc); + this.renderOverrides(gc); + this.renderFocusCell(gc); + gc.closePath(); + }, + + focusLineStep: [ + [5, 5], + [0, 1, 5, 4], + [0, 2, 5, 3], + [0, 3, 5, 2], + [0, 4, 5, 1], + [0, 5, 5, 0], + [1, 5, 4, 0], + [2, 5, 3, 0], + [3, 5, 2, 0], + [4, 5, 1, 0] + ], + + renderFocusCell: function(gc) { + gc.beginPath(); + this._renderFocusCell(gc); + gc.closePath(); + }, + + _renderFocusCell: function(gc) { - if (!grid.isScrollingNow()) { + var selections = this.grid.selectionModel.getSelections(); + if (!selections || selections.length === 0) { + return; + } + var selection = selections[selections.length - 1]; + var mouseDown = selection.origin; + if (mouseDown.x === -1) { + //no selected area, lets exit return; } - var lastDragCell = this.lastDragCell; - var b = grid.getDataBounds(); - var xOffset = 0; - var yOffset = 0; + var visibleColumns = this.getVisibleColumns(); + var visibleRows = this.getVisibleRows(); + var lastVisibleColumn = visibleColumns[visibleColumns.length - 1]; + var lastVisibleRow = visibleRows[visibleRows.length - 1]; - var numFixedColumns = grid.getFixedColumnCount(); - var numFixedRows = grid.getFixedRowCount(); + var extent = selection.extent; - var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns; - var dragEndInFixedAreaY = lastDragCell.y < numFixedRows; + var dpOX = Math.min(mouseDown.x, mouseDown.x + extent.x); + var dpOY = Math.min(mouseDown.y, mouseDown.y + extent.y); - if (this.currentDrag.x < b.origin.x) { - xOffset = -1; + //lets check if our selection rectangle is scrolled outside of the visible area + if (dpOX > lastVisibleColumn) { + return; //the top of our rectangle is below visible } - - if (this.currentDrag.x > b.origin.x + b.extent.x) { - xOffset = 1; + if (dpOY > lastVisibleRow) { + return; //the left of our rectangle is to the right of being visible } - var dragCellOffsetX = xOffset; - var dragCellOffsetY = yOffset; + var dpEX = Math.max(mouseDown.x, mouseDown.x + extent.x) + 1; + dpEX = Math.min(dpEX, 1 + lastVisibleColumn); - if (dragEndInFixedAreaX) { - dragCellOffsetX = 0; - } + var dpEY = Math.max(mouseDown.y, mouseDown.y + extent.y) + 1; + dpEY = Math.min(dpEY, 1 + lastVisibleRow); - if (dragEndInFixedAreaY) { - dragCellOffsetY = 0; + var o = this._getBoundsOfCell(dpOX, dpOY); + var ox = Math.round((o.x === undefined) ? this.grid.getFixedColumnsWidth() : o.x); + var oy = Math.round((o.y === undefined) ? this.grid.getFixedRowsHeight() : o.y); + // var ow = o.width; + // var oh = o.height; + var e = this._getBoundsOfCell(dpEX, dpEY); + var ex = Math.round((e.x === undefined) ? this.grid.getFixedColumnsWidth() : e.x); + var ey = Math.round((e.y === undefined) ? this.grid.getFixedRowsHeight() : e.y); + // var ew = e.width; + // var eh = e.height; + var x = Math.min(ox, ex); + var y = Math.min(oy, ey); + var width = 1 + ex - ox; + var height = 1 + ey - oy; + if (x === ex) { + width = ox - ex; + } + if (y === ey) { + height = oy - ey; + } + if (width * height < 1) { + //if we are only a skinny line, don't render anything + return; } - this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY); - grid.scrollBy(xOffset, yOffset); - this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection - grid.repaint(); - setTimeout(this.scrollDrag.bind(this, grid), 25); - }, - - /** - * @memberOf ColumnSelection.prototype - * @desc extend a selection or create one if there isnt yet - * @param {Hypergrid} grid - * @param {Object} gridCell - the event details - * @param {Array} keys - array of the keys that are currently pressed down - */ - extendSelection: function(grid, gridCell, keys) { - grid.stopEditing(); - //var hasCTRL = keys.indexOf('CTRL') !== -1; - var hasSHIFT = keys.indexOf('SHIFT') !== -1; - - // var scrollTop = grid.getVScrollValue(); - // var scrollLeft = grid.getHScrollValue(); + gc.rect(x, y, width, height); + gc.fillStyle = this.resolveProperty('selectionRegionOverlayColor'); + gc.fill(); + gc.lineWidth = 1; + gc.strokeStyle = this.resolveProperty('selectionRegionOutlineColor'); - // var numFixedColumns = 0;//grid.getFixedColumnCount(); - // var numFixedRows = 0;//grid.getFixedRowCount(); + // animate the dashed line a bit here for fun - var mousePoint = grid.getMouseDown(); - var x = gridCell.x; // - numFixedColumns + scrollLeft; - var y = gridCell.y; // - numFixedRows + scrollTop; + gc.stroke(); - //were outside of the grid do nothing - if (x < 0 || y < 0) { - return; - } + //gc.rect(x, y, width, height); - //we have repeated a click in the same spot deslect the value from last time - // if (mousePoint && x === mousePoint.x && y === mousePoint.y) { - // grid.clearSelections(); - // grid.popMouseDown(); - // grid.repaint(); - // return; - // } + //gc.strokeStyle = 'white'; - // if (!hasCTRL && !hasSHIFT) { - // grid.clearSelections(); - // } + // animate the dashed line a bit here for fun + //gc.setLineDash(this.focusLineStep[Math.floor(10 * (Date.now() / 300 % 1)) % this.focusLineStep.length]); - if (hasSHIFT) { - grid.clearMostRecentColumnSelection(); - grid.selectColumn(x, mousePoint.x); - grid.setDragExtent(grid.newPoint(x - mousePoint.x, 0)); - } else { - grid.toggleSelectColumn(x, keys); - grid.setMouseDown(grid.newPoint(x, y)); - grid.setDragExtent(grid.newPoint(0, 0)); - } - grid.repaint(); + //gc.stroke(); }, - /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid + * @memberOf Renderer.prototype + * @desc iterate the renderering overrides and manifest each + * @param {CanvasRenderingContext2D} gc */ - handleDOWNSHIFT: function(grid) {}, + renderOverrides: function(gc) { + var cache = this.grid.renderOverridesCache; + for (var key in cache) { + if (cache.hasOwnProperty(key)) { + var override = cache[key]; + if (override) { + this.renderOverride(gc, override); + } + } + } + }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @desc copy each overrides specified area to it's target and blank out the source area + * @param {CanvasRenderingContext2D} gc + * @param {OverrideObject} override - an object with details contain an area and a target context */ - handleUPSHIFT: function(grid) {}, + renderOverride: function(gc, override) { + //lets blank out the drag row + var hdpiRatio = override.hdpiratio; + //var edges = this.getColumnEdges(); + var startX = override.startX; //hdpiRatio * edges[override.columnIndex]; + var width = override.width + 1; + var height = override.height; + var targetCTX = override.ctx; + var imgData = gc.getImageData(startX, 0, Math.round(width * hdpiRatio), Math.round(height * hdpiRatio)); + targetCTX.putImageData(imgData, 0, 0); + gc.fillStyle = this.resolveProperty('backgroundColor2'); + gc.fillRect(Math.round(startX / hdpiRatio), 0, width, height); + }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {number} Current vertical scroll value. */ - handleLEFTSHIFT: function(grid) { - this.moveShiftSelect(grid, -1); + getScrollTop: function() { + return this.grid.getVScrollValue(); }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {number} Current horizontal scroll value. */ - handleRIGHTSHIFT: function(grid) { - this.moveShiftSelect(grid, 1); + getScrollLeft: function() { + return this.grid.getHScrollValue(); }, - /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + /** @deprecated Use `.grid.behavior` property instead. + * @memberOf Renderer.prototype + * @returns {Behavior} The behavior (model). */ - handleDOWN: function(grid) { + getBehavior: function() { + return this.deprecated('grid.behavior', { since: '0.2' }); + }, - // var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); - // var maxRows = grid.getRowCount() - 1; + getColumnEdges: function() { + return this.columnEdges; + }, - // var newX = mouseCorner.x; - // var newY = grid.getHeaderRowCount() + grid.getVScrollValue(); + getRowEdges: function() { + return this.rowEdges; + }, - // newY = Math.min(maxRows, newY); + /** + * @memberOf Renderer.prototype + * @returns {boolean} The last col was rendered (is visible) + */ + isLastColumnVisible: function() { + var lastColumnIndex = this.getColumnCount() - 1; + return this.visibleColumns.indexOf(lastColumnIndex) !== -1; + }, - // grid.clearSelections(); - // grid.select(newX, newY, 0, 0); - // grid.setMouseDown(new grid.rectangular.Point(newX, newY)); - // grid.setDragExtent(new grid.rectangular.Point(0, 0)); + /** + * @memberOf Renderer.prototype + * @returns {number} The rendered column width at index + */ + getRenderedWidth: function(index) { + return this.getColumnEdges()[index]; + }, - // grid.repaint(); + /** + * @memberOf Renderer.prototype + * @returns {number} The rendered row height at index + */ + getRenderedHeight: function(index) { + return this.rowEdges[index]; }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {fin-canvas} my [fin-canvas](https://github.com/stevewirts/fin-canvas) */ - handleUP: function(grid) {}, + getCanvas: function() { + return this.grid.getCanvas(); + }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {boolean} User is currently dragging a column for reordering. */ - handleLEFT: function(grid) { - this.moveSingleSelect(grid, -1); + isDraggingColumn: function() { + return this.grid.isDraggingColumn(); + }, + + /** + * @memberOf Renderer.prototype + * @returns {number} The row to goto for a page up. + */ + getPageUpRow: function() { + var grid = this.grid, + scrollHeight = this.getVisibleScrollHeight(), + headerRows = this.grid.getFixedRowCount(), + top = this.dataWindow.origin.y - headerRows, + scanHeight = 0; + while (scanHeight < scrollHeight && top > -1) { + scanHeight = scanHeight + grid.getRowHeight(top); + top--; + } + return top + 1; }, /** - * @memberOf ColumnSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @returns {number} The row to goto for a page down. */ - handleRIGHT: function(grid) { - this.moveSingleSelect(grid, 1); + getPageDownRow: function() { + var headerRows = this.grid.getFixedRowCount(); + var rowNum = this.dataWindow.corner.y - headerRows - 1; + return rowNum; }, /** - * @memberOf ColumnSelection.prototype - * @desc If we are holding down the same navigation key, accelerate the increment we scroll - * #### returns: integer + * @memberOf Renderer.prototype + * @returns {number} The number of columns. */ - getAutoScrollAcceleration: function() { - var count = 1; - var elapsed = this.getAutoScrollDuration() / 2000; - count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed)); - return count; + getColumnCount: function() { + return this.grid.getColumnCount(); }, /** - * @memberOf ColumnSelection.prototype - * @desc set the start time to right now when we initiate an auto scroll + * @memberOf Renderer.prototype + * @returns {number} The number of rows. */ - setAutoScrollStartTime: function() { - this.sbAutoStart = Date.now(); + getRowCount: function() { + return this.grid.getRowCount(); }, /** - * @memberOf ColumnSelection.prototype - * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time + * @memberOf Renderer.prototype + * @returns {number} The number of fixed columns. */ - pingAutoScroll: function() { - var now = Date.now(); - if (now - this.sbLastAuto > 500) { - this.setAutoScrollStartTime(); - } - this.sbLastAuto = Date.now(); + getFixedColumnCount: function() { + return this.grid.getFixedColumnCount(); }, /** - * @memberOf ColumnSelection.prototype - * @desc answer how long we have been auto scrolling - * #### returns: integer + * @memberOf Renderer.prototype + * @returns {number} The number of fixed rows. */ - getAutoScrollDuration: function() { - if (Date.now() - this.sbLastAuto > 500) { - return 0; - } - return Date.now() - this.sbAutoStart; + getFixedRowCount: function() { + return this.grid.getFixedRowCount(); }, /** - * @memberOf ColumnSelection.prototype - * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary. - * @param {Hypergrid} grid - * @param {number} offsetX - x coordinate to start at - * @param {number} offsetY - y coordinate to start at + * @memberOf Renderer.prototype + * @returns {number} The number of header rows. */ - moveShiftSelect: function(grid, offsetX) { - - var maxColumns = grid.getColumnCount() - 1; - - var maxViewableColumns = grid.getVisibleColumns() - 1; - - if (!grid.resolveProperty('scrollingEnabled')) { - maxColumns = Math.min(maxColumns, maxViewableColumns); - } - - var origin = grid.getMouseDown(); - var extent = grid.getDragExtent(); - - var newX = extent.x + offsetX; - //var newY = grid.getRowCount(); - - newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX)); - - grid.clearMostRecentColumnSelection(); - grid.selectColumn(origin.x, origin.x + newX); - - grid.setDragExtent(grid.newPoint(newX, 0)); - - if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) { - this.pingAutoScroll(); - } - - grid.repaint(); - + getHeaderRowCount: function() { + return this.grid.getHeaderRowCount(); }, /** - * @memberOf ColumnSelection.prototype - * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent. - * @param {Hypergrid} grid - * @param {number} offsetX - x coordinate to start at - * @param {number} offsetY - y coordinate to start at + * @memberOf Renderer.prototype + * @returns {number} The number of header columns. */ - moveSingleSelect: function(grid, offsetX) { - - var maxColumns = grid.getColumnCount() - 1; - - var maxViewableColumns = grid.getVisibleColumnsCount() - 1; - - if (!grid.resolveProperty('scrollingEnabled')) { - maxColumns = Math.min(maxColumns, maxViewableColumns); - } - - var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); - - var newX = mouseCorner.x + offsetX; - //var newY = grid.getRowCount(); + getHeaderColumnCount: function() { + return this.grid.getHeaderColumnCount(); + }, - newX = Math.min(maxColumns, Math.max(0, newX)); + /** @summary Smart render the grid. + * @desc Paint all the cells of a grid, including all "fixed" columns and rows. + * We snapshot the context to insure against its pollution. + * `try...catch` surrounds each cell paint in case a cell editor throws an error. + * The error message is error-logged to console AND displayed in cell. + * @memberOf Renderer.prototype + * @param {CanvasRenderingContext2D} gc + */ + paintCells: function(gc) { + var renderCellError, + message, + x, y, + c, r, - grid.clearSelections(); - grid.selectColumn(newX); - grid.setMouseDown(grid.newPoint(newX, 0)); - grid.setDragExtent(grid.newPoint(0, 0)); + columnEdges = this.getColumnEdges(), + rowEdges = this.rowEdges, - if (grid.insureModelColIsVisible(newX, offsetX)) { - this.pingAutoScroll(); - } + visibleCols = this.getVisibleColumns(), + visibleRows = this.getVisibleRows(), - grid.repaint(); + behavior = this.grid.behavior, - }, + clipX = 0, + clipY = 0, + clipWidth, + clipHeight = this.getBounds().height, - isColumnDragging: function(grid) { - var dragger = grid.lookupFeature('ColumnMoving'); - if (!dragger) { - return false; - } - var isActivated = dragger.dragging && !this.dragging; - return isActivated; - } + loopStart = this.grid.isShowRowNumbers() ? -1 : 0, + loopLength = visibleCols.length; // regardless of loopStart, due to definition of .length -}); + this.buttonCells = {}; -module.exports = ColumnSelection; + if (loopLength) { // this if prevents painting just the fixed columns when there are no visible columns -},{"./Feature.js":59}],58:[function(require,module,exports){ -'use strict'; + // For each column... + for (x = loopStart; x < loopLength; x++, clipX += clipWidth) { -var Feature = require('./Feature.js'); + c = visibleCols[x]; + this.renderedColumnMinWidths[c] = 0; + renderCellError = behavior.getColumnProperties(c).renderCellError; -/** - * @constructor - */ -var ColumnSorting = Feature.extend('ColumnSorting', { + gc.save(); - alias: 'ColumnSorting', + // Clip to visible portion of column to prevent overflow to right. Previously we clipped to entire visible grid and dealt with overflow by overpainting with next column. However, this strategy fails when transparent background (no background color). + // TODO: if extra clip() calls per column affect performance (not the clipping itself which was happening anyway, but the clip calls which set up the clipping), use previous strategy when there is a background color + clipWidth = columnEdges[x - loopStart] - clipX; + gc.beginPath(); + gc.rect(clipX, clipY, clipWidth, clipHeight); + gc.clip(); - /** - * @memberOf ColumnSorting.prototype - * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ + // For each row (of each column)... + for (y = 0; y < visibleRows.length; y++) { - handleDoubleClick: function(grid, event) { - var gridCell = event.gridCell; - if (grid.isShowHeaderRow() && gridCell.y === 0 && gridCell.x !== -1) { - var keys = event.primitiveEvent.detail.keys; - grid.toggleSort(gridCell.x, keys); - } else if (this.next) { - this.next.handleDoubleClick(grid, event); - } - }, + r = visibleRows[y]; - /** - * @memberOf ColumnSorting.prototype - * @desc * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseMove: function(grid, event) { - var y = event.gridCell.y; - if (this.isFixedRow(grid, event) && y < 1) { - this.cursor = 'pointer'; - } else { - this.cursor = null; - } - if (this.next) { - this.next.handleMouseMove(grid, event); - } - } + try { -}); + this._paintCell(gc, c, r); -module.exports = ColumnSorting; + //if (r === 9 && c === 2) { throw Error('She sells sea shells by the sea shore.'); } -},{"./Feature.js":59}],59:[function(require,module,exports){ -'use strict'; + } catch (e) { -var Base = require('extend-me').Base; + message = e && (e.message || e) || 'Unknown error.'; -/** - * @constructor - * @desc instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid. - */ -var Feature = Base.extend('Feature', { + console.error(message); - /** - * the next feature to be given a chance to handle incoming events - * @type {Feature} - * @default null - * @memberOf Feature.prototype - */ - next: null, + if (renderCellError) { + var rawGc = gc.gc || gc, // Don't log these canvas calls + errY = rowEdges[y], + errHeight = rowEdges[y + 1] - errY; - /** - * a temporary holding field for my next feature when I'm in a disconnected state - * @type {Feature} - * @default null - * @memberOf Feature.prototype - */ - detached: null, + rawGc.save(); // define clipping region + rawGc.beginPath(); + rawGc.rect(clipX, errY, clipWidth, errHeight); + rawGc.clip(); - /** - * the cursor I want to be displayed - * @type {string} - * @default null - * @memberOf Feature.prototype - */ - cursor: null, + renderCellError(rawGc, message, clipX, errY, clipWidth, errHeight); - /** - * the cell location where the cursor is currently - * @type {Point} - * @default null - * @memberOf Feature.prototype - */ - currentHoverCell: null, + rawGc.restore(); // discard clipping region + } - /** - * @memberOf Feature.prototype - * @desc set my next field, or if it's populated delegate to the feature in my next field - * @param {Feature} nextFeature - this is how we build the chain of responsibility - */ - setNext: function(nextFeature) { - if (this.next) { - this.next.setNext(nextFeature); - } else { - this.next = nextFeature; - this.detached = nextFeature; + } + } + + // Bottom totals rows... + for (y = -behavior.getDataModel().getBottomTotals().length; y; y++) { + this._paintCell(gc, c, y); + } + + gc.restore(); // Remove column's clip region (and anything else renderCellError() might have set) + } } - }, - /** - * @memberOf Feature.prototype - * @desc disconnect my child - */ - detachChain: function() { - this.next = null; + setNumberColumnWidth(gc, behavior, this.grid.getRowCount()); }, /** - * @memberOf Feature.prototype - * @desc reattach my child from the detached reference + * @memberOf Renderer.prototype + * @desc We opted to not paint borders for each cell as that was extremely expensive. Instead we draw gridlines here. Also we record the widths and heights for later. + * @param {CanvasRenderingContext2D} gc */ - attachChain: function() { - this.next = this.detached; - }, + paintGridlines: function(gc) { + var x, y, c, r = 0; - /** - * @memberOf Feature.prototype - * @desc handle mouse move down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseMove: function(grid, event) { - if (this.next) { - this.next.handleMouseMove(grid, event); - } - }, + var colWidths = this.getColumnEdges(); + var rowHeights = this.rowEdges; - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseExit: function(grid, event) { - if (this.next) { - this.next.handleMouseExit(grid, event); - } - }, + var viewWidth = colWidths[colWidths.length - 1]; + var viewHeight = this.getBounds().height; //rowHeights[rowHeights.length - 1]; - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseEnter: function(grid, event) { - if (this.next) { - this.next.handleMouseEnter(grid, event); - } - }, + var drawThemH = this.resolveProperty('gridLinesH'); + var drawThemV = this.resolveProperty('gridLinesV'); + var lineColor = this.resolveProperty('lineColor'); - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDown: function(grid, event) { - if (this.next) { - this.next.handleMouseDown(grid, event); - } - }, + gc.beginPath(); - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseUp: function(grid, event) { - if (this.next) { - this.next.handleMouseUp(grid, event); + if (drawThemV) { + for (c = 0; c < colWidths.length + 1; c++) { + x = colWidths[c] + 0.5; + gc.moveTo(x, 0); + gc.lineTo(x, viewHeight); + } } - }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleKeyDown: function(grid, event) { - if (this.next) { - this.next.handleKeyDown(grid, event); - } - }, + if (drawThemH) { + for (r = 0; r < rowHeights.length - 1; r++) { + y = rowHeights[r] + 0.5; + gc.moveTo(0, y); + gc.lineTo(viewWidth, y); + } - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleKeyUp: function(grid, event) { - if (this.next) { - this.next.handleKeyUp(grid, event); + // Bottom totals rows... + var behavior = this.grid.behavior, + rowHeight = behavior.getDefaultRowHeight(); + for (r = -behavior.getDataModel().getBottomTotals().length, y = this.getBounds().height; r; r++) { + y -= rowHeight; + gc.moveTo(0, y); + gc.lineTo(viewWidth, y); + } } - }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleWheelMoved: function(grid, event) { - if (this.next) { - this.next.handleWheelMoved(grid, event); - } - }, + gc.closePath(); - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleDoubleClick: function(grid, event) { - if (this.next) { - this.next.handleDoubleClick(grid, event); - } + gc.strokeStyle = lineColor; + gc.lineWidth = this.resolveProperty('lineWidth'); + gc.stroke(); }, /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf Renderer.prototype + * @param {CanvasRenderingContext2D} gc + * @param x + * @param y */ - handleHoldPulse: function(grid, event) { - if (this.next) { - this.next.handleHoldPulse(grid, event); + paintCell: function(gc, x, y) { + gc.moveTo(0, 0); + + var c = this.getVisibleColumns()[x], + r = this.getVisibleRows()[y]; + + if (c) { //something is being viewed at at the moment (otherwise returns undefined) + this._paintCell(gc, c, r); } }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleTap: function(grid, event) { - if (this.next) { - this.next.handleTap(grid, event); + _paintCell: function(gc, c, r) { + + var grid = this.grid, + behavior = grid.behavior, + baseProperties = behavior.getColumnProperties(c); + + if (baseProperties.isNull) { + return; } - }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleMouseDrag: function(grid, event) { - if (this.next) { - this.next.handleMouseDrag(grid, event); + var columnProperties = baseProperties, + + headerRowCount = behavior.getHeaderRowCount(), + isGridRow = r >= headerRowCount, + isFooterRow = r < 0, + isHeaderRow = !isGridRow && !isFooterRow, + isFilterRow = grid.isFilterRow(r), + + headerColumnCount = behavior.getHeaderColumnCount(), + isGridColumn = c >= headerColumnCount, + isShowRowNumbers = grid.isShowRowNumbers(), + isHierarchyColumn = grid.isHierarchyColumn(c), + + isRowSelected = grid.isRowSelected(r), + isColumnSelected = grid.isColumnSelected(c), + isCellSelected = grid.isCellSelected(c, r), + isCellSelectedInColumn = grid.isCellSelectedInColumn(c), + isCellSelectedInRow = grid.isCellSelectedInRow(r), + areAllRowsSelected = grid.areAllRowsSelected(), + cellProperties; + + if ((isShowRowNumbers && c === -1) || isHierarchyColumn) { + if (isRowSelected) { + cellProperties = Object.create(baseProperties.rowHeaderRowSelection); + cellProperties.isSelected = true; + } else { + cellProperties = Object.create(baseProperties.rowHeader); + cellProperties.isSelected = isCellSelectedInRow; + } + cellProperties.isUserDataArea = false; + } else if (isHeaderRow || isFooterRow) { + if (isFilterRow) { + cellProperties = Object.create(baseProperties.filterProperties); + cellProperties.isSelected = false; + } else if (isColumnSelected) { + cellProperties = Object.create(baseProperties.columnHeaderColumnSelection); + cellProperties.isSelected = true; + } else { + cellProperties = Object.create(baseProperties.columnHeader); + cellProperties.isSelected = isCellSelectedInColumn; + } + cellProperties.isUserDataArea = false; + } else if (isHierarchyColumn) { + cellProperties = Object.create(baseProperties.rowHeader); + cellProperties.isSelected = isCellSelectedInRow; + } else { + cellProperties = Object.create(baseProperties); + cellProperties.isSelected = isCellSelected || isRowSelected || isColumnSelected; + cellProperties.isUserDataArea = true; } - }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleContextMenu: function(grid, event) { - if (this.next) { - this.next.handleContextMenu(grid, event); + var rowNum = r - headerRowCount + 1; + + if (c === -1) { + if (r === 0) { // header label row gets "master" checkbox + cellProperties.value = [images.checkbox(areAllRowsSelected), '', null]; + } else if (isFilterRow) { // no checkbox but show filter icon + cellProperties.value = [images.filter(false), '', null]; + } else if (isHeaderRow || isFooterRow) { // no checkbox on "totals" rows + cellProperties.value = ''; + } else { + cellProperties.value = [images.checkbox(isRowSelected), rowNum, null]; + } + cellProperties.halign = 'right'; + } else { + cellProperties.value = grid.getValue(c, r); + cellProperties.halign = grid.getColumnAlignment(c); } - }, - /** - * @memberOf Feature.prototype - * @desc toggle the column picker - */ + cellProperties.isGridColumn = isGridColumn; + cellProperties.isGridRow = isGridRow; + cellProperties.isColumnHovered = grid.isColumnHovered(c) && isGridColumn; + cellProperties.isRowHovered = grid.isRowHovered(r) && isGridRow; + cellProperties.isCellHovered = grid.isHovered(c, r) && isGridColumn && isGridRow; + cellProperties.bounds = this._getBoundsOfCell(c, r); + cellProperties.isCellSelected = isCellSelected; + cellProperties.isRowSelected = isRowSelected; + cellProperties.isColumnSelected = isColumnSelected; + cellProperties.isInCurrentSelectionRectangle = grid.isInCurrentSelectionRectangle(c, r); - moveSingleSelect: function(grid, x, y) { - if (this.next) { - this.next.moveSingleSelect(grid, x, y); + if (grid.mouseDownState) { + var point = grid.mouseDownState.gridCell; + cellProperties.mouseDown = point.x === c && point.y === r; } - }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isFixedRow: function(grid, event) { - var gridCell = event.viewPoint; - var isFixed = gridCell.y < grid.getFixedRowCount(); - return isFixed; + cellProperties.x = c; + cellProperties.y = r; + + behavior.cellPropertiesPrePaintNotification(cellProperties); + + var cell = behavior.getCellRenderer(cellProperties, c, r); + var overrides = behavior.getCellProperties(c, r); + + //declarative cell properties + _(cellProperties).extendOwn(overrides); + + //allow the renderer to identify itself if it's a button + cellProperties.buttonCells = this.buttonCells; + var formatType = cellProperties.isUserDataArea ? cellProperties.format : 'default'; + cellProperties.formatter = grid.getFormatter(formatType); + cell.paint(gc, cellProperties); + + this.renderedColumnMinWidths[c] = Math.max(cellProperties.minWidth || 0, this.renderedColumnMinWidths[c]); + columnProperties.preferredWidth = this.renderedColumnMinWidths[c]; }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isFirstFixedRow: function(grid, event) { - var gridCell = event.viewPoint; - var isFixed = gridCell.y < 1; - return isFixed; + isViewableButton: function(c, r) { + var key = c + ',' + r; + return this.buttonCells[key] === true; }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isFixedColumn: function(grid, event) { - var gridCell = event.viewPoint; - var isFixed = gridCell.x < grid.getFixedColumnCount(); - return isFixed; + getRowNumbersWidth: function() { + var colEdges = this.getColumnEdges(); + if (colEdges.length === 0) { + return 0; + } + return colEdges[0]; }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isFirstFixedColumn: function(grid, event) { - var gridCell = event.viewPoint; - var edge = grid.isShowRowNumbers() ? 0 : 1; - var isFixed = gridCell.x < edge; - return isFixed; + startAnimator: function() { + var animate; + var self = this; + animate = function() { + self.animate(); + requestAnimationFrame(animate); + }; + requestAnimationFrame(animate); }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - isTopLeft: function(grid, event) { - var isTopLeft = this.isFixedRow(grid, event) && this.isFixedColumn(grid, event); - return isTopLeft; + animate: function() { + var ctx = this.getCanvas().canvasCTX; + ctx.beginPath(); + ctx.save(); + this.renderFocusCell(ctx); + ctx.restore(); + ctx.closePath(); }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - setCursor: function(grid) { - if (this.next) { - this.next.setCursor(grid); - } - if (this.cursor) { - grid.beCursor(this.cursor); - } + getBounds: function() { + return this.bounds; }, - /** - * @memberOf Feature.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - initializeOn: function(grid) { - if (this.next) { - this.next.initializeOn(grid); - } + setBounds: function(bounds) { + return (this.bounds = bounds); } }); -module.exports = Feature; +function setNumberColumnWidth(gc, behavior, maxRow) { + var columnProperties = behavior.getColumnProperties(-1), + cellProperties = columnProperties.rowHeader, + icon = images.checked; + + gc.font = cellProperties.font; + + columnProperties.preferredWidth = icon.width + 7 + cellProperties.getTextWidth(gc, maxRow + 1); +} -},{"extend-me":4}],60:[function(require,module,exports){ +module.exports = Renderer; + +},{"../../images/index":3,"./Base":66,"object-iterators":21}],72:[function(require,module,exports){ 'use strict'; -var Feature = require('./Feature.js'); +var RangeSelectionModel = require('sparse-boolean-array'); /** + * * @constructor + * @desc We represent selections as a list of rectangles because large areas can be represented and tested against quickly with a minimal amount of memory usage. Also we need to maintain the selection rectangles flattened counter parts so we can test for single dimension contains. This is how we know to highlight the fixed regions on the edges of the grid. */ -var Filters = Feature.extend('Filters', { - - alias: 'Filters', - handleDoubleClick: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleDoubleClick(grid, event); - } - }, +function SelectionModel(grid) { - handleTap: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleTap(grid, event); - } - }, + this.grid = grid; /** - * @memberOf CellEditing.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @name selections + * @type {Rectangle[]} + * @summary The selection rectangles. + * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}. + * @memberOf SelectionModel.prototype */ - handleHoldPulse: function(grid, event) { - var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick'); - if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) { - grid._activateEditor(event); - } else if (this.next) { - this.next.handleHoldPulse(grid, event); - } - }, - - checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) { - var isFilterRow = grid.isFilterRow(event.gridCell.y); - var activateEditor = isDoubleClickEditorActivation && isFilterRow; - return activateEditor; - } - -}); - -module.exports = Filters; - -},{"./Feature.js":59}],61:[function(require,module,exports){ -'use strict'; - -var Feature = require('./Feature.js'); - -var commands = { - PAGEDOWN: function(grid) { grid.pageDown(); }, - PAGEUP: function(grid) { grid.pageUp(); }, - PAGELEFT: function(grid) { grid.pageLeft(); }, - PAGERIGHT: function(grid) { grid.pageRight(); } -}; + this.selections = []; -/** - * @constructor - */ -var KeyPaging = Feature.extend('KeyPaging', { + /** + * @name flattenedX + * @type {Rectangle[]} + * @summary The selection rectangles flattened in the horizontal direction (no width). + * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}. + * @memberOf SelectionModel.prototype + */ + this.flattenedX = []; - alias: 'KeyPaging', + /** + * @name flattenedY + * @type {Rectangle[]} + * @summary The selection rectangles flattened in the vertical direction (no height). + * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}. + * @memberOf SelectionModel.prototype + */ + this.flattenedY = []; /** - * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details - * @memberOf KeyPaging.prototype + * @name rowSelectionModel + * @type {RangeSelectionModel} + * @summary The selection rectangles. + * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}. + * @memberOf SelectionModel.prototype */ - handleKeyDown: function(grid, event) { - var detail = event.detail.char; - var func = commands[detail]; - if (func) { - func(grid); - } else if (this.next) { - this.next.handleKeyDown(grid, event); - } - } - -}); + this.rowSelectionModel = new RangeSelectionModel(); -module.exports = KeyPaging; + /** + * @name columnSelectionModel + * @type {RangeSelectionModel} + * @summary The selection rectangles. + * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}. + * @memberOf SelectionModel.prototype + */ + this.columnSelectionModel = new RangeSelectionModel(); -},{"./Feature.js":59}],62:[function(require,module,exports){ -'use strict'; + this.setLastSelectionType(''); +} -var Feature = require('./Feature.js'); +SelectionModel.prototype = { -/** - * @constructor - */ -var OnHover = Feature.extend('OnHover', { + constructor: SelectionModel.prototype.constructor, - alias: 'OnHover', + /** + * @type {boolean} + * @memberOf SelectionModel.prototype + */ + allRowsSelected: false, /** - * @desc Hhandle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details - * @memberOf OnHover.prototype + * @memberOf SelectionModel.prototype + * @returns {*} */ - handleMouseMove: function(grid, event) { - var currentHoverCell = grid.getHoverCell(); - if (!event.gridCell.equals(currentHoverCell)) { - if (currentHoverCell) { - this.handleMouseExit(grid, currentHoverCell); - } - this.handleMouseEnter(grid, event); - grid.setHoverCell(event.gridCell); - } else { - if (this.next) { - this.next.handleMouseMove(grid, event); - } - } - } + getLastSelection: function() { + var sels = this.selections; + var sel = sels[sels.length - 1]; + return sel; + }, -}); + /** + * @memberOf SelectionModel.prototype + * @returns {*} + */ + getLastSelectionType: function() { + return this.lastSelectionType; + }, -module.exports = OnHover; + /** + * @param type + * @memberOf SelectionModel.prototype + */ + setLastSelectionType: function(type) { + this.lastSelectionType = type; + }, -},{"./Feature.js":59}],63:[function(require,module,exports){ -'use strict'; + /** + * @memberOf SelectionModel.prototype + * @description Select the region described by the given coordinates. + * + * @param {number} ox - origin x coordinate + * @param {number} oy - origin y coordinate + * @param {number} ex - extent x coordinate + * @param {number} ey - extent y coordinate + */ + select: function(ox, oy, ex, ey) { + var newSelection = this.grid.newRectangle(ox, oy, ex, ey); + this.selections.push(newSelection); + this.flattenedX.push(newSelection.flattenXAt(0)); + this.flattenedY.push(newSelection.flattenYAt(0)); + this.setLastSelectionType('cell'); + this.grid.selectionChanged(); + }, -var ColumnResizing = require('./ColumnResizing'); + /** + * @memberOf SelectionModel.prototype + * @param {number} ox - origin x coordinate + * @param {number} oy - origin y coordinate + * @param {number} ex - extent x coordinate + * @param {number} ey - extent y coordinate + */ + toggleSelect: function(ox, oy, ex, ey) { -/** - * @constructor - */ -var RowResizing = ColumnResizing.extend('RowResizing', { + var selected, index; - alias: 'RowResizing', + selected = this.selections.find(function(selection, idx) { + index = idx; + return ( + selection.origin.x === ox && selection.origin.y === oy && + selection.extent.x === ex && selection.extent.y === ey + ); + }); - /** - * the index of the row/column we are dragging - * @type {number} - * @default -1 - * @memberOf RowResizing.prototype - */ - dragArea: -1, + if (selected) { + this.selections.splice(index, 1); + this.flattenedX.splice(index, 1); + this.flattenedY.splice(index, 1); + this.grid.selectionChanged(); + } else { + this.select(ox, oy, ex, ey); + } + }, /** - * the pixel location of the where the drag was initiated - * @type {number} - * @default -1 - * @memberOf RowResizing.prototype + * @memberOf SelectionModel.prototype + * @desc Remove the last selection that was created. */ - dragStart: -1, + clearMostRecentSelection: function(dontClearRowSelections) { + dontClearRowSelections = dontClearRowSelections === true; + if (!dontClearRowSelections) { + this.setAllRowsSelected(false); + } + if (this.selections.length) { --this.selections.length; } + if (this.flattenedX.length) { --this.flattenedX.length; } + if (this.flattenedY.length) { --this.flattenedY.length; } + //this.getGrid().selectionChanged(); + }, /** - * the starting width/height of the row/column we are dragging - * @type {number} - * @default -1 - * @memberOf RowResizing.prototype + * @memberOf SelectionModel.prototype */ - dragAreaStartingSize: -1, + clearMostRecentColumnSelection: function() { + this.columnSelectionModel.clearMostRecentSelection(); + this.setLastSelectionType('column'); + }, /** - * @memberOf RowResizing.prototype - * @desc get the mouse x,y coordinate - * @returns {number} - * @param {MouseEvent} event - the mouse event to query + * @memberOf SelectionModel.prototype */ - getMouseValue: function(event) { - return event.primitiveEvent.detail.mouse.y; + clearMostRecentRowSelection: function() { + this.rowSelectionModel.clearMostRecentSelection(); + this.setLastSelectionType('row'); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc get the grid cell x,y coordinate - * @returns {number} - * @param {Point} gridCell + * @memberOf SelectionModel.prototype */ - getGridCellValue: function(gridCell) { - return gridCell.x; + clearRowSelection: function() { + this.rowSelectionModel.clear(); + this.setLastSelectionType('row'); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc return the grids x,y scroll value - * @returns {number} - * @param {Hypergrid} grid + * @memberOf SelectionModel.prototype + * @returns {*} */ - getScrollValue: function(grid) { - return grid.getVScrollValue(); + getSelections: function() { + return this.selections; }, /** - * @function - * @memberOf RowResizing.prototype - * @desc return the width/height of the row/column of interest - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest + * @memberOf SelectionModel.prototype + * @returns {boolean} There are active selection(s). */ - getAreaSize: function(grid, index) { - return grid.getRowHeight(index); + hasSelections: function() { + return this.selections.length !== 0; }, /** - * @function - * @memberOf RowResizing.prototype - * @desc set the width/height of the row/column at index - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest - * @param {number} value - the width/height to set to + * @memberOf SelectionModel.prototype + * @returns {boolean} */ - setAreaSize: function(grid, index, value) { - grid.setRowHeight(index, value); + hasRowSelections: function() { + return !this.rowSelectionModel.isEmpty(); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc returns the index of which divider I'm over - * @returns {number} - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype + * @returns {boolean} */ - overAreaDivider: function(grid, event) { - return grid.overRowDivider(event); + hasColumnSelections: function() { + return !this.columnSelectionModel.isEmpty(); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc am I over the column/row area - * @returns {boolean} - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype + * @return {boolean} Selection covers a specific column. + * @param {number} y */ - isFirstFixedOtherArea: function(grid, event) { - return this.isFirstFixedColumn(grid, event); + isCellSelectedInRow: function(y) { + return this._isCellSelected(this.flattenedX, 0, y); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc return the cursor name - * @returns {string} + * @memberOf SelectionModel.prototype + * @returns Selection covers a specific row. + * @param {number} x */ - getCursorName: function() { - return 'row-resize'; + isCellSelectedInColumn: function(x) { + return this._isCellSelected(this.flattenedY, x, 0); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc return the recently rendered area's width/height - * @returns {number} - * @param {Hypergrid} grid - * @param {number} index - the row/column index of interest + * @memberOf SelectionModel.prototype + * @summary Selection query function. + * @returns {boolean} The given cell is selected (part of an active selection). + * @param {Rectangle[]} selections - Selection rectangles to search through. + * @param {number} x + * @param {number} y */ - getPreviousAbsoluteSize: function(grid, index) { - return grid.getRenderedHeight(index); + isSelected: function(x, y) { + return ( + this.isColumnSelected(x) || + this.isRowSelected(y) || + this._isCellSelected(this.selections, x, y) + ); }, /** - * @function - * @memberOf RowResizing.prototype - * @desc return the fixed area rows/columns count - * @returns {number} - * @param {Hypergrid} grid + * @memberOf SelectionModel.prototype + * @param x + * @param y + * @returns {*} */ - getOtherFixedAreaCount: function(grid) { - return grid.getFixedColumnCount(); + isCellSelected: function(x, y) { + return this._isCellSelected(this.selections, x, y); }, /** - * - * @param {Hypergrid} grid - * @returns {number} - * @default -2 - * @memberOf ColumnResizing.prototype + * @memberOf SelectionModel.prototype + * @param selections + * @param x + * @param y + * @returns {boolean} + * @private */ - getFixedAreaCount: function(grid) { - return grid.getFixedRowCount() + grid.getHeaderRowCount(); + _isCellSelected: function(selections, x, y) { + var self = this; + return !!selections.find(function(selection) { + return self.rectangleContains(selection, x, y); + }); }, /** + * @memberOf SelectionModel.prototype + * @desc empty out all our state * - * @param {Hypergrid} grid - * @returns {boolean} - * @default -2 - * @memberOf ColumnResizing.prototype */ - isEnabled: function(grid) { - return grid.isRowResizeable(); - } - -}); - -module.exports = RowResizing; - -},{"./ColumnResizing":56}],64:[function(require,module,exports){ -'use strict'; - -var Feature = require('./Feature.js'); - -/** - * @constructor - */ -var RowSelection = Feature.extend('RowSelection', { - - alias: 'RowSelection', + clear: function(dontClearRowSelections) { + dontClearRowSelections = dontClearRowSelections === true; + this.selections.length = 0; + this.flattenedX.length = 0; + this.flattenedY.length = 0; + this.columnSelectionModel.clear(); + if (!dontClearRowSelections) { + this.setAllRowsSelected(false); + this.rowSelectionModel.clear(); + } + //this.getGrid().selectionChanged(); + }, /** - * The pixel location of the mouse pointer during a drag operation. - * @type {Point} - * @default null - * @memberOf RowSelection.prototype + * @memberOf SelectionModel.prototype + * @param {number} ox - origin x coordinate + * @param {number} oy - origin y coordinate + * @param {number} ex - extent x coordinate + * @param {number} ey - extent y coordinate + * @returns {boolean} */ - currentDrag: null, + isRectangleSelected: function(ox, oy, ex, ey) { + return !!this.selections.find(function(selection) { + return ( + selection.origin.x === ox && selection.origin.y === oy && + selection.extent.x === ex && selection.extent.y === ey + ); + }); + }, /** - * The cell coordinates of the where the mouse pointer is during a drag operation. - * @type {Object} - * @default null - * @memberOf RowSelection.prototype + * @memberOf SelectionModel.prototype + * @param x + * @returns {*} */ - lastDragCell: null, + isColumnSelected: function(x) { + return this.columnSelectionModel.isSelected(x); + }, /** - * a millisecond value representing the previous time an autoscroll started - * @type {number} - * @default 0 - * @memberOf RowSelection.prototype + * @memberOf SelectionModel.prototype + * @param y + * @returns {boolean|*} */ - sbLastAuto: 0, + isRowSelected: function(y) { + return this.allRowsSelected || this.rowSelectionModel.isSelected(y); + }, /** - * a millisecond value representing the time the current autoscroll started - * @type {number} - * @default 0 - * @memberOf RowSelection.prototype + * @memberOf SelectionModel.prototype + * @param x1 + * @param x2 */ - sbAutoStart: 0, - - dragArmed: false, + selectColumn: function(x1, x2) { + this.columnSelectionModel.select(x1, x2); + this.setLastSelectionType('column'); + }, /** - * @memberOf RowSelection.prototype - * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype */ - handleMouseUp: function(grid, event) { - if (this.dragArmed) { - this.dragArmed = false; - //global row selection - if (event.gridCell.x === -1 && event.gridCell.y === 0) { - grid.toggleSelectAllRows(); - } - grid.fireSyntheticRowSelectionChangedEvent(); - } else if (this.dragging) { - this.dragging = false; - grid.fireSyntheticRowSelectionChangedEvent(); - } else if (this.next) { - this.next.handleMouseUp(grid, event); - } + selectAllRows: function() { + this.clear(); + this.setAllRowsSelected(true); }, /** - * @memberOf RowSelection.prototype - * @desc * @desc Handle this event down the feature chain of responsibility. - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype + * @returns {boolean} */ - handleMouseDown: function(grid, event) { - - var isRightClick = event.primitiveEvent.detail.isRightClick; - var cell = event.gridCell; - var viewCell = event.viewPoint; - var dx = cell.x; - var dy = cell.y; - - - var isHeader = grid.isShowRowNumbers() && dx < 0; - - if (!grid.isRowSelection() || isRightClick || !isHeader) { - if (this.next) { - this.next.handleMouseDown(grid, event); - } - } else { - - var numFixedRows = grid.getFixedRowCount(); - - //if we are in the fixed area do not apply the scroll values - //check both x and y values independently - if (viewCell.y < numFixedRows) { - dy = viewCell.y; - } - var dCell = grid.newPoint(0, dy); + setAllRowsSelected: function(isIt) { + this.allRowsSelected = isIt; + }, - var primEvent = event.primitiveEvent; - var keys = primEvent.detail.keys; - this.dragArmed = true; - this.extendSelection(grid, dCell, keys); - } + areAllRowsSelected: function() { + return this.allRowsSelected; }, /** - * @memberOf RowSelection.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype + * @param y1 + * @param y2 */ - handleMouseDrag: function(grid, event) { - var isRightClick = event.primitiveEvent.detail.isRightClick; - - if (!this.dragArmed || !grid.isRowSelection() || isRightClick) { - if (this.next) { - this.next.handleMouseDrag(grid, event); - } - } else { - this.dragging = true; - var numFixedRows = grid.getFixedRowCount(); - - var cell = event.gridCell; - var viewCell = event.viewPoint; - //var dx = cell.x; - var dy = cell.y; - - //if we are in the fixed area do not apply the scroll values - //check both x and y values independently - if (viewCell.y < numFixedRows) { - dy = viewCell.y; - } - - var dCell = grid.newPoint(0, dy); + selectRow: function(y1, y2) { + this.rowSelectionModel.select(y1, y2); + this.setLastSelectionType('row'); + }, - var primEvent = event.primitiveEvent; - this.currentDrag = primEvent.detail.mouse; - this.lastDragCell = dCell; + /** + * @memberOf SelectionModel.prototype + * @param x1 + * @param x2 + */ + deselectColumn: function(x1, x2) { + this.columnSelectionModel.deselect(x1, x2); + this.setLastSelectionType('column'); + }, - this.checkDragScroll(grid, this.currentDrag); - this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys); - } + /** + * @memberOf SelectionModel.prototype + * @param y1 + * @param y2 + */ + deselectRow: function(y1, y2) { + this.rowSelectionModel.deselect(y1, y2); + this.setLastSelectionType('row'); }, /** - * @memberOf RowSelection.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details + * @memberOf SelectionModel.prototype + * @returns {*} */ - handleKeyDown: function(grid, event) { - if (grid.getLastSelectionType() !== 'row') { - if (this.next) { - this.next.handleKeyDown(grid, event); + getSelectedRows: function() { + if (this.areAllRowsSelected()) { + var headerRows = this.grid.getHeaderRowCount(); + var rowCount = this.grid.getRowCount() - headerRows; + var result = new Array(rowCount); + for (var i = 0; i < rowCount; i++) { + result[i] = i + headerRows; } - return; - } - var command = 'handle' + event.detail.char; - if (this[command]) { - this[command].call(this, grid, event.detail); + return result; } + return this.rowSelectionModel.getSelections(); }, /** - * @memberOf RowSelection.prototype - * @desc Handle a mousedrag selection - * @param {Hypergrid} grid - * @param {Object} mouse - the event details - * @param {Array} keys - array of the keys that are currently pressed down + * @memberOf SelectionModel.prototype + * @returns {*|Array.Array.number} */ - handleMouseDragCellSelection: function(grid, gridCell, keys) { - - //var behavior = grid.getBehavior(); - var y = gridCell.y; - // var previousDragExtent = grid.getDragExtent(); - var mouseDown = grid.getMouseDown(); - - var newY = y - mouseDown.y; - //var newY = y - mouseDown.y; - - // if (previousDragExtent.x === newX && previousDragExtent.y === newY) { - // return; - // } - - grid.clearMostRecentRowSelection(); - - grid.selectRow(mouseDown.y, y); - grid.setDragExtent(grid.newPoint(0, newY)); + getSelectedColumns: function() { + return this.columnSelectionModel.getSelections(); + }, - grid.repaint(); + /** + * @memberOf SelectionModel.prototype + * @returns {boolean} + */ + isColumnOrRowSelected: function() { + return !this.columnSelectionModel.isEmpty() || !this.rowSelectionModel.isEmpty(); }, /** - * @memberOf RowSelection.prototype - * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above) - * @param {Hypergrid} grid - * @param {Object} mouse - the event details + * @memberOf SelectionModel.prototype + * @returns {Array} */ - checkDragScroll: function(grid, mouse) { - if (!grid.resolveProperty('scrollingEnabled')) { - return; - } - var b = grid.getDataBounds(); - var inside = b.contains(mouse); - if (inside) { - if (grid.isScrollingNow()) { - grid.setScrollingNow(false); + getFlattenedYs: function() { + var result = []; + var set = {}; + this.selections.forEach(function(selection) { + var top = selection.origin.y; + var size = selection.extent.y + 1; + for (var r = 0; r < size; r++) { + var ti = r + top; + if (!set[ti]) { + result.push(ti); + set[ti] = true; + } } - } else if (!grid.isScrollingNow()) { - grid.setScrollingNow(true); - this.scrollDrag(grid); - } + }); + result.sort(function(x, y) { + return x - y; + }); + return result; }, /** - * @memberOf RowSelection.prototype - * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly - * @param {Hypergrid} grid + * @memberOf SelectionModel.prototype + * @param offset */ - scrollDrag: function(grid) { - if (!grid.isScrollingNow()) { - return; - } - - var lastDragCell = this.lastDragCell; - var b = grid.getDataBounds(); - var xOffset = 0; - var yOffset = 0; - - var numFixedColumns = grid.getFixedColumnCount(); - var numFixedRows = grid.getFixedRowCount(); - - var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns; - var dragEndInFixedAreaY = lastDragCell.y < numFixedRows; + selectRowsFromCells: function(offset, dontClearRowSelections) { + offset = offset || 0; + dontClearRowSelections = dontClearRowSelections === true; - if (this.currentDrag.y < b.origin.y) { - yOffset = -1; - } + var sm = this.rowSelectionModel; - if (this.currentDrag.y > b.origin.y + b.extent.y) { - yOffset = 1; + if (!dontClearRowSelections) { + this.setAllRowsSelected(false); + sm.clear(); } - var dragCellOffsetX = xOffset; - var dragCellOffsetY = yOffset; + this.selections.forEach(function(selection) { + var top = selection.origin.y, + extent = selection.extent.y; + top += offset; + sm.select(top, top + extent); + }); + }, - if (dragEndInFixedAreaX) { - dragCellOffsetX = 0; - } + /** + * @memberOf SelectionModel.prototype + * @param offset + */ + selectColumnsFromCells: function(offset) { + offset = offset || 0; - if (dragEndInFixedAreaY) { - dragCellOffsetY = 0; - } + var sm = this.columnSelectionModel; + sm.clear(); - this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY); - grid.scrollBy(xOffset, yOffset); - this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection - grid.repaint(); - setTimeout(this.scrollDrag.bind(this, grid), 25); + this.selections.forEach(function(selection) { + var left = selection.origin.x, + extent = selection.extent.x; + left += offset; + sm.select(left, left + extent); + }); }, /** - * @memberOf RowSelection.prototype - * @desc extend a selection or create one if there isnt yet - * @param {Hypergrid} grid - * @param {Object} gridCell - the event details - * @param {Array} keys - array of the keys that are currently pressed down + * @memberOf SelectionModel.prototype + * @param x + * @param y + * @returns {*} */ - extendSelection: function(grid, gridCell, keys) { - grid.stopEditing(); - //var hasCTRL = keys.indexOf('CTRL') !== -1; - var hasSHIFT = keys.indexOf('SHIFT') !== -1; + isInCurrentSelectionRectangle: function(x, y) { + var last = this.selections[this.selections.length - 1]; + return last && this.rectangleContains(last, x, y); + }, - var mousePoint = grid.getMouseDown(); - var x = gridCell.x; // - numFixedColumns + scrollLeft; - var y = gridCell.y; // - numFixedRows + scrollTop; + /** + * @memberOf SelectionModel.prototype + * @param rect + * @param x + * @param y + * @returns {boolean} + */ + rectangleContains: function(rect, x, y) { //TODO: explore why this works and contains on rectanglular does not + var minX = rect.origin.x; + var minY = rect.origin.y; + var maxX = minX + rect.extent.x; + var maxY = minY + rect.extent.y; - //were outside of the grid do nothing - if (x < 0 || y < 0) { - return; + if (rect.extent.x < 0) { + minX = maxX; + maxX = rect.origin.x; } - if (hasSHIFT) { - grid.clearMostRecentRowSelection(); - grid.selectRow(y, mousePoint.y); - grid.setDragExtent(grid.newPoint(0, y - mousePoint.y)); - } else { - grid.toggleSelectRow(y, keys); - grid.setMouseDown(grid.newPoint(x, y)); - grid.setDragExtent(grid.newPoint(0, 0)); + if (rect.extent.y < 0) { + minY = maxY; + maxY = rect.origin.y; } - grid.repaint(); - }, + var result = + x >= minX && + y >= minY && + x <= maxX && + y <= maxY; - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - */ - handleDOWNSHIFT: function(grid) { - this.moveShiftSelect(grid, 1); - }, + return result; + } +}; - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleUPSHIFT: function(grid) { - this.moveShiftSelect(grid, -1); - }, +module.exports = SelectionModel; - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleLEFTSHIFT: function(grid) {}, +},{"sparse-boolean-array":24}],73:[function(require,module,exports){ +/* eslint-env browser */ - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleRIGHTSHIFT: function(grid) {}, +'use strict'; + +var Base = require('./Base'); + + +var ANIMATION_TIME = 500, + TRANSITION = ANIMATION_TIME + 'ms ease-in'; + +/** @constructor + * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid. + * + * See {@link TableDialog#initialize|initialize} which is called by the constructor. + */ +var TableDialog = Base.extend('TableDialog', { - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleDOWN: function(grid) { - this.moveSingleSelect(grid, 1); + initialize: function(grid) { + this.grid = grid; + this.initializeOverlaySurface(); + this.openNow = false; }, /** - * @memberOf RowSelection.prototype - * @desc handle this event + * @memberOf Overlay.prototype + * @desc returns true if the overlay is open + * @returns {boolean} * @param {Hypergrid} grid - * @param {Object} event - the event details */ - handleUP: function(grid) { - this.moveSingleSelect(grid, -1); + isOpen: function() { + return this.openNow; }, - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleLEFT: function(grid) {}, - - /** - * @memberOf RowSelection.prototype - * @desc handle this event - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleRIGHT: function(grid) { + open: function() { + if (this.isOpen()) { + return; + } - var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); - var maxColumns = grid.getColumnCount() - 1; + this.openNow = true; + var self = this; + this.overlay.style.backgroundColor = this.grid.resolveProperty('backgroundColor'); - var newX = grid.getHeaderColumnCount() + grid.getHScrollValue(); - var newY = mouseCorner.y; + this.overlay.style.top = this.overlay.style.bottom = this.overlay.style.right = this.overlay.style.left = 0; - newX = Math.min(maxColumns, newX); + self.overlay.style.webkitTransition = ''; - grid.clearSelections(); - grid.select(newX, newY, 0, 0); - grid.setMouseDown(grid.newPoint(newX, newY)); - grid.setDragExtent(grid.newPoint(0, 0)); + this.overlay.style.margin = '15px 35px 35px 15px'; + this.overlay.style.opacity = 0; + this.overlay.style.zIndex = 100; - grid.repaint(); - }, + this.closeTransition = function() { + this.overlay.style.opacity = 0; + }; - /** - * @memberOf RowSelection.prototype - * @desc If we are holding down the same navigation key, accelerate the increment we scroll - * #### returns: integer - */ - getAutoScrollAcceleration: function() { - var count = 1; - var elapsed = this.getAutoScrollDuration() / 2000; - count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed)); - return count; - }, + if (!this._closer) { + this._closer = function(e) { + var key = self.getCharFor(e.keyCode).toLowerCase(); + var keys = self.grid.resolveProperty('editorActivationKeys'); + if (keys.indexOf(key) > -1 || e.keyCode === 27) { + e.preventDefault(); + self.close(); + } + }; + } + requestAnimationFrame(function() { + self.overlay.style.webkitTransition = 'opacity ' + ANIMATION_TIME + 'ms ease-in'; + requestAnimationFrame(function() { + document.addEventListener('keydown', self._closer, false); + self.overlay.style.opacity = 0.95; + }); + }); - /** - * @memberOf RowSelection.prototype - * @desc set the start time to right now when we initiate an auto scroll - */ - setAutoScrollStartTime: function() { - this.sbAutoStart = Date.now(); + setTimeout(function() { + self.overlay.focus(); + }, 100); }, - /** - * @memberOf RowSelection.prototype - * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time + * @memberOf Overlay.prototype + * @desc open the overlay + * #### returns: type + * @param {Hypergrid} grid */ - pingAutoScroll: function() { - var now = Date.now(); - if (now - this.sbLastAuto > 500) { - this.setAutoScrollStartTime(); + openFrom: function(rect) { + if (this.isOpen()) { + return; } - this.sbLastAuto = Date.now(); - }, + this.openNow = true; + var self = this; + var style = this.overlay.style; + style.backgroundColor = this.grid.resolveProperty('backgroundColor'); - /** - * @memberOf RowSelection.prototype - * @desc answer how long we have been auto scrolling - * #### returns: integer - */ - getAutoScrollDuration: function() { - if (Date.now() - this.sbLastAuto > 500) { - return 0; + var bounds = this.grid.div.getBoundingClientRect(), + margins = rect.y + 'px ' + + (bounds.width - (rect.x + rect.width)) + 'px ' + + (bounds.height - (rect.y + rect.height)) + 'px ' + + rect.x + 'px'; + + style.webkitTransition = ''; + + style.top = style.right = style.bottom = style.left = 0; + + style.margin = margins; + style.zIndex = 100; + style.opacity = 1; + + this.closeTransition = function() { + style.margin = margins; + }; + + if (!this._closer) { + this._closer = function(e) { + var key = self.getCharFor(e.keyCode).toLowerCase(); + var keys = self.grid.resolveProperty('editorActivationKeys'); + if (keys.indexOf(key) > -1 || e.keyCode === 27) { + e.preventDefault(); + self.close(); + } + }; } - return Date.now() - this.sbAutoStart; + + //grid.setFocusable(false); + requestAnimationFrame(function() { + document.addEventListener('keydown', self._closer, false); + requestAnimationFrame(function() { + requestAnimationFrame(function() { + style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION; + style.margin = '15px 35px 35px 15px'; + }); + }); + }); + setTimeout(function() { + self.overlay.focus(); + }, 100); }, /** - * @memberOf RowSelection.prototype - * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary. + * @memberOf Overlay.prototype + * @desc close the overlay * @param {Hypergrid} grid - * @param {number} offsetX - x coordinate to start at - * @param {number} offsetY - y coordinate to start at */ - moveShiftSelect: function(grid, offsetY) { - - var maxRows = grid.getRowCount() - 1; - - var maxViewableRows = grid.getVisibleRows() - 1; + close: function() { + //grid.setFocusable(true); + this.openNow = false; + document.removeEventListener('keydown', this._closer, false); - if (!grid.resolveProperty('scrollingEnabled')) { - maxRows = Math.min(maxRows, maxViewableRows); - } + var self = this; - var origin = grid.getMouseDown(); - var extent = grid.getDragExtent(); + requestAnimationFrame(function() { + self.closeTransition(); + }); - var newY = extent.y + offsetY; - //var newY = grid.getRowCount(); + setTimeout(function() { + self.clear(); + self.overlay.style.zIndex = -1000; + if (self.onClose) { + self.onClose(); + self.onClose = undefined; + } + self.grid.takeFocus(); + }, ANIMATION_TIME); + }, - newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY)); + /** + * @memberOf Overlay.prototype + * @desc initialize the overlay surface into the grid + * #### returns: type + * @param {Hypergrid} grid + */ + initializeOverlaySurface: function() { + this.overlay = document.createElement('div'); + this.overlay.setAttribute('tabindex', 0); + this.overlay.addEventListener('wheel', function(evt) { evt.stopPropagation(); }); - grid.clearMostRecentRowSelection(); - grid.selectRow(origin.y, origin.y + newY); + var style = this.overlay.style; + style.outline = 'none'; + style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)'; + style.position = 'absolute'; - grid.setDragExtent(grid.newPoint(0, newY)); + style.margin = 0; + style.overflow = 'hidden'; - if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) { - this.pingAutoScroll(); - } + //style.display = 'none'; - grid.fireSyntheticRowSelectionChangedEvent(); - grid.repaint(); + //style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION; + style.opacity = 0; + style.zIndex = 10; + this.grid.div.appendChild(this.overlay); + //document.body.appendChild(this.overlay); }, /** - * @memberOf RowSelection.prototype - * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent. + * @memberOf Overlay.prototype + * @desc get a human readable description of the key pressed from it's integer representation + * @returns {string} * @param {Hypergrid} grid - * @param {number} offsetX - x coordinate to start at - * @param {number} offsetY - y coordinate to start at + * @param {number} integer - the integer we want the char for */ - moveSingleSelect: function(grid, offsetY) { + getCharFor: function(integer) { + var charMap = this.grid.getCanvas().getCharMap(); + return charMap[integer][0]; + }, + + clear: function() { + this.overlay.innerHTML = ''; + }, - var maxRows = grid.getRowCount() - 1; + querySelector: function(selector) { + var elements = this.overlay.querySelector(selector); + return elements; + }, - var maxViewableRows = grid.getVisibleRowsCount() - 1; + getAnimationTime: function() { + return ANIMATION_TIME; + } +}); - if (!grid.resolveProperty('scrollingEnabled')) { - maxRows = Math.min(maxRows, maxViewableRows); - } +module.exports = TableDialog; - var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent()); +},{"./Base":66}],74:[function(require,module,exports){ +'use strict'; - var newY = mouseCorner.y + offsetY; - //var newY = grid.getRowCount(); +// console.warn polyfill as needed +// used for deprecation warnings +if (!console.warn) { + console.warn = function() { + console.log.apply(console, ['WARNING:'].concat(Array.prototype.slice.call(arguments))); + }; +} - newY = Math.min(maxRows, Math.max(0, newY)); +var deprecated = function(dotProps, options) { + var chain = dotProps.split('.'), + method = chain[chain.length - 1], + asOfVersion = options && options.asOfVersion, + result = this, + warning; - grid.clearSelections(); - grid.selectRow(newY); - grid.setMouseDown(grid.newPoint(0, newY)); - grid.setDragExtent(grid.newPoint(0, 0)); + method = 'get' + method[0].toUpperCase() + method.substr(1); - if (grid.insureModelRowIsVisible(newY, offsetY)) { - this.pingAutoScroll(); - } + warning = '.' + method + '() method is deprecated'; - grid.fireSyntheticRowSelectionChangedEvent(); - grid.repaint(); + if (asOfVersion) { + warning += ' as of v' + options.asOfVersion; + } - }, + warning += '. Use .' + dotProps; - isSingleRowSelection: function() { - return true; + if (dotProps[dotProps.length - 1] !== ')') { + warning += ' property'; } -}); + warning += ' instead. (Will be removed in a future release.)'; -module.exports = RowSelection; + console.warn(warning); + + chain.forEach(function(link) { + result = result[link]; + }); -},{"./Feature.js":59}],65:[function(require,module,exports){ + return result; +}; + +module.exports = deprecated; + +},{}],75:[function(require,module,exports){ 'use strict'; -var Feature = require('./Feature.js'); +var images = require('../../images/index'); /** - * @constructor + * @summary Writes error message into cell. + * + * @desc This funciton is guaranteed to be called as follows: + * + * ```javascript + * gc.save(); + * gc.beginPath(); + * gc.rect(x, y, width, height); + * gc.clip(); + * renderCellError(gc, message, x, y, width, height); + * gc.restore(); + * ``` + * + * Before doing anything else, this function should clear the cell by setting `gc.fillStyle` and calling `gc.fill()`. + * + * @param {CanvasRenderingContext2D} gc + * @param {string} message + * @param {number} x + * @param {number} y + * @param {number} width + * @param {number} height */ -var ThumbwheelScrolling = Feature.extend('ThumbwheelScrolling', { - - alias: 'ThumbwheelScrolling', +function renderCellError(gc, message, x, y, width, height) { - /** - * @memberOf ThumbwheelScrolling.prototype - * @desc handle this event down the feature chain of responsibility - * @param {Hypergrid} grid - * @param {Object} event - the event details - */ - handleWheelMoved: function(grid, e) { - if (!grid.resolveProperty('scrollingEnabled')) { - return; - } - var primEvent = e.primitiveEvent; - var deltaY = primEvent.wheelDeltaY || -primEvent.deltaY; - var deltaX = primEvent.wheelDeltaX || -primEvent.deltaX; - if (deltaY > 0) { - grid.scrollBy(0, -1); - } else if (deltaY < -0) { - grid.scrollBy(0, 1); - } else if (deltaX > 0) { - grid.scrollBy(-1, 0); - } else if (deltaX < -0) { - grid.scrollBy(1, 0); - } - } + // clear the cell + // (this makes use of the rect path defined by the caller) + gc.fillStyle = '#FFD500'; + gc.fill(); -}); + // render cell border + gc.strokeStyle = gc.createPattern(images.caution, 'repeat'); + gc.lineWidth = 5; + gc.beginPath(); + gc.moveTo(x, y); // caution: do not use rect() here because Chrome does not clip its stroke properly + gc.lineTo(x + width, y); + gc.lineTo(x + width, y + height); + gc.lineTo(x, y + height); + gc.lineTo(x, y); + gc.stroke(); + // adjust clip region to prevent text from rendering over right border should it overflow + gc.beginPath(); + gc.rect(x, y, width - 2, height); + gc.clip(); -module.exports = ThumbwheelScrolling; + // render message text + gc.fillStyle = '#A00'; + gc.textAlign = 'start'; + gc.textBaseline = 'middle'; + gc.font = 'bold 6pt "arial narrow", verdana, geneva'; + gc.fillText(message, x + 4, y + height / 2 + 0.5); -},{"./Feature.js":59}],66:[function(require,module,exports){ -'use strict'; +} -module.exports = { - Feature: require('./Feature'), // abstract base class - CellClick: require('./CellClick'), - CellEditing: require('./CellEditing'), - CellSelection: require('./CellSelection'), - ColumnAutosizing: require('./ColumnAutosizing'), - ColumnMoving: require('./ColumnMoving'), - ColumnResizing: require('./ColumnResizing'), - ColumnSelection: require('./ColumnSelection'), - ColumnSorting: require('./ColumnSorting'), - Filters: require('./Filters'), - KeyPaging: require('./KeyPaging'), - OnHover: require('./OnHover'), - ColumnPicker: require('./ColumnPicker'), - RowResizing: require('./RowResizing'), - RowSelection: require('./RowSelection'), - ThumbwheelScrolling: require('./ThumbwheelScrolling') -}; +module.exports = renderCellError; -},{"./CellClick":50,"./CellEditing":51,"./CellSelection":52,"./ColumnAutosizing":53,"./ColumnMoving":54,"./ColumnPicker":55,"./ColumnResizing":56,"./ColumnSelection":57,"./ColumnSorting":58,"./Feature":59,"./Filters":60,"./KeyPaging":61,"./OnHover":62,"./RowResizing":63,"./RowSelection":64,"./ThumbwheelScrolling":65}],67:[function(require,module,exports){ +},{"../../images/index":3}],76:[function(require,module,exports){ 'use strict'; module.exports = (function() { @@ -23124,7 +23822,7 @@ module.exports = (function() { })(); -},{}],68:[function(require,module,exports){ +},{}],77:[function(require,module,exports){ 'use strict'; var Map = require('./Map'); @@ -23223,7 +23921,7 @@ module.exports = (function() { })(); -},{"./DataNodeBase":67,"./Map":78}],69:[function(require,module,exports){ +},{"./DataNodeBase":76,"./Map":87}],78:[function(require,module,exports){ 'use strict'; var DataNodeBase = require('./DataNodeBase'); @@ -23261,7 +23959,7 @@ module.exports = (function() { })(); -},{"./DataNodeBase":67}],70:[function(require,module,exports){ +},{"./DataNodeBase":76}],79:[function(require,module,exports){ 'use strict'; var DataNodeGroup = require('./DataNodeGroup'); @@ -23306,7 +24004,7 @@ module.exports = (function() { })(); -},{"./DataNodeGroup":68}],71:[function(require,module,exports){ +},{"./DataNodeGroup":77}],80:[function(require,module,exports){ 'use strict'; var DataSourceSorter = require('./DataSourceSorter'); @@ -23554,7 +24252,7 @@ module.exports = (function() { })(); -},{"./DataNodeGroup":68,"./DataNodeLeaf":69,"./DataNodeTree":70,"./DataSourceSorter":75}],72:[function(require,module,exports){ +},{"./DataNodeGroup":77,"./DataNodeLeaf":78,"./DataNodeTree":79,"./DataSourceSorter":84}],81:[function(require,module,exports){ 'use strict'; module.exports = (function() { @@ -23645,7 +24343,7 @@ module.exports = (function() { })(); -},{}],73:[function(require,module,exports){ +},{}],82:[function(require,module,exports){ 'use strict'; var DataSourceDecorator = require('./DataSourceDecorator'); @@ -23714,7 +24412,7 @@ module.exports = (function() { })(); -},{"./DataSourceDecorator":72}],74:[function(require,module,exports){ +},{"./DataSourceDecorator":81}],83:[function(require,module,exports){ 'use strict'; var DataSourceDecorator = require('./DataSourceDecorator'); @@ -23786,7 +24484,7 @@ module.exports = (function() { })(); -},{"./DataSourceDecorator":72}],75:[function(require,module,exports){ +},{"./DataSourceDecorator":81}],84:[function(require,module,exports){ 'use strict'; var Utils = require('./Utils.js'); @@ -23824,7 +24522,7 @@ module.exports = (function() { })(); -},{"./DataSourceDecorator":72,"./Utils.js":79}],76:[function(require,module,exports){ +},{"./DataSourceDecorator":81,"./Utils.js":88}],85:[function(require,module,exports){ 'use strict'; var DataSourceDecorator = require('./DataSourceDecorator'); @@ -23876,7 +24574,7 @@ module.exports = (function() { })(); -},{"./DataSourceDecorator":72,"./DataSourceSorter":75}],77:[function(require,module,exports){ +},{"./DataSourceDecorator":81,"./DataSourceSorter":84}],86:[function(require,module,exports){ 'use strict'; module.exports = (function() { @@ -23977,9 +24675,9 @@ module.exports = (function() { })(); -},{}],78:[function(require,module,exports){ -module.exports=require(26) -},{}],79:[function(require,module,exports){ +},{}],87:[function(require,module,exports){ +module.exports=require(70) +},{}],88:[function(require,module,exports){ 'use strict'; var stableSort = require('./stableSort.js'); @@ -23994,7 +24692,7 @@ module.exports = (function() { })(); -},{"./Map.js":78,"./stableSort.js":82}],80:[function(require,module,exports){ +},{"./Map.js":87,"./stableSort.js":91}],89:[function(require,module,exports){ 'use strict'; module.exports = (function() { @@ -24088,7 +24786,7 @@ module.exports = (function() { })(); -},{}],81:[function(require,module,exports){ +},{}],90:[function(require,module,exports){ 'use strict'; var JSDataSource = require('./JSDataSource'); @@ -24113,7 +24811,7 @@ module.exports = (function() { })(); -},{"./DataSourceAggregator":71,"./DataSourceFilter":73,"./DataSourceGlobalFilter":74,"./DataSourceSorter":75,"./DataSourceSorterComposite":76,"./JSDataSource":77,"./aggregations":80}],82:[function(require,module,exports){ +},{"./DataSourceAggregator":80,"./DataSourceFilter":82,"./DataSourceGlobalFilter":83,"./DataSourceSorter":84,"./DataSourceSorterComposite":85,"./JSDataSource":86,"./aggregations":89}],91:[function(require,module,exports){ 'use strict'; var stabilize = function(comparator, descending) { @@ -24207,123 +24905,5 @@ module.exports = (function() { return sort; })(); -},{}],83:[function(require,module,exports){ -'use strict'; - -var images = require('../images'); - -/** - * @summary Writes error message into cell. - * - * @desc This funciton is guaranteed to be called as follows: - * - * ```javascript - * gc.save(); - * gc.beginPath(); - * gc.rect(x, y, width, height); - * gc.clip(); - * renderCellError(gc, message, x, y, width, height); - * gc.restore(); - * ``` - * - * Before doing anything else, this function should clear the cell by setting `gc.fillStyle` and calling `gc.fill()`. - * - * @param {CanvasRenderingContext2D} gc - * @param {string} message - * @param {number} x - * @param {number} y - * @param {number} width - * @param {number} height - */ -function renderCellError(gc, message, x, y, width, height) { - - // clear the cell - // (this makes use of the rect path defined by the caller) - gc.fillStyle = '#FFD500'; - gc.fill(); - - // render cell border - gc.strokeStyle = gc.createPattern(images.caution, 'repeat'); - gc.lineWidth = 5; - gc.beginPath(); - gc.moveTo(x, y); // caution: do not use rect() here because Chrome does not clip its stroke properly - gc.lineTo(x + width, y); - gc.lineTo(x + width, y + height); - gc.lineTo(x, y + height); - gc.lineTo(x, y); - gc.stroke(); - - // adjust clip region to prevent text from rendering over right border should it overflow - gc.beginPath(); - gc.rect(x, y, width - 2, height); - gc.clip(); - - // render message text - gc.fillStyle = '#A00'; - gc.textAlign = 'start'; - gc.textBaseline = 'middle'; - gc.font = 'bold 6pt "arial narrow", verdana, geneva'; - gc.fillText(message, x + 4, y + height / 2 + 0.5); - -} - -module.exports = renderCellError; - -},{"../images":2}],84:[function(require,module,exports){ -'use strict'; - -var cssInjector = require('css-injector'); - -var stylesheets = { - grid: [ - 'div#grid-container {', - ' position: relative;', - ' display: inline-block;', - ' -webkit-user-select: none;', - ' -moz-user-select: none;', - ' -ms-user-select: none;', - ' -o-user-select: none;', - ' user-select: none;', - ' overflow: hidden; }', - 'visible { opacity: 0.75; }', - 'hidden { opacity: 0.0; }', - 'editor {', - ' position: absolute;', - ' display: none;', - ' border: solid 2px black;', - ' outline: 0;', - ' padding: 0;', - ' z-index: 1000; }' - ], - 'list-dragon': [ - 'div.dragon-list, li.dragon-pop {', - ' font-family: Roboto, sans-serif;', - ' text-transform: capitalize; }', - 'div.dragon-list {', - ' position: absolute;', - ' top: 4%;', - ' left: 4%;', - ' height: 92%;', - ' width: 20%; }', - 'div.dragon-list:nth-child(2) { left: 28%; }', - 'div.dragon-list:nth-child(3) { left: 52%; }', - 'div.dragon-list:nth-child(4) { left: 76%; }', - 'div.dragon-list > div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }', - 'div.dragon-list > ul { top: 46px; }', - 'div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {', - ' content: \'\\2b24\';', // BLACK LARGE CIRCLE - ' color: #b6b6b6;', - ' font-size: 30px;', - ' margin: 8px 14px 8px 8px; }', - 'li.dragon-pop { opacity:.8; }' - ] -}; - -function addStylesheet(key, referenceElement) { - cssInjector(stylesheets[key], key, referenceElement); -} - -module.exports = addStylesheet; - -},{"css-injector":3}]},{},[49]) -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/browser-pack/_prelude.js","/Users/stevewirts/Projects/dev/fin-hypergrid/images/images.js","/Users/stevewirts/Projects/dev/fin-hypergrid/images/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/css-injector/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/extend-me/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/filter-tree/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/filter-tree/js/FilterLeaf.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/filter-tree/js/FilterNode.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/filter-tree/js/template.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/filter-tree/js/tree-operators.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/finbars/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/fincanvas/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/fincanvas/js/GraphicsContext.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/fincanvas/js/gc-console-logger.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/fincanvas/js/polymergestures.dev.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/list-dragon/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/lru-cache/lib/lru-cache.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/mustache/mustache.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/object-iterators/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/rectangular/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/regexp-like/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/sparse-boolean-array/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/node_modules/templex/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/CellProvider.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/Formatters.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/Hypergrid.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/Mappy.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/Renderer.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/SelectionModel.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/TableDialog.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/Behavior.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/Column.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/DataModelDecorator.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/JSON.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/Null.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/behaviors/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/CellEditor.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Choice.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Color.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Date.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Filter.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Simple.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Slider.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Spinner.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/Textfield.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/cellEditors/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/dataModels/DataModel.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/dataModels/JSON.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/defaults.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/fake_6a71c23b.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/CellClick.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/CellEditing.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/CellSelection.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnAutosizing.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnMoving.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnPicker.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnResizing.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnSelection.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ColumnSorting.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/Feature.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/Filters.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/KeyPaging.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/OnHover.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/RowResizing.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/RowSelection.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/ThumbwheelScrolling.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/features/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeBase.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeGroup.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeLeaf.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeTree.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceAggregator.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceDecorator.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceFilter.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceGlobalFilter.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceSorter.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceSorterComposite.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/JSDataSource.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/Utils.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/aggregations.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/index.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/local_node_modules/finanalytics/stableSort.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/renderCellError.js","/Users/stevewirts/Projects/dev/fin-hypergrid/src/stylesheets.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/xBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxrEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1kBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxiBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpyGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7qCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACphBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/yBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxiBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7qBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzhBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","module.exports = { // This file generated by gulp-imagine-64 at 1:11:59 PM on 1/6/2016\n\t\"calendar\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAAc0lEQVR4nIXQwQkCMRSE4U9ZLMCT9Xjaq2AfNhfYU5oQLMAOtoN48EWei5iBIRPe/yYQ3qrhf1lFG7iKcEaJxSfukUvMWgdHavt0uWHtg2QwxXnAnJZ2uOLyVZtybzzhgWNmfoFl0/YB87NbzR1cjP9xeQHSDC6mcL1xFQAAAABJRU5ErkJggg==\"\n\t},\n\t\"checked\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAYJJREFUOE+NkstLglEQxf0fahG0iFrUxm2ElFDYLohCqCDaCAkWPaxIRbFFEJEaGEKLDCoMETRFUAMLyaIHBUG6sSKIMtKFqEhLT818ZUgmDhzu3DPn9z0uV1RrmUwmyGQyqNVqfFvViwBxu5RFPZuLSyGMKhz/qlEsRV19K8xm6y+w7bpBPFnAferjj3bdQX6DpHcAUwavAHUN2RGIZxBJZHH2mC/TUeydwwTZvBegLENNgw7sX6Wh1FswNmPEmjPCDyGRRwCtW9E3tMgdAtQw7GZjYcNX+gza2wJ3ZXsSZUuQ0vWCOV8SHfJJ/uluhbHUj1v8PKNMszIoQNRMHCShD6Wh8zyhrbOPwz8w+STKlCCJ7oRNUzQH63kBs5thBghePXxlj2aUoSxDPcuXPNiLAc5EEZ6HIkbmV2DYiXBPHs0o079+K0DTVj/s11mE00A0L+g4VcDp10qKZMAzytBhMaTRaPmYg885DlcSzSij0eoEiIouoUqlqqqaL2rlEok+Ad4vlfzPoVDsAAAAAElFTkSuQmCC\"\n\t},\n\t\"down-rectangle\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAAkAAAAECAYAAABcDxXOAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAadEVYdFNvZnR3YXJlAFBhaW50Lk5FVCB2My41LjExR/NCNwAAABpJREFUGFdjgIL/eDAKIKgABggqgAE0BQwMAPTlD/Fpi0JfAAAAAElFTkSuQmCC\"\n\t},\n\t\"filter-off\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAAChSURBVChTzZHBCoUgFET9TqEiskgyWoutQvRLRIr+cR7XQAjiJW/1BgZmMUevXsY5xy9OoDEGMcYiUzeB67qibVuwQjVNA6311V+WBeM4vsLDMEApde/1fY9pmtI453neHEKAlBJd1z0fXtc16PbjODK07zvmeUZVVd8nooc75zJIOX3Gm6i0bVsGKf8xKIRIuyJTLgJJ3nvQzsjW2geIsQ/pr9hMVrSncAAAAABJRU5ErkJggg==\"\n\t},\n\t\"filter-on\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAACoSURBVChTY3BqfP2fHAzWmDbj7f8p294RhVOBasEa02e+/e/VBmQQCTxaX/9PnvYGoj5ywpv/Qd2ENft3vv4f1gfVBAP+nW/+h/a+ATtn1q73KHjytvdgg3070DTBgHvL6/8g22fsQGiaDmSHA21xaybgIpDHixa8hWssnA8NDEIApCh3LkIjiD2INYJCL2X6W3B8gdhEaQQBUOCA4gyE8+e9xaKJgQEA/74BNE3cElkAAAAASUVORK5CYII=\"\n\t},\n\t\"unchecked\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAARBJREFUOE+9krtug1AQRPldSio7FQ1tZImOkoKOBomGT0EURC5ino54yTw90WywQhTkIkVWGoF2zuxdrlD+t0zThKZpT0Vmxb8CQRCg6zr0fb8rer7vfwcPxxdcrx+YpgnzPGNZlh9ibxxHlGUJshLSdV0at9tNpg7DIBrX5+OkPM9BVkKGYSBJEtR1jbZrBdiqbVtUVYU0TUFWQq+nE+I4xvvlImGaW7FHjwxZCVmWhbfzGVmWoSgKWXUr9uiRISshx3FkEldomubXauzRI0NWQp7nyUR+NG/rfr/jUXxnjx5vmKyEbNuWox9Xvid6ZMhK6HA4wnVdhGGIKIp2RY8MWQmx+JuoqvpUZFb8L6UonyYL3uOtrFH+AAAAAElFTkSuQmCC\"\n\t},\n\t\"up-down-spin\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGJJREFUOE+lkwEKACEIBH2Zb/PnHsoGeaVJDUjGOgRRpKpkiIj+y4MME3eDR7kaKOVNsJyMNjIHzGy9YnW6J7qIcrriQimeCqORNABd0fpRTkt8uVUj7EsxC6vs/q3e/Q6iD2bwnByjPXHNAAAAAElFTkSuQmCC\"\n\t},\n\t\"up-down\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGFJREFUOE+lkkEKQCEIRD2ZJ3Ph3iN4WD9GflpYhj0YYowpGgJmbikd3gjMDFokwbuT1iAiurG5nomgqo5QaPo9ERQRI6Jf7sfGjudy2je23+i0Wl2oQ85TOdlfrJQOazF8br+rqTXQKn0AAAAASUVORK5CYII=\"\n\t},\n};\n","/* eslint-env browser */\n\n'use strict';\n\nvar _ = require('object-iterators');\n\nvar images = require('./images'); // this is the file generated by gulpfile.js (and ignored by git)\n\n_(images).each(function(image, key) {\n    var element = new Image();\n    element.src = 'data:' + image.type + ';base64,' + image.data;\n    images[key] = element;\n});\n\nimages.checkbox = function(state) {\n    return images[state ? 'checked' : 'unchecked'];\n};\n\nimages.filter = function(state) {\n    return images[state ? 'filter-on' : 'filter-off'];\n};\n\nmodule.exports = images;\n","'use strict';\n\n/* eslint-env browser */\n\n/** @namespace cssInjector */\n\n/**\n * @summary Insert base stylesheet into DOM\n *\n * @desc Creates a new `<style>...</style>` element from the named text string(s) and inserts it but only if it does not already exist in the specified container as per `referenceElement`.\n *\n * > Caveat: If stylesheet is for use in a shadow DOM, you must specify a local `referenceElement`.\n *\n * @returns A reference to the newly created `<style>...</style>` element.\n *\n * @param {string|string[]} cssRules\n * @param {string} [ID]\n * @param {undefined|null|Element|string} [referenceElement] - Container for insertion. Overloads:\n * * `undefined` type (or omitted): injects stylesheet at top of `<head>...</head>` element\n * * `null` value: injects stylesheet at bottom of `<head>...</head>` element\n * * `Element` type: injects stylesheet immediately before given element, wherever it is found.\n * * `string` type: injects stylesheet immediately before given first element found that matches the given css selector.\n *\n * @memberOf cssInjector\n */\nfunction cssInjector(cssRules, ID, referenceElement) {\n    if (typeof referenceElement === 'string') {\n        referenceElement = document.querySelector(referenceElement);\n        if (!referenceElement) {\n            throw 'Cannot find reference element for CSS injection.';\n        }\n    } else if (referenceElement && !(referenceElement instanceof Element)) {\n        throw 'Given value not a reference element.';\n    }\n\n    var container = referenceElement && referenceElement.parentNode || document.head || document.getElementsByTagName('head')[0];\n\n    if (ID) {\n        ID = cssInjector.idPrefix + ID;\n\n        if (container.querySelector('#' + ID)) {\n            return; // stylesheet already in DOM\n        }\n    }\n\n    var style = document.createElement('style');\n    style.type = 'text/css';\n    if (ID) {\n        style.id = ID;\n    }\n    if (cssRules instanceof Array) {\n        cssRules = cssRules.join('\\n');\n    }\n    cssRules = '\\n' + cssRules + '\\n';\n    if (style.styleSheet) {\n        style.styleSheet.cssText = cssRules;\n    } else {\n        style.appendChild(document.createTextNode(cssRules));\n    }\n\n    if (referenceElement === undefined) {\n        referenceElement = container.firstChild;\n    }\n\n    container.insertBefore(style, referenceElement);\n\n    return style;\n}\n\n/**\n * @summary Optional prefix for `<style>` tag IDs.\n * @desc Defaults to `'injected-stylesheet-'`.\n * @type {string}\n * @memberOf cssInjector\n */\ncssInjector.idPrefix = 'injected-stylesheet-';\n\n// Interface\nmodule.exports = cssInjector;\n","'use strict';\n\n/** @namespace extend-me **/\n\n/** @summary Extends an existing constructor into a new constructor.\n *\n * @returns {ChildConstructor} A new constructor, extended from the given context, possibly with some prototype additions.\n *\n * @desc Extends \"objects\" (constructors), with optional additional code, optional prototype additions, and optional prototype member aliases.\n *\n * > CAVEAT: Not to be confused with Underscore-style .extend() which is something else entirely. I've used the name \"extend\" here because other packages (like Backbone.js) use it this way. You are free to call it whatever you want when you \"require\" it, such as `var inherits = require('extend')`.\n *\n * Provide a constructor as the context and any prototype additions you require in the first argument.\n *\n * For example, if you wish to be able to extend `BaseConstructor` to a new constructor with prototype overrides and/or additions, basic usage is:\n *\n * ```javascript\n * var Base = require('extend-me').Base;\n * var BaseConstructor = Base.extend(basePrototype); // mixes in .extend\n * var ChildConstructor = BaseConstructor.extend(childPrototypeOverridesAndAdditions);\n * var GrandchildConstructor = ChildConstructor.extend(grandchildPrototypeOverridesAndAdditions);\n * ```\n *\n * This function (`extend()`) is added to the new extended object constructor as a property `.extend`, essentially making the object constructor itself easily \"extendable.\" (Note: This is a property of each constructor and not a method of its prototype!)\n *\n * @param {string} [extendedClassName] - This is simply added to the prototype as $$CLASS_NAME. Useful for debugging because all derived constructors appear to have the same name (\"Constructor\") in the debugger. This property is ignored unless `extend.debug` is explicitly set to a truthy value.\n *\n * @param {extendedPrototypeAdditionsObject} [prototypeAdditions] - Object with members to copy to new constructor's prototype. Most members will be copied to the prototype. Some members, however, have special meanings as explained in the {@link extendedPrototypeAdditionsObject|type definition} (and may or may not be copied to the prototype).\n *\n * @property {boolean} [debug] - See parameter `extendedClassName` _(above)_.\n *\n * @property {object} Base - A convenient base class from which all other classes can be extended.\n *\n * @memberOf extend-me\n */\nfunction extend(extendedClassName, prototypeAdditions) {\n    switch (arguments.length) {\n        case 0:\n            prototypeAdditions = {};\n            break;\n        case 1:\n            prototypeAdditions = extendedClassName;\n            if (typeof prototypeAdditions !== 'object') {\n                throw 'Single parameter overload must be object.';\n            }\n            extendedClassName = undefined;\n            break;\n        case 2:\n            if (typeof extendedClassName !== 'string' || typeof prototypeAdditions !== 'object') {\n                throw 'Two parameter overload must be string, object.';\n            }\n            break;\n        default:\n            throw 'Too many parameters';\n    }\n\n    function Constructor() {\n        if (prototypeAdditions.preInitialize) {\n            prototypeAdditions.preInitialize.apply(this, arguments);\n        }\n\n        initializePrototypeChain.apply(this, arguments);\n\n        if (prototypeAdditions.postInitialize) {\n            prototypeAdditions.postInitialize.apply(this, arguments);\n        }\n    }\n\n    Constructor.extend = extend;\n\n    var prototype = Constructor.prototype = Object.create(this.prototype);\n    prototype.constructor = Constructor;\n\n    if (extendedClassName && extend.debug) {\n        prototype.$$CLASS_NAME = extendedClassName;\n    }\n\n    for (var key in prototypeAdditions) {\n        if (prototypeAdditions.hasOwnProperty(key)) {\n            var value = prototypeAdditions[key];\n            switch (key) {\n                case 'initializeOwn':\n                    // already called above; not needed in prototype\n                    break;\n                case 'aliases':\n                    for (var alias in value) {\n                        if (value.hasOwnProperty(alias)) {\n                            makeAlias(value[alias], alias);\n                        }\n                    }\n                    break;\n                default:\n                    if (typeof value === 'string' && value[0] === '#') {\n                        makeAlias(value, key.substr(1));\n                    } else {\n                        prototype[key] = value;\n                    }\n            }\n        }\n    }\n\n    return Constructor;\n\n    function makeAlias(value, key) { // eslint-disable-line no-shadow\n        prototype[key] = prototypeAdditions[value];\n    }\n}\n\nextend.Base = function () {};\nextend.Base.extend = extend;\n\n/** @typedef {function} extendedConstructor\n * @property prototype.super - A reference to the prototype this constructor was extended from.\n * @property [extend] - If `prototypeAdditions.extendable` was truthy, this will be a reference to {@link extend.extend|extend}.\n */\n\n/** @typedef {object} extendedPrototypeAdditionsObject\n * @property {function} [initialize] - Additional constructor code for new object. This method is added to the new constructor's prototype. Gets passed new object as context + same args as constructor itself. Called on instantiation after similar function in all ancestors called with same signature.\n * @property {function} [initializeOwn] - Additional constructor code for new object. This method is added to the new constructor's prototype. Gets passed new object as context + same args as constructor itself. Called on instantiation after (all) the `initialize` function(s).\n * @property {object} [aliases] - Hash of aliases for prototype members in form `{ key: 'member', ... }` where `key` is the name of an alieas and `'member'` is the name of an existing member in the prototype. Each such key is added to the prototype as a reference to the named member. (The `aliases` object itself is *not* added to prototype.) Alternatively:\n * @property {string} [keys] - Arbitrary property names defined here with string values starting with a `#` character will alias the actual properties named in the strings (following the `#`). This is an alternative to providing an `aliases` hash, perhaps simpler (though subtler). (Use arbitrary identifiers here; don't use the name `keys`!)\n * @property {*} [arbitraryProperties] - Any additional arbitrary properties defined here will be added to the new constructor's prototype. (Use arbitrary identifiers here; don't use the name `aribitraryProperties`!)\n */\n\n/** @summary Call all `initialize` methods found in prototype chain.\n * @desc This recursive routine is called by the constructor.\n * 1. Walks back the prototype chain to `Object`'s prototype\n * 2. Walks forward to new object, calling any `initialize` methods it finds along the way with the same context and arguments with which the constructor was called.\n * @private\n * @memberOf extend-me\n */\nfunction initializePrototypeChain() {\n    var term = this,\n        args = arguments;\n    recur(term);\n\n    function recur(obj) {\n        var proto = Object.getPrototypeOf(obj);\n        if (proto.constructor !== Object) {\n            recur(proto);\n            if (proto.hasOwnProperty('initialize')) {\n                proto.initialize.apply(term, args);\n            }\n        }\n    }\n}\n\nmodule.exports = extend;\n","/* eslint-env browser */\n\n// This is the main file, usable as is, such as by /test/index.js.\n// For npm: gulpfile.js copies this file to ../index.js, adjusting the require paths and defining the `css` local.\n// For CDN: gulpfile.js then browserifies ../index.js with sourcemap to /build/filter-tree.js and uglified without sourcemap to /build/filter-tree.min.js. The CDN is https://joneit.github.io/filter-tree.\n\n'use strict';\n\nvar cssInjector = require('css-injector');\n\nvar FilterNode = require('./js/FilterNode');\nvar DefaultFilter = require('./js/FilterLeaf');\nvar template = require('./js/template');\nvar operators = require('./js/tree-operators');\n\nvar css; // defined by code inserted by gulpfile between following comments\n/* inject:css */\ncss = '.filter-tree{font-family:sans-serif;font-size:10pt;line-height:1.5em}.filter-tree label{font-weight:400}.filter-tree input[type=checkbox],.filter-tree input[type=radio]{left:3px;margin-right:3px}.filter-tree ol{margin-top:0}.filter-tree-add,.filter-tree-add-filter,.filter-tree-remove{cursor:pointer}.filter-tree-add,.filter-tree-add-filter{font-style:italic;color:#444;font-size:90%}.filter-tree-add-filter{margin:3px 0 3px 3em;width:120px;display:inline-block}.filter-tree-add-filter:hover,.filter-tree-add:hover{text-decoration:underline}.filter-tree-add-filter.as-menu-header,.filter-tree-add.as-menu-header{background-color:#fff;font-weight:700;font-style:normal}.filter-tree-add-filter.as-menu-header:hover{text-decoration:inherit}.filter-tree-add-filter>div,.filter-tree-add>div,.filter-tree-remove{display:inline-block;width:15px;height:15px;border-radius:8px;background-color:#8c8;font-size:11.5px;font-weight:700;color:#fff;text-align:center;line-height:normal;font-style:normal;font-family:sans-serif;text-shadow:0 0 1.5px grey;margin-right:4px}.filter-tree-add-filter>div:before,.filter-tree-add>div:before{content:\\'\\\\ff0b\\'}.filter-tree-remove{background-color:#e88;border:0}.filter-tree-remove:before{content:\\'\\\\2212\\'}.filter-tree li::after{font-size:70%;font-style:italic;font-weight:700;color:#080}.filter-tree>ol>li:last-child::after{display:none}.op-or>ol>li::after{margin-left:2.5em;content:\\'— OR —\\'}.op-and>ol>li::after{margin-left:2.5em;content:\\'— AND —\\'}.op-nor>ol>li::after{margin-left:2.5em;content:\\'— NOR —\\'}.filter-tree-default>:enabled{margin:0 .4em;background-color:#ddd;border:0}.filter-tree-default>input[type=text]{width:8em;padding:0 5px}.filter-tree-default>select{border:0}.filter-tree-default>.filter-tree-warning{background-color:#ffc}.filter-tree-default>.filter-tree-error{background-color:#Fcc}.filter-tree .footnotes{font-size:6pt;margin:2px 0 0;line-height:normal;white-space:normal;color:#999}.filter-tree .footnotes>ol{margin:0;padding-left:2em}.filter-tree .footnotes>ol>li{margin:2px 0}.filter-tree .footnotes .field-name,.filter-tree .footnotes .field-value{font-weight:700;color:#777}.filter-tree .footnotes .field-value:after,.filter-tree .footnotes .field-value:before{content:\\'\\\"\\'}.filter-tree .footnotes .field-value{font-family:monospace}.filter-tree-chooser{position:absolute;font-size:9pt;outline:0;box-shadow:5px 5px 10px grey}';\n/* endinject */\n\nvar ordinal = 0;\nvar reFilterTreeErrorString = /^filter-tree: /;\n\n/** @constructor\n *\n * @summary A node in a filter tree (including the root node), representing a complex filter expression.\n *\n * @desc A `FilterTree` is an n-ary tree with a single `operator` to be applied to all its `children`.\n *\n * Also known as a \"subtree\" or a \"subexpression\".\n *\n * Each of the `children` can be either:\n *\n * * a terminal node `Filter` (or an object inheriting from `Filter`) representing a simple conditional expression; or\n * * a nested `FilterTree` representing a complex subexpression.\n *\n * The `operator` must be one of the {@link operators|tree operators} or may be left undefined iff there is only one child node.\n *\n * Notes:\n * 1. A `FilterTree` may consist of a single leaf, in which case the `operator` is not used and may be left undefined. However, if a second child is added and the operator is still undefined, it will be set to the default (`'op-and'`).\n * 2. The order of the children is undefined as all operators are commutative. For the '`op-or`' operator, evaluation ceases on the first positive result and for efficiency, all simple conditional expressions will be evaluated before any complex subexpressions.\n * 3. A nested `FilterTree` is distinguished in the JSON object from a `Filter` by the presence of a `children` member.\n * 4. Nesting a `FilterTree` containing a single child is valid (albeit pointless).\n *\n * See {@link FilterNode} for additional `options` properties.\n *\n * @param {object} [options.editors] - Editor hash to override prototype's. These are constructors for objects that extend from `FilterTree.prototype.editors.Default`. Typically, you would include the default editor itself: `{ Default: FilterTree.prototype.editors.Default, ... }`. Alternatively, before instantiating, you might add your additional editors to `FilterTree.prototype.editors` for use by all filter tree objects.\n *\n * @property {FilterTree} parent\n * @property {number} ordinal\n * @property {string} operator\n * @property {FilterNode[]} children - Each one is either a `Filter` (or an object inheriting from `Filter`) or another `FilterTree`..\n * @property {Element} el - The root element of this (sub)tree.\n */\nvar FilterTree = FilterNode.extend('FilterTree', {\n\n    initialize: function(options) {\n        cssInjector(css, 'filter-tree-base', options && options.cssStylesheetReferenceElement);\n\n        if (options.editors) {\n            this.editors = options.editors;\n        }\n    },\n\n    destroy: function() {\n        detachChooser.call(this);\n    },\n\n    editors: {\n        Default: DefaultFilter\n    },\n\n    addEditor: function(key, overrides) {\n        if (overrides) {\n            this.editors[key] = DefaultFilter.extend(overrides);\n        } else {\n            delete this.editors[key];\n        }\n    },\n\n    newView: function() {\n        this.el = template('tree', ++ordinal);\n        this.el.addEventListener('click', catchClick.bind(this));\n    },\n\n    load: function(state) {\n        if (!state) {\n            var filterEditorNames = Object.keys(this.editors),\n                onlyOneFilterEditor = filterEditorNames.length === 1;\n            this.children = onlyOneFilterEditor ? [new this.editors[filterEditorNames[0]]({\n                parent: this\n            })] : [];\n            this.operator = 'op-and';\n        } else {\n            throwIfJSON(state);\n\n            // Validate `state.operator`\n            if (!(operators[state.operator] || state.operator === undefined && state.children.length === 1)) {\n                throw this.Error('Expected `operator` property to be one of: ' + Object.keys(operators));\n            }\n            this.operator = state.operator;\n\n            // Validate `state.children`\n            if (!(state.children instanceof Array && state.children.length)) {\n                throw this.Error('Expected `children` property to be a non-empty array.');\n            }\n            this.children = [];\n            var self = this;\n            state.children.forEach(function(state) { // eslint-disable-line no-shadow\n                var Constructor;\n                if (typeof state !== 'object') {\n                    throw self.Error('Expected child to be an object containing either `children`, `editor`, or neither.');\n                }\n                if (state.children) {\n                    Constructor = FilterTree;\n                } else {\n                    Constructor = self.editors[state.editor || 'Default'];\n                }\n                self.children.push(new Constructor({\n                    state: state,\n                    parent: self\n                }));\n            });\n        }\n    },\n\n    render: function() {\n        // simulate click on the operator to display strike-through and operator between filters\n        var radioButton = this.el.querySelector('input[value=' + this.operator + ']');\n        radioButton.checked = true;\n        this['filter-tree-op-choice']({\n            target: radioButton\n        });\n\n        // when multiple filter editors available, simulate click on the new \"add conditional\" link\n        if (!this.children.length && Object.keys(this.editors).length > 1) {\n            var addFilterLink = this.el.querySelector('.filter-tree-add-filter');\n            this['filter-tree-add-filter']({\n                target: addFilterLink\n            });\n        }\n\n        // proceed with render\n        FilterNode.prototype.render.call(this);\n    },\n\n    'filter-tree-op-choice': function(evt) {\n        var radioButton = evt.target;\n\n        this.operator = radioButton.value;\n\n        // display strike-through\n        var radioButtons = this.el.querySelectorAll('label>input.filter-tree-op-choice[name=' + radioButton.name + ']');\n        Array.prototype.slice.call(radioButtons).forEach(function(radioButton) { // eslint-disable-line no-shadow\n            radioButton.parentElement.style.textDecoration = radioButton.checked ? 'none' : 'line-through';\n        });\n\n        // display operator between filters by adding operator string as a CSS class of this tree\n        for (var key in operators) {\n            this.el.classList.remove(key);\n        }\n        this.el.classList.add(this.operator);\n    },\n\n    'filter-tree-add-filter': function(evt) {\n        var filterEditorNames = Object.keys(this.editors);\n        if (filterEditorNames.length === 1) {\n            this.children.push(new this.editors[filterEditorNames[0]]({\n                parent: this\n            }));\n        } else {\n            attachChooser.call(this, evt);\n        }\n    },\n\n    'filter-tree-add': function() {\n        this.children.push(new FilterTree({\n            parent: this\n        }));\n    },\n\n    'filter-tree-remove': function(evt) {\n        var deleteButton = evt.target,\n            listItem = deleteButton.parentElement,\n            children = this.children,\n            el = deleteButton.nextElementSibling;\n\n        children.forEach(function(child, idx) {\n            if (child.el === el) {\n                delete children[idx];\n                listItem.remove();\n            }\n        });\n    },\n\n    /**\n     * @param {boolean} [object.alert=true] - Announce error via window.alert() before returning.\n     * @param {boolean} [object.focus=true] - Place the focus on the offending control and give it error color.\n     * @returns {undefined|string} `undefined` means valid or string containing error message.\n     */\n    validate: function(options) {\n        options = options || {};\n\n        var focus = options.focus === undefined || options.focus,\n            alert = options.alert === undefined || options.alert,\n            result;\n\n        try {\n            validate.call(this, focus);\n        } catch (err) {\n            result = err.message;\n\n            // Throw when not a filter tree error\n            if (!reFilterTreeErrorString.test(result)) {\n                throw err;\n            }\n\n            if (alert) {\n                result = result.replace(reFilterTreeErrorString, '');\n                window.alert(result); // eslint-disable-line no-alert\n            }\n        }\n\n        return result;\n    },\n\n    test: function test(dataRow) {\n        var operator = operators[this.operator],\n            result = operator.seed;\n\n        this.children.find(function(child) {\n            if (child) {\n                if (child instanceof DefaultFilter) {\n                    result = operator.reduce(result, child.test(dataRow));\n                } else if (child.children.length) {\n                    result = operator.reduce(result, test.call(child, dataRow));\n                }\n                return result === operator.abort;\n            }\n\n            return false;\n        });\n\n        return operator.negate ? !result : result;\n    },\n\n    toJSON: function toJSON() {\n        var state = {\n            operator: this.operator,\n            children: []\n        };\n\n        this.children.forEach(function(child) {\n            if (child) {\n                if (child instanceof DefaultFilter) {\n                    state.children.push(child);\n                } else if (child.children.length) {\n                    state.children.push(toJSON.call(child));\n                }\n            }\n        });\n\n        var metadata = FilterNode.prototype.toJSON.call(this);\n        Object.keys(metadata).forEach(function(key) {\n            state[key] = metadata[key];\n        });\n\n        return state;\n    },\n\n    toSQL: function toSQL() {\n        var lexeme = operators[this.operator].SQL,\n            where = lexeme.beg;\n\n        this.children.forEach(function(child, idx) {\n            var op = idx ? ' ' + lexeme.op + ' ' : '';\n            if (child) {\n                if (child instanceof DefaultFilter) {\n                    where += op + child.toSQL();\n                } else if (child.children.length) {\n                    where += op + toSQL.call(child);\n                }\n            }\n        });\n\n        where += lexeme.end;\n\n        return where;\n    }\n\n});\n\n/**\n * Checks to make sure `state` is defined as a plain object and not a JSON string.\n * If not, throws error and does not return.\n * @param {object} state\n * @private\n */\nfunction throwIfJSON(state) {\n    if (typeof state !== 'object') {\n        var errMsg = 'Expected `state` parameter to be an object.';\n        if (typeof state === 'string') {\n            errMsg += ' See `JSON.parse()`.';\n        }\n        throw this.Error(errMsg);\n    }\n}\n\nfunction catchClick(evt) { // must be called with context\n    var elt = evt.target;\n\n    var handler = this[elt.className] || this[elt.parentNode.className];\n    if (handler) {\n        if (this.detachChooser) {\n            this.detachChooser();\n        }\n        handler.call(this, evt);\n        evt.stopPropagation();\n    }\n}\n\n/**\n * Throws error if invalid expression tree.\n * Caught by {@link FilterTree#validate|FilterTree.prototype.validate()}.\n * @param {boolean} focus - Move focus to offending control.\n * @returns {undefined} if valid\n * @private\n */\nfunction validate(focus) { // must be called with context\n    if (this instanceof FilterTree && !this.children.length) {\n        throw new FilterNode.Error('Empty subexpression (no filters).');\n    }\n\n    this.children.forEach(function(child) {\n        if (child instanceof DefaultFilter) {\n            child.validate(focus);\n        } else if (child.children.length) {\n            validate.call(child, focus);\n        }\n    });\n}\n\nfunction attachChooser(evt) { // must be called with context\n    var tree = this,\n        rect = evt.target.getBoundingClientRect();\n\n    if (!rect.width) {\n        // not in DOM yet so try again later\n        setTimeout(function() {\n            attachChooser.call(tree, evt);\n        }, 50);\n        return;\n    }\n\n    // Create it\n    var editors = Object.keys(FilterTree.prototype.editors),\n        chooser = this.chooser = document.createElement('select');\n\n    chooser.className = 'filter-tree-chooser';\n    chooser.size = editors.length;\n\n    editors.forEach(function(key) {\n        var name = tree.editors[key].prototype.name || key;\n        name = name.replace('?', '\\u225F'); // make question mark into \"? over equals\" UNICODE char\n        chooser.add(new Option(name, key));\n    });\n\n    chooser.onmouseover = function(evt) { // eslint-disable-line no-shadow\n        evt.target.selected = true;\n    };\n\n    // Position it\n    chooser.style.left = rect.left + 19 + 'px';\n    chooser.style.top = rect.bottom + 'px';\n\n    this.detachChooser = detachChooser.bind(this);\n    window.addEventListener('click', this.detachChooser); // detach chooser if click outside\n\n    chooser.onclick = function() {\n        tree.children.push(new tree.editors[chooser.value]({\n            parent: tree\n        }));\n        // click bubbles up to window where it detaches chooser\n    };\n\n    chooser.onmouseout = function() {\n        chooser.selectedIndex = -1;\n    };\n\n    // Add it to the DOM\n    this.el.appendChild(chooser);\n\n    // Color the link similarly\n    this.chooserTarget = evt.target;\n    this.chooserTarget.classList.add('as-menu-header');\n}\n\nfunction detachChooser() { // must be called with context\n    var chooser = this.chooser;\n    if (chooser) {\n        this.el.removeChild(chooser);\n        this.chooserTarget.classList.remove('as-menu-header');\n\n        chooser.onclick = chooser.onmouseout = null;\n        window.removeEventListener('click', this.detachChooser);\n\n        delete this.detachChooser;\n        delete this.chooser;\n    }\n}\n\nmodule.exports = FilterTree;\n","/* eslint-env browser */\n/* eslint-disable key-spacing */\n\n'use strict';\n\nvar regExpLIKE = require('regexp-like').cached;\n\nvar FilterNode = require('./FilterNode');\nvar template = require('./template');\n\n\n/** @typedef {object} converter\n * @property {function} to - Returns input value converted to type. Fails silently.\n * @property {function} not - Tests input value against type, returning `false if type or `true` if not type.\n */\n/** @type converter */\nvar numberConverter = { to: Number, not: isNaN };\n/** @type converter */\nvar dateConverter = { to: function(s) { return new Date(s); }, not: isNaN };\n\n/** @constructor\n * @summary A terminal node in a filter tree, representing a conditional expression.\n * @desc Also known as a \"filter.\"\n */\nvar FilterLeaf = FilterNode.extend('FilterLeaf', {\n\n    name: 'Column ? Literal',\n\n    operators: {\n        '<'       : { test: function(a, b) { return a < b; } },\n        '\\u2264'  : { test: function(a, b) { return a <= b; }, SQL: '<=' },\n        '='       : { test: function(a, b) { return a === b; } },\n        '\\u2265'  : { test: function(a, b) { return a >= b; }, SQL: '>=' },\n        '>'       : { test: function(a, b) { return a > b; } },\n        '\\u2260'  : { test: function(a, b) { return a !== b; }, SQL: '<>' },\n        LIKE      : { test: function(a, b) { return regExpLIKE(b).test(a); } },\n        'NOT LIKE': { test: function(a, b) { return !regExpLIKE(b).test(a); } }\n    },\n\n    destroy: function() {\n        if (this.controls) {\n            for (var key in this.controls) {\n                this.controls[key].removeEventListener('change', cleanUpAndMoveOn);\n            }\n        }\n    },\n\n    newView: function() {\n        var fields = this.parent.nodeFields || this.fields;\n\n        if (!fields) {\n            throw this.Error('Terminal node requires a fields list.');\n        }\n\n        var root = this.el = document.createElement('span');\n        root.className = 'filter-tree-default';\n\n        this.controls = {\n            column: this.makeElement(root, fields, 'column'),\n            operator: this.makeElement(root, Object.keys(this.operators), 'operator'),\n            argument: this.makeElement(root)\n        };\n\n        root.appendChild(document.createElement('br'));\n    },\n\n    /** @typedef {object} valueOption\n     * You should supply both `name` and `alias` but you could omit one or the other and whichever you provide will be used for both. (In such case you might as well just give a string for {@link fieldOption} rather than this object.)\n     * @property {string} [name]\n     * @property {string} [alias]\n     * @property {string} [type] One of the keys of `this.converters`. If not one of these (including `undefined`), field values will be tested with a string comparison.\n     * @property {boolean} [hidden=false]\n     */\n    /** @typedef {object} optionGroup\n     * @property {string} label\n     * @property {fieldOption[]} options\n     */\n    /** @typedef {string|valueOption|optionGroup} fieldOption\n     * The three possible types specify either an `<option>....</option>` element or an `<optgroup>....</optgroup>` element as follows:\n     * * `string` - specifies only the text of an `<option>....</option>` element (the value naturally defaults to the text)\n     * * {@link valueOption} - specifies both the text (`.name`) and the value (`.alias`) of an `<option....</option>` element\n     * * {@link optionGroup} - specifies an `<optgroup>....</optgroup>` element\n     */\n    /**\n     * @summary HTML form controls factory.\n     * @desc Creates and appends a text box or a drop-down.\n     * @returns The new element.\n     * @param {Element} container - An element to which to append the new element.\n     * @param {fieldOption[]} [options] - Overloads:\n     * * If omitted, will create an `<input/>` (text box) element.\n     * * If contains only a single option, will create a `<span>...</span>` element containing the string and a `<input type=hidden>` containing the value.\n     * * Otherwise, creates a `<select>...</select>` element with these options.\n     * @param {null|string} [prompt=''] - Adds an initial `<option>...</option>` element to the drop-down with this value, parenthesized, as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses.\n     */\n    makeElement: function(container, options, prompt) {\n        var el, option, span,\n            tagName = options ? 'select' : 'input';\n\n        if (options && options.length === 1) {\n            option = options[0];\n\n            el = document.createElement('input');\n            el.type = 'hidden';\n            el.value = option.name || option.alias || option;\n\n            span = document.createElement('span');\n            span.innerHTML = option.alias || option.name || option;\n            span.appendChild(el);\n\n            container.appendChild(span);\n        } else {\n            el = addOptions(tagName, options, prompt);\n            this.el.addEventListener('change', cleanUpAndMoveOn);\n            FilterNode.setWarningClass(el);\n            container.appendChild(el);\n        }\n\n        return el;\n    },\n\n    load: function(state) {\n        if (state) {\n            var value, el, i, b, selected, notes = [];\n            for (var key in state) {\n                if (key !== 'fields' && key !== 'editor') {\n                    value = state[key];\n                    el = this.controls[key];\n                    switch (el.type) {\n                        case 'checkbox':\n                        case 'radio':\n                            el = document.querySelectorAll('input[name=\\'' + el.name + '\\']');\n                            for (i = 0; i < el.length; i++) {\n                                el[i].checked = value.indexOf(el[i].value) >= 0;\n                            }\n                            break;\n                        case 'select-multiple':\n                            el = el.options;\n                            for (i = 0, b = false; i < el.length; i++, b = b || selected) {\n                                selected = value.indexOf(el[i].value) >= 0;\n                                el[i].selected = selected;\n                            }\n                            FilterNode.setWarningClass(el, b);\n                            break;\n                        default:\n                            el.value = value;\n                            if (!FilterNode.setWarningClass(el) && el.value !== value) {\n                                notes.push({ key: key, value: value });\n                            }\n                    }\n                }\n            }\n            if (notes.length) {\n                var multiple = notes.length > 1,\n                    footnotes = template(multiple ? 'notes' : 'note'),\n                    inner = footnotes.lastElementChild;\n                notes.forEach(function(note) {\n                    var footnote = multiple ? document.createElement('li') : inner;\n                    note = template('optionMissing', note.key, note.value);\n                    while (note.length) { footnote.appendChild(note[0]); }\n                    if (multiple) { inner.appendChild(footnote); }\n                });\n                el.parentNode.replaceChild(footnotes, el.parentNode.lastElementChild);\n            }\n        }\n    },\n\n    /**\n     * @property {converter} number\n     * @property {converter} date\n     */\n    converters: {\n        number: numberConverter,\n        int: numberConverter, // pseudo-type: really just a Number\n        float: numberConverter, // pseudo-type: really just a Number\n        date: dateConverter\n    },\n\n    /**\n     * Throws error if invalid expression.\n     * Caught by {@link FilterTree#validate|FilterTree.prototype.validate()}.\n     *\n     * Also performs the following compilation actions:\n     * * Copies all the `this.controls`'s values from the DOM to similarly named properties of `this`.\n     * * Pre-sets `this.operation`, `this.converter` and `this.sqlOperator` for efficient access in walks.\n     *\n     * @param {boolean} focus - Move focus to offending control.\n     * @returns {undefined} if valid\n     */\n    validate: function(focus) {\n        for (var elementName in this.controls) {\n            var el = this.controls[elementName],\n                value = controlValue(el).trim();\n\n            if (value === '') {\n                if (focus) { focusOn(el); }\n                throw new FilterNode.Error('Blank ' + elementName + ' control.\\nComplete the filter or delete it.');\n            } else {\n                // Copy each controls's value to property of this object.\n                this[elementName] = value;\n\n                switch (elementName) {\n                    case 'operator':\n                        var operator = this.operators[value];\n                        this.operation = operator.test; // for efficient access in this.test()\n                        this.sqlOperator = operator.SQL || value;\n                        break;\n                    case 'column':\n                        var fields = this.parent.nodeFields || this.fields,\n                            field = findField(fields, value);\n                        if (field && field.type) {\n                            this.converter = this.converters[field.type];\n                        }\n                }\n            }\n        }\n    },\n\n    p: function(dataRow) { return dataRow[this.column]; },\n    q: function() { return this.argument; },\n\n    test: function(dataRow) {\n        var p = this.p(dataRow),\n            q = this.q(dataRow),\n            P, Q, // typed versions of p and q\n            convert = this.converter;\n\n        return (\n            convert &&\n            !convert.not(P = convert.to(p)) &&\n            !convert.not(Q = convert.to(q))\n        )\n            ? this.operation(P, Q)\n            : this.operation(p, q);\n    },\n\n    toJSON: function(options) { // eslint-disable-line no-unused-vars\n        var state = {};\n        if (this.editor) {\n            state.editor = this.editor;\n        }\n        for (var key in this.controls) {\n            state[key] = this[key];\n        }\n        if (!this.parent.nodeFields && this.fields !== this.parent.fields) {\n            state.fields = this.fields;\n        }\n        return state;\n    },\n\n    toSQL: function() {\n        return [\n            this.SQL_QUOTED_IDENTIFIER + this.column + this.SQL_QUOTED_IDENTIFIER,\n            this.sqlOperator,\n            ' \\'' + this.argument.replace(/'/g, '\\'\\'') + '\\''\n        ].join(' ');\n    }\n});\n\nfunction findField(fields, name) {\n    var complex, simple;\n\n    simple = fields.find(function(field) {\n        if ((field.options || field) instanceof Array) {\n            return (complex = findField(field.options || field, name));\n        } else {\n            return field.name === name;\n        }\n    });\n\n    return complex || simple;\n}\n\n/** `change` or `click` event handler for all form controls.\n * Removes error CSS class from control.\n * Adds warning CSS class from control if blank; removes if not blank.\n * Moves focus to next non-blank sibling control.\n */\nfunction cleanUpAndMoveOn(evt) {\n    var el = evt.target;\n\n    // remove `error` CSS class, which may have been added by `FilterLeaf.prototype.validate`\n    el.classList.remove('filter-tree-error');\n\n    // set or remove 'warning' CSS class, as per el.value\n    FilterNode.setWarningClass(el);\n\n    // find next sibling control, if any\n    if (!el.multiple && el.value) {\n        while ((el = el.nextElementSibling) && (!('name' in el) || el.value.trim() !== '')); // eslint-disable-line curly\n    }\n\n    // and click in it (opens select list)\n    if (el && el.value.trim() === '') {\n        el.value = ''; // rid of any white space\n        FilterNode.clickIn(el);\n    }\n}\n\nfunction focusOn(el) {\n    setTimeout(function() {\n        el.classList.add('filter-tree-error');\n        FilterNode.clickIn(el);\n    }, 0);\n}\n\nfunction controlValue(el) {\n    var value, i;\n\n    switch (el.type) {\n        case 'checkbox':\n        case 'radio':\n            el = document.querySelectorAll('input[name=\\'' + el.name + '\\']:enabled:checked');\n            for (value = [], i = 0; i < el.length; i++) {\n                value.push(el[i].value);\n            }\n            break;\n\n        case 'select-multiple':\n            el = el.options;\n            for (value = [], i = 0; i < el.length; i++) {\n                if (!el.disabled && el.selected) {\n                    value.push(el[i].value);\n                }\n            }\n            break;\n\n        default:\n            value = el.value;\n    }\n\n    return value;\n}\n\n/**\n * @summary Creates a new element and adds options to it.\n * @param {string} tagName - Must be one of:\n * * `'input'` for a text box\n * * `'select'` for a drop-down\n * * `'optgroup'` (for internal use only)\n * @param {fieldOption[]} [options] - Strings to add as `<option>...</option>` elements. Omit to create a text box.\n * @param {null|string} [prompt=''] - Adds an initial `<option>...</option>` element to the drop-down with this value in parentheses as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses.\n * @returns {Element} Either a `<select>` or `<optgroup>` element.\n */\nfunction addOptions(tagName, options, prompt) {\n    var el = document.createElement(tagName);\n\n    if (options) {\n        var add, newOption;\n        if (tagName === 'select') {\n            add = el.add;\n            if (prompt) {\n                newOption = new Option('(' + prompt, '');\n                newOption.innerHTML += '&hellip;)';\n                el.add(newOption);\n            } else if (prompt !== null) {\n                el.add(new Option());\n            }\n        } else {\n            add = el.appendChild;\n            el.label = prompt;\n        }\n\n        options = options.slice().sort(fieldComparator); // clone it and sort the clone\n\n        options.forEach(function(option) {\n            if ((option.options || option) instanceof Array) {\n                var optgroup = addOptions('optgroup', option.options || option, option.label);\n                el.add(optgroup);\n            } else {\n                var newElement = typeof option !== 'object'\n                    ? new Option(option)\n                    : new Option(\n                        option.alias || option.name,\n                        option.name || option.alias\n                    );\n                add.call(el, newElement);\n            }\n        });\n    } else {\n        el.type = 'text';\n    }\n\n    return el;\n}\n\nfunction fieldComparator(a, b) {\n    a = a.alias || a.name || a.label || a;\n    b = b.alias || b.name || b.label || b;\n    return a < b ? -1 : a > b ? 1 : 0;\n}\n\nmodule.exports = FilterLeaf;\n","/* eslint-env browser */\n\n'use strict';\n\nvar extend = require('extend-me');\nvar Base = extend.Base;\n\nvar template = require('./template');\n\nextend.debug = true;\n\nvar CHILDREN_TAG = 'OL',\n    CHILD_TAG = 'LI';\n\n/**\n * @constructor\n *\n * @description A filter tree represents a _complex conditional expression_ and consists of a single `FilterNode` object serving as the _root_ of an _n_-ary tree.\n *\n * Each `FilterNode` represents a node in tree. Each node is one of two types of objects extended from `FilterNode`:\n *\n * * The non-terminal (@link FilterTree} nodes represent _complex subexpressions_, each consisting of two or more _conditional_ (boolean expressions), all concatenated together with one of the _tree operators_.\n * * The terminal {@link FilterLeaf} nodes represent _simple expressions_.\n *\n * Tree operators currently include **_AND_** (labeled \"all\" in the UI; and \"op-and\" internally), **_OR_** (\"any\"; \"op-or\"), and **_NOR_** (\"none\"; \"op-nor\").\n *\n * Each conditional in a _subexpression_ (non-terminal node) is represented by a child node which may be either a _simple expression_ (terminal node) or another (\"nested\") subexpression non-terminal node.\n *\n * The `FilterLeaf` object is the default type of simple expression, which is in the form _field-property operator-property argument-property_ where:\n *\n * * _field-property_ - the name of a column, selected from a drop-down;\n * * _operator-property_ - an equality (=), inequality (<, ≤, ≠, ≥, >), or pattern operator (LIKE, NOT LIKE), also selected from a drop-down; and\n * * _argument-property_ is a constant typed into a text box.\n *\n * The `FilterTree` object has polymorphic methods that operate on the entire tree using recursion. When the recursion reaches a terminal node, it calls the methods on the `FilterLeaf` object instead. Calling `test()` on the root tree therefore returns a boolean that determines if the row passes through the entire filter expression (`true`) or is blocked by it (`false`).\n *\n * The programmer may define a new type of simple expression by extending from `FilterLeaf`. An example is the `FilterField` object. Such an implementation must include methods:\n *\n * * Save and subsequently reload the state of the conditional as entered by the user (`toJSON()` and `fromJSON()`, respectively).\n * * Create the DOM objects that represent the UI filter editor and render them to the UI (`newView()` and `render()`, respectively).\n * * Filter a table by implementing one or more of the following:\n *   * Apply the conditional logic to available table row data (`test()`).\n *   * Apply the conditional logic to a remote data-store by generating a **SQL** or **Q** _WHERE_ clause (`toSQL()` and `toQ()`, respectively).\n *\n * Some of the above-named methods as already implemented in `FilterLeaf` and/or `FilterNode` may be sufficient to handle your needs as is (without further code).\n *\n * @param {string[]} [options.fields] - A default list of column names for field drop-downs of all descendant terminal nodes. Overrides `options.state.fields` (see). May be defined for any node and pertains to all descendants of that node (including terminal nodes). If omitted (and no `nodeFields`), will use the nearest ancestor `fields` definition. However, descendants with their own definition of `types` will override any ancestor definition.\n *\n * > Typically only used by the caller for the top-level (root) tree.\n *\n * @param {string[]} [options.nodeFields] - A default list of column names for field drop-downs of immediate descendant terminal nodes _only_. Overrides `options.state.nodeFields` (see).\n *\n * Although both `options.fields` and `options.nodeFields` are notated as optional herein, by the time a terminal node tries to render a fields drop-down, a `fields` list _must_ be defined through (in order of priority):\n *\n * * Terminal node's own `options.fields` (or `options.state.fields`) definition.\n * * Terminal node's parent node's `option.nodeFields` (or `option.state.nodesFields`) definition.\n * * Any of terminal node's ancestor's `options.fields` (or `options.state.fields`) definition.\n *\n * @param {object} [options.state] - A data structure that describes a tree, subtree, or leaf:\n *\n * * May describe a terminal node with properties:\n *   * `fields` - Overridden on instantiation by `options.fields`. If both unspecified, uses parent's definition.\n *   * `editor` - A string identifying the type of conditional. Must be in the tree's (see {@link FilterTree#editors|editors}) hash. If omitted, defaults to `'Default'`.\n *   * misc. - Other properties peculiar to this filter type (but typically including at least a `field` property).\n * * May describe a non-terminal node with properties:\n *   * `fields` - Overridden on instantiation by `options.fields`. If both unspecified, uses parent's definition.\n *   * `operator` - One of {@link treeOperators}.\n *   * `children` -  Array containing additional terminal and non-terminal nodes.\n *\n * If this `options.state` object is omitted altogether, loads an empty filter, which is a `FilterTree` node consisting the default `operator` value (`'op-and'`).\n *\n * > Note that this is a JSON object; not a JSON string (_i.e.,_ \"parsed\"; not \"stringified\").\n *\n * @param {function} [options.editor='Default'] - Type of simple expression.\n *\n * @param {FilterTree} [options.parent] - Used internally to insert element when creating nested subtrees. For the top level tree, you don't give a value for `parent`; you are responsible for inserting the top-level `.el` into the DOM.\n */\nvar FilterNode = Base.extend({\n\n    initialize: function(options) {\n        var parent = options && options.parent,\n            state = options && options.state;\n\n        this.parent = parent;\n\n        /** Default list of fields only for direct child terminal-node drop-downs.\n         * @type {string[]}\n         * @memberOf FilterNode.prototype\n         */\n        this.nodeFields = setOption('nodeFields', options, state);\n\n        /** Default list of fields for all descending terminal-node drop-downs.\n         * @type {string[]}\n         * @memberOf FilterNode.prototype\n         */\n        this.fields = setOption('fields', options, state, parent);\n\n        /** Type of filter editor.\n         * @type {string}\n         * @memberOf FilterNode.prototype\n         */\n        this.editor = setOption('editor', options, state, parent);\n\n        this.fromJSON(state);\n    },\n\n    /** Insert each subtree into its parent node along with a \"delete\" button.\n     * > The root tree is has no parent and is inserted into the DOM by the instantiating code (without a delete button).\n     */\n    render: function() {\n        if (this.parent) {\n            var newListItem = document.createElement(CHILD_TAG);\n            newListItem.appendChild(template('removeButton'));\n            newListItem.appendChild(this.el);\n            this.parent.el.querySelector(CHILDREN_TAG).appendChild(newListItem);\n        }\n    },\n\n    toJSON: function toJSON() {\n        var state = {};\n\n        if (this.toJsonOptions) {\n            var tree = this, metadata = [];\n            if (this.toJsonOptions.fields) {\n                metadata.push('fields');\n                metadata.push('nodeFields');\n            }\n            if (this.toJsonOptions.editor) {\n                metadata.push('editor');\n            }\n            metadata.forEach(function(prop) {\n                if (!tree.parent || tree[prop] && tree[prop] !== tree.parent[prop]) {\n                    state[prop] = tree[prop];\n                }\n            });\n        }\n\n        return state;\n    },\n\n    fromJSON: function(state) {\n        var oldEl = this.el;\n        this.newView();\n        this.load(state);\n        this.render();\n        if (oldEl && !this.parent) {\n            oldEl.parentNode.replaceChild(this.el, oldEl);\n        }\n    },\n\n    SQL_QUOTED_IDENTIFIER: '\"'\n\n});\n\nfunction setOption(key, options, state, parent) {\n    return (\n        options && options[key] ||\n        state && state[key] ||\n        parent && parent[key] // reference parent value now so we don't have to search up the tree later\n    );\n}\n\nFilterNode.setWarningClass = function(el, value) {\n    if (arguments.length < 2) {\n        value = el.value;\n    }\n    el.classList[value ? 'remove' : 'add']('filter-tree-warning');\n    return value;\n\n};\n\nFilterNode.Error = function(msg) {\n    return new Error('filter-tree: ' + msg);\n};\n\nFilterNode.clickIn = function(el) {\n    if (el) {\n        if (el.tagName === 'SELECT') {\n            setTimeout(function() { el.dispatchEvent(new MouseEvent('mousedown')); }, 0);\n        } else {\n            el.focus();\n        }\n    }\n};\n\nmodule.exports = FilterNode;\n","/* eslint-env browser */\n\n'use strict';\n\nvar templex = require('templex');\n\nvar templates = {\n\n    tree: function() {\n        /*\n        <span class=\"filter-tree\">\n            Match\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-or\">any</label>\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-and\">all</label>\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-nor\">none</label>\n            of:<br/>\n            <span class=\"filter-tree-add-filter\" title=\"Add a new conditional to this match.\">\n                <div></div>conditional\n            </span>\n            <span class=\"filter-tree-add\" title=\"Add a new sub-match under this match.\">\n                <div></div>subexpression\n            </span>\n            <ol></ol>\n        </span>\n        */\n    },\n\n    removeButton: function() {\n        /*\n        <div class=\"filter-tree-remove\" title=\"delete conditional\"></div>\n        */\n    },\n\n    note: function() {\n        /*\n        <div class=\"footnotes\">\n            <em>Note regarding the above expression:</em>\n            <span></span>\n            Select a new value or delete the expression altogether.\n        </div>\n        */\n    },\n\n    notes: function() {\n        /*\n         <div class=\"footnotes\">\n            <em>Notes regarding the above expression:</em>\n            <ol></ol>\n            Select new values or delete the expression altogether.\n         </div>\n         */\n    },\n\n    optionMissing: function() {\n        /*\n        The previous <span class=\"field-name\">{1:encode}</span>\n        value <span class=\"field-value\">{2:encode}</span>\n        is no longer valid.\n        */\n    }\n\n};\n\nvar extract = /\\/\\*\\s*([^]+?)\\s+\\*\\//; // finds the string inside the /* ... */; the group excludes the whitespace\nvar encoders = /\\{(\\d+)\\:encode\\}/g;\n\nfunction get(templateName) {\n    var temp = document.createElement('div');\n    var text = templates[templateName].toString().match(extract)[1];\n    var templexArgs = [text].concat(Array.prototype.slice.call(arguments, 1));\n    var keys, encoder = {};\n\n    encoders.lastIndex = 0;\n    while ((keys = encoders.exec(text))) {\n        encoder[keys[1]] = true;\n    }\n    keys = Object.keys(encoder);\n    if (keys.length) {\n        keys.forEach(function(key) {\n            temp.textContent = templexArgs[key];\n            templexArgs[key] = temp.innerHTML;\n        });\n        templexArgs[0] = text.replace(encoders, '{$1}');\n    }\n\n    temp.innerHTML = templex.apply(this, templexArgs);\n\n    // if only one HTMLElement, return it; otherwise entire list of nodes\n    return temp.children.length === 1 && temp.childNodes.length === 1 ? temp.firstChild : temp.childNodes;\n}\n\nmodule.exports = get;\n","'use strict';\n\n/** @typedef {function} operationReducer\n * @param {boolean} p\n * @param {boolean} q\n * @returns {boolean} The result of applying the operator to the two parameters.\n */\n\n/**\n * @private\n * @type {operationReducer}\n */\nfunction AND(p, q) {\n    return p && q;\n}\n\n/**\n * @private\n * @type {operationReducer}\n */\nfunction OR(p, q) {\n    return p || q;\n}\n\n/** @typedef {obejct} treeOperator\n * @desc Each `treeOperator` object describes two things:\n *\n * 1. How to take the test results of _n_ child nodes by applying the operator to all the results to \"reduce\" it down to a single result.\n * 2. How to generate SQL WHERE clause syntax that applies the operator to _n_ child nodes.\n *\n * @property {operationReducer} reduce\n * @property {boolean} seed -\n * @property {boolean} abort -\n * @property {boolean} negate -\n * @property {string} SQL.op -\n * @property {string} SQL.beg -\n * @property {string} SQL.end -\n */\n\n/** A hash of {@link treeOperator} objects.\n * @type {object}\n */\nvar treeOperators = {\n    'op-and': {\n        reduce: AND,\n        seed: true,\n        abort: false,\n        negate: false,\n        SQL: {\n            op: 'AND',\n            beg: '(',\n            end: ')'\n        }\n    },\n    'op-or': {\n        reduce: OR,\n        seed: false,\n        abort: true,\n        negate: false,\n        SQL: {\n            op: 'OR',\n            beg: '(',\n            end: ')'\n        }\n    },\n    'op-nor': {\n        reduce: OR,\n        seed: false,\n        abort: true,\n        negate: true,\n        SQL: {\n            op: 'OR',\n            beg: 'NOT  (',\n            end: ')'\n        }\n    }\n};\n\nmodule.exports = treeOperators;\n","'use strict';\n\n/* eslint-env node, browser */\n\nvar cssInjector = require('css-injector');\n\n/**\n * @constructor FinBar\n * @summary Create a scrollbar object.\n * @desc Creating a scrollbar is a three-step process:\n *\n * 1. Instantiate the scrollbar object by calling this constructor function. Upon instantiation, the DOM element for the scrollbar (with a single child element for the scrollbar \"thumb\") is created but is not insert it into the DOM.\n * 2. After instantiation, it is the caller's responsibility to insert the scrollbar, {@link FinBar#bar|this.bar}, into the DOM.\n * 3. After insertion, the caller must call {@link FinBar#resize|resize()} at least once to size and position the scrollbar and its thumb. After that, `resize()` should also be called repeatedly on resize events (as the content element is being resized).\n *\n * Suggested configurations:\n * * _**Unbound**_<br/>\n * The scrollbar serves merely as a simple range (slider) control. Omit both `options.onchange` and `options.content`.\n * * _**Bound to virtual content element**_<br/>\n * Virtual content is projected into the element using a custom event handler supplied by the programmer in `options.onchange`. A typical use case would be to handle scrolling of the virtual content. Other use cases include data transformations, graphics transformations, _etc._\n * * _**Bound to real content**_<br/>\n * Set `options.content` to the \"real\" content element but omit `options.onchange`. This will cause the scrollbar to use the built-in event handler (`this.scrollRealContent`) which implements smooth scrolling of the content element within the container.\n *\n * @param {finbarOptions} [options={}] - Options object. See the type definition for member details.\n */\nfunction FinBar(options) {\n\n    // make bound versions of all the mouse event handler\n    var bound = this._bound = {};\n    for (key in handlersToBeBound) {\n        bound[key] = handlersToBeBound[key].bind(this);\n    }\n\n    /**\n     * @name thumb\n     * @summary The generated scrollbar thumb element.\n     * @desc The thumb element's parent element is always the {@link FinBar#bar|bar} element.\n     *\n     * This property is typically referenced internally only. The size and position of the thumb element is maintained by `_calcThumb()`.\n     * @type {Element}\n     * @memberOf FinBar.prototype\n     */\n    var thumb = document.createElement('div');\n    thumb.classList.add('thumb');\n    thumb.onclick = bound.shortStop;\n    thumb.onmouseover = bound.onmouseover;\n    this.thumb = thumb;\n\n    /**\n     * @name bar\n     * @summary The generated scrollbar element.\n     * @desc The caller inserts this element into the DOM (typically into the content container) and then calls its {@link FinBar#resize|resize()} method.\n     *\n     * Thus the node tree is typically:\n     * * A **content container** element, which contains:\n     *    * The content element(s)\n     *    * This **scrollbar element**, which in turn contains:\n     *        * The **thumb element**\n     *\n     * @type {Element}\n     * @memberOf FinBar.prototype\n     */\n    var bar = document.createElement('div');\n\n    bar.classList.add('finbar-vertical');\n\n    bar.appendChild(thumb);\n    if (this.paging) {\n        bar.onclick = bound.onclick;\n    }\n    this.bar = bar;\n\n    options = options || {};\n\n    // presets\n    this.orientation = 'vertical';\n    this._min = this._index = 0;\n    this._max = 100;\n\n    // options\n    for (var key in options) {\n        if (options.hasOwnProperty(key)) {\n            var option = options[key];\n            switch (key) {\n\n            case 'index':\n                this._index = option;\n                break;\n\n            case 'range':\n                validRange(option);\n                this._min = option.min;\n                this._max = option.max;\n                this.contentSize = option.max - option.min + 1;\n                break;\n\n            default:\n                if (\n                    key.charAt(0) !== '_' &&\n                    typeof FinBar.prototype[key] !== 'function'\n                ) {\n                    // override prototype defaults for standard ;\n                    // extend with additional properties (for use in onchange event handlers)\n                    this[key] = option;\n                }\n                break;\n\n            }\n        }\n    }\n\n    cssInjector(cssFinBars, 'finbar-base', options.cssStylesheetReferenceElement);\n}\n\nFinBar.prototype = {\n\n    /**\n     * @summary The scrollbar orientation.\n     * @desc Set by the constructor to either `'vertical'` or `'horizontal'`. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * Useful values are `'vertical'` (the default) or `'horizontal'`.\n     *\n     * Setting this property resets `this.oh` and `this.deltaProp` and changes the class names so as to reposition the scrollbar as per the CSS rules for the new orientation.\n     * @default 'vertical'\n     * @type {string}\n     * @memberOf FinBar.prototype\n     */\n    set orientation(orientation) {\n        if (orientation === this._orientation) {\n            return;\n        }\n\n        this._orientation = orientation;\n\n        /**\n         * @readonly\n         * @name oh\n         * @summary <u>O</u>rientation <u>h</u>ash for this scrollbar.\n         * @desc Set by the `orientation` setter to either the vertical or the horizontal orientation hash. The property should always be synchronized with `orientation`; do not update directly!\n         *\n         * This object is used internally to access scrollbars' DOM element properties in a generalized way without needing to constantly query the scrollbar orientation. For example, instead of explicitly coding `this.bar.top` for a vertical scrollbar and `this.bar.left` for a horizontal scrollbar, simply code `this.bar[this.oh.leading]` instead. See the {@link orientationHashType} definition for details.\n         *\n         * This object is useful externally for coding generalized {@link finbarOnChange} event handler functions that serve both horizontal and vertical scrollbars.\n         * @type {orientationHashType}\n         * @memberOf FinBar.prototype\n         */\n        this.oh = orientationHashes[this._orientation];\n\n        if (!this.oh) {\n            error('Invalid value for `options._orientation.');\n        }\n\n        /**\n         * @name deltaProp\n         * @summary The name of the `WheelEvent` property this scrollbar should listen to.\n         * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n         *\n         * Useful values are `'deltaX'`, `'deltaY'`, or `'deltaZ'`. A value of `null` means to ignore mouse wheel events entirely.\n         *\n         * The mouse wheel is one-dimensional and only emits events with `deltaY` data. This property is provided so that you can override the default of `'deltaX'` with a value of `'deltaY'` on your horizontal scrollbar primarily to accommodate certain \"panoramic\" interface designs where the mouse wheel should control horizontal rather than vertical scrolling. Just give `{ deltaProp: 'deltaY' }` in your horizontal scrollbar instantiation.\n         *\n         * Caveat: Note that a 2-finger drag on an Apple trackpad emits events with _both_ `deltaX ` and `deltaY` data so you might want to delay making the above adjustment until you can determine that you are getting Y data only with no X data at all (which is a sure bet you on a mouse wheel rather than a trackpad).\n\n         * @type {object|null}\n         * @memberOf FinBar.prototype\n         */\n        this.deltaProp = this.oh.delta;\n\n        this.bar.className = this.bar.className.replace(/(vertical|horizontal)/g, orientation);\n\n        if (this.bar.style.cssText || this.thumb.style.cssText) {\n            this.bar.removeAttribute('style');\n            this.thumb.removeAttribute('style');\n            this.resize();\n        }\n    },\n    get orientation() {\n        return this._orientation;\n    },\n\n    /**\n     * @summary Callback for scroll events.\n     * @desc Set by the constructor via the similarly named property in the {@link finbarOptions} object. After instantiation, `this.onchange` may be updated directly.\n     *\n     * This event handler is called whenever the value of the scrollbar is changed through user interaction. The typical use case is when the content is scrolled. It is called with the `FinBar` object as its context and the current value of the scrollbar (its index, rounded) as the only parameter.\n     *\n     * Set this property to `null` to stop emitting such events.\n     * @type {function(number)|null}\n     * @memberOf FinBar.prototype\n     */\n    onchange: null,\n\n    /**\n     * @summary Add a CSS class name to the bar element's class list.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * The bar element's class list will always include `finbar-vertical` (or `finbar-horizontal` based on the current orientation). Whenever this property is set to some value, first the old prefix+orientation is removed from the bar element's class list; then the new prefix+orientation is added to the bar element's class list. This property causes _an additional_ class name to be added to the bar element's class list. Therefore, this property will only add at most one additional class name to the list.\n     *\n     * To remove _classname-orientation_ from the bar element's class list, set this property to a falsy value, such as `null`.\n     *\n     * > NOTE: You only need to specify an additional class name when you need to have mulltiple different styles of scrollbars on the same page. If this is not a requirement, then you don't need to make a new class; you would just create some additional rules using the same selectors in the built-in stylesheet (../css/finbars.css):\n     * *`div.finbar-vertical` (or `div.finbar-horizontal`) for the scrollbar\n     * *`div.finbar-vertical > div` (or `div.finbar-horizontal > div`) for the \"thumb.\"\n     *\n     * Of course, your rules should come after the built-ins.\n     * @type {string}\n     * @memberOf FinBar.prototype\n     */\n    set classPrefix(prefix) {\n        if (this._classPrefix) {\n            this.bar.classList.remove(this._classPrefix + this.orientation);\n        }\n\n        this._classPrefix = prefix;\n\n        if (prefix) {\n            this.bar.classList.add(prefix + '-' + this.orientation);\n        }\n    },\n    get classPrefix() {\n        return this._classPrefix;\n    },\n\n    /**\n     * @name increment\n     * @summary Number of scrollbar index units representing a pageful. Used exclusively for paging up and down and for setting thumb size relative to content size.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * Can also be given as a parameter to the {@link FinBar#resize|resize} method, which is pertinent because content area size changes affect the definition of a \"pageful.\" However, you only need to do this if this value is being used. It not used when:\n     * * you define `paging.up` and `paging.down`\n     * * your scrollbar is using `scrollRealContent`\n     * @type {number}\n     * @memberOf FinBar.prototype\n     */\n    increment: 1,\n\n    /**\n     * @name barStyles\n     * @summary Scrollbar styles to be applied by {@link FinBar#resize|resize()}.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * This is a value to be assigned to {@link FinBar#styles|styles} on each call to {@link FinBar#resize|resize()}. That is, a hash of values to be copied to the scrollbar element's style object on resize; or `null` for none.\n     *\n     * @see {@link FinBar#style|style}\n     * @type {finbarStyles|null}\n     * @memberOf FinBar.prototype\n     */\n    barStyles: null,\n\n    /**\n     * @name style\n     * @summary Additional scrollbar styles.\n     * @desc See type definition for more details. These styles are applied directly to the scrollbar's `bar` element.\n     *\n     * Values are adjusted as follows before being applied to the element:\n     * 1. Included \"pseudo-property\" names from the scrollbar's orientation hash, {@link FinBar#oh|oh}, are translated to actual property names before being applied.\n     * 2. When there are margins, percentages are translated to absolute pixel values because CSS ignores margins in its percentage calculations.\n     * 3. If you give a value without a unit (a raw number), \"px\" unit is appended.\n     *\n     * General notes:\n     * 1. It is always preferable to specify styles via a stylesheet. Only set this property when you need to specifically override (a) stylesheet value(s).\n     * 2. Can be set directly or via calls to the {@link FinBar#resize|resize} method.\n     * 3. Should only be set after the scrollbar has been inserted into the DOM.\n     * 4. Before applying these new values to the element, _all_ in-line style values are reset (by removing the element's `style` attribute), exposing inherited values (from stylesheets).\n     * 5. Empty object has no effect.\n     * 6. Falsey value in place of object has no effect.\n     *\n     * > CAVEAT: Do not attempt to treat the object you assign to this property as if it were `this.bar.style`. Specifically, changing this object after assigning it will have no effect on the scrollbar. You must assign it again if you want it to have an effect.\n     *\n     * @see {@link FinBar#barStyles|barStyles}\n     * @type {finbarStyles}\n     * @memberOf FinBar.prototype\n     */\n    set style(styles) {\n        var keys = Object.keys(styles = extend({}, styles, this._auxStyles));\n\n        if (keys.length) {\n            var bar = this.bar,\n                barRect = bar.getBoundingClientRect(),\n                container = this.container || bar.parentElement,\n                containerRect = container.getBoundingClientRect(),\n                oh = this.oh;\n\n            // Before applying new styles, revert all styles to values inherited from stylesheets\n            bar.removeAttribute('style');\n\n            keys.forEach(function (key) {\n                var val = styles[key];\n\n                if (key in oh) {\n                    key = oh[key];\n                }\n\n                if (!isNaN(Number(val))) {\n                    val = (val || 0) + 'px';\n                } else if (/%$/.test(val)) {\n                    // When bar size given as percentage of container, if bar has margins, restate size in pixels less margins.\n                    // (If left as percentage, CSS's calculation will not exclude margins.)\n                    var oriented = axis[key],\n                        margins = barRect[oriented.marginLeading] + barRect[oriented.marginTrailing];\n                    if (margins) {\n                        val = parseInt(val, 10) / 100 * containerRect[oriented.size] - margins + 'px';\n                    }\n                }\n\n                bar.style[key] = val;\n            });\n        }\n    },\n\n    /**\n     * @readonly\n     * @name paging\n     * @summary Enable page up/dn clicks.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * If truthy, listen for clicks in page-up and page-down regions of scrollbar.\n     *\n     * If an object, call `.paging.up()` on page-up clicks and `.paging.down()` will be called on page-down clicks.\n     *\n     * Changing the truthiness of this value after instantiation currently has no effect.\n     * @type {boolean|object}\n     * @memberOf FinBar.prototype\n     */\n    paging: true,\n\n    /**\n     * @name range\n     * @summary Setter for the minimum and maximum scroll values.\n     * @desc Set by the constructor. These values are the limits for {@link FooBar#index|index}.\n     *\n     * The setter accepts an object with exactly two numeric properties: `.min` which must be less than `.max`. The values are extracted and the object is discarded.\n     *\n     * The getter returns a new object with `.min` and '.max`.\n     *\n     * @type {rangeType}\n     * @memberOf FinBar.prototype\n     */\n    set range(range) {\n        validRange(range);\n        this._min = range.min;\n        this._max = range.max;\n        this.contentSize = range.max - range.min + 1;\n        this.index = this.index; // re-clamp\n    },\n    get range() {\n        return {\n            min: this._min,\n            max: this._max\n        };\n    },\n\n    /**\n     * @summary Index value of the scrollbar.\n     * @desc This is the position of the scroll thumb.\n     *\n     * Setting this value clamps it to {@link FinBar#min|min}..{@link FinBar#max|max}, scroll the content, and moves thumb.\n     *\n     * Getting this value returns the current index. The returned value will be in the range `min`..`max`. It is intentionally not rounded.\n     *\n     * Use this value as an alternative to (or in addition to) using the {@link FinBar#onchange|onchange} callback function.\n     *\n     * @see {@link FinBar#_setScroll|_setScroll}\n     * @type {number}\n     * @memberOf FinBar.prototype\n     */\n    set index(idx) {\n        idx = Math.min(this._max, Math.max(this._min, idx)); // clamp it\n        this._setScroll(idx);\n        // this._setThumbSize();\n    },\n    get index() {\n        return this._index;\n    },\n\n    /**\n     * @private\n     * @summary Move the thumb.\n     * @desc Also displays the index value in the test panel and invokes the callback.\n     * @param idx - The new scroll index, a value in the range `min`..`max`.\n     * @param [scaled=f(idx)] - The new thumb position in pixels and scaled relative to the containing {@link FinBar#bar|bar} element, i.e., a proportional number in the range `0`..`thumbMax`. When omitted, a function of `idx` is used.\n     * @memberOf FinBar.prototype\n     */\n    _setScroll: function (idx, scaled) {\n        this._index = idx;\n\n        // Display the index value in the test panel\n        if (this.testPanelItem && this.testPanelItem.index instanceof Element) {\n            this.testPanelItem.index.innerHTML = Math.round(idx);\n        }\n\n        // Call the callback\n        if (this.onchange) {\n            this.onchange.call(this, Math.round(idx));\n        }\n\n        // Move the thumb\n        if (scaled === undefined) {\n            scaled = (idx - this._min) / (this._max - this._min) * this._thumbMax;\n        }\n        this.thumb.style[this.oh.leading] = scaled + 'px';\n    },\n\n    scrollRealContent: function (idx) {\n        var containerRect = this.content.parentElement.getBoundingClientRect(),\n            sizeProp = this.oh.size,\n            maxScroll = Math.max(0, this.content[sizeProp] - containerRect[sizeProp]),\n            //scroll = Math.min(idx, maxScroll);\n            scroll = (idx - this._min) / (this._max - this._min) * maxScroll;\n        //console.log('scroll: ' + scroll);\n        this.content.style[this.oh.leading] = -scroll + 'px';\n    },\n\n    /**\n     * @summary Recalculate thumb position.\n     *\n     * @desc This method recalculates the thumb size and position. Call it once after inserting your scrollbar into the DOM, and repeatedly while resizing the scrollbar (which typically happens when the scrollbar's parent is resized by user.\n     *\n     * > This function shifts args if first arg omitted.\n     *\n     * @param {number} [increment=this.increment] - Resets {@link FooBar#increment|increment} (see).\n     *\n     * @param {finbarStyles} [barStyles=this.barStyles] - (See type definition for details.) Scrollbar styles to be applied to the bar element.\n     *\n     * Only specify a `barStyles` object when you need to override stylesheet values. If provided, becomes the new default (`this.barStyles`), for use as a default on subsequent calls.\n     *\n     * It is generally the case that the scrollbar's new position is sufficiently described by the current styles. Therefore, it is unusual to need to provide a `barStyles` object on every call to `resize`.\n     *\n     * @returns {FinBar} Self for chaining.\n     * @memberOf FinBar.prototype\n     */\n    resize: function (increment, barStyles) {\n        var bar = this.bar;\n\n        if (!bar.parentNode) {\n            return; // not in DOM yet so nothing to do\n        }\n\n        var container = this.container || bar.parentElement,\n            containerRect = container.getBoundingClientRect();\n\n        // shift args if if 1st arg omitted\n        if (typeof increment === 'object') {\n            barStyles = increment;\n            increment = undefined;\n        }\n\n        this.style = this.barStyles = barStyles || this.barStyles;\n\n        // Bound to real content: Content was given but no onchange handler.\n        // Set up .onchange, .containerSize, and .increment.\n        // Note this only makes sense if your index unit is pixels.\n        if (this.content) {\n            if (!this.onchange) {\n                this.onchange = this.scrollRealContent;\n                this.contentSize = this.content[this.oh.size];\n                this._min = 0;\n                this._max = this.contentSize - 1;\n            }\n        }\n        if (this.onchange === this.scrollRealContent) {\n            this.containerSize = containerRect[this.oh.size];\n            this.increment = this.containerSize / (this.contentSize - this.containerSize) * (this._max - this._min);\n        } else {\n            this.containerSize = 1;\n            this.increment = increment || this.increment;\n        }\n\n        var index = this.index;\n        this.testPanelItem = this.testPanelItem || this._addTestPanelItem();\n        this._setThumbSize();\n        this.index = index;\n\n        if (this.deltaProp !== null) {\n            container.addEventListener('wheel', this._bound.onwheel);\n        }\n\n        return this;\n    },\n\n    /**\n     * @summary Shorten trailing end of scrollbar by thickness of some other scrollbar.\n     * @desc In the \"classical\" scenario where vertical scroll bar is on the right and horizontal scrollbar is on the bottom, you want to shorten the \"trailing end\" (bottom and right ends, respectively) of at least one of them so they don't overlay.\n     *\n     * This convenience function is an programmatic alternative to hardcoding the correct style with the correct value in your stylesheet; or setting the correct style with the correct value in the {@link FinBar#barStyles|barStyles} object.\n     *\n     * @see {@link FinBar#foreshortenBy|foreshortenBy}.\n     *\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    shortenBy: function (otherFinBar) { return this.shortenEndBy('trailing', otherFinBar); },\n\n    /**\n     * @summary Shorten leading end of scrollbar by thickness of some other scrollbar.\n     * @desc Supports non-classical scrollbar scenarios where vertical scroll bar may be on left and horizontal scrollbar may be on top, in which case you want to shorten the \"leading end\" rather than the trailing end.\n     * @see {@link FinBar#shortenBy|shortenBy}.\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    foreshortenBy: function (otherFinBar) { return this.shortenEndBy('leading', otherFinBar); },\n\n    /**\n     * @summary Generalized shortening function.\n     * @see {@link FinBar#shortenBy|shortenBy}.\n     * @see {@link FinBar#foreshortenBy|foreshortenBy}.\n     * @param {string} whichEnd - a CSS style property name or an orientation hash name that translates to a CSS style property name.\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    shortenEndBy: function (whichEnd, otherFinBar) {\n        if (!otherFinBar) {\n            delete this._auxStyles;\n        } else if (otherFinBar instanceof FinBar && otherFinBar.orientation !== this.orientation) {\n            var otherStyle = window.getComputedStyle(otherFinBar.bar),\n                ooh = orientationHashes[otherFinBar.orientation];\n            this._auxStyles = {};\n            this._auxStyles[whichEnd] = otherStyle[ooh.thickness];\n        }\n        return this; // for chaining\n    },\n\n    /**\n     * @private\n     * @summary Sets the proportional thumb size and hides thumb when 100%.\n     * @desc The thumb size has an absolute minimum of 20 (pixels).\n     * @memberOf FinBar.prototype\n     */\n    _setThumbSize: function () {\n        var oh = this.oh,\n            thumbComp = window.getComputedStyle(this.thumb),\n            thumbMarginLeading = parseInt(thumbComp[oh.marginLeading]),\n            thumbMarginTrailing = parseInt(thumbComp[oh.marginTrailing]),\n            thumbMargins = thumbMarginLeading + thumbMarginTrailing,\n            barSize = this.bar.getBoundingClientRect()[oh.size],\n            thumbSize = Math.max(20, barSize * this.containerSize / this.contentSize);\n\n        if (this.containerSize < this.contentSize) {\n            this.bar.style.visibility = 'visible';\n            this.thumb.style[oh.size] = thumbSize + 'px';\n        } else {\n            this.bar.style.visibility = 'hidden';\n        }\n\n        /**\n         * @private\n         * @name _thumbMax\n         * @summary Maximum offset of thumb's leading edge.\n         * @desc This is the pixel offset within the scrollbar of the thumb when it is at its maximum position at the extreme end of its range.\n         *\n         * This value takes into account the newly calculated size of the thumb element (including its margins) and the inner size of the scrollbar (the thumb's containing element, including _its_ margins).\n         *\n         * NOTE: Scrollbar padding is not taken into account and assumed to be 0 in the current implementation and is assumed to be `0`; use thumb margins in place of scrollbar padding.\n         * @type {number}\n         * @memberOf FinBar.prototype\n         */\n        this._thumbMax = barSize - thumbSize - thumbMargins;\n\n        this._thumbMarginLeading = thumbMarginLeading; // used in mousedown\n    },\n\n    /**\n     * @summary Remove the scrollbar.\n     * @desc Unhooks all the event handlers and then removes the element from the DOM. Always call this method prior to disposing of the scrollbar object.\n     * @memberOf FinBar.prototype\n     */\n    remove: function () {\n        this._removeEvt('mousedown');\n        this._removeEvt('mousemove');\n        this._removeEvt('mouseup');\n\n        (this.container || this.bar.parentElement)._removeEvt('wheel', this._bound.onwheel);\n\n        this.bar.onclick =\n            this.thumb.onclick =\n                this.thumb.onmouseover =\n                    this.thumb.transitionend =\n                        this.thumb.onmouseout = null;\n\n        this.bar.remove();\n    },\n\n    /**\n     * @private\n     * @function _addTestPanelItem\n     * @summary Append a test panel element.\n     * @desc If there is a test panel in the DOM (typically an `<ol>...</ol>` element) with class names of both `this.classPrefix` and `'test-panel'` (or, barring that, any element with class name `'test-panel'`), an `<li>...</li>` element will be created and appended to it. This new element will contain a span for each class name given.\n     *\n     * You should define a CSS selector `.listening` for these spans. This class will be added to the spans to alter their appearance when a listener is added with that class name (prefixed with 'on').\n     *\n     * (This is an internal function that is called once by the constructor on every instantiation.)\n     * @returns {Element|undefined} The appended `<li>...</li>` element or `undefined` if there is no test panel.\n     * @memberOf FinBar.prototype\n     */\n    _addTestPanelItem: function () {\n        var testPanelItem,\n            testPanelElement = document.querySelector('.' + this._classPrefix + '.test-panel') || document.querySelector('.test-panel');\n\n        if (testPanelElement) {\n            var testPanelItemPartNames = [ 'mousedown', 'mousemove', 'mouseup', 'index' ],\n                item = document.createElement('li');\n\n            testPanelItemPartNames.forEach(function (partName) {\n                item.innerHTML += '<span class=\"' + partName + '\">' + partName.replace('mouse', '') + '</span>';\n            });\n\n            testPanelElement.appendChild(item);\n\n            testPanelItem = {};\n            testPanelItemPartNames.forEach(function (partName) {\n                testPanelItem[partName] = item.getElementsByClassName(partName)[0];\n            });\n        }\n\n        return testPanelItem;\n    },\n\n    _addEvt: function (evtName) {\n        var spy = this.testPanelItem && this.testPanelItem[evtName];\n        if (spy) { spy.classList.add('listening'); }\n        window.addEventListener(evtName, this._bound['on' + evtName]);\n    },\n\n    _removeEvt: function (evtName) {\n        var spy = this.testPanelItem && this.testPanelItem[evtName];\n        if (spy) { spy.classList.remove('listening'); }\n        window.removeEventListener(evtName, this._bound['on' + evtName]);\n    }\n};\n\nfunction extend(obj) {\n    for (var i = 1; i < arguments.length; ++i) {\n        var objn = arguments[i];\n        if (objn) {\n            for (var key in objn) {\n                obj[key] = objn[key];\n            }\n        }\n    }\n    return obj;\n}\n\nfunction validRange(range) {\n    var keys = Object.keys(range),\n        valid =  keys.length === 2 &&\n            typeof range.min === 'number' &&\n            typeof range.max === 'number' &&\n            range.min <= range.max;\n\n    if (!valid) {\n        error('Invalid .range object.');\n    }\n}\n\n/**\n * @private\n * @name handlersToBeBound\n * @type {object}\n * @desc The functions defined in this object are all DOM event handlers that are bound by the FinBar constructor to each new instance. In other words, the `this` value of these handlers, once bound, refer to the FinBar object and not to the event emitter. \"Do not consume raw.\"\n */\nvar handlersToBeBound = {\n    shortStop: function (evt) {\n        evt.stopPropagation();\n    },\n\n    onwheel: function (evt) {\n        this.index += evt[this.deltaProp];\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onclick: function (evt) {\n        var thumbBox = this.thumb.getBoundingClientRect(),\n            goingUp = evt[this.oh.coordinate] < thumbBox[this.oh.leading];\n\n        if (typeof this.paging === 'object') {\n            this.index = this.paging[goingUp ? 'up' : 'down'](Math.round(this.index));\n        } else {\n            this.index += goingUp ? -this.increment : this.increment;\n        }\n\n        // make the thumb glow momentarily\n        this.thumb.classList.add('hover');\n        var self = this;\n        this.thumb.addEventListener('transitionend', function waitForIt() {\n            this.removeEventListener('transitionend', waitForIt);\n            self._bound.onmouseup(evt);\n        });\n\n        evt.stopPropagation();\n    },\n\n    onmouseover: function () {\n        this.thumb.classList.add('hover');\n        this.thumb.onmouseout = this._bound.onmouseout;\n        this._addEvt('mousedown');\n    },\n\n    onmouseout: function () {\n        this._removeEvt('mousedown');\n        this.thumb.onmouseover = this._bound.onmouseover;\n        this.thumb.classList.remove('hover');\n    },\n\n    onmousedown: function (evt) {\n        this._removeEvt('mousedown');\n        this.thumb.onmouseover = this.thumb.onmouseout = null;\n\n        var thumbBox = this.thumb.getBoundingClientRect();\n        this.pinOffset = evt[this.oh.axis] - thumbBox[this.oh.leading] + this.bar.getBoundingClientRect()[this.oh.leading] + this._thumbMarginLeading;\n        document.documentElement.style.cursor = 'default';\n\n        this._addEvt('mousemove');\n        this._addEvt('mouseup');\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onmousemove: function (evt) {\n        var scaled = Math.min(this._thumbMax, Math.max(0, evt[this.oh.axis] - this.pinOffset));\n        var idx = scaled / this._thumbMax * (this._max - this._min) + this._min;\n\n        this._setScroll(idx, scaled);\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onmouseup: function (evt) {\n        this._removeEvt('mousemove');\n        this._removeEvt('mouseup');\n\n        document.documentElement.style.cursor = 'auto';\n\n        var thumbBox = this.thumb.getBoundingClientRect();\n        if (\n            thumbBox.left <= evt.clientX && evt.clientX <= thumbBox.right &&\n            thumbBox.top <= evt.clientY && evt.clientY <= thumbBox.bottom\n        ) {\n            this._bound.onmouseover(evt);\n        } else {\n            this._bound.onmouseout(evt);\n        }\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    }\n};\n\nvar orientationHashes = {\n    vertical: {\n        coordinate:     'clientY',\n        axis:           'pageY',\n        size:           'height',\n        outside:        'right',\n        inside:         'left',\n        leading:        'top',\n        trailing:       'bottom',\n        marginLeading:  'marginTop',\n        marginTrailing: 'marginBottom',\n        thickness:      'width',\n        delta:          'deltaY'\n    },\n    horizontal: {\n        coordinate:     'clientX',\n        axis:           'pageX',\n        size:           'width',\n        outside:        'bottom',\n        inside:         'top',\n        leading:        'left',\n        trailing:       'right',\n        marginLeading:  'marginLeft',\n        marginTrailing: 'marginRight',\n        thickness:      'height',\n        delta:          'deltaX'\n    }\n};\n\nvar axis = {\n    top:    'vertical',\n    bottom: 'vertical',\n    height: 'vertical',\n    left:   'horizontal',\n    right:  'horizontal',\n    width:  'horizontal'\n};\n\nvar cssFinBars; // definition inserted by gulpfile between following comments\n/* inject:css */\ncssFinBars = 'div.finbar-horizontal,div.finbar-vertical{position:absolute;margin:3px}div.finbar-horizontal>.thumb,div.finbar-vertical>.thumb{position:absolute;background-color:#d3d3d3;-webkit-box-shadow:0 0 1px #000;-moz-box-shadow:0 0 1px #000;box-shadow:0 0 1px #000;border-radius:4px;margin:2px;opacity:.4;transition:opacity .5s}div.finbar-horizontal>.thumb.hover,div.finbar-vertical>.thumb.hover{opacity:1;transition:opacity .5s}div.finbar-vertical{top:0;bottom:0;right:0;width:11px}div.finbar-vertical>.thumb{top:0;right:0;width:7px}div.finbar-horizontal{left:0;right:0;bottom:0;height:11px}div.finbar-horizontal>.thumb{left:0;bottom:0;height:7px}';\n/* endinject */\n\nfunction error(msg) {\n    throw 'finbars: ' + msg;\n}\n\n// Interface\nmodule.exports = FinBar;\n","/* eslint-env browser */\n\n'use strict';\n\nvar rectangular = require('rectangular');\n\nvar gestures = require('./js/polymergestures.dev.js');\nvar GraphicsContext = require('./js/GraphicsContext.js');\n\nvar RESIZE_POLLING_INTERVAL = 200,\n    paintables = [],\n    resizables = [],\n    paintLoopRunning = true,\n    resizeLoopRunning = true,\n    charMap = makeCharMap();\n\nfunction Canvas(div, component) {\n    var self = this;\n\n    this.div = div;\n    this._component = component;\n\n    this.dragEndtime = Date.now();\n\n    this.canvas = document.createElement('canvas');\n    this.div.appendChild(this.canvas);\n\n    this.canvas.style.outline = 'none';\n\n    // this.focuser = document.createElement('button');\n    // this.focuser.style.position = 'absolute';\n    // this.focuser.style.top = '0px';\n    // this.focuser.style.left = '0px';\n    // this.focuser.style.zIndex = '-1';\n    // this.focuser.style.outline = 'none';\n    // this.div.appendChild(this.focuser);\n\n    this.canvasCTX = this.canvas.getContext('2d');\n    this.gc = new GraphicsContext(this.canvasCTX);\n\n    this.buffer = document.createElement('canvas');\n    this.bufferCTX = this.buffer.getContext('2d');\n    this.bufferGC = new GraphicsContext(this.bufferCTX);\n\n    this.mouseLocation = new rectangular.Point(-1, -1);\n    this.dragstart = new rectangular.Point(-1, -1);\n    //this.origin = new rectangular.Point(0, 0);\n    this.bounds = new rectangular.Rectangle(0, 0, 0, 0);\n    this.hasMouse = false;\n\n    document.addEventListener('mousemove', function(e) {\n        if (self.hasMouse || self.isDragging()) {\n            self.finmousemove(e);\n        }\n    });\n    document.addEventListener('mouseup', function(e) {\n        self.finmouseup(e);\n    });\n    document.addEventListener('wheel', function(e) {\n        self.finwheelmoved(e);\n    });\n    document.addEventListener('keydown', function(e) {\n        self.finkeydown(e);\n    });\n    document.addEventListener('keyup', function(e) {\n        self.finkeyup(e);\n    });\n\n    this.canvas.onmouseover = function() {\n        self.hasMouse = true;\n    };\n    this.canvas.addEventListener('focus', function(e) {\n        self.finfocusgained(e);\n    });\n    this.canvas.addEventListener('blur', function(e) {\n        self.finfocuslost(e);\n    });\n    this.canvas.addEventListener('mousedown', function(e) {\n        self.finmousedown(e);\n    });\n    this.canvas.addEventListener('mouseout', function(e) {\n        self.hasMouse = false;\n        self.finmouseout(e);\n    });\n    this.canvas.addEventListener('click', function(e) {\n        self.finclick(e);\n    });\n    this.canvas.addEventListener('contextmenu', function(e) {\n        self.fincontextmenu(e);\n        e.preventDefault();\n        return false;\n    });\n\n    gestures.addEventListener(this.canvas, 'tap', function(e) {\n        self.fintap(e);\n    });\n    gestures.addEventListener(this.canvas, 'holdpulse', function(e) {\n        self.finholdpulse(e);\n    });\n    gestures.addEventListener(this.canvas, 'flick', function(e) {\n        self.finflick(e);\n    });\n    gestures.addEventListener(this.canvas, 'release', function(e) {\n        self.finrelease(e);\n    });\n    gestures.addEventListener(this.canvas, 'trackstart', function(e) {\n        self.fintrackstart(e);\n    });\n    gestures.addEventListener(this.canvas, 'track', function(e) {\n        self.fintrack(e);\n    });\n    gestures.addEventListener(this.canvas, 'trackend', function(e) {\n        self.fintrackend(e);\n    });\n\n    this.canvas.setAttribute('tabindex', 0);\n    this.canvas.contentEditable = true;\n\n    this.resize();\n\n    this.beginResizing();\n    this.beginPainting();\n}\n\nCanvas.prototype = {\n    constructor: Canvas.prototype.constructor,\n    div: null,\n    _component: null,\n    gestures: gestures, // TODO: why do we need this? (was previously at bottom of file)\n    canvas: null,\n    canvasCTX: null,\n    focuser: null,\n    buffer: null,\n    ctx: null,\n    mouseLocation: null,\n    holdPulseCount: -1,\n    dragstart: null,\n    origin: null,\n    bounds: null,\n    dirty: false,\n    size: null,\n    mousedown: false,\n    dragging: false,\n    repeatKeyCount: 0,\n    repeatKey: null,\n    repeatKeyStartTime: 0,\n    currentKeys: [],\n    hasMouse: false,\n    lastDoubleClickTime: 0,\n    dragEndTime: 0,\n    lastRepaintTime: 0,\n\n    addEventListener: function(name, callback) {\n        this.canvas.addEventListener(name, callback);\n    },\n\n    stopPaintLoop: function() {\n        paintLoopRunning = false;\n    },\n\n    restartPaintLoop: function() {\n        if (paintLoopRunning) {\n            return; // already running\n        }\n        paintLoopRunning = true;\n        requestAnimationFrame(paintLoopFunction);\n    },\n\n    stopResizeLoop: function() {\n        resizeLoopRunning = false;\n    },\n\n    restartResizeLoop: function() {\n        if (resizeLoopRunning) {\n            return; // already running\n        }\n        resizeLoopRunning = true;\n        setInterval(resizablesLoopFunction, 200);\n    },\n\n    detached: function() {\n        this.stopPainting();\n        this.stopResizing();\n    },\n\n    useHiDPI: function() {\n        return this._component.resolveProperty('useHiDPI');\n    },\n\n    useBitBlit: function() {\n        return this._component.resolveProperty('useBitBlit');\n    },\n\n    getFPS: function() {\n        var fps = this._component.resolveProperty('repaintIntervalRate');\n        return fps ? parseInt(fps) : 0;\n    },\n\n    tickPaint: function(now) {\n        var fps = this.getFPS();\n        if (fps === 0) {\n            return;\n        }\n        var interval = 1000 / fps;\n\n        var elapsed = now - this.lastRepaintTime;\n        if (elapsed > interval && this.dirty) {\n            this.lastRepaintTime = now - (elapsed % interval);\n            this.paintNow();\n        }\n    },\n\n    beginPainting: function() {\n        var self = this;\n        this.dirty = true;\n        this.tickPainter = function(now) {\n            self.tickPaint(now);\n        };\n        paintables.push(this);\n    },\n\n    stopPainting: function() {\n        paintables.splice(paintables.indexOf(this), 1);\n    },\n\n    beginResizing: function() {\n        var self = this;\n        this.tickResizer = function() {\n            self.checksize();\n        };\n        resizables.push(this);\n    },\n\n    stopResizing: function() {\n        resizables.splice(resizables.indexOf(this), 1);\n    },\n\n    start: function() {\n        this.beginPainting();\n        this.beginResizing();\n    },\n\n    stop: function() {\n        this.stopPainting();\n        this.stopResizing();\n    },\n\n    checksize: function() {\n        //this is expensive lets do it at some modulo\n        var sizeNow = this.div.getBoundingClientRect();\n        if (sizeNow.width !== this.size.width || sizeNow.height !== this.size.height) {\n            this.sizeChangedNotification();\n        }\n    },\n\n    sizeChangedNotification: function() {\n        this.resize();\n    },\n\n    resize: function() {\n        var box = this.size = this.div.getBoundingClientRect();\n\n        this.canvas.width = this.buffer.width = box.width;\n        this.canvas.height = this.buffer.height = box.height;\n\n        //fix ala sir spinka, see\n        //http://www.html5rocks.com/en/tutorials/canvas/hidpi/\n        //just add 'hdpi' as an attribute to the fin-canvas tag\n        var ratio = 1;\n        var useBitBlit = this.useBitBlit();\n        var isHIDPI = window.devicePixelRatio && this.useHiDPI();\n        if (isHIDPI) {\n            var devicePixelRatio = window.devicePixelRatio || 1;\n            var backingStoreRatio = this.canvasCTX.webkitBackingStorePixelRatio ||\n                this.canvasCTX.mozBackingStorePixelRatio ||\n                this.canvasCTX.msBackingStorePixelRatio ||\n                this.canvasCTX.oBackingStorePixelRatio ||\n                this.canvasCTX.backingStorePixelRatio || 1;\n\n            ratio = devicePixelRatio / backingStoreRatio;\n            //this.canvasCTX.scale(ratio, ratio);\n        }\n        var width = this.canvas.getAttribute('width');\n        var height = this.canvas.getAttribute('height');\n        this.canvas.width = width * ratio;\n        this.canvas.height = height * ratio;\n        this.buffer.width = width * ratio;\n        this.buffer.height = height * ratio;\n\n        this.canvas.style.width = width + 'px';\n        this.canvas.style.height = height + 'px';\n        this.buffer.style.width = width + 'px';\n        this.buffer.style.height = height + 'px';\n\n        this.bufferCTX.scale(ratio, ratio);\n        if (isHIDPI && !useBitBlit) {\n            this.canvasCTX.scale(ratio, ratio);\n        }\n\n        //this.origin = new rectangular.Point(Math.round(this.size.left), Math.round(this.size.top));\n        this.bounds = new rectangular.Rectangle(0, 0, box.width, box.height);\n        //setTimeout(function() {\n        var comp = this._component;\n        if (comp) {\n            comp.setBounds(this.bounds);\n        }\n        this.resizeNotification();\n        this.paintNow();\n        //});\n    },\n\n    resizeNotification: function() {\n        //to be overridden\n    },\n\n    getBounds: function() {\n        return this.bounds;\n    },\n\n    paintNow: function() {\n        var self = this;\n        this.safePaintImmediately(function(gc) {\n            gc.clearRect(0, 0, self.canvas.width, self.canvas.height);\n\n            var comp = self._component;\n            if (comp) {\n                comp._paint(gc);\n            }\n\n            self.dirty = false;\n        });\n    },\n\n    safePaintImmediately: function(paintFunction) {\n        var useBitBlit = this.useBitBlit(),\n            gc = useBitBlit ? this.bufferGC : this.gc;\n        try {\n            gc.save();\n            paintFunction(gc);\n        } catch (e) {\n            console.error(e);\n        } finally {\n            gc.restore();\n        }\n        if (useBitBlit) {\n            this.flushBuffer();\n        }\n    },\n\n    flushBuffer: function() {\n        if (this.buffer.width > 0 && this.buffer.height > 0) {\n            this.canvasCTX.drawImage(this.buffer, 0, 0);\n        }\n    },\n\n    dispatchNewEvent: function(event, name, detail) {\n        detail = {\n            detail: detail || {}\n        };\n        detail.detail.primitiveEvent = event;\n        return this.canvas.dispatchEvent(new CustomEvent(name, detail));\n    },\n\n    dispatchNewMouseKeysEvent: function(event, name, detail) {\n        detail = detail || {};\n        detail.mouse = this.mouseLocation;\n        detail.keys = this.currentKeys;\n        return this.dispatchNewEvent(event, name, detail);\n    },\n\n    finmousemove: function(e) {\n        if (!this.isDragging() && this.mousedown) {\n            this.beDragging();\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dragstart', {\n                isRightClick: this.isRightClick(e)\n            });\n            this.dragstart = new rectangular.Point(this.mouseLocation.x, this.mouseLocation.y);\n        }\n        this.mouseLocation = this.getLocal(e);\n        //console.log(this.mouseLocation);\n        if (this.isDragging()) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-drag', {\n                dragstart: this.dragstart,\n                isRightClick: this.isRightClick(e)\n            });\n        }\n        if (this.bounds.contains(this.mouseLocation)) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mousemove');\n        }\n    },\n\n    finmousedown: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.mousedown = true;\n\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mousedown', {\n            isRightClick: this.isRightClick(e)\n        });\n        this.takeFocus();\n    },\n\n    finmouseup: function(e) {\n        if (this.isDragging()) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dragend', {\n                dragstart: this.dragstart,\n                isRightClick: this.isRightClick(e)\n            });\n            this.beNotDragging();\n            this.dragEndtime = Date.now();\n        }\n        this.mousedown = false;\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mouseup', {\n            isRightClick: this.isRightClick(e)\n        });\n        //this.mouseLocation = new rectangular.Point(-1, -1);\n    },\n\n    finmouseout: function(e) {\n        if (!this.mousedown) {\n            this.mouseLocation = new rectangular.Point(-1, -1);\n        }\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mouseout');\n    },\n\n    finwheelmoved: function(e) {\n        if (this.isDragging() || !this.hasFocus()) {\n            return;\n        }\n        e.preventDefault();\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-wheelmoved', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    finclick: function(e) {\n        if (Date.now() - this.lastClickTime < 250) {\n            //this is a double click...\n            this.findblclick(e);\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-click', {\n            isRightClick: this.isRightClick(e)\n        });\n        this.lastClickTime = Date.now();\n    },\n\n    finrelease: function(e) {\n        this.holdPulseCount = 0;\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-release');\n    },\n\n    finflick: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-flick', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    fintrackstart: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-trackstart');\n    },\n\n    fintrack: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-track');\n    },\n\n    fintrackend: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-trackend');\n    },\n\n    finhold: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-hold', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    finholdpulse: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-holdpulse', {\n            count: this.holdPulseCount++\n        });\n    },\n\n    fintap: function(e) {\n        //this nonsense is to hold a tap if it's really a double click\n        var self = this;\n        var now = Date.now();\n        var dif = now - this.lastDoubleClickTime;\n        if (dif < 300) {\n            return;\n        }\n        //dragend is also causing a tap\n        //lets fix this here\n        if (now - this.dragEndtime < 100) {\n            return;\n        }\n        setTimeout(function() {\n            self._fintap(e);\n        }, 180);\n    },\n\n    _fintap: function(e) {\n        //this nonsense is to hold a tap if it's really a double click\n        var now = Date.now();\n        var dif = now - this.lastDoubleClickTime;\n        if (dif < 300) {\n            return;\n        }\n        //this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-tap', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    findblclick: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.lastDoubleClickTime = Date.now();\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dblclick', {\n            isRightClick: this.isRightClick(e)\n        });\n        //console.log('dblclick', this.currentKeys);\n    },\n\n    getCharMap: function() { //TODO: This is static. Make it a property of the constructor.\n        return charMap;\n    },\n\n    finkeydown: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n\n        //e.preventDefault();\n        var keyChar = e.shiftKey ? charMap[e.keyCode][1] : charMap[e.keyCode][0];\n        if (e.repeat) {\n            if (this.repeatKey === keyChar) {\n                this.repeatKeyCount++;\n            } else {\n                this.repeatKey = keyChar;\n                this.repeatKeyStartTime = Date.now();\n            }\n        } else {\n            this.repeatKey = null;\n            this.repeatKeyCount = 0;\n            this.repeatKeyStartTime = 0;\n        }\n        if (this.currentKeys.indexOf(keyChar) === -1) {\n            this.currentKeys.push(keyChar);\n        }\n        //console.log(keyChar, e.keyCode);\n        this.dispatchNewEvent(e, 'fin-canvas-keydown', {\n            alt: e.altKey,\n            ctrl: e.ctrlKey,\n            char: keyChar,\n            code: e.charCode,\n            key: e.keyCode,\n            meta: e.metaKey,\n            repeatCount: this.repeatKeyCount,\n            repeatStartTime: this.repeatKeyStartTime,\n            shift: e.shiftKey,\n            identifier: e.keyIdentifier,\n            currentKeys: this.currentKeys.slice(0)\n        });\n    },\n\n    finkeyup: function(e) {\n        var keyChar = e.shiftKey ? charMap[e.keyCode][1] : charMap[e.keyCode][0];\n        this.currentKeys.splice(this.currentKeys.indexOf(keyChar), 1);\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.repeatKeyCount = 0;\n        this.repeatKey = null;\n        this.repeatKeyStartTime = 0;\n        this.dispatchNewEvent(e, 'fin-canvas-keyup', {\n            alt: e.altKey,\n            ctrl: e.ctrlKey,\n            char: keyChar,\n            code: e.charCode,\n            key: e.keyCode,\n            meta: e.metaKey,\n            repeat: e.repeat,\n            shift: e.shiftKey,\n            identifier: e.keyIdentifier,\n            currentKeys: this.currentKeys.slice(0)\n        });\n    },\n\n    finfocusgained: function(e) {\n        this.dispatchNewEvent(e, 'fin-canvas-focus-gained');\n    },\n\n    finfocuslost: function(e) {\n        this.dispatchNewEvent(e, 'fin-canvas-focus-lost');\n    },\n\n    fincontextmenu: function(e) {\n        if (e.ctrlKey && this.currentKeys.indexOf('CTRL') === -1) {\n            this.currentKeys.push('CTRL');\n        }\n        if (Date.now() - this.lastClickTime < 250) {\n            //this is a double click...\n            this.findblclick(e);\n            return;\n        }\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-context-menu', {\n            isRightClick: this.isRightClick(e)\n        });\n        this.lastClickTime = Date.now();\n    },\n\n    repaint: function() {\n        var fps = this.getFPS();\n        this.dirty = true;\n        if (!paintLoopRunning || fps === 0) {\n            this.paintNow();\n        }\n    },\n\n    getMouseLocation: function() {\n        return this.mouseLocation;\n    },\n\n    getOrigin: function() {\n        var rect = this.canvas.getBoundingClientRect();\n        var p = new rectangular.Point(rect.left, rect.top);\n        return p;\n    },\n\n    getLocal: function(e) {\n        var rect = this.canvas.getBoundingClientRect();\n        var p = new rectangular.Point(e.clientX - rect.left, e.clientY - rect.top);\n        return p;\n    },\n\n    hasFocus: function() {\n        return document.activeElement === this.canvas;\n    },\n\n    takeFocus: function() {\n        var self = this;\n        if (!this.hasFocus()) {\n            setTimeout(function() {\n                self.canvas.focus();\n            }, 10);\n        }\n    },\n\n    beDragging: function() {\n        this.dragging = true;\n        this.disableDocumentElementSelection();\n    },\n\n    beNotDragging: function() {\n        this.dragging = false;\n        this.enableDocumentElementSelection();\n    },\n\n    isDragging: function() {\n        return this.dragging;\n    },\n\n    disableDocumentElementSelection: function() {\n        var style = document.body.style;\n        style.cssText = style.cssText + '-webkit-user-select: none';\n    },\n\n    enableDocumentElementSelection: function() {\n        var style = document.body.style;\n        style.cssText = style.cssText.replace('-webkit-user-select: none', '');\n    },\n\n    setFocusable: function(truthy) {\n        this.focuser.style.display = truthy ? '' : 'none';\n    },\n\n    isRightClick: function(e) {\n        var isRightMB;\n        e = e || window.event;\n\n        if ('which' in e) { // Gecko (Firefox), WebKit (Safari/Chrome) & Opera\n            isRightMB = e.which === 3;\n        } else if ('button' in e) { // IE, Opera\n            isRightMB = e.button === 2;\n        }\n        return isRightMB;\n    },\n\n    dispatchEvent: function(e) {\n        return this.canvas.dispatchEvent(e);\n    }\n};\n\nfunction paintLoopFunction(now) {\n    if (!paintLoopRunning) {\n        return;\n    }\n    for (var i = 0; i < paintables.length; i++) {\n        try {\n            paintables[i].tickPainter(now);\n        } catch (e) {\n            console.error(e);\n        }\n    }\n    requestAnimationFrame(paintLoopFunction);\n}\nrequestAnimationFrame(paintLoopFunction);\n\nfunction resizablesLoopFunction(now) {\n    if (!resizeLoopRunning) {\n        return;\n    }\n    for (var i = 0; i < resizables.length; i++) {\n        try {\n            resizables[i].tickResizer(now);\n        } catch (e) {\n            console.error(e);\n        }\n    }\n}\nsetInterval(resizablesLoopFunction, RESIZE_POLLING_INTERVAL);\n\nfunction makeCharMap() {\n    var map = [];\n\n    var empty = ['', ''];\n\n    for (var i = 0; i < 256; i++) {\n        map[i] = empty;\n    }\n\n    map[27] = ['ESC', 'ESCSHIFT'];\n    map[192] = ['`', '~'];\n    map[49] = ['1', '!'];\n    map[50] = ['2', '@'];\n    map[51] = ['3', '#'];\n    map[52] = ['4', '$'];\n    map[53] = ['5', '%'];\n    map[54] = ['6', '^'];\n    map[55] = ['7', '&'];\n    map[56] = ['8', '*'];\n    map[57] = ['9', '('];\n    map[48] = ['0', ')'];\n    map[189] = ['-', '_'];\n    map[187] = ['=', '+'];\n    map[8] = ['DELETE', 'DELETESHIFT'];\n    map[9] = ['TAB', 'TABSHIFT'];\n    map[81] = ['q', 'Q'];\n    map[87] = ['w', 'W'];\n    map[69] = ['e', 'E'];\n    map[82] = ['r', 'R'];\n    map[84] = ['t', 'T'];\n    map[89] = ['y', 'Y'];\n    map[85] = ['u', 'U'];\n    map[73] = ['i', 'I'];\n    map[79] = ['o', 'O'];\n    map[80] = ['p', 'P'];\n    map[219] = ['[', '{'];\n    map[221] = [']', '}'];\n    map[220] = ['\\\\', '|'];\n    map[220] = ['CAPSLOCK', 'CAPSLOCKSHIFT'];\n    map[65] = ['a', 'A'];\n    map[83] = ['s', 'S'];\n    map[68] = ['d', 'D'];\n    map[70] = ['f', 'F'];\n    map[71] = ['g', 'G'];\n    map[72] = ['h', 'H'];\n    map[74] = ['j', 'J'];\n    map[75] = ['k', 'K'];\n    map[76] = ['l', 'L'];\n    map[186] = [';', ':'];\n    map[222] = ['\\'', '|'];\n    map[13] = ['RETURN', 'RETURNSHIFT'];\n    map[16] = ['SHIFT', 'SHIFT'];\n    map[90] = ['z', 'Z'];\n    map[88] = ['x', 'X'];\n    map[67] = ['c', 'C'];\n    map[86] = ['v', 'V'];\n    map[66] = ['b', 'B'];\n    map[78] = ['n', 'N'];\n    map[77] = ['m', 'M'];\n    map[188] = [',', '<'];\n    map[190] = ['.', '>'];\n    map[191] = ['/', '?'];\n    map[16] = ['SHIFT', 'SHIFT'];\n    map[17] = ['CTRL', 'CTRLSHIFT'];\n    map[18] = ['ALT', 'ALTSHIFT'];\n    map[91] = ['COMMANDLEFT', 'COMMANDLEFTSHIFT'];\n    map[32] = ['SPACE', 'SPACESHIFT'];\n    map[93] = ['COMMANDRIGHT', 'COMMANDRIGHTSHIFT'];\n    map[18] = ['ALT', 'ALTSHIFT'];\n    map[38] = ['UP', 'UPSHIFT'];\n    map[37] = ['LEFT', 'LEFTSHIFT'];\n    map[40] = ['DOWN', 'DOWNSHIFT'];\n    map[39] = ['RIGHT', 'RIGHTSHIFT'];\n\n    map[33] = ['PAGEUP', 'PAGEUPSHIFT'];\n    map[34] = ['PAGEDOWN', 'PAGEDOWNSHIFT'];\n    map[35] = ['PAGERIGHT', 'PAGERIGHTSHIFT'];\n    map[36] = ['PAGELEFT', 'PAGELEFTSHIFT'];\n\n    return map;\n}\n\nmodule.exports = Canvas;\n","'use strict';\n\nvar consoleLogger = require('./gc-console-logger');\n\n/**\n * @constructor\n * @param gc - The 2-D graphics context from your canvas\n * @param {boolean|apiLogger} [logger=true]\n * * `true` uses `gc-console-logger` function bound to 'gc.' as prefix\n * * string uses `gc-console-logger` function bound to string\n * * function used as is\n */\nfunction GraphicsContext(gc, logger) {\n    this.gc = gc;\n\n    var self = this;\n    var reWEBKIT = /^webkit/;\n\n    switch (typeof logger) {\n\n        case 'string':\n            logger =  consoleLogger.bind(undefined, logger + '.');\n            break;\n\n        case 'boolean':\n            if (logger === true) {\n                logger = consoleLogger.bind(undefined, 'gc.');\n            }\n            break;\n\n        case 'function':\n            if (logger.length !== 3) {\n                throw 'GraphicsContext: User-supplied API logger function does not accept three parameters.';\n            }\n            break;\n\n        default:\n            logger = false;\n    }\n\n    // Stub out all the prototype members of the canvas 2D graphics context:\n    Object.keys(Object.getPrototypeOf(gc)).forEach(MakeStub);\n\n    // Some older browsers (e.g., Chrome 40) did not have all members of canvas\n    // 2D graphics context in the prototype so we make this additional call:\n    Object.keys(gc).forEach(MakeStub);\n\n    function MakeStub(key) {\n        if (key in GraphicsContext.prototype || reWEBKIT.test(key)) {\n            return;\n        }\n        if (typeof gc[key] === 'function') {\n            self[key] = !logger ? gc[key].bind(gc) : function() {\n                return logger(key, arguments, gc[key].apply(gc, arguments));\n            };\n        } else {\n            Object.defineProperty(self, key, {\n                get: function() {\n                    var result = gc[key];\n                    return logger ? logger(key, 'getter', result) : result;\n                },\n                set: function(value) {\n                    gc[key] = logger ? logger(key, 'setter', value) : value;\n                }\n            });\n        }\n    }\n}\n\nmodule.exports = GraphicsContext;\n","'use strict';\n\nvar YIELDS = '\\u27F9'; // LONG RIGHTWARDS DOUBLE ARROW\n\nfunction consoleLogger(prefix, name, args, value) {\n    var result = value;\n\n    if (typeof value === 'string') {\n        result = '\"' + result + '\"';\n    }\n\n    name = prefix + name;\n\n    switch (args) {\n        case 'getter':\n            console.log(name, '=', result);\n            break;\n\n        case 'setter':\n            console.log(name, YIELDS, result);\n            break;\n\n        default: // method call\n            name += '(' + Array.prototype.slice.call(args).join(', ') + ')';\n            if (result === undefined) {\n                console.log(name);\n            } else {\n                console.log(name, YIELDS, result);\n            }\n    }\n\n    return value;\n}\n\nmodule.exports = consoleLogger;\n","/* eslint-disable */\n\n/**\n * @license\n * Copyright (c) 2014 The Polymer Project Authors. All rights reserved.\n * This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt\n * The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt\n * The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt\n * Code distributed by Google as part of the polymer project is also\n * subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt\n */\n//module.exports = {};\n\n(function(scope) {\n    var hasFullPath = false;\n\n    // test for full event path support\n    var pathTest = document.createElement('meta');\n    if (pathTest.createShadowRoot) {\n        var sr = pathTest.createShadowRoot();\n        var s = document.createElement('span');\n        sr.appendChild(s);\n        pathTest.addEventListener('testpath', function(ev) {\n            if (ev.path) {\n                // if the span is in the event path, then path[0] is the real source for all events\n                hasFullPath = ev.path[0] === s;\n            }\n            ev.stopPropagation();\n        });\n        var ev = new CustomEvent('testpath', {\n            bubbles: true\n        });\n        // must add node to DOM to trigger event listener\n        document.head.appendChild(pathTest);\n        s.dispatchEvent(ev);\n        pathTest.parentNode.removeChild(pathTest);\n        sr = s = null;\n    }\n    pathTest = null;\n\n    var target = {\n        shadow: function(inEl) {\n            if (inEl) {\n                return inEl.shadowRoot || inEl.webkitShadowRoot;\n            }\n        },\n        canTarget: function(shadow) {\n            return shadow && Boolean(shadow.elementFromPoint);\n        },\n        targetingShadow: function(inEl) {\n            var s = this.shadow(inEl);\n            if (this.canTarget(s)) {\n                return s;\n            }\n        },\n        olderShadow: function(shadow) {\n            var os = shadow.olderShadowRoot;\n            if (!os) {\n                var se = shadow.querySelector('shadow');\n                if (se) {\n                    os = se.olderShadowRoot;\n                }\n            }\n            return os;\n        },\n        allShadows: function(element) {\n            var shadows = [],\n                s = this.shadow(element);\n            while (s) {\n                shadows.push(s);\n                s = this.olderShadow(s);\n            }\n            return shadows;\n        },\n        searchRoot: function(inRoot, x, y) {\n            var t, st, sr, os;\n            if (inRoot) {\n                t = inRoot.elementFromPoint(x, y);\n                if (t) {\n                    // found element, check if it has a ShadowRoot\n                    sr = this.targetingShadow(t);\n                } else if (inRoot !== document) {\n                    // check for sibling roots\n                    sr = this.olderShadow(inRoot);\n                }\n                // search other roots, fall back to light dom element\n                return this.searchRoot(sr, x, y) || t;\n            }\n        },\n        owner: function(element) {\n            if (!element) {\n                return document;\n            }\n            var s = element;\n            // walk up until you hit the shadow root or document\n            while (s.parentNode) {\n                s = s.parentNode;\n            }\n            // the owner element is expected to be a Document or ShadowRoot\n            if (s.nodeType != Node.DOCUMENT_NODE && s.nodeType != Node.DOCUMENT_FRAGMENT_NODE) {\n                s = document;\n            }\n            return s;\n        },\n        findTarget: function(inEvent) {\n            if (hasFullPath && inEvent.path && inEvent.path.length) {\n                return inEvent.path[0];\n            }\n            var x = inEvent.clientX,\n                y = inEvent.clientY;\n            // if the listener is in the shadow root, it is much faster to start there\n            var s = this.owner(inEvent.target);\n            // if x, y is not in this root, fall back to document search\n            if (!s.elementFromPoint(x, y)) {\n                s = document;\n            }\n            return this.searchRoot(s, x, y);\n        },\n        findTouchAction: function(inEvent) {\n            var n;\n            if (hasFullPath && inEvent.path && inEvent.path.length) {\n                var path = inEvent.path;\n                for (var i = 0; i < path.length; i++) {\n                    n = path[i];\n                    if (n.nodeType === Node.ELEMENT_NODE && n.hasAttribute('touch-action')) {\n                        return n.getAttribute('touch-action');\n                    }\n                }\n            } else {\n                n = inEvent.target;\n                while (n) {\n                    if (n.nodeType === Node.ELEMENT_NODE && n.hasAttribute('touch-action')) {\n                        return n.getAttribute('touch-action');\n                    }\n                    n = n.parentNode || n.host;\n                }\n            }\n            // auto is default\n            return \"auto\";\n        },\n        LCA: function(a, b) {\n            if (a === b) {\n                return a;\n            }\n            if (a && !b) {\n                return a;\n            }\n            if (b && !a) {\n                return b;\n            }\n            if (!b && !a) {\n                return document;\n            }\n            // fast case, a is a direct descendant of b or vice versa\n            if (a.contains && a.contains(b)) {\n                return a;\n            }\n            if (b.contains && b.contains(a)) {\n                return b;\n            }\n            var adepth = this.depth(a);\n            var bdepth = this.depth(b);\n            var d = adepth - bdepth;\n            if (d >= 0) {\n                a = this.walk(a, d);\n            } else {\n                b = this.walk(b, -d);\n            }\n            while (a && b && a !== b) {\n                a = a.parentNode || a.host;\n                b = b.parentNode || b.host;\n            }\n            return a;\n        },\n        walk: function(n, u) {\n            for (var i = 0; n && (i < u); i++) {\n                n = n.parentNode || n.host;\n            }\n            return n;\n        },\n        depth: function(n) {\n            var d = 0;\n            while (n) {\n                d++;\n                n = n.parentNode || n.host;\n            }\n            return d;\n        },\n        deepContains: function(a, b) {\n            var common = this.LCA(a, b);\n            // if a is the common ancestor, it must \"deeply\" contain b\n            return common === a;\n        },\n        insideNode: function(node, x, y) {\n            var rect = node.getBoundingClientRect();\n            return (rect.left <= x) && (x <= rect.right) && (rect.top <= y) && (y <= rect.bottom);\n        },\n        path: function(event) {\n            var p;\n            if (hasFullPath && event.path && event.path.length) {\n                p = event.path;\n            } else {\n                p = [];\n                var n = this.findTarget(event);\n                while (n) {\n                    p.push(n);\n                    n = n.parentNode || n.host;\n                }\n            }\n            return p;\n        }\n    };\n    scope.targetFinding = target;\n    /**\n     * Given an event, finds the \"deepest\" node that could have been the original target before ShadowDOM retargetting\n     *\n     * @param {Event} Event An event object with clientX and clientY properties\n     * @return {Element} The probable event origninator\n     */\n    scope.findTarget = target.findTarget.bind(target);\n    /**\n     * Determines if the \"container\" node deeply contains the \"containee\" node, including situations where the \"containee\" is contained by one or more ShadowDOM\n     * roots.\n     *\n     * @param {Node} container\n     * @param {Node} containee\n     * @return {Boolean}\n     */\n    scope.deepContains = target.deepContains.bind(target);\n\n    /**\n     * Determines if the x/y position is inside the given node.\n     *\n     * Example:\n     *\n     *     function upHandler(event) {\n     *       var innode = PolymerGestures.insideNode(event.target, event.clientX, event.clientY);\n     *       if (innode) {\n     *         // wait for tap?\n     *       } else {\n     *         // tap will never happen\n     *       }\n     *     }\n     *\n     * @param {Node} node\n     * @param {Number} x Screen X position\n     * @param {Number} y screen Y position\n     * @return {Boolean}\n     */\n    scope.insideNode = target.insideNode;\n\n})(exports);\n\n(function() {\n    function shadowSelector(v) {\n        return 'html /deep/ ' + selector(v);\n    }\n\n    function selector(v) {\n        return '[touch-action=\"' + v + '\"]';\n    }\n\n    function rule(v) {\n        return '{ -ms-touch-action: ' + v + '; touch-action: ' + v + ';}';\n    }\n    var attrib2css = [\n        'none',\n        'auto',\n        'pan-x',\n        'pan-y', {\n            rule: 'pan-x pan-y',\n            selectors: [\n                'pan-x pan-y',\n                'pan-y pan-x'\n            ]\n        },\n        'manipulation'\n    ];\n    var styles = '';\n    // only install stylesheet if the browser has touch action support\n    var hasTouchAction = typeof document.head.style.touchAction === 'string';\n    // only add shadow selectors if shadowdom is supported\n    var hasShadowRoot = !window.ShadowDOMPolyfill && document.head.createShadowRoot;\n\n    if (hasTouchAction) {\n        attrib2css.forEach(function(r) {\n            if (String(r) === r) {\n                styles += selector(r) + rule(r) + '\\n';\n                if (hasShadowRoot) {\n                    styles += shadowSelector(r) + rule(r) + '\\n';\n                }\n            } else {\n                styles += r.selectors.map(selector) + rule(r.rule) + '\\n';\n                if (hasShadowRoot) {\n                    styles += r.selectors.map(shadowSelector) + rule(r.rule) + '\\n';\n                }\n            }\n        });\n\n        var el = document.createElement('style');\n        el.textContent = styles;\n        document.head.appendChild(el);\n    }\n})();\n\n/**\n * This is the constructor for new PointerEvents.\n *\n * New Pointer Events must be given a type, and an optional dictionary of\n * initialization properties.\n *\n * Due to certain platform requirements, events returned from the constructor\n * identify as MouseEvents.\n *\n * @constructor\n * @param {String} inType The type of the event to create.\n * @param {Object} [inDict] An optional dictionary of initial event properties.\n * @return {Event} A new PointerEvent of type `inType` and initialized with properties from `inDict`.\n */\n(function(scope) {\n\n    var MOUSE_PROPS = [\n        'bubbles',\n        'cancelable',\n        'view',\n        'detail',\n        'screenX',\n        'screenY',\n        'clientX',\n        'clientY',\n        'ctrlKey',\n        'altKey',\n        'shiftKey',\n        'metaKey',\n        'button',\n        'relatedTarget',\n        'pageX',\n        'pageY'\n    ];\n\n    var MOUSE_DEFAULTS = [\n        false,\n        false,\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        false,\n        false,\n        false,\n        false,\n        0,\n        null,\n        0,\n        0\n    ];\n\n    var NOP_FACTORY = function() {\n        return function() {};\n    };\n\n    var eventFactory = {\n        // TODO(dfreedm): this is overridden by tap recognizer, needs review\n        preventTap: NOP_FACTORY,\n        makeBaseEvent: function(inType, inDict) {\n            var e = document.createEvent('Event');\n            e.initEvent(inType, inDict.bubbles || false, inDict.cancelable || false);\n            e.preventTap = eventFactory.preventTap(e);\n            return e;\n        },\n        makeGestureEvent: function(inType, inDict) {\n            inDict = inDict || Object.create(null);\n\n            var e = this.makeBaseEvent(inType, inDict);\n            for (var i = 0, keys = Object.keys(inDict), k; i < keys.length; i++) {\n                k = keys[i];\n                if (k !== 'bubbles' && k !== 'cancelable') {\n                    e[k] = inDict[k];\n                }\n            }\n            return e;\n        },\n        makePointerEvent: function(inType, inDict) {\n            inDict = inDict || Object.create(null);\n\n            var e = this.makeBaseEvent(inType, inDict);\n            // define inherited MouseEvent properties\n            for (var i = 2, p; i < MOUSE_PROPS.length; i++) {\n                p = MOUSE_PROPS[i];\n                e[p] = inDict[p] || MOUSE_DEFAULTS[i];\n            }\n            e.buttons = inDict.buttons || 0;\n\n            // Spec requires that pointers without pressure specified use 0.5 for down\n            // state and 0 for up state.\n            var pressure = 0;\n            if (inDict.pressure) {\n                pressure = inDict.pressure;\n            } else {\n                pressure = e.buttons ? 0.5 : 0;\n            }\n\n            // add x/y properties aliased to clientX/Y\n            e.x = e.clientX;\n            e.y = e.clientY;\n\n            // define the properties of the PointerEvent interface\n            e.pointerId = inDict.pointerId || 0;\n            e.width = inDict.width || 0;\n            e.height = inDict.height || 0;\n            e.pressure = pressure;\n            e.tiltX = inDict.tiltX || 0;\n            e.tiltY = inDict.tiltY || 0;\n            e.pointerType = inDict.pointerType || '';\n            e.hwTimestamp = inDict.hwTimestamp || 0;\n            e.isPrimary = inDict.isPrimary || false;\n            e._source = inDict._source || '';\n            return e;\n        }\n    };\n\n    scope.eventFactory = eventFactory;\n})(exports);\n\n/**\n * This module implements an map of pointer states\n */\n(function(scope) {\n    var USE_MAP = window.Map && window.Map.prototype.forEach;\n    var POINTERS_FN = function() {\n        return this.size;\n    };\n\n    function PointerMap() {\n        if (USE_MAP) {\n            var m = new Map();\n            m.pointers = POINTERS_FN;\n            return m;\n        } else {\n            this.keys = [];\n            this.values = [];\n        }\n    }\n\n    PointerMap.prototype = {\n        set: function(inId, inEvent) {\n            var i = this.keys.indexOf(inId);\n            if (i > -1) {\n                this.values[i] = inEvent;\n            } else {\n                this.keys.push(inId);\n                this.values.push(inEvent);\n            }\n        },\n        has: function(inId) {\n            return this.keys.indexOf(inId) > -1;\n        },\n        'delete': function(inId) {\n            var i = this.keys.indexOf(inId);\n            if (i > -1) {\n                this.keys.splice(i, 1);\n                this.values.splice(i, 1);\n            }\n        },\n        get: function(inId) {\n            var i = this.keys.indexOf(inId);\n            return this.values[i];\n        },\n        clear: function() {\n            this.keys.length = 0;\n            this.values.length = 0;\n        },\n        // return value, key, map\n        forEach: function(callback, thisArg) {\n            this.values.forEach(function(v, i) {\n                callback.call(thisArg, v, this.keys[i], this);\n            }, this);\n        },\n        pointers: function() {\n            return this.keys.length;\n        }\n    };\n\n    scope.PointerMap = PointerMap;\n})(exports);\n\n(function(scope) {\n    var CLONE_PROPS = [\n        // MouseEvent\n        'bubbles',\n        'cancelable',\n        'view',\n        'detail',\n        'screenX',\n        'screenY',\n        'clientX',\n        'clientY',\n        'ctrlKey',\n        'altKey',\n        'shiftKey',\n        'metaKey',\n        'button',\n        'relatedTarget',\n        // DOM Level 3\n        'buttons',\n        // PointerEvent\n        'pointerId',\n        'width',\n        'height',\n        'pressure',\n        'tiltX',\n        'tiltY',\n        'pointerType',\n        'hwTimestamp',\n        'isPrimary',\n        // event instance\n        'type',\n        'target',\n        'currentTarget',\n        'which',\n        'pageX',\n        'pageY',\n        'timeStamp',\n        // gesture addons\n        'preventTap',\n        'tapPrevented',\n        '_source'\n    ];\n\n    var CLONE_DEFAULTS = [\n        // MouseEvent\n        false,\n        false,\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        false,\n        false,\n        false,\n        false,\n        0,\n        null,\n        // DOM Level 3\n        0,\n        // PointerEvent\n        0,\n        0,\n        0,\n        0,\n        0,\n        0,\n        '',\n        0,\n        false,\n        // event instance\n        '',\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        function() {},\n        false\n    ];\n\n    var HAS_SVG_INSTANCE = (typeof SVGElementInstance !== 'undefined');\n\n    var eventFactory = scope.eventFactory;\n\n    // set of recognizers to run for the currently handled event\n    var currentGestures;\n\n    /**\n     * This module is for normalizing events. Mouse and Touch events will be\n     * collected here, and fire PointerEvents that have the same semantics, no\n     * matter the source.\n     * Events fired:\n     *   - pointerdown: a pointing is added\n     *   - pointerup: a pointer is removed\n     *   - pointermove: a pointer is moved\n     *   - pointerover: a pointer crosses into an element\n     *   - pointerout: a pointer leaves an element\n     *   - pointercancel: a pointer will no longer generate events\n     */\n    var dispatcher = {\n        IS_IOS: false,\n        pointermap: new scope.PointerMap(),\n        requiredGestures: new scope.PointerMap(),\n        eventMap: Object.create(null),\n        // Scope objects for native events.\n        // This exists for ease of testing.\n        eventSources: Object.create(null),\n        eventSourceList: [],\n        gestures: [],\n        // map gesture event -> {listeners: int, index: gestures[int]}\n        dependencyMap: {\n            // make sure down and up are in the map to trigger \"register\"\n            down: {\n                listeners: 0,\n                index: -1\n            },\n            up: {\n                listeners: 0,\n                index: -1\n            }\n        },\n        gestureQueue: [],\n        /**\n         * Add a new event source that will generate pointer events.\n         *\n         * `inSource` must contain an array of event names named `events`, and\n         * functions with the names specified in the `events` array.\n         * @param {string} name A name for the event source\n         * @param {Object} source A new source of platform events.\n         */\n        registerSource: function(name, source) {\n            var s = source;\n            var newEvents = s.events;\n            if (newEvents) {\n                newEvents.forEach(function(e) {\n                    if (s[e]) {\n                        this.eventMap[e] = s[e].bind(s);\n                    }\n                }, this);\n                this.eventSources[name] = s;\n                this.eventSourceList.push(s);\n            }\n        },\n        registerGesture: function(name, source) {\n            var obj = Object.create(null);\n            obj.listeners = 0;\n            obj.index = this.gestures.length;\n            for (var i = 0, g; i < source.exposes.length; i++) {\n                g = source.exposes[i].toLowerCase();\n                this.dependencyMap[g] = obj;\n            }\n            this.gestures.push(source);\n        },\n        register: function(element, initial) {\n            var l = this.eventSourceList.length;\n            for (var i = 0, es;\n                (i < l) && (es = this.eventSourceList[i]); i++) {\n                // call eventsource register\n                es.register.call(es, element, initial);\n            }\n        },\n        unregister: function(element) {\n            var l = this.eventSourceList.length;\n            for (var i = 0, es;\n                (i < l) && (es = this.eventSourceList[i]); i++) {\n                // call eventsource register\n                es.unregister.call(es, element);\n            }\n        },\n        // EVENTS\n        down: function(inEvent) {\n            this.requiredGestures.set(inEvent.pointerId, currentGestures);\n            this.fireEvent('down', inEvent);\n        },\n        move: function(inEvent) {\n            // pipe move events into gesture queue directly\n            inEvent.type = 'move';\n            this.fillGestureQueue(inEvent);\n        },\n        up: function(inEvent) {\n            this.fireEvent('up', inEvent);\n            this.requiredGestures.delete(inEvent.pointerId);\n        },\n        cancel: function(inEvent) {\n            inEvent.tapPrevented = true;\n            this.fireEvent('up', inEvent);\n            this.requiredGestures.delete(inEvent.pointerId);\n        },\n        addGestureDependency: function(node, currentGestures) {\n            var gesturesWanted = node._pgEvents;\n            if (gesturesWanted && currentGestures) {\n                var gk = Object.keys(gesturesWanted);\n                for (var i = 0, r, ri, g; i < gk.length; i++) {\n                    // gesture\n                    g = gk[i];\n                    if (gesturesWanted[g] > 0) {\n                        // lookup gesture recognizer\n                        r = this.dependencyMap[g];\n                        // recognizer index\n                        ri = r ? r.index : -1;\n                        currentGestures[ri] = true;\n                    }\n                }\n            }\n        },\n        // LISTENER LOGIC\n        eventHandler: function(inEvent) {\n            // This is used to prevent multiple dispatch of events from\n            // platform events. This can happen when two elements in different scopes\n            // are set up to create pointer events, which is relevant to Shadow DOM.\n\n            var type = inEvent.type;\n\n            // only generate the list of desired events on \"down\"\n            if (type === 'touchstart' || type === 'mousedown' || type === 'pointerdown' || type === 'MSPointerDown') {\n                if (!inEvent._handledByPG) {\n                    currentGestures = {};\n                }\n\n                // in IOS mode, there is only a listener on the document, so this is not re-entrant\n                if (this.IS_IOS) {\n                    var ev = inEvent;\n                    if (type === 'touchstart') {\n                        var ct = inEvent.changedTouches[0];\n                        // set up a fake event to give to the path builder\n                        ev = {\n                            target: inEvent.target,\n                            clientX: ct.clientX,\n                            clientY: ct.clientY,\n                            path: inEvent.path\n                        };\n                    }\n                    // use event path if available, otherwise build a path from target finding\n                    var nodes = inEvent.path || scope.targetFinding.path(ev);\n                    for (var i = 0, n; i < nodes.length; i++) {\n                        n = nodes[i];\n                        this.addGestureDependency(n, currentGestures);\n                    }\n                } else {\n                    this.addGestureDependency(inEvent.currentTarget, currentGestures);\n                }\n            }\n\n            if (inEvent._handledByPG) {\n                return;\n            }\n            var fn = this.eventMap && this.eventMap[type];\n            if (fn) {\n                fn(inEvent);\n            }\n            inEvent._handledByPG = true;\n        },\n        // set up event listeners\n        listen: function(target, events) {\n            for (var i = 0, l = events.length, e;\n                (i < l) && (e = events[i]); i++) {\n                this.addEvent(target, e);\n            }\n        },\n        // remove event listeners\n        unlisten: function(target, events) {\n            for (var i = 0, l = events.length, e;\n                (i < l) && (e = events[i]); i++) {\n                this.removeEvent(target, e);\n            }\n        },\n        addEvent: function(target, eventName) {\n            target.addEventListener(eventName, this.boundHandler);\n        },\n        removeEvent: function(target, eventName) {\n            target.removeEventListener(eventName, this.boundHandler);\n        },\n        // EVENT CREATION AND TRACKING\n        /**\n         * Creates a new Event of type `inType`, based on the information in\n         * `inEvent`.\n         *\n         * @param {string} inType A string representing the type of event to create\n         * @param {Event} inEvent A platform event with a target\n         * @return {Event} A PointerEvent of type `inType`\n         */\n        makeEvent: function(inType, inEvent) {\n            var e = eventFactory.makePointerEvent(inType, inEvent);\n            e.preventDefault = inEvent.preventDefault;\n            e.tapPrevented = inEvent.tapPrevented;\n            e._target = e._target || inEvent.target;\n            return e;\n        },\n        // make and dispatch an event in one call\n        fireEvent: function(inType, inEvent) {\n            var e = this.makeEvent(inType, inEvent);\n            return this.dispatchEvent(e);\n        },\n        /**\n         * Returns a snapshot of inEvent, with writable properties.\n         *\n         * @param {Event} inEvent An event that contains properties to copy.\n         * @return {Object} An object containing shallow copies of `inEvent`'s\n         *    properties.\n         */\n        cloneEvent: function(inEvent) {\n            var eventCopy = Object.create(null),\n                p;\n            for (var i = 0; i < CLONE_PROPS.length; i++) {\n                p = CLONE_PROPS[i];\n                eventCopy[p] = inEvent[p] || CLONE_DEFAULTS[i];\n                // Work around SVGInstanceElement shadow tree\n                // Return the <use> element that is represented by the instance for Safari, Chrome, IE.\n                // This is the behavior implemented by Firefox.\n                if (p === 'target' || p === 'relatedTarget') {\n                    if (HAS_SVG_INSTANCE && eventCopy[p] instanceof SVGElementInstance) {\n                        eventCopy[p] = eventCopy[p].correspondingUseElement;\n                    }\n                }\n            }\n            // keep the semantics of preventDefault\n            eventCopy.preventDefault = function() {\n                inEvent.preventDefault();\n            };\n            return eventCopy;\n        },\n        /**\n         * Dispatches the event to its target.\n         *\n         * @param {Event} inEvent The event to be dispatched.\n         * @return {Boolean} True if an event handler returns true, false otherwise.\n         */\n        dispatchEvent: function(inEvent) {\n            var t = inEvent._target;\n            if (t) {\n                t.dispatchEvent(inEvent);\n                // clone the event for the gesture system to process\n                // clone after dispatch to pick up gesture prevention code\n                var clone = this.cloneEvent(inEvent);\n                clone.target = t;\n                this.fillGestureQueue(clone);\n            }\n        },\n        gestureTrigger: function() {\n            // process the gesture queue\n            for (var i = 0, e, rg; i < this.gestureQueue.length; i++) {\n                e = this.gestureQueue[i];\n                rg = e._requiredGestures;\n                if (rg) {\n                    for (var j = 0, g, fn; j < this.gestures.length; j++) {\n                        // only run recognizer if an element in the source event's path is listening for those gestures\n                        if (rg[j]) {\n                            g = this.gestures[j];\n                            fn = g[e.type];\n                            if (fn) {\n                                fn.call(g, e);\n                            }\n                        }\n                    }\n                }\n            }\n            this.gestureQueue.length = 0;\n        },\n        fillGestureQueue: function(ev) {\n            // only trigger the gesture queue once\n            if (!this.gestureQueue.length) {\n                requestAnimationFrame(this.boundGestureTrigger);\n            }\n            ev._requiredGestures = this.requiredGestures.get(ev.pointerId);\n            this.gestureQueue.push(ev);\n        }\n    };\n    dispatcher.boundHandler = dispatcher.eventHandler.bind(dispatcher);\n    dispatcher.boundGestureTrigger = dispatcher.gestureTrigger.bind(dispatcher);\n    scope.dispatcher = dispatcher;\n\n    /**\n     * Listen for `gesture` on `node` with the `handler` function\n     *\n     * If `handler` is the first listener for `gesture`, the underlying gesture recognizer is then enabled.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @return Boolean `gesture` is a valid gesture\n     */\n    scope.activateGesture = function(node, gesture) {\n        var g = gesture.toLowerCase();\n        var dep = dispatcher.dependencyMap[g];\n        if (dep) {\n            var recognizer = dispatcher.gestures[dep.index];\n            if (!node._pgListeners) {\n                dispatcher.register(node);\n                node._pgListeners = 0;\n            }\n            // TODO(dfreedm): re-evaluate bookkeeping to avoid using attributes\n            if (recognizer) {\n                var touchAction = recognizer.defaultActions && recognizer.defaultActions[g];\n                var actionNode;\n                switch (node.nodeType) {\n                    case Node.ELEMENT_NODE:\n                        actionNode = node;\n                        break;\n                    case Node.DOCUMENT_FRAGMENT_NODE:\n                        actionNode = node.host;\n                        break;\n                    default:\n                        actionNode = null;\n                        break;\n                }\n                if (touchAction && actionNode && !actionNode.hasAttribute('touch-action')) {\n                    actionNode.setAttribute('touch-action', touchAction);\n                }\n            }\n            if (!node._pgEvents) {\n                node._pgEvents = {};\n            }\n            node._pgEvents[g] = (node._pgEvents[g] || 0) + 1;\n            node._pgListeners++;\n        }\n        return Boolean(dep);\n    };\n\n    /**\n     *\n     * Listen for `gesture` from `node` with `handler` function.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @param {Function} handler\n     * @param {Boolean} capture\n     */\n    scope.addEventListener = function(node, gesture, handler, capture) {\n        if (handler) {\n            scope.activateGesture(node, gesture);\n            node.addEventListener(gesture, handler, capture);\n        }\n    };\n\n    /**\n     * Tears down the gesture configuration for `node`\n     *\n     * If `handler` is the last listener for `gesture`, the underlying gesture recognizer is disabled.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @return Boolean `gesture` is a valid gesture\n     */\n    scope.deactivateGesture = function(node, gesture) {\n        var g = gesture.toLowerCase();\n        var dep = dispatcher.dependencyMap[g];\n        if (dep) {\n            if (node._pgListeners > 0) {\n                node._pgListeners--;\n            }\n            if (node._pgListeners === 0) {\n                dispatcher.unregister(node);\n            }\n            if (node._pgEvents) {\n                if (node._pgEvents[g] > 0) {\n                    node._pgEvents[g]--;\n                } else {\n                    node._pgEvents[g] = 0;\n                }\n            }\n        }\n        return Boolean(dep);\n    };\n\n    /**\n     * Stop listening for `gesture` from `node` with `handler` function.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @param {Function} handler\n     * @param {Boolean} capture\n     */\n    scope.removeEventListener = function(node, gesture, handler, capture) {\n        if (handler) {\n            scope.deactivateGesture(node, gesture);\n            node.removeEventListener(gesture, handler, capture);\n        }\n    };\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    // radius around touchend that swallows mouse events\n    var DEDUP_DIST = 25;\n\n    var WHICH_TO_BUTTONS = [0, 1, 4, 2];\n\n    var currentButtons = 0;\n\n    var FIREFOX_LINUX = /Linux.*Firefox\\//i;\n\n    var HAS_BUTTONS = (function() {\n        // firefox on linux returns spec-incorrect values for mouseup.buttons\n        // https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent.buttons#See_also\n        // https://codereview.chromium.org/727593003/#msg16\n        if (FIREFOX_LINUX.test(navigator.userAgent)) {\n            return false;\n        }\n        try {\n            return new MouseEvent('test', {\n                buttons: 1\n            }).buttons === 1;\n        } catch (e) {\n            return false;\n        }\n    })();\n\n    // handler block for native mouse events\n    var mouseEvents = {\n        POINTER_ID: 1,\n        POINTER_TYPE: 'mouse',\n        events: [\n            'mousedown',\n            'mousemove',\n            'mouseup'\n        ],\n        exposes: [\n            'down',\n            'up',\n            'move'\n        ],\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        lastTouches: [],\n        // collide with the global mouse listener\n        isEventSimulatedFromTouch: function(inEvent) {\n            var lts = this.lastTouches;\n            var x = inEvent.clientX,\n                y = inEvent.clientY;\n            for (var i = 0, l = lts.length, t; i < l && (t = lts[i]); i++) {\n                // simulated mouse events will be swallowed near a primary touchend\n                var dx = Math.abs(x - t.x),\n                    dy = Math.abs(y - t.y);\n                if (dx <= DEDUP_DIST && dy <= DEDUP_DIST) {\n                    return true;\n                }\n            }\n        },\n        prepareEvent: function(inEvent) {\n            var e = dispatcher.cloneEvent(inEvent);\n            e.pointerId = this.POINTER_ID;\n            e.isPrimary = true;\n            e.pointerType = this.POINTER_TYPE;\n            e._source = 'mouse';\n            if (!HAS_BUTTONS) {\n                var type = inEvent.type;\n                var bit = WHICH_TO_BUTTONS[inEvent.which] || 0;\n                if (type === 'mousedown') {\n                    currentButtons |= bit;\n                } else if (type === 'mouseup') {\n                    currentButtons &= ~bit;\n                }\n                e.buttons = currentButtons;\n            }\n            return e;\n        },\n        mousedown: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var p = pointermap.has(this.POINTER_ID);\n                var e = this.prepareEvent(inEvent);\n                e.target = scope.findTarget(inEvent);\n                pointermap.set(this.POINTER_ID, e.target);\n                dispatcher.down(e);\n            }\n        },\n        mousemove: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var target = pointermap.get(this.POINTER_ID);\n                if (target) {\n                    var e = this.prepareEvent(inEvent);\n                    e.target = target;\n                    // handle case where we missed a mouseup\n                    if ((HAS_BUTTONS ? e.buttons : e.which) === 0) {\n                        if (!HAS_BUTTONS) {\n                            currentButtons = e.buttons = 0;\n                        }\n                        dispatcher.cancel(e);\n                        this.cleanupMouse(e.buttons);\n                    } else {\n                        dispatcher.move(e);\n                    }\n                }\n            }\n        },\n        mouseup: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var e = this.prepareEvent(inEvent);\n                e.relatedTarget = scope.findTarget(inEvent);\n                e.target = pointermap.get(this.POINTER_ID);\n                dispatcher.up(e);\n                this.cleanupMouse(e.buttons);\n            }\n        },\n        cleanupMouse: function(buttons) {\n            if (buttons === 0) {\n                pointermap.delete(this.POINTER_ID);\n            }\n        }\n    };\n\n    scope.mouseEvents = mouseEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var allShadows = scope.targetFinding.allShadows.bind(scope.targetFinding);\n    var pointermap = dispatcher.pointermap;\n    var touchMap = Array.prototype.map.call.bind(Array.prototype.map);\n    // This should be long enough to ignore compat mouse events made by touch\n    var DEDUP_TIMEOUT = 2500;\n    var DEDUP_DIST = 25;\n    var CLICK_COUNT_TIMEOUT = 200;\n    var HYSTERESIS = 20;\n    var ATTRIB = 'touch-action';\n    // TODO(dfreedm): disable until http://crbug.com/399765 is resolved\n    // var HAS_TOUCH_ACTION = ATTRIB in document.head.style;\n    var HAS_TOUCH_ACTION = false;\n\n    // handler block for native touch events\n    var touchEvents = {\n        IS_IOS: false,\n        events: [\n            'touchstart',\n            'touchmove',\n            'touchend',\n            'touchcancel'\n        ],\n        exposes: [\n            'down',\n            'up',\n            'move'\n        ],\n        register: function(target, initial) {\n            if (this.IS_IOS ? initial : !initial) {\n                dispatcher.listen(target, this.events);\n            }\n        },\n        unregister: function(target) {\n            if (!this.IS_IOS) {\n                dispatcher.unlisten(target, this.events);\n            }\n        },\n        scrollTypes: {\n            EMITTER: 'none',\n            XSCROLLER: 'pan-x',\n            YSCROLLER: 'pan-y',\n        },\n        touchActionToScrollType: function(touchAction) {\n            var t = touchAction;\n            var st = this.scrollTypes;\n            if (t === st.EMITTER) {\n                return 'none';\n            } else if (t === st.XSCROLLER) {\n                return 'X';\n            } else if (t === st.YSCROLLER) {\n                return 'Y';\n            } else {\n                return 'XY';\n            }\n        },\n        POINTER_TYPE: 'touch',\n        firstTouch: null,\n        isPrimaryTouch: function(inTouch) {\n            return this.firstTouch === inTouch.identifier;\n        },\n        setPrimaryTouch: function(inTouch) {\n            // set primary touch if there no pointers, or the only pointer is the mouse\n            if (pointermap.pointers() === 0 || (pointermap.pointers() === 1 && pointermap.has(1))) {\n                this.firstTouch = inTouch.identifier;\n                this.firstXY = {\n                    X: inTouch.clientX,\n                    Y: inTouch.clientY\n                };\n                this.firstTarget = inTouch.target;\n                this.scrolling = null;\n                this.cancelResetClickCount();\n            }\n        },\n        removePrimaryPointer: function(inPointer) {\n            if (inPointer.isPrimary) {\n                this.firstTouch = null;\n                this.firstXY = null;\n                this.resetClickCount();\n            }\n        },\n        clickCount: 0,\n        resetId: null,\n        resetClickCount: function() {\n            var fn = function() {\n                this.clickCount = 0;\n                this.resetId = null;\n            }.bind(this);\n            this.resetId = setTimeout(fn, CLICK_COUNT_TIMEOUT);\n        },\n        cancelResetClickCount: function() {\n            if (this.resetId) {\n                clearTimeout(this.resetId);\n            }\n        },\n        typeToButtons: function(type) {\n            var ret = 0;\n            if (type === 'touchstart' || type === 'touchmove') {\n                ret = 1;\n            }\n            return ret;\n        },\n        findTarget: function(touch, id) {\n            if (this.currentTouchEvent.type === 'touchstart') {\n                if (this.isPrimaryTouch(touch)) {\n                    var fastPath = {\n                        clientX: touch.clientX,\n                        clientY: touch.clientY,\n                        path: this.currentTouchEvent.path,\n                        target: this.currentTouchEvent.target\n                    };\n                    return scope.findTarget(fastPath);\n                } else {\n                    return scope.findTarget(touch);\n                }\n            }\n            // reuse target we found in touchstart\n            return pointermap.get(id);\n        },\n        touchToPointer: function(inTouch) {\n            var cte = this.currentTouchEvent;\n            var e = dispatcher.cloneEvent(inTouch);\n            // Spec specifies that pointerId 1 is reserved for Mouse.\n            // Touch identifiers can start at 0.\n            // Add 2 to the touch identifier for compatibility.\n            var id = e.pointerId = inTouch.identifier + 2;\n            e.target = this.findTarget(inTouch, id);\n            e.bubbles = true;\n            e.cancelable = true;\n            e.detail = this.clickCount;\n            e.buttons = this.typeToButtons(cte.type);\n            e.width = inTouch.webkitRadiusX || inTouch.radiusX || 0;\n            e.height = inTouch.webkitRadiusY || inTouch.radiusY || 0;\n            e.pressure = inTouch.webkitForce || inTouch.force || 0.5;\n            e.isPrimary = this.isPrimaryTouch(inTouch);\n            e.pointerType = this.POINTER_TYPE;\n            e._source = 'touch';\n            // forward touch preventDefaults\n            var self = this;\n            e.preventDefault = function() {\n                self.scrolling = false;\n                self.firstXY = null;\n                cte.preventDefault();\n            };\n            return e;\n        },\n        processTouches: function(inEvent, inFunction) {\n            var tl = inEvent.changedTouches;\n            this.currentTouchEvent = inEvent;\n            for (var i = 0, t, p; i < tl.length; i++) {\n                t = tl[i];\n                p = this.touchToPointer(t);\n                if (inEvent.type === 'touchstart') {\n                    pointermap.set(p.pointerId, p.target);\n                }\n                if (pointermap.has(p.pointerId)) {\n                    inFunction.call(this, p);\n                }\n                if (inEvent.type === 'touchend' || inEvent._cancel) {\n                    this.cleanUpPointer(p);\n                }\n            }\n        },\n        // For single axis scrollers, determines whether the element should emit\n        // pointer events or behave as a scroller\n        shouldScroll: function(inEvent) {\n            if (this.firstXY) {\n                var ret;\n                var touchAction = scope.targetFinding.findTouchAction(inEvent);\n                var scrollAxis = this.touchActionToScrollType(touchAction);\n                if (scrollAxis === 'none') {\n                    // this element is a touch-action: none, should never scroll\n                    ret = false;\n                } else if (scrollAxis === 'XY') {\n                    // this element should always scroll\n                    ret = true;\n                } else {\n                    var t = inEvent.changedTouches[0];\n                    // check the intended scroll axis, and other axis\n                    var a = scrollAxis;\n                    var oa = scrollAxis === 'Y' ? 'X' : 'Y';\n                    var da = Math.abs(t['client' + a] - this.firstXY[a]);\n                    var doa = Math.abs(t['client' + oa] - this.firstXY[oa]);\n                    // if delta in the scroll axis > delta other axis, scroll instead of\n                    // making events\n                    ret = da >= doa;\n                }\n                return ret;\n            }\n        },\n        findTouch: function(inTL, inId) {\n            for (var i = 0, l = inTL.length, t; i < l && (t = inTL[i]); i++) {\n                if (t.identifier === inId) {\n                    return true;\n                }\n            }\n        },\n        // In some instances, a touchstart can happen without a touchend. This\n        // leaves the pointermap in a broken state.\n        // Therefore, on every touchstart, we remove the touches that did not fire a\n        // touchend event.\n        // To keep state globally consistent, we fire a\n        // pointercancel for this \"abandoned\" touch\n        vacuumTouches: function(inEvent) {\n            var tl = inEvent.touches;\n            // pointermap.pointers() should be < tl.length here, as the touchstart has not\n            // been processed yet.\n            if (pointermap.pointers() >= tl.length) {\n                var d = [];\n                pointermap.forEach(function(value, key) {\n                    // Never remove pointerId == 1, which is mouse.\n                    // Touch identifiers are 2 smaller than their pointerId, which is the\n                    // index in pointermap.\n                    if (key !== 1 && !this.findTouch(tl, key - 2)) {\n                        var p = value;\n                        d.push(p);\n                    }\n                }, this);\n                d.forEach(function(p) {\n                    this.cancel(p);\n                    pointermap.delete(p.pointerId);\n                }, this);\n            }\n        },\n        touchstart: function(inEvent) {\n            this.vacuumTouches(inEvent);\n            this.setPrimaryTouch(inEvent.changedTouches[0]);\n            this.dedupSynthMouse(inEvent);\n            if (!this.scrolling) {\n                this.clickCount++;\n                this.processTouches(inEvent, this.down);\n            }\n        },\n        down: function(inPointer) {\n            dispatcher.down(inPointer);\n        },\n        touchmove: function(inEvent) {\n            if (HAS_TOUCH_ACTION) {\n                // touchevent.cancelable == false is sent when the page is scrolling under native Touch Action in Chrome 36\n                // https://groups.google.com/a/chromium.org/d/msg/input-dev/wHnyukcYBcA/b9kmtwM1jJQJ\n                if (inEvent.cancelable) {\n                    this.processTouches(inEvent, this.move);\n                }\n            } else {\n                if (!this.scrolling) {\n                    if (this.scrolling === null && this.shouldScroll(inEvent)) {\n                        this.scrolling = true;\n                    } else {\n                        this.scrolling = false;\n                        inEvent.preventDefault();\n                        this.processTouches(inEvent, this.move);\n                    }\n                } else if (this.firstXY) {\n                    var t = inEvent.changedTouches[0];\n                    var dx = t.clientX - this.firstXY.X;\n                    var dy = t.clientY - this.firstXY.Y;\n                    var dd = Math.sqrt(dx * dx + dy * dy);\n                    if (dd >= HYSTERESIS) {\n                        this.touchcancel(inEvent);\n                        this.scrolling = true;\n                        this.firstXY = null;\n                    }\n                }\n            }\n        },\n        move: function(inPointer) {\n            dispatcher.move(inPointer);\n        },\n        touchend: function(inEvent) {\n            this.dedupSynthMouse(inEvent);\n            this.processTouches(inEvent, this.up);\n        },\n        up: function(inPointer) {\n            inPointer.relatedTarget = scope.findTarget(inPointer);\n            dispatcher.up(inPointer);\n        },\n        cancel: function(inPointer) {\n            dispatcher.cancel(inPointer);\n        },\n        touchcancel: function(inEvent) {\n            inEvent._cancel = true;\n            this.processTouches(inEvent, this.cancel);\n        },\n        cleanUpPointer: function(inPointer) {\n            pointermap['delete'](inPointer.pointerId);\n            this.removePrimaryPointer(inPointer);\n        },\n        // prevent synth mouse events from creating pointer events\n        dedupSynthMouse: function(inEvent) {\n            var lts = scope.mouseEvents.lastTouches;\n            var t = inEvent.changedTouches[0];\n            // only the primary finger will synth mouse events\n            if (this.isPrimaryTouch(t)) {\n                // remember x/y of last touch\n                var lt = {\n                    x: t.clientX,\n                    y: t.clientY\n                };\n                lts.push(lt);\n                var fn = (function(lts, lt) {\n                    var i = lts.indexOf(lt);\n                    if (i > -1) {\n                        lts.splice(i, 1);\n                    }\n                }).bind(null, lts, lt);\n                setTimeout(fn, DEDUP_TIMEOUT);\n            }\n        }\n    };\n\n    // prevent \"ghost clicks\" that come from elements that were removed in a touch handler\n    var STOP_PROP_FN = Event.prototype.stopImmediatePropagation || Event.prototype.stopPropagation;\n    document.addEventListener('click', function(ev) {\n        var x = ev.clientX,\n            y = ev.clientY;\n        // check if a click is within DEDUP_DIST px radius of the touchstart\n        var closeTo = function(touch) {\n            var dx = Math.abs(x - touch.x),\n                dy = Math.abs(y - touch.y);\n            return (dx <= DEDUP_DIST && dy <= DEDUP_DIST);\n        };\n        // if click coordinates are close to touch coordinates, assume the click came from a touch\n        var wasTouched = scope.mouseEvents.lastTouches.some(closeTo);\n        // if the click came from touch, and the touchstart target is not in the path of the click event,\n        // then the touchstart target was probably removed, and the click should be \"busted\"\n        var path = scope.targetFinding.path(ev);\n        if (wasTouched) {\n            for (var i = 0; i < path.length; i++) {\n                if (path[i] === touchEvents.firstTarget) {\n                    return;\n                }\n            }\n            ev.preventDefault();\n            STOP_PROP_FN.call(ev);\n        }\n    }, true);\n\n    scope.touchEvents = touchEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    var HAS_BITMAP_TYPE = window.MSPointerEvent && typeof window.MSPointerEvent.MSPOINTER_TYPE_MOUSE === 'number';\n    var msEvents = {\n        events: [\n            'MSPointerDown',\n            'MSPointerMove',\n            'MSPointerUp',\n            'MSPointerCancel',\n        ],\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        POINTER_TYPES: [\n            '',\n            'unavailable',\n            'touch',\n            'pen',\n            'mouse'\n        ],\n        prepareEvent: function(inEvent) {\n            var e = inEvent;\n            e = dispatcher.cloneEvent(inEvent);\n            if (HAS_BITMAP_TYPE) {\n                e.pointerType = this.POINTER_TYPES[inEvent.pointerType];\n            }\n            e._source = 'ms';\n            return e;\n        },\n        cleanup: function(id) {\n            pointermap['delete'](id);\n        },\n        MSPointerDown: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.target = scope.findTarget(inEvent);\n            pointermap.set(inEvent.pointerId, e.target);\n            dispatcher.down(e);\n        },\n        MSPointerMove: function(inEvent) {\n            var target = pointermap.get(inEvent.pointerId);\n            if (target) {\n                var e = this.prepareEvent(inEvent);\n                e.target = target;\n                dispatcher.move(e);\n            }\n        },\n        MSPointerUp: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.up(e);\n            this.cleanup(inEvent.pointerId);\n        },\n        MSPointerCancel: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.cancel(e);\n            this.cleanup(inEvent.pointerId);\n        }\n    };\n\n    scope.msEvents = msEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    var pointerEvents = {\n        events: [\n            'pointerdown',\n            'pointermove',\n            'pointerup',\n            'pointercancel'\n        ],\n        prepareEvent: function(inEvent) {\n            var e = dispatcher.cloneEvent(inEvent);\n            e._source = 'pointer';\n            return e;\n        },\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        cleanup: function(id) {\n            pointermap['delete'](id);\n        },\n        pointerdown: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.target = scope.findTarget(inEvent);\n            pointermap.set(e.pointerId, e.target);\n            dispatcher.down(e);\n        },\n        pointermove: function(inEvent) {\n            var target = pointermap.get(inEvent.pointerId);\n            if (target) {\n                var e = this.prepareEvent(inEvent);\n                e.target = target;\n                dispatcher.move(e);\n            }\n        },\n        pointerup: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.up(e);\n            this.cleanup(inEvent.pointerId);\n        },\n        pointercancel: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.cancel(e);\n            this.cleanup(inEvent.pointerId);\n        }\n    };\n\n    scope.pointerEvents = pointerEvents;\n})(exports);\n\n/**\n * This module contains the handlers for native platform events.\n * From here, the dispatcher is called to create unified pointer events.\n * Included are touch events (v1), mouse events, and MSPointerEvents.\n */\n(function(scope) {\n\n    var dispatcher = scope.dispatcher;\n    var nav = window.navigator;\n\n    if (window.PointerEvent) {\n        dispatcher.registerSource('pointer', scope.pointerEvents);\n    } else if (nav.msPointerEnabled) {\n        dispatcher.registerSource('ms', scope.msEvents);\n    } else {\n        dispatcher.registerSource('mouse', scope.mouseEvents);\n        if (window.ontouchstart !== undefined) {\n            dispatcher.registerSource('touch', scope.touchEvents);\n        }\n    }\n\n    // Work around iOS bugs https://bugs.webkit.org/show_bug.cgi?id=135628 and https://bugs.webkit.org/show_bug.cgi?id=136506\n    var ua = navigator.userAgent;\n    var IS_IOS = ua.match(/iPad|iPhone|iPod/) && 'ontouchstart' in window;\n\n    dispatcher.IS_IOS = IS_IOS;\n    scope.touchEvents.IS_IOS = IS_IOS;\n\n    dispatcher.register(document, true);\n})(exports);\n\n/**\n * This event denotes the beginning of a series of tracking events.\n *\n * @module PointerGestures\n * @submodule Events\n * @class trackstart\n */\n/**\n * Pixels moved in the x direction since trackstart.\n * @type Number\n * @property dx\n */\n/**\n * Pixes moved in the y direction since trackstart.\n * @type Number\n * @property dy\n */\n/**\n * Pixels moved in the x direction since the last track.\n * @type Number\n * @property ddx\n */\n/**\n * Pixles moved in the y direction since the last track.\n * @type Number\n * @property ddy\n */\n/**\n * The clientX position of the track gesture.\n * @type Number\n * @property clientX\n */\n/**\n * The clientY position of the track gesture.\n * @type Number\n * @property clientY\n */\n/**\n * The pageX position of the track gesture.\n * @type Number\n * @property pageX\n */\n/**\n * The pageY position of the track gesture.\n * @type Number\n * @property pageY\n */\n/**\n * The screenX position of the track gesture.\n * @type Number\n * @property screenX\n */\n/**\n * The screenY position of the track gesture.\n * @type Number\n * @property screenY\n */\n/**\n * The last x axis direction of the pointer.\n * @type Number\n * @property xDirection\n */\n/**\n * The last y axis direction of the pointer.\n * @type Number\n * @property yDirection\n */\n/**\n * A shared object between all tracking events.\n * @type Object\n * @property trackInfo\n */\n/**\n * The element currently under the pointer.\n * @type Element\n * @property relatedTarget\n */\n/**\n * The type of pointer that make the track gesture.\n * @type String\n * @property pointerType\n */\n/**\n *\n * This event fires for all pointer movement being tracked.\n *\n * @class track\n * @extends trackstart\n */\n/**\n * This event fires when the pointer is no longer being tracked.\n *\n * @class trackend\n * @extends trackstart\n */\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var track = {\n        events: [\n            'down',\n            'move',\n            'up',\n        ],\n        exposes: [\n            'trackstart',\n            'track',\n            'trackx',\n            'tracky',\n            'trackend'\n        ],\n        defaultActions: {\n            'track': 'none',\n            'trackx': 'pan-y',\n            'tracky': 'pan-x'\n        },\n        WIGGLE_THRESHOLD: 4,\n        clampDir: function(inDelta) {\n            return inDelta > 0 ? 1 : -1;\n        },\n        calcPositionDelta: function(inA, inB) {\n            var x = 0,\n                y = 0;\n            if (inA && inB) {\n                x = inB.pageX - inA.pageX;\n                y = inB.pageY - inA.pageY;\n            }\n            return {\n                x: x,\n                y: y\n            };\n        },\n        fireTrack: function(inType, inEvent, inTrackingData) {\n            var t = inTrackingData;\n            var d = this.calcPositionDelta(t.downEvent, inEvent);\n            var dd = this.calcPositionDelta(t.lastMoveEvent, inEvent);\n            if (dd.x) {\n                t.xDirection = this.clampDir(dd.x);\n            } else if (inType === 'trackx') {\n                return;\n            }\n            if (dd.y) {\n                t.yDirection = this.clampDir(dd.y);\n            } else if (inType === 'tracky') {\n                return;\n            }\n            var gestureProto = {\n                bubbles: true,\n                cancelable: true,\n                trackInfo: t.trackInfo,\n                relatedTarget: inEvent.relatedTarget,\n                pointerType: inEvent.pointerType,\n                pointerId: inEvent.pointerId,\n                _source: 'track'\n            };\n            if (inType !== 'tracky') {\n                gestureProto.x = inEvent.x;\n                gestureProto.dx = d.x;\n                gestureProto.ddx = dd.x;\n                gestureProto.clientX = inEvent.clientX;\n                gestureProto.pageX = inEvent.pageX;\n                gestureProto.screenX = inEvent.screenX;\n                gestureProto.xDirection = t.xDirection;\n            }\n            if (inType !== 'trackx') {\n                gestureProto.dy = d.y;\n                gestureProto.ddy = dd.y;\n                gestureProto.y = inEvent.y;\n                gestureProto.clientY = inEvent.clientY;\n                gestureProto.pageY = inEvent.pageY;\n                gestureProto.screenY = inEvent.screenY;\n                gestureProto.yDirection = t.yDirection;\n            }\n            var e = eventFactory.makeGestureEvent(inType, gestureProto);\n            t.downTarget.dispatchEvent(e);\n        },\n        down: function(inEvent) {\n            if (inEvent.isPrimary && (inEvent.pointerType === 'mouse' ? inEvent.buttons === 1 : true)) {\n                var p = {\n                    downEvent: inEvent,\n                    downTarget: inEvent.target,\n                    trackInfo: {},\n                    lastMoveEvent: null,\n                    xDirection: 0,\n                    yDirection: 0,\n                    tracking: false\n                };\n                pointermap.set(inEvent.pointerId, p);\n            }\n        },\n        move: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            if (p) {\n                if (!p.tracking) {\n                    var d = this.calcPositionDelta(p.downEvent, inEvent);\n                    var move = d.x * d.x + d.y * d.y;\n                    // start tracking only if finger moves more than WIGGLE_THRESHOLD\n                    if (move > this.WIGGLE_THRESHOLD) {\n                        p.tracking = true;\n                        p.lastMoveEvent = p.downEvent;\n                        this.fireTrack('trackstart', inEvent, p);\n                    }\n                }\n                if (p.tracking) {\n                    this.fireTrack('track', inEvent, p);\n                    this.fireTrack('trackx', inEvent, p);\n                    this.fireTrack('tracky', inEvent, p);\n                }\n                p.lastMoveEvent = inEvent;\n            }\n        },\n        up: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            if (p) {\n                if (p.tracking) {\n                    this.fireTrack('trackend', inEvent, p);\n                }\n                pointermap.delete(inEvent.pointerId);\n            }\n        }\n    };\n    dispatcher.registerGesture('track', track);\n})(exports);\n\n/**\n * This event is fired when a pointer is held down for 200ms.\n *\n * @module PointerGestures\n * @submodule Events\n * @class hold\n */\n/**\n * Type of pointer that made the holding event.\n * @type String\n * @property pointerType\n */\n/**\n * Screen X axis position of the held pointer\n * @type Number\n * @property clientX\n */\n/**\n * Screen Y axis position of the held pointer\n * @type Number\n * @property clientY\n */\n/**\n * Type of pointer that made the holding event.\n * @type String\n * @property pointerType\n */\n/**\n * This event is fired every 200ms while a pointer is held down.\n *\n * @class holdpulse\n * @extends hold\n */\n/**\n * Milliseconds pointer has been held down.\n * @type Number\n * @property holdTime\n */\n/**\n * This event is fired when a held pointer is released or moved.\n *\n * @class release\n */\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var hold = {\n        // wait at least HOLD_DELAY ms between hold and pulse events\n        HOLD_DELAY: 200,\n        // pointer can move WIGGLE_THRESHOLD pixels before not counting as a hold\n        WIGGLE_THRESHOLD: 16,\n        events: [\n            'down',\n            'move',\n            'up',\n        ],\n        exposes: [\n            'hold',\n            'holdpulse',\n            'release'\n        ],\n        heldPointer: null,\n        holdJob: null,\n        pulse: function() {\n            var hold = Date.now() - this.heldPointer.timeStamp;\n            var type = this.held ? 'holdpulse' : 'hold';\n            this.fireHold(type, hold);\n            this.held = true;\n        },\n        cancel: function() {\n            clearInterval(this.holdJob);\n            if (this.held) {\n                this.fireHold('release');\n            }\n            this.held = false;\n            this.heldPointer = null;\n            this.target = null;\n            this.holdJob = null;\n        },\n        down: function(inEvent) {\n            if (inEvent.isPrimary && !this.heldPointer) {\n                this.heldPointer = inEvent;\n                this.target = inEvent.target;\n                this.holdJob = setInterval(this.pulse.bind(this), this.HOLD_DELAY);\n            }\n        },\n        up: function(inEvent) {\n            if (this.heldPointer && this.heldPointer.pointerId === inEvent.pointerId) {\n                this.cancel();\n            }\n        },\n        move: function(inEvent) {\n            if (this.heldPointer && this.heldPointer.pointerId === inEvent.pointerId) {\n                var x = inEvent.clientX - this.heldPointer.clientX;\n                var y = inEvent.clientY - this.heldPointer.clientY;\n                if ((x * x + y * y) > this.WIGGLE_THRESHOLD) {\n                    this.cancel();\n                }\n            }\n        },\n        fireHold: function(inType, inHoldTime) {\n            var p = {\n                bubbles: true,\n                cancelable: true,\n                pointerType: this.heldPointer.pointerType,\n                pointerId: this.heldPointer.pointerId,\n                x: this.heldPointer.clientX,\n                y: this.heldPointer.clientY,\n                _source: 'hold'\n            };\n            if (inHoldTime) {\n                p.holdTime = inHoldTime;\n            }\n            var e = eventFactory.makeGestureEvent(inType, p);\n            this.target.dispatchEvent(e);\n        }\n    };\n    dispatcher.registerGesture('hold', hold);\n})(exports);\n\n/**\n * This event is fired when a pointer quickly goes down and up, and is used to\n * denote activation.\n *\n * Any gesture event can prevent the tap event from being created by calling\n * `event.preventTap`.\n *\n * Any pointer event can prevent the tap by setting the `tapPrevented` property\n * on itself.\n *\n * @module PointerGestures\n * @submodule Events\n * @class tap\n */\n/**\n * X axis position of the tap.\n * @property x\n * @type Number\n */\n/**\n * Y axis position of the tap.\n * @property y\n * @type Number\n */\n/**\n * Type of the pointer that made the tap.\n * @property pointerType\n * @type String\n */\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var tap = {\n        events: [\n            'down',\n            'up'\n        ],\n        exposes: [\n            'tap'\n        ],\n        down: function(inEvent) {\n            if (inEvent.isPrimary && !inEvent.tapPrevented) {\n                pointermap.set(inEvent.pointerId, {\n                    target: inEvent.target,\n                    buttons: inEvent.buttons,\n                    x: inEvent.clientX,\n                    y: inEvent.clientY\n                });\n            }\n        },\n        shouldTap: function(e, downState) {\n            var tap = true;\n            if (e.pointerType === 'mouse') {\n                // only allow left click to tap for mouse\n                tap = (e.buttons ^ 1) && (downState.buttons & 1);\n            }\n            return tap && !e.tapPrevented;\n        },\n        up: function(inEvent) {\n            var start = pointermap.get(inEvent.pointerId);\n            if (start && this.shouldTap(inEvent, start)) {\n                // up.relatedTarget is target currently under finger\n                var t = scope.targetFinding.LCA(start.target, inEvent.relatedTarget);\n                if (t) {\n                    var e = eventFactory.makeGestureEvent('tap', {\n                        bubbles: true,\n                        cancelable: true,\n                        x: inEvent.clientX,\n                        y: inEvent.clientY,\n                        detail: inEvent.detail,\n                        pointerType: inEvent.pointerType,\n                        pointerId: inEvent.pointerId,\n                        altKey: inEvent.altKey,\n                        ctrlKey: inEvent.ctrlKey,\n                        metaKey: inEvent.metaKey,\n                        shiftKey: inEvent.shiftKey,\n                        _source: 'tap'\n                    });\n                    t.dispatchEvent(e);\n                }\n            }\n            pointermap.delete(inEvent.pointerId);\n        }\n    };\n    // patch eventFactory to remove id from tap's pointermap for preventTap calls\n    eventFactory.preventTap = function(e) {\n        return function() {\n            e.tapPrevented = true;\n            pointermap.delete(e.pointerId);\n        };\n    };\n    dispatcher.registerGesture('tap', tap);\n})(exports);\n\n/*\n * Basic strategy: find the farthest apart points, use as diameter of circle\n * react to size change and rotation of the chord\n */\n\n/**\n * @module pointer-gestures\n * @submodule Events\n * @class pinch\n */\n/**\n * Scale of the pinch zoom gesture\n * @property scale\n * @type Number\n */\n/**\n * Center X position of pointers causing pinch\n * @property centerX\n * @type Number\n */\n/**\n * Center Y position of pointers causing pinch\n * @property centerY\n * @type Number\n */\n\n/**\n * @module pointer-gestures\n * @submodule Events\n * @class rotate\n */\n/**\n * Angle (in degrees) of rotation. Measured from starting positions of pointers.\n * @property angle\n * @type Number\n */\n/**\n * Center X position of pointers causing rotation\n * @property centerX\n * @type Number\n */\n/**\n * Center Y position of pointers causing rotation\n * @property centerY\n * @type Number\n */\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var RAD_TO_DEG = 180 / Math.PI;\n    var pinch = {\n        events: [\n            'down',\n            'up',\n            'move',\n            'cancel'\n        ],\n        exposes: [\n            'pinchstart',\n            'pinch',\n            'pinchend',\n            'rotate'\n        ],\n        defaultActions: {\n            'pinch': 'none',\n            'rotate': 'none'\n        },\n        reference: {},\n        down: function(inEvent) {\n            pointermap.set(inEvent.pointerId, inEvent);\n            if (pointermap.pointers() == 2) {\n                var points = this.calcChord();\n                var angle = this.calcAngle(points);\n                this.reference = {\n                    angle: angle,\n                    diameter: points.diameter,\n                    target: scope.targetFinding.LCA(points.a.target, points.b.target)\n                };\n\n                this.firePinch('pinchstart', points.diameter, points);\n            }\n        },\n        up: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            var num = pointermap.pointers();\n            if (p) {\n                if (num === 2) {\n                    // fire 'pinchend' before deleting pointer\n                    var points = this.calcChord();\n                    this.firePinch('pinchend', points.diameter, points);\n                }\n                pointermap.delete(inEvent.pointerId);\n            }\n        },\n        move: function(inEvent) {\n            if (pointermap.has(inEvent.pointerId)) {\n                pointermap.set(inEvent.pointerId, inEvent);\n                if (pointermap.pointers() > 1) {\n                    this.calcPinchRotate();\n                }\n            }\n        },\n        cancel: function(inEvent) {\n            this.up(inEvent);\n        },\n        firePinch: function(type, diameter, points) {\n            var zoom = diameter / this.reference.diameter;\n            var e = eventFactory.makeGestureEvent(type, {\n                bubbles: true,\n                cancelable: true,\n                scale: zoom,\n                centerX: points.center.x,\n                centerY: points.center.y,\n                _source: 'pinch'\n            });\n            this.reference.target.dispatchEvent(e);\n        },\n        fireRotate: function(angle, points) {\n            var diff = Math.round((angle - this.reference.angle) % 360);\n            var e = eventFactory.makeGestureEvent('rotate', {\n                bubbles: true,\n                cancelable: true,\n                angle: diff,\n                centerX: points.center.x,\n                centerY: points.center.y,\n                _source: 'pinch'\n            });\n            this.reference.target.dispatchEvent(e);\n        },\n        calcPinchRotate: function() {\n            var points = this.calcChord();\n            var diameter = points.diameter;\n            var angle = this.calcAngle(points);\n            if (diameter != this.reference.diameter) {\n                this.firePinch('pinch', diameter, points);\n            }\n            if (angle != this.reference.angle) {\n                this.fireRotate(angle, points);\n            }\n        },\n        calcChord: function() {\n            var pointers = [];\n            pointermap.forEach(function(p) {\n                pointers.push(p);\n            });\n            var dist = 0;\n            // start with at least two pointers\n            var points = {\n                a: pointers[0],\n                b: pointers[1]\n            };\n            var x, y, d;\n            for (var i = 0; i < pointers.length; i++) {\n                var a = pointers[i];\n                for (var j = i + 1; j < pointers.length; j++) {\n                    var b = pointers[j];\n                    x = Math.abs(a.clientX - b.clientX);\n                    y = Math.abs(a.clientY - b.clientY);\n                    d = x + y;\n                    if (d > dist) {\n                        dist = d;\n                        points = {\n                            a: a,\n                            b: b\n                        };\n                    }\n                }\n            }\n            x = Math.abs(points.a.clientX + points.b.clientX) / 2;\n            y = Math.abs(points.a.clientY + points.b.clientY) / 2;\n            points.center = {\n                x: x,\n                y: y\n            };\n            points.diameter = dist;\n            return points;\n        },\n        calcAngle: function(points) {\n            var x = points.a.clientX - points.b.clientX;\n            var y = points.a.clientY - points.b.clientY;\n            return (360 + Math.atan2(y, x) * RAD_TO_DEG) % 360;\n        }\n    };\n    dispatcher.registerGesture('pinch', pinch);\n})(exports);","// list-dragon node module\n// https://github.com/openfin/list-dragon\n\n/* eslint-env node, browser */\n\n'use strict';\n\nvar cssInjector = require('css-injector');\nvar format = require('templex');\n\nvar REVERT_TO_STYLESHEET_VALUE = null;  // null removes the style\n\nvar transform, timer, scrollVelocity, cssListDragon;\n\n/* inject:css */\ncssListDragon = 'div.dragon-list{position:relative;background-color:#fff}div.dragon-list>div,div.dragon-list>ul{position:absolute;left:0;right:0}div.dragon-list>div{text-align:center;background-color:#00796b;color:#fff;box-shadow:0 3px 6px rgba(0,0,0,.16),0 3px 6px rgba(0,0,0,.23);overflow:hidden;white-space:nowrap}div.dragon-list>ul{overflow-y:auto;bottom:0;margin:0;padding:0;box-shadow:0 1px 3px rgba(0,0,0,.12),0 1px 2px rgba(0,0,0,.24)}div.dragon-list>ul>li,li.dragon-pop{white-space:nowrap;list-style-type:none;border:0 solid #f4f4f4;border-bottom:1px solid #e0e0e0;cursor:move;transition:border-top-width .2s}div.dragon-list>ul>li:last-child{height:0;border-bottom:none}li.dragon-pop{position:fixed;background-color:#fff;border:1px solid #e0e0e0;left:0;top:0;overflow-x:hidden;box-shadow:rgba(0,0,0,.188235) 0 10px 20px,rgba(0,0,0,.227451) 0 6px 6px}';\n/* endinject */\n\n/**\n * @constructor ListDragon\n *\n * @desc This object services a set of item lists that allow dragging and dropping items within and between lists in a set.\n *\n * Two strategies are supported:\n *\n * 1. Supply your own HTML markup and let the API build the item models for you.\n *    To use this strategy, script your HTML and provide one of these:\n *    * an array of all the list item (`<li>`) tags\n *    * a CSS selector that points to all the list item tags\n * 2. Supply your own item models and let the API build the HTML markup for you.\n *    To use this strategy, provide an array of model lists.\n *\n * The new ListDragon object's `modelLists` property references the array of model lists the API constructed for you in strategy #1 or the array of model lists you supplied for strategy #2.\n *\n * After the user performs a successful drag-and-drop operation, the position of the model references within the `modelLists` array is rearranged. (The models themselves are the original objects as supplied in the model lists; they are not rebuilt or altered in any way. Just the references to them are moved around.)\n *\n * @param {string|Element[]|modelListType[]} selectorOrModelLists - You must supply one of the items in **bold** below:\n *\n * 1. _For strategy #1 above (API creates models from supplied elements):_ All the list item (`<li>`) DOM elements of all the lists you want the new object to manage, as either:\n *    1. **A CSS selector;** _or_\n *    2. **An array of DOM elements**\n * 2. _For strategy #2 above (API creates elements from supplied models):_ **An array of model lists,** each of which is in one of the following forms:\n *    1. An array of item models (with various option properties hanging off of it); _and/or_\n *    2. A {@link modelListType} object with those same various option properties including the required `models` property containing that same array of item models.\n *\n * In either case (2.1 or 2.2), each element of such arrays of item models may take the form of:\n * * A string primitive; _or_\n * * A {@link itemModelType} object with a various option properties including the required `label` property containing a string primitive.\n *\n * Regarding these string primitives, each is either:\n * * A string to be displayed in the list item; _or_\n * * A format string with other property values merged in, the result of which is to be displayed in the list item.\n *\n * @param {object} [options={}] - You may supply \"global\" template variables here, representing the \"outer scope,\" after first searching each model and then each model list.\n * @param {undefined|null|Element|string} [cssStylesheetReferenceElement] - Determines where to insert the stylesheet. (This is the only formal option.) Passed to css-injector, the overloads are (from css-injector docs):\n * * `undefined` type (or omitted): injects stylesheet at top of `<head>...</head>` element\n * * `null` value: injects stylesheet at bottom of `<head>...</head>` element\n * * `Element` type: injects stylesheet immediately before given element, wherever it is found.\n * * `string` type: injects stylesheet immediately before given first element found that matches the given css selector.\n */\nfunction ListDragon(selectorOrModelLists, options) {\n\n    if (!(this instanceof ListDragon)) {\n        throw error('Not called with \"new\" keyword.');\n    }\n\n    var self = this, modelLists, items;\n\n    options = options || {};\n\n    if (typeof selectorOrModelLists === 'string') {\n        items = toArray(document.querySelectorAll(selectorOrModelLists));\n        modelLists = createModelListsFromListElements(items);\n    } else if (selectorOrModelLists[0] instanceof Element) {\n        items = toArray(selectorOrModelLists);\n        modelLists = createModelListsFromListElements(items);\n    } else {\n        // param is array of model lists\n        // build new <ul> element(s) for each list and put in `.modelLists`;\n        // fill `.items` array with <li> elements from these new <ul> elements\n        items = [];\n        modelLists = createListElementsFromModelLists(selectorOrModelLists, options);\n        modelLists.forEach(function (list) {\n            items = items.concat(toArray(list.element.querySelectorAll('li')));\n        });\n    }\n\n    // grab wheel events and don't let 'em bubble\n    modelLists.forEach(function (modelList) {\n        modelList.element.addEventListener('wheel', captureEvent);\n    });\n\n    items.forEach(function (itemElement, index) {\n        var item = (itemElement !== itemElement.parentElement.lastElementChild)\n            ? self.addEvt(itemElement, 'mousedown', itemElement, true)\n            : { element: itemElement };\n\n        /* `item.model` not currently needed so commented out here.\n         * (Originally used for rebuilding modelLists for final\n         * reporting, modelLists are now spliced on every successful\n         * drag-and-drop operation so they're always up to date.)\n\n         var origin = this.itemCoordinates(itemElement);\n         item.model = this.modelLists[origin.list].models[origin.item];\n\n         */\n\n        items[index] = item;\n    });\n\n    transform = 'transform' in items[0].element.style\n        ? 'transform' // Chrome 45 and Firefox 40\n        : '-webkit-transform'; // Safari 8\n\n    // set up the new object\n    this.modelLists = modelLists;\n    this.items = items;\n    this.bindings = {};\n    this.callback = {};\n\n    cssInjector(cssListDragon, 'list-dragon-base', options.cssStylesheetReferenceElement);\n\n}\n\nListDragon.prototype = {\n\n    addEvt: function (target, type, listener, doNotBind) {\n        var binding = {\n            handler: handlers[type].bind(target, this),\n            element: listener || window\n        };\n\n        if (!doNotBind) {\n            this.bindings[type] = binding;\n        }\n\n        binding.element.addEventListener(type, binding.handler);\n\n        return binding;\n    },\n\n    removeEvt: function (type) {\n        var binding = this.bindings[type];\n        delete this.bindings[type];\n        binding.element.removeEventListener(type, binding.handler);\n    },\n\n    removeAllEventListeners: function () {\n        // remove drag & drop events (mousemove, mouseup, and transitionend)\n        for (var type in this.bindings) {\n            var binding = this.bindings[type];\n            binding.element.removeEventListener(type, binding.handler);\n        }\n        // remove the mousedown events from all list items\n        this.items.forEach(function (item) {\n            if (item.handler) {\n                item.element.removeEventListener('mousedown', item.handler);\n            }\n        });\n        // wheel events on the list elements\n        this.modelLists.forEach(function (modelList) {\n            modelList.element.removeEventListener('wheel', captureEvent);\n        });\n    },\n\n    pointInListRects: function (point) {\n        return this.modelLists.find(function (modelList) {\n            var rect = modelList.element.getBoundingClientRect();\n\n            rect = {\n                left:   window.scrollX + rect.left,\n                top:    window.scrollY + rect.top,\n                right:  window.scrollX + rect.right,\n                bottom: window.scrollY + rect.bottom,\n                width:  rect.width,\n                height: rect.height\n            };\n\n            modelList.rect = rect;\n\n            if (pointInRect(point, rect)) {\n                modelList.rect = rect;\n                return true; // found\n            } else {\n                return false;\n            }\n        });\n    },\n\n    pointInItemRects: function (point, except1, except2) {\n        return this.items.find(function (item) {\n            var element = item.element;\n            return (\n                element !== except1 &&\n                element !== except2 &&\n                pointInRect(point, item.rect)\n            );\n        });\n    },\n\n    // get positions of all list items in page coords (normalized for window and list scrolling)\n    getAllItemBoundingRects: function () {\n        var modelLists = this.modelLists, height;\n        this.items.forEach(function (item) {\n            var itemElement = item.element,\n                listElement = itemElement.parentElement,\n                list = modelLists.find(function (list) { return list.element === listElement; });\n\n            if (\n                // omitted: default to true\n                list.isDropTarget === undefined ||\n\n                // function: use return value\n                typeof list.isDropTarget === 'function' && list.isDropTarget() ||\n\n                // otherwise: use truthiness of given value\n                list.isDropTarget\n            ) {\n                var rect = itemElement.getBoundingClientRect(),\n                    bottom = rect.bottom;\n\n                if (itemElement === listElement.lastElementChild) {\n                    bottom = listElement.getBoundingClientRect().bottom;\n                    if (bottom < rect.top) {\n                        bottom = rect.top + (height || 50);\n                    }\n                } else {\n                    height = rect.height;\n                }\n\n                rect = {\n                    left:   window.scrollX + rect.left,\n                    right:  window.scrollX + rect.right,\n                    top:    window.scrollY + rect.top    + listElement.scrollTop,\n                    bottom: window.scrollY + bottom + listElement.scrollTop\n                };\n\n                item.rect = rect;\n            }\n        });\n    },\n\n    reinsert: function (target) {\n        var style = target.style;\n        style.width = style[transform] = style.transition = REVERT_TO_STYLESHEET_VALUE;\n\n        target.classList.remove('dragon-pop');\n\n        this.drop.style.transitionDuration = '0s';\n        this.drop.style.borderTopWidth = REVERT_TO_STYLESHEET_VALUE;\n        this.drop.parentElement.insertBefore(target, this.drop);\n\n        delete this.drop;\n    },\n\n    // return an object { item: <item index within list>, list: <list index within list of lists> }\n    itemCoordinates: function (item) {\n        var listElement = item.parentElement,\n            coords = { item: 0 };\n\n        while ((item = item.previousElementSibling)) {\n            ++coords.item;\n        }\n\n        this.modelLists.find(function (list, index) {\n            coords.list = index;\n            return list.element === listElement; // stop when we find the one we belong to\n        });\n\n        return coords;\n    }\n\n};\n\nvar handlers = {\n    mousedown: function (dragon, evt) {\n\n        evt.stopPropagation();\n        evt.preventDefault();  //prevents user selection of rendered nodes during drag\n\n        if (dragon.drop) {\n            return;\n        }\n\n        var rect = this.getBoundingClientRect();\n\n        dragon.rect = rect = {\n            left:   Math.round(rect.left - 1),\n            top:    Math.round(rect.top - 1),\n            right:  Math.round(rect.right),\n            bottom: Math.round(rect.bottom),\n            width:  Math.round(rect.width),\n            height: Math.round(rect.height)\n        };\n\n        dragon.pin = {\n            x: window.scrollX + evt.clientX,\n            y: window.scrollY + evt.clientY\n        };\n\n        dragon.origin = dragon.itemCoordinates(this);\n\n        if (dragon.callback.grabbed) {\n            dragon.callback.grabbed.call(this, dragon);\n        }\n\n        dragon.getAllItemBoundingRects();\n\n        dragon.drop = this.nextElementSibling;\n        dragon.drop.style.transitionDuration = '0s';\n        dragon.drop.style.borderTopWidth = rect.height + 'px';\n\n        this.style.width = rect.width + 'px';\n        this.style.transitionDuration = '0s';\n        this.style[transform] = translate(\n            rect.left - window.scrollX,\n            rect.top  - window.scrollY\n        );\n        this.classList.add('dragon-pop');\n        this.style.zIndex = window.getComputedStyle(dragon.modelLists[0].container.parentElement).zIndex;\n\n        if (!dragon.container) {\n            // walk back to closest shadow root OR body tag OR root tag\n            var container = this;\n            while (container.parentNode) {\n                container = container.parentNode;\n                if (container instanceof ShadowRoot || container.tagName === 'BODY'){\n                    break;\n                }\n            }\n            dragon.container = container;\n        }\n\n        dragon.container.appendChild(this);\n\n        rect.left   += window.scrollX;\n        rect.top    += window.scrollY;\n        rect.right  += window.scrollX;\n        rect.bottom += window.scrollY;\n\n        dragon.addEvt(this, 'mousemove');\n        dragon.addEvt(this, 'mouseup');\n    },\n\n    mousemove: function (dragon, evt) {\n        dragon.drop.style.transition = REVERT_TO_STYLESHEET_VALUE;\n\n        var hoverList = dragon.pointInListRects({ x: evt.clientX, y: evt.clientY }) || dragon.mostRecentHoverList;\n\n        if (hoverList) {\n            var dx = evt.clientX - dragon.pin.x,\n                dy = evt.clientY - dragon.pin.y;\n\n            dragon.mostRecentHoverList = hoverList;\n\n            var maxScrollY = hoverList.element.scrollHeight - hoverList.rect.height,\n                y = evt.clientY + window.scrollY,\n                magnitude;\n\n            if (maxScrollY > 0) {\n                // list is scrollable (is taller than rect)\n                if (hoverList.element.scrollTop > 0 && (magnitude = y - (hoverList.rect.top + 5)) < 0) {\n                    // mouse near or above top and list is not scrolled to top yet\n                    resetAutoScrollTimer(magnitude, 0, hoverList.element);\n                } else if (hoverList.element.scrollTop < maxScrollY && (magnitude = y - (hoverList.rect.bottom - 1 - 5)) > 0) {\n                    // mouse near or below bottom and list not scrolled to bottom yet\n                    resetAutoScrollTimer(magnitude, maxScrollY, hoverList.element);\n                } else {\n                    // mouse inside\n                    resetAutoScrollTimer();\n                }\n            }\n\n            var other = dragon.pointInItemRects({\n                x: evt.clientX,\n                y: dragon.rect.bottom + window.scrollY + dy + hoverList.element.scrollTop\n            }, this, dragon.drop);\n\n            this.style[transform] = translate(\n                dragon.rect.left - window.scrollX + dx,\n                dragon.rect.top - window.scrollY + dy\n            );\n\n            if (other) {\n                var element = other.element;\n                element.style.transition = REVERT_TO_STYLESHEET_VALUE;\n                element.style.borderTopWidth = dragon.drop.style.borderTopWidth;\n                dragon.drop.style.borderTopWidth = null;\n                dragon.drop = element;\n            }\n        }\n    },\n\n    mouseup: function (dragon, evt) {\n        resetAutoScrollTimer();\n        dragon.removeEvt('mousemove');\n        dragon.removeEvt('mouseup');\n\n        evt.stopPropagation();\n\n        var newRect = this.getBoundingClientRect();\n\n        if (\n            window.scrollX + newRect.left === dragon.rect.left &&\n            window.scrollY + newRect.top === dragon.rect.top\n        ) {\n            dragon.reinsert(this);\n        } else {\n            var dropRect = dragon.drop.getBoundingClientRect();\n\n            dragon.addEvt(this, 'transitionend', this);\n            this.style.transitionDuration = REVERT_TO_STYLESHEET_VALUE; //reverts to 200ms\n            this.style.transitionProperty = transform;\n            this.style[transform] = translate(\n                dropRect.left - window.scrollX,\n                dropRect.top - window.scrollY\n            );\n        }\n    },\n\n    transitionend: function (dragon, evt) {\n        if (evt.propertyName === transform) {\n            dragon.removeEvt('transitionend');\n            dragon.reinsert(this);\n\n            this.style.transitionProperty = REVERT_TO_STYLESHEET_VALUE; //reverts to border-top-width\n\n            var model = dragon.modelLists[dragon.origin.list].splice(dragon.origin.item, 1)[0];\n            var destination = dragon.itemCoordinates(this);\n            dragon.modelLists[destination.list].splice(destination.item, 0, model);\n\n            if (dragon.callback.dropped) {\n                dragon.callback.dropped.call(this, dragon);\n            }\n        }\n    }\n};\n\nfunction resetAutoScrollTimer(magnitude, limit, element) {\n    if (!magnitude) {\n        clearInterval(timer);\n        scrollVelocity = 0;\n    } else {\n        var changeDirection =\n            scrollVelocity  <  0 && magnitude  >= 0 ||\n            scrollVelocity === 0 && magnitude !== 0 ||\n            scrollVelocity  >  0 && magnitude  <= 0;\n        scrollVelocity = magnitude > 0 ? Math.min(50, magnitude) : Math.max(-50, magnitude);\n        if (changeDirection) {\n            clearInterval(timer);\n            timer = setInterval(function (limit) {\n                var scrollTop = element.scrollTop + scrollVelocity;\n                if (scrollVelocity < 0 && scrollTop < limit || scrollVelocity > 0 && scrollTop > limit) {\n                    element.scrollTop = limit;\n                    clearInterval(timer);\n                } else {\n                    element.scrollTop = scrollTop;\n                }\n            }, 125);\n        }\n    }\n}\n\nfunction toArray(arrayLikeObject) {\n    return Array.prototype.slice.call(arrayLikeObject);\n}\n\nfunction pointInRect(point, rect) {\n    return rect.top <= point.y && point.y <= rect.bottom\n        && rect.left <= point.x && point.x <= rect.right;\n}\n\nfunction translate(left, top) {\n    return 'translate('\n        + Math.floor(left + window.scrollX) + 'px,'\n        + Math.floor(top + window.scrollY) + 'px)';\n}\n\nfunction htmlEncode(string) {\n    var textNode = document.createTextNode(string);\n\n    return document\n        .createElement('a')\n        .appendChild(textNode)\n        .parentNode\n        .innerHTML;\n}\n\n/**\n * Creates `<ul>...</ul>` elements and inserts them into an `element` property on each model.\n * @param {object} modelLists\n * @returns `modelLists`\n */\nfunction createListElementsFromModelLists(modelLists, options) {\n    var templateLabel = options.label || '{label}';\n\n    modelLists.forEach(function (modelList, listIndex) {\n        var listLabel = modelList.label || templateLabel,\n            listHtmlEncode = modelList.htmlEncode !== undefined && modelList.htmlEncode || options.htmlEncode,\n            container = document.createElement('div'),\n            listElement = document.createElement('ul');\n\n        if (modelList.models) {\n            Object.keys(modelList).forEach(function (key) {\n                if (key !== 'models') {\n                    modelList.models[key] = modelList[key];\n                }\n            });\n            modelLists[listIndex] = modelList = modelList.models;\n        } else if (modelList instanceof Array) {\n            modelList.models = modelList; // point to self\n        } else {\n            throw error('List [{1}] not an array of models (with or without additional properties) OR ' +\n                'an object (with a `models` property containing an array of models).', listIndex);\n        }\n\n        modelList.forEach(function (model) {\n            var modelLabel = model.label || listLabel,\n                modelHtmlEncode = model.htmlEncode !== undefined && model.htmlEncode || listHtmlEncode,\n                modelObject = typeof model === 'object' ? model : { label: model},\n                label = format.call([modelObject, modelList, options], modelLabel),\n                itemElement = document.createElement('li');\n\n            itemElement.innerHTML = modelHtmlEncode ? htmlEncode(label) : label;\n\n            listElement.appendChild(itemElement);\n        });\n\n        // append the final \"fencepost\" item -- drop target at bottom of list after all items\n        var itemElement = document.createElement('li');\n        itemElement.innerHTML = '&nbsp;';\n        listElement.appendChild(itemElement);\n\n        // append header to container\n        if (modelList.title) {\n            var header = document.createElement('div');\n            header.innerHTML = listHtmlEncode ? htmlEncode(modelList.title) : modelList.title;\n            container.appendChild(header);\n        }\n\n        container.appendChild(listElement);\n        container.className = modelList.cssClassNames || options.cssClassNames || 'dragon-list';\n        modelList.element = listElement;\n        modelList.container = container;\n    });\n\n    return modelLists;\n}\n\n/**\n * Create a `.modelLists` array with these <li> elements' parent <ul> elements\n * @param {Element[]} listItemElements\n * @returns {Array}\n */\nfunction createModelListsFromListElements(listItemElements) {\n    var modelLists = [];\n\n    listItemElements.forEach(function (itemElement) {\n        var listElement = itemElement.parentElement,\n            container = listElement.parentElement,\n            models = [];\n        if (!modelLists.find(function (list) { return list.element === listElement; })) {\n            toArray(listElement.querySelectorAll('li')).forEach(function (itemElement) {\n                if (itemElement !== listElement.lastElementChild) {\n                    models.push(itemElement.innerHTML);\n                }\n            });\n            models.element = listElement;\n            models.container = container;\n            modelLists.push(models);\n        }\n    });\n\n    return modelLists;\n}\n\nfunction captureEvent(evt) {\n    evt.stopPropagation();\n}\n\nfunction error() {\n    return 'list-dragon: ' + format.apply(this, Array.prototype.slice.call(arguments));\n}\n\n// this interface consists solely of the prototypal object constructor\nmodule.exports = ListDragon;\n",";(function () { // closure for web browsers\n\nif (typeof module === 'object' && module.exports) {\n  module.exports = LRUCache\n} else {\n  // just set the global for non-node platforms.\n  this.LRUCache = LRUCache\n}\n\nfunction hOP (obj, key) {\n  return Object.prototype.hasOwnProperty.call(obj, key)\n}\n\nfunction naiveLength () { return 1 }\n\nvar didTypeWarning = false\nfunction typeCheckKey(key) {\n  if (!didTypeWarning && typeof key !== 'string' && typeof key !== 'number') {\n    didTypeWarning = true\n    console.error(new TypeError(\"LRU: key must be a string or number. Almost certainly a bug! \" + typeof key).stack)\n  }\n}\n\nfunction LRUCache (options) {\n  if (!(this instanceof LRUCache))\n    return new LRUCache(options)\n\n  if (typeof options === 'number')\n    options = { max: options }\n\n  if (!options)\n    options = {}\n\n  this._max = options.max\n  // Kind of weird to have a default max of Infinity, but oh well.\n  if (!this._max || !(typeof this._max === \"number\") || this._max <= 0 )\n    this._max = Infinity\n\n  this._lengthCalculator = options.length || naiveLength\n  if (typeof this._lengthCalculator !== \"function\")\n    this._lengthCalculator = naiveLength\n\n  this._allowStale = options.stale || false\n  this._maxAge = options.maxAge || null\n  this._dispose = options.dispose\n  this.reset()\n}\n\n// resize the cache when the max changes.\nObject.defineProperty(LRUCache.prototype, \"max\",\n  { set : function (mL) {\n      if (!mL || !(typeof mL === \"number\") || mL <= 0 ) mL = Infinity\n      this._max = mL\n      if (this._length > this._max) trim(this)\n    }\n  , get : function () { return this._max }\n  , enumerable : true\n  })\n\n// resize the cache when the lengthCalculator changes.\nObject.defineProperty(LRUCache.prototype, \"lengthCalculator\",\n  { set : function (lC) {\n      if (typeof lC !== \"function\") {\n        this._lengthCalculator = naiveLength\n        this._length = this._itemCount\n        for (var key in this._cache) {\n          this._cache[key].length = 1\n        }\n      } else {\n        this._lengthCalculator = lC\n        this._length = 0\n        for (var key in this._cache) {\n          this._cache[key].length = this._lengthCalculator(this._cache[key].value)\n          this._length += this._cache[key].length\n        }\n      }\n\n      if (this._length > this._max) trim(this)\n    }\n  , get : function () { return this._lengthCalculator }\n  , enumerable : true\n  })\n\nObject.defineProperty(LRUCache.prototype, \"length\",\n  { get : function () { return this._length }\n  , enumerable : true\n  })\n\n\nObject.defineProperty(LRUCache.prototype, \"itemCount\",\n  { get : function () { return this._itemCount }\n  , enumerable : true\n  })\n\nLRUCache.prototype.forEach = function (fn, thisp) {\n  thisp = thisp || this\n  var i = 0\n  var itemCount = this._itemCount\n\n  for (var k = this._mru - 1; k >= 0 && i < itemCount; k--) if (this._lruList[k]) {\n    i++\n    var hit = this._lruList[k]\n    if (isStale(this, hit)) {\n      del(this, hit)\n      if (!this._allowStale) hit = undefined\n    }\n    if (hit) {\n      fn.call(thisp, hit.value, hit.key, this)\n    }\n  }\n}\n\nLRUCache.prototype.keys = function () {\n  var keys = new Array(this._itemCount)\n  var i = 0\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    keys[i++] = hit.key\n  }\n  return keys\n}\n\nLRUCache.prototype.values = function () {\n  var values = new Array(this._itemCount)\n  var i = 0\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    values[i++] = hit.value\n  }\n  return values\n}\n\nLRUCache.prototype.reset = function () {\n  if (this._dispose && this._cache) {\n    for (var k in this._cache) {\n      this._dispose(k, this._cache[k].value)\n    }\n  }\n\n  this._cache = Object.create(null) // hash of items by key\n  this._lruList = Object.create(null) // list of items in order of use recency\n  this._mru = 0 // most recently used\n  this._lru = 0 // least recently used\n  this._length = 0 // number of items in the list\n  this._itemCount = 0\n}\n\nLRUCache.prototype.dump = function () {\n  var arr = []\n  var i = 0\n\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    if (!isStale(this, hit)) {\n      //Do not store staled hits\n      ++i\n      arr.push({\n        k: hit.key,\n        v: hit.value,\n        e: hit.now + (hit.maxAge || 0)\n      });\n    }\n  }\n  //arr has the most read first\n  return arr\n}\n\nLRUCache.prototype.dumpLru = function () {\n  return this._lruList\n}\n\nLRUCache.prototype.set = function (key, value, maxAge) {\n  maxAge = maxAge || this._maxAge\n  typeCheckKey(key)\n\n  var now = maxAge ? Date.now() : 0\n  var len = this._lengthCalculator(value)\n\n  if (hOP(this._cache, key)) {\n    if (len > this._max) {\n      del(this, this._cache[key])\n      return false\n    }\n    // dispose of the old one before overwriting\n    if (this._dispose)\n      this._dispose(key, this._cache[key].value)\n\n    this._cache[key].now = now\n    this._cache[key].maxAge = maxAge\n    this._cache[key].value = value\n    this._length += (len - this._cache[key].length)\n    this._cache[key].length = len\n    this.get(key)\n\n    if (this._length > this._max)\n      trim(this)\n\n    return true\n  }\n\n  var hit = new Entry(key, value, this._mru++, len, now, maxAge)\n\n  // oversized objects fall out of cache automatically.\n  if (hit.length > this._max) {\n    if (this._dispose) this._dispose(key, value)\n    return false\n  }\n\n  this._length += hit.length\n  this._lruList[hit.lu] = this._cache[key] = hit\n  this._itemCount ++\n\n  if (this._length > this._max)\n    trim(this)\n\n  return true\n}\n\nLRUCache.prototype.has = function (key) {\n  typeCheckKey(key)\n  if (!hOP(this._cache, key)) return false\n  var hit = this._cache[key]\n  if (isStale(this, hit)) {\n    return false\n  }\n  return true\n}\n\nLRUCache.prototype.get = function (key) {\n  typeCheckKey(key)\n  return get(this, key, true)\n}\n\nLRUCache.prototype.peek = function (key) {\n  typeCheckKey(key)\n  return get(this, key, false)\n}\n\nLRUCache.prototype.pop = function () {\n  var hit = this._lruList[this._lru]\n  del(this, hit)\n  return hit || null\n}\n\nLRUCache.prototype.del = function (key) {\n  typeCheckKey(key)\n  del(this, this._cache[key])\n}\n\nLRUCache.prototype.load = function (arr) {\n  //reset the cache\n  this.reset();\n\n  var now = Date.now()\n  //A previous serialized cache has the most recent items first\n  for (var l = arr.length - 1; l >= 0; l-- ) {\n    var hit = arr[l]\n    typeCheckKey(hit.k)\n    var expiresAt = hit.e || 0\n    if (expiresAt === 0) {\n      //the item was created without expiration in a non aged cache\n      this.set(hit.k, hit.v)\n    } else {\n      var maxAge = expiresAt - now\n      //dont add already expired items\n      if (maxAge > 0) this.set(hit.k, hit.v, maxAge)\n    }\n  }\n}\n\nfunction get (self, key, doUse) {\n  typeCheckKey(key)\n  var hit = self._cache[key]\n  if (hit) {\n    if (isStale(self, hit)) {\n      del(self, hit)\n      if (!self._allowStale) hit = undefined\n    } else {\n      if (doUse) use(self, hit)\n    }\n    if (hit) hit = hit.value\n  }\n  return hit\n}\n\nfunction isStale(self, hit) {\n  if (!hit || (!hit.maxAge && !self._maxAge)) return false\n  var stale = false;\n  var diff = Date.now() - hit.now\n  if (hit.maxAge) {\n    stale = diff > hit.maxAge\n  } else {\n    stale = self._maxAge && (diff > self._maxAge)\n  }\n  return stale;\n}\n\nfunction use (self, hit) {\n  shiftLU(self, hit)\n  hit.lu = self._mru ++\n  self._lruList[hit.lu] = hit\n}\n\nfunction trim (self) {\n  while (self._lru < self._mru && self._length > self._max)\n    del(self, self._lruList[self._lru])\n}\n\nfunction shiftLU (self, hit) {\n  delete self._lruList[ hit.lu ]\n  while (self._lru < self._mru && !self._lruList[self._lru]) self._lru ++\n}\n\nfunction del (self, hit) {\n  if (hit) {\n    if (self._dispose) self._dispose(hit.key, hit.value)\n    self._length -= hit.length\n    self._itemCount --\n    delete self._cache[ hit.key ]\n    shiftLU(self, hit)\n  }\n}\n\n// classy, since V8 prefers predictable objects.\nfunction Entry (key, value, lu, length, now, maxAge) {\n  this.key = key\n  this.value = value\n  this.lu = lu\n  this.length = length\n  this.now = now\n  if (maxAge) this.maxAge = maxAge\n}\n\n})()\n","/*!\n * mustache.js - Logic-less {{mustache}} templates with JavaScript\n * http://github.com/janl/mustache.js\n */\n\n/*global define: false Mustache: true*/\n\n(function defineMustache (global, factory) {\n  if (typeof exports === 'object' && exports && typeof exports.nodeName !== 'string') {\n    factory(exports); // CommonJS\n  } else if (typeof define === 'function' && define.amd) {\n    define(['exports'], factory); // AMD\n  } else {\n    global.Mustache = {};\n    factory(global.Mustache); // script, wsh, asp\n  }\n}(this, function mustacheFactory (mustache) {\n\n  var objectToString = Object.prototype.toString;\n  var isArray = Array.isArray || function isArrayPolyfill (object) {\n    return objectToString.call(object) === '[object Array]';\n  };\n\n  function isFunction (object) {\n    return typeof object === 'function';\n  }\n\n  /**\n   * More correct typeof string handling array\n   * which normally returns typeof 'object'\n   */\n  function typeStr (obj) {\n    return isArray(obj) ? 'array' : typeof obj;\n  }\n\n  function escapeRegExp (string) {\n    return string.replace(/[\\-\\[\\]{}()*+?.,\\\\\\^$|#\\s]/g, '\\\\$&');\n  }\n\n  /**\n   * Null safe way of checking whether or not an object,\n   * including its prototype, has a given property\n   */\n  function hasProperty (obj, propName) {\n    return obj != null && typeof obj === 'object' && (propName in obj);\n  }\n\n  // Workaround for https://issues.apache.org/jira/browse/COUCHDB-577\n  // See https://github.com/janl/mustache.js/issues/189\n  var regExpTest = RegExp.prototype.test;\n  function testRegExp (re, string) {\n    return regExpTest.call(re, string);\n  }\n\n  var nonSpaceRe = /\\S/;\n  function isWhitespace (string) {\n    return !testRegExp(nonSpaceRe, string);\n  }\n\n  var entityMap = {\n    '&': '&amp;',\n    '<': '&lt;',\n    '>': '&gt;',\n    '\"': '&quot;',\n    \"'\": '&#39;',\n    '/': '&#x2F;',\n    '`': '&#x60;',\n    '=': '&#x3D;'\n  };\n\n  function escapeHtml (string) {\n    return String(string).replace(/[&<>\"'`=\\/]/g, function fromEntityMap (s) {\n      return entityMap[s];\n    });\n  }\n\n  var whiteRe = /\\s*/;\n  var spaceRe = /\\s+/;\n  var equalsRe = /\\s*=/;\n  var curlyRe = /\\s*\\}/;\n  var tagRe = /#|\\^|\\/|>|\\{|&|=|!/;\n\n  /**\n   * Breaks up the given `template` string into a tree of tokens. If the `tags`\n   * argument is given here it must be an array with two string values: the\n   * opening and closing tags used in the template (e.g. [ \"<%\", \"%>\" ]). Of\n   * course, the default is to use mustaches (i.e. mustache.tags).\n   *\n   * A token is an array with at least 4 elements. The first element is the\n   * mustache symbol that was used inside the tag, e.g. \"#\" or \"&\". If the tag\n   * did not contain a symbol (i.e. {{myValue}}) this element is \"name\". For\n   * all text that appears outside a symbol this element is \"text\".\n   *\n   * The second element of a token is its \"value\". For mustache tags this is\n   * whatever else was inside the tag besides the opening symbol. For text tokens\n   * this is the text itself.\n   *\n   * The third and fourth elements of the token are the start and end indices,\n   * respectively, of the token in the original template.\n   *\n   * Tokens that are the root node of a subtree contain two more elements: 1) an\n   * array of tokens in the subtree and 2) the index in the original template at\n   * which the closing tag for that section begins.\n   */\n  function parseTemplate (template, tags) {\n    if (!template)\n      return [];\n\n    var sections = [];     // Stack to hold section tokens\n    var tokens = [];       // Buffer to hold the tokens\n    var spaces = [];       // Indices of whitespace tokens on the current line\n    var hasTag = false;    // Is there a {{tag}} on the current line?\n    var nonSpace = false;  // Is there a non-space char on the current line?\n\n    // Strips all whitespace tokens array for the current line\n    // if there was a {{#tag}} on it and otherwise only space.\n    function stripSpace () {\n      if (hasTag && !nonSpace) {\n        while (spaces.length)\n          delete tokens[spaces.pop()];\n      } else {\n        spaces = [];\n      }\n\n      hasTag = false;\n      nonSpace = false;\n    }\n\n    var openingTagRe, closingTagRe, closingCurlyRe;\n    function compileTags (tagsToCompile) {\n      if (typeof tagsToCompile === 'string')\n        tagsToCompile = tagsToCompile.split(spaceRe, 2);\n\n      if (!isArray(tagsToCompile) || tagsToCompile.length !== 2)\n        throw new Error('Invalid tags: ' + tagsToCompile);\n\n      openingTagRe = new RegExp(escapeRegExp(tagsToCompile[0]) + '\\\\s*');\n      closingTagRe = new RegExp('\\\\s*' + escapeRegExp(tagsToCompile[1]));\n      closingCurlyRe = new RegExp('\\\\s*' + escapeRegExp('}' + tagsToCompile[1]));\n    }\n\n    compileTags(tags || mustache.tags);\n\n    var scanner = new Scanner(template);\n\n    var start, type, value, chr, token, openSection;\n    while (!scanner.eos()) {\n      start = scanner.pos;\n\n      // Match any text between tags.\n      value = scanner.scanUntil(openingTagRe);\n\n      if (value) {\n        for (var i = 0, valueLength = value.length; i < valueLength; ++i) {\n          chr = value.charAt(i);\n\n          if (isWhitespace(chr)) {\n            spaces.push(tokens.length);\n          } else {\n            nonSpace = true;\n          }\n\n          tokens.push([ 'text', chr, start, start + 1 ]);\n          start += 1;\n\n          // Check for whitespace on the current line.\n          if (chr === '\\n')\n            stripSpace();\n        }\n      }\n\n      // Match the opening tag.\n      if (!scanner.scan(openingTagRe))\n        break;\n\n      hasTag = true;\n\n      // Get the tag type.\n      type = scanner.scan(tagRe) || 'name';\n      scanner.scan(whiteRe);\n\n      // Get the tag value.\n      if (type === '=') {\n        value = scanner.scanUntil(equalsRe);\n        scanner.scan(equalsRe);\n        scanner.scanUntil(closingTagRe);\n      } else if (type === '{') {\n        value = scanner.scanUntil(closingCurlyRe);\n        scanner.scan(curlyRe);\n        scanner.scanUntil(closingTagRe);\n        type = '&';\n      } else {\n        value = scanner.scanUntil(closingTagRe);\n      }\n\n      // Match the closing tag.\n      if (!scanner.scan(closingTagRe))\n        throw new Error('Unclosed tag at ' + scanner.pos);\n\n      token = [ type, value, start, scanner.pos ];\n      tokens.push(token);\n\n      if (type === '#' || type === '^') {\n        sections.push(token);\n      } else if (type === '/') {\n        // Check section nesting.\n        openSection = sections.pop();\n\n        if (!openSection)\n          throw new Error('Unopened section \"' + value + '\" at ' + start);\n\n        if (openSection[1] !== value)\n          throw new Error('Unclosed section \"' + openSection[1] + '\" at ' + start);\n      } else if (type === 'name' || type === '{' || type === '&') {\n        nonSpace = true;\n      } else if (type === '=') {\n        // Set the tags for the next time around.\n        compileTags(value);\n      }\n    }\n\n    // Make sure there are no open sections when we're done.\n    openSection = sections.pop();\n\n    if (openSection)\n      throw new Error('Unclosed section \"' + openSection[1] + '\" at ' + scanner.pos);\n\n    return nestTokens(squashTokens(tokens));\n  }\n\n  /**\n   * Combines the values of consecutive text tokens in the given `tokens` array\n   * to a single token.\n   */\n  function squashTokens (tokens) {\n    var squashedTokens = [];\n\n    var token, lastToken;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      token = tokens[i];\n\n      if (token) {\n        if (token[0] === 'text' && lastToken && lastToken[0] === 'text') {\n          lastToken[1] += token[1];\n          lastToken[3] = token[3];\n        } else {\n          squashedTokens.push(token);\n          lastToken = token;\n        }\n      }\n    }\n\n    return squashedTokens;\n  }\n\n  /**\n   * Forms the given array of `tokens` into a nested tree structure where\n   * tokens that represent a section have two additional items: 1) an array of\n   * all tokens that appear in that section and 2) the index in the original\n   * template that represents the end of that section.\n   */\n  function nestTokens (tokens) {\n    var nestedTokens = [];\n    var collector = nestedTokens;\n    var sections = [];\n\n    var token, section;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      token = tokens[i];\n\n      switch (token[0]) {\n        case '#':\n        case '^':\n          collector.push(token);\n          sections.push(token);\n          collector = token[4] = [];\n          break;\n        case '/':\n          section = sections.pop();\n          section[5] = token[2];\n          collector = sections.length > 0 ? sections[sections.length - 1][4] : nestedTokens;\n          break;\n        default:\n          collector.push(token);\n      }\n    }\n\n    return nestedTokens;\n  }\n\n  /**\n   * A simple string scanner that is used by the template parser to find\n   * tokens in template strings.\n   */\n  function Scanner (string) {\n    this.string = string;\n    this.tail = string;\n    this.pos = 0;\n  }\n\n  /**\n   * Returns `true` if the tail is empty (end of string).\n   */\n  Scanner.prototype.eos = function eos () {\n    return this.tail === '';\n  };\n\n  /**\n   * Tries to match the given regular expression at the current position.\n   * Returns the matched text if it can match, the empty string otherwise.\n   */\n  Scanner.prototype.scan = function scan (re) {\n    var match = this.tail.match(re);\n\n    if (!match || match.index !== 0)\n      return '';\n\n    var string = match[0];\n\n    this.tail = this.tail.substring(string.length);\n    this.pos += string.length;\n\n    return string;\n  };\n\n  /**\n   * Skips all text until the given regular expression can be matched. Returns\n   * the skipped string, which is the entire tail if no match can be made.\n   */\n  Scanner.prototype.scanUntil = function scanUntil (re) {\n    var index = this.tail.search(re), match;\n\n    switch (index) {\n      case -1:\n        match = this.tail;\n        this.tail = '';\n        break;\n      case 0:\n        match = '';\n        break;\n      default:\n        match = this.tail.substring(0, index);\n        this.tail = this.tail.substring(index);\n    }\n\n    this.pos += match.length;\n\n    return match;\n  };\n\n  /**\n   * Represents a rendering context by wrapping a view object and\n   * maintaining a reference to the parent context.\n   */\n  function Context (view, parentContext) {\n    this.view = view;\n    this.cache = { '.': this.view };\n    this.parent = parentContext;\n  }\n\n  /**\n   * Creates a new context using the given view with this context\n   * as the parent.\n   */\n  Context.prototype.push = function push (view) {\n    return new Context(view, this);\n  };\n\n  /**\n   * Returns the value of the given name in this context, traversing\n   * up the context hierarchy if the value is absent in this context's view.\n   */\n  Context.prototype.lookup = function lookup (name) {\n    var cache = this.cache;\n\n    var value;\n    if (cache.hasOwnProperty(name)) {\n      value = cache[name];\n    } else {\n      var context = this, names, index, lookupHit = false;\n\n      while (context) {\n        if (name.indexOf('.') > 0) {\n          value = context.view;\n          names = name.split('.');\n          index = 0;\n\n          /**\n           * Using the dot notion path in `name`, we descend through the\n           * nested objects.\n           *\n           * To be certain that the lookup has been successful, we have to\n           * check if the last object in the path actually has the property\n           * we are looking for. We store the result in `lookupHit`.\n           *\n           * This is specially necessary for when the value has been set to\n           * `undefined` and we want to avoid looking up parent contexts.\n           **/\n          while (value != null && index < names.length) {\n            if (index === names.length - 1)\n              lookupHit = hasProperty(value, names[index]);\n\n            value = value[names[index++]];\n          }\n        } else {\n          value = context.view[name];\n          lookupHit = hasProperty(context.view, name);\n        }\n\n        if (lookupHit)\n          break;\n\n        context = context.parent;\n      }\n\n      cache[name] = value;\n    }\n\n    if (isFunction(value))\n      value = value.call(this.view);\n\n    return value;\n  };\n\n  /**\n   * A Writer knows how to take a stream of tokens and render them to a\n   * string, given a context. It also maintains a cache of templates to\n   * avoid the need to parse the same template twice.\n   */\n  function Writer () {\n    this.cache = {};\n  }\n\n  /**\n   * Clears all cached templates in this writer.\n   */\n  Writer.prototype.clearCache = function clearCache () {\n    this.cache = {};\n  };\n\n  /**\n   * Parses and caches the given `template` and returns the array of tokens\n   * that is generated from the parse.\n   */\n  Writer.prototype.parse = function parse (template, tags) {\n    var cache = this.cache;\n    var tokens = cache[template];\n\n    if (tokens == null)\n      tokens = cache[template] = parseTemplate(template, tags);\n\n    return tokens;\n  };\n\n  /**\n   * High-level method that is used to render the given `template` with\n   * the given `view`.\n   *\n   * The optional `partials` argument may be an object that contains the\n   * names and templates of partials that are used in the template. It may\n   * also be a function that is used to load partial templates on the fly\n   * that takes a single argument: the name of the partial.\n   */\n  Writer.prototype.render = function render (template, view, partials) {\n    var tokens = this.parse(template);\n    var context = (view instanceof Context) ? view : new Context(view);\n    return this.renderTokens(tokens, context, partials, template);\n  };\n\n  /**\n   * Low-level method that renders the given array of `tokens` using\n   * the given `context` and `partials`.\n   *\n   * Note: The `originalTemplate` is only ever used to extract the portion\n   * of the original template that was contained in a higher-order section.\n   * If the template doesn't use higher-order sections, this argument may\n   * be omitted.\n   */\n  Writer.prototype.renderTokens = function renderTokens (tokens, context, partials, originalTemplate) {\n    var buffer = '';\n\n    var token, symbol, value;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      value = undefined;\n      token = tokens[i];\n      symbol = token[0];\n\n      if (symbol === '#') value = this.renderSection(token, context, partials, originalTemplate);\n      else if (symbol === '^') value = this.renderInverted(token, context, partials, originalTemplate);\n      else if (symbol === '>') value = this.renderPartial(token, context, partials, originalTemplate);\n      else if (symbol === '&') value = this.unescapedValue(token, context);\n      else if (symbol === 'name') value = this.escapedValue(token, context);\n      else if (symbol === 'text') value = this.rawValue(token);\n\n      if (value !== undefined)\n        buffer += value;\n    }\n\n    return buffer;\n  };\n\n  Writer.prototype.renderSection = function renderSection (token, context, partials, originalTemplate) {\n    var self = this;\n    var buffer = '';\n    var value = context.lookup(token[1]);\n\n    // This function is used to render an arbitrary template\n    // in the current context by higher-order sections.\n    function subRender (template) {\n      return self.render(template, context, partials);\n    }\n\n    if (!value) return;\n\n    if (isArray(value)) {\n      for (var j = 0, valueLength = value.length; j < valueLength; ++j) {\n        buffer += this.renderTokens(token[4], context.push(value[j]), partials, originalTemplate);\n      }\n    } else if (typeof value === 'object' || typeof value === 'string' || typeof value === 'number') {\n      buffer += this.renderTokens(token[4], context.push(value), partials, originalTemplate);\n    } else if (isFunction(value)) {\n      if (typeof originalTemplate !== 'string')\n        throw new Error('Cannot use higher-order sections without the original template');\n\n      // Extract the portion of the original template that the section contains.\n      value = value.call(context.view, originalTemplate.slice(token[3], token[5]), subRender);\n\n      if (value != null)\n        buffer += value;\n    } else {\n      buffer += this.renderTokens(token[4], context, partials, originalTemplate);\n    }\n    return buffer;\n  };\n\n  Writer.prototype.renderInverted = function renderInverted (token, context, partials, originalTemplate) {\n    var value = context.lookup(token[1]);\n\n    // Use JavaScript's definition of falsy. Include empty arrays.\n    // See https://github.com/janl/mustache.js/issues/186\n    if (!value || (isArray(value) && value.length === 0))\n      return this.renderTokens(token[4], context, partials, originalTemplate);\n  };\n\n  Writer.prototype.renderPartial = function renderPartial (token, context, partials) {\n    if (!partials) return;\n\n    var value = isFunction(partials) ? partials(token[1]) : partials[token[1]];\n    if (value != null)\n      return this.renderTokens(this.parse(value), context, partials, value);\n  };\n\n  Writer.prototype.unescapedValue = function unescapedValue (token, context) {\n    var value = context.lookup(token[1]);\n    if (value != null)\n      return value;\n  };\n\n  Writer.prototype.escapedValue = function escapedValue (token, context) {\n    var value = context.lookup(token[1]);\n    if (value != null)\n      return mustache.escape(value);\n  };\n\n  Writer.prototype.rawValue = function rawValue (token) {\n    return token[1];\n  };\n\n  mustache.name = 'mustache.js';\n  mustache.version = '2.2.1';\n  mustache.tags = [ '{{', '}}' ];\n\n  // All high-level mustache.* functions use this writer.\n  var defaultWriter = new Writer();\n\n  /**\n   * Clears all cached templates in the default writer.\n   */\n  mustache.clearCache = function clearCache () {\n    return defaultWriter.clearCache();\n  };\n\n  /**\n   * Parses and caches the given template in the default writer and returns the\n   * array of tokens it contains. Doing this ahead of time avoids the need to\n   * parse templates on the fly as they are rendered.\n   */\n  mustache.parse = function parse (template, tags) {\n    return defaultWriter.parse(template, tags);\n  };\n\n  /**\n   * Renders the `template` with the given `view` and `partials` using the\n   * default writer.\n   */\n  mustache.render = function render (template, view, partials) {\n    if (typeof template !== 'string') {\n      throw new TypeError('Invalid template! Template should be a \"string\" ' +\n                          'but \"' + typeStr(template) + '\" was given as the first ' +\n                          'argument for mustache#render(template, view, partials)');\n    }\n\n    return defaultWriter.render(template, view, partials);\n  };\n\n  // This is here for backwards compatibility with 0.4.x.,\n  /*eslint-disable */ // eslint wants camel cased function name\n  mustache.to_html = function to_html (template, view, partials, send) {\n    /*eslint-enable*/\n\n    var result = mustache.render(template, view, partials);\n\n    if (isFunction(send)) {\n      send(result);\n    } else {\n      return result;\n    }\n  };\n\n  // Export the escaping function so that the user may override it.\n  // See https://github.com/janl/mustache.js/issues/244\n  mustache.escape = escapeHtml;\n\n  // Export these mainly for testing, but also for advanced usage.\n  mustache.Scanner = Scanner;\n  mustache.Context = Context;\n  mustache.Writer = Writer;\n\n}));\n","/* object-iterators.js - Mini Underscore library\n * by Jonathan Eiten\n *\n * The methods below operate on objects (but not arrays) similarly\n * to Underscore (http://underscorejs.org/#collections).\n *\n * For more information:\n * https://github.com/joneit/object-iterators\n */\n\n'use strict';\n\n/**\n * @constructor\n * @summary Wrap an object for one method call.\n * @Desc Note that the `new` keyword is not necessary.\n * @param {object|null|undefined} object - `null` or `undefined` is treated as an empty plain object.\n * @return {Wrapper} The wrapped object.\n */\nfunction Wrapper(object) {\n    if (object instanceof Wrapper) {\n        return object;\n    }\n    if (!(this instanceof Wrapper)) {\n        return new Wrapper(object);\n    }\n    this.originalValue = object;\n    this.o = object || {};\n}\n\n/**\n * @name Wrapper.chain\n * @summary Wrap an object for a chain of method calls.\n * @Desc Calls the constructor `Wrapper()` and modifies the wrapper for chaining.\n * @param {object} object\n * @return {Wrapper} The wrapped object.\n */\nWrapper.chain = function (object) {\n    var wrapped = Wrapper(object); // eslint-disable-line new-cap\n    wrapped.chaining = true;\n    return wrapped;\n};\n\nWrapper.prototype = {\n    /**\n     * Unwrap an object wrapped with {@link Wrapper.chain|Wrapper.chain()}.\n     * @return {object|null|undefined} The value originally wrapped by the constructor.\n     * @memberOf Wrapper.prototype\n     */\n    value: function () {\n        return this.originalValue;\n    },\n\n    /**\n     * @desc Mimics Underscore's [each](http://underscorejs.org/#each) method: Iterate over the members of the wrapped object, calling `iteratee()` with each.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function is undefined; an `.each` loop cannot be broken out of (use {@link Wrapper#find|.find} instead).\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `iteratee`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {Wrapper} The wrapped object for chaining.\n     * @memberOf Wrapper.prototype\n     */\n    each: function (iteratee, context) {\n        var o = this.o;\n        Object.keys(o).forEach(function (key) {\n            iteratee.call(this, o[key], key, o);\n        }, context || o);\n        return this;\n    },\n\n    /**\n     * @desc Mimics Underscore's [find](http://underscorejs.org/#find) method: Look through each member of the wrapped object, returning the first one that passes a truth test (`predicate`), or `undefined` if no value passes the test. The function returns the value of the first acceptable member, and doesn't necessarily traverse the entire object.\n     * @param {function} predicate - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function should be truthy if the member passes the test and falsy otherwise.\n     * @param {object} [context] - If given, `predicate` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} The found property's value, or undefined if not found.\n     * @memberOf Wrapper.prototype\n     */\n    find: function (predicate, context) {\n        var o = this.o;\n        var result;\n        if (o) {\n            result = Object.keys(o).find(function (key) {\n                return predicate.call(this, o[key], key, o);\n            }, context || o);\n            if (result !== undefined) {\n                result = o[result];\n            }\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [filter](http://underscorejs.org/#filter) method: Look through each member of the wrapped object, returning the values of all members that pass a truth test (`predicate`), or empty array if no value passes the test. The function always traverses the entire object.\n     * @param {function} predicate - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function should be truthy if the member passes the test and falsy otherwise.\n     * @param {object} [context] - If given, `predicate` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} An array containing the filtered values.\n     * @memberOf Wrapper.prototype\n     */\n    filter: function (predicate, context) {\n        var o = this.o;\n        var result = [];\n        if (o) {\n            Object.keys(o).forEach(function (key) {\n                if (predicate.call(this, o[key], key, o)) {\n                    result.push(o[key]);\n                }\n            }, context || o);\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [map](http://underscorejs.org/#map) method: Produces a new array of values by mapping each value in list through a transformation function (`iteratee`). The function always traverses the entire object.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function is concatenated to the end of the new array.\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} An array containing the filtered values.\n     * @memberOf Wrapper.prototype\n     */\n    map: function (iteratee, context) {\n        var o = this.o;\n        var result = [];\n        if (o) {\n            Object.keys(o).forEach(function (key) {\n                result.push(iteratee.call(this, o[key], key, o));\n            }, context || o);\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [reduce](http://underscorejs.org/#reduce) method: Boil down the values of all the members of the wrapped object into a single value. `memo` is the initial state of the reduction, and each successive step of it should be returned by `iteratee()`.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with four arguments: `(memo, value, key, object)`. The return value of this function becomes the new value of `memo` for the next iteration.\n     * @param {*} [memo] - If no memo is passed to the initial invocation of reduce, the iteratee is not invoked on the first element of the list. The first element is instead passed as the memo in the invocation of the iteratee on the next element in the list.\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `iteratee`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} The value of `memo` \"reduced\" as per `iteratee`.\n     * @memberOf Wrapper.prototype\n     */\n    reduce: function (iteratee, memo, context) {\n        var o = this.o;\n        if (o) {\n            Object.keys(o).forEach(function (key, idx) {\n                memo = (!idx && memo === undefined) ? o[key] : iteratee(memo, o[key], key, o);\n            }, context || o);\n        }\n        return memo;\n    },\n\n    /**\n     * @desc Mimics Underscore's [extend](http://underscorejs.org/#extend) method: Copy all of the properties in each of the `source` object parameter(s) over to the (wrapped) destination object (thus mutating it). It's in-order, so the properties of the last `source` object will override properties with the same name in previous arguments or in the destination object.\n     * > This method copies own members as well as members inherited from prototype chain.\n     * @param {...object|null|undefined} source - Values of `null` or `undefined` are treated as empty plain objects.\n     * @return {Wrapper|object} The wrapped destination object if chaining is in effect; otherwise the unwrapped destination object.\n     * @memberOf Wrapper.prototype\n     */\n    extend: function (source) {\n        var o = this.o;\n        Array.prototype.slice.call(arguments).forEach(function (object) {\n            if (object) {\n                for (var key in object) {\n                    o[key] = object[key];\n                }\n            }\n        });\n        return this.chaining ? this : o;\n    },\n\n    /**\n     * @desc Mimics Underscore's [extendOwn](http://underscorejs.org/#extendOwn) method: Like {@link Wrapper#extend|extend}, but only copies its \"own\" properties over to the destination object.\n     * @param {...object|null|undefined} source - Values of `null` or `undefined` are treated as empty plain objects.\n     * @return {Wrapper|object} The wrapped destination object if chaining is in effect; otherwise the unwrapped destination object.\n     * @memberOf Wrapper.prototype\n     */\n    extendOwn: function (source) {\n        var o = this.o;\n        Array.prototype.slice.call(arguments).forEach(function (object) {\n            Wrapper(object).each(function (val, key) { // eslint-disable-line new-cap\n                o[key] = val;\n            });\n        });\n        return this.chaining ? this : o;\n    }\n};\n\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find\nif (!Array.prototype.find) {\n    Array.prototype.find = function (predicate) { // eslint-disable-line no-extend-native\n        if (this === null) {\n            throw new TypeError('Array.prototype.find called on null or undefined');\n        }\n        if (typeof predicate !== 'function') {\n            throw new TypeError('predicate must be a function');\n        }\n        var list = Object(this);\n        var length = list.length >>> 0;\n        var thisArg = arguments[1];\n        var value;\n\n        for (var i = 0; i < length; i++) {\n            value = list[i];\n            if (predicate.call(thisArg, value, i, list)) {\n                return value;\n            }\n        }\n        return undefined;\n    };\n}\n\nmodule.exports = Wrapper;\n","'use strict';\n\n/* eslint-env node, browser */\n\n/**\n * Creates a new read-only property and attaches it to the provided context.\n * @private\n * @param {string} name - Name for new property.\n * @param {*} [value] - Value of new property.\n */\nfunction addReadOnlyProperty(name, value) {\n    Object.defineProperty(this, name, {\n        value: value,\n        writable: false,\n        enumerable: true,\n        configurable: false\n    });\n}\n\n/**\n * @constructor Point\n *\n * @desc This object represents a single point in an abstract 2-dimensional matrix.\n *\n * The unit of measure is typically pixels.\n * (If used to model computer graphics, vertical coordinates are typically measured downwards\n * from the top of the window. This convention however is not inherent in this object.)\n *\n * Note: This object should be instantiated with the `new` keyword.\n *\n * @param {number} x - the new point's `x` property\n * @param {number} y - the new point's `y` property\n */\nfunction Point(x, y) {\n\n    /**\n     * @name x\n     * @type {number}\n     * @summary This point's horizontal coordinate.\n     * @desc Created upon instantiation by the {@link Point|constructor}.\n     * @memberOf Point.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'x', Number(x) || 0);\n\n    /**\n     * @name y\n     * @type {number}\n     * @summary This point's vertical coordinate.\n     * @desc Created upon instantiation by the {@link Point|constructor}.\n     * @memberOf Point.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'y', Number(y) || 0);\n\n}\n\nPoint.prototype = {\n\n    /**\n     * @returns {Point} A new point which is this point's position increased by coordinates of given `offset`.\n     * @param {Point} offset - Horizontal and vertical values to add to this point's coordinates.\n     * @memberOf Point.prototype\n     */\n    plus: function(offset) {\n        return new Point(\n            this.x + offset.x,\n            this.y + offset.y\n        );\n    },\n\n    /**\n     * @returns {Point} A new point which is this point's position increased by given offsets.\n     * @param {number} [offsetX=0] - Value to add to this point's horizontal coordinate.\n     * @param {number} [offsetY=0] - Value to add to this point's horizontal coordinate.\n     * @memberOf Point.prototype\n     */\n    plusXY: function(offsetX, offsetY) {\n        return new Point(\n            this.x + (offsetX || 0),\n            this.y + (offsetY || 0)\n        );\n    },\n\n    /**\n     * @returns {Point} A new point which is this point's position decreased by coordinates of given `offset`.\n     * @param {Point} offset - Horizontal and vertical values to subtract from this point's coordinates.\n     * @memberOf Point.prototype\n     */\n    minus: function(offset) {\n        return new Point(\n            this.x - offset.x,\n            this.y - offset.y\n        );\n    },\n\n    /**\n     * @returns {Point} A new `Point` positioned to least x and least y of this point and given `offset`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    min: function(point) {\n        return new Point(\n            Math.min(this.x, point.x),\n            Math.min(this.y, point.y)\n        );\n    },\n\n    /**\n     * @returns {Point} A new `Point` positioned to greatest x and greatest y of this point and given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    max: function(point) {\n        return new Point(\n            Math.max(this.x, point.x),\n            Math.max(this.y, point.y)\n        );\n    },\n\n    /**\n     * @returns {number} Distance between given `point` and this point using Pythagorean Theorem formula.\n     * @param {Point} point - A point from which to compute the distance to this point.\n     * @memberOf Point.prototype\n     */\n    distance: function(point) {\n        var deltaX = point.x - this.x,\n            deltaY = point.y - this.y;\n\n        return Math.sqrt(\n            deltaX * deltaX +\n            deltaY * deltaY\n        );\n    },\n\n    /**\n     * _(Formerly: `equal`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are exactly equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    equals: function(point) {\n        var result = false;\n\n        if (point) {\n            result =\n                this.x === point.x &&\n                this.y === point.y;\n        }\n\n        return result;\n    },\n\n    /**\n     * @returns {boolean} `true` iff _both_ coordinates of this point are greater than those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    greaterThan: function(point) {\n        return (\n            this.x > point.x &&\n            this.y > point.y\n        );\n    },\n\n    /**\n     * @returns {boolean} `true` iff _both_ coordinates of this point are less than those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    lessThan: function(point) {\n        return (\n            this.x < point.x &&\n            this.y < point.y\n        );\n    },\n\n    /**\n     * _(Formerly `greaterThanEqualTo`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are greater than or equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    greaterThanOrEqualTo: function(point) {\n        return (\n            this.x >= point.x &&\n            this.y >= point.y\n        );\n    },\n\n    /**\n     * _(Formerly `lessThanEqualTo`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are less than or equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    lessThanOrEqualTo: function(point) {\n        return (\n            this.x <= point.x &&\n            this.y <= point.y\n        );\n    },\n\n    /**\n     * _(Formerly `isContainedWithinRectangle`.)_\n     * @param rect {Rectangle} - Rectangle to test this point against.\n     * @returns {boolean} `true` iff this point is within given `rect`.\n     * @memberOf Point.prototype\n     */\n    within: function(rect) {\n        var minX = rect.origin.x,\n            maxX = minX + rect.extent.x;\n        var minY = rect.origin.y,\n            maxY = minY + rect.extent.y;\n\n        if (rect.extent.x < 0) {\n            minX = maxX;\n            maxX = rect.origin.x;\n        }\n\n        if (rect.extent.y < 0) {\n            minY = maxY;\n            maxY = rect.origin.y;\n        }\n\n        return (\n            minX <= this.x && this.x < maxX &&\n            minY <= this.y && this.y < maxY\n        );\n    }\n};\n\nPoint.prototype.EQ = Point.prototype.equals;\nPoint.prototype.GT = Point.prototype.greaterThan;\nPoint.prototype.LT = Point.prototype.lessThan;\nPoint.prototype.GE = Point.prototype.greaterThanOrEqualTo;\nPoint.prototype.LE = Point.prototype.lessThanOrEqualTo;\n\n\n/**\n * @constructor Rectangle\n *\n * @desc This object represents a rectangular area within an abstract 2-dimensional matrix.\n *\n * The unit of measure is typically pixels.\n * (If used to model computer graphics, vertical coordinates are typically measured downwards\n * from the top of the window. This convention however is not inherent in this object.)\n *\n * Normally, the `x` and `y` parameters to the constructor describe the upper left corner of the rect.\n * However, negative values of `width` and `height` will be added to the given `x` and `y`. That is,\n * a negative value of the `width` parameter will extend the rect to the left of the given `x` and\n * a negative value of the `height` parameter will extend the rect above the given `y`.\n * In any case, after instantiation the following are guaranteed to always be true:\n * * The `extent`, `width`, and `height` properties _always_ give positive values.\n * * The `origin`, `top`, and `left` properties _always_ reflect the upper left corner.\n * * The `corner`, `bottom`, and `right` properties _always_ reflect the lower right corner.\n *\n * Note: This object should be instantiated with the `new` keyword.\n *\n * @param {number} [x=0] - Horizontal coordinate of some corner of the rect.\n * @param {number} [y=0] - Vertical coordinate of some corner of the rect.\n * @param {number} [width=0] - Width of the new rect. May be negative (see above).\n * @param {number} [height=0] - Height of the new rect. May be negative (see above).\n */\nfunction Rectangle(x, y, width, height) {\n\n    x = Number(x) || 0;\n    y = Number(y) || 0;\n    width = Number(width) || 0;\n    height = Number(height) || 0;\n\n    if (width < 0) {\n        x += width;\n        width = -width;\n    }\n\n    if (height < 0) {\n        y += height;\n        height = -height;\n    }\n\n    /**\n     * @name origin\n     * @type {Point}\n     * @summary Upper left corner of this rect.\n     * @desc Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'origin', new Point(x, y));\n\n    /**\n     * @name extent\n     * @type {Point}\n     * @summary this rect's width and height.\n     * @desc Unlike the other `Point` properties, `extent` is not a global coordinate pair; rather it consists of a _width_ (`x`, always positive) and a _height_ (`y`, always positive).\n     *\n     * This object might be more legitimately typed as something like `Area` with properties `width` and `height`; however we wanted it to be able to use it efficiently with a point's `plus` and `minus` methods (that is, without those methods having to check and branch on the type of its parameter).\n     *\n     * Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @see The {@link Rectangle#corner|corner} method.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'extent', new Point(width, height));\n\n    /**\n     * @name corner\n     * @type {Point}\n     * @summary Lower right corner of this rect.\n     * @desc This is a calculated value created upon instantiation by the {@linkplain Rectangle|constructor}. It is `origin` offset by `extent`.\n     *\n     * **Note:** These coordinates actually point to the pixel one below and one to the right of the rect's actual lower right pixel.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'corner', new Point(x + width, y + height));\n\n    /**\n     * @name center\n     * @type {Point}\n     * @summary Center of this rect.\n     * @desc Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'center', new Point(x + (width / 2), y + (height / 2)));\n\n}\n\nRectangle.prototype = {\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Minimum vertical coordinate of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get top() {\n        return this.origin.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Minimum horizontal coordinate of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get left() {\n        return this.origin.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Maximum vertical coordinate of this rect + 1.\n     * @memberOf Rectangle.prototype\n     */\n    get bottom() {\n        return this.corner.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Maximum horizontal coordinate of this rect + 1.\n     * @memberOf Rectangle.prototype\n     */\n    get right() {\n        return this.corner.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Width of this rect (always positive).\n     * @memberOf Rectangle.prototype\n     */\n    get width() {\n        return this.extent.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Height of this rect (always positive).\n     * @memberOf Rectangle.prototype\n     */\n    get height() {\n        return this.extent.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Area of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get area() {\n        return this.width * this.height;\n    },\n\n    /**\n     * @returns {Rectangle} A copy of this rect but with horizontal position reset to given `x` and no width.\n     * @param {number} x - Horizontal coordinate of the new rect.\n     * @memberOf Rectangle.prototype\n     */\n    flattenXAt: function(x) {\n        return new Rectangle(x, this.origin.y, 0, this.extent.y);\n    },\n\n    /**\n     * @returns {Rectangle} A copy of this rect but with vertical position reset to given `y` and no height.\n     * @param {number} y - Vertical coordinate of the new rect.\n     * @memberOf Rectangle.prototype\n     */\n    flattenYAt: function(y) {\n        return new Rectangle(this.origin.x, y, this.extent.x, 0);\n    },\n\n    /**\n     * @returns {boolean} `true` iff given `point` entirely contained within this rect.\n     * @param {Point} pointOrRect - The point or rect to test for containment.\n     * @memberOf Rectangle.prototype\n     */\n    contains: function(pointOrRect) {\n        return pointOrRect.within(this);\n    },\n\n    /**\n     * _(Formerly `isContainedWithinRectangle`.)_\n     * @returns {boolean} `true` iff `this` rect is entirely contained within given `rect`.\n     * @param {Rectangle} rect - Rectangle to test against this rect.\n     * @memberOf Rectangle.prototype\n     */\n    within: function(rect) {\n        return (\n            rect.origin.lessThanOrEqualTo(this.origin) &&\n            rect.corner.greaterThanOrEqualTo(this.corner)\n        );\n    },\n\n    /**\n     * _(Formerly: `insetBy`.)_\n     * @returns {Rectangle} That is enlarged/shrunk by given `padding`.\n     * @param {number} padding - Amount by which to increase (+) or decrease (-) this rect\n     * @see The {@link Rectangle#shrinkBy|shrinkBy} method.\n     * @memberOf Rectangle.prototype\n     */\n    growBy: function(padding) {\n        return new Rectangle(\n            this.origin.x + padding,\n            this.origin.y + padding,\n            this.extent.x - padding - padding,\n            this.extent.y - padding - padding);\n    },\n\n    /**\n     * @returns {Rectangle} That is enlarged/shrunk by given `padding`.\n     * @param {number} padding - Amount by which to decrease (+) or increase (-) this rect.\n     * @see The {@link Rectangle#growBy|growBy} method.\n     * @memberOf Rectangle.prototype\n     */\n    shrinkBy: function(padding) {\n        return this.growBy(-padding);\n    },\n\n    /**\n     * @returns {Rectangle} Bounding rect that contains both this rect and the given `rect`.\n     * @param {Rectangle} rect - The rectangle to union with this rect.\n     * @memberOf Rectangle.prototype\n     */\n    union: function(rect) {\n        var origin = this.origin.min(rect.origin),\n            corner = this.corner.max(rect.corner),\n            extent = corner.minus(origin);\n\n        return new Rectangle(\n            origin.x, origin.y,\n            extent.x, extent.y\n        );\n    },\n\n    /**\n     * iterate over all points within this rect, invoking `iteratee` for each.\n     * @param {function(number,number)} iteratee - Function to call for each point.\n     * Bound to `context` when given; otherwise it is bound to this rect.\n     * Each invocation of `iteratee` is called with two arguments:\n     * the horizontal and vertical coordinates of the point.\n     * @param {object} [context=this] - Context to bind to `iteratee` (when not `this`).\n     * @memberOf Rectangle.prototype\n     */\n    forEach: function(iteratee, context) {\n        context = context || this;\n        for (var x = this.origin.x, x2 = this.corner.x; x < x2; x++) {\n            for (var y = this.origin.y, y2 = this.corner.y; y < y2; y++) {\n                iteratee.call(context, x, y);\n            }\n        }\n    },\n\n    /**\n     * @returns {Rectangle} One of:\n     * * _If this rect intersects with the given `rect`:_\n     *      a new rect representing that intersection.\n     * * _If it doesn't intersect and `ifNoneAction` defined:_\n     *      result of calling `ifNoneAction`.\n     * * _If it doesn't intersect and `ifNoneAction` undefined:_\n     *      `null`.\n     * @param {Rectangle} rect - The rectangle to intersect with this rect.\n     * @param {function(Rectangle)} [ifNoneAction] - When no intersection, invoke and return result.\n     * Bound to `context` when given; otherwise bound to this rect.\n     * Invoked with `rect` as sole parameter.\n     * @param {object} [context=this] - Context to bind to `ifNoneAction` (when not `this`).\n     * @memberOf Rectangle.prototype\n     */\n    intersect: function(rect, ifNoneAction, context) {\n        var result = null,\n            origin = this.origin.max(rect.origin),\n            corner = this.corner.min(rect.corner),\n            extent = corner.minus(origin);\n\n        if (extent.x > 0 && extent.y > 0) {\n            result = new Rectangle(\n                origin.x, origin.y,\n                extent.x, extent.y\n            );\n        } else if (typeof ifNoneAction === 'function') {\n            result = ifNoneAction.call(context || this, rect);\n        }\n\n        return result;\n    },\n\n    /**\n     * @returns {boolean} `true` iff this rect overlaps with given `rect`.\n     * @param {Rectangle} rect - The rectangle to intersect with this rect.\n     * @memberOf Rectangle.prototype\n     */\n    intersects: function(rect) {\n        return (\n            rect.corner.x > this.origin.x &&\n            rect.corner.y > this.origin.y &&\n            rect.origin.x < this.corner.x &&\n            rect.origin.y < this.corner.y\n        );\n    }\n};\n\n// Interface\nexports.Point = Point;\nexports.Rectangle = Rectangle;\n","'use strict';\n\nvar // a regex search pattern that matches all the reserved chars of a regex search pattern\n    reserved = /([\\.\\\\\\+\\*\\?\\^\\$\\(\\)\\{\\}\\=\\!\\<\\>\\|\\:\\[\\]])/g,\n\n    // regex wildcard search patterns\n    REGEXP_WILDCARD = '.*',\n    REGEXP_WILDCHAR = '.',\n    REGEXP_WILDCARD_MATCHER = '(' + REGEXP_WILDCARD + ')',\n\n    // LIKE search patterns\n    LIKE_WILDCHAR = '_',\n    LIKE_WILDCARD = '%',\n\n    // regex search patterns that match LIKE search patterns\n    REGEXP_LIKE_PATTERN_MATCHER = new RegExp('(' + [\n        LIKE_WILDCHAR,\n        LIKE_WILDCARD,\n        '\\\\[\\\\^?[^-\\\\]]+]', // matches a LIKE set (same syntax as a RegExp set)\n        '\\\\[\\\\^?[^-\\\\]]\\\\-[^\\\\]]]' // matches a LIKE range (same syntax as a RegExp range)\n    ].join('|') + ')', 'g');\n\nfunction reserve(s) {\n    return s.replace(reserved, '\\\\$1');\n}\n\nfunction regExpLIKE(pattern) {\n    var i, parts;\n\n    // Find all LIKE patterns\n    parts = pattern.match(REGEXP_LIKE_PATTERN_MATCHER);\n\n    if (parts) {\n        // Translate found LIKE patterns to regex patterns, escaped intervening non-patterns, and interleave the two\n\n        for (i = 0; i < parts.length; ++i) {\n            // Escape left brackets (unpaired right brackets are OK)\n            if (parts[i][0] === '[') {\n                parts[i] = reserve(parts[i]);\n            }\n\n            // Make each found pattern matchable by enclosing in parentheses\n            parts[i] = '(' + parts[i] + ')';\n        }\n\n        // Match these precise patterns again with their intervening non-patterns (i.e., text)\n        parts = pattern.match(new RegExp(\n            REGEXP_WILDCARD_MATCHER +\n            parts.join(REGEXP_WILDCARD_MATCHER)  +\n            REGEXP_WILDCARD_MATCHER\n        ));\n\n        // Discard first match of non-global search (which is the whole string)\n        parts.shift();\n\n        // For each re-found pattern part, translate % and _ to regex equivalent\n        for (i = 1; i < parts.length; i += 2) {\n            var part = parts[i];\n            switch (part) {\n                case LIKE_WILDCARD: part = REGEXP_WILDCARD; break;\n                case LIKE_WILDCHAR: part = REGEXP_WILDCHAR; break;\n                default:\n                    var j = part[1] === '^' ? 2 : 1;\n                    part = '[' + reserve(part.substr(j, part.length - (j + 1))) + ']';\n            }\n            parts[i] = part;\n        }\n    } else {\n        parts = [pattern];\n    }\n\n    // For each surrounding text part, escape reserved regex chars\n    for (i = 0; i < parts.length; i += 2) {\n        parts[i] = reserve(parts[i]);\n    }\n\n    // Join all the interleaved parts\n    parts = parts.join('');\n\n    // Optimize or anchor the pattern at each end as needed\n    if (parts.substr(0, 2) === REGEXP_WILDCARD) { parts = parts.substr(2); } else { parts = '^' + parts; }\n    if (parts.substr(-2, 2) === REGEXP_WILDCARD) { parts = parts.substr(0, parts.length - 2); } else { parts += '$'; }\n\n    // Return the new regex\n    return new RegExp(parts);\n}\n\nvar cache, size;\n\n/**\n * @summary Delete a pattern from the cache; or clear the whole cache.\n * @param {string} [pattern] - The LIKE pattern to remove from the cache. Fails silently if not found in the cache. If pattern omitted, clears whole cache.\n */\n(regExpLIKE.clearCache = function (pattern) {\n    if (!pattern) {\n        cache = {};\n        size = 0;\n    } else if (cache[pattern]) {\n        delete cache[pattern];\n        size--;\n    }\n    return size;\n})(); // init the cache\n\nregExpLIKE.getCacheSize = function () { return size; };\n\n/**\n * @summary Cached version of `regExpLIKE()`.\n * @desc Cached entries are subject to garbage collection if `keep` is `undefined` or `false` on insertion or `false` on most recent reference. Garbage collection will occur iff `regExpLIKE.cacheMax` is defined and it equals the number of cached patterns. The garbage collector sorts the patterns based on most recent reference; the oldest 10% of the entries are deleted. Alternatively, you can manage the cache yourself to a limited extent (see {@link regeExpLIKE.clearCache|clearCache}).\n * @param pattern - the LIKE pattern (to be) converted to a RegExp\n * @param [keep] - If given, changes the keep status for this pattern as follows:\n * * `true` permanently caches the pattern (not subject to garbage collection) until `false` is given on a subsequent call\n * * `false` allows garbage collection on the cached pattern\n * * `undefined` no change to keep status\n * @returns {RegExp}\n */\nregExpLIKE.cached = function (pattern, keep) {\n    var item = cache[pattern];\n    if (item) {\n        item.when = new Date().getTime();\n        if (keep !== undefined) {\n            item.keep = keep;\n        }\n    } else {\n        if (size === regExpLIKE.cacheMax) {\n            var age = [], ages = 0, key, i;\n            for (key in cache) {\n                item = cache[key];\n                if (!item.keep) {\n                    for (i = 0; i < ages; ++i) {\n                        if (item.when < age[i].item.when) {\n                            break;\n                        }\n                    }\n                    age.splice(i, 0, { key: key, item: item });\n                    ages++;\n                }\n            }\n            if (!age.length) {\n                return regExpLIKE(pattern); // cache is full!\n            }\n            i = Math.ceil(age.length / 10); // will always be at least 1\n            size -= i;\n            while (i--) {\n                delete cache[age[i].key];\n            }\n        }\n        item = cache[pattern] = {\n            regex: regExpLIKE(pattern),\n            keep: keep,\n            when: new Date().getTime()\n        };\n        size++;\n    }\n    return item.regex;\n};\n\nmodule.exports = regExpLIKE;\n","'use strict';\n\n/* eslint-env node, browser */\n\n(function (module) {  // eslint-disable-line no-unused-expressions\n\n    // This closure supports NodeJS-less client side includes with <script> tags. See https://github.com/joneit/mnm.\n\n    /**\n     * @constructor RangeSelectionModel\n     *\n     * @desc This object models selection of \"cells\" within an abstract single-dimensional matrix.\n     *\n     * Disjoint selections can be built with calls to the following methods:\n     * * {@link RangeSelectionModel#select|select(start, stop)} - Add a range to the matrix.\n     * * {@link RangeSelectionModel#deselect|deselect(start, stop)} - Remove a range from the matrix.\n     *\n     * Two more methods are available:\n     * * Test a cell to see if it {@link RangeSelectionModel#isSelected|isSelected(cell)}\n     * * {@link RangeSelectionModel#clear|clear()} the matrix\n     *\n     * Internally, the selection is run-length-encoded. It is therefore a \"sparse\" matrix\n     * with undefined bounds. A single data property called `selection` is an array that\n     * contains all the \"runs\" (ranges) of selected cells albeit in no particular order.\n     * This property should not normally need to be accessed directly.\n     *\n     * Note: This object should be instantiated with the `new` keyword.\n     *\n     * @returns {RangeSelectionModel} Self (i.e., `this` object).\n     */\n    function RangeSelectionModel() {\n        /**\n         * @name selection\n         * @type {Array.Array.number}\n         * @summary Unordered list of runs.\n         * @desc A \"run\" is defined as an Array(2) where:\n         * * element [0] is the beginning of the run\n         * * element [1] is the end of the run (inclusive) and is always >= element [0]\n         * The order of the runs within is undefined.\n         * @memberOf RangeSelectionModel.prototype\n         * @abstract\n         */\n        this.selection = [];\n\n        //we need to be able to go back in time\n        //the states field\n        this.states = [];\n\n        //clone and store my current state\n        //so we can unwind changes if need be\n        this.storeState = function () {\n            var sels = this.selection;\n            var state = [];\n            var copy;\n            for (var i = 0; i < sels.length; i++) {\n                copy = [].concat(sels[i]);\n                state.push(copy);\n            }\n            this.states.push(state);\n        };\n    }\n\n    RangeSelectionModel.prototype = {\n\n        /**\n         * @summary Add a contiguous run of points to the selection.\n         * @desc Insert a new run into `this.selection`.\n         * The new run will be merged with overlapping and adjacent runs.\n         *\n         * The two parameters may be given in either order.\n         * The start and stop elements in the resulting run will however always be ordered.\n         * (However, note that the order of the runs within `this.selection` is itself always unordered.)\n         *\n         * Note that `this.selection` is updated in place, preserving validity of any external references.\n         * @param {number} start - Start of run. May be greater than `stop`.\n         * @param {number} [stop=stop] - End of run (inclusive). May be less than `start`.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        select: function (start, stop) {\n            this.storeState();\n            var run = makeRun(start, stop);\n            var splicer = [0, 1];\n            this.selection.forEach(function (each) {\n                if (overlaps(each, run) || abuts(each, run)) {\n                    run = merge(each, run);\n                } else {\n                    splicer.push(each);\n                }\n            });\n            splicer.push(run);\n            splicer[1] = this.selection.length;\n            this.selection.splice.apply(this.selection, splicer); // update in place to preserve external references\n            return this;\n        },\n\n        /**\n         * @summary Remove a contiguous run of points from the selection.\n         * @desc Truncate and/or remove run(s) from `this.selection`.\n         * Removing part of existing runs will (correctly) shorten them or break them into two fragments.\n         *\n         * The two parameters may be given in either order.\n         *\n         * Note that `this.selection` is updated in place, preserving validity of any external references.\n         * @param {number} start - Start of run. May be greater than `stop`.\n         * @param {number} [stop=stop] - End of run (inclusive). May be less than `start`.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        deselect: function (start, stop) {\n            var run = makeRun(start, stop);\n            var splicer = [0, 0];\n            this.selection.forEach(function (each) {\n                if (overlaps(each, run)) {\n                    var pieces = subtract(each, run);\n                    splicer = splicer.concat(pieces);\n                } else {\n                    splicer.push(each);\n                }\n            });\n            splicer[1] = this.selection.length;\n            this.selection.splice.apply(this.selection, splicer); // update in place to preserve external references\n            return this;\n        },\n\n        /**\n         * @summary Empties `this.selection`, effectively removing all runs.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        clear: function () {\n            this.states.length = 0;\n            this.selection.length = 0;\n            return this;\n        },\n\n        clearMostRecentSelection: function () {\n            if (this.states.length === 0) {\n                return;\n            }\n            this.selection = this.states.pop();\n        },\n\n        /**\n         * @summary Determines if the given `cell` is selected.\n         * @returns {boolean} `true` iff given `cell` is within any of the runs in `this.selection`.\n         * @param {number} cell - The cell to test for inclusion in the selection.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        isSelected: function (cell) {\n            return this.selection.some(function (each) {\n                return each[0] <= cell && cell <= each[1];\n            });\n        },\n\n        isEmpty: function (){\n            return this.selection.length === 0;\n        },\n\n        /**\n         * @summary Return the indexes that are selected.\n         * @desc Return the indexes that are selected.\n         * @returns {Array.Array.number}\n         * @memberOf RangeSelectionModel.prototype\n         */\n        getSelections: function (){\n            var result = [];\n            this.selection.forEach(function (each) {\n                for (var i = each[0]; i <= each[1]; i++) {\n                    result.push(i);\n                }\n            });\n            result.sort(function (a, b){\n                return a - b;\n            });\n            return result;\n        }\n\n    };\n\n    /**\n     * @private\n     * @summary Preps `start` and `stop` params into order array\n     * @function makeRun\n     * @desc Utility function called by both `select()` and `deselect()`.\n     * @param {number|number[]} start - Start of run. if array, `start` and `stop` are taken from first two elements.\n     * @param {number} [stop=start] - End of run (inclusive).\n     */\n    function makeRun(start, stop) {\n        return (\n            start instanceof Array\n                ? makeRun.apply(this, start) // extract params from given array\n                : stop === undefined\n                ? [ start, start ] // single param is a run that stops where it starts\n                : start <= stop\n                ? [ start, stop ]\n                : [ stop, start ] // reverse descending params into ascending order\n        );\n    }\n\n    /**\n     * @private\n     * @function overlaps\n     * @returns {boolean} `true` iff `run1` overlaps `run2`\n     * @summary Comparison operator that determines if given runs overlap with one another.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * Overlap is defined to include the case where one run completely contains the other.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - first run\n     * @param {number[]} run2 - second run\n     */\n    function overlaps(run1, run2) {\n        return (\n            run1[0] <= run2[0] && run2[0] <= run1[1] || // run2's start is within run1 OR...\n            run1[0] <= run2[1] && run2[1] <= run1[1] || // run2's stop is within run1 OR...\n            run2[0] <  run1[0] && run1[1] <  run2[1]    // run2 completely contains run1\n        );\n    }\n\n    /**\n     * @private\n     * @function abuts\n     * @summary Comparison operator that determines if given runs are consecutive with one another.\n     * @returns {boolean} `true` iff `run1` is consecutive with `run2`\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - first run\n     * @param {number[]} run2 - second run\n     */\n    function abuts(run1, run2) {\n        return (\n            run1[1] === run2[0] - 1 || // run1's top immediately precedes run2's start OR...\n            run2[1] === run1[0] - 1    // run2's top immediately precedes run1's start\n        );\n    }\n\n    /**\n     * @private\n     * @function subtract\n     * @summary Operator that subtracts one run from another.\n     * @returns {Array.Array.number} The remaining pieces of `minuend` after removing `subtrahend`.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * This function _does not assumes_ that `overlap()` has already been called with the same runs and has returned `true`.\n     *\n     * Returned array contains 0, 1, or 2 runs which are the portion(s) of `minuend` that do _not_ include `subtrahend`.\n     *\n     * Caveat: This operator is *not* commutative.\n     * @param {number[]} minuend - a run from which to \"subtract\" `subtrahend`\n     * @param {number[]} subtrahend - a run to \"subtracted\" from `minuend`\n     */\n    function subtract(minuend, subtrahend) {\n        var m0 = minuend[0];\n        var m1 = minuend[1];\n        var s0 = subtrahend[0];\n        var s1 = subtrahend[1];\n        var result = [];\n\n        if (s0 <= m0 && s1 < m1) {\n            //subtrahend extends before minuend: return remaining piece of `minuend`\n            result.push([s1 + 1, m1]);\n        } else if (s0 > m0 && s1 >= m1) {\n            //subtrahend extends after minuend: return remaining piece of `minuend`\n            result.push([m0, s0 - 1]);\n        } else if (m0 < s0 && s1 < m1) {\n            //completely inside: return 2 smaller pieces resulting from the hole\n            result.push([m0, s0 - 1]);\n            result.push([s1 + 1, m1]);\n        } else if (s1 < m0 || s0 > m1) {\n            // completely outside: return `minuend` untouched\n            result.push(minuend);\n        }\n\n        //else subtrahend must completely overlap minuend so return no pieces\n\n        return result;\n    }\n\n\n    // Local utility functions\n\n    /**\n     * @private\n     * @function merge\n     * @summary Operator that merges given runs.\n     * @returns {number[]} A single merged run.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * The runs are assumed to be overlapping or adjacent to one another.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - a run to merge with `run2`\n     * @param {number[]} run2 - a run to merge with `run1`\n     */\n    function merge(run1, run2) {\n        var min = Math.min(Math.min.apply(Math, run1), Math.min.apply(Math, run2));\n        var max = Math.max(Math.max.apply(Math, run1), Math.max.apply(Math, run2));\n        return [min, max];\n    }\n\n    // Interface\n    module.exports = RangeSelectionModel;\n})(\n    typeof module === 'object' && module || (window.RangeSelectionModel = {}),\n    typeof module === 'object' && module.exports || (window.RangeSelectionModel.exports = {})\n) || (\n    typeof module === 'object' || (window.RangeSelectionModel = window.RangeSelectionModel.exports)\n);\n\n/* About the above IIFE:\n * This file is a \"modified node module.\" It functions as usual in Node.js *and* is also usable directly in the browser.\n * 1. Node.js: The IIFE is superfluous but innocuous.\n * 2. In the browser: The IIFE closure serves to keep internal declarations private.\n * 2.a. In the browser as a global: The logic in the actual parameter expressions + the post-invocation expression\n * will put your API in `window.RangeSelectionModel`.\n * 2.b. In the browser as a module: If you predefine a `window.module` object, the results will be in `module.exports`.\n * The bower component `mnm` makes this easy and also provides a global `require()` function for referencing your module\n * from other closures. In either case, this works with both NodeJs-style export mechanisms -- a single API assignment,\n * `module.exports = yourAPI` *or* a series of individual property assignments, `module.exports.property = property`.\n *\n * Before the IIFE runs, the actual parameter expressions are executed:\n * 1. If `window` object undefined, we're in NodeJs so assume there is a `module` object with an `exports` property\n * 2. If `window` object defined, we're in browser\n * 2.a. If `module` object predefined, use it\n * 2.b. If `module` object undefined, create a `RangeSelectionModel` object\n *\n * After the IIFE returns:\n * Because it always returns undefined, the expression after the || will execute:\n * 1. If `window` object undefined, then we're in NodeJs so we're done\n * 2. If `window` object defined, then we're in browser\n * 2.a. If `module` object predefined, we're done; results are in `moudule.exports`\n * 2.b. If `module` object undefined, redefine`RangeSelectionModel` to be the `RangeSelectionModel.exports` object\n */\n","// templex node module\n// https://github.com/joneit/templex\n\n/* eslint-env node */\n\n/**\n * Merges values of execution context properties named in template by {prop1},\n * {prop2}, etc., or any javascript expression incorporating such prop names.\n * The context always includes the global object. In addition you can specify a single\n * context or an array of contexts to search (in the order given) before finally\n * searching the global context.\n *\n * Merge expressions consisting of simple numeric terms, such as {0}, {1}, etc., deref\n * the first context given, which is assumed to be an array. As a convenience feature,\n * if additional args are given after `template`, `arguments` is unshifted onto the context\n * array, thus making first additional arg available as {1}, second as {2}, etc., as in\n * `templex('Hello, {1}!', 'World')`. ({0} is the template so consider this to be 1-based.)\n *\n * If you prefer something other than braces, redefine `templex.regexp`.\n *\n * See tests for examples.\n *\n * @param {string} template\n * @param {...string} [args]\n */\nfunction templex(template) {\n    var contexts = this instanceof Array ? this : [this];\n    if (arguments.length > 1) { contexts.unshift(arguments); }\n    return template.replace(templex.regexp, templex.merger.bind(contexts));\n}\n\ntemplex.regexp = /\\{(.*?)\\}/g;\n\ntemplex.with = function (i, s) {\n    return 'with(this[' + i + ']){' + s + '}';\n};\n\ntemplex.cache = [];\n\ntemplex.deref = function (key) {\n    if (!(this.length in templex.cache)) {\n        var code = 'return eval(expr)';\n\n        for (var i = 0; i < this.length; ++i) {\n            code = templex.with(i, code);\n        }\n\n        templex.cache[this.length] = eval('(function(expr){' + code + '})'); // eslint-disable-line no-eval\n    }\n    return templex.cache[this.length].call(this, key);\n};\n\ntemplex.merger = function (match, key) {\n    // Advanced features: Context can be a list of contexts which are searched in order.\n    var replacement;\n\n    try {\n        replacement = isNaN(key) ? templex.deref.call(this, key) : this[0][key];\n    } catch (e) {\n        replacement = '{' + key + '}';\n    }\n\n    return replacement;\n};\n\n// this interface consists solely of the templex function (and it's properties)\nmodule.exports = templex;\n","'use strict';\n\nvar Base = require('extend-me').Base;\n\n/** @constructor\n * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n *\n * See {@link CellProvider#initialize|initialize} which is called by the constructor.\n */\nvar CellProvider = Base.extend('CellProvider', {\n\n    /**\n     * @summary Constructor logic\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     * @memberOf CellProvider.prototype\n     */\n    initialize: function() {\n        this.cellCache = {};\n        this.initializeCells();\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getColumnHeaderCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getRowHeaderCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    paintButton: function(gc, config) {\n        var val = config.value;\n        var c = config.x;\n        var r = config.y;\n        var bounds = config.bounds;\n        var x = bounds.x + 2;\n        var y = bounds.y + 2;\n        var width = bounds.width - 3;\n        var height = bounds.height - 3;\n        var radius = height / 2;\n        var arcGradient = gc.createLinearGradient(x, y, x, y + height);\n        if (config.mouseDown) {\n            arcGradient.addColorStop(0, '#B5CBED');\n            arcGradient.addColorStop(1, '#4d74ea');\n        } else {\n            arcGradient.addColorStop(0, '#ffffff');\n            arcGradient.addColorStop(1, '#aaaaaa');\n        }\n        gc.fillStyle = arcGradient;\n        gc.strokeStyle = '#000000';\n        roundRect(gc, x, y, width, height, radius, arcGradient, true);\n\n        var ox = (width - config.getTextWidth(gc, val)) / 2;\n        var oy = (height - config.getTextHeight(gc.font).descent) / 2;\n\n        if (gc.textBaseline !== 'middle') {\n            gc.textBaseline = 'middle';\n        }\n\n        gc.fillStyle = '#000000';\n\n        config.backgroundColor = 'rgba(0,0,0,0)';\n        gc.fillText(val, x + ox, y + oy);\n\n        //identify that we are a button\n        config.buttonCells[c + ',' + r] = true;\n    },\n\n    /**\n     * @summary The default cell rendering function for rendering a vanilla cell.\n     * @desc Great care has been taken in crafting this function as it needs to perform extremely fast. Reads on the gc object are expensive but not quite as expensive as writes to it. We do our best to avoid writes, then avoid reads. Clipping bounds are not set here as this is also an expensive operation. Instead, we truncate overflowing text and content by filling a rectangle with background color column by column instead of cell by cell.  This column by column fill happens higher up on the stack in a calling function from fin-hypergrid-renderer.  Take note we do not do cell by cell border renderering as that is expensive.  Instead we render many fewer gridlines after all cells are rendered.\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} config.bounds.x - the x screen coordinate of my origin\n     * @param {number} config.bounds.y - the y screen coordinate of my origin\n     * @param {number} config.bounds.width - the width I'm allowed to draw within\n     * @param {number} config.bounds.height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    defaultCellPaint: function(gc, config) {\n        var val = config.value,\n            x = config.bounds.x,\n            y = config.bounds.y,\n            width = config.bounds.width,\n            height = config.bounds.height,\n            wrapHeaders = config.headerTextWrapping,\n            leftPadding = 2, //TODO: fix this\n            isHeader = config.y === 0;\n\n        var leftIcon, rightIcon, centerIcon, ixoffset, iyoffset;\n\n        // setting gc properties are expensive, let's not do it needlessly\n\n        if (val && val.constructor === Array) {\n            leftIcon = val[0];\n            rightIcon = val[2];\n            val = val[1];\n            if (val && typeof val === 'object') {\n                if (val.constructor.name === 'HTMLImageElement') { // must be an image\n                    centerIcon = val;\n                    val = null;\n                }\n            }\n            if (leftIcon && leftIcon.nodeName !== 'IMG') {\n                leftIcon = null;\n            }\n            if (rightIcon && rightIcon.nodeName !== 'IMG') {\n                rightIcon = null;\n            }\n            if (centerIcon && centerIcon.nodeName !== 'IMG') {\n                centerIcon = null;\n            }\n        }\n\n        val = valueOrFunctionExecute(config, val);\n\n        val = config.formatter(val);\n\n        if (gc.font !== config.font) {\n            gc.font = config.font;\n        }\n        if (gc.textAlign !== 'left') {\n            gc.textAlign = 'left';\n        }\n        if (gc.textBaseline !== 'middle') {\n            gc.textBaseline = 'middle';\n        }\n\n        // fill background only if our bgColor is populated or we are a selected cell\n        var backgroundColor;\n        if (config.backgroundColor) {\n            gc.fillStyle = backgroundColor = valueOrFunctionExecute(config, config.isSelected ? config.backgroundSelectionColor : config.backgroundColor);\n            if (config.isColumnHovered) {\n                gc.fillStyle = config.hoverColumnColor;\n            }\n            if (config.isRowHovered) {\n                gc.fillStyle = config.hoverRowColor;\n            }\n            if (config.isCellHovered) {\n                gc.fillStyle = config.hoverCellColor;\n            }\n            gc.fillRect(x, y, width, height);\n        }\n\n        // draw text\n        var theColor = valueOrFunctionExecute(config, config.isSelected ? config.foregroundSelectionColor : config.color);\n        if (gc.fillStyle !== theColor) {\n            gc.fillStyle = theColor;\n            gc.strokeStyle = theColor;\n        }\n\n        if (isHeader && wrapHeaders) {\n            this.renderMultiLineText(gc, x, y, height, width, config, val);\n        } else {\n            this.renderSingleLineText(gc, x, y, height, width, config, val);\n        }\n\n        var iconWidth = 0;\n        if (leftIcon) {\n            iyoffset = Math.round((height - leftIcon.height) / 2);\n            gc.drawImage(leftIcon, x + leftPadding, y + iyoffset);\n            iconWidth = Math.max(leftIcon.width + 2);\n        }\n        if (rightIcon && width > 1.75 * height) {\n            iyoffset = Math.round((height - rightIcon.height) / 2);\n            var rightX = x + width - rightIcon.width;\n            if (backgroundColor) {\n                gc.fillStyle = backgroundColor;\n                gc.fillRect(rightX, y, rightIcon.width, height);\n            } else {\n                gc.clearRect(rightX, y, rightIcon.width, height);\n            }\n            gc.drawImage(rightIcon, rightX, y + iyoffset);\n            iconWidth = Math.max(rightIcon.width + 2);\n        }\n        if (centerIcon) {\n            iyoffset = Math.round((height - centerIcon.height) / 2);\n            ixoffset = Math.round((width - centerIcon.width) / 2);\n            gc.drawImage(centerIcon, x + width - ixoffset - centerIcon.width, y + iyoffset);\n            iconWidth = Math.max(centerIcon.width + 2);\n        }\n        if (config.cellBorderThickness) {\n            gc.beginPath();\n            gc.rect(x, y, width, height);\n            gc.lineWidth = config.cellBorderThickness;\n            gc.strokeStyle = config.cellBorderStyle;\n\n            // animate the dashed line a bit here for fun\n\n            gc.stroke();\n            gc.closePath();\n        }\n        config.minWidth = config.minWidth + 2 * (iconWidth);\n    },\n\n    renderMultiLineText: function(gc, x, y, height, width, config, val) {\n        var lines = fitText(gc, config, val, width);\n        if (lines.length === 1) {\n            return this.renderSingleLineText(gc, x, y, height, width, config, squeeze(val));\n        }\n\n        var colHEdgeOffset = config.cellPadding,\n            halignOffset = 0,\n            valignOffset = config.voffset,\n            halign = config.halign,\n            textHeight = config.getTextHeight(config.font).height;\n\n        switch (halign) {\n            case 'right':\n                halignOffset = width - colHEdgeOffset;\n                break;\n            case 'center':\n                halignOffset = width / 2;\n                break;\n            case 'left':\n                halignOffset = colHEdgeOffset;\n                break;\n        }\n\n        var hMin = 0, vMin = Math.ceil(textHeight / 2);\n\n        valignOffset += Math.ceil((height - (lines.length - 1) * textHeight) / 2);\n\n        halignOffset = Math.max(hMin, halignOffset);\n        valignOffset = Math.max(vMin, valignOffset);\n\n        gc.save(); // define a clipping region for cell\n        gc.rect(x, y, width, height);\n        gc.clip();\n\n        gc.textAlign = halign;\n\n        for (var i = 0; i < lines.length; i++) {\n            gc.fillText(lines[i], x + halignOffset, y + valignOffset + (i * textHeight));\n        }\n\n        gc.restore(); // discard clipping region\n    },\n\n    renderSingleLineText: function(gc, x, y, height, width, config, val) {\n        var colHEdgeOffset = config.cellPadding,\n            halignOffset = 0,\n            valignOffset = config.voffset,\n            halign = config.halign,\n            isCellHovered = config.isCellHovered,\n            isLink = config.link;\n\n        var fontMetrics = config.getTextHeight(config.font);\n        var textWidth = config.getTextWidth(gc, val);\n\n        //we must set this in order to compute the minimum width\n        //for column autosizing purposes\n        config.minWidth = textWidth + (2 * colHEdgeOffset);\n\n        switch (halign) {\n            case 'right':\n                //textWidth = config.getTextWidth(gc, config.value);\n                halignOffset = width - colHEdgeOffset - textWidth;\n                break;\n            case 'center':\n                //textWidth = config.getTextWidth(gc, config.value);\n                halignOffset = (width - textWidth) / 2;\n                break;\n            case 'left':\n                halignOffset = colHEdgeOffset;\n                break;\n        }\n\n        halignOffset = Math.max(0, halignOffset);\n        valignOffset = valignOffset + Math.ceil(height / 2);\n\n        if (val !== null) {\n            gc.fillText(val, x + halignOffset, y + valignOffset);\n        }\n\n        if (isCellHovered) {\n            gc.beginPath();\n            if (isLink) {\n                underline(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1);\n                gc.stroke();\n            }\n            gc.closePath();\n        }\n        if (config.strikeThrough === true) {\n            gc.beginPath();\n            strikeThrough(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1);\n            gc.stroke();\n            gc.closePath();\n        }\n    },\n\n    /**\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     * @desc Emerson's paint function for a slider button. currently the user cannot interact with it\n     */\n    paintSlider: function(gc, x, y, width, height) {\n        // gc.strokeStyle = 'white';\n        // var val = this.config.value;\n        // var radius = height / 2;\n        // var offset = width * val;\n        // var bgColor = this.config.isSelected ? this.config.bgSelColor : '#333333';\n        // var btnGradient = gc.createLinearGradient(x, y, x, y + height);\n        // btnGradient.addColorStop(0, bgColor);\n        // btnGradient.addColorStop(1, '#666666');\n        // var arcGradient = gc.createLinearGradient(x, y, x, y + height);\n        // arcGradient.addColorStop(0, '#aaaaaa');\n        // arcGradient.addColorStop(1, '#777777');\n        // gc.fillStyle = btnGradient;\n        // roundRect(gc, x, y, width, height, radius, btnGradient);\n        // if (val < 1.0) {\n        //     gc.fillStyle = arcGradient;\n        // } else {\n        //     gc.fillStyle = '#eeeeee';\n        // }\n        // gc.beginPath();\n        // gc.arc(x + Math.max(offset - radius, radius), y + radius, radius, 0, 2 * Math.PI);\n        // gc.fill();\n        // gc.closePath();\n        // this.config.minWidth = 100;\n    },\n\n    /**\n     * @desc A simple implementation of a sparkline, because it's a barchart we've changed the name ;).\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    paintSparkbar: function(gc, x, y, width, height) {\n        gc.beginPath();\n        var val = this.config.value;\n        if (!val || !val.length) {\n            return;\n        }\n        var count = val.length;\n        var eWidth = width / count;\n        var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n        gc.fillStyle = fgColor;\n        for (var i = 0; i < val.length; i++) {\n            var barheight = val[i] / 110 * height;\n            gc.fillRect(x + 5, y + height - barheight, eWidth * 0.6666, barheight);\n            x = x + eWidth;\n        }\n        gc.closePath();\n        this.config.minWidth = count * 10;\n\n    },\n\n    /**\n     * @desc A simple implementation of a sparkline.  see [Edward Tufte sparkline](http://www.edwardtufte.com/bboard/q-and-a-fetch-msg?msg_id=0001OR)\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    paintSparkline: function(gc, x, y, width, height) {\n        gc.beginPath();\n        var val = this.config.value;\n        if (!val || !val.length) {\n            return;\n        }\n        var count = val.length;\n        var eWidth = width / count;\n\n        var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n        gc.strokeStyle = fgColor;\n        gc.fillStyle = fgColor;\n        gc.beginPath();\n        var prev;\n        for (var i = 0; i < val.length; i++) {\n            var barheight = val[i] / 110 * height;\n            if (!prev) {\n                prev = barheight;\n            }\n            gc.lineTo(x + 5, y + height - barheight);\n            gc.arc(x + 5, y + height - barheight, 1, 0, 2 * Math.PI, false);\n            x = x + eWidth;\n        }\n        this.config.minWidth = count * 10;\n        gc.stroke();\n        gc.closePath();\n    },\n\n    /**\n     * @desc A simple implementation of a tree cell renderer for use mainly with the qtree.\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    treeCellRenderer: function(gc, x, y, width, height) {\n        var val = this.config.value.data;\n        var indent = this.config.value.indent;\n        var icon = this.config.value.icon;\n\n        //fill background only if our bgColor is populated or we are a selected cell\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n\n        if (!val || !val.length) {\n            return;\n        }\n        var valignOffset = Math.ceil(height / 2);\n\n        gc.fillStyle = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        gc.fillText(icon + val, x + indent, y + valignOffset);\n\n        var textWidth = this.config.getTextWidth(gc, icon + val);\n        var minWidth = x + indent + textWidth + 10;\n        this.config.minWidth = minWidth;\n    },\n\n    /**\n     * @desc An empty implementation of a cell renderer, see [the null object pattern](http://c2.com/cgi/wiki?NullObject).\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    emptyCellRenderer: function(gc, x, y, width, height) {},\n\n    /**\n     * @memberOf CellProvider.prototype\n     * @private\n     */\n    initializeCells: function() {\n        var self = this;\n        this.cellCache.simpleCellRenderer = {\n            paint: this.defaultCellPaint,\n            renderSingleLineText: this.renderSingleLineText,\n            renderMultiLineText: this.renderMultiLineText\n        };\n        this.cellCache.sliderCellRenderer = {\n            paint: this.paintSlider\n        };\n        this.cellCache.sparkbarCellRenderer = {\n            paint: this.paintSparkbar\n        };\n        this.cellCache.sparklineCellRenderer = {\n            paint: this.paintSparkline\n        };\n        this.cellCache.treeCellRenderer = {\n            paint: this.treeCellRenderer\n        };\n        this.cellCache.emptyCellRenderer = {\n            paint: this.emptyCellRenderer\n        };\n        this.cellCache.buttonRenderer = {\n            paint: this.paintButton,\n            //defaultCellPaint: this.defaultCellPaint\n        };\n        this.cellCache.linkCellRenderer = {\n            paint: function(gc, x, y, width, height) {\n                self.config = this.config;\n                self.defaultCellPaint(gc, x, y, width, height, true);\n            }\n        };\n    }\n});\n\nfunction valueOrFunctionExecute(config, valueOrFunction) {\n    var isFunction = (((typeof valueOrFunction)[0]) === 'f');\n    var result = isFunction ? valueOrFunction(config) : valueOrFunction;\n    if (!result && result !== 0) {\n        return '';\n    }\n    return result;\n}\n\nfunction underline(config, gc, text, x, y, thickness) {\n    var width = config.getTextWidth(gc, text);\n\n    switch (gc.textAlign) {\n        case 'center':\n            x -= (width / 2);\n            break;\n        case 'right':\n            x -= width;\n            break;\n    }\n\n    //gc.beginPath();\n    gc.lineWidth = thickness;\n    gc.moveTo(x + 0.5, y + 0.5);\n    gc.lineTo(x + width + 0.5, y + 0.5);\n}\n\nfunction strikeThrough(config, gc, text, x, y, thickness) {\n    var fontMetrics = config.getTextHeight(config.font);\n    var width = config.getTextWidth(gc, text);\n    y = y - (fontMetrics.height * 0.4);\n\n    switch (gc.textAlign) {\n        case 'center':\n            x -= (width / 2);\n            break;\n        case 'right':\n            x -= width;\n            break;\n    }\n\n    //gc.beginPath();\n    gc.lineWidth = thickness;\n    gc.moveTo(x + 0.5, y + 0.5);\n    gc.lineTo(x + width + 0.5, y + 0.5);\n}\n\nfunction findLines(gc, config, words, width) {\n\n    if (words.length === 1) {\n        return words;\n    }\n\n    // starting with just the first word…\n    var stillFits, line = [words.shift()];\n    while (\n        // so lone as line still fits within current column…\n    (stillFits = config.getTextWidth(gc, line.join(' ')) < width)\n        // …AND there are more words available…\n    && words.length\n        ) {\n        // …add another word to end of line and retest\n        line.push(words.shift());\n    }\n\n    if (\n        !stillFits // if line is now too long…\n        && line.length > 1 // …AND is multiple words…\n    ) {\n        words.unshift(line.pop()); // …back off by (i.e., remove) one word\n    }\n\n    line = [line.join(' ')];\n\n    if (words.length) { // if there's anything left…\n        line = line.concat(findLines(gc, config, words, width)); // …break it up as well\n    }\n\n    return line;\n}\n\nfunction fitText(gc, config, string, width) {\n    return findLines(gc, config, squeeze(string).split(' '), width);\n}\n\n// trim string; then reduce all runs of multiple spaces to a single space\nfunction squeeze(string) {\n    return string.toString().trim().replace(/\\s\\s+/g, ' ');\n}\n\nfunction roundRect(gc, x, y, width, height, radius, fill, stroke) {\n\n    if (!stroke) {\n        stroke = true;\n    }\n    if (!radius) {\n        radius = 5;\n    }\n    gc.beginPath();\n    gc.moveTo(x + radius, y);\n    gc.lineTo(x + width - radius, y);\n    gc.quadraticCurveTo(x + width, y, x + width, y + radius);\n    gc.lineTo(x + width, y + height - radius);\n    gc.quadraticCurveTo(x + width, y + height, x + width - radius, y + height);\n    gc.lineTo(x + radius, y + height);\n    gc.quadraticCurveTo(x, y + height, x, y + height - radius);\n    gc.lineTo(x, y + radius);\n    gc.quadraticCurveTo(x, y, x + radius, y);\n    gc.closePath();\n    if (stroke) {\n        gc.stroke();\n    }\n    if (fill) {\n        gc.fill();\n    }\n    gc.closePath();\n}\n\nmodule.exports = CellProvider;\n","'use strict';\n\nvar leadingZeroIfNecessary = function(number) {\n    return number < 10 ? '0' + number : number + '';\n};\n\nmodule.exports = {\n    date: function(value) {\n        var dateString = value.getFullYear() + '-' + leadingZeroIfNecessary(value.getMonth() + 1) + '-' + leadingZeroIfNecessary(value.getDay());\n        return dateString;\n    },\n    default: function(value) {\n        return value + '';\n    }\n};\n","/* eslint-env browser */\n\n'use strict';\n\nvar extend = require('extend-me');\nextend.debug = true;\n\nvar FinBar = require('finbars');\nvar Canvas = require('fincanvas');\nvar Point = require('rectangular').Point;\nvar Rectangle = require('rectangular').Rectangle;\nvar _ = require('object-iterators');\n\nvar defaults = require('./defaults');\nvar Renderer = require('./Renderer');\nvar SelectionModel = require('./SelectionModel');\nvar addStylesheet = require('./stylesheets');\nvar TableDialog = require('./TableDialog');\nvar Formatters = require('./Formatters');\n\nvar themeInitialized = false,\n    polymerTheme = Object.create(defaults),\n    globalProperties = Object.create(polymerTheme),\n    customFilters = {};\n\n/**\n * @constructor\n * @param {string|Element} div - CSS selector or Element\n * @param {string} behaviorName - name of a behavior constructor from ./behaviors\n * @param {object} [margin] - optional canvas margins\n * @param {string} [margin.top]\n * @param {string} [margin.right='-200px']\n * @param {string} [margin.bottom]\n * @param {string} [margin.left]\n */\nfunction Hypergrid(div, behaviorFactory, margin) {\n    var self = this;\n\n    this.div = (typeof div === 'string') ? document.querySelector(div) : div;\n\n    addStylesheet('grid');\n\n    this.lastEdgeSelection = [0, 0];\n\n    this.lnfProperties = Object.create(globalProperties);\n\n    this.isWebkit = navigator.userAgent.toLowerCase().indexOf('webkit') > -1;\n    this.selectionModel = new SelectionModel();\n    this.localCellEditors = {};\n    this.selectionModel.getGrid = function() {\n        return self;\n    };\n    this.cellEditors = Object.create(this.localCellEditors);\n    this.renderOverridesCache = {};\n    this.behavior = behaviorFactory(this);\n\n    //prevent the default context menu for appearing\n    this.div.oncontextmenu = function(event) {\n        event.preventDefault();\n        return false;\n    };\n\n    this.clearMouseDown();\n    this.dragExtent = new Point(0, 0);\n    this.numRows = 0;\n    this.numColumns = 0;\n\n    //install any plugins\n    this.pluginsDo(function(each) {\n        if (each.installOn) {\n            each.installOn(self);\n        }\n    });\n\n    margin = margin || {};\n    margin.top = margin.top || 0;\n    margin.right = margin.right || '-200px';\n    margin.bottom = margin.bottom || 0;\n    margin.left = margin.left || 0;\n\n    //initialize our various pieces\n    if (!themeInitialized) {\n        themeInitialized = true;\n        buildPolymerTheme();\n    }\n    this.initRenderer();\n    this.initCanvas(margin);\n    this.initScrollbars();\n    this.initLocalCellEditors();\n\n    //Register a listener for the copy event so we can copy our selected region to the pastebuffer if conditions are right.\n    document.body.addEventListener('copy', function(evt) {\n        self.checkClipboardCopy(evt);\n    });\n    this.getCanvas().resize();\n\n    this.dialog = new TableDialog(this);\n    //this.computeCellsBounds();\n}\n\nHypergrid.prototype = {\n    constructor: Hypergrid.prototype.constructor,\n\n    /**\n     *\n     * A null object behavior serves as a place holder.\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n    behavior: null,\n\n    /**\n     * Cached resulan}\n     * @memberOf Hypergrid.prototype\n     */\n    isWebkit: true,\n\n    /**\n     * The pixel location of an initial mousedown click, either for editing a cell or for dragging a selection.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n    mouseDown: [],\n\n    /**\n     * The extent from the mousedown point during a drag operation.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n\n    dragExtent: null,\n\n    /**\n     * A float value between 0.0 - 1.0 of the vertical scroll position.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    vScrollValue: 0,\n\n    /**\n     * A float value between 0.0 - 1.0 of the horizontal scroll position.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    hScrollValue: 0,\n\n    /**\n     * @property {window.fin.rectangular} rectangular - Namespace for Point and Rectangle \"classes\" (constructors).\n     * @memberOf Hypergrid.prototype\n     */\n    rectangular: null,\n\n    /**\n     * @property {fin-hypergrid-selection-model} selectionModel - A [fin-hypergrid-selection-model](module-._selection-model.html) instance.\n     * @memberOf Hypergrid.prototype\n     */\n    selectionModel: null,\n\n    /**\n     * @property {fin-hypergrid-cell-editor} cellEditor - The current instance of [fin-hypergrid-cell-editor](module-cell-editors_base.html).\n     * @memberOf Hypergrid.prototype\n     */\n    cellEditor: null,\n\n    /**\n     * @property {fin-vampire-bar} sbHScroller - An instance of [fin-vampire-bar](http://datamadic.github.io/fin-vampire-bar/components/fin-vampire-bar/).\n     * @memberOf Hypergrid.prototype\n     */\n    sbHScroller: null,\n\n    /**\n     * @property {fin-vampire-bar} sbVScroller - An instance of [fin-vampire-bar](http://datamadic.github.io/fin-vampire-bar/components/fin-vampire-bar/).\n     * @memberOf Hypergrid.prototype\n     */\n    sbVScroller: null,\n\n    /**\n     * The previous value of sbVScrollVal.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    sbPrevVScrollValue: null,\n\n    /**\n     * The previous value of sbHScrollValue.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    sbPrevHScrollValue: null,\n\n    /**\n     * The cache of singleton cellEditors.\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n    cellEditors: null,\n\n    /**\n     * is the short term memory of what column I might be dragging around\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n\n    renderOverridesCache: {},\n\n    /**\n     * The pixel location of the current hovered cell.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n    hoverCell: null,\n\n    scrollingNow: false,\n\n    lastEdgeSelection: null,\n\n    /**\n     * @memberOf Hypergrid.prototype\n    clear out the LRU cache of text widths\n     */\n    setAttribute: function(attribute, value) {\n        this.div.setAttribute(attribute, value);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n    clear out all state and data of the grid\n     */\n    reset: function() {\n        var self = this;\n        this.lastEdgeSelection = [0, 0];\n        this.lnfProperties = Object.create(globalProperties);\n        this.selectionModel = new SelectionModel();\n        this.selectionModel.getGrid = function() {\n            return self;\n        };\n        this.cellEditors = Object.create(this.localCellEditors);\n        this.renderOverridesCache = {};\n        this.clearMouseDown();\n        this.dragExtent = new Point(0, 0);\n\n        this.numRows = 0;\n        this.numColumns = 0;\n\n        this.vScrollValue = 0;\n        this.hScrollValue = 0;\n\n        this.cellEditor = null;\n\n        this.sbPrevVScrollValue = null;\n        this.sbPrevHScrollValue = null;\n\n        this.hoverCell = null;\n        this.scrollingNow = false;\n        this.lastEdgeSelection = [0, 0];\n\n        this.getBehavior().reset();\n        this.getRenderer().reset();\n        this.getCanvas().resize();\n        this.behaviorChanged();\n    },\n\n    //resetTextWidthCache: function() {\n    //    textWidthCache = new LRUCache(2000);\n    //},\n\n    getProperties: function() {\n        return this.getPrivateState();\n    },\n\n    _getProperties: function() {\n        return this.lnfProperties;\n    },\n\n    computeCellsBounds: function() {\n        var renderer = this.getRenderer();\n        if (!renderer) {\n            return;\n        }\n        renderer.computeCellsBounds();\n    },\n\n    initCellEditor: function(cellEditor) {\n        this.localCellEditors[cellEditor.alias] = cellEditor;\n        cellEditor.grid = this;\n    },\n\n    initLocalCellEditors: function() {\n\n        var cellEditors = [\n            'Textfield',\n            'Choice',\n            //'Combo',\n            'Color',\n            'Date',\n            'Slider',\n            'Spinner',\n            'Filter'\n        ];\n\n        var self = this;\n        cellEditors.forEach(function(name) {\n            self.initCellEditor(new Hypergrid.cellEditors[name]);\n        });\n\n        this.localCellEditors.int = this.localCellEditors.spinner;\n        this.localCellEditors.float = this.localCellEditors.spinner;\n        this.localCellEditors.date = this.localCellEditors.date;\n        this.localCellEditors.string = this.localCellEditors.extfield;\n    },\n\n    toggleColumnPicker: function() {\n        this.getBehavior().toggleColumnPicker();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The pointer is over the given cell.\n     * @param {number} x - The x cell coordinate.\n     * @param {number} y - The y cell coordinate.\n     */\n    isHovered: function(x, y) {\n        var p = this.getHoverCell();\n        if (!p) {\n            return false;\n        }\n        return p.x === x && p.y === y;\n    },\n\n    registerFormatter: function(name, func) {\n        Formatters[name] = func;\n    },\n\n    getFormatter: function(type) {\n        var formatter = Formatters[type];\n        if (formatter) {\n            return formatter;\n        }\n        return Formatters.default;\n    },\n\n    formatValue: function(type, value) {\n        var formatter = this.getFormatter(type);\n        return formatter(value);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns boolean} The pointer is hovering over the given column.\n     * @param {number} x - The horizontal cell coordinate.\n     */\n    isColumnHovered: function(x) {\n        var p = this.getHoverCell();\n        if (!p) {\n            return false;\n        }\n        return p.x === x;\n    },\n\n    isRowResizeable: function() {\n        return this.resolveProperty('rowResize');\n    },\n\n    isCheckboxOnlyRowSelections: function() {\n        return this.resolveProperty('checkboxOnlyRowSelections');\n    },\n\n    /**\n     *\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The pointer is hovering over the row `y`.\n     * @param {number} y - The vertical cell coordinate.\n     */\n    isRowHovered: function(y) {\n        var p = this.getHoverCell();\n        if (!p) {\n            return false;\n        }\n        return p.y === y;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Point} The cell over which the cursor is hovering.\n     */\n    getHoverCell: function() {\n        return this.hoverCell;\n    },\n\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the cell under the cursor.\n     * @param {Point} point\n     */\n    setHoverCell: function(point) {\n        var me = this.hoverCell;\n        var newPoint = new Point(point.x, point.y);\n        if (me && me.equals(newPoint)) {\n            return;\n        }\n        this.hoverCell = newPoint;\n        this.fireSyntheticOnCellEnterEvent(newPoint);\n        this.repaint();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for all hypergrids in this process.\n     * @param {object} properties - A simple properties hash.\n     */\n    addGlobalProperties: function(properties) {\n        //we check for existence to avoid race condition in initialization\n        if (!globalProperties) {\n            var self = this;\n            setTimeout(function() {\n                self.addGlobalProperties(properties);\n            }, 10);\n        } else {\n            this._addGlobalProperties(properties);\n        }\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for all hypergrids in this process.\n     * @param {object} properties - A simple properties hash.\n     * @private\n     */\n    _addGlobalProperties: function(properties) {\n        _(properties).each(function(property, key) {\n            globalProperties[key] = property;\n        });\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for this hypergrid only.\n     * @param {object} properties - A simple properties hash.\n     */\n    addProperties: function(moreProperties) {\n        var properties = this.getProperties();\n        _(moreProperties).each(function(property, key) {\n            properties[key] = moreProperties[key];\n        });\n        this.refreshProperties();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Utility function to push out properties if we change them.\n     * @param {object} properties - An object of various key value pairs.\n     */\n\n    refreshProperties: function() {\n        // this.canvas = this.shadowRoot.querySelector('fin-canvas');\n        //this.canvas = new Canvas(this.divCanvas, this.renderer); //TODO: Do we really need to be recreating it here?\n        this.checkScrollbarVisibility();\n        this.getBehavior().defaultRowHeight = null;\n        if (this.isColumnAutosizing()) {\n            this.getBehavior().autosizeAllColumns();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} The state object for remembering our state.\n     * @see [Memento pattern](http://en.wikipedia.org/wiki/Memento_pattern)\n     */\n    getPrivateState: function() {\n        return this.getBehavior().getPrivateState();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the state object to return to the given user configuration.\n     * @param {object} state - A memento object.\n     * @see [Memento pattern](http://en.wikipedia.org/wiki/Memento_pattern)\n     */\n    setState: function(state) {\n        var self = this;\n        this.getBehavior().setState(state);\n        setTimeout(function() {\n            self.behaviorChanged();\n            self.synchronizeScrollingBoundries();\n        }, 100);\n    },\n\n    getState: function() {\n        return this.getBehavior().getState();\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} The initial mouse position on a mouse down event for cell editing or a drag operation.\n     * @memberOf Hypergrid.prototype\n     */\n    getMouseDown: function() {\n        var last = this.mouseDown.length - 1;\n        if (last < 0) {\n            return null;\n        }\n        return this.mouseDown[last];\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Remove the last item from the mouse down stack.\n     */\n    popMouseDown: function() {\n        if (this.mouseDown.length === 0) {\n            return;\n        }\n        this.mouseDown.length = this.mouseDown.length - 1;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Empty out the mouse down stack.\n     */\n    clearMouseDown: function() {\n        this.mouseDown = [new Point(-1, -1)];\n        this.dragExtent = null;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     set the mouse point that initated a cell edit or drag operation\n     *\n     * @param {Point} point\n     */\n    setMouseDown: function(point) {\n        this.mouseDown.push(point);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Point} The extent point of the current drag selection rectangle.\n     */\n    getDragExtent: function() {\n        return this.dragExtent;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Sets the extent point of the current drag selection operation.\n     * @param {Point} point\n     */\n    setDragExtent: function(point) {\n        this.dragExtent = point;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Iterate over the plugins invoking the given function with each.\n     * @todo We need a new plugin mechanism!\n     * @param {function} func - The function to invoke on all the plugins.\n     */\n    pluginsDo: function(func) {\n        //TODO: We need a new plugin mechanism!\n        //var userPlugins = this.children.array();\n        //var pluginsTag = this.shadowRoot.querySelector('fin-plugins');\n        //\n        //var plugins = userPlugins;\n        //if (pluginsTag) {\n        //    var systemPlugins = pluginsTag.children.array();\n        //    plugins = systemPlugins.concat(plugins);\n        //}\n        //\n        //plugins.forEach(function(plugin) {\n        //    func(plugin);\n        //});\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The CellProvider is accessed through Hypergrid because Hypergrid is the mediator and should have ultimate control on where it comes from. The default is to delegate through the behavior object.\n     * @returns {fin-hypergrid-cell-provider}\n     */\n    getCellProvider: function() {\n        var provider = this.getBehavior().getCellProvider();\n        return provider;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc This function is a callback from the HypergridRenderer sub-component. It is called after each paint of the canvas.\n     */\n    gridRenderedNotification: function() {\n        this.updateRenderedSizes();\n        if (this.cellEditor) {\n            this.cellEditor.gridRenderedNotification();\n        }\n        this.checkColumnAutosizing();\n        this.fireSyntheticGridRenderedEvent();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The grid has just been rendered, make sure the column widths are optimal.\n     */\n    checkColumnAutosizing: function() {\n        var behavior = this.getBehavior();\n        behavior.autoSizeRowNumberColumn();\n        if (this.isColumnAutosizing()) {\n            behavior.checkColumnAutosizing(false);\n        }\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Notify the GridBehavior how many rows and columns we just rendered.\n     */\n    updateRenderedSizes: function() {\n        var behavior = this.getBehavior();\n        //add one to each of these values as we want also to include\n        //the columns and rows that are partially visible\n        behavior.setRenderedColumnCount(this.getVisibleColumns() + 1);\n        behavior.setRenderedRowCount(this.getVisibleRows() + 1);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Conditionally copy to clipboard.\n     * @desc If we have focus, copy our current selection data to the system clipboard.\n     * @param {event} event - The copy system event.\n     */\n    checkClipboardCopy: function(event) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        event.preventDefault();\n        var csvData = this.getSelectionAsTSV();\n        event.clipboardData.setData('text/plain', csvData);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have any selections.\n     */\n    hasSelections: function() {\n        if (!this.getSelectionModel) {\n            return; // were not fully initialized yet\n        }\n        return this.getSelectionModel().hasSelections();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {string} Tab separated value string from the selection and our data.\n     */\n    getSelectionAsTSV: function() {\n        var sm = this.getSelectionModel();\n        if (sm.hasSelections()) {\n            var selections = this.getSelectionMatrix();\n            selections = selections[selections.length - 1];\n            return this.getMatrixSelectionAsTSV(selections);\n        } else if (sm.hasRowSelections()) {\n            return this.getMatrixSelectionAsTSV(this.getRowSelectionMatrix());\n        } else if (sm.hasColumnSelections()) {\n            return this.getMatrixSelectionAsTSV(this.getColumnSelectionMatrix());\n        }\n    },\n\n    getMatrixSelectionAsTSV: function(selections) {\n        //only use the data from the last selection\n        if (selections.length) {\n            var width = selections.length,\n                height = selections[0].length,\n                area = width * height,\n                collector = [];\n\n            //disallow if selection is too big\n            if (area > 20000) {\n                alert('selection size is too big to copy to the paste buffer'); // eslint-disable-line no-alert\n                return '';\n            }\n\n            for (var h = 0; h < height; h++) {\n                for (var w = 0; w < width; w++) {\n                    collector.push(selections[w][h]);\n                    if (w < width) {\n                        collector.push('\\t');\n                    }\n                }\n                if (h < height) {\n                    collector.push('\\n');\n                }\n            }\n\n            var result = collector.join('');\n\n            return result;\n        }\n        return '';\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have focus.\n     */\n    hasFocus: function() {\n        return this.getCanvas().hasFocus();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear all the selections.\n     */\n    clearSelections: function() {\n        var dontClearRows = this.isCheckboxOnlyRowSelections();\n        this.getSelectionModel().clear(dontClearRows);\n        this.clearMouseDown();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent selection.\n     */\n    clearMostRecentSelection: function() {\n        var dontClearRows = this.isCheckboxOnlyRowSelections();\n        this.getSelectionModel().clearMostRecentSelection(dontClearRows);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent column selection.\n     */\n    clearMostRecentColumnSelection: function() {\n        this.getSelectionModel().clearMostRecentColumnSelection();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent row selection.\n     */\n    clearMostRecentRowSelection: function() {\n        this.getSelectionModel().clearMostRecentRowSelection();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Select given region.\n     * @param {number} ox - origin x\n     * @param {number} oy - origin y\n     * @param {number} ex - extent x\n     * @param {number} ex - extent y\n     */\n    select: function(ox, oy, ex, ey) {\n        if (ox < 0 || oy < 0) {\n            //we don't select negative area\n            //also this means there is no origin mouse down for a selection rect\n            return;\n        }\n        this.getSelectionModel().select(ox, oy, ex, ey);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} Given point is selected.\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     */\n    isSelected: function(x, y) {\n        return this.getSelectionModel().isSelected(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given column is selected anywhere in the entire table.\n     * @param {number} col - The column index.\n     */\n    isCellSelectedInRow: function(col) {\n        var selectionModel = this.getSelectionModel();\n        var isSelected = selectionModel.isCellSelectedInRow(col);\n        return isSelected;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given row is selected anywhere in the entire table.\n     * @param {number} row - The row index.\n     */\n    isCellSelectedInColumn: function(row) {\n        var selectionModel = this.getSelectionModel();\n        var isSelected = selectionModel.isCellSelectedInColumn(row);\n        return isSelected;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {fin-hypergrid-selection-model} The selection model.\n     */\n    getSelectionModel: function() {\n        return this.selectionModel;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Behavior} The behavior (model).\n     */\n    getBehavior: function() {\n        return this.behavior;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Set the Behavior (model) object for this grid control.\n     * @desc This can be done dynamically.\n     * @param {Behavior} The behavior (model).\n     */\n    setBehavior: function(newBehavior) {\n\n        this.behavior = newBehavior;\n        this.behavior.setGrid(this);\n\n        this.behavior.changed = this.behaviorChanged.bind(this);\n        this.behavior.shapeChanged = this.behaviorShapeChanged.bind(this);\n        this.behavior.stateChanged = this.behaviorStateChanged.bind(this);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc I've been notified that the behavior has changed.\n     */\n    behaviorChanged: function() {\n        if (this.numColumns !== this.getColumnCount() || this.numRows !== this.getRowCount()) {\n            this.numColumns = this.getColumnCount();\n            this.numRows = this.getRowCount();\n            this.behaviorShapeChanged();\n        }\n        this.computeCellsBounds();\n        this.repaint();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Rectangle} My bounds.\n     */\n    getBounds: function() {\n        var renderer = this.getRenderer();\n        if (!renderer) {\n            return;\n        }\n        return renderer.getBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {string} The value of a lnf property.\n     * @param {string} key - A look-and-feel key.\n     */\n    resolveProperty: function(key) {\n        return this.getProperties()[key];\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The dimensions of the grid data have changed. You've been notified.\n     */\n    behaviorShapeChanged: function() {\n        this.synchronizeScrollingBoundries();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The dimensions of the grid data have changed. You've been notified.\n     */\n    behaviorStateChanged: function() {\n        this.getRenderer().computeCellsBounds();\n        this.repaint();\n    },\n\n    repaint: function() {\n        var now = this.resolveProperty('repaintImmediately');\n        var canvas = this.getCanvas();\n        if (canvas) {\n            if (now === true) {\n                canvas.paintNow();\n            } else {\n                canvas.repaint();\n            }\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Paint immediately in this microtask.\n     */\n    paintNow: function() {\n        var canvas = this.getCanvas();\n        canvas.paintNow();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} In HiDPI mode (has an attribute as such).\n     */\n    useHiDPI: function() {\n        return this.resolveProperty('useHiDPI') !== false;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Initialize drawing surface.\n     * @private\n     */\n    initCanvas: function(margin) {\n\n        var self = this;\n\n        var divCanvas = this.divCanvas = document.createElement('div');\n        this.div.appendChild(divCanvas);\n        this.canvas = new Canvas(divCanvas, this.renderer);\n\n        var style = divCanvas.style;\n        style.position = 'absolute';\n        style.top = margin.top;\n        style.right = margin.right;\n        style.bottom = margin.bottom;\n        style.left = margin.left;\n\n        this.canvas.resizeNotification = function() {\n            self.resized();\n        };\n\n        this.addFinEventListener('fin-canvas-mousemove', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseMove(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-mousedown', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.keys = e.detail.keys;\n            mouseEvent.primitiveEvent = e;\n            self.mouseDownState = mouseEvent;\n            self.delegateMouseDown(mouseEvent);\n            self.fireSyntheticMouseDownEvent(mouseEvent);\n            self.repaint();\n        });\n\n\n        // this.addFinEventListener('fin-canvas-click', function(e) {\n        //     if (self.resolveProperty('readOnly')) {\n        //         return;\n        //     }\n        //     //self.stopEditing();\n        //     var mouse = e.detail.mouse;\n        //     var mouseEvent = self.getGridCellFromMousePoint(mouse);\n        //     mouseEvent.primitiveEvent = e;\n        //     self.fireSyntheticClickEvent(mouseEvent);\n        // });\n\n        this.addFinEventListener('fin-canvas-mouseup', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.dragging = false;\n            if (self.isScrollingNow()) {\n                self.setScrollingNow(false);\n            }\n            if (self.columnDragAutoScrolling) {\n                self.columnDragAutoScrolling = false;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseUp(mouseEvent);\n            if (self.mouseDownState) {\n                self.fireSyntheticButtonPressedEvent(self.mouseDownState);\n            }\n            self.mouseDownState = null;\n            self.fireSyntheticMouseUpEvent(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-dblclick', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.fireSyntheticDoubleClickEvent(mouseEvent, e);\n            self.delegateDoubleClick(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-tap', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var tapEvent = self.getGridCellFromMousePoint(mouse);\n            tapEvent.primitiveEvent = e;\n            tapEvent.keys = e.detail.keys;\n            self.fireSyntheticClickEvent(tapEvent);\n            self.delegateTap(tapEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-drag', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.dragging = true;\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseDrag(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-keydown', function(e) {\n            var key = e.detail.char;\n            if (['DELETE'].indexOf(key) !== -1) {\n                if (!self.isEditing()) {\n                    setTimeout(function() {\n                        self.takeFocus();\n                    }, 50);\n                }\n            }\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.fireSyntheticKeydownEvent(e);\n            self.delegateKeyDown(e);\n        });\n\n        this.addFinEventListener('fin-canvas-keyup', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.fireSyntheticKeyupEvent(e);\n            self.delegateKeyUp(e);\n        });\n\n        this.addFinEventListener('fin-canvas-track', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            if (self.dragging) {\n                return;\n            }\n            var primEvent = e.detail.primitiveEvent;\n            if (Math.abs(primEvent.dy) > Math.abs(primEvent.dx)) {\n                if (primEvent.yDirection > 0) {\n                    self.scrollVBy(-2);\n                } else if (primEvent.yDirection < -0) {\n                    self.scrollVBy(2);\n                }\n            } else {\n                if (primEvent.xDirection > 0) {\n                    self.scrollHBy(-1);\n                } else if (primEvent.xDirection < -0) {\n                    self.scrollHBy(1);\n                }\n            }\n        });\n\n        // this.addFinEventListener('fin-canvas-holdpulse', function(e) {\n        //     console.log('holdpulse');\n        //     if (self.resolveProperty('readOnly')) {\n        //         return;\n        //     }\n        //     var mouse = e.detail.mouse;\n        //     var mouseEvent = self.getGridCellFromMousePoint(mouse);\n        //     mouseEvent.primitiveEvent = e;\n        //     self.delegateHoldPulse(mouseEvent);\n        // });\n\n        this.addFinEventListener('fin-canvas-wheelmoved', function(e) {\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateWheelMoved(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-mouseout', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateMouseExit(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-context-menu', function(e) {\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateContextMenu(mouseEvent);\n        });\n\n        this.div.removeAttribute('tabindex');\n\n    },\n\n    convertViewPointToDataPoint: function(viewPoint) {\n        return this.getBehavior().convertViewPointToDataPoint(viewPoint);\n    },\n\n    convertDataPointToViewPoint: function(dataPoint) {\n        return this.getBehavior().convertDataPointToViewPoint(dataPoint);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Add an event listener to me.\n     * @param {string} eventName - The type of event we are interested in.\n     * @param {function} callback - The event handler.\n     */\n    addFinEventListener: function(eventName, callback) {\n        this.canvas.addEventListener(eventName, callback);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Set for `scrollingNow` field.\n     * @param {boolean} isItNow - The type of event we are interested in.\n     */\n    setScrollingNow: function(isItNow) {\n        this.scrollingNow = isItNow;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The `scrollingNow` field.\n     */\n    isScrollingNow: function() {\n        return this.scrollingNow;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The index of the column divider under the mouse coordinates.\n     * @param {MouseEvent} mouseEvent - The event to interogate.\n     */\n    overColumnDivider: function(mouseEvent) {\n        var x = mouseEvent.primitiveEvent.detail.mouse.x;\n        var whichCol = this.getRenderer().overColumnDivider(x);\n        return whichCol;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The index of the row divider under the mouse coordinates.\n     * @param {MouseEvent} mouseEvent - The event to interogate.\n     */\n    overRowDivider: function(mouseEvent) {\n        var y = mouseEvent.primitiveEvent.detail.mouse.y;\n        var which = this.getRenderer().overRowDivider(y);\n        return which;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Switch the cursor for the grid.\n     * @param {string} cursorName - A well know cursor name.\n     * @see [cursor names](http://www.javascripter.net/faq/stylesc.htm)\n     */\n    beCursor: function(cursorName) {\n        if (!cursorName) {\n            cursorName = 'default';\n        }\n        this.div.style.cursor = cursorName;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate the wheel moved event to the behavior.\n     * @param {Event} event - The pertinent event.\n     */\n    delegateWheelMoved: function(event) {\n        var behavior = this.getBehavior();\n        behavior.onWheelMoved(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseExit to the behavior (model).\n     * @param {Event} event - The pertinent event.\n     */\n    delegateMouseExit: function(event) {\n        var behavior = this.getBehavior();\n        behavior.handleMouseExit(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseExit to the behavior (model).\n     * @param {Event} event - The pertinent event.\n     */\n    delegateContextMenu: function(event) {\n        var behavior = this.getBehavior();\n        behavior.onContextMenu(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseMove to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseMove: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onMouseMove(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mousedown to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseDown: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.handleMouseDown(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mouseup to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseUp: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onMouseUp(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate tap to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateTap: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onTap(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mouseDrag to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseDrag: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onMouseDrag(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc We've been doubleclicked on. Delegate through the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateDoubleClick: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onDoubleClick(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate holdpulse through the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateHoldPulse: function(mouseDetails) {\n        var behavior = this.getBehavior();\n        behavior.onHoldPulse(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Generate a function name and call it on self.\n     * @desc This should also be delegated through Behavior keeping the default implementation here though.\n     * @param {event} event - The pertinent event.\n     */\n    delegateKeyDown: function(event) {\n        var behavior = this.getBehavior();\n        behavior.onKeyDown(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Generate a function name and call it on self.\n     * @desc This should also be delegated through Behavior keeping the default implementation here though.\n     * @param {event} event - The pertinent event.\n     */\n    delegateKeyUp: function(event) {\n        var behavior = this.getBehavior();\n        behavior.onKeyUp(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Shut down the current cell editor.\n     */\n    stopEditing: function() {\n        if (this.cellEditor && this.isEditing()) {\n            if (this.cellEditor.stopEditing) {\n                this.cellEditor.stopEditing();\n            }\n            this.cellEditor = null;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Register a cell editor.\n     * @desc This is typically called from within a cell-editor's `installOn` method, when it is being initialized as a plugin.\n     * @param {string} alias - The name/id of the cell editor.\n     * @param {fin-hypergrid-cell-editor-base} cellEditor - see [fin-hypergrid-cell-editor-base](module-cell-editors_base.html)\n     */\n    registerCellEditor: function(alias, cellEditor) {\n        this.cellEditors[alias] = cellEditor;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Rectangle} The pixel coordinates of just the center 'main\" data area.\n     */\n    getDataBounds: function() {\n        var colDNDHackWidth = 200; //this was a hack to help with column dnd, need to factor this into a shared variable\n        //var behavior = this.getBehavior();\n        var b = this.canvas.bounds;\n\n        //var x = this.getRowNumbersWidth();\n        // var y = behavior.getFixedRowsHeight() + 2;\n\n        var result = new Rectangle(0, 0, b.origin.x + b.extent.x - colDNDHackWidth, b.origin.y + b.extent.y);\n        return result;\n    },\n\n    getRowNumbersWidth: function() {\n        if (this.isShowRowNumbers()) {\n            return this.getRenderer().getRowNumbersWidth();\n        } else {\n            return 0;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Canvas} Our fin-canvas instance.\n     */\n    getCanvas: function() {\n        return this.canvas;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Open the given cell-editor at the provided model coordinates.\n     * @param {string} cellEditor - The specific cell editor to use.\n     * @param {Point} coordinates - The pixel locaiton of the cell to edit at.\n     */\n    editAt: function(cellEditor, coordinates) {\n\n        this.cellEditor = cellEditor;\n\n        var cell = coordinates.gridCell;\n\n        var x = cell.x;\n        var y = cell.y;\n\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        var editPoint = new Point(x, y);\n        this.setMouseDown(editPoint);\n        this.setDragExtent(new Point(0, 0));\n        cellEditor.beginEditAt(editPoint);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given column is fully visible.\n     * @param {number} columnIndex - The column index in question.\n     */\n    isColumnVisible: function(columnIndex) {\n        var isVisible = this.getRenderer().isColumnVisible(columnIndex);\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given row is fully visible.\n     * @param {number} rowIndex - The row index in question.\n     */\n    isDataRowVisible: function(rowIndex) {\n        var isVisible = this.getRenderer().isRowVisible(rowIndex);\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given cell is fully is visible.\n     * @param {number} columnIndex - The column index in question.\n     * @param {number} rowIndex - The row index in question.\n     */\n    isDataVisible: function(columnIndex, rowIndex) {\n        var isVisible = this.isDataRowVisible(rowIndex) && this.isColumnVisible(columnIndex);\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll in the `offsetX` direction if column index `colIndex` is not visible.\n     * @param {number} colIndex - The column index in question.\n     * @param {number} offsetX - The direction and magnitude to scroll if we need to.\n     */\n    insureModelColIsVisible: function(colIndex, offsetX) {\n        //-1 because we want only fully visible columns, don't include partially\n        //visible columns\n        var maxCols = this.getColumnCount() - 1;\n        var indexToCheck = colIndex;\n\n        if (offsetX > 0) {\n            indexToCheck++;\n        }\n\n        if (!this.isColumnVisible(indexToCheck) || colIndex === maxCols) {\n            //the scroll position is the leftmost column {\n            this.scrollBy(offsetX, 0);\n            return true;\n        }\n        return false;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll in the offsetY direction if column index c is not visible.\n     * @param {number} rowIndex - The column index in question.\n     * @param {number} offsetX - The direction and magnitude to scroll if we need to.\n     */\n    insureModelRowIsVisible: function(rowIndex, offsetY) {\n        //-1 because we want only fully visible rows, don't include partially\n        //viewable rows\n        var maxRows = this.getRowCount() - 1;\n        var indexToCheck = rowIndex;\n\n        if (offsetY > 0) {\n            indexToCheck++;\n        }\n\n        if (!this.isDataRowVisible(indexToCheck) || rowIndex === maxRows) {\n            //the scroll position is the topmost row\n            this.scrollBy(0, offsetY);\n            return true;\n        }\n        return false;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll horizontal and vertically by the provided offsets.\n     * @param {number} offsetX - Scroll in the x direction this much.\n     * @param {number} offsetY - Scroll in the y direction this much.\n     */\n    scrollBy: function(offsetX, offsetY) {\n        this.scrollHBy(offsetX);\n        this.scrollVBy(offsetY);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll vertically by the provided offset.\n     * @param {number} offsetY - Scroll in the y direction this much.\n     */\n    scrollVBy: function(offsetY) {\n        var max = this.sbVScroller.range.max;\n        var oldValue = this.getVScrollValue();\n        var newValue = Math.min(max, Math.max(0, oldValue + offsetY));\n        if (newValue === oldValue) {\n            return;\n        }\n        this.setVScrollValue(newValue);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll horizontally by the provided offset.\n     * @param {number} offsetX - Scroll in the x direction this much.\n     */\n    scrollHBy: function(offsetX) {\n        var max = this.sbHScroller.range.max;\n        var oldValue = this.getHScrollValue();\n        var newValue = Math.min(max, Math.max(0, oldValue + offsetX));\n        if (newValue === oldValue) {\n            return;\n        }\n        this.setHScrollValue(newValue);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Answer which data cell is under a pixel value mouse point.\n     * @param {mousePoint} mouse - The mouse point to interrogate.\n     */\n\n    getGridCellFromMousePoint: function(mouse) {\n        var cell = this.getRenderer().getGridCellFromMousePoint(mouse);\n        return cell;\n    },\n\n    /**\n     * @returns {Rectangle} The pixel based bounds rectangle given a data cell point.\n     * @param {Point} cell - The pixel location of the mouse.\n     * @memberOf Hypergrid.prototype\n     */\n    getBoundsOfCell: function(cell) {\n        var b = this.getRenderer().getBoundsOfCell(cell);\n\n        //we need to convert this to a proper rectangle\n        var newBounds = new Rectangle(b.x, b.y, b.width, b.height);\n        return newBounds;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc This is called by the fin-canvas when a resize occurs.\n     */\n    resized: function() {\n        this.synchronizeScrollingBoundries();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A click event occured.\n     * @desc Determine the cell and delegate to the behavior (model).\n     * @param {MouseEvent} event - The mouse event to interrogate.\n     */\n    cellClicked: function(event) {\n        var cell = event.gridCell;\n        var colCount = this.getColumnCount();\n        var rowCount = this.getRowCount();\n\n        //click occured in background area\n        if (cell.x > colCount || cell.y > rowCount) {\n            return;\n        }\n\n        //var behavior = this.getBehavior();\n        var hovered = this.getHoverCell();\n        var sy = this.getVScrollValue();\n        var x = hovered.x;\n        // if (hovered.x > -1) {\n        //     x = behavior.translateColumnIndex(hovered.x + this.getHScrollValue());\n        // }\n        if (hovered.y < 0) {\n            sy = 0;\n        }\n        hovered = new Point(x, hovered.y + sy);\n        this.getBehavior().cellClicked(hovered, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} atBottom - this value is in the \"bottom\" totals area\n     */\n    setTotalsValueNotification: function(x, y, value, atBottom) {\n        this.fireSyntheticSetTotalsValue(x, y, value, atBottom);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} atBottom - this value is in the \"bottom\" totals area\n     */\n    fireSyntheticSetTotalsValue: function(x, y, value, atBottom) {\n        var clickEvent = new CustomEvent('fin-set-totals-value', {\n            detail: {\n                x: x,\n                y: y,\n                value: value,\n                area: atBottom ? 'bottom' : 'top'\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyUpEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-key-up', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyDownEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-key-down', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            },\n\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyPressEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-key-press', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            },\n\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorDataChangeEvent: function(inputControl, oldValue, newValue) {\n        var clickEvent = new CustomEvent('fin-editor-data-change', {\n            detail: {\n                input: inputControl,\n                oldValue: oldValue,\n                newValue: newValue\n            },\n            cancelable: true\n        });\n        return this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-row-selection-changed` event.\n     */\n    fireSyntheticRowSelectionChangedEvent: function() {\n        var selectionEvent = new CustomEvent('fin-row-selection-changed', {\n            detail: {\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections(),\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n\n    fireSyntheticColumnSelectionChangedEvent: function() {\n        var selectionEvent = new CustomEvent('fin-column-selection-changed', {\n            detail: {\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and dispatch a `fin-selection-changed` event.\n     */\n    selectionChanged: function() {\n        var selectedRows = this.getSelectedRows();\n        var selectionEvent = new CustomEvent('fin-selection-changed', {\n            detail: {\n                rows: selectedRows,\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections(),\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n\n\n    getRowSelection: function() {\n        var c, column, self = this,\n            selectedRowIndexes = this.getSelectionModel().getSelectedRows(),\n            numCols = this.getColumnCount(),\n            result = {};\n\n        function setValue(selectedRowIndex, r) {\n            column[r] = valueOrFunctionExecute(self.getValue(c, selectedRowIndex));\n        }\n\n        for (c = 0; c < numCols; c++) {\n            column = new Array(selectedRowIndexes.length);\n            result[this.getField(c)] = column;\n            selectedRowIndexes.forEach(setValue);\n        }\n\n        return result;\n    },\n\n    getRowSelectionMatrix: function() {\n        var c, self = this,\n            selectedRowIndexes = this.getSelectionModel().getSelectedRows(),\n            numCols = this.getColumnCount(),\n            result = new Array(numCols);\n\n        function getValue(selectedRowIndex, r) {\n            result[c][r] = valueOrFunctionExecute(self.getValue(c, selectedRowIndex));\n        }\n\n        for (c = 0; c < numCols; c++) {\n            result[c] = new Array(selectedRowIndexes.length);\n            selectedRowIndexes.forEach(getValue);\n        }\n\n        return result;\n    },\n\n    getColumnSelectionMatrix: function() {\n        var selectedColumnIndexes = this.getSelectedColumns();\n        var numRows = this.getRowCount();\n        var result = new Array(selectedColumnIndexes.length);\n        var self = this;\n        selectedColumnIndexes.forEach(function(selectedColumnIndex, c) {\n            result[c] = new Array(numRows);\n            for (var r = 0; r < numRows; r++) {\n                result[c][r] = valueOrFunctionExecute(self.getValue(selectedColumnIndex, r));\n            }\n        });\n        return result;\n    },\n\n    getColumnSelection: function() {\n        var selectedColumnIndexes = this.getSelectedColumns();\n        var result = {};\n        var rowCount = this.getRowCount();\n        var self = this;\n        selectedColumnIndexes.forEach(function(selectedColumnIndex) {\n            var column = new Array(rowCount);\n            result[self.getField(selectedColumnIndex)] = column;\n            for (var r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(self.getValue(selectedColumnIndex, r));\n            }\n        });\n        return result;\n    },\n\n    getSelection: function() {\n        var self = this;\n        var selections = this.getSelections();\n        var result = new Array(selections.length);\n        selections.forEach(function(selectionRect, i) {\n            result[i] = self._getSelection(selectionRect);\n        });\n        return result;\n    },\n\n    _getSelection: function(rect) {\n        rect = normalizeRect(rect);\n        var colCount = rect.extent.x + 1;\n        var rowCount = rect.extent.y + 1;\n        var ox = rect.origin.x;\n        var oy = rect.origin.y;\n        var result = {};\n        var r;\n        for (var c = 0; c < colCount; c++) {\n            var column = new Array(rowCount);\n            result[this.getField(c + ox)] = column;\n            for (r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(this.getValue(ox + c, oy + r));\n            }\n        }\n        return result;\n    },\n\n    getSelectionMatrix: function() {\n        var self = this;\n        var selections = this.getSelections();\n        var result = new Array(selections.length);\n        selections.forEach(function(selectionRect, i) {\n            result[i] = self._getSelectionMatrix(selectionRect);\n        });\n        return result;\n    },\n\n    _getSelectionMatrix: function(rect) {\n        rect = normalizeRect(rect);\n        var colCount = rect.extent.x + 1;\n        var rowCount = rect.extent.y + 1;\n        var ox = rect.origin.x;\n        var oy = rect.origin.y;\n        var result = [];\n        for (var c = 0; c < colCount; c++) {\n            var column = new Array(rowCount);\n            result[c] = column;\n            for (var r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(this.getValue(ox + c, oy + r));\n            }\n        }\n        return result;\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-context-menu` event\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticContextMenuEvent: function(e) {\n        e.gridCell = this.convertViewPointToDataPoint(e.gridCell);\n        var event = new CustomEvent('fin-context-menu', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    fireSyntheticMouseUpEvent: function(e) {\n        var event = new CustomEvent('fin-mouseup', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    fireSyntheticMouseDownEvent: function(e) {\n        var event = new CustomEvent('fin-mousedown', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.getSelectionModel().getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    isViewableButton: function(c, r) {\n        return this.getRenderer().isViewableButton(c, r);\n    },\n\n    fireSyntheticButtonPressedEvent: function(evt) {\n        var dataCell = evt.dataCell;\n        var gridCell = evt.gridCell;\n        if (!this.isViewableButton(dataCell.x, dataCell.y)) {\n            return;\n        }\n        var event = new CustomEvent('fin-button-pressed', {\n            detail: {\n                gridCell: gridCell\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-keydown` event.\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticKeydownEvent: function(keyEvent) {\n        var clickEvent = new CustomEvent('fin-keydown', {\n            detail: keyEvent.detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-keyup` event.\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticKeyupEvent: function(keyEvent) {\n        var clickEvent = new CustomEvent('fin-keyup', {\n            detail: keyEvent.detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticFilterAppliedEvent: function(details) {\n        var event = new CustomEvent('fin-filter-applied', {\n            detail: details\n        });\n        if (this.canvas) {\n            this.canvas.dispatchEvent(event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-enter` event\n     * @param {Point} cell - The pixel location of the cell in which the click event occurred.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticOnCellEnterEvent: function(cell) {\n        var detail = {\n            gridCell: cell,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-cell-enter', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticGroupsChangedEvent: function(groups) {\n        var detail = {\n            groups: groups,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-groups-changed', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-exit` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticOnCellExitEvent: function(cell) {\n        var detail = {\n            gridCell: cell,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-cell-exit', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-click` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticClickEvent: function(mouseEvent) {\n        this.stopEditing();\n        var cell = mouseEvent.gridCell;\n        var detail = {\n            gridCell: cell,\n            mousePoint: mouseEvent.mousePoint,\n            keys: mouseEvent.keys,\n            primitiveEvent: mouseEvent,\n            time: Date.now(),\n            grid: this\n        };\n        this.getBehavior().enhanceDoubleClickEvent(detail);\n        var clickEvent = new CustomEvent('fin-click', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-double-click` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticDoubleClickEvent: function(mouseEvent) {\n        this.stopEditing();\n        var cell = mouseEvent.gridCell;\n        var behavior = this.getBehavior();\n        var detail = {\n            gridCell: cell,\n            mousePoint: mouseEvent.mousePoint,\n            time: Date.now(),\n            grid: this\n        };\n        behavior.enhanceDoubleClickEvent(mouseEvent);\n        var clickEvent = new CustomEvent('fin-double-click', {\n            detail: detail\n        });\n        behavior.cellDoubleClicked(cell, mouseEvent);\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a rendered event.\n     */\n    fireSyntheticGridRenderedEvent: function() {\n        var event = new CustomEvent('fin-grid-rendered', {\n            detail: {\n                source: this,\n                time: Date.now()\n            }\n        });\n        if (this.canvas) {\n            this.canvas.dispatchEvent(event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a scroll event.\n     * @param {string} type - Should be either `fin-scroll-x` or `fin-scroll-y`.\n     * @param {number} oldValue - The old scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    fireScrollEvent: function(type, oldValue, newValue) {\n        var event = new CustomEvent(type, {\n            detail: {\n                oldValue: oldValue,\n                value: newValue,\n                time: Date.now()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the vertical scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    setVScrollValue: function(y) {\n        y = Math.round(y);\n        var max = this.sbVScroller.range.max;\n        y = Math.min(max, Math.max(0, y));\n        var self = this;\n        if (y === this.vScrollValue) {\n            return;\n        }\n        this.getBehavior()._setScrollPositionY(y);\n        var oldY = this.vScrollValue;\n        this.vScrollValue = y;\n        this.scrollValueChangedNotification();\n        setTimeout(function() {\n            // self.sbVRangeAdapter.subjectChanged();\n            self.fireScrollEvent('fin-scroll-y', oldY, y);\n        });\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @return {number} The vertical scroll value.\n     */\n    getVScrollValue: function() {\n        return this.vScrollValue;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the horizontal scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    setHScrollValue: function(x) {\n        x = Math.round(x);\n        var max = this.sbHScroller.range.max;\n        x = Math.min(max, Math.max(0, x));\n        var self = this;\n        if (x === this.hScrollValue) {\n            return;\n        }\n        this.getBehavior()._setScrollPositionX(x);\n        var oldX = this.hScrollValue;\n        this.hScrollValue = x;\n        this.scrollValueChangedNotification();\n        setTimeout(function() {\n            //self.sbHRangeAdapter.subjectChanged();\n            self.fireScrollEvent('fin-scroll-x', oldX, x);\n            self.synchronizeScrollingBoundries();\n        });\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns The vertical scroll value.\n     */\n    getHScrollValue: function() {\n        return this.hScrollValue;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Request input focus.\n     */\n    takeFocus: function() {\n        if (this.isEditing()) {\n            this.stopEditing();\n        } else {\n            this.getCanvas().takeFocus();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Request focus for our cell editor.\n     */\n    editorTakeFocus: function() {\n        if (this.cellEditor) {\n            return this.cellEditor.takeFocus();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have a currently active cell editor.\n     */\n    isEditing: function() {\n        if (this.cellEditor) {\n            return this.cellEditor.isEditing;\n        }\n        return false;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Initialize the scroll bars.\n     */\n    initScrollbars: function() {\n\n        var self = this;\n\n        var horzBar = new FinBar({\n            orientation: 'horizontal',\n            onchange: self.setHScrollValue.bind(self),\n            cssStylesheetReferenceElement: this.div\n        });\n\n        var vertBar = new FinBar({\n            orientation: 'vertical',\n            onchange: self.setVScrollValue.bind(self),\n            paging: {\n                up: self.pageUp.bind(self),\n                down: self.pageDown.bind(self)\n            }\n        });\n\n        this.sbHScroller = horzBar;\n        this.sbVScroller = vertBar;\n\n        var hPrefix = this.resolveProperty('hScrollbarClassPrefix');\n        var vPrefix = this.resolveProperty('vScrollbarClassPrefix');\n\n        if (hPrefix && hPrefix !== '') {\n            this.sbHScroller.classPrefix = hPrefix;\n        }\n\n        if (vPrefix && vPrefix !== '') {\n            this.sbVScroller.classPrefix = vPrefix;\n        }\n\n        this.div.appendChild(horzBar.bar);\n        this.div.appendChild(vertBar.bar);\n\n        this.resizeScrollbars();\n\n    },\n\n    resizeScrollbars: function() {\n        this.sbHScroller.shortenBy(this.sbVScroller).resize();\n        //this.sbVScroller.shortenBy(this.sbHScroller);\n        this.sbVScroller.resize();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll values have changed, we've been notified.\n     */\n    setVScrollbarValues: function(max) {\n        this.sbVScroller.range = {\n            min: 0,\n            max: max\n        };\n    },\n\n    setHScrollbarValues: function(max) {\n        this.sbHScroller.range = {\n            min: 0,\n            max: max\n        };\n    },\n\n    scrollValueChangedNotification: function() {\n\n        if (this.hScrollValue === this.sbPrevHScrollValue && this.vScrollValue === this.sbPrevVScrollValue) {\n            return;\n        }\n\n        this.sbPrevHScrollValue = this.hScrollValue;\n        this.sbPrevVScrollValue = this.vScrollValue;\n\n        if (this.cellEditor) {\n            this.cellEditor.scrollValueChangedNotification();\n        }\n\n        this.computeCellsBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Get data value at given cell.\n     * @desc Delegates to the behavior.\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     * @param {*} value\n     */\n    getValue: function(x, y) {\n        return this.getBehavior().getValue(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set a data value into the behavior (model) at the given point\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     */\n    setValue: function(x, y, value) {\n        this.getBehavior().setValue(x, y, value);\n    },\n\n    getColumnAlignment: function(c) {\n        return this.getBehavior().getColumnAlignment(c);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The data dimensions have changed, or our pixel boundries have changed.\n     * Adjust the scrollbar properties as necessary.\n     */\n    synchronizeScrollingBoundries: function() {\n        //327/664\n        var behavior = this.getBehavior();\n\n        var numFixedColumns = this.getFixedColumnCount();\n        var numFixedRows = this.getFixedRowCount();\n\n        var numColumns = this.getColumnCount();\n        var numRows = this.getRowCount();\n\n        var bounds = this.getBounds();\n        if (!bounds) {\n            return;\n        }\n        var scrollableHeight = bounds.height - behavior.getFixedRowsMaxHeight() - 15; //5px padding at bottom and right side\n        var scrollableWidth = (bounds.width - 200) - behavior.getFixedColumnsMaxWidth() - 15;\n\n        var lastPageColumnCount = 0;\n        var columnsWidth = 0;\n        for (; lastPageColumnCount < numColumns; lastPageColumnCount++) {\n            var eachWidth = this.getColumnWidth(numColumns - lastPageColumnCount - 1);\n            columnsWidth = columnsWidth + eachWidth;\n            if (columnsWidth > scrollableWidth) {\n                break;\n            }\n        }\n\n        var lastPageRowCount = 0;\n        var rowsHeight = 0;\n        for (; lastPageRowCount < numRows; lastPageRowCount++) {\n            var eachHeight = this.getRowHeight(numRows - lastPageRowCount - 1);\n            rowsHeight = rowsHeight + eachHeight;\n            if (rowsHeight > scrollableHeight) {\n                break;\n            }\n        }\n\n        var hMax = Math.max(0, numColumns - numFixedColumns - lastPageColumnCount);\n        this.setHScrollbarValues(hMax);\n\n        var vMax = 1 + Math.max(0, numRows - numFixedRows - lastPageRowCount);\n        this.setVScrollbarValues(vMax);\n\n        this.setHScrollValue(Math.min(this.getHScrollValue(), hMax));\n        this.setVScrollValue(Math.min(this.getVScrollValue(), vMax));\n\n        //this.getCanvas().resize();\n        this.computeCellsBounds();\n        this.repaint();\n\n        this.resizeScrollbars();\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Note that \"viewable rows\" includes any partially viewable rows.\n     * @returns {number} The number of viewable rows.\n     */\n    getVisibleRows: function() {\n        return this.getRenderer().getVisibleRows();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Note that \"viewable columns\" includes any partially viewable columns.\n     * @returns {number} The number of viewable columns.\n     */\n    getVisibleColumns: function() {\n        return this.getRenderer().getVisibleColumns();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Initialize the renderer sub-component.\n     */\n    initRenderer: function() {\n        this.renderer = new Renderer(this);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Renderer} sub-component\n     */\n    getRenderer: function() {\n        return this.renderer;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The width of the given column.\n     * @param {number} columnIndex - The untranslated column index.\n     */\n    getColumnWidth: function(columnIndex) {\n        return this.getBehavior().getColumnWidth(columnIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the width of the given column.\n     * @param {number} columnIndex - The untranslated column index.\n     * @param {number} columnWidth - The width in pixels.\n     */\n    setColumnWidth: function(columnIndex, columnWidth) {\n        this.stopEditing();\n        this.getBehavior().setColumnWidth(columnIndex, columnWidth);\n    },\n\n    getColumnEdge: function(c) {\n        return this.getBehavior().getColumnEdge(c, this.getRenderer());\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The total width of all the fixed columns.\n     */\n    getFixedColumnsWidth: function() {\n        return this.getBehavior().getFixedColumnsWidth();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The height of the given row\n     * @param {number} rowIndex - The untranslated fixed column index.\n     */\n    getRowHeight: function(rowIndex) {\n        return this.getBehavior().getRowHeight(rowIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the height of the given row.\n     * @param {number} rowIndex - The row index.\n     * @param {number} rowHeight - The width in pixels.\n     */\n    setRowHeight: function(rowIndex, rowHeight) {\n        this.stopEditing();\n        this.getBehavior().setRowHeight(rowIndex, rowHeight);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The total fixed rows height\n     */\n    getFixedRowsHeight: function() {\n        return this.getBehavior().getFixedRowsHeight();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of columns.\n     */\n    getColumnCount: function() {\n        return this.getBehavior().getColumnCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of fixed rows.\n     */\n    getRowCount: function() {\n        return this.getBehavior().getRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of unfiltered rows.\n     */\n    getUnfilteredRowCount: function() {\n        return this.getBehavior().getUnfilteredRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        return this.getBehavior().getFixedColumnCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        return this.getBehavior().getFixedRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary The top left area has been clicked on\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    topLeftClicked: function(mouse) {\n        this.getBehavior().topLeftClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A fixed row has been clicked.\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    rowHeaderClicked: function(mouse) {\n        this.getBehavior().rowHeaderClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A fixed column has been clicked.\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    columnHeaderClicked: function(mouse) {\n        this.getBehavior().columnHeaderClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc An edit event has occurred. Activate the editor.\n     * @param {event} event - The event details.\n     */\n    _activateEditor: function(event) {\n        var gridCell = event.gridCell;\n        this.activateEditor(gridCell.x, gridCell.y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Activate the editor at the given coordinates.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     */\n    activateEditor: function(x, y) {\n        if (!this.isEditable() && !this.isFilterRow(y)) {\n            return;\n        }\n        var editor = this.getCellEditorAt(x, y);\n        if (!editor) {\n            return;\n        }\n        var point = editor.getEditorPoint();\n        if (editor) {\n            if (point.x === x && point.y === y && editor.isEditing) {\n                return; //we're already open at this location\n            } else if (this.isEditing()) {\n                this.stopEditing(); //other editor is open, close it first\n            }\n            event.gridCell = {\n                x: x,\n                y: y\n            };\n            this.editAt(editor, event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Get the cell editor.\n     * @desc Delegates to the behavior.\n     * @returns The cell editor at the given coordinates.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     */\n    getCellEditorAt: function(x, y) {\n        return this.getBehavior()._getCellEditorAt(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Toggle HiDPI support.\n     * @desc HiDPI support is now *on* by default.\n     * > There used to be a bug in Chrome that caused severe slow down on bit blit of large images, so this HiDPI needed to be optional.\n     */\n    toggleHiDPI: function() {\n        if (this.useHiDPI()) {\n            this.removeAttribute('hidpi');\n        } else {\n            this.setAttribute('hidpi', null);\n        }\n        this.canvas.resize();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} Te HiDPI ratio.\n     */\n    getHiDPI: function(ctx) {\n        if (window.devicePixelRatio && this.useHiDPI()) {\n            var devicePixelRatio = window.devicePixelRatio || 1;\n            var backingStoreRatio = ctx.webkitBackingStorePixelRatio ||\n                ctx.mozBackingStorePixelRatio ||\n                ctx.msBackingStorePixelRatio ||\n                ctx.oBackingStorePixelRatio ||\n                ctx.backingStorePixelRatio || 1;\n\n            var ratio = devicePixelRatio / backingStoreRatio;\n            return ratio;\n        } else {\n            return 1;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The width of the given (recently rendered) column.\n     * @param {number} colIndex - The column index.\n     */\n    getRenderedWidth: function(colIndex) {\n        return this.renderer.getRenderedWidth(colIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The height of the given (recently rendered) row.\n     * @param {number} rowIndex - Tthe row index.\n     */\n    getRenderedHeight: function(rowIndex) {\n        return this.renderer.getRenderedHeight(rowIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {CellEditor} The cell editor at alias \"name\" (a sub-component).\n     * @param {string} name\n     */\n    resolveCellEditor: function(name) {\n        return this.cellEditors[name];\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n    update the cursor under the hover cell\n     */\n    updateCursor: function() {\n        var translate = this.getBehavior();\n        var cursor = translate.getCursorAt(-1, -1);\n        var hoverCell = this.getHoverCell();\n        if (hoverCell && hoverCell.x > -1 && hoverCell.y > -1) {\n            var x = hoverCell.x + this.getHScrollValue();\n            cursor = translate.getCursorAt(x, hoverCell.y + this.getVScrollValue());\n        }\n        this.beCursor(cursor);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Repaint the given cell.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     */\n    repaintCell: function(x, y) {\n        this.getRenderer().repaintCell(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The user is currently dragging a column to reorder it.\n     */\n    isDraggingColumn: function() {\n        return !!this.renderOverridesCache.dragger;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll up one full page.\n     * @returns {number}\n     */\n    pageUp: function() {\n        var rowNum = this.getRenderer().getPageUpRow();\n        this.setVScrollValue(rowNum);\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll down one full page.\n     * @returns {number}\n     */\n    pageDown: function() {\n        var rowNum = this.getRenderer().getPageDownRow();\n        this.setVScrollValue(rowNum);\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Not yet implemented.\n     */\n    pageLeft: function() {\n        console.log('page left');\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Not yet implemented.\n     */\n    pageRight: function() {\n        console.log('page right');\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object[]} Objects with the values that were just rendered.\n     */\n    getRenderedData: function() {\n        // assumes one row of headers\n        var behavior = this.getBehavior(),\n            renderer = this.getRenderer(),\n            colCount = this.getColumnCount(),\n            rowCount = renderer.getVisibleRows(),\n            headers = new Array(colCount),\n            results = new Array(rowCount),\n            row;\n\n        headers.forEach(function(header, c) {\n            headers[c] = behavior.getColumnId(c, 0);\n        });\n\n        results.forEach(function(result, r) {\n            row = results[r] = {\n                hierarchy: behavior.getFixedColumnValue(0, r)\n            };\n            headers.forEach(function(field, c) {\n                row[field] = behavior.getValue(c, r);\n            });\n        });\n\n        return results;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} An object that represents the currently selection row.\n     */\n    getSelectedRow: function() {\n        var sels = this.getSelectionModel().getSelections();\n        if (sels.length) {\n            var behavior = this.getBehavior(),\n                colCount = this.getColumnCount(),\n                topRow = sels[0].origin.y,\n                row = {\n                    //hierarchy: behavior.getFixedColumnValue(0, topRow)\n                };\n\n            for (var c = 0; c < colCount; c++) {\n                row[behavior.getColumnId(c, 0)] = behavior.getValue(c, topRow);\n            }\n\n            return row;\n        }\n    },\n\n    fireRequestCellEdit: function(cell, value) {\n        var clickEvent = new CustomEvent('fin-request-cell-edit', {\n            cancelable: true,\n            detail: {\n                value: value,\n                gridCell: cell,\n                time: Date.now()\n            }\n        });\n        return this.canvas.dispatchEvent(clickEvent); //I wasn't cancelled\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a fin-before-cell-edit event.\n     * @param {Point} cell - The x,y coordinates.\n     * @param {Object} value - The current value.\n     */\n    fireBeforeCellEdit: function(cell, oldValue, newValue, control) {\n        var clickEvent = new CustomEvent('fin-before-cell-edit', {\n            cancelable: true,\n            detail: {\n                oldValue: oldValue,\n                newValue: newValue,\n                gridCell: cell,\n                time: Date.now(),\n                input: control,\n                row: this.getRow(cell.y)\n            }\n        });\n        var proceed = this.canvas.dispatchEvent(clickEvent);\n        return proceed; //I wasn't cancelled\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Renderer} sub-component\n     * @param {Point} cell - The x,y coordinates.\n     * @param {Object} oldValue - The old value.\n     * @param {Object} newValue - The new value.\n     */\n    fireAfterCellEdit: function(cell, oldValue, newValue, control) {\n        var clickEvent = new CustomEvent('fin-after-cell-edit', {\n            detail: {\n                newValue: newValue,\n                oldValue: oldValue,\n                gridCell: cell,\n                time: Date.now(),\n                input: control,\n                row: this.getRow(cell.y)\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Autosize the column at colIndex for best fit.\n     * @param {number} colIndex - The column index to modify at\n     */\n    autosizeColumn: function(colIndex) {\n        var column = this.getBehavior().getColumn(colIndex);\n        column.checkColumnAutosizing(true);\n        this.computeCellsBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Enable/disable if this component can receive the focus.\n     * @param {boolean} - canReceiveFocus\n     */\n    setFocusable: function(canReceiveFocus) {\n        this.getCanvas().setFocusable(canReceiveFocus);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of columns that were just rendered\n     */\n    getVisibleColumnsCount: function() {\n        return this.getRenderer().getVisibleColumnsCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of rows that were just rendered\n     */\n    getVisibleRowsCount: function() {\n        return this.getRenderer().getVisibleRowsCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n    update the size of the grid\n     *\n     * #### returns: integer\n     */\n    updateSize: function() {\n        this.canvas.checksize();\n    },\n\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Stop the global repainting flag thread.\n     */\n    stopPaintThread: function() {\n        this.canvas.stopPaintThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Stop the global resize check flag thread.\n     */\n    stopResizeThread: function() {\n        this.canvas.stopResizeThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Restart the global resize check flag thread.\n     */\n    restartResizeThread: function() {\n        this.canvas.restartResizeThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Restart the global repainting check flag thread.\n     */\n    restartPaintThread: function() {\n        this.canvas.restartPaintThread();\n    },\n\n    swapColumns: function(source, target) {\n        this.getBehavior().swapColumns(source, target);\n    },\n\n    endDragColumnNotification: function() {\n        this.getBehavior().endDragColumnNotification();\n    },\n\n    getFixedColumnsMaxWidth: function() {\n        return this.getBehavior().getFixedColumnsMaxWidth();\n    },\n\n    isMouseDownInHeaderArea: function() {\n        var numHeaderColumns = this.getHeaderColumnCount();\n        var numHeaderRows = this.getHeaderRowCount();\n        var mouseDown = this.getMouseDown();\n        return mouseDown.x < numHeaderColumns || mouseDown.y < numHeaderRows;\n    },\n\n    isHeaderWrapping: function() {\n        return this.resolveProperty('headerTextWrapping');\n    },\n\n    _getBoundsOfCell: function(x, y) {\n        var bounds = this.getRenderer()._getBoundsOfCell(x, y);\n        return bounds;\n    },\n\n    getColumnProperties: function(columnIndex) {\n        var properties = this.getBehavior().getColumnProperties(columnIndex);\n        return properties;\n    },\n\n    setColumnProperties: function(columnIndex, properties) {\n        this.getBehavior().setColumnProperties(columnIndex, properties);\n    },\n\n    moveSingleSelect: function(x, y) {\n        this.getBehavior().moveSingleSelect(this, x, y);\n    },\n\n    selectCell: function(x, y) {\n        this.getSelectionModel().clear();\n        this.getSelectionModel().select(x, y, 0, 0);\n    },\n\n    getHeaderColumnCount: function() {\n        return this.getBehavior().getHeaderColumnCount();\n    },\n\n    toggleSort: function(x, keys) {\n        this.stopEditing();\n        var behavior = this.getBehavior();\n        var self = this;\n        behavior.toggleSort(x, keys);\n\n        setTimeout(function() {\n            self.synchronizeScrollingBoundries();\n            //self.behaviorChanged();\n            if (self.isColumnAutosizing()) {\n                behavior.autosizeAllColumns();\n            }\n            self.repaint();\n        }, 10);\n    },\n\n    toggleSelectColumn: function(x, keys) {\n        keys = keys || [];\n        var model = this.getSelectionModel();\n        var alreadySelected = model.isColumnSelected(x);\n        var hasCTRL = keys.indexOf('CTRL') > -1;\n        var hasSHIFT = keys.indexOf('SHIFT') > -1;\n        if (!hasCTRL && !hasSHIFT) {\n            model.clear();\n            if (!alreadySelected) {\n                model.selectColumn(x);\n            }\n        } else {\n            if (hasCTRL) {\n                if (alreadySelected) {\n                    model.deselectColumn(x);\n                } else {\n                    model.selectColumn(x);\n                }\n            }\n            if (hasSHIFT) {\n                model.clear();\n                model.selectColumn(this.lastEdgeSelection[0], x);\n            }\n        }\n        if (!alreadySelected && !hasSHIFT) {\n            this.lastEdgeSelection[0] = x;\n        }\n        this.repaint();\n        this.fireSyntheticColumnSelectionChangedEvent();\n    },\n\n    toggleSelectRow: function(y, keys) {\n\n        //we can select the totals rows if they exist,\n        //but not rows above that\n        var selectionEdge = this.getFilterRowIndex() + 1;\n        if (y < selectionEdge) {\n            return;\n        }\n\n        keys = keys || [];\n\n        var isSingleRowSelection = this.isSingleRowSelectionMode();\n        var model = this.getSelectionModel();\n        var alreadySelected = model.isRowSelected(y);\n        var hasCTRL = keys.indexOf('CTRL') > -1;\n        var hasSHIFT = keys.indexOf('SHIFT') > -1;\n\n        if (!hasCTRL && !hasSHIFT) {\n            //model.clear();\n            if (alreadySelected) {\n                model.deselectRow(y);\n            } else {\n                model.selectRow(y);\n            }\n        } else {\n            if (hasCTRL) {\n                if (alreadySelected) {\n                    model.deselectRow(y);\n                } else {\n                    if (isSingleRowSelection) {\n                        model.clearRowSelection();\n                    }\n                    model.selectRow(y);\n                }\n            }\n            if (hasSHIFT) {\n                model.clear();\n                model.selectRow(this.lastEdgeSelection[1], y);\n            }\n        }\n        if (!alreadySelected && !hasSHIFT) {\n            this.lastEdgeSelection[1] = y;\n        }\n        this.repaint();\n    },\n\n    selectViewportCell: function(x, y) {\n        var headerRowCount = this.getHeaderRowCount();\n        var renderer = this.getRenderer();\n        var realX = renderer.getVisibleColumns()[x];\n        var realY = renderer.getVisibleRows()[y];\n        this.clearSelections();\n        this.select(realX, realY + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(realX, realY + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToViewportCell: function(x, y) {\n        var selections = this.getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var headerRowCount = this.getHeaderRowCount();\n        var renderer = this.getRenderer();\n        var realX = renderer.getVisibleColumns()[x];\n        var realY = renderer.getVisibleRows()[y] + headerRowCount;\n        var selection = selections[0];\n        var origin = selection.origin;\n        this.setDragExtent(this.newPoint(realX - origin.x, realY - origin.y));\n        this.select(origin.x, origin.y, realX - origin.x, realY - origin.y);\n        this.repaint();\n    },\n\n    selectFinalCellOfCurrentRow: function() {\n        var x = this.getColumnCount() - 1;\n        var y = this.getSelectedRows()[0];\n        var headerRowCount = this.getHeaderRowCount();\n        this.clearSelections();\n        this.scrollBy(this.getColumnCount(), 0);\n        this.select(x, y + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(x, y + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToFinalCellOfCurrentRow: function() {\n        var selections = this.getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var selection = selections[0];\n        var origin = selection.origin;\n        var extent = selection.extent;\n        var columnCount = this.getColumnCount();\n        this.scrollBy(columnCount, 0);\n\n        this.clearSelections();\n        this.select(origin.x, origin.y, columnCount - origin.x - 1, extent.y);\n\n        this.repaint();\n    },\n\n    selectFirstCellOfCurrentRow: function() {\n        var x = 0;\n        var y = this.getSelectedRows()[0];\n        var headerRowCount = this.getHeaderRowCount();\n        this.clearSelections();\n        this.setHScrollValue(0);\n        this.select(x, y + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(x, y + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToFirstCellOfCurrentRow: function() {\n        var selections = this.getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var selection = selections[0];\n        var origin = selection.origin;\n        var extent = selection.extent;\n        this.clearSelections();\n        this.select(origin.x, origin.y, -origin.x, extent.y);\n        this.setHScrollValue(0);\n        this.repaint();\n    },\n\n    selectFinalCell: function() {\n        this.selectCell(this.getColumnCount() - 1, this.getRowCount() - 1);\n        this.scrollBy(this.getColumnCount(), this.getRowCount());\n        this.repaint();\n    },\n\n    selectToFinalCell: function() {\n\n        var selections = this.getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var selection = selections[0];\n        var origin = selection.origin;\n        var columnCount = this.getColumnCount();\n        var rowCount = this.getRowCount();\n\n        this.clearSelections();\n        this.select(origin.x, origin.y, columnCount - origin.x - 1, rowCount - origin.y - 1);\n        this.scrollBy(columnCount, rowCount);\n        this.repaint();\n    },\n\n    isShowRowNumbers: function() {\n        return this.resolveProperty('showRowNumbers');\n    },\n    isEditable: function() {\n        return this.resolveProperty('editable') === true;\n    },\n    isShowFilterRow: function() {\n        return this.resolveProperty('showFilterRow');\n    },\n    isShowHeaderRow: function() {\n        return this.resolveProperty('showHeaderRow');\n    },\n    getHeaderRowCount: function() {\n        return this.getBehavior().getHeaderRowCount();\n    },\n    isFilterRow: function(y) {\n        return y === this.getFilterRowIndex();\n    },\n    getFilterRowIndex: function() {\n        if (!this.isShowFilterRow()) {\n            return -1;\n        }\n        if (this.isShowHeaderRow()) {\n            return 1;\n        } else {\n            return 0;\n        }\n    },\n    setGroups: function(arrayOfColumnIndexes) {\n        this.getBehavior().setGroups(arrayOfColumnIndexes);\n    },\n    filterClicked: function(event) {\n        this.activateEditor(event.gridCell.x, event.gridCell.y);\n    },\n    hasHierarchyColumn: function() {\n        return this.getBehavior().hasHierarchyColumn();\n    },\n    isHierarchyColumn: function(x) {\n        if (!this.hasHierarchyColumn()) {\n            return false;\n        }\n        return x === 0;\n    },\n    checkScrollbarVisibility: function() {\n        // var hoverClassOver = this.resolveProperty('scrollbarHoverOver');\n        // var hoverClassOff = this.resolveProperty('scrollbarHoverOff');\n\n        // if (hoverClassOff === 'visible') {\n        //     this.sbHScroller.classList.remove(hoverClassOver);\n        //     this.sbVScroller.classList.remove(hoverClassOff);\n        //     this.sbHScroller.classList.add('visible');\n        //     this.sbVScroller.classList.add('visible');\n        // }\n    },\n    isColumnOrRowSelected: function() {\n        return this.getSelectionModel().isColumnOrRowSelected();\n    },\n    selectColumn: function(x1, x2) {\n        this.getSelectionModel().selectColumn(x1, x2);\n    },\n    selectRow: function(y1, y2) {\n        if (this.isSingleRowSelectionMode()) {\n            this.getSelectionModel().clearRowSelection();\n            y1 = y2;\n        } else {\n            y2 = y2 || y1;\n        }\n        var min = Math.min(y1, y2);\n        var max = Math.max(y1, y2);\n        var selectionEdge = this.getFilterRowIndex() + 1;\n        if (min < selectionEdge) {\n            return;\n        }\n        this.getSelectionModel().selectRow(min, max);\n    },\n    isRowNumberAutosizing: function() {\n        return this.resolveProperty('rowNumberAutosizing');\n    },\n    isRowSelected: function(r) {\n        return this.getSelectionModel().isRowSelected(r);\n    },\n    isColumnSelected: function(c) {\n        return this.getSelectionModel().isColumnSelected(c);\n    },\n    lookupFeature: function(key) {\n        return this.getBehavior().lookupFeature(key);\n    },\n    getRow: function(y) {\n        return this.getBehavior().getRow(y);\n    },\n    getFieldName: function(index) {\n        return this.getBehavior().getFieldName(index);\n    },\n\n    getColumnIndex: function(fieldName) {\n        return this.getBehavior().getColumnIndex(fieldName);\n    },\n    isCellSelection: function() {\n        return this.resolveProperty('cellSelection') === true;\n    },\n    isRowSelection: function() {\n        return this.resolveProperty('rowSelection') === true;\n    },\n    isColumnSelection: function() {\n        return this.resolveProperty('columnSelection') === true;\n    },\n    getComputedRow: function(y) {\n        return this.getBehavior().getComputedRow(y);\n    },\n    isColumnAutosizing: function() {\n        return this.resolveProperty('columnAutosizing') === true;\n    },\n    setGlobalFilter: function(string) {\n        this.getBehavior().setGlobalFilter(string);\n    },\n    selectRowsFromCells: function() {\n        if (this.isCheckboxOnlyRowSelections()) {\n            return;\n        }\n        var sm = this.getSelectionModel();\n        if (this.isSingleRowSelectionMode()) {\n            var last = sm.getLastSelection();\n            if (!last) {\n                sm.clearRowSelection();\n            } else {\n                this.selectRow(null, last.corner.y);\n            }\n        } else {\n            sm.selectRowsFromCells();\n        }\n    },\n    selectColumnsFromCells: function() {\n        this.getSelectionModel().selectColumnsFromCells();\n    },\n    getSelectedRows: function() {\n        return this.getBehavior().getSelectedRows();\n    },\n    getSelectedColumns: function() {\n        return this.getBehavior().getSelectedColumns();\n    },\n    getSelections: function() {\n        return this.getBehavior().getSelections();\n    },\n    getLastSelectionType: function() {\n        return this.getSelectionModel().getLastSelectionType();\n    },\n    isCellSelected: function(x, y) {\n        return this.getSelectionModel().isCellSelected(x, y);\n    },\n    isInCurrentSelectionRectangle: function(x, y) {\n        return this.getSelectionModel().isInCurrentSelectionRectangle(x, y);\n    },\n    selectAllRows: function() {\n        this.getSelectionModel().selectAllRows();\n    },\n    areAllRowsSelected: function() {\n        return this.getSelectionModel().areAllRowsSelected();\n    },\n    toggleSelectAllRows: function() {\n        if (this.areAllRowsSelected()) {\n            this.getSelectionModel().clear();\n        } else {\n            this.selectAllRows();\n        }\n        this.repaint();\n    },\n    getField: function(x) {\n        return this.getBehavior().getField(x);\n    },\n    isSingleRowSelectionMode: function() {\n        return this.resolveProperty('singleRowSelectionMode');\n    },\n    newPoint: function(x, y) {\n        return new Point(x, y);\n    },\n    newRectangle: function(x, y, width, height) {\n        return new Rectangle(x, y, width, height);\n    },\n    registerFilter: function(filter) {\n        customFilters[filter.alias] = filter;\n    },\n    getFilterFor: function(columnIndex) { //TODO: fix this\n        return customFilters.MyCustomFilter;\n    },\n    resolveFilter: function(alias) { //TODO: fix this\n        return customFilters[alias];\n    },\n    getFormattedValue: function(x, y) {\n        y = y + this.getHeaderRowCount();\n        var formatType = this.getColumnProperties(x).format;\n        var value = this.getValue(x, y);\n        var formatter = this.getFormatter(formatType);\n        var string = formatter(value);\n        return string;\n    }\n};\n\nfunction normalizeRect(rect) {\n    var o = rect.origin;\n    var c = rect.corner;\n\n    var ox = Math.min(o.x, c.x);\n    var oy = Math.min(o.y, c.y);\n\n    var cx = Math.max(o.x, c.x);\n    var cy = Math.max(o.y, c.y);\n\n    var result = new Rectangle(ox, oy, cx - ox, cy - oy);\n\n    return result;\n}\n\nfunction buildPolymerTheme() {\n    clearObjectProperties(polymerTheme);\n    var pb = document.createElement('paper-button');\n\n    pb.style.display = 'none';\n    pb.setAttribute('disabled', true);\n    document.body.appendChild(pb);\n    var p = window.getComputedStyle(pb);\n\n    var section = document.createElement('section');\n    section.style.display = 'none';\n    section.setAttribute('hero', true);\n    document.body.appendChild(section);\n\n    var h = window.getComputedStyle(document.querySelector('html'));\n    var hb = window.getComputedStyle(document.querySelector('html, body'));\n    var s = window.getComputedStyle(section);\n\n    polymerTheme.columnHeaderBackgroundColor = p.color;\n    polymerTheme.rowHeaderBackgroundColor = p.color;\n    polymerTheme.topLeftBackgroundColor = p.color;\n    polymerTheme.lineColor = p.backgroundColor;\n\n    polymerTheme.backgroundColor2 = hb.backgroundColor;\n\n    polymerTheme.color = h.color;\n    polymerTheme.fontFamily = h.fontFamily;\n    polymerTheme.backgroundColor = s.backgroundColor;\n\n    pb.setAttribute('disabled', false);\n    pb.setAttribute('secondary', true);\n    pb.setAttribute('raised', true);\n    p = window.getComputedStyle(pb);\n\n    polymerTheme.columnHeaderColor = p.color;\n    polymerTheme.rowHeaderColor = p.color;\n    polymerTheme.topLeftColor = p.color;\n\n\n    polymerTheme.backgroundSelectionColor = p.backgroundColor;\n    polymerTheme.foregroundSelectionColor = p.color;\n\n    pb.setAttribute('secondary', false);\n    pb.setAttribute('warning', true);\n\n    polymerTheme.columnHeaderForegroundSelectionColor = p.color;\n    polymerTheme.columnHeaderBackgroundSelectionColor = p.backgroundColor;\n    polymerTheme.rowHeaderForegroundSelectionColor = p.color;\n    polymerTheme.fixedColumnBackgroundSelectionColor = p.backgroundColor;\n\n    //check if there is actually a theme loaded if not, clear out all bogus values\n    //from my cache\n    if (polymerTheme.columnHeaderBackgroundSelectionColor === 'rgba(0, 0, 0, 0)' ||\n        polymerTheme.lineColor === 'transparent') {\n        clearObjectProperties(polymerTheme);\n    }\n\n    document.body.removeChild(pb);\n    document.body.removeChild(section);\n}\n\nfunction clearObjectProperties(obj) {\n    for (var prop in obj) {\n        if (obj.hasOwnProperty(prop)) {\n            delete obj[prop];\n        }\n    }\n}\n\nfunction valueOrFunctionExecute(valueOrFunction) {\n    var result = typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction;\n    return result || result === 0 ? result : '';\n}\n\nmodule.exports = Hypergrid;\n","'use strict';\n\nmodule.exports = (function() {\n\n    var oidPrefix = '.~.#%_'; //this should be something we never will see at the begining of a string\n    var counter = 0;\n\n    var hash = function(key) {\n        var typeOf = typeof key;\n        switch (typeOf) {\n            case 'number':\n                return oidPrefix + typeOf + '_' + key;\n            case 'string':\n                return oidPrefix + typeOf + '_' + key;\n            case 'boolean':\n                return oidPrefix + typeOf + '_' + key;\n            case 'symbol':\n                return oidPrefix + typeOf + '_' + key;\n            case 'undefined':\n                return oidPrefix + 'undefined';\n            case 'object':\n                /*eslint-disable */\n                if (key.___finhash) {\n                    return key.___finhash;\n                }\n                key.___finhash = oidPrefix + counter++;\n                return key.___finhash;\n            case 'function':\n                if (key.___finhash) {\n                    return key.___finhash;\n                }\n                key.___finhash = oidPrefix + counter++;\n                return key.___finhash; /*eslint-enable */\n        }\n    };\n\n    // Object.is polyfill, courtesy of @WebReflection\n    var is = Object.is ||\n        function(a, b) {\n            return a === b ? a !== 0 || 1 / a == 1 / b : a != a && b != b; // eslint-disable-line\n        };\n\n    // More reliable indexOf, courtesy of @WebReflection\n    var betterIndexOf = function(arr, value) {\n        if (value != value || value === 0) { // eslint-disable-line\n            for (var i = arr.length; i-- && !is(arr[i], value);) { // eslint-disable-line\n            }\n        } else {\n            i = [].indexOf.call(arr, value);\n        }\n        return i;\n    };\n\n    function Mappy() {\n        this.keys = [];\n        this.data = {};\n        this.values = [];\n    }\n\n    Mappy.prototype.set = function(key, value) {\n        var hashCode = hash(key);\n        if (this.data[hashCode] === undefined) {\n            this.keys.push(key);\n            this.values.push(value);\n        }\n        this.data[hashCode] = value;\n    };\n\n    Mappy.prototype.get = function(key) {\n        var hashCode = hash(key);\n        return this.data[hashCode];\n    };\n\n    Mappy.prototype.getIfAbsent = function(key, ifAbsentFunc) {\n        var value = this.get(key);\n        if (value === undefined) {\n            value = ifAbsentFunc(key, this);\n        }\n        return value;\n    };\n\n    Mappy.prototype.size = function() {\n        return this.keys.length;\n    };\n\n    Mappy.prototype.clear = function() {\n        this.keys.length = 0;\n        this.data = {};\n    };\n\n    Mappy.prototype.delete = function(key) {\n        var hashCode = hash(key);\n        if (this.data[hashCode] === undefined) {\n            return;\n        }\n        var index = betterIndexOf(this.keys, key);\n        this.keys.splice(index, 1);\n        this.values.splice(index, 1);\n        delete this.data[hashCode];\n    };\n\n    Mappy.prototype.forEach = function(func) {\n        var keys = this.keys;\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            func(value, key, this);\n        }\n    };\n\n    Mappy.prototype.map = function(func) {\n        var keys = this.keys;\n        var newMap = new Mappy();\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            var transformed = func(value, key, this);\n            newMap.set(key, transformed);\n        }\n        return newMap;\n    };\n\n    Mappy.prototype.copy = function() {\n        var keys = this.keys;\n        var newMap = new Mappy();\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            newMap.set(key, value);\n        }\n        return newMap;\n    };\n\n    return Mappy;\n\n})();\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\nvar _ = require('object-iterators');\nvar Base = require('extend-me').Base;\n\nvar images = require('../images');\n\n/** @typedef {object} CanvasRenderingContext2D\n * @see [CanvasRenderingContext2D](https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D)\n */\n\n/**\n * @constructor\n * @desc fin-hypergrid-renderer is the canvas enabled top level sub component that handles the renderering of the Grid.\n *\n * It relies on two other external subprojects\n *\n * 1. fin-canvas: a wrapper to provide a simpler interface to the HTML5 canvas component\n * 2. rectangular: a small npm module providing Point and Rectangle objects\n *\n * The fin-hypergrid-renderer is in a unique position to provide critical functionality to the fin-hypergrid in a hightly performant manner.\n * Because it MUST iterate over all the visible cells it can store various bits of information that can be encapsulated as a service for consumption by the fin-hypergrid component.\n *\n * Instances of this object have basically four main functions.\n *\n * 1. render fixed row headers\n * 2. render fixed col headers\n * 3. render main data cells\n * 4. render grid lines\n *\n * Same parameters as {@link Renderer#initialize|initialize}, which is called by this constructor.\n *\n */\nvar Renderer = Base.extend('Renderer', {\n\n    //the shared single item \"pooled\" cell object for drawing each cell\n    cell: {\n        x: 0,\n        y: 0,\n        width: 0,\n        height: 0\n    },\n\n    scrollHeight: 0,\n\n    viewHeight: 0,\n\n    reset: function() {\n        this.bounds = {\n            width: 0,\n            height: 0\n        };\n        this.columnEdges = [];\n        this.columnEdgesIndexMap = [];\n        this.renderedColumnMinWidths = [];\n        this.rowEdges = [];\n        this.rowEdgesIndexMap = [];\n        this.visibleColumns = [];\n        this.visibleRows = [];\n        this.insertionBounds = [];\n    },\n\n    /**\n     * @summary Constructor logic\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     * @memberOf Renderer.prototype\n     */\n    initialize: function(grid) {\n        this.grid = grid;\n        this.reset();\n    },\n\n    //this function computes the grid coordinates used for extremely fast iteration over\n    //painting the grid cells. this function is very fast, for thousand rows X 100 columns\n    //on a modest machine taking usually 0ms and no more that 3 ms.\n    computeCellsBounds: function() {\n\n        //var startTime = Date.now();\n\n        var grid = this.getGrid(),\n            scrollTop = this.getScrollTop(),\n            scrollLeft = this.getScrollLeft(),\n\n            numColumns = this.getColumnCount(),\n            numFixedColumns = this.getFixedColumnCount(),\n\n            numRows = this.getRowCount(),\n            numFixedRows = this.getFixedRowCount(),\n\n            behavior = this.getBehavior(),\n            bounds = this.getBounds(),\n            numberOfBottomTotalsRows = behavior.getDataModel().getBottomTotals().length,\n            viewWidth = bounds.width || grid.canvas.width, // if 0, we must be in bootstrap\n            viewHeight = bounds.height - numberOfBottomTotalsRows * behavior.getDefaultRowHeight(),\n\n            insertionBoundsCursor = 0,\n            previousInsertionBoundsCursorValue = 0,\n\n            start = 0,\n            x = 0, y = 0,\n            c, r,\n            vx, vy,\n            width, height,\n            firstVX, lastVX,\n            firstVY, lastVY;\n\n        this.getColumnEdges().length = 0;\n        this.rowEdges.length = 0;\n\n        this.columnEdges[0] = 0;\n        this.rowEdges[0] = 0;\n        this.scrollHeight = 0;\n\n        this.visibleColumns.length = 0;\n        this.visibleRows.length = 0;\n        this.columnEdgesIndexMap = [];\n        this.rowEdgesIndexMap = [];\n\n        this.insertionBounds = [];\n\n        if (grid.isShowRowNumbers()) {\n            start--;\n            this.columnEdges[-1] = -1;\n        }\n\n        for (c = start; c < numColumns; c++) {\n            vx = c;\n            if (c >= numFixedColumns) {\n                vx = vx + scrollLeft;\n                if (firstVX === undefined) {\n                    firstVX = vx;\n                }\n                lastVX = vx;\n            }\n            if (x > viewWidth || numColumns <= vx) {\n                break;\n            }\n            width = this.getColumnWidth(vx);\n            x = x + width;\n            this.columnEdges[c + 1] = Math.round(x);\n            this.visibleColumns[c] = vx;\n            this.columnEdgesIndexMap[vx] = c;\n\n            insertionBoundsCursor = insertionBoundsCursor + Math.round(width / 2) + previousInsertionBoundsCursorValue;\n            this.insertionBounds.push(insertionBoundsCursor);\n            previousInsertionBoundsCursorValue = Math.round(width / 2);\n        }\n\n        for (r = 0; r < numRows; r++) {\n            vy = r;\n            if (r >= numFixedRows) {\n                vy = vy + scrollTop;\n                if (firstVY === undefined) {\n                    firstVY = vy;\n                }\n                lastVY = vy;\n            }\n            if (y > viewHeight || numRows <= vy) {\n                break;\n            }\n            height = this.getRowHeight(vy);\n            y = y + height;\n            this.rowEdges[r + 1] = Math.round(y);\n            this.visibleRows[r] = vy;\n            this.rowEdgesIndexMap[vy] = r;\n        }\n        this.viewHeight = viewHeight;\n        this.dataWindow = grid.newRectangle(firstVX, firstVY, lastVX - firstVX, lastVY - firstVY);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {Object} a property value at a key, delegates to the grid\n     */\n    resolveProperty: function(key) {\n        return this.getGrid().resolveProperty(key);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {Hypergrid} grid\n     */\n    getGrid: function() {\n        return this.grid;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Notify the fin-hypergrid everytime we've repainted.\n     * @desc This is the entry point from fin-canvas.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    _paint: function(gc) {\n        if (this.grid) {\n            this.renderGrid(gc);\n            this.getGrid().gridRenderedNotification();\n        }\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Answer how many rows we rendered\n     */\n    getVisibleRowsCount: function() {\n        return this.visibleRows.length - 1;\n    },\n\n    getVisibleScrollHeight: function() {\n        var grid = this.getGrid(),\n            frh = grid.getFixedRowsHeight();\n\n        return this.viewHeight - frh;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number[]} Rows we just rendered.\n     */\n    getVisibleRows: function() {\n        return this.visibleRows;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Numer of columns we just rendered.\n     */\n    getVisibleColumnsCount: function() {\n        return this.visibleColumns.length - 1;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Columns we just rendered.\n     */\n    getVisibleColumns: function() {\n        return this.visibleColumns;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The column index whne the mouseEvent coordinates are over a column divider.\n     */\n    overColumnDivider: function(x) {\n        x = Math.round(x);\n        var edges = this.getColumnEdges();\n        var whichCol = edges.indexOf(x - 1);\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x - 2);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x + 1);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x - 3);\n        }\n\n        return whichCol;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row index when the mouseEvent coordinates are over a row divider.\n     */\n    overRowDivider: function(y) {\n        y = Math.round(y);\n        var which = this.rowEdges.indexOf(y + 1);\n        if (which < 0) {\n            which = this.rowEdges.indexOf(y);\n        }\n        if (which < 0) {\n            which = this.rowEdges.indexOf(y - 1);\n        }\n        return which;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {Point} cell\n     * @returns {Rectangle} Bounding rect of the given `cell`.\n     */\n    getBoundsOfCell: function(cell) {\n        return this._getBoundsOfCell(cell.x, cell.y);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {number} c - The horizontal coordinate.\n     * @param {number} r - The vertical coordinate.\n     * @returns {Rectangle} Bounding rect of cell with the given coordinates.\n     */\n    _getBoundsOfCell: function(c, r) {\n        var xOutside = false,\n            yOutside = false,\n            cell = this.cell;\n\n        var y, x = this.columnEdgesIndexMap[c];\n        if (x === undefined) {\n            x = this.columnEdgesIndexMap[c - 1];\n            xOutside = true;\n        }\n\n        var oy, ox = this.columnEdges[x],\n            cy, cx = this.columnEdges[x + 1],\n            ey, ex = cx - ox;\n\n        cell.x = xOutside ? cx : ox;\n        cell.width = xOutside ? 0 : ex;\n\n        if (r < 0) { // bottom totals rows\n            var grid = this.getGrid(),\n                behavior = grid.getBehavior(),\n                bounds = this.getBounds();\n\n            ey = behavior.getDefaultRowHeight();\n            oy = bounds.height + r * ey;\n            cy = oy + ey;\n        } else {\n            y = this.rowEdgesIndexMap[r];\n            if (y === undefined) {\n                y = this.rowEdgesIndexMap[r - 1];\n                yOutside = true;\n            }\n\n            oy = this.rowEdges[y];\n            cy = this.rowEdges[y + 1];\n            ey = cy - oy;\n        }\n\n        cell.y = yOutside ? cy : oy;\n        cell.height = yOutside ? 0 : ey;\n\n        return cell;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc answer the column index under the coordinate at pixelX\n     * @param {number} pixelX - The horizontal coordinate.\n     * @returns {number} The column index under the coordinate at pixelX.\n     */\n    getColumnFromPixelX: function(pixelX) {\n        var width = 0,\n            grid = this.getGrid(),\n            fixedColumnCount = this.getFixedColumnCount(),\n            scrollLeft = grid.getHScrollValue(),\n            edges = this.getColumnEdges();\n\n        for (var c = 1; c < edges.length - 1; c++) {\n            width = edges[c] - (edges[c] - edges[c - 1]) / 2;\n            if (pixelX < width) {\n                if (c > fixedColumnCount) {\n                    c = c + scrollLeft;\n                }\n                return c - 1;\n            }\n        }\n        if (c > fixedColumnCount) {\n            c = c + scrollLeft;\n        }\n        return c - 1;\n    },\n\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc Answer specific data cell coordinates given mouse coordinates in pixels.\n     * @param {Point} point\n     * @returns {Point} Cell coordinates\n     */\n    getGridCellFromMousePoint: function(point) {\n\n        var grid = this.getGrid();\n        var behavior = grid.getBehavior();\n        var width = 0;\n        var height = 0;\n        var x, y, c, r;\n        var previous = 0;\n        var columnEdges = this.getColumnEdges();\n        var fixedColumnCount = this.getFixedColumnCount(); // + gridSize;\n        var fixedRowCount = this.getFixedRowCount();\n\n        // var fixedColumnCount = this.getFixedColumnCount();\n        // var fixedRowCount = this.getFixedRowCount();\n        var scrollX = this.getScrollLeft();\n        var scrollY = this.getScrollTop();\n\n        for (c = 0; c < columnEdges.length; c++) {\n            width = columnEdges[c];\n            if (point.x < width) {\n                x = Math.max(0, point.x - previous - 2);\n                break;\n            }\n            previous = width;\n        }\n        c--;\n        previous = 0;\n        for (r = 0; r < this.rowEdges.length; r++) {\n            height = this.rowEdges[r];\n            if (point.y < height) {\n                y = Math.max(0, point.y - previous - 2);\n                break;\n            }\n            previous = height;\n        }\n        r--;\n        if (point.x < 0) {\n            c = -1;\n        }\n        if (point.y < 0) {\n            r = -1;\n        }\n\n        var viewPoint = grid.newPoint(c, r);\n\n        //compensate if we are scrolled\n        if (c >= fixedColumnCount) {\n            c = c + scrollX;\n        }\n        if (r >= fixedRowCount) {\n            r = r + scrollY;\n        }\n\n        var translatedIndex = -1;\n\n        var column = behavior.getColumn(c);\n        if (column) {\n            translatedIndex = column.index;\n        }\n\n        return {\n            gridCell: grid.newPoint(c, r),\n            mousePoint: grid.newPoint(x, y),\n            viewPoint: viewPoint,\n            dataCell: grid.newPoint(translatedIndex, r),\n        };\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines if a column is visible.\n     * @param {number} colIndex - the column index*\n     * @returns {boolean} The given column is fully visible.\n     */\n    isColumnVisible: function(colIndex) {\n        var isVisible = this.visibleColumns.indexOf(colIndex) !== -1;\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The width x coordinate of the last rendered column\n     */\n    getFinalVisableColumnBoundary: function() {\n        var isMaxX = this.isLastColumnVisible();\n        var chop = isMaxX ? 2 : 1;\n        var colWall = this.getColumnEdges()[this.getColumnEdges().length - chop];\n        var result = Math.min(colWall, this.getBounds().width - 200);\n        return result;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines visibility of a row.\n     * @param {number} rowIndex - the row index\n     * @returns {boolean} The given row is fully visible.\n     */\n    isRowVisible: function(rowIndex) {\n        var isVisible = this.visibleRows.indexOf(rowIndex) !== -1;\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines if a cell is selected.\n     * @param {number} x - the x cell coordinate\n     * @param {number} y - the y cell coordinate*\n     * @returns {boolean} The given cell is fully visible.\n     */\n    isSelected: function(x, y) {\n        return this.getGrid().isSelected(x, y);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc This is the main forking of the renderering task.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    renderGrid: function(gc) {\n        gc.beginPath();\n\n        this.paintCells(gc);\n        this.paintGridlines(gc);\n        this.renderOverrides(gc);\n        this.renderFocusCell(gc);\n        gc.closePath();\n    },\n\n    focusLineStep: [\n        [5, 5],\n        [0, 1, 5, 4],\n        [0, 2, 5, 3],\n        [0, 3, 5, 2],\n        [0, 4, 5, 1],\n        [0, 5, 5, 0],\n        [1, 5, 4, 0],\n        [2, 5, 3, 0],\n        [3, 5, 2, 0],\n        [4, 5, 1, 0]\n    ],\n\n    renderFocusCell: function(gc) {\n        gc.beginPath();\n        this._renderFocusCell(gc);\n        gc.closePath();\n    },\n\n    _renderFocusCell: function(gc) {\n\n        var grid = this.getGrid();\n        var selections = grid.getSelectionModel().getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var selection = selections[selections.length - 1];\n        var mouseDown = selection.origin;\n        if (mouseDown.x === -1) {\n            //no selected area, lets exit\n            return;\n        }\n\n        var visibleColumns = this.getVisibleColumns();\n        var visibleRows = this.getVisibleRows();\n        var lastVisibleColumn = visibleColumns[visibleColumns.length - 1];\n        var lastVisibleRow = visibleRows[visibleRows.length - 1];\n\n        var extent = selection.extent;\n\n        var dpOX = Math.min(mouseDown.x, mouseDown.x + extent.x);\n        var dpOY = Math.min(mouseDown.y, mouseDown.y + extent.y);\n\n        //lets check if our selection rectangle is scrolled outside of the visible area\n        if (dpOX > lastVisibleColumn) {\n            return; //the top of our rectangle is below visible\n        }\n        if (dpOY > lastVisibleRow) {\n            return; //the left of our rectangle is to the right of being visible\n        }\n\n        var dpEX = Math.max(mouseDown.x, mouseDown.x + extent.x) + 1;\n        dpEX = Math.min(dpEX, 1 + lastVisibleColumn);\n\n        var dpEY = Math.max(mouseDown.y, mouseDown.y + extent.y) + 1;\n        dpEY = Math.min(dpEY, 1 + lastVisibleRow);\n\n        var o = this._getBoundsOfCell(dpOX, dpOY);\n        var ox = Math.round((o.x === undefined) ? grid.getFixedColumnsWidth() : o.x);\n        var oy = Math.round((o.y === undefined) ? grid.getFixedRowsHeight() : o.y);\n        // var ow = o.width;\n        // var oh = o.height;\n        var e = this._getBoundsOfCell(dpEX, dpEY);\n        var ex = Math.round((e.x === undefined) ? grid.getFixedColumnsWidth() : e.x);\n        var ey = Math.round((e.y === undefined) ? grid.getFixedRowsHeight() : e.y);\n        // var ew = e.width;\n        // var eh = e.height;\n        var x = Math.min(ox, ex);\n        var y = Math.min(oy, ey);\n        var width = 1 + ex - ox;\n        var height = 1 + ey - oy;\n        if (x === ex) {\n            width = ox - ex;\n        }\n        if (y === ey) {\n            height = oy - ey;\n        }\n        if (width * height < 1) {\n            //if we are only a skinny line, don't render anything\n            return;\n        }\n\n        gc.rect(x, y, width, height);\n        gc.fillStyle = this.resolveProperty('selectionRegionOverlayColor');\n        gc.fill();\n        gc.lineWidth = 1;\n        gc.strokeStyle = this.resolveProperty('selectionRegionOutlineColor');\n\n        // animate the dashed line a bit here for fun\n\n        gc.stroke();\n\n        //gc.rect(x, y, width, height);\n\n        //gc.strokeStyle = 'white';\n\n        // animate the dashed line a bit here for fun\n        //gc.setLineDash(this.focusLineStep[Math.floor(10 * (Date.now() / 300 % 1)) % this.focusLineStep.length]);\n\n        //gc.stroke();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc iterate the renderering overrides and manifest each\n     * @param {CanvasRenderingContext2D} gc\n     */\n    renderOverrides: function(gc) {\n        var grid = this.getGrid();\n        var cache = grid.renderOverridesCache;\n        for (var key in cache) {\n            if (cache.hasOwnProperty(key)) {\n                var override = cache[key];\n                if (override) {\n                    this.renderOverride(gc, override);\n                }\n            }\n        }\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc copy each overrides specified area to it's target and blank out the source area\n     * @param {CanvasRenderingContext2D} gc\n     * @param {OverrideObject} override - an object with details contain an area and a target context\n     */\n    renderOverride: function(gc, override) {\n        //lets blank out the drag row\n        var hdpiRatio = override.hdpiratio;\n        //var edges = this.getColumnEdges();\n        var startX = override.startX; //hdpiRatio * edges[override.columnIndex];\n        var width = override.width + 1;\n        var height = override.height;\n        var targetCTX = override.ctx;\n        var imgData = gc.getImageData(startX, 0, Math.round(width * hdpiRatio), Math.round(height * hdpiRatio));\n        targetCTX.putImageData(imgData, 0, 0);\n        gc.fillStyle = this.resolveProperty('backgroundColor2');\n        gc.fillRect(Math.round(startX / hdpiRatio), 0, width, height);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} mouse is currently over cell x, y\n     * @param {number} offsetX - x coordinate\n     * @param {number} offsetY - y coordinate\n     */\n    isHovered: function(x, y) {\n        var grid = this.getGrid();\n        return grid.isHovered(x, y) && (grid.resolveProperty('hoverCellHighlight') === true);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} mouse is currently over row y\n     * @param {number} offsetY - y coordinate\n     */\n    isRowHovered: function(y) {\n        var grid = this.getGrid();\n        return grid.isRowHovered(y) && (grid.resolveProperty('hoverRowHighlight') === true);\n     },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} mouse is currently over column x\n     * @param {number} offsetX - x coordinate\n     */\n    isColumnHovered: function(x) {\n        var grid = this.getGrid();\n        return grid.isColumnHovered(x) && (grid.resolveProperty('hoverColumnHighlight') === true);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {number} colIndex\n     * @returns {boolean} The given column within the fixed row area is selected.\n     */\n    isCellSelectedInRow: function(colIndex) {\n        return this.getGrid().isCellSelectedInRow(colIndex);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {number} rowIndex\n     * @returns {boolean} The given row within the fixed column area is selected.\n     */\n    isCellSelectedInColumn: function(rowIndex) {\n        return this.getGrid().isCellSelectedInColumn(rowIndex);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Current vertical scroll value.\n     */\n    getScrollTop: function() {\n        var st = this.getGrid().getVScrollValue();\n        return st;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Current horizontal scroll value.\n     */\n    getScrollLeft: function() {\n        var st = this.getGrid().getHScrollValue();\n        return st;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {Behavior}\n     */\n    getBehavior: function() {\n        return this.getGrid().getBehavior();\n    },\n\n    getColumnEdges: function() {\n        return this.columnEdges;\n    },\n\n    getRowEdges: function() {\n        return this.rowEdges;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row height of the row at index rowIndex\n     * @param {number} rowIndex\n     */\n    getRowHeight: function(rowIndex) {\n        var height = this.getBehavior().getRowHeight(rowIndex);\n        return height;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The columnWidth of the column at index columnIndex\n     * @param {number} columnIndex\n     */\n    getColumnWidth: function(columnIndex) {\n        var width = this.getGrid().getColumnWidth(columnIndex);\n        return width;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} The last col was rendered (is visible)\n     */\n    isLastColumnVisible: function() {\n        var lastColumnIndex = this.getColumnCount() - 1;\n        var isMax = this.visibleColumns.indexOf(lastColumnIndex) !== -1;\n        return isMax;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The rendered column width at index\n     */\n    getRenderedWidth: function(index) {\n        return this.getColumnEdges()[index];\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The rendered row height at index\n     */\n    getRenderedHeight: function(index) {\n        return this.rowEdges[index];\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {fin-canvas} my [fin-canvas](https://github.com/stevewirts/fin-canvas)\n     */\n    getCanvas: function() {\n        return this.getGrid().getCanvas();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} User is currently dragging a column for reordering.\n     */\n    isDraggingColumn: function() {\n        return this.getGrid().isDraggingColumn();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row to goto for a page up.\n     */\n    getPageUpRow: function() {\n        var behavior = this.getBehavior();\n        var scrollHeight = this.getVisibleScrollHeight();\n        var headerRows = this.getGrid().getFixedRowCount();\n        var top = this.dataWindow.origin.y - headerRows;\n        var scanHeight = 0;\n        while (scanHeight < scrollHeight && top > -1) {\n            scanHeight = scanHeight + behavior.getRowHeight(top);\n            top--;\n        }\n        return top + 1;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row to goto for a page down.\n     */\n    getPageDownRow: function() {\n        var headerRows = this.getGrid().getFixedRowCount();\n        var rowNum = this.dataWindow.corner.y - headerRows - 1;\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of columns.\n     */\n    getColumnCount: function() {\n        return this.getGrid().getColumnCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of rows.\n     */\n    getRowCount: function() {\n        return this.getGrid().getRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        return this.getGrid().getFixedColumnCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        return this.getGrid().getFixedRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of header rows.\n     */\n    getHeaderRowCount: function() {\n        return this.getGrid().getHeaderRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of header columns.\n     */\n    getHeaderColumnCount: function() {\n        return this.getGrid().getHeaderColumnCount();\n    },\n\n    /** @summary Smart render the grid.\n     * @desc Paint all the cells of a grid, including all \"fixed\" columns and rows.\n     * We snapshot the context to insure against its pollution.\n     * `try...catch` surrounds each cell paint in case a cell editor throws an error.\n     * The error message is error-logged to console AND displayed in cell.\n     * @memberOf Renderer.prototype\n     * @param {CanvasRenderingContext2D} gc\n     */\n    paintCells: function(gc) {\n        var renderCellError,\n            message,\n            x, y,\n            c, r,\n\n            columnEdges = this.getColumnEdges(),\n            rowEdges = this.rowEdges,\n\n            visibleCols = this.getVisibleColumns(),\n            visibleRows = this.getVisibleRows(),\n\n            behavior = this.getBehavior(),\n\n            clipX = 0,\n            clipY = 0,\n            clipWidth,\n            clipHeight = this.getBounds().height,\n\n            loopStart = this.getGrid().isShowRowNumbers() ? -1 : 0,\n            loopLength = visibleCols.length; // regardless of loopStart, due to definition of .length\n\n        this.buttonCells = {};\n\n        if (loopLength) { // this if prevents painting just the fixed columns when there are no visible columns\n\n            // For each column...\n            for (x = loopStart; x < loopLength; x++, clipX += clipWidth) {\n\n                c = visibleCols[x];\n                this.renderedColumnMinWidths[c] = 0;\n                renderCellError = behavior.getColumnProperties(c).renderCellError;\n\n                gc.save();\n\n                // Clip to visible portion of column to prevent overflow to right. Previously we clipped to entire visible grid and dealt with overflow by overpainting with next column. However, this strategy fails when transparent background (no background color).\n                // TODO: if extra clip() calls per column affect performance (not the clipping itself which was happening anyway, but the clip calls which set up the clipping), use previous strategy when there is a background color\n                clipWidth = columnEdges[x - loopStart] - clipX;\n                gc.beginPath();\n                gc.rect(clipX, clipY, clipWidth, clipHeight);\n                gc.clip();\n\n                // For each row (of each column)...\n                for (y = 0; y < visibleRows.length; y++) {\n\n                    r = visibleRows[y];\n\n                    try {\n\n                        this._paintCell(gc, c, r);\n\n                        //if (r === 9 && c === 2) { throw Error('She sells sea shells by the sea shore.'); }\n\n                    } catch (e) {\n\n                        message = e && (e.message || e) || 'Unknown error.';\n\n                        console.error(message);\n\n                        if (renderCellError) {\n                            var rawGc = gc.gc || gc, // Don't log these canvas calls\n                                errY = rowEdges[y],\n                                errHeight = rowEdges[y + 1] - errY;\n\n                            rawGc.save(); // define clipping region\n                            rawGc.beginPath();\n                            rawGc.rect(clipX, errY, clipWidth, errHeight);\n                            rawGc.clip();\n\n                            renderCellError(rawGc, message, clipX, errY, clipWidth, errHeight);\n\n                            rawGc.restore(); // discard clipping region\n                        }\n\n                    }\n                }\n\n                // Bottom totals rows...\n                for (y = -behavior.getDataModel().getBottomTotals().length; y; y++) {\n                    this._paintCell(gc, c, y);\n                }\n\n                gc.restore(); // Remove column's clip region (and anything else renderCellError() might have set)\n            }\n        }\n\n        setNumberColumnWidth(gc, behavior, this.getGrid().getRowCount());\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc We opted to not paint borders for each cell as that was extremely expensive. Instead we draw gridlines here. Also we record the widths and heights for later.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    paintGridlines: function(gc) {\n        var x, y, c, r = 0;\n\n        var colWidths = this.getColumnEdges();\n        var rowHeights = this.rowEdges;\n\n        var viewWidth = colWidths[colWidths.length - 1];\n        var viewHeight = this.getBounds().height; //rowHeights[rowHeights.length - 1];\n\n        var drawThemH = this.resolveProperty('gridLinesH');\n        var drawThemV = this.resolveProperty('gridLinesV');\n        var lineColor = this.resolveProperty('lineColor');\n\n        gc.beginPath();\n\n        if (drawThemV) {\n            for (c = 0; c < colWidths.length + 1; c++) {\n                x = colWidths[c] + 0.5;\n                gc.moveTo(x, 0);\n                gc.lineTo(x, viewHeight);\n            }\n        }\n\n        if (drawThemH) {\n            for (r = 0; r < rowHeights.length - 1; r++) {\n                y = rowHeights[r] + 0.5;\n                gc.moveTo(0, y);\n                gc.lineTo(viewWidth, y);\n            }\n\n            // Bottom totals rows...\n            var behavior = this.getBehavior(),\n                rowHeight = behavior.getDefaultRowHeight();\n            for (r = -behavior.getDataModel().getBottomTotals().length, y = this.getBounds().height; r; r++) {\n                y -= rowHeight;\n                gc.moveTo(0, y);\n                gc.lineTo(viewWidth, y);\n            }\n        }\n\n        gc.closePath();\n\n        gc.strokeStyle = lineColor;\n        gc.lineWidth = this.resolveProperty('lineWidth');\n        gc.stroke();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {CanvasRenderingContext2D} gc\n     * @param x\n     * @param y\n     */\n    paintCell: function(gc, x, y) {\n        gc.moveTo(0, 0);\n\n        var c = this.getVisibleColumns()[x],\n            r = this.getVisibleRows()[y];\n\n        if (c) { //something is being viewed at at the moment (otherwise returns undefined)\n            this._paintCell(gc, c, r);\n        }\n    },\n\n    _paintCell: function(gc, c, r) {\n\n        var grid = this.getGrid(),\n            behavior = this.getBehavior(),\n            baseProperties = behavior.getColumnProperties(c);\n\n        if (baseProperties.isNull) {\n            return;\n        }\n\n        var columnProperties = baseProperties,\n            headerRowCount = behavior.getHeaderRowCount(),\n            isShowRowNumbers = grid.isShowRowNumbers(),\n            isHeaderRow = r >= 0 && r < headerRowCount,\n            isFooterRow = r < 0,\n            isFilterRow = grid.isFilterRow(r),\n            isHierarchyColumn = grid.isHierarchyColumn(c),\n            isRowSelected = grid.isRowSelected(r),\n            isColumnSelected = grid.isColumnSelected(c),\n            isCellSelected = grid.isCellSelected(c, r),\n            isCellSelectedInColumn = grid.isCellSelectedInColumn(c),\n            isCellSelectedInRow = grid.isCellSelectedInRow(r),\n            areAllRowsSelected = grid.areAllRowsSelected(),\n            cellProperties;\n\n        if ((isShowRowNumbers && c === -1) || isHierarchyColumn) {\n            if (isRowSelected) {\n                baseProperties = baseProperties.rowHeaderRowSelection;\n                cellProperties = Object.create(baseProperties);\n                cellProperties.isSelected = true;\n            } else {\n                baseProperties = baseProperties.rowHeader;\n                cellProperties = Object.create(baseProperties);\n                cellProperties.isSelected = isCellSelectedInRow;\n            }\n            cellProperties.isUserDataArea = false;\n        } else if (isHeaderRow || isFooterRow) {\n            if (isFilterRow) {\n                baseProperties = baseProperties.filterProperties;\n                cellProperties = Object.create(baseProperties);\n                cellProperties.isSelected = false;\n            } else if (isColumnSelected) {\n                baseProperties = baseProperties.columnHeaderColumnSelection;\n                cellProperties = Object.create(baseProperties);\n                cellProperties.isSelected = true;\n            } else {\n                baseProperties = baseProperties.columnHeader;\n                cellProperties = Object.create(baseProperties);\n                cellProperties.isSelected = isCellSelectedInColumn;\n            }\n            cellProperties.isUserDataArea = false;\n        } else if (isHierarchyColumn) {\n            baseProperties = baseProperties.rowHeader;\n            cellProperties = Object.create(baseProperties);\n            cellProperties.isSelected = isCellSelectedInRow;\n        } else {\n            cellProperties = Object.create(baseProperties);\n            cellProperties.isSelected = isCellSelected || isRowSelected || isColumnSelected;\n            cellProperties.isUserDataArea = true;\n        }\n\n        var rowNum = r - headerRowCount + 1;\n\n        if (c === -1) {\n            if (r === 0) { // header label row gets \"master\" checkbox\n                cellProperties.value = [images.checkbox(areAllRowsSelected), '', null];\n            } else if (isFilterRow) { // no checkbox but show filter icon\n                cellProperties.value = [images.filter(false), '', null];\n            } else if (isHeaderRow || isFooterRow) { // no checkbox on \"totals\" rows\n                cellProperties.value = '';\n            } else {\n                cellProperties.value = [images.checkbox(isRowSelected), rowNum, null];\n            }\n            cellProperties.halign = 'right';\n        } else {\n            cellProperties.value = grid.getValue(c, r);\n            cellProperties.halign = grid.getColumnAlignment(c);\n        }\n        cellProperties.isColumnHovered = this.isColumnHovered(c);\n        cellProperties.isRowHovered = this.isRowHovered(r);\n        cellProperties.isCellHovered = this.isHovered(c, r);\n        cellProperties.bounds = this._getBoundsOfCell(c, r);\n        cellProperties.isCellSelected = isCellSelected;\n        cellProperties.isRowSelected = isRowSelected;\n        cellProperties.isColumnSelected = isColumnSelected;\n        cellProperties.isInCurrentSelectionRectangle = grid.isInCurrentSelectionRectangle(c, r);\n\n        var mouseDownState = grid.mouseDownState;\n        if (mouseDownState) {\n            var point = mouseDownState.gridCell;\n            cellProperties.mouseDown = point.x === c && point.y === r;\n        }\n\n        cellProperties.x = c;\n        cellProperties.y = r;\n\n        behavior.cellPropertiesPrePaintNotification(cellProperties);\n\n        var cell = behavior.getCellRenderer(cellProperties, c, r);\n        var overrides = behavior.getCellProperties(c, r);\n\n        //declarative cell properties\n        _(cellProperties).extendOwn(overrides);\n\n        //allow the renderer to identify itself if it's a button\n        cellProperties.buttonCells = this.buttonCells;\n        var formatType = cellProperties.isUserDataArea ? cellProperties.format : 'default';\n        cellProperties.formatter = this.getGrid().getFormatter(formatType);\n        cell.paint(gc, cellProperties);\n\n        this.renderedColumnMinWidths[c] = Math.max(cellProperties.minWidth || 0, this.renderedColumnMinWidths[c]);\n        columnProperties.preferredWidth = this.renderedColumnMinWidths[c];\n    },\n\n    isViewableButton: function(c, r) {\n        var key = c + ',' + r;\n        return this.buttonCells[key] === true;\n    },\n\n    getRowNumbersWidth: function() {\n        var colEdges = this.getColumnEdges();\n        if (colEdges.length === 0) {\n            return 0;\n        }\n        return colEdges[0];\n    },\n\n    startAnimator: function() {\n        var animate;\n        var self = this;\n        animate = function() {\n            self.animate();\n            requestAnimationFrame(animate);\n        };\n        requestAnimationFrame(animate);\n    },\n\n    animate: function() {\n        var ctx = this.getCanvas().canvasCTX;\n        ctx.beginPath();\n        ctx.save();\n        this.renderFocusCell(ctx);\n        ctx.restore();\n        ctx.closePath();\n    },\n\n    getBounds: function() {\n        return this.bounds;\n    },\n\n    setBounds: function(bounds) {\n        return (this.bounds = bounds);\n    }\n\n});\n\nfunction setNumberColumnWidth(gc, behavior, maxRow) {\n    var columnProperties = behavior.getColumnProperties(-1),\n        cellProperties = columnProperties.rowHeader,\n        icon = images.checked;\n\n    gc.font = cellProperties.font;\n\n    columnProperties.preferredWidth = icon.width + 7 + cellProperties.getTextWidth(gc, maxRow + 1);\n}\n\nmodule.exports = Renderer;\n","'use strict';\n\nvar RangeSelectionModel = require('sparse-boolean-array');\n\n/**\n *\n * @constructor\n * @desc We represent selections as a list of rectangles because large areas can be represented and tested against quickly with a minimal amount of memory usage. Also we need to maintain the selection rectangles flattened counter parts so we can test for single dimension contains. This is how we know to highlight the fixed regions on the edges of the grid.\n */\n\nfunction SelectionModel() {\n\n    /**\n     * @name selections\n     * @type {Rectangle[]}\n     * @summary The selection rectangles.\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.selections = [];\n\n    /**\n     * @name flattenedX\n     * @type {Rectangle[]}\n     * @summary The selection rectangles flattened in the horizontal direction (no width).\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.flattenedX = [];\n\n    /**\n     * @name flattenedY\n     * @type {Rectangle[]}\n     * @summary The selection rectangles flattened in the vertical direction (no height).\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.flattenedY = [];\n\n    /**\n     * @name rowSelectionModel\n     * @type {RangeSelectionModel}\n     * @summary The selection rectangles.\n     * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.rowSelectionModel = new RangeSelectionModel();\n\n    /**\n     * @name columnSelectionModel\n     * @type {RangeSelectionModel}\n     * @summary The selection rectangles.\n     * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.columnSelectionModel = new RangeSelectionModel();\n\n    this.setLastSelectionType('');\n}\n\nSelectionModel.prototype = {\n\n    /**\n     * @type {boolean}\n     * @memberOf SelectionModel.prototype\n     */\n    allRowsSelected: false,\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @desc getter for the [fin-hypergrid](module-._fin-hypergrid.html)\n     * #### returns: fin-hypergrid\n     */\n    getGrid: function() {\n        return null;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getLastSelection: function() {\n        var sels = this.selections;\n        var sel = sels[sels.length - 1];\n        return sel;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getLastSelectionType: function() {\n        return this.lastSelectionType;\n    },\n\n    /**\n     * @param type\n     * @memberOf SelectionModel.prototype\n     */\n    setLastSelectionType: function(type) {\n        this.lastSelectionType = type;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @description Select the region described by the given coordinates.\n     *\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     */\n    select: function(ox, oy, ex, ey) {\n        var newSelection = this.getGrid().newRectangle(ox, oy, ex, ey);\n        this.selections.push(newSelection);\n        this.flattenedX.push(newSelection.flattenXAt(0));\n        this.flattenedY.push(newSelection.flattenYAt(0));\n        this.setLastSelectionType('cell');\n        this.getGrid().selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     */\n    toggleSelect: function(ox, oy, ex, ey) {\n\n        var selected, index;\n\n        selected = this.selections.find(function(selection, idx) {\n            index = idx;\n            return (\n                selection.origin.x === ox && selection.origin.y === oy &&\n                selection.extent.x === ex && selection.extent.y === ey\n            );\n        });\n\n        if (selected) {\n            this.selections.splice(index, 1);\n            this.flattenedX.splice(index, 1);\n            this.flattenedY.splice(index, 1);\n            this.getGrid().selectionChanged();\n        } else {\n            this.select(ox, oy, ex, ey);\n        }\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @desc Remove the last selection that was created.\n     */\n    clearMostRecentSelection: function(dontClearRows) {\n        dontClearRows = dontClearRows === true;\n        if (!dontClearRows) {\n            this.setAllRowsSelected(false);\n        }\n        this.selections.length = Math.max(0, this.selections.length - 1);\n        this.flattenedX.length = Math.max(0, this.flattenedX.length - 1);\n        this.flattenedY.length = Math.max(0, this.flattenedY.length - 1);\n        //this.getGrid().selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearMostRecentColumnSelection: function() {\n        this.columnSelectionModel.clearMostRecentSelection();\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearMostRecentRowSelection: function() {\n        this.rowSelectionModel.clearMostRecentSelection();\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearRowSelection: function() {\n        this.rowSelectionModel.clear();\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getSelections: function() {\n        return this.selections;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean} There are active selection(s).\n     */\n    hasSelections: function() {\n        return this.selections.length !== 0;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n    hasRowSelections: function() {\n        return !this.rowSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n    hasColumnSelections: function() {\n        return !this.columnSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @return {boolean} Selection covers a specific column.\n     * @param {number} y\n     */\n    isCellSelectedInRow: function(y) {\n        return this._isCellSelected(this.flattenedX, 0, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns Selection covers a specific row.\n     * @param {number} x\n     */\n    isCellSelectedInColumn: function(x) {\n        return this._isCellSelected(this.flattenedY, x, 0);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @summary Selection query function.\n     * @returns {boolean} The given cell is selected (part of an active selection).\n     * @param {Rectangle[]} selections - Selection rectangles to search through.\n     * @param {number} x\n     * @param {number} y\n     */\n    isSelected: function(x, y) {\n        return (\n            this.isColumnSelected(x) ||\n            this.isRowSelected(y) ||\n            this._isCellSelected(this.selections, x, y)\n        );\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @param y\n     * @returns {*}\n     */\n    isCellSelected: function(x, y) {\n        return this._isCellSelected(this.selections, x, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param selections\n     * @param x\n     * @param y\n     * @returns {boolean}\n     * @private\n     */\n    _isCellSelected: function(selections, x, y) {\n        var self = this;\n        return !!selections.find(function(selection) {\n            return self.rectangleContains(selection, x, y);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @desc empty out all our state\n     *\n     */\n    clear: function(dontClearRowSelections) {\n        dontClearRowSelections = dontClearRowSelections === true;\n        this.selections.length = 0;\n        this.flattenedX.length = 0;\n        this.flattenedY.length = 0;\n        this.columnSelectionModel.clear();\n        if (!dontClearRowSelections) {\n            this.setAllRowsSelected(false);\n            this.rowSelectionModel.clear();\n        }\n        //this.getGrid().selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     * @returns {boolean}\n     */\n    isRectangleSelected: function(ox, oy, ex, ey) {\n        return !!this.selections.find(function(selection) {\n            return (\n                selection.origin.x === ox && selection.origin.y === oy &&\n                selection.extent.x === ex && selection.extent.y === ey\n            );\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @returns {*}\n     */\n    isColumnSelected: function(x) {\n        return this.columnSelectionModel.isSelected(x);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y\n     * @returns {boolean|*}\n     */\n    isRowSelected: function(y) {\n        return this.allRowsSelected || this.rowSelectionModel.isSelected(y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x1\n     * @param x2\n     */\n    selectColumn: function(x1, x2) {\n        this.columnSelectionModel.select(x1, x2);\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    selectAllRows: function() {\n        this.clear();\n        this.setAllRowsSelected(true);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n\n    setAllRowsSelected: function(isIt) {\n        this.allRowsSelected = isIt;\n    },\n\n    areAllRowsSelected: function() {\n        return this.allRowsSelected;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y1\n     * @param y2\n     */\n    selectRow: function(y1, y2) {\n        this.rowSelectionModel.select(y1, y2);\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x1\n     * @param x2\n     */\n    deselectColumn: function(x1, x2) {\n        this.columnSelectionModel.deselect(x1, x2);\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y1\n     * @param y2\n     */\n    deselectRow: function(y1, y2) {\n        this.rowSelectionModel.deselect(y1, y2);\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getSelectedRows: function() {\n        if (this.areAllRowsSelected()) {\n            var grid = this.getGrid();\n            var headerRows = grid.getHeaderRowCount();\n            var rowCount = grid.getRowCount() - headerRows;\n            var result = new Array(rowCount);\n            for (var i = 0; i < rowCount; i++) {\n                result[i] = i + headerRows;\n            }\n            return result;\n        }\n        return this.rowSelectionModel.getSelections();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*|Array.Array.number}\n     */\n    getSelectedColumns: function() {\n        return this.columnSelectionModel.getSelections();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n     isColumnOrRowSelected: function() {\n        return !this.columnSelectionModel.isEmpty() || !this.rowSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {Array}\n     */\n    getFlattenedYs: function() {\n        var result = [];\n        var set = {};\n        this.selections.forEach(function(selection) {\n            var top = selection.origin.y;\n            var size = selection.extent.y + 1;\n            for (var r = 0; r < size; r++) {\n                var ti = r + top;\n                if (!set[ti]) {\n                    result.push(ti);\n                    set[ti] = true;\n                }\n            }\n        });\n        result.sort(function(x, y) {\n            return x - y;\n        });\n        return result;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param offset\n     */\n    selectRowsFromCells: function(offset) {\n        offset = offset || 0;\n\n        var sm = this.rowSelectionModel;\n        this.setAllRowsSelected(false);\n        sm.clear();\n\n        this.selections.forEach(function(selection) {\n            var top = selection.origin.y,\n                size = selection.extent.y;\n            sm.select(top + offset, top + size + offset);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param offset\n     */\n    selectColumnsFromCells: function(offset) {\n        offset = offset || 0;\n\n        var sm = this.columnSelectionModel;\n        sm.clear();\n\n        this.selections.forEach(function(selection) {\n            var left = selection.origin.x;\n            var size = selection.extent.x;\n            sm.select(left + offset, left + size + offset);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @param y\n     * @returns {*}\n     */\n    isInCurrentSelectionRectangle: function(x, y) {\n        var last = this.selections[this.selections.length - 1];\n        return last && this.rectangleContains(last, x, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param rect\n     * @param x\n     * @param y\n     * @returns {boolean}\n     */\n    rectangleContains: function(rect, x, y) { //TODO: explore why this works and contains on rectanglular does not\n        var minX = rect.origin.x;\n        var minY = rect.origin.y;\n        var maxX = minX + rect.extent.x;\n        var maxY = minY + rect.extent.y;\n\n        if (rect.extent.x < 0) {\n            minX = maxX;\n            maxX = rect.origin.x;\n        }\n\n        if (rect.extent.y < 0) {\n            minY = maxY;\n            maxY = rect.origin.y;\n        }\n\n        var result =\n            x >= minX &&\n            y >= minY &&\n            x <= maxX &&\n            y <= maxY;\n\n        return result;\n    }\n};\n\nmodule.exports = SelectionModel;\n","/* eslint-env browser */\n\n'use strict';\n\nvar Base = require('extend-me').Base;\n\n\nvar ANIMATION_TIME = 500,\n    TRANSITION = ANIMATION_TIME + 'ms ease-in';\n\n/** @constructor\n * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n *\n * See {@link TableDialog#initialize|initialize} which is called by the constructor.\n */\nvar TableDialog = Base.extend('TableDialog', {\n\n    initialize: function(grid) {\n        this.grid = grid;\n        this.initializeOverlaySurface();\n        this.openNow = false;\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc returns true if the overlay is open\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     */\n    isOpen: function() {\n        return this.openNow;\n    },\n\n    open: function() {\n        if (this.isOpen()) {\n            return;\n        }\n\n        this.openNow = true;\n        var self = this;\n        this.overlay.style.backgroundColor = this.grid.resolveProperty('backgroundColor');\n\n        this.overlay.style.top = this.overlay.style.bottom = this.overlay.style.right = this.overlay.style.left = 0;\n\n        self.overlay.style.webkitTransition = '';\n\n        this.overlay.style.margin = '15px 35px 35px 15px';\n        this.overlay.style.opacity = 0;\n        this.overlay.style.zIndex = 100;\n\n        this.closeTransition = function() {\n            this.overlay.style.opacity = 0;\n        };\n\n        if (!this._closer) {\n            this._closer = function(e) {\n                var key = self.getCharFor(e.keyCode).toLowerCase();\n                var keys = self.grid.resolveProperty('editorActivationKeys');\n                if (keys.indexOf(key) > -1 || e.keyCode === 27) {\n                    e.preventDefault();\n                    self.close();\n                }\n            };\n        }\n        requestAnimationFrame(function() {\n            self.overlay.style.webkitTransition = 'opacity ' + ANIMATION_TIME + 'ms ease-in';\n            requestAnimationFrame(function() {\n                document.addEventListener('keydown', self._closer, false);\n                self.overlay.style.opacity = 0.95;\n            });\n        });\n\n        setTimeout(function() {\n            self.overlay.focus();\n        }, 100);\n    },\n    /**\n     * @memberOf Overlay.prototype\n     * @desc open the overlay\n     * #### returns: type\n     * @param {Hypergrid} grid\n     */\n    openFrom: function(rect) {\n        if (this.isOpen()) {\n            return;\n        }\n        this.openNow = true;\n        var self = this;\n        var style = this.overlay.style;\n        style.backgroundColor = this.grid.resolveProperty('backgroundColor');\n\n        var bounds = this.grid.div.getBoundingClientRect(),\n            margins = rect.y + 'px ' +\n                (bounds.width - (rect.x + rect.width)) + 'px ' +\n                (bounds.height - (rect.y + rect.height)) + 'px ' +\n                rect.x + 'px';\n\n        style.webkitTransition = '';\n\n        style.top = style.right = style.bottom = style.left = 0;\n\n        style.margin = margins;\n        style.zIndex = 100;\n        style.opacity = 1;\n\n        this.closeTransition = function() {\n            style.margin = margins;\n        };\n\n        if (!this._closer) {\n            this._closer = function(e) {\n                var key = self.getCharFor(e.keyCode).toLowerCase();\n                var keys = self.grid.resolveProperty('editorActivationKeys');\n                if (keys.indexOf(key) > -1 || e.keyCode === 27) {\n                    e.preventDefault();\n                    self.close();\n                }\n            };\n        }\n\n        //grid.setFocusable(false);\n        requestAnimationFrame(function() {\n            document.addEventListener('keydown', self._closer, false);\n            requestAnimationFrame(function() {\n                requestAnimationFrame(function() {\n                    style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION;\n                    style.margin = '15px 35px 35px 15px';\n                });\n            });\n        });\n        setTimeout(function() {\n            self.overlay.focus();\n        }, 100);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc close the overlay\n     * @param {Hypergrid} grid\n     */\n    close: function() {\n        //grid.setFocusable(true);\n        this.openNow = false;\n        document.removeEventListener('keydown', this._closer, false);\n\n        var self = this;\n\n        requestAnimationFrame(function() {\n            self.closeTransition();\n        });\n\n        setTimeout(function() {\n            self.clear();\n            self.overlay.style.zIndex = -1000;\n            if (self.onClose) {\n                self.onClose();\n                self.onClose = undefined;\n            }\n            self.grid.takeFocus();\n        }, ANIMATION_TIME);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc initialize the overlay surface into the grid\n     * #### returns: type\n     * @param {Hypergrid} grid\n     */\n    initializeOverlaySurface: function() {\n        this.overlay = document.createElement('div');\n        this.overlay.setAttribute('tabindex', 0);\n        this.overlay.addEventListener('wheel', function(evt) { evt.stopPropagation(); });\n\n        var style = this.overlay.style;\n        style.outline = 'none';\n        style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)';\n        style.position = 'absolute';\n\n        style.margin = 0;\n        style.overflow = 'hidden';\n\n        //style.display = 'none';\n\n        //style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION;\n\n        style.opacity = 0;\n        style.zIndex = 10;\n        this.grid.div.appendChild(this.overlay);\n        //document.body.appendChild(this.overlay);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc get a human readable description of the key pressed from it's integer representation\n     * @returns {string}\n     * @param {Hypergrid} grid\n     * @param {number} integer - the integer we want the char for\n     */\n    getCharFor: function(integer) {\n        var charMap = this.grid.getCanvas().getCharMap();\n        return charMap[integer][0];\n    },\n\n    clear: function() {\n        this.overlay.innerHTML = '';\n    },\n\n    querySelector: function(selector) {\n        var elements = this.overlay.querySelector(selector);\n        return elements;\n    },\n\n    getAnimationTime: function() {\n        return ANIMATION_TIME;\n    }\n});\n\nmodule.exports = TableDialog;\n","/* eslint-env browser */\n\n'use strict';\n\nvar _ = require('object-iterators');\nvar Base = require('extend-me').Base;\n\nvar Column = require('./Column');\nvar CellProvider = require('../CellProvider');\n\nvar noExportProperties = [\n    'columnHeader',\n    'columnHeaderColumnSelection',\n    'filterProperties',\n    'rowHeader',\n    'rowHeaderRowSelection',\n    'rowNumbersProperties',\n    'treeColumnProperties',\n    'treeColumnPropertiesColumnSelection',\n];\n\n/**\n * @constructor\n * @desc This is the base class for creating behaviors.  a behavior can be thought of as a model++.\nit contains all code/data that's necessary for easily implementing a virtual data source and it's manipulation/analytics\n */\nvar Behavior = Base.extend('Behavior', {\n\n    /**\n     * @desc this is the callback for the plugin pattern of nested tags\n     * @param {Hypergrid} grid\n     * @memberOf Behavior.prototype\n     */\n    initialize: function(grid) { //formerly installOn\n        grid.setBehavior(this);\n        this.initializeFeatureChain(grid);\n\n        this.getDataModel();\n        this.cellProvider = this.createCellProvider();\n        this.renderedColumnCount = 30;\n        this.renderedRowCount = 60;\n        this.dataUpdates = {}; //for overriding with edit values;\n    },\n\n    /**\n     * @desc create the feature chain - this is the [chain of responsibility](http://c2.com/cgi/wiki?ChainOfResponsibilityPattern) pattern.\n     * @param {Hypergrid} grid\n     * @memberOf Behavior.prototype\n     */\n    initializeFeatureChain: function(grid) {\n        var self = this;\n        this.features.forEach(function(FeatureConstructor) {\n            self.setNextFeature(new FeatureConstructor);\n        });\n\n        this.featureChain.initializeOn(grid);\n    },\n\n    features: [], // in case implementing class has no features TODO: Will this ever happen?\n\n    /**\n     * memento for the user configured visual properties of the table\n     * @type {object}\n     * @memberOf Behavior.prototype\n     */\n    tableState: null,\n\n    /**\n     * @type {Hypergrid}\n     * @memberOf Behavior.prototype\n     */\n    grid: null,\n\n    /**\n     * list of default cell editor names\n     * @type {string[]}\n     * @memberOf Behavior.prototype\n     */\n    editorTypes: ['choice', 'textfield', 'color', 'slider', 'spinner', 'date'],\n\n    /**\n     * controller chain of command\n     * @type {object}\n     * @memberOf Behavior.prototype\n     */\n    featureChain: null,\n\n    dataModel: null,\n    baseModel: null,\n\n    scrollPositionX: 0,\n    scrollPositionY: 0,\n\n    featureMap: {},\n    allColumns: [],\n    columns: [],\n\n    reset: function() {\n\n        this.cellProvider = this.createCellProvider();\n        this.renderedColumnCount = 30;\n        this.renderedRowCount = 60;\n        this.dataUpdates = {}; //for overriding with edit values;\n        this.clearColumns();\n        this.clearState();\n        this.getDataModel().reset();\n        this.createColumns();\n    },\n\n    clearColumns: function() {\n        this.columns = [];\n        this.allColumns = [];\n        this.columns[-1] = this.newColumn(-1, '');\n        this.columns[-2] = this.newColumn(-2, 'Tree');\n        this.allColumns[-1] = this.columns[-1];\n        this.allColumns[-2] = this.columns[-2];\n    },\n\n    getColumn: function(x) {\n        return this.columns[x];\n    },\n\n    getColumnId: function(x) {\n        return this.getColumn(x).label;\n    },\n\n    newColumn: function(index, label) {\n        var properties = this.createColumnProperties();\n        this.getPrivateState().columnProperties[index] = properties;\n        return new Column(this, index, label);\n    },\n\n    addColumn: function(index, label) {\n        var column = this.newColumn(index, label);\n        this.columns.push(column);\n        this.allColumns.push(column);\n        return column;\n    },\n\n    createColumns: function() {\n        //concrete implementation here\n    },\n\n    createColumnProperties: function() {\n        var tableState = this.getPrivateState();\n        var properties = Object.create(tableState);\n\n        properties.rowNumbersProperties = Object.create(properties, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.rowHeader = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderFont;\n                },\n                set: function(value) {\n                    this.rowHeaderFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderColor;\n                },\n                set: function(value) {\n                    this.rowHeaderColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.columnHeader = Object.create(properties, {\n            format: {\n                value: 'default'\n            },\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderFont;\n                },\n                set: function(value) {\n                    this.columnHeaderFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderColor;\n                },\n                set: function(value) {\n                    this.columnHeaderColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.columnHeaderColumnSelection = Object.create(properties.columnHeader, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundColumnSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundColumnSelectionColor = value;\n                }\n            }\n        });\n\n        properties.rowHeaderRowSelection = Object.create(properties.rowHeader, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderForegroundRowSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderForegroundRowSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundRowSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundRowSelectionColor = value;\n                }\n            }\n        });\n\n        properties.filterProperties = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.filterFont;\n                },\n                set: function(value) {\n                    this.filterFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.filterColor;\n                },\n                set: function(value) {\n                    this.filterColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterBackgroundColor;\n                },\n                set: function(value) {\n                    this.filterBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.filterForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.filterBackgroundSelectionColor = value;\n                }\n            },\n            cellBorderStyle: {\n                configurable: true,\n                get: function() {\n                    return this.filterCellBorderStyle;\n                },\n                set: function(value) {\n                    this.filterCellBorderStyle = value;\n                }\n            },\n            cellBorderThickness: {\n                configurable: true,\n                get: function() {\n                    return this.filterCellBorderThickness;\n                },\n                set: function(value) {\n                    this.filterCellBorderThickness = value;\n                }\n            }\n        });\n\n        properties.treeColumnProperties = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnFont;\n                },\n                set: function(value) {\n                    this.treeColumnFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnColor;\n                },\n                set: function(value) {\n                    this.treeColumnColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.treeColumnPropertiesColumnSelection = Object.create(properties.treeColumnProperties, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnForegroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnForegroundColumnSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundColumnSelectionColor = value;\n                }\n            }\n        });\n\n        return properties;\n    },\n\n    getColumnWidth: function(x) {\n        var col = this.getColumn(x);\n        if (!col) {\n            return this.resolveProperty('defaultColumnWidth');\n        }\n        var width = col.getWidth();\n        return width;\n    },\n\n    setColumnWidth: function(x, width) {\n        this.getColumn(x).setWidth(width);\n        this.stateChanged();\n    },\n\n    getDataModel: function() {\n        if (this.dataModel === null) {\n            var dataModel = this.getDefaultDataModel();\n            this.setDataModel(dataModel);\n        }\n        return this.dataModel;\n    },\n\n    getCellRenderer: function(config, x, y) {\n        return this.getColumn(x).getCellRenderer(config, y);\n    },\n\n    setDataModel: function(newDataModel) {\n        this.dataModel = newDataModel;\n    },\n\n    setComplexFilter: function(columnIndex, complexFilter) {\n        var col = this.getColumn(columnIndex);\n        if (col) {\n            col.setComplexFilter(complexFilter);\n        }\n    },\n\n    getComplexFilter: function(columnIndex) {\n        var col = this.getColumn(columnIndex);\n        if (col) {\n            return col.getComplexFilter();\n        }\n        return;\n    },\n\n    applyFilters: function() {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc utility function to empty an object of its members\n     * @param {object} obj - the object to empty\n     * @param {boolean} [exportProps]\n     * * `undefined` (omitted) - delete *all* properties\n     * * **falsy** - delete *only* the export properties\n     * * **truthy** - delete all properties *except* the export properties\n     */\n    clearObjectProperties: function(obj, exportProps) {\n        for (var key in obj) {\n            if (\n                obj.hasOwnProperty(key) && (\n                    exportProps === undefined ||\n                    !exportProps && noExportProperties.indexOf(key) >= 0 ||\n                    exportProps && noExportProperties.indexOf(key) < 0\n                )\n            ) {\n                delete obj[key];\n            }\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc getter for a [Memento](http://c2.com/cgi/wiki?MementoPattern) Object\n     * @returns {object}\n     */\n    getPrivateState: function() {\n        if (!this.tableState) {\n            this.tableState = this.getDefaultState();\n        }\n        return this.tableState;\n    },\n\n    //this is effectively a clone, with certain things removed....\n    getState: function() {\n        var copy = JSON.parse(JSON.stringify(this.getPrivateState()));\n        this.clearObjectProperties(copy.columnProperties, false);\n        return copy;\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc clear all table state\n     */\n    clearState: function() {\n        this.tableState = null;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {object} Newly created default empty tablestate.\n     */\n    getDefaultState: function() {\n        var tableProperties = this.getGrid()._getProperties();\n        var state = Object.create(tableProperties);\n\n        _(state).extendOwn({\n            rowHeights: {},\n            cellProperties: {},\n            columnProperties: []\n        });\n\n        return state;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Restore this table to a previous state.\n     * See the [memento pattern](http://c2.com/cgi/wiki?MementoPattern).\n     * @param {Object} memento - an encapsulated representation of table state\n     */\n    setState: function(memento) {\n\n        //we don't want to clobber the column properties completely\n        if (!memento.columnIndexes) {\n            var fields = this.getFields();\n            memento.columnIndexes = [];\n            for (var i = 0; i < fields.length; i++) {\n                memento.columnIndexes[i] = i;\n            }\n        }\n        var colProperties = memento.columnProperties;\n        delete memento.columnProperties;\n        this.tableState = null;\n        var state = this.getPrivateState();\n        this.createColumns();\n        this.setColumnOrder(memento.columnIndexes);\n        _(state).extendOwn(memento);\n        this.setAllColumnProperties(colProperties);\n        memento.columnProperties = colProperties;\n        //memento.columnProperties = colProperties;\n\n        // this.getDataModel().setState(memento);\n        // var self = this;\n        // requestAnimationFrame(function() {\n        //     self.applySorts();\n        //     self.changed();\n        //     self.stateChanged();\n        // });\n\n        //just to be close/ it's easier on the eyes\n        this.setColumnWidth(-1, 24.193359375);\n        this.getDataModel().applyState();\n    },\n\n    setAllColumnProperties: function(properties) {\n        properties = properties || [];\n        for (var i = 0; i < properties.length; i++) {\n            var current = this.getPrivateState().columnProperties[i];\n            this.clearObjectProperties(current, false);\n            _(current).extendOwn(properties[i]);\n        }\n    },\n\n    setColumnOrder: function(indexes) {\n        if (!indexes) {\n            this.columns.length = 0;\n            return;\n        }\n        this.columns.length = indexes.length;\n        for (var i = 0; i < indexes.length; i++) {\n            this.columns[i] = this.allColumns[indexes[i]];\n        }\n    },\n\n    applySorts: function() {\n        //if I have sorts, apply them now//\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc fetch the value for a property key\n     * @returns {*} The value of the given property.\n     * @param {string} key - a property name\n     */\n    resolveProperty: function(key) {\n        return this.grid.resolveProperty(key);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc A specific cell was clicked; you've been notified.\n     * @param {Point} cell - point of cell coordinates\n     * @param {Object} event - all event information\n     */\n    cellClicked: function(cell, event) {\n        this.getDataModel().cellClicked(cell, event);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc A specific cell was le double-clicked; you've been notified.\n     * @param {Point} cell - point of cell coordinates\n     * @param {Object} event - all event information\n     */\n    cellDoubleClicked: function(cell, event) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc add nextFeature to me If I don't have a next node, otherwise pass it along\n     * @param {Feature}\n     */\n    setNextFeature: function(nextFeature) {\n        this.featureMap[nextFeature.alias] = nextFeature;\n        if (this.featureChain) {\n            this.featureChain.setNext(nextFeature);\n        } else {\n            this.featureChain = nextFeature;\n        }\n    },\n\n    lookupFeature: function(key) {\n        return this.featureMap[key];\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc getter for the cell provider\n     * @return {CellProvider}\n     */\n    getCellProvider: function() {\n        return this.cellProvider;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc setter for the hypergrid\n     * @param {Hypergrid} grid\n     */\n    setGrid: function(finGrid) {\n        this.grid = finGrid;\n        this.getDataModel().setGrid(finGrid);\n        this.clearColumns();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @returns: {Hypergrid} The hypergrid to which this behavior is attached.\n     * @param {type} varname - descripton\n     */\n    getGrid: function() {\n        return this.grid;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc You can override this function and substitute your own cell provider.\n     * @return {CellProvider}\n     */\n    createCellProvider: function() {\n        return new CellProvider();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc First check to see if something was overridden.\n     * @return {*} The value at `x,y` for the top left section of the hypergrid.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     */\n    getValue: function(x, y) {\n        var column = this.getColumn(x);\n        if (!column) {\n            return undefined;\n        }\n        return column.getValue(y);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        var column = this.getColumn(x);\n        if (!column) {\n            return undefined;\n        }\n        return column.getUnfilteredValue(y);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc update the data at point x, y with value\n     * @return The data.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     * @param {Object} value - the value to use\n     */\n    setValue: function(x, y, value) {\n        var column = this.getColumn(x);\n        if (!column) {\n            return;\n        }\n        return column.setValue(y, value);\n    },\n\n    getDataValue: function(x, y) {\n        return this.getDataModel().getValue(x, y);\n    },\n\n    setDataValue: function(x, y, value) {\n        this.getDataModel().setValue(x, y, value);\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc First checks to see if something was overridden.\n     * @return {*} The value at x,y for the top left section of the hypergrid.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     */\n    getCellProperties: function(x, y) {\n        var col = this.getColumn(x);\n        return col.getCellProperties(y);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc update the data at point x, y with value\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     * @param {Object} value - the value to use\n     */\n    setCellProperties: function(x, y, value) {\n        var col = this.getColumn(x);\n        if (col) {\n            col.setCellProperties(y, value);\n        }\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of rows in the hypergrid.\n     */\n    getRowCount: function() {\n        return this.getDataModel().getRowCount();\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.getDataModel().getUnfilteredRowCount();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The height in pixels of the fixed rows area  of the hypergrid.\n     */\n    getFixedRowsHeight: function() {\n        var count = this.getFixedRowCount();\n        var total = 0;\n        for (var i = 0; i < count; i++) {\n            total = total + this.getRowHeight(i);\n        }\n        //var footerHeight = this.getDefaultRowHeight();\n        //total = total + (footerHeight * this.getFooterRowCount());\n        return total;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The height in pixels of a specific row in the hypergrid.\n     * @param {number} rowNum - row index of interest\n     */\n    getRowHeight: function(rowNum) {\n        var tableState = this.getPrivateState();\n        if (tableState.rowHeights) {\n            var override = tableState.rowHeights[rowNum];\n            if (override) {\n                return override;\n            }\n        }\n        return this.getDefaultRowHeight();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The value is lazily initialized and comes from the properties mechanism for '`defaultRowHeight`', which should be ~20px.\n     * @returns {number} The row height in pixels.\n     */\n    getDefaultRowHeight: function() {\n        if (!this.defaultRowHeight) {\n            this.defaultRowHeight = this.resolveProperty('defaultRowHeight');\n        }\n        return this.defaultRowHeight;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc set the pixel height of a specific row\n     * @param {number} rowNum - the row index of interest\n     * @param {number} height - pixel height\n     */\n    setRowHeight: function(rowNum, height) {\n        var tableState = this.getPrivateState();\n        tableState.rowHeights[rowNum] = Math.max(5, height);\n        this.stateChanged();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc This will allow 'floating' fixed rows.\n     * @return {number} The maximum height of the fixed rows area in the hypergrid.\n     */\n    getFixedRowsMaxHeight: function() {\n        return this.getFixedRowsHeight();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The width of the fixed column area in the hypergrid.\n     */\n    getFixedColumnsWidth: function() {\n        var count = this.getFixedColumnCount();\n        var total = 0;\n        if (this.getGrid().isShowRowNumbers()) {\n            total = this.getColumnWidth(-1);\n        }\n        for (var i = 0; i < count; i++) {\n            total = total + this.getColumnWidth(i);\n        }\n        return total;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc This exists to support \"floating\" columns.\n     * @return {number} The total width of the fixed columns area.\n     */\n    getFixedColumnsMaxWidth: function() {\n        var width = this.getFixedColumnsWidth();\n        return width;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the scroll position in vertical dimension and notify listeners.\n     * @param {number} y - the new y value\n     */\n    _setScrollPositionY: function(y) {\n        this.setScrollPositionY(y);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the scroll position in horizontal dimension and notify listeners.\n     * @param {number} x - the new x value\n     */\n    _setScrollPositionX: function(x) {\n        this.setScrollPositionX(x);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of columns just rendered, including partially rendered columns.\n     * @param {number} count - how many columns were just rendered\n     */\n    setRenderedColumnCount: function(count) {\n        this.renderedColumnCount = count;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of rows just rendered, including partially rendered rows.\n     * @param {number} count - how many rows were just rendered\n     */\n    setRenderedRowCount: function(count) {\n        this.renderedRowCount = count;\n    },\n\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The fixed row area has been clicked, massage the details and call the real function.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - event details\n     */\n    _fixedRowClicked: function(grid, mouse) {\n        var x = this.translateColumnIndex(this.getScrollPositionX() + mouse.gridCell.x - this.getFixedColumnCount());\n        var translatedPoint = this.grid.newPoint(x, mouse.gridCell.y);\n        mouse.gridCell = translatedPoint;\n        this.fixedRowClicked(grid, mouse);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The fixed column area has been clicked, massage the details and call the real function.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - event details\n     */\n    _fixedColumnClicked: function(grid, mouse) {\n        var translatedPoint = this.grid.newPoint(mouse.gridCell.x, this.getScrollPositionY() + mouse.gridCell.y - this.getFixedRowCount());\n        mouse.gridCell = translatedPoint;\n        this.fixedColumnClicked(grid, mouse);\n    },\n\n    moveSingleSelect: function(grid, x, y) {\n        if (this.featureChain) {\n            this.featureChain.moveSingleSelect(grid, x, y);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate setting the cursor up the feature chain of responsibility\n     * @param {Hypergrid} grid\n     */\n    setCursor: function(grid) {\n        grid.updateCursor();\n        this.featureChain.setCursor(grid);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse move to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseMove: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseMove(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling tap to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onTap: function(grid, event) {\n\n        if (this.featureChain) {\n            this.featureChain.handleTap(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling tap to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onContextMenu: function(grid, event) {\n        var proceed = grid.fireSyntheticContextMenuEvent(event);\n        if (proceed && this.featureChain) {\n            this.featureChain.handleContextMenu(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling wheel moved to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onWheelMoved: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleWheelMoved(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse up to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseUp: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseUp(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse drag to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseDrag: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseDrag(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling key down to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onKeyDown: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleKeyDown(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling key up to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onKeyUp: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleKeyUp(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling double click to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onDoubleClick: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleDoubleClick(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling hold pulse to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onHoldPulse: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleHoldPulse(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling double click to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    toggleColumnPicker: function() {\n        var dialog = this.grid.dialog;\n        var self = this;\n        if (dialog.isOpen()) {\n            dialog.close();\n        } else {\n            this.buildColumnPicker(dialog.overlay);\n            dialog.onClose = function() {\n                self.updateFromColumnPicker(dialog.overlay);\n            };\n            dialog.open();\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse down to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseDown(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse exit to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseExit: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseExit(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is replaced by the grid on initialization and serves as the callback\n     */\n    changed: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is replaced by the grid on initialization and serves as the callback\n     */\n    shapeChanged: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {boolean} Can re-order columns.\n     */\n    isColumnReorderable: function() {\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {Object} The properties for a specific column. These are used if no cell properties are specified.\n     * @param {index} columnIndex - the column index of interest\n     */\n    getColumnProperties: function(columnIndex) {\n        var col = this.columns[columnIndex];\n        if (!col) {\n            return {\n                isNull: true\n            };\n        }\n        var properties = col.getProperties(); //TODO: returns `null` on Hypergrid.reset();\n        if (!properties) {\n            return {\n                isNull: true\n            };\n        }\n        return properties;\n    },\n    setColumnProperties: function(columnIndex, properties) {\n        var columnProperties = this.allColumns[columnIndex].getProperties();\n        _(columnProperties).extendOwn(properties);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The field at `colIndex`.\n     * @param {number} colIndex - the column index of interest\n     */\n    getField: function(colIndex) {\n        if (colIndex === -1) {\n            return 'tree';\n        }\n        var col = this.getColumn(colIndex);\n        return col.getField();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The column heading at `colIndex'.\n     * @param {number} colIndex - the column index of interest\n     */\n    getHeader: function(colIndex) {\n        if (colIndex === -1) {\n            return 'Tree';\n        }\n        var col = this.getColumn(colIndex);\n        return col.getHeader();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this is called by the column editor post closing; rebuild the column order indexes\n     * @param {Array} list - list of column objects from the column editor\n     */\n    setColumnDescriptors: function(lists) {\n        //assumes there is one row....\n        var visible = lists.visible;\n        var tableState = this.getPrivateState();\n\n        var columnCount = visible.length;\n        var indexes = [];\n        var i;\n        for (i = 0; i < columnCount; i++) {\n            indexes.push(visible[i].id);\n        }\n        tableState.columnIndexes = indexes;\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string[]} All the currently hidden column header labels.\n     */\n    getHiddenColumnDescriptors: function() {\n        var tableState = this.getPrivateState();\n        var indexes = tableState.columnIndexes;\n        var labels = [];\n        var columnCount = this.getColumnCount();\n        for (var i = 0; i < columnCount; i++) {\n            if (indexes.indexOf(i) === -1) {\n                labels.push({\n                    id: i,\n                    label: this.getHeader(i),\n                    field: this.getField(i)\n                });\n            }\n        }\n        return labels;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc hide columns that are specified by their indexes\n     * @param {Array} arrayOfIndexes - an array of column indexes to hide\n     */\n    hideColumns: function(arrayOfIndexes) {\n        var tableState = this.getPrivateState();\n        var order = tableState.columnIndexes;\n        for (var i = 0; i < arrayOfIndexes.length; i++) {\n            var each = arrayOfIndexes[i];\n            if (order.indexOf(each) !== -1) {\n                order.splice(order.indexOf(each), 1);\n            }\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {integer} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        var tableState = this.getPrivateState();\n        return tableState.fixedColumnCount || 0;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc set the number of fixed columns\n     * @param {number} n - the integer count of how many columns to be fixed\n     */\n    setFixedColumnCount: function(n) {\n        var tableState = this.getPrivateState();\n        tableState.fixedColumnCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {integer} The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        if (!this.tableState) {\n            return 0;\n        }\n        var headers = this.getGrid().getHeaderRowCount();\n        var usersSize = this.tableState.fixedRowCount || 0;\n        return headers + usersSize;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of fixed rows, which includes (top to bottom order):\n     * 1. The header rows\n     *    1. The header labels row (optional)\n     *    2. The filter row (optional)\n     *    3. The top total rows (0 or more)\n     * 2. The non-scrolling rows (externally called \"the fixed rows\")\n     *\n     * @returns {number} Sum of the above or 0 if none of the above are in use.\n     *\n     * @param {number} n - The number of rows.\n     */\n    setFixedRowCount: function(n) {\n        this.tableState.fixedRowCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of header rows.\n     * A portion of the number returned by {@link Behavior#getFixedRowCount()|getFixedRowCount()}.\n     * (The remaining _fixed rows_ are the _top totals_ rows.)\n     */\n    getHeaderRowCount: function() {\n        var grid = this.getGrid();\n        var header = grid.isShowHeaderRow() ? 1 : 0;\n        var filter = grid.isShowFilterRow() ? 1 : 0;\n        var totals = this.getTopTotals().length;\n        return header + filter + totals;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of footer rows, consisting entirely of 0 or more _bottom totals_ rows.\n     */\n    getFooterRowCount: function() {\n        return this.getBottomTotals().length;\n    },\n\n    getTopTotals: function() {\n        return this.getDataModel().getTopTotals();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @summary Set the number of header rows.\n     * @param {number} n - The number of _fixed rows_ to reserve as header rows.\n     * (The remaining _fixed rows_ are the _top totals_ rows.)\n     */\n    setHeaderRowCount: function(n) {\n        this.tableState.headerRowCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of fixed rows.\n     */\n    getHeaderColumnCount: function() {\n        var grid = this.getGrid();\n        var count = grid.resolveProperty('headerColumnCount');\n        return count;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} The number of fixed rows.\n     */\n    setHeaderColumnCount: function(numberOfHeaderColumns) {\n        this.tableState.headerColumnCount = numberOfHeaderColumns;\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc build and open the editor within the container div argument\n     * @return {boolean} `false` prevents editor from opening\n     * @param {HTMLDivElement} div - the containing div element\n     */\n    buildColumnPicker: function(div) {\n        var container = document.createElement('div');\n\n        var hidden = document.createElement('fin-hypergrid-dnd-list');\n        var visible = document.createElement('fin-hypergrid-dnd-list');\n\n        container.appendChild(hidden);\n        container.appendChild(visible);\n\n        this.beColumnStyle(hidden.style);\n        hidden.title = 'hidden columns';\n        hidden.list = this.getHiddenColumnDescriptors();\n\n        this.beColumnStyle(visible.style);\n        visible.style.left = '50%';\n        visible.title = 'visible columns';\n        visible.list = this.getColumnDescriptors();\n\n        div.lists = {\n            hidden: hidden.list,\n            visible: visible.list\n        };\n        div.appendChild(container);\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc the editor is requesting close; deal with the edits\n     * @return `true`\n     * @param {HTMLDivElement} div - the containing div element\n     */\n    updateFromColumnPicker: function(div) {\n        var lists = div.lists;\n        this.setColumnDescriptors(lists);\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc a dnd column has just been dropped, we've been notified\n     */\n    endDragColumnNotification: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc bind column editor appropriate css values to arg style\n     * @param {HTMLStyleElement} style - the style object to enhance\n     */\n    beColumnStyle: function(style) {\n        style.top = '5%';\n        style.position = 'absolute';\n        style.width = '50%';\n        style.height = '100%';\n        style.whiteSpace = 'nowrap';\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {null} the cursor at a specific x,y coordinate\n     * @param {number} x - the x coordinate\n     * @param {number} y - the y coordinate\n     */\n    getCursorAt: function(x, y) {\n        return null;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The total number of columns.\n     */\n    getColumnCount: function() {\n        return this.columns.length;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The column alignment at column `x`: `'left'`, `'center'` , or `'right'`\n     * @param {number} x - The column index of interest.\n     */\n    getColumnAlignment: function(x) {\n        return 'center';\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Quietly set the horizontal scroll position.\n     * @param {number} x - The new position in pixels.\n     */\n    setScrollPositionX: function(x) {\n        this.scrollPositionX = x;\n    },\n\n    getScrollPositionX: function() {\n        return this.scrollPositionX;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Quietly set the vertical scroll position.\n     * @param {number} y - The new position in pixels.\n     */\n    setScrollPositionY: function(y) {\n        this.scrollPositionY = y;\n    },\n\n    getScrollPositionY: function() {\n        return this.scrollPositionY;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {cellEditor} The cell editor for the cell at cell coordinates `x,y`\n     * @param {number} x - The horizontal cell coordinate.\n     * @param {number} y - The vertical cell coordinate.\n     */\n    _getCellEditorAt: function(x, y) {\n        var editor = this.getColumn(x).getCellEditorAt(x, y);\n        if (editor) {\n            return editor;\n        }\n        var grid = this.getGrid();\n        var column = this.getColumn(x);\n        var type = grid.isFilterRow(y) ? column.getFilterType() : column.getType();\n        editor = grid.resolveCellEditor(type);\n        return editor;\n    },\n\n    getCellEditorAt: function(x, y) {\n        var grid = this.getGrid();\n        if (grid.isFilterRow(y)) {\n            return grid.cellEditors.textfield;\n        }\n        var editor = this.getDataModel().getCellEditorAt(x, y);\n        return editor;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} x - The column index.\n     * @param {string[]} keys\n     */\n    toggleSort: function(x, keys) {\n        this.getColumn(x).toggleSort(keys);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {boolean} `true` if we should highlight on hover\n     * @param {boolean} isColumnHovered - the column is hovered or not\n     * @param {boolean} isRowHovered - the row is hovered or not\n     */\n    highlightCellOnHover: function(isColumnHovered, isRowHovered) {\n        return isColumnHovered && isRowHovered;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellPropertiesPrePaintNotification: function(cellProperties) {\n        var row = this.getRow(cellProperties.y);\n        var columnId = this.getHeader(cellProperties.x);\n        cellProperties.row = row;\n        cellProperties.columnId = columnId;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a fixed row cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellFixedRowPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a fixed column cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellFixedColumnPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a top left cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellTopLeftPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function enhance the double click event just before it's broadcast to listeners\n     * @param {Object} event - event to enhance\n     */\n    enhanceDoubleClickEvent: function(event) {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc swap src and tar columns\n     * @param {number} src - column index\n     * @param {number} tar - column index\n     */\n    swapColumns: function(source, target) {\n        var columns = this.columns;\n        var tmp = columns[source];\n        columns[source] = columns[target];\n        columns[target] = tmp;\n        this.changed();\n    },\n\n    getColumnEdge: function(c, renderer) {\n        return this.getDataModel().getColumnEdge(c, renderer);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} [atBottom=false] - this value is in the \"bottom\" totals area\n     */\n    setTotalsValue: function(x, y, value, atBottom) {\n        this.getGrid().setTotalsValueNotification(x, y, value, !!atBottom);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {object} The object at y index.\n     * @param {number} y - the row index of interest\n     */\n    getRow: function(y) {\n        return this.getDataModel().getRow(y);\n    },\n\n    convertViewPointToDataPoint: function(viewPoint) {\n        var newX = this.getColumn(viewPoint.x).index;\n        var newPoint = this.getGrid().newPoint(newX, viewPoint.y);\n        return newPoint;\n    },\n\n    setGroups: function(arrayOfColumnIndexes) {\n        this.getDataModel().setGroups(arrayOfColumnIndexes);\n        this.createColumns();\n        this.changed();\n    },\n\n    setAggregates: function(mapOfKeysToFunctions) {\n        var self = this;\n        this.getDataModel().setAggregates(mapOfKeysToFunctions);\n        this.createColumns();\n        setTimeout(function() {\n            self.changed();\n        }, 100);\n    },\n\n    hasHierarchyColumn: function() {\n        return false;\n    },\n\n    getRowContextFunction: function(selectedRows) {\n        return function() {\n            return null;\n        };\n    },\n\n    getSelectionMatrixFunction: function(selectedRows) {\n        return function() {\n            return null;\n        };\n    },\n\n    getFieldName: function(index) {\n        return this.getFields()[index];\n    },\n\n    getColumnIndex: function(fieldName) {\n        return this.getFields().indexOf(fieldName);\n    },\n\n    getComputedRow: function(y) {\n        return this.getDataModel().getComputedRow(y);\n    },\n\n    autosizeAllColumns: function() {\n        this.checkColumnAutosizing(true);\n        this.changed();\n    },\n\n    checkColumnAutosizing: function(force) {\n        force = force === true;\n        this.autoSizeRowNumberColumn();\n        this.allColumns[-2].checkColumnAutosizing(force);\n        this.allColumns.forEach(function(column) {\n            column.checkColumnAutosizing(force);\n        });\n    },\n\n    autoSizeRowNumberColumn: function() {\n        if (this.getGrid().isRowNumberAutosizing()) {\n            this.allColumns[-1].checkColumnAutosizing(true);\n        }\n    },\n\n    setGlobalFilter: function(string) {\n        this.getDataModel().setGlobalFilter(string);\n    },\n\n    getSelectedRows: function() {\n        return this.getGrid().getSelectionModel().getSelectedRows();\n    },\n\n    getSelectedColumns: function() {\n        return this.getGrid().getSelectionModel().getSelectedColumns();\n    },\n\n    getSelections: function() {\n        return this.getGrid().getSelectionModel().getSelections();\n    },\n\n    getData: function() {\n        return this.getDataModel().getData();\n    },\n\n    getFilteredData: function() {\n        return this.getDataModel().getFilteredData();\n    },\n});\n\nmodule.exports = Behavior;\n","/* eslint-env browser */\n\n'use strict';\n\nvar _ = require('object-iterators');\n\nfunction Column(behavior, index, label) {\n    this.behavior = behavior;\n    this.dataModel = behavior.getDataModel();\n    this.index = index;\n    this.label = label;\n}\n\nColumn.prototype = {\n    constructor: Column.prototype.constructor,\n\n    getUnfilteredValue: function(y) {\n        return this.dataModel.getUnfilteredValue(this.index, y);\n    },\n\n    getValue: function(y) {\n        return this.dataModel.getValue(this.index, y);\n    },\n\n    setValue: function(y, value) {\n        return this.dataModel.setValue(this.index, y, value);\n    },\n\n    getWidth: function() {\n        var properties = this.getProperties();\n        if (properties) {\n            var override = properties.width;\n            if (override) {\n                return override;\n            }\n        }\n        return this.behavior.resolveProperty('defaultColumnWidth');\n    },\n\n    setWidth: function(width) {\n        this.getProperties().width = Math.max(5, width);\n    },\n\n    getCellRenderer: function(config, y) {\n        return this.dataModel.getCellRenderer(config, this.index, y);\n    },\n\n    getCellProperties: function(y) {\n        return this.behavior.getPrivateState().cellProperties[this.index + ',' + y];\n    },\n\n    setCellProperties: function(y, value) {\n        this.behavior.getPrivateState().cellProperties[this.index + ',' + y] = value;\n    },\n\n    setComplexFilter: function(data) {\n        this.getProperties().complexFilter = data;\n    },\n\n    getComplexFilter: function() {\n        return this.getProperties().complexFilter;\n    },\n\n    checkColumnAutosizing: function(force) {\n        var properties = this.getProperties();\n        var a, b, d;\n        if (properties) {\n            a = properties.width;\n            b = properties.preferredWidth || properties.width;\n            d = properties.columnAutosized && !force;\n            if (a !== b || !d) {\n                properties.width = !d ? b : Math.max(a, b);\n                properties.columnAutosized = !isNaN(properties.width);\n            }\n        }\n    },\n\n    getCellType: function(y) {\n        var value = this.getValue(y);\n        var type = this.typeOf(value);\n        return type;\n    },\n\n    getFilterType: function() {\n        // var props = this.getProperties();\n        // var type = props.filterType;\n        // if (!type) {\n        //     type = this.getType();\n        //     if (type !== 'unkknown') {\n        //         props.type = type;\n        //     }\n        // }\n        // return type;\n        return 'filter';\n    },\n\n    getType: function() {\n        var props = this.getProperties();\n        var type = props.type;\n        if (!type) {\n            type = this.computeColumnType();\n            if (type !== 'unkknown') {\n                props.type = type;\n            }\n        }\n        return type;\n    },\n\n    computeColumnType: function() {\n        var headerRowCount = this.behavior.getHeaderRowCount();\n        var height = this.behavior.getRowCount();\n        var value = this.getValue(headerRowCount);\n        var eachType = this.typeOf(value);\n        if (!eachType) {\n            return 'unknown';\n        }\n        var type = this.typeOf(value);\n        var isNumber = ((typeof value) === 'number');\n        for (var y = headerRowCount; y < height; y++) {\n            value = this.getValue(y);\n            eachType = this.typeOf(value);\n            if (type !== eachType) {\n                if (isNumber && (typeof value === 'number')) {\n                    type = 'float';\n                } else {\n                    return 'mixed';\n                }\n            }\n        }\n        return type;\n    },\n\n    typeOf: function(something) {\n        var typeOf = typeof something;\n        switch (typeOf) {\n            case 'object':\n                return something.constructor.name.toLowerCase();\n            case 'number':\n                return parseInt(something) === something ? 'int' : 'float';\n            default:\n                return typeOf;\n        }\n    },\n\n    getProperties: function() {\n        return this.behavior.getPrivateState().columnProperties[this.index];\n    },\n\n    setProperties: function(properties) {\n        var current = this.behavior.getPrivateState().columnProperties[this.index];\n        this.clearObjectProperties(current, false);\n        _(current).extendOwn(properties);\n    },\n\n    toggleSort: function(keys) {\n        this.dataModel.toggleSort(this.index, keys);\n    },\n\n    getCellEditorAt: function(x, y) {\n        return this.dataModel.getCellEditorAt(this.index, y);\n    },\n\n    getHeader: function() {\n        return this.label;\n    },\n\n    getField: function() {\n        return this.dataModel.getFields()[this.index];\n    }\n};\n\nmodule.exports = Column;\n","'use strict';\n\nfunction DataModelDecorator(grid, component) {\n    this.setComponent(component);\n    this.setGrid(grid);\n}\n\nDataModelDecorator.prototype = {\n    constructor: DataModelDecorator.prototype.constructor,\n\n    component: null,\n    grid: null,\n\n    getGrid: function() {\n        return this.grid;\n    },\n\n    setGrid: function(newGrid) {\n        this.grid = newGrid;\n        this.getComponent().setGrid(newGrid);\n    },\n\n    getBehavior: function() {\n        return this.getGrid().getBehavior();\n    },\n\n    changed: function() {\n        this.getBehavior().changed();\n    },\n\n    getPrivateState: function() {\n        return this.getGrid().getPrivateState();\n    },\n\n    applyState: function() {\n\n    },\n\n    setComponent: function(newComponent) {\n        this.component = newComponent;\n    },\n\n    getComponent: function() {\n        return this.component;\n    },\n\n    setGlobalFilter: function(string) {\n        return this.getComponent().setGlobalFilter(string);\n    },\n\n    getData: function() {\n        return this.getComponent().getData();\n    },\n\n    getFilteredData: function() {\n        return this.getComponent().getFilteredData();\n    },\n\n    getValue: function(x, y) {\n        return this.getComponent().getValue(x, y);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        return this.getComponent().getUnfilteredValue(x, y);\n    },\n\n    setValue: function(x, y, value) {\n        this.getComponent().setValue(x, y, value);\n    },\n\n    getColumnCount: function() {\n        return this.getComponent().getColumnCount();\n    },\n\n    applyFilters: function() {\n        return this.getComponent().applyFilters();\n    },\n\n    getRowCount: function() {\n        return this.getComponent().getRowCount();\n    },\n\n    getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) {\n        return this.getComponent().getCellRenderer(config, x, y, untranslatedX, untranslatedY);\n    },\n\n    getRowHeight: function(y) {\n        return this.getComponent().getRowHeight(y);\n    },\n\n    getColumnEdge: function(x, renderer) {\n        return this.getComponent().getColumnEdge(x, renderer);\n    },\n\n    getColumnWidth: function(x) {\n        return this.getComponent().getColumnWidth(x);\n    },\n\n    setColumnWidth: function(x, width) {\n        this.getComponent().setColumnWidth(x, width);\n    },\n\n    toggleSort: function(x, keys) {\n        this.getComponent().toggleSort(x, keys);\n    },\n\n    getColumnProperties: function(columnIndex) {\n        return this.getComponent().getColumnProperties(columnIndex);\n    },\n\n    setColumnProperties: function(columnIndex, properties) {\n        this.getComponent().setColumnProperties(columnIndex, properties);\n    },\n\n    getHeaders: function() {\n        return this.getComponent().getHeaders();\n    },\n\n    getFields: function() {\n        return this.getComponent().getFields();\n    },\n\n    setFields: function(fields) {\n        this.getComponent().setFields(fields);\n    },\n\n    getCellProperties: function(x, y) {\n        return this.getComponent().getCellProperties(x, y);\n    },\n\n    setCellProperties: function(x, y, value) {\n        this.getComponent().setCellProperties(x, y, value);\n    },\n\n    getRow: function(y) {\n        return this.getComponent().getRow(y);\n    },\n\n    getTopTotals: function() {\n        return this.getComponent().getTopTotals();\n    },\n\n    setTopTotals: function(totalRows) {\n        this.getComponent().setTopTotals(totalRows);\n    },\n\n    getBottomTotals: function() {\n        return this.getComponent().getBottomTotals();\n    },\n\n    setBottomTotals: function(totalRows) {\n        this.getComponent().setBottomTotals(totalRows);\n    },\n\n    setData: function(y) {\n        return this.getComponent().setData(y);\n    },\n\n    hasHierarchyColumn: function() {\n        return this.getComponent().hasHierarchyColumn();\n    },\n\n    setHeaders: function(headerLabels) {\n        return this.getComponent().setHeaders(headerLabels);\n    },\n\n    cellClicked: function(cell, event) {\n        return this.getComponent().cellClicked(cell, event);\n    },\n\n    getAvailableGroups: function() {\n        return this.getComponent().getAvailableGroups();\n    },\n\n    getGroups: function() {\n        return this.getComponent().getGroups();\n    },\n\n    setGroups: function(groups) {\n        this.getComponent().setGroups(groups);\n    },\n\n    getHiddenColumns: function() {\n        return this.getComponent().getHiddenColumns();\n    },\n\n    getVisibleColumns: function() {\n        return this.getComponent().getVisibleColumns();\n    },\n\n    setAggregates: function(aggregates) {\n        return this.getComponent().setAggregates(aggregates);\n    },\n\n    reset: function() {\n        this.getComponent().reset();\n    },\n\n    getCellEditorAt: function(x, y) {\n        return this.getComponent().getCellEditorAt(x, y);\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.getComponent().getUnfilteredRowCount();\n    }\n};\n\nmodule.exports = DataModelDecorator;\n","'use strict';\n\nvar ListDragon = require('list-dragon');\n\nvar Behavior = require('./Behavior');\nvar DataModelDecorator = require('./DataModelDecorator');\nvar DataModelJSON = require('../dataModels/JSON');\nvar features = require('../features/index');\nvar addStylesheet = require('../stylesheets');\n//var aggregations = require('hyper-analytics').util.aggregations;\n//var aggregations = require('../local_node_modules/hyper-analytics').util.aggregations;\nvar aggregations = require('../local_node_modules/finanalytics').aggregations;\n\n/**\n * @name behaviors.JSON\n * @desc > Same parameters as {@link behaviors.JSON#initialize|initialize}, which is called by this constructor.\n * @constructor\n */\nvar JSON = Behavior.extend('behaviors.JSON', {\n\n    /**\n     * @summary Constructor logic, called _after_{@link Behavior#initialize|Behavior.initialize()}.\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     *\n     * @param grid - the hypergrid\n     * @param {object[]} dataRows - array of uniform data objects\n     * @memberOf behaviors.JSON.prototype\n     */\n    initialize: function(grid, dataRows) {\n        this.setData(dataRows);\n    },\n\n    features: [\n        features.CellSelection,\n        features.KeyPaging,\n        features.ColumnPicker,\n        features.ColumnResizing,\n        features.RowResizing,\n        features.Filters,\n        features.RowSelection,\n        features.ColumnSelection,\n        features.ColumnMoving,\n        features.ColumnSorting,\n        features.CellEditing,\n        features.CellClick,\n        features.OnHover\n    ],\n\n    aggregations: aggregations,\n\n    createColumns: function() {\n        var dataModel = this.getDataModel();\n        var columnCount = dataModel.getColumnCount();\n        var headers = dataModel.getHeaders();\n        var fields = dataModel.getFields();\n        this.clearColumns();\n        for (var i = 0; i < columnCount; i++) {\n            var header = headers[i];\n            var column = this.addColumn(i, header);\n            var properties = column.getProperties();\n            properties.field = fields[i];\n            properties.header = header;\n            properties.complexFilter = null;\n        }\n    },\n\n    getDefaultDataModel: function() {\n        var model = new DataModelJSON();\n        var wrapper = new DataModelDecorator(this.getGrid(), model);\n        wrapper.setComponent(model);\n        return wrapper;\n    },\n\n    applyFilters: function() {\n        this.dataModel.applyFilters();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the header labels.\n     * @param {string[]} headerLabels - The header labels.\n     */\n    setHeaders: function(headerLabels) {\n        this.getDataModel().setHeaders(headerLabels);\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @desc * @returns {string[]} The header labels.\n     */\n    getHeaders: function() {\n        return this.getDataModel().getHeaders();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the fields array.\n     * @param {string[]} fieldNames - The field names.\n     */\n    setFields: function(fieldNames) {\n        //were defining the columns based on field names....\n        //we must rebuild the column definitions\n        this.getDataModel().setFields(fieldNames);\n        this.createColumns();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Get the field names.\n     * @returns {string[]}\n     */\n    getFields: function() {\n        return this.getDataModel().getFields();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the data field.\n     * @param {object[]} objects - An array of uniform objects, each being a row in the grid.\n     */\n    setData: function(dataRows) {\n        this.getDataModel().setData(dataRows);\n        this.createColumns();\n        var self = this;\n        if (this.getGrid().isColumnAutosizing()) {\n            setTimeout(function() {\n                self.autosizeAllColumns();\n            }, 100);\n            self.changed();\n        } else {\n            setTimeout(function() {\n                self.allColumns[-1].checkColumnAutosizing(true);\n                self.changed();\n            });\n        }\n    },\n\n    /**\n     * @summary Set the top totals.\n     * @memberOf behaviors.JSON.prototype\n     * @param {Array<Array>} totalRows - array of rows (arrays) of totals\n     */\n    setTopTotals: function(totalRows) {\n        this.getDataModel().setTopTotals(totalRows);\n    },\n\n    /**\n     * @summary Get the top totals.\n     * @memberOf behaviors.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getTopTotals: function() {\n        return this.getDataModel().getTopTotals();\n    },\n\n    /**\n     * @summary Set the bottom totals.\n     * @memberOf behaviors.JSON.prototype\n     * @param {Array<Array>} totalRows - array of rows (arrays) of totals\n     */\n    setBottomTotals: function(totalRows) {\n        this.getDataModel().setBottomTotals(totalRows);\n    },\n\n    /**\n     * @summary Get the bottom totals.\n     * @memberOf behaviors.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getBottomTotals: function() {\n        return this.getDataModel().getBottomTotals();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Build the fields and headers from the supplied column definitions.\n     * ```javascript\n     * myJsonBehavior.setColumns([\n     *     { title: 'Stock Name', field: 'short_description' },\n     *     { title: 'Status', field: 'trading_phase' },\n     *     { title: 'Reference Price', field: 'reference_price' }\n     * ]);\n     * ```\n     * @param {Array} columnDefinitions - an array of objects with fields 'title', and 'field'\n     */\n    setColumns: function(columnDefinitions) {\n        this.getDataModel().setColumns(columnDefinitions);\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Enhance the double-click event just before it's broadcast to listeners.\n     * @param {Point} event\n     */\n    enhanceDoubleClickEvent: function(event) {\n        event.row = this.getRow(event.gridCell.y);\n    },\n\n    setDataProvider: function(dataProvider) {\n        this.getDataModel().setDataProvider(dataProvider);\n    },\n\n    hasHierarchyColumn: function() {\n        return this.getDataModel().hasHierarchyColumn();\n    },\n\n    getColumnAlignment: function(x) {\n        if (x === 0 && this.hasHierarchyColumn()) {\n            return 'left';\n        } else {\n            return 'center';\n        }\n    },\n\n    getRowSelectionMatrix: function(selectedRows) {\n        return this.getDataModel().getRowSelectionMatrix(selectedRows);\n    },\n\n    getColumnSelectionMatrix: function(selectedColumns) {\n        return this.getDataModel().getColumnSelectionMatrix(selectedColumns);\n    },\n\n    getSelectionMatrix: function(selections) {\n        return this.getDataModel().getSelectionMatrix(selections);\n    },\n\n    getRowSelection: function() {\n        var selectedRows = this.getSelectedRows();\n        return this.getDataModel().getRowSelection(selectedRows);\n    },\n\n    getColumnSelection: function() {\n        var selectedColumns = this.getSelectedColumns();\n        return this.getDataModel().getColumnSelection(selectedColumns);\n    },\n\n    getSelection: function() {\n        var selections = this.getSelections();\n        return this.getDataModel().getSelection(selections);\n    },\n\n    buildColumnPicker: function(div) {\n        if (!this.isColumnReorderable()) {\n            return false;\n        }\n\n        var listOptions = {\n            cssStylesheetReferenceElement: div\n        };\n\n        var groups = { models: this.getGroups(), title: 'Groups' },\n            availableGroups = { models: this.getAvailableGroups(), title: 'Available Groups' },\n            hiddenColumns = { models: this.getHiddenColumns(), title: 'Hidden Columns' },\n            visibleColumns = { models: this.getVisibleColumns(), title: 'Visible Columns'},\n            groupLists = new ListDragon([groups, availableGroups], listOptions),\n            columnLists = new ListDragon([hiddenColumns, visibleColumns], listOptions),\n            listSets = [groupLists, columnLists];\n\n        addStylesheet('list-dragon', div);\n\n        listSets.forEach(function(listSet) {\n            listSet.modelLists.forEach(function(list) {\n                div.appendChild(list.container);\n            });\n        });\n\n        //attach for later retrieval\n        div.lists = {\n            group: groups.models,\n            availableGroups: availableGroups.models,\n            hidden: hiddenColumns.models,\n            visible: visibleColumns.models\n        };\n\n        return true;\n    },\n    getGroups: function() {\n        return this.getDataModel().getGroups();\n    },\n    getAvailableGroups: function() {\n        return this.getDataModel().getAvailableGroups();\n    },\n    getHiddenColumns: function() {\n        return this.getDataModel().getHiddenColumns();\n    },\n    getVisibleColumns: function() {\n        return this.getDataModel().getVisibleColumns();\n    },\n    setColumnDescriptors: function(lists) {\n        //assumes there is one row....\n        var tree = this.columns[0];\n        this.columns.length = 0;\n        if (tree && tree.label === 'Tree') {\n            this.columns.push(tree);\n        }\n        for (var i = 0; i < lists.visible.length; i++) {\n            this.columns.push(lists.visible[i]);\n        }\n\n        var groupBys = lists.group.map(function(e) {\n            return e.id;\n        });\n        this.getDataModel().setGroups(groupBys);\n\n        this.changed();\n    },\n\n    getSelectedRows: function() {\n        var offset = -this.getGrid().getHeaderRowCount();\n        var selections = this.getGrid().getSelectionModel().getSelectedRows();\n        var result = selections.map(function(each) {\n            return each + offset;\n        });\n        return result;\n    },\n\n    getSelectedColumns: function() {\n        return this.getGrid().getSelectionModel().getSelectedColumns();\n    },\n\n    getSelections: function() {\n        return this.getGrid().getSelectionModel().getSelections();\n    }\n\n});\n\nmodule.exports = JSON;\n","'use strict';\n\nvar Behavior = require('./Behavior');\n\nvar noop = function() {},\n    n00p = function() { return 0; };\n\n/**\n * @constructor\n */\nvar Null = Behavior.extend('Null', {\n\n    //initalize: function(grid, component) {},\n\n    setScrollPositionY: noop,\n    setScrollPositionX: noop,\n    getColumnCount: n00p,\n    getFixedColumnCount: n00p,\n    getFixedColumnsWidth: n00p,\n    getFixedColumnsMaxWidth: n00p,\n    setRenderedWidth: n00p,\n    getRowCount: n00p,\n    getFixedRowCount: n00p,\n    getFixedRowsHeight: n00p,\n    getFixedRowsMaxHeight: n00p,\n    setRenderedHeight: n00p,\n    getCellProvider: noop,\n    click: noop,\n    doubleClick: noop\n});\n\nmodule.exports = Null;\n","'use strict';\n\nmodule.exports = {\n    Behavior: require('./Behavior'), // abstract base class\n    JSON: require('./JSON'),\n    Null: require('./Null')\n};","/* eslint-env browser */\n\n'use strict';\n\nvar mustache = require('mustache');\nvar Base = require('extend-me').Base;\n\n/**\n * @constructor\n */\nvar CellEditor = Base.extend('CellEditor', {\n\n    alias: 'base',\n\n    /**\n     * am I currently editing (i.e., between calls to `beginEditAt` and either `stopEditing` or `cancelEditing`)\n     * @type {boolean}\n     * @default false\n     * @memberOf CellEditor.prototype\n     */\n    isEditing: false,\n\n    /**\n     * the point that I am editing at right now\n     * @type {Point}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    editorPoint: {\n        x: -1,\n        y: -1\n    },\n\n    /**\n     * if true, check that the editor is in the right location\n     * @type {boolean}\n     * @default false\n     * @memberOf CellEditor.prototype\n     */\n    checkEditorPositionFlag: false,\n\n    /**\n     * my instance of hypergrid\n     * @type {Hypergrid}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    grid: null,\n\n    /**\n     * the value before editing\n     * @type {type}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    initialValue: null,\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc return the behavior (model)\n     * @returns {Behavior} The behavior (model).\n     */\n    getBehavior: function() {\n        return this.grid.getBehavior();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc This function is a callback from the fin-hypergrid.   It is called after each paint of the canvas.\n     */\n    gridRenderedNotification: function() {\n        this.checkEditor();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc scroll values have changed, we've been notified\n     */\n    scrollValueChangedNotification: function() {\n        this.setCheckEditorPositionFlag();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc turn on checkEditorPositionFlag boolean field\n     */\n    setCheckEditorPositionFlag: function() {\n        this.checkEditorPositionFlag = true;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc begin editing at location point\n     * @param {Point} point - the location to start editing at\n     */\n    beginEditAt: function(point) {\n        this.editorPoint = point;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc put value into our editor\n     * @param {object} value - whatever value we want to edit\n     */\n    setEditorValue: function(value) {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc returns the point at which we are currently editing\n     * @returns {Point}\n     */\n    getEditorPoint: function() {\n        return this.editorPoint;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc set the current editor location\n     * @param {Point} point - the data location of the current editor\n     */\n    setEditorPoint: function(point) {\n        this.editorPoint = point;\n        this.modelPoint = this.getGrid().convertViewPointToDataPoint(point);\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc display the editor\n     */\n    showEditor: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc hide the editor\n     */\n    hideEditor: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc stop editing\n     */\n    stopEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        var proceed = this.getGrid().fireSyntheticEditorDataChangeEvent(this, this.initialValue, this.getEditorValue, this);\n        if (!proceed) {\n            return;\n        }\n        this.saveEditorValue();\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    cancelEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc save the new value into the behavior(model)\n     */\n    saveEditorValue: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc return the current editor's value\n     */\n    getEditorValue: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc request focus\n     */\n    takeFocus: function() {},\n\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc check that the editor is in the correct location, and is showing/hidden appropriately\n     */\n    checkEditor: function() {\n    },\n\n    getGrid: function() {\n        return this.grid;\n    },\n\n    template: function() {\n        /*\n\n         */\n    },\n\n    getHTML: function() {\n        var string = this.template.toString().split('\\n');\n        string.shift();\n        string.shift();\n        string.length = string.length - 2;\n        string = string.join('\\n').trim();\n        return mustache.render(string, this);\n    },\n\n});\n\nmodule.exports = CellEditor;\n","'use strict';\n\nvar Simple = require('./Simple');\nvar Map = require('../Mappy');\n\n/**\n * @constructor\n */\nvar Choice = Simple.extend('Choice', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Choice.prototype\n     */\n    alias: 'choice',\n\n    /**\n     * the list of items to pick from\n     * @type {Array}\n     * @memberOf Choice.prototype\n     */\n    items: ['a', 'b', 'c'],\n\n    template: function() {\n        /*\n                <select id=\"editor\">\n                    {{#items}}\n                        <option value=\"{{.}}\">{{.}}</option>\n                    {{/items}}\n                </select>\n            */\n    },\n\n    autopopulate: function() {\n        var grid = this.getGrid();\n        var behavior = grid.getBehavior();\n        var point = this.getEditorPoint();\n        var colProps = grid.getColumnProperties(point.x);\n        if (!colProps.autopopulateEditor) {\n            return;\n        }\n        var headerCount = grid.getHeaderRowCount();\n        var rowCount = grid.getUnfilteredRowCount() - headerCount;\n        var column = point.x;\n        var map = new Map();\n        for (var r = 0; r < rowCount; r++) {\n            var each = behavior.getUnfilteredValue(column, r);\n            map.set(each, each);\n        }\n        var values = map.values;\n        values.sort();\n\n        if (values.length > 0 && values[0].length > 0) {\n            values.unshift('');\n        }\n\n        this.setItems(values);\n    },\n\n    //no events are fired while the dropdown is open\n    //see http://jsfiddle.net/m4tndtu4/6/\n\n    /**\n     * @memberOf Choice.prototype\n     */\n    showEditor: function() {\n        var self = this;\n        this.input.style.display = 'inline';\n        setTimeout(function() {\n            self.showDropdown(self.input);\n        }, 50);\n    },\n\n    preShowEditorNotification: function() {\n        this.autopopulate();\n        this.setEditorValue(this.initialValue);\n    },\n\n    /**\n     * @memberOf Choice.prototype\n     * @param items\n     */\n    setItems: function(items) {\n        this.items = items;\n        this.updateView();\n    },\n\n    /**\n     * @memberOf Choice.prototype\n     * @param input\n     */\n    initializeInput: function(input) {\n        var self = this;\n        Simple.prototype.initializeInput.apply(this, [input]);\n        input.onchange = function() {\n            self.stopEditing();\n        };\n    }\n\n});\n\nmodule.exports = Choice;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Color = Simple.extend('Color', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Color.prototype\n     */\n    alias: 'color',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"color\">\n        */\n    }\n\n});\n\nmodule.exports = Color;\n","/* eslint-env browser */\n\n'use strict';\n\nvar Simple = require('./Simple');\n\nvar parseDate = function(input) {\n  var parts = input.match(/(\\d+)/g);\n  // new Date(year, month [, date [, hours[, minutes[, seconds[, ms]]]]])\n  return new window.Date(parts[0], parts[1] - 1, parts[2]); // months are 0-based\n};\n\nvar leadingZeroIfNecessary = function(number) {\n    return number < 10 ? '0' + number : number + '';\n};\n/**\n * @constructor\n */\nvar Date = Simple.extend('Date', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Date.prototype\n     */\n    alias: 'date',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"date\">\n        */\n    },\n\n    setEditorValue: function(value) {\n        if (value && value.constructor.name === 'Date') {\n            value = value.getFullYear() + '-' + leadingZeroIfNecessary(value.getMonth() + 1) + '-' + leadingZeroIfNecessary(value.getDay());\n        }\n        this.getInput().value = value + '';\n    },\n\n    getEditorValue: function() {\n        var value = this.getInput().value;\n        value = parseDate(value);\n        return value;\n    },\n\n});\n\nmodule.exports = Date;\n","/* eslint-env browser */\n'use strict';\n\nvar CellEditor = require('./CellEditor');\n\n/**\n * @constructor\n */\nvar Filter = CellEditor.extend('Filter', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Textfield.prototype\n     */\n    alias: 'filter',\n\n    initialize: function() {\n        var data = document.createElement('div');\n        var style = data.style;\n        style.position = 'absolute';\n        style.top = style.bottom = '44px';\n        style.right = style.left = '1em';\n        style.overflowY = 'scroll';\n\n        var table = document.createElement('table');\n        data.appendChild(table);\n\n        style = table.style;\n        style.width = style.height = '100%';\n\n        var tr = document.createElement('tr');\n        var td = document.createElement('td');\n        table.appendChild(tr);\n        tr.appendChild(td);\n\n\n        this.title = document.createElement('div');\n        this.title.innerHTML = 'Filter Editor';\n\n        this.dialog = document.createElement('div');\n\n        this.content = td;\n        this.buttons = document.createElement('div');\n\n        style = this.dialog.style;\n        style.position = 'absolute';\n        style.top = style.left = style.right = style.bottom = 0;\n        style.whiteSpace = 'nowrap';\n\n        style = this.title.style;\n        style.position = 'absolute';\n        style.top = style.left = style.right = 0;\n        style.height = '44px';\n        style.whiteSpace = 'nowrap';\n        style.textAlign = 'center';\n        style.padding = '11px';\n\n        style = this.buttons.style;\n        style.position = 'absolute';\n        style.left = style.right = style.bottom = 0;\n        style.height = '44px';\n        style.whiteSpace = 'nowrap';\n        style.textAlign = 'center';\n        style.padding = '8px';\n\n        this.dialog.appendChild(this.title);\n        this.dialog.appendChild(data);\n        this.dialog.appendChild(this.buttons);\n\n        this.ok = document.createElement('button');\n        this.ok.style.borderRadius = '8px';\n        this.ok.style.width = '5.5em';\n\n        this.cancel = document.createElement('button');\n        this.cancel.style.marginLeft = '2em';\n        this.cancel.style.borderRadius = '8px';\n        this.cancel.style.width = '5.5em';\n\n        this.delete = document.createElement('button');\n        this.delete.style.marginLeft = '2em';\n        this.delete.style.borderRadius = '8px';\n        this.delete.style.width = '5.5em';\n\n        this.reset = document.createElement('button');\n        this.reset.style.marginLeft = '2em';\n        this.reset.style.borderRadius = '8px';\n        this.reset.style.width = '5.5em';\n\n        this.ok.innerHTML = 'ok';\n        this.cancel.innerHTML = 'cancel';\n        this.delete.innerHTML = 'delete';\n        this.reset.innerHTML = 'reset';\n\n        this.buttons.appendChild(this.ok);\n        this.buttons.appendChild(this.reset);\n        this.buttons.appendChild(this.delete);\n        this.buttons.appendChild(this.cancel);\n\n        var self = this;\n        this.ok.onclick = function() {\n            self.okPressed();\n        };\n        this.cancel.onclick = function() {\n            self.cancelPressed();\n        };\n        this.delete.onclick = function() {\n            self.deletePressed();\n        };\n        this.reset.onclick = function() {\n            self.resetPressed();\n        };\n    },\n\n    tearDown: function() {\n        this.content.innerHTML = '';\n    },\n\n    okPressed: function() {\n        var dialog = this.getGrid().dialog;\n        dialog.onOkPressed();\n    },\n\n    cancelPressed: function() {\n        var dialog = this.getGrid().dialog;\n        dialog.onCancelPressed();\n    },\n\n    deletePressed: function() {\n        var dialog = this.getGrid().dialog;\n        dialog.onDeletePressed();\n    },\n\n    resetPressed: function() {\n        var dialog = this.getGrid().dialog;\n        dialog.onResetPressed();\n    },\n\n    beginEditAt: function(editorPoint) {\n        var grid = this.getGrid();\n        var behavior = grid.getBehavior();\n        var dialog = grid.dialog;\n\n        var columnIndex = editorPoint.x,\n            title = behavior.getColumnId(columnIndex),\n            field = behavior.getField(columnIndex),\n            type = behavior.getColumn(columnIndex).getType();\n\n        var fieldsProvider = function() {\n            return [{\n                name: field,\n                alias: title,\n                type: type\n            }];\n        };\n        this.title.innerHTML = 'filter for <strong><em>' + title + '</em></strong> column';\n        var filter = grid.getFilterFor(columnIndex);\n        //var self = this;\n        if (dialog.isOpen()) {\n            dialog.close();\n        } else {\n            var self = this;\n\n            dialog.clear();\n            dialog.overlay.appendChild(this.dialog);\n\n            filter.initialize(fieldsProvider);\n\n            dialog.onOkPressed = function() {\n                if (filter.onOk && filter.onOk()) { // onOK() truthy result means abort; falsy means proceed\n                    return;\n                }\n                self.tearDown();\n                behavior.setComplexFilter(columnIndex, {\n                    type: filter.alias,\n                    state: filter.getState()\n                });\n                dialog.close();\n                behavior.applyFilters();\n                behavior.changed();\n            };\n\n            dialog.onCancelPressed = function() {\n                if (filter.onCancel && filter.onCancel()) {\n                    return;\n                }\n                self.tearDown();\n                dialog.close();\n                filter = undefined;\n            };\n\n            dialog.onDeletePressed = function() {\n                if (filter.onDelete && filter.onDelete()) {\n                    return;\n                }\n                self.tearDown();\n                behavior.setComplexFilter(columnIndex, undefined);\n                dialog.close();\n                behavior.applyFilters();\n                behavior.changed();\n            };\n\n            dialog.onResetPressed = function() {\n                if (filter.onReset && filter.onReset()) {\n                    return;\n                }\n                self.tearDown();\n                filter.initialize(dialog);\n                if (filter.onShow) {\n                    filter.onShow(self.content);\n                }\n            };\n\n            var cellBounds = grid._getBoundsOfCell(columnIndex, editorPoint.y);\n\n            //hack to accomodate bootstrap margin issues...\n            var xOffset = grid.div.getBoundingClientRect().left - grid.divCanvas.getBoundingClientRect().left;\n            cellBounds.x = cellBounds.x - xOffset;\n            dialog.openFrom(cellBounds);\n            var previousState = behavior.getComplexFilter(columnIndex);\n            if (previousState) {\n                filter.setState(previousState.state);\n            }\n            setTimeout(function() {\n                if (filter.onShow) {\n                    filter.onShow(self.content);\n                }\n            }, dialog.getAnimationTime() + 10);\n        }\n    },\n\n});\n\nmodule.exports = Filter;\n","/* eslint-env browser */\n\n'use strict';\n\nvar CellEditor = require('./CellEditor.js');\n\n/**\n * @constructor\n */\nvar Simple = CellEditor.extend('Simple', {\n\n    /**\n     * my main input control\n     * @type {Element}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    input: null,\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Simple.prototype\n     */\n    alias: 'simple',\n\n    /**\n     * @memberOf Simple.prototype\n     */\n    initialize: function() {\n        this.editorPoint = {\n            x: 0,\n            y: 0\n        };\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc  the function to override for initialization\n     */\n    initializeInput: function(input) {\n        var self = this;\n        input.addEventListener('keyup', function(e) {\n            if (e && (e.keyCode === 13 || e.keyCode === 27 || e.keyCode === 8)) {\n                e.preventDefault();\n                if (e.keyCode === 8) {\n                    self.clearStopEditing();\n                } else if (e.keyCode === 27) {\n                    self.cancelEditing();\n                } else {\n                    self.stopEditing();\n                }\n                self.getGrid().repaint();\n                self.getGrid().takeFocus();\n            }\n            self.getGrid().fireSyntheticEditorKeyUpEvent(self, e);\n        });\n        input.addEventListener('keydown', function(e) {\n            self.getGrid().fireSyntheticEditorKeyDownEvent(self, e);\n        });\n        input.addEventListener('keypress', function(e) {\n            self.getGrid().fireSyntheticEditorKeyPressEvent(self, e);\n        });\n        input.onblur = function(e) {\n            self.cancelEditing();\n        };\n\n        // input.addEventListener('focusout', function() {\n        //     self.stopEditing();\n        // });\n        // input.addEventListener('blur', function() {\n        //     self.stopEditing();\n        // });\n        input.style.position = 'absolute';\n        input.style.display = 'none';\n        input.style.border = 'solid 2px black';\n        input.style.outline = 0;\n        input.style.padding = 0;\n        input.style.zIndex = 1000;\n        //input.style.fontSize = '8pt';\n        input.style.boxShadow = 'white 0px 0px 1px 1px';\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @returns {object} the current editor's value\n     */\n    getEditorValue: function() {\n        var value = this.getInput().value;\n        return value;\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc save the new value into the behavior(model)\n     */\n    setEditorValue: function(value) {\n        this.getInput().value = value + '';\n    },\n\n    clearStopEditing: function() {\n        this.setEditorValue('');\n        this.stopEditing();\n    },\n\n    cancelEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        this.getInput().value = null;\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc display the editor\n     */\n    showEditor: function() {\n        this.getInput().style.display = 'inline';\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc hide the editor\n     */\n    hideEditor: function() {\n        this.getInput().style.display = 'none';\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc request focus for my input control\n     */\n    takeFocus: function() {\n        var self = this;\n        setTimeout(function() {\n            self.input.focus();\n            self.selectAll();\n        }, 300);\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc select everything\n     */\n    selectAll: function() {\n\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc how much should I offset my bounds from 0,0\n     */\n    originOffset: function() {\n        return [0, 0];\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc set the bounds of my input control\n     * @param {rectangle} rectangle - the bounds to move to\n     */\n    setBounds: function(cellBounds) {\n        var originOffset = this.originOffset();\n        var translation = 'translate('\n            + (cellBounds.x - 1 + originOffset[0]) + 'px,'\n            + (cellBounds.y - 1 + originOffset[1]) + 'px)';\n\n        var input = this.getInput();\n\n        input.style.boxSizing = 'border-box';\n\n        input.style.webkitTransform = translation;\n        input.style.MozTransform = translation;\n        input.style.msTransform = translation;\n        input.style.OTransform = translation;\n\n        // input.style.left = cellBounds.x + originOffset[0] + 'px';\n        // input.style.top = cellBounds.y + originOffset[1] + 'px';\n\n        input.style.width = (cellBounds.width + 2) + 'px';\n        input.style.height = (cellBounds.height + 2) + 'px';\n        //var xOffset = this.grid.canvas.getBoundingClientRect().left;\n    },\n\n    saveEditorValue: function() {\n        var point = this.getEditorPoint();\n        var value = this.getEditorValue();\n        if (value === this.initialValue) {\n            return; //data didn't change do nothing\n        }\n        if (parseFloat(this.initialValue) === this.initialValue) { // I'm a number\n            value = parseFloat(value);\n        }\n        var continued = this.getGrid().fireBeforeCellEdit(point, this.initialValue, value, this);\n        if (!continued) {\n            return;\n        }\n        this.getBehavior().setValue(point.x, point.y, value);\n        this.getGrid().fireAfterCellEdit(point, this.initialValue, value, this);\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc move the editor to the current editor point\n     */\n    _moveEditor: function() {\n        var grid = this.getGrid();\n        var editorPoint = this.getEditorPoint();\n        var cellBounds = grid._getBoundsOfCell(editorPoint.x, editorPoint.y);\n\n        //hack to accomodate bootstrap margin issues...\n        var xOffset = grid.div.getBoundingClientRect().left - grid.divCanvas.getBoundingClientRect().left;\n        cellBounds.x = cellBounds.x - xOffset;\n\n        this.setBounds(cellBounds);\n    },\n\n    moveEditor: function() {\n        this._moveEditor();\n        this.takeFocus();\n    },\n\n    beginEditAt: function(point) {\n\n        if (!this.isAdded) {\n            this.isAdded = true;\n            this.grid.div.appendChild(this.getInput());\n        }\n\n        this.setEditorPoint(point);\n        var model = this.getBehavior();\n        var value = model.getValue(point.x, point.y);\n        if (value.constructor.name === 'Array') {\n            value = value[1]; //it's a nested object\n        }\n        var proceed = this.grid.fireRequestCellEdit(point, value);\n        if (!proceed) {\n            //we were cancelled\n            return;\n        }\n        this.initialValue = value;\n        this.isEditing = true;\n        this.setCheckEditorPositionFlag();\n        this.checkEditor();\n    },\n\n    checkEditor: function() {\n        if (!this.checkEditorPositionFlag) {\n            return;\n        } else {\n            this.checkEditorPositionFlag = false;\n        }\n        if (!this.isEditing) {\n            return;\n        }\n        var editorPoint = this.getEditorPoint();\n        if (this.grid.isDataVisible(editorPoint.x, editorPoint.y)) {\n            this.preShowEditorNotification();\n            this.attachEditor();\n            this.moveEditor();\n            this.showEditor();\n        } else {\n            this.hideEditor();\n        }\n    },\n\n    attachEditor: function() {\n        var input = this.getInput();\n        this.grid.div.appendChild(input);\n    },\n\n    preShowEditorNotification: function() {\n        this.setEditorValue(this.initialValue);\n    },\n\n    getInput: function() {\n        if (!this.input) {\n            this.input = this.getDefaultInput();\n        }\n        return this.input;\n    },\n\n    getDefaultInput: function() {\n        var div = document.createElement('DIV');\n        div.innerHTML = this.getHTML();\n        var input = div.firstChild;\n        this.initializeInput(input);\n        return input;\n    },\n\n    updateView: function() {\n        var oldGuy = this.getInput();\n        var parent = oldGuy.parentNode;\n        var newGuy = this.getDefaultInput();\n        this.input = newGuy;\n        parent.replaceChild(newGuy, oldGuy);\n    },\n\n    showDropdown: function(element) {\n        var event;\n        event = document.createEvent('MouseEvents');\n        event.initMouseEvent('mousedown', true, true, window);\n        element.dispatchEvent(event);\n    }\n});\n\nmodule.exports = Simple;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Slider = Simple.extend('Slider', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Slider.prototype\n     */\n    alias: 'slider',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"range\">\n        */\n    }\n\n});\n\nmodule.exports = Slider;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Spinner = Simple.extend('Spinner', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Spinner.prototype\n     */\n    alias: 'spinner',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"number\">\n        */\n    }\n\n});\n\nmodule.exports = Spinner;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Textfield = Simple.extend('Textfield', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Textfield.prototype\n     */\n    alias: 'textfield',\n\n    template: function() {\n        /*\n            <input id=\"editor\">\n        */\n    },\n\n    selectAll: function() {\n        this.input.setSelectionRange(0, this.input.value.length);\n    },\n\n    initializeInput: function(input) {\n        var self = this;\n        input.addEventListener('keyup', function(e) {\n            if (e && (e.keyCode === 13 || e.keyCode === 27)) {\n                e.preventDefault();\n                if (e.keyCode === 27) {\n                    self.cancelEditing();\n                } else {\n                    self.stopEditing();\n                }\n                self.getGrid().repaint();\n                self.getGrid().takeFocus();\n            }\n            if (self.getGrid().isFilterRow(self.getEditorPoint().y)) {\n                setTimeout(function() {\n                    self.saveEditorValue();\n                    self._moveEditor();\n                });\n            }\n            self.getGrid().fireSyntheticEditorKeyUpEvent(self, e);\n        });\n        input.addEventListener('keydown', function(e) {\n            self.getGrid().fireSyntheticEditorKeyDownEvent(self, e);\n        });\n        input.addEventListener('keypress', function(e) {\n            self.getGrid().fireSyntheticEditorKeyPressEvent(self, e);\n        });\n        input.onblur = function(e) {\n            self.cancelEditing();\n        };\n        // input.addEventListener('focusout', function() {\n        //     self.stopEditing();\n        // });\n        // input.addEventListener('blur', function() {\n        //     self.stopEditing();\n        // });\n        input.style.position = 'absolute';\n        input.style.display = 'none';\n        input.style.border = 'solid 2px black';\n        input.style.outline = 0;\n        input.style.padding = 0;\n        input.style.zIndex = 1000;\n        //input.style.fontSize = '8pt';\n        input.style.boxShadow = 'white 0px 0px 1px 1px';\n    },\n});\n\nmodule.exports = Textfield;\n","'use strict';\n\nmodule.exports = {\n    CellEditor: require('./CellEditor'), // abstract base class\n    Textfield: require('./Textfield'),\n    Choice: require('./Choice'),\n    //Combo: require('./Combo'),\n    Color: require('./Color'),\n    Date: require('./Date'),\n    Slider: require('./Slider'),\n    Spinner: require('./Spinner'),\n    Filter: require('./Filter')\n};\n","'use strict';\n\nvar Base = require('extend-me').Base;\n\nvar A = 'A'.charCodeAt(0);\n\n/**\n * @constructor\n */\nvar DataModel = Base.extend('DataModel', {\n\n    next: null,\n\n    grid: null,\n\n    setGrid: function(newGrid) {\n        this.grid = newGrid;\n    },\n\n    getGrid: function() {\n        return this.grid;\n    },\n\n    getBehavior: function() {\n        return this.getGrid().getBehavior();\n    },\n\n    changed: function() {\n        this.getBehavior().changed();\n    },\n\n    getPrivateState: function() {\n        return this.getGrid().getPrivateState();\n    },\n\n    applyState: function() {\n\n    },\n\n    alphaFor: function(i) {\n        // Name the column headers in A, .., AA, AB, AC, .., AZ format\n        // quotient/remainder\n        //var quo = Math.floor(col/27);\n        var quo = Math.floor(i / 26);\n        var rem = i % 26;\n        var code = '';\n        if (quo > 0) {\n            code += this.alpha(quo - 1);\n        }\n        code += this.alpha(rem);\n        return code;\n    },\n\n    alpha: function(i) {\n        return String.fromCharCode(A + i);\n    },\n\n    getCellEditorAt: function(x, y) {\n    },\n\n});\n\nmodule.exports = DataModel;\n","'use strict';\n\n//var analytics = require('hyper-analytics');\n//var analytics = require('../local_node_modules/hyper-analytics');\nvar analytics = require('../local_node_modules/finanalytics');\nvar DataModel = require('./DataModel');\nvar images = require('../../images');\n\nvar UPWARDS_BLACK_ARROW = '\\u25b2', // aka '▲'\n    DOWNWARDS_BLACK_ARROW = '\\u25bc'; // aka '▼'\n\nvar nullDataSource = {\n    isNullObject: function() {\n        return true;\n    },\n    getFields: function() {\n        return [];\n    },\n    getHeaders: function() {\n        return [];\n    },\n    getColumnCount: function() {\n        return 0;\n    },\n    getRowCount: function() {\n        return 0;\n    },\n    getAggregateTotals: function() {\n        return [];\n    },\n    hasAggregates: function() {\n        return false;\n    },\n    hasGroups: function() {\n        return false;\n    },\n    getRow: function() {\n        return null;\n    }\n};\n\n/**\n * @name dataModels.JSON\n * @constructor\n */\nvar JSON = DataModel.extend('dataModels.JSON', {\n\n    //null object pattern for the source object\n    source: nullDataSource,\n\n    preglobalfilter: nullDataSource,\n    prefilter: nullDataSource,\n\n    presorter: nullDataSource,\n    analytics: nullDataSource,\n    postglobalfilter: nullDataSource,\n    postfilter: nullDataSource,\n    postsorter: nullDataSource,\n\n    topTotals: [],\n    bottomTotals: [],\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasAggregates: function() {\n        return this.analytics.hasAggregates();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasGroups: function() {\n        return this.analytics.hasGroups();\n    },\n\n    getDataSource: function() {\n        return this.postsorter; //this.hasAggregates() ? this.analytics : this.presorter;\n    },\n\n    getFilterSource: function() {\n        return this.postfilter; //this.hasAggregates() ? this.postfilter : this.prefilter;\n    },\n\n    getGlobalFilterSource: function() {\n        return this.postglobalfilter; //this.hasAggregates() ? this.postfilter : this.prefilter;\n    },\n\n    getSortingSource: function() {\n        return this.postsorter; //this.hasAggregates() ? this.postsorter : this.presorter;\n    },\n\n    getData: function() {\n        return this.source.data;\n    },\n\n    getFilteredData: function() {\n        var ds = this.getDataSource();\n        var count = ds.getRowCount();\n        var result = new Array(count);\n        for (var y = 0; y < count; y++) {\n            result[y] = ds.getRow(y);\n        }\n        return result;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @returns {*}\n     */\n    getValue: function(x, y) {\n        var hasHierarchyColumn = this.hasHierarchyColumn();\n        var grid = this.getGrid();\n        var headerRowCount = grid.getHeaderRowCount();\n        var value;\n        if (hasHierarchyColumn) {\n            if (x === -2) {\n                x = 0;\n            }\n        } else if (this.hasAggregates()) {\n            x += 1;\n        }\n        if (y < headerRowCount) {\n            value = this.getHeaderRowValue(x, y);\n            return value;\n        }\n        // if (hasHierarchyColumn) {\n        //     y += 1;\n        // }\n        value = this.getDataSource().getValue(x, y - headerRowCount);\n        return value;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y - negative values refer to _bottom totals_ rows\n     * @returns {*}\n     */\n    getHeaderRowValue: function(x, y) {\n        var value;\n        if (y === undefined) {\n            value = this.getHeaders()[Math.max(x, 0)];\n        } else if (y < 0) { // bottom totals rows\n            var bottomTotals = this.getBottomTotals();\n            value = bottomTotals[bottomTotals.length + y][x];\n        } else {\n            var grid = this.getGrid(),\n                isFilterRow = grid.isShowFilterRow(),\n                isHeaderRow = grid.isShowHeaderRow(),\n                topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0);\n            if (y >= topTotalsOffset) { // top totals rows\n                value = this.getTopTotals()[y - topTotalsOffset][x];\n            } else if (isHeaderRow && y === 0) {\n                value = this.getHeaders()[x];\n                var sortString = this.getSortImageForColumn(x);\n                if (sortString) { value = sortString + value; }\n            } else { // must be filter row\n                value = this.getFilter(x);\n                var icon = images.filter(value.length);\n                return [null, value, icon];\n            }\n        }\n        return value;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @param value\n     */\n    setValue: function(x, y, value) {\n        var hasHierarchyColumn = this.hasHierarchyColumn();\n        var grid = this.getGrid();\n        var headerRowCount = grid.getHeaderRowCount();\n        if (hasHierarchyColumn) {\n            if (x === -2) {\n                x = 0;\n            }\n        } else if (this.hasAggregates()) {\n            x += 1;\n        }\n        if (y < headerRowCount) {\n            this.setHeaderRowValue(x, y, value);\n        } else {\n            this.getDataSource().setValue(x, y - headerRowCount, value);\n        }\n        this.changed();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @param value\n     * @returns {*}\n     */\n    setHeaderRowValue: function(x, y, value) {\n        if (value === undefined) {\n            return this._setHeader(x, y); // y is really the value\n        }\n        var grid = this.getGrid();\n        var isFilterRow = grid.isShowFilterRow();\n        var isHeaderRow = grid.isShowHeaderRow();\n        var isBoth = isFilterRow && isHeaderRow;\n        var topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0);\n        if (y >= topTotalsOffset) {\n            this.getTopTotals()[y - topTotalsOffset][x] = value;\n        } else if (x === -1) {\n            return; // can't change the row numbers\n        } else if (isBoth) {\n            if (y === 0) {\n                return this._setHeader(x, value);\n            } else {\n                this.setFilter(x, value);\n            }\n        } else if (isFilterRow) {\n            this.setFilter(x, value);\n        } else {\n            return this._setHeader(x, value);\n        }\n        return '';\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @returns {*}\n     */\n    getColumnProperties: function(colIndex) {\n        //access directly because we want it ordered\n        var column = this.getBehavior().allColumns[colIndex];\n        if (column) {\n            return column.getProperties();\n        }\n        return undefined;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @returns {*}\n     */\n    getFilter: function(colIndex) {\n        var columnProperties = this.getColumnProperties(colIndex);\n        if (!columnProperties) {\n            return '';\n        }\n        return columnProperties.filter || '';\n    },\n\n    getComplexFilter: function(colIndex) {\n        var columnProperties = this.getColumnProperties(colIndex);\n        if (!columnProperties) {\n            return '';\n        }\n        var filter = columnProperties.complexFilter;\n        if (filter) {\n            var filterObject = this.getGrid().resolveFilter(filter.type);\n            var newFilter = filterObject.create(filter.state);\n            return function(data) {\n                var transformed = valueOrFunctionExecute(data);\n                return newFilter(transformed);\n            };\n        } else {\n            return;\n        }\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param value\n     */\n    setFilter: function(colIndex, value) {\n        var columnProperties = this.getColumnProperties(colIndex);\n        columnProperties.filter = value;\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {number}\n     */\n    getColumnCount: function() {\n        var showTree = this.getGrid().resolveProperty('showTreeColumn') === true;\n        var hasAggregates = this.hasAggregates();\n        var offset = (hasAggregates && !showTree) ? -1 : 0;\n        return this.analytics.getColumnCount() + offset;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {number}\n     */\n    getRowCount: function() {\n        var grid = this.getGrid();\n        var count = this.getDataSource().getRowCount();\n        count += grid.getHeaderRowCount();\n        return count;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {string[]}\n     */\n    getHeaders: function() {\n        return this.analytics.getHeaders();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string[]} headers\n     */\n    setHeaders: function(headers) {\n        this.getDataSource().setHeaders(headers);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string[]} fields\n     */\n    setFields: function(fields) {\n        this.getDataSource().setFields(fields);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {string[]}\n     */\n    getFields: function() {\n        return this.getDataSource().getFields();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {object[]} dataRows\n     */\n    setData: function(dataRows) {\n        this.source = new analytics.JSDataSource(dataRows);\n        //this.preglobalfilter = new analytics.DataSourceGlobalFilter(this.source);\n        //this.prefilter = new analytics.DataSourceFilter(this.preglobalfilter);\n        //this.presorter = new analytics.DataSourceSorterComposite(this.prefilter);\n\n        this.analytics = new analytics.DataSourceAggregator(this.source);\n\n        this.postglobalfilter = new analytics.DataSourceGlobalFilter(this.analytics);\n        this.postfilter = new analytics.DataSourceFilter(this.postglobalfilter);\n        this.postsorter = new analytics.DataSourceSorterComposite(this.postfilter);\n\n        this.applyAnalytics();\n\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {Array<Array>} totalRows\n     */\n    setTopTotals: function(totalRows) {\n        this.topTotals = totalRows;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getTopTotals: function() {\n        if (!this.hasAggregates()) {\n            return this.topTotals;\n        }\n        return this.getDataSource().getGrandTotals();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {Array<Array>} totalRows\n     */\n    setBottomTotals: function(totalRows) {\n        this.bottomTotals = totalRows;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getBottomTotals: function() {\n        if (!this.hasAggregates()) {\n            return this.bottomTotals;\n        }\n        return this.getDataSource().getGrandTotals();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param groups\n     */\n    setGroups: function(groups) {\n        this.analytics.setGroupBys(groups);\n        this.applyAnalytics();\n        this.getGrid().fireSyntheticGroupsChangedEvent(this.getGroups());\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getGroups: function() {\n        var headers = this.getHeaders().slice(0);\n        var fields = this.getFields().slice(0);\n        var groupBys = this.analytics.groupBys;\n        var groups = [];\n        for (var i = 0; i < groupBys.length; i++) {\n            var field = headers[groupBys[i]];\n            groups.push({\n                id: groupBys[i],\n                label: field,\n                field: fields\n            });\n        }\n        return groups;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getAvailableGroups: function() {\n        var headers = this.source.getHeaders().slice(0);\n        var groupBys = this.analytics.groupBys;\n        var groups = [];\n        for (var i = 0; i < headers.length; i++) {\n            if (groupBys.indexOf(i) === -1) {\n                var field = headers[i];\n                groups.push({\n                    id: i,\n                    label: field,\n                    field: field\n                });\n            }\n        }\n        return groups;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getVisibleColumns: function() {\n        var items = this.getBehavior().columns;\n        items = items.filter(function(each) {\n            return each.label !== 'Tree';\n        });\n        return items;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getHiddenColumns: function() {\n        var visible = this.getBehavior().columns;\n        var all = this.getBehavior().allColumns;\n        var hidden = [];\n        for (var i = 0; i < all.length; i++) {\n            if (visible.indexOf(all[i]) === -1) {\n                hidden.push(all[i]);\n            }\n        }\n        hidden.sort(function(a, b) {\n            return a.label < b.label;\n        });\n        return hidden;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param aggregations\n     */\n    setAggregates: function(aggregations) {\n        this.quietlySetAggregates(aggregations);\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param aggregations\n     */\n    quietlySetAggregates: function(aggregations) {\n        this.analytics.setAggregates(aggregations);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasHierarchyColumn: function() {\n        var showTree = this.getGrid().resolveProperty('showTreeColumn') === true;\n        return this.hasAggregates() && this.hasGroups() && showTree;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyAnalytics: function() {\n        this.applyGroupBysAndAggregations();\n        this.applyFilters();\n        this.applySorts();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyGroupBysAndAggregations: function() {\n        if (this.analytics.aggregates.length === 0) {\n            this.quietlySetAggregates({});\n        }\n        this.analytics.apply();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyFilters: function() {\n        var grid = this.getGrid();\n        var visibleColumns = this.getVisibleColumns();\n        this.getGlobalFilterSource().apply(visibleColumns);\n        var details = [];\n        var filterSource = this.getFilterSource();\n        var groupOffset = 0; //this.hasHierarchyColumn() ? 0 : 1;\n\n        // apply column filters\n        filterSource.clearAll();\n\n        visibleColumns.forEach(function(column) {\n            var columnIndex = column.index,\n                filterText = this.getFilter(columnIndex),\n                formatterType = column.getProperties().format,\n                formatter = grid.getFormatter(formatterType),\n                complexFilter = this.getComplexFilter(columnIndex),\n                filter = complexFilter || filterText.length > 0 && textMatchFilter(filterText);\n\n            if (filter) {\n                filterSource.add(columnIndex - groupOffset, this.createFormattedFilter(formatter, filter));\n                details.push({\n                    column: column.label,\n                    format: complexFilter ? 'complex' : formatterType\n                });\n            }\n        }.bind(this));\n\n        filterSource.applyAll();\n\n        grid.fireSyntheticFilterAppliedEvent({\n            details: details\n        });\n    },\n\n    createFormattedFilter: function(formatter, filter) {\n        return function(value) {\n            var formattedValue = formatter(value);\n            return filter(formattedValue);\n        };\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param keys\n     */\n    toggleSort: function(colIndex, keys) {\n        this.incrementSortState(colIndex, keys);\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param {string[]} keys\n     */\n    incrementSortState: function(colIndex, keys) {\n        colIndex++; //hack to get around 0 index\n        var state = this.getPrivateState();\n        var hasCTRL = keys.indexOf('CTRL') > -1;\n        state.sorts = state.sorts || [];\n        var already = state.sorts.indexOf(colIndex);\n        if (already === -1) {\n            already = state.sorts.indexOf(-1 * colIndex);\n        }\n        if (already > -1) {\n            if (state.sorts[already] > 0) {\n                state.sorts[already] = -1 * state.sorts[already];\n            } else {\n                state.sorts.splice(already, 1);\n            }\n        } else if (hasCTRL || state.sorts.length === 0) {\n            state.sorts.unshift(colIndex);\n        } else {\n            state.sorts.length = 0;\n            state.sorts.unshift(colIndex);\n        }\n        if (state.sorts.length > 3) {\n            state.sorts.length = 3;\n        }\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applySorts: function() {\n        var sortingSource = this.getSortingSource();\n        var sorts = this.getPrivateState().sorts;\n        var groupOffset = this.hasAggregates() ? 1 : 0;\n        if (!sorts || sorts.length === 0) {\n            sortingSource.clearSorts();\n        } else {\n            for (var i = 0; i < sorts.length; i++) {\n                var colIndex = Math.abs(sorts[i]) - 1;\n                var type = sorts[i] < 0 ? -1 : 1;\n                sortingSource.sortOn(colIndex - groupOffset, type);\n            }\n        }\n        sortingSource.applySorts();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param index\n     * @param returnAsString\n     * @returns {*}\n     */\n    getSortImageForColumn: function(index) {\n        index++;\n        var up = true;\n        var sorts = this.getPrivateState().sorts;\n        if (!sorts) {\n            return null;\n        }\n        var position = sorts.indexOf(index);\n        if (position < 0) {\n            position = sorts.indexOf(-1 * index);\n            up = false;\n        }\n        if (position < 0) {\n            return null;\n        }\n        var rank = sorts.length - position;\n        var arrow = up ? UPWARDS_BLACK_ARROW : DOWNWARDS_BLACK_ARROW;\n        return rank + arrow + ' ';\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param cell\n     * @param event\n     */\n    cellClicked: function(cell, event) {\n        if (!this.hasAggregates()) {\n            return;\n        }\n        if (event.gridCell.x !== 0) {\n            return; // this wasn't a click on the hierarchy column\n        }\n        var grid = this.getGrid();\n        var headerRowCount = grid.getHeaderRowCount();\n        var y = event.gridCell.y - headerRowCount;\n        this.getDataSource().click(y);\n        this.applyFilters();\n        this.applySorts();\n        this.changed();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    getRow: function(y) {\n        var grid = this.getGrid();\n        var headerRowCount = grid.getHeaderRowCount();\n        if (y < headerRowCount && !this.hasAggregates()) {\n            var topTotals = this.getTopTotals();\n            return topTotals[y - (headerRowCount - topTotals.length)];\n        }\n        return this.getDataSource().getRow(y - headerRowCount);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    buildRow: function(y) {\n        var colCount = this.getColumnCount();\n        var fields = [].concat(this.getFields());\n        var result = {};\n        if (this.hasAggregates()) {\n            result.tree = this.getValue(-2, y);\n            fields.shift();\n        }\n        for (var i = 0; i < colCount; i++) {\n            result[fields[i]] = this.getValue(i, y);\n        }\n        return result;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    getComputedRow: function(y) {\n        var rcf = this.getRowContextFunction([y]);\n        var fields = this.getFields();\n        var row = {};\n        for (var i = 0; i < fields.length; i++) {\n            var field = fields[i];\n            row[field] = rcf(field)[0];\n        }\n        return row;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string} fieldName\n     * @param {number} y\n     * @returns {*}\n     */\n    getValueByField: function(fieldName, y) {\n        var index = this.getFields().indexOf(fieldName);\n        if (this.hasAggregates()) {\n            y += 1;\n        }\n        return this.getDataSource().getValue(index, y);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {sring} string\n     */\n    setGlobalFilter: function(string) {\n        var globalFilterSource = this.getGlobalFilterSource();\n        if (!string || string.length === 0) {\n            globalFilterSource.clear();\n        } else {\n            globalFilterSource.set(textMatchFilter(string));\n        }\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {object} config\n     * @param {number} x\n     * @param {number} y\n     * @param {number} untranslatedX\n     * @param {number} untranslatedY\n     * @returns {object}\n     */\n    getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) {\n        var renderer;\n        var provider = this.getGrid().getCellProvider();\n\n        config.x = x;\n        config.y = y;\n        config.untranslatedX = untranslatedX;\n        config.untranslatedY = untranslatedY;\n\n        renderer = provider.getCell(config);\n        renderer.config = config;\n\n        return renderer;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyState: function() {\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    reset: function() {\n        this.setData([]);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        return this.source.getValue(x, y);\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.source.getRowCount();\n    },\n\n});\n\nfunction valueOrFunctionExecute(valueOrFunction) {\n    return typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction;\n}\n\nfunction textMatchFilter(string) {\n    string = string.toLowerCase();\n    return function(each) {\n        each = valueOrFunctionExecute(each);\n        return (each + '').toLowerCase().indexOf(string) > -1;\n    };\n}\n\nmodule.exports = JSON;\n","/* eslint-env browser */\n\n'use strict';\n\nvar LRUCache = require('lru-cache');\n\nvar renderCellError = require('./renderCellError');\n\n/**\n * This module lists the properties that can be set on a {@link Hypergrid} along with their default values.\n * Edit this file to override the defaults.\n * @module defaults\n */\n\nmodule.exports = {\n\n    //these are for the theme\n\n    /**\n     * The font for data cells.\n     * @default '13px Tahoma, Geneva, sans-serif'\n     * @type {string}\n     * @instance\n     */\n    font: '13px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    color: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(241, 241, 241)'\n     * @type {string}\n     * @instance\n     */\n    backgroundColor: 'rgb(241, 241, 241)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    foregroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(183, 219, 255)'\n     * @type {string}\n     * @instance\n     */\n    backgroundSelectionColor: 'rgb(183, 219, 255)',\n\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 220, 97)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderBackgroundSelectionColor: 'rgb(255, 220, 97)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderForegroundColumnSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {string}\n     * @instance\n     */\n    columnHeaderBackgroundColumnSelectionColor: 'rgb(255, 180, 0)',\n\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 220, 97)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderBackgroundSelectionColor: 'rgb(255, 220, 97)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderForegroundRowSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {string}\n     * @instance\n     */\n    rowHeaderBackgroundRowSelectionColor: 'rgb(255, 180, 0)',\n\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {string}\n     * @instance\n     */\n    filterFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    filterColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'white'\n     * @type {string}\n     * @instance\n     */\n    filterBackgroundColor: 'white',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    filterForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 220, 97)'\n     * @type {string}\n     * @instance\n     */\n    filterBackgroundSelectionColor: 'rgb(255, 220, 97)',\n\n    /**\n     * @default 'rgba(0,0,0,0.8)'\n     * @type {string}\n     * @instance\n     */\n    filterCellBorderStyle: 'rgba(0,0,0,0.8)',\n\n    /**\n     * @default 0.4\n     * @type {number}\n     * @instance\n     */\n    filterCellBorderThickness: 0.4,\n\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {string}\n     * @instance\n     */\n    treeColumnFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 220, 97)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnBackgroundSelectionColor: 'rgb(255, 220, 97)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnForegroundColumnSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {string}\n     * @instance\n     */\n    treeColumnBackgroundColumnSelectionColor: 'rgb(255, 180, 0)',\n\n    /**\n     * @default 'rgb(201, 201, 201)'\n     * @type {string}\n     * @instance\n     */\n    backgroundColor2: 'rgb(201, 201, 201)',\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    voffset: 0,\n\n    /**\n     * @default 'visible'\n     * @type {string}\n     * @instance\n     */\n    scrollbarHoverOver: 'visible',\n\n    /**\n     * @default 'hidden'\n     * @type {string}\n     * @instance\n     */\n    scrollbarHoverOff: 'hidden',\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    scrollingEnabled: true,\n\n    /**\n     * @default ''\n     * @type {string}\n     * @instance\n     */\n    vScrollbarClassPrefix: '',\n\n    /**\n     * @default ''\n     * @type {string}\n     * @instance\n     */\n    hScrollbarClassPrefix: '',\n\n    //these used to be in the constants element\n\n    /**\n     * @default 'center'\n     * @type {string}\n     * @instance\n     */\n    fixedRowAlign: 'center',\n\n    /**\n     * @default 'center'\n     * @type {string}\n     * @instance\n     */\n    fixedColAlign: 'center',\n\n    /**\n     * @default 5\n     * @type {number}\n     * @instance\n     */\n    cellPadding: 5,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    gridLinesH: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    gridLinesV: true,\n\n    /**\n     * @default 'rgb(199, 199 199)'\n     * @type {string}\n     * @instance\n     */\n    lineColor: 'rgb(199, 199, 199)',\n\n    /**\n     * @default 0.4\n     * @type {number}\n     * @instance\n     */\n    lineWidth: 0.4,\n\n\n    /**\n     * @default 15\n     * @type {number}\n     * @instance\n     */\n    defaultRowHeight: 15,\n\n    /**\n     * @default 100\n     * @type {number}\n     * @instance\n     */\n    defaultColumnWidth: 100,\n\n    //for immediate painting, set these values to 0, true respectively\n\n    /**\n     * @default 60\n     * @type {number}\n     * @instance\n     */\n    repaintIntervalRate: 60,\n\n    /**\n     *\n     * @instance\n     */\n    repaintImmediately: false,\n\n    //enable or disable double buffering\n\n    /**\n     * @default false\n     * @type {boolean}\n     * @instance\n     */\n    useBitBlit: false,\n\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    useHiDPI: true,\n\n    /**\n     * @default ['alt', 'esc']\n     * @type {string}\n     * @instance\n     */\n    editorActivationKeys: ['alt', 'esc'],\n\n    /**\n     * @default false\n     * @type {boolean}\n     * @instance\n     */\n    readOnly: false,\n\n    //inhertied by cell renderers\n\n    /**\n     * @default getTextWidth\n     * @type {function}\n     * @instance\n     */\n    getTextWidth: getTextWidth,\n\n    /**\n     * @default getTextHeight\n     * @type {function}\n     * @instance\n     */\n    getTextHeight: getTextHeight,\n\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    fixedColumnCount: 0,\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    fixedRowCount: 0,\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    headerColumnCount: 0,\n\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    showRowNumbers: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    showTreeColumn: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    showHeaderRow: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    showFilterRow: true,\n\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    cellSelection: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    columnSelection: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    rowSelection: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    singleRowSelectionMode: true,\n\n    /**\n     * @default 'rgba(0, 0, 0, 0.2)'\n     * @type {string}\n     * @instance\n     */\n    selectionRegionOverlayColor: 'rgba(0, 0, 0, 0.2)',\n\n    /**\n     * @default 'black'\n     * @type {string}\n     * @instance\n     */\n    selectionRegionOutlineColor: 'black',\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    columnAutosizing: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    rowNumberAutosizing: true,\n\n    /**\n     * @default false\n     * @type {boolean}\n     * @instance\n     */\n    headerTextWrapping: false,\n\n    /**\n     * @default false\n     * @type {boolean}\n     * @instance\n     */\n    rowResize: false,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    editable: true,\n\n    /**\n     * @default true\n     * @type {boolean}\n     * @instance\n     */\n    editOnDoubleClick: true,\n\n    /**\n     * @default renderCellError\n     * @type {function}\n     */\n    renderCellError: renderCellError,\n\n    /**\n     * @default false\n     * @type {boolean}\n     */\n    checkboxOnlyRowSelections: false,\n\n    format: 'default',\n\n    hoverCellHighlight: true,\n    hoverRowHighlight: true,\n    hoverColumnHighlight: true,\n\n    hoverCellColor: 'lightgray',\n    hoverRowColor: 'gray',\n    hoverColumnColor: 'gray',\n\n    link: false,\n    strikeThrough: false,\n\n};\n\nvar textWidthCache = new LRUCache(2000);\n\nfunction getTextWidth(gc, string) {\n    if (string === null || string === undefined) {\n        return 0;\n    }\n    string = string + '';\n    if (string.length === 0) {\n        return 0;\n    }\n    var key = gc.font + string;\n    var width = textWidthCache.get(key);\n    if (!width) {\n        width = gc.measureText(string).width;\n        textWidthCache.set(key, width);\n    }\n    return width;\n}\n\nvar fontData = {};\n\nfunction getTextHeight(font) {\n    var result = fontData[font];\n\n    if (!result) {\n        result = {};\n\n        var text = document.createElement('span');\n        text.textContent = 'Hg';\n        text.style.font = font;\n\n        var block = document.createElement('div');\n        block.style.display = 'inline-block';\n        block.style.width = '1px';\n        block.style.height = '0px';\n\n        var div = document.createElement('div');\n        div.appendChild(text);\n        div.appendChild(block);\n\n        div.style.position = 'absolute';\n        document.body.appendChild(div);\n\n        try {\n\n            block.style.verticalAlign = 'baseline';\n\n            var blockRect = block.getBoundingClientRect();\n            var textRect = text.getBoundingClientRect();\n\n            result.ascent = blockRect.top - textRect.top;\n\n            block.style.verticalAlign = 'bottom';\n            result.height = blockRect.top - textRect.top;\n\n            result.descent = result.height - result.ascent;\n\n        } finally {\n            document.body.removeChild(div);\n        }\n        if (result.height !== 0) {\n            fontData[font] = result;\n        }\n    }\n\n    return result;\n}\n","/* eslint-env browser */\n\n'use strict';\n\nrequire('object-iterators'); // Install the Array.find polyfill, as needed\n\nvar Hypergrid = require('./Hypergrid');\n\nHypergrid.images = require('../images');\nHypergrid.behaviors = require('./behaviors/index');\nHypergrid.cellEditors = require('./cellEditors/index');\nHypergrid.features = require('./features/index');\n\nwindow.fin = {\n    Hypergrid: Hypergrid,\n    FilterTree: require('filter-tree')\n};\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellClick = Feature.extend('CellClick', {\n\n    alias: 'CellClick',\n\n    /**\n     * @memberOf CellClick.prototype\n     * @desc Handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleTap: function(grid, event) {\n        var gridCell = event.gridCell;\n        var behavior = grid.getBehavior();\n        var headerRowCount = behavior.getHeaderRowCount();\n        var headerColumnCount = behavior.getHeaderColumnCount();\n        if ((gridCell.y >= headerRowCount) &&\n            (gridCell.x >= headerColumnCount)) {\n            grid.cellClicked(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    }\n});\n\nmodule.exports = CellClick;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellEditing = Feature.extend('CellEditing', {\n\n    alias: 'CellEditing',\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDoubleClick: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    handleTap: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n           grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) {\n        var behavior = grid.getBehavior();\n        var headerRowCount = behavior.getHeaderRowCount();\n        var headerColumnCount = behavior.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        var isFilterRow = grid.isFilterRow(gridCell.y);\n        var activateEditor = isDoubleClickEditorActivation && gridCell.x >= headerColumnCount && (isFilterRow || gridCell.y >= headerRowCount);\n        return activateEditor;\n    }\n\n});\n\nmodule.exports = CellEditing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellSelection = Feature.extend('CellSelection', {\n\n    alias: 'CellSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {window.fin.rectangular.Point}\n     * @memberOf CellSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * the cell coordinates of the where the mouse pointer is during a drag operation\n     * @type {Object}\n     * @memberOf CellSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf CellSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf CellSelection.prototype\n     */\n    sbAutoStart: 0,\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragging) {\n            this.dragging = false;\n        }\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var behavior = grid.getBehavior();\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n        var headerRowCount = behavior.getHeaderRowCount();\n        var headerColumnCount = behavior.getHeaderColumnCount();\n        var columnCount = behavior.getColumnCount();\n        var isOutside = viewCell.x >= columnCount;\n\n        var isHeader = dy < headerRowCount || dx < headerColumnCount;\n\n        if (!grid.isCellSelection() || isRightClick || isHeader || isOutside) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n            var numFixedRows = grid.getFixedRowCount();\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            var keys = primEvent.detail.keys;\n            this.dragging = true;\n            this.extendSelection(grid, dCell, keys);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (!grid.isCellSelection() || isRightClick || !this.dragging) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n            var numFixedRows = grid.getFixedRowCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle a mousedrag selection.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n\n        var behavior = grid.getBehavior();\n        var headerRowCount = behavior.getHeaderRowCount();\n        var headerColumnCount = behavior.getHeaderColumnCount();\n        var x = gridCell.x;\n        var y = gridCell.y;\n        x = Math.max(headerColumnCount, x);\n        y = Math.max(headerRowCount, y);\n\n        var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        //var scrollingNow = grid.isScrollingNow();\n\n        var newX = x - mouseDown.x;\n        var newY = y - mouseDown.y;\n\n        if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n            return;\n        }\n\n        grid.clearMostRecentSelection();\n\n        grid.select(mouseDown.x, mouseDown.y, newX, newY);\n        grid.setDragExtent(grid.newPoint(newX, newY));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var dragStartedInHeaderArea = grid.isMouseDownInHeaderArea();\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (!dragStartedInHeaderArea) {\n            if (this.currentDrag.x < b.origin.x) {\n                xOffset = -1;\n            }\n            if (this.currentDrag.y < b.origin.y) {\n                yOffset = -1;\n            }\n        }\n        if (this.currentDrag.x > b.origin.x + b.extent.x) {\n            xOffset = 1;\n        }\n        if (this.currentDrag.y > b.origin.y + b.extent.y) {\n            yOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        var hasCTRL = keys.indexOf('CTRL') !== -1;\n        var hasSHIFT = keys.indexOf('SHIFT') !== -1;\n        // var scrollTop = grid.getVScrollValue();\n        // var scrollLeft = grid.getHScrollValue();\n\n        // var numFixedColumns = 0;//grid.getFixedColumnCount();\n        // var numFixedRows = 0;//grid.getFixedRowCount();\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        //we have repeated a click in the same spot deslect the value from last time\n        if (x === mousePoint.x && y === mousePoint.y) {\n            grid.clearMostRecentSelection();\n            grid.popMouseDown();\n            grid.repaint();\n            return;\n        }\n\n        if (!hasCTRL && !hasSHIFT) {\n            grid.clearSelections();\n        }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentSelection();\n            grid.select(mousePoint.x, mousePoint.y, x - mousePoint.x + 1, y - mousePoint.y + 1);\n            grid.setDragExtent(grid.newPoint(x - mousePoint.x + 1, y - mousePoint.y));\n        } else {\n            grid.select(x, y, 0, 0);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 0, 1);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 0, -1);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid, event) {\n        //keep the browser viewport from auto scrolling on key event\n        event.primitiveEvent.preventDefault();\n\n        var count = this.getAutoScrollAcceleration();\n        this.moveSingleSelect(grid, 0, count);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid, event) {\n        //keep the browser viewport from auto scrolling on key event\n        event.primitiveEvent.preventDefault();\n\n        var count = this.getAutoScrollAcceleration();\n        this.moveSingleSelect(grid, 0, -count);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {\n        this.moveSingleSelect(grid, -1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n        this.moveSingleSelect(grid, 1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetX, offsetY) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumns() - 1;\n        var maxViewableRows = grid.getVisibleRows() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newX = extent.x + offsetX;\n        var newY = extent.y + offsetY;\n\n        newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX));\n        newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY));\n\n        grid.clearMostRecentSelection();\n        grid.select(origin.x, origin.y, newX, newY);\n\n        grid.setDragExtent(grid.newPoint(newX, newY));\n\n        if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) {\n            this.pingAutoScroll();\n        }\n        if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetX, offsetY) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumnsCount() - 1;\n        var maxViewableRows = grid.getVisibleRowsCount() - 1;\n\n        var minRows = grid.getHeaderRowCount();\n        var minCols = grid.getHeaderColumnCount();\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newX = mouseCorner.x + offsetX;\n        var newY = mouseCorner.y + offsetY;\n\n        newX = Math.min(maxColumns, Math.max(minCols, newX));\n        newY = Math.min(maxRows, Math.max(minRows, newY));\n\n        grid.clearSelections();\n        grid.select(newX, newY, 0, 0);\n        grid.setMouseDown(grid.newPoint(newX, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelColIsVisible(newX, offsetX)) {\n            this.pingAutoScroll();\n        }\n        if (grid.insureModelRowIsVisible(newY, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    }\n\n});\n\nmodule.exports = CellSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnAutosizing = Feature.extend('ColumnAutosizing', {\n\n    alias: 'ColumnAutosizing',\n\n    /**\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf ColumnAutosizing.prototype\n     */\n    handleDoubleClick: function(grid, event) {\n        var headerRowCount = grid.getHeaderRowCount();\n        //var headerColCount = grid.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        if (gridCell.y <= headerRowCount) {\n            grid.autosizeColumn(gridCell.x);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = ColumnAutosizing;\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\n// This feature is responsible for column drag and drop reordering.\n// This object is a mess and desperately needs a complete rewrite.....\n\nvar Feature = require('./Feature.js');\n\nvar columnAnimationTime = 150;\nvar dragger;\nvar draggerCTX;\nvar floatColumn;\nvar floatColumnCTX;\n\n/**\n * @constructor\n */\nvar ColumnMoving = Feature.extend('ColumnMoving', {\n\n    alias: 'ColumnMoving',\n\n    /**\n     * queue up the animations that need to play so they are done synchronously\n     * @type {Array}\n     * @memberOf CellMoving.prototype\n     */\n    floaterAnimationQueue: [],\n\n    /**\n     * am I currently auto scrolling right\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    columnDragAutoScrollingRight: false,\n\n    /**\n     * am I currently auto scrolling left\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    columnDragAutoScrollingLeft: false,\n\n    /**\n     * is the drag mechanism currently enabled (\"armed\")\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    dragArmed: false,\n\n    /**\n     * am I dragging right now\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    dragging: false,\n\n    /**\n     * the column index of the currently dragged column\n     * @type {number}\n     * @memberOf CellMoving.prototype\n     */\n    dragCol: -1,\n\n    /**\n     * an offset to position the dragged item from the cursor\n     * @type {number}\n     * @memberOf CellMoving.prototype\n     */\n    dragOffset: 0,\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc give me an opportunity to initialize stuff on the grid\n     * @param {Hypergrid} grid\n     */\n    initializeOn: function(grid) {\n        this.isFloatingNow = false;\n        this.initializeAnimationSupport(grid);\n        if (this.next) {\n            this.next.initializeOn(grid);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc initialize animation support on the grid\n     * @param {Hypergrid} grid\n     */\n    initializeAnimationSupport: function(grid) {\n        if (!dragger) {\n            dragger = document.createElement('canvas');\n            dragger.setAttribute('width', '0px');\n            dragger.setAttribute('height', '0px');\n\n            document.body.appendChild(dragger);\n            draggerCTX = dragger.getContext('2d');\n        }\n        if (!floatColumn) {\n            floatColumn = document.createElement('canvas');\n            floatColumn.setAttribute('width', '0px');\n            floatColumn.setAttribute('height', '0px');\n\n            document.body.appendChild(floatColumn);\n            floatColumnCTX = floatColumn.getContext('2d');\n        }\n\n    },\n\n    getCanDragCursorName: function() {\n        return '-webkit-grab';\n    },\n\n    getDraggingCursorName: function() {\n        return '-webkit-grabbing';\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n\n        var gridCell = event.gridCell;\n        var x;\n        //var y;\n\n        var distance = Math.abs(event.primitiveEvent.detail.dragstart.x - event.primitiveEvent.detail.mouse.x);\n\n        if (distance < 10) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n            return;\n        }\n\n        if (this.isHeaderRow(grid, event) && this.dragArmed && !this.dragging) {\n            this.dragging = true;\n            this.dragCol = gridCell.x;\n            this.dragOffset = event.mousePoint.x;\n            this.detachChain();\n            x = event.primitiveEvent.detail.mouse.x - this.dragOffset;\n            //y = event.primitiveEvent.detail.mouse.y;\n            this.createDragColumn(grid, x, this.dragCol);\n        } else if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n\n        if (this.dragging) {\n            x = event.primitiveEvent.detail.mouse.x - this.dragOffset;\n            //y = event.primitiveEvent.detail.mouse.y;\n            this.dragColumn(grid, x);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (grid.getBehavior().isColumnReorderable()) {\n            if (this.isHeaderRow(grid, event) && event.gridCell.x !== -1) {\n                this.dragArmed = true;\n                this.cursor = this.getDraggingCursorName();\n                grid.clearSelections();\n            }\n        }\n        if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        //var col = event.gridCell.x;\n        if (this.dragging) {\n            this.cursor = null;\n            //delay here to give other events a chance to be dropped\n            var self = this;\n            this.endDragColumn(grid);\n            setTimeout(function() {\n                self.attachChain();\n            }, 200);\n        }\n        this.dragCol = -1;\n        this.dragging = false;\n        this.dragArmed = false;\n        this.cursor = null;\n        grid.repaint();\n\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n\n        if (!this.dragging && event.mousePoint.y < 5 && event.viewPoint.y === 0) {\n            this.cursor = this.getCanDragCursorName();\n        } else {\n            this.cursor = null;\n        }\n\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n\n        if (this.isHeaderRow(grid, event) && this.dragging) {\n            this.cursor = this.getDraggingCursorName(); //move';\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc this is the main event handler that manages the dragging of the column\n     * @param {Hypergrid} grid\n     * @param {boolean} draggedToTheRight - are we moving to the right\n     */\n    floatColumnTo: function(grid, draggedToTheRight) {\n        this.floatingNow = true;\n\n        var renderer = grid.getRenderer();\n        var colEdges = renderer.getColumnEdges();\n        //var behavior = grid.getBehavior();\n        var scrollLeft = grid.getHScrollValue();\n        var floaterIndex = grid.renderOverridesCache.floater.columnIndex;\n        var draggerIndex = grid.renderOverridesCache.dragger.columnIndex;\n        var hdpiratio = grid.renderOverridesCache.dragger.hdpiratio;\n\n        var draggerStartX;\n        var floaterStartX;\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var draggerWidth = grid.getColumnWidth(draggerIndex);\n        var floaterWidth = grid.getColumnWidth(floaterIndex);\n\n        var max = grid.getVisibleColumnsCount();\n\n        var doffset = 0;\n        var foffset = 0;\n\n        if (draggerIndex >= fixedColumnCount) {\n            doffset = scrollLeft;\n        }\n        if (floaterIndex >= fixedColumnCount) {\n            foffset = scrollLeft;\n        }\n\n        if (draggedToTheRight) {\n            draggerStartX = colEdges[Math.min(max, draggerIndex - doffset)];\n            floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)];\n\n            grid.renderOverridesCache.dragger.startX = (draggerStartX + floaterWidth) * hdpiratio;\n            grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio;\n\n        } else {\n            floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)];\n            draggerStartX = floaterStartX + draggerWidth;\n\n            grid.renderOverridesCache.dragger.startX = floaterStartX * hdpiratio;\n            grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio;\n        }\n        grid.swapColumns(draggerIndex, floaterIndex);\n        grid.renderOverridesCache.dragger.columnIndex = floaterIndex;\n        grid.renderOverridesCache.floater.columnIndex = draggerIndex;\n\n\n        this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(grid, floaterStartX, draggerStartX));\n\n        this.doFloaterAnimation(grid);\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc manifest the column drag and drop animation\n     * @param {Hypergrid} grid\n     * @param {number} floaterStartX - the x start coordinate of the column underneath that floats behind the dragged column\n     * @param {number} draggerStartX - the x start coordinate of the dragged column\n     */\n    doColumnMoveAnimation: function(grid, floaterStartX, draggerStartX) {\n        var self = this;\n        return function() {\n            var d = floatColumn;\n            d.style.display = 'inline';\n            self.setCrossBrowserProperty(d, 'transform', 'translate(' + floaterStartX + 'px, ' + 0 + 'px)');\n\n            //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)';\n            //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)';\n\n            requestAnimationFrame(function() {\n                self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease');\n                self.setCrossBrowserProperty(d, 'transform', 'translate(' + draggerStartX + 'px, ' + -2 + 'px)');\n            });\n            grid.repaint();\n            //need to change this to key frames\n\n            setTimeout(function() {\n                self.setCrossBrowserProperty(d, 'transition', '');\n                grid.renderOverridesCache.floater = null;\n                grid.repaint();\n                self.doFloaterAnimation(grid);\n                requestAnimationFrame(function() {\n                    d.style.display = 'none';\n                    self.isFloatingNow = false;\n                });\n            }, columnAnimationTime + 50);\n        };\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc manifest the floater animation\n     * @param {Hypergrid} grid\n     */\n    doFloaterAnimation: function(grid) {\n        if (this.floaterAnimationQueue.length === 0) {\n            this.floatingNow = false;\n            grid.repaint();\n            return;\n        }\n        var animation = this.floaterAnimationQueue.pop();\n        animation();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc create the float column at columnIndex underneath the dragged column\n     * @param {Hypergrid} grid\n     * @param {number} columnIndex - the index of the column that will be floating\n     */\n    createFloatColumn: function(grid, columnIndex) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n\n        var columnWidth = grid.getColumnWidth(columnIndex);\n        var colHeight = grid.div.clientHeight;\n        var d = floatColumn;\n        var style = d.style;\n        var location = grid.div.getBoundingClientRect();\n\n        style.top = (location.top - 2) + 'px';\n        style.left = location.left + 'px';\n        style.position = 'fixed';\n\n        var hdpiRatio = grid.getHiDPI(floatColumnCTX);\n\n        d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px');\n        d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px');\n        style.boxShadow = '0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)';\n        style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px';\n        style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px';\n        style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor');\n        style.backgroundColor = renderer.resolveProperty('backgroundColor');\n\n        var startX = columnEdges[columnIndex - scrollLeft];\n        startX = startX * hdpiRatio;\n\n        floatColumnCTX.scale(hdpiRatio, hdpiRatio);\n\n        grid.renderOverridesCache.floater = {\n            columnIndex: columnIndex,\n            ctx: floatColumnCTX,\n            startX: startX,\n            width: columnWidth,\n            height: colHeight,\n            hdpiratio: hdpiRatio\n        };\n\n        style.zIndex = '4';\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -2 + 'px)');\n        style.cursor = this.getDraggingCursorName();\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc utility function for setting cross browser css properties\n     * @param {HTMLElement} element - descripton\n     * @param {string} property - the property\n     * @param {string} value - the value to assign\n     */\n    setCrossBrowserProperty: function(element, property, value) {\n        var uProperty = property[0].toUpperCase() + property.substr(1);\n        this.setProp(element, 'webkit' + uProperty, value);\n        this.setProp(element, 'Moz' + uProperty, value);\n        this.setProp(element, 'ms' + uProperty, value);\n        this.setProp(element, 'O' + uProperty, value);\n        this.setProp(element, property, value);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc utility function for setting properties on HTMLElements\n     * @param {HTMLElement} element - descripton\n     * @param {string} property - the property\n     * @param {string} value - the value to assign\n     */\n    setProp: function(element, property, value) {\n        if (property in element.style) {\n            element.style[property] = value;\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc create the dragged column at columnIndex above the floated column\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     * @param {number} columnIndex - the index of the column that will be floating\n     */\n    createDragColumn: function(grid, x, columnIndex) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n        var hdpiRatio = grid.getHiDPI(draggerCTX);\n        var columnWidth = grid.getColumnWidth(columnIndex);\n        var colHeight = grid.div.clientHeight;\n        var d = dragger;\n        var location = grid.div.getBoundingClientRect();\n        var style = d.style;\n\n        style.top = location.top + 'px';\n        style.left = location.left + 'px';\n        style.position = 'fixed';\n        style.opacity = 0.85;\n        style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)';\n        //style.zIndex = 100;\n        style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor');\n        style.backgroundColor = grid.renderer.resolveProperty('backgroundColor');\n\n        d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px');\n        d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px');\n\n        style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px';\n        style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px';\n\n        var startX = columnEdges[columnIndex - scrollLeft];\n        startX = startX * hdpiRatio;\n\n        draggerCTX.scale(hdpiRatio, hdpiRatio);\n\n        grid.renderOverridesCache.dragger = {\n            columnIndex: columnIndex,\n            ctx: draggerCTX,\n            startX: startX,\n            width: columnWidth,\n            height: colHeight,\n            hdpiratio: hdpiRatio\n        };\n\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, -5px)');\n        style.zIndex = '5';\n        style.cursor = this.getDraggingCursorName();\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc this function is the main dragging logic\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    dragColumn: function(grid, x) {\n\n        //TODO: this function is overly complex, refactor this in to something more reasonable\n        var self = this;\n        //var renderer = grid.getRenderer();\n        //var columnEdges = renderer.getColumnEdges();\n\n        var autoScrollingNow = this.columnDragAutoScrollingRight || this.columnDragAutoScrollingLeft;\n\n        var hdpiRatio = grid.getHiDPI(draggerCTX);\n\n        var dragColumnIndex = grid.renderOverridesCache.dragger.columnIndex;\n        var columnWidth = grid.renderOverridesCache.dragger.width;\n\n        var minX = 0; //grid.getFixedColumnsWidth();\n        var maxX = grid.renderer.getFinalVisableColumnBoundary() - columnWidth;\n        x = Math.min(x, maxX + 15);\n        x = Math.max(minX - 15, x);\n\n        //am I at my lower bound\n        var atMin = x < minX && dragColumnIndex !== 0;\n\n        //am I at my upper bound\n        var atMax = x > maxX;\n\n        var d = dragger;\n\n        this.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + 0 + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease');\n\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, ' + -10 + 'px)');\n        requestAnimationFrame(function() {\n            d.style.display = 'inline';\n        });\n\n        var overCol = grid.renderer.getColumnFromPixelX(x + (d.width / 2 / hdpiRatio));\n\n        if (atMin) {\n            overCol = 0;\n        }\n\n        if (atMax) {\n            overCol = grid.getColumnCount() - 1;\n        }\n\n        var doAFloat = dragColumnIndex > overCol;\n        doAFloat = doAFloat || (overCol - dragColumnIndex >= 1);\n\n        if (doAFloat && !atMax && !autoScrollingNow) {\n            var draggedToTheRight = dragColumnIndex < overCol;\n            // if (draggedToTheRight) {\n            //     overCol = overCol - 1;\n            // }\n            if (this.isFloatingNow) {\n                return;\n            }\n\n            this.isFloatingNow = true;\n            this.createFloatColumn(grid, overCol);\n            this.floatColumnTo(grid, draggedToTheRight);\n        } else {\n\n            if (x < minX - 10) {\n                this.checkAutoScrollToLeft(grid, x);\n            }\n            if (x > minX - 10) {\n                this.columnDragAutoScrollingLeft = false;\n            }\n            //lets check for autoscroll to right if were up against it\n            if (atMax || x > maxX + 10) {\n                this.checkAutoScrollToRight(grid, x);\n                return;\n            }\n            if (x < maxX + 10) {\n                this.columnDragAutoScrollingRight = false;\n            }\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc autoscroll to the right if necessary\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    checkAutoScrollToRight: function(grid, x) {\n        if (this.columnDragAutoScrollingRight) {\n            return;\n        }\n        this.columnDragAutoScrollingRight = true;\n        this._checkAutoScrollToRight(grid, x);\n    },\n\n    _checkAutoScrollToRight: function(grid, x) {\n        if (!this.columnDragAutoScrollingRight) {\n            return;\n        }\n        var scrollLeft = grid.getHScrollValue();\n        if (!grid.dragging || scrollLeft > (grid.sbHScrollConfig.rangeStop - 2)) {\n            return;\n        }\n        var draggedIndex = grid.renderOverridesCache.dragger.columnIndex;\n        grid.scrollBy(1, 0);\n        var newIndex = draggedIndex + 1;\n        console.log(newIndex, draggedIndex);\n        grid.swapColumns(newIndex, draggedIndex);\n        grid.renderOverridesCache.dragger.columnIndex = newIndex;\n\n        setTimeout(this._checkAutoScrollToRight.bind(this, grid, x), 250);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc autoscroll to the left if necessary\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    checkAutoScrollToLeft: function(grid, x) {\n        if (this.columnDragAutoScrollingLeft) {\n            return;\n        }\n        this.columnDragAutoScrollingLeft = true;\n        this._checkAutoScrollToLeft(grid, x);\n    },\n\n    _checkAutoScrollToLeft: function(grid, x) {\n        if (!this.columnDragAutoScrollingLeft) {\n            return;\n        }\n\n        var scrollLeft = grid.getHScrollValue();\n        if (!grid.dragging || scrollLeft < 1) {\n            return;\n        }\n        var draggedIndex = grid.renderOverridesCache.dragger.columnIndex;\n        grid.swapColumns(draggedIndex + scrollLeft, draggedIndex + scrollLeft - 1);\n        grid.scrollBy(-1, 0);\n        setTimeout(this._checkAutoScrollToLeft.bind(this, grid, x), 250);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc a column drag has completed, update data and cleanup\n     * @param {Hypergrid} grid\n     */\n    endDragColumn: function(grid) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        var columnIndex = grid.renderOverridesCache.dragger.columnIndex;\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n        var self = this;\n        var startX = columnEdges[columnIndex - scrollLeft];\n        var d = dragger;\n\n        self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease');\n        self.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -1 + 'px)');\n        d.style.boxShadow = '0px 0px 0px #888888';\n\n        setTimeout(function() {\n            grid.renderOverridesCache.dragger = null;\n            grid.repaint();\n            requestAnimationFrame(function() {\n                d.style.display = 'none';\n                grid.endDragColumnNotification();\n            });\n        }, columnAnimationTime + 50);\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isHeaderRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y === 0;\n        return isFixed;\n    }\n\n});\n\nmodule.exports = ColumnMoving;\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnPicker = Feature.extend('ColumnPicker', {\n\n    alias: 'ColumnPicker',\n\n    /**\n     * @memberOf ColumnPicker.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyUp: function(grid, event) {\n        var key = event.detail.char.toLowerCase();\n        var keys = grid.resolveProperty('editorActivationKeys');\n        if (keys.indexOf(key) > -1) {\n           grid.toggleColumnPicker();\n        }\n    },\n\n});\n\nmodule.exports = ColumnPicker;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnResizing = Feature.extend('ColumnResizing', {\n\n    alias: 'ColumnResizing',\n\n    /**\n     * the index of the column wall were currently dragging\n     * @type {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    dragIndex: -2,\n\n    /**\n     * the pixel location of the where the drag was initiated\n     * @type {number}\n     * @default -1\n     * @memberOf ColumnResizing.prototype\n     */\n    dragStart: -1,\n\n    /**\n     * the starting width/height of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf ColumnResizing.prototype\n     */\n    dragIndexStartingSize: -1,\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the mouse x,y coordinate\n     * @returns {number}\n     * @param {MouseEvent} event - the mouse event to query\n     */\n    getMouseValue: function(event) {\n        return event.primitiveEvent.detail.mouse.x;\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the grid cell x,y coordinate\n     * @returns {number}\n     * @param {window.fin.rectangular.Point} gridCell\n     */\n    getGridCellValue: function(gridCell) {\n        return gridCell.y;\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the grids x,y scroll value\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getScrollValue: function(grid) {\n        return grid.getHScrollValue();\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the width/height of the row/column of interest\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getAreaSize: function(grid, index) {\n        return grid.getColumnWidth(index);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc set the width/height of the row/column at index\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     * @param {number} value - the width/height to set to\n     */\n    setAreaSize: function(grid, index, value) {\n        grid.setColumnWidth(index, value);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the recently rendered area's width/height\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getPreviousAbsoluteSize: function(grid, index) {\n        return grid.getRenderedWidth(index);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc returns the index of which divider I'm over\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    overAreaDivider: function(grid, event) {\n        return grid.overColumnDivider(event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc am I over the column/row area\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedOtherArea: function(grid, event) {\n        return this.isFirstFixedRow(grid, event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the cursor name\n     * @returns {string}\n     */\n    getCursorName: function() {\n        return 'col-resize';\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        if (this.dragIndex > -2) {\n            //var fixedAreaCount = this.getFixedAreaCount(grid);\n            //var offset = this.getFixedAreaSize(grid, fixedAreaCount + areaIndex);\n            var mouse = this.getMouseValue(event);\n            var scrollValue = this.getScrollValue(grid);\n            if (this.dragIndex < this.getFixedAreaCount(grid)) {\n                scrollValue = 0;\n            }\n            var previous = this.getPreviousAbsoluteSize(grid, this.dragIndex - scrollValue);\n            var distance = mouse - previous;\n            this.setAreaSize(grid, this.dragIndex, distance);\n        } else if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the width/height of a specific row/column\n     * @param {Hypergrid} grid\n     * @param {number} areaIndex - the row/column index of interest\n     */\n    getSize: function(grid, areaIndex) {\n        return this.getAreaSize(grid, areaIndex);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the fixed area rows/columns count\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getOtherFixedAreaCount: function(grid) {\n        return grid.getFixedRowCount();\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        var overArea = this.overAreaDivider(grid, event);\n        if (isEnabled && overArea > -1 && this.isFirstFixedOtherArea(grid, event)) {\n            var scrollValue = this.getScrollValue(grid);\n            if (overArea < this.getFixedAreaCount(grid)) {\n                scrollValue = 0;\n            }\n            this.dragIndex = overArea - 1 + scrollValue;\n            this.dragStart = this.getMouseValue(event);\n            this.dragIndexStartingSize = 0;\n            this.detachChain();\n        } else if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        if (isEnabled && this.dragIndex > -2) {\n            this.cursor = null;\n            this.dragIndex = -2;\n\n            event.primitiveEvent.stopPropagation();\n            //delay here to give other events a chance to be dropped\n            var self = this;\n            grid.synchronizeScrollingBoundries();\n            setTimeout(function() {\n                self.attachChain();\n            }, 200);\n        } else if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        if (this.dragIndex > -2) {\n            return;\n        }\n        this.cursor = null;\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n        this.checkForAreaResizeCursorChange(grid, event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc fill this in\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    checkForAreaResizeCursorChange: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        if (isEnabled && this.overAreaDivider(grid, event) > -1 && this.isFirstFixedOtherArea(grid, event)) {\n            this.cursor = this.getCursorName();\n        } else {\n            this.cursor = null;\n        }\n\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @returns {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    getFixedAreaCount: function(grid) {\n        var count = grid.getFixedColumnCount() + (grid.isShowRowNumbers() ? 1 : 0) + (grid.hasHierarchyColumn() ? 1 : 0);\n        return count;\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @param event\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    handleDoubleClick: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        var hasCursor = this.overAreaDivider(grid, event) > -1; //this.cursor !== null;\n        var headerRowCount = grid.getHeaderRowCount();\n        //var headerColCount = grid.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        if (isEnabled && hasCursor && (gridCell.y <= headerRowCount)) {\n            grid.autosizeColumn(gridCell.x - 1);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @returns {boolean}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    isEnabled: function(grid) {\n        return true;\n    }\n\n});\n\nmodule.exports = ColumnResizing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnSelection = Feature.extend('ColumnSelection', {\n\n    alias: 'ColumnSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {window.fin.rectangular.Point}\n     * @default null\n     * @memberOf ColumnSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * The cell coordinates of the where the mouse pointer is during a drag operation.\n     * @type {Object}\n     * @default null\n     * @memberOf ColumnSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf ColumnSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf ColumnSelection.prototype\n     */\n    sbAutoStart: 0,\n\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragging) {\n            this.dragging = false;\n        }\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n            return;\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n\n        if ((!grid.isColumnSelection() || event.mousePoint.y < 5) && this.next) {\n            this.next.handleMouseDown(grid, event);\n            return;\n        }\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n\n        var isHeader = grid.isShowHeaderRow() && dy === 0 && dx !== -1;\n\n        if (isRightClick || !isHeader) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            var dCell = grid.newPoint(dx, 0);\n\n            var primEvent = event.primitiveEvent;\n            var keys = primEvent.detail.keys;\n            this.dragging = true;\n            this.extendSelection(grid, dCell, keys);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n\n        if ((!grid.isColumnSelection() || this.isColumnDragging(grid)) && this.next) {\n            this.next.handleMouseDrag(grid, event);\n            return;\n        }\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (isRightClick || !this.dragging) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (grid.getLastSelectionType() !== 'column') {\n            if (this.next) {\n                this.next.handleKeyDown(grid, event);\n            }\n            return;\n        }\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Handle a mousedrag selection\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n\n        //var behavior = grid.getBehavior();\n        var x = gridCell.x;\n        //            var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        var newX = x - mouseDown.x;\n        //var newY = y - mouseDown.y;\n\n        // if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n        //     return;\n        // }\n\n        grid.clearMostRecentColumnSelection();\n\n        grid.selectColumn(mouseDown.x, x);\n        grid.setDragExtent(grid.newPoint(newX, 0));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (this.currentDrag.x < b.origin.x) {\n            xOffset = -1;\n        }\n\n        if (this.currentDrag.x > b.origin.x + b.extent.x) {\n            xOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        grid.stopEditing();\n        //var hasCTRL = keys.indexOf('CTRL') !== -1;\n        var hasSHIFT = keys.indexOf('SHIFT') !== -1;\n\n        // var scrollTop = grid.getVScrollValue();\n        // var scrollLeft = grid.getHScrollValue();\n\n        // var numFixedColumns = 0;//grid.getFixedColumnCount();\n        // var numFixedRows = 0;//grid.getFixedRowCount();\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        //we have repeated a click in the same spot deslect the value from last time\n        // if (mousePoint && x === mousePoint.x && y === mousePoint.y) {\n        //     grid.clearSelections();\n        //     grid.popMouseDown();\n        //     grid.repaint();\n        //     return;\n        // }\n\n        // if (!hasCTRL && !hasSHIFT) {\n        //     grid.clearSelections();\n        // }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentColumnSelection();\n            grid.selectColumn(x, mousePoint.x);\n            grid.setDragExtent(grid.newPoint(x - mousePoint.x, 0));\n        } else {\n            grid.toggleSelectColumn(x, keys);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid) {\n\n        // var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n        // var maxRows = grid.getRowCount() - 1;\n\n        // var newX = mouseCorner.x;\n        // var newY = grid.getHeaderRowCount() + grid.getVScrollValue();\n\n        // newY = Math.min(maxRows, newY);\n\n        // grid.clearSelections();\n        // grid.select(newX, newY, 0, 0);\n        // grid.setMouseDown(new grid.rectangular.Point(newX, newY));\n        // grid.setDragExtent(new grid.rectangular.Point(0, 0));\n\n        // grid.repaint();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {\n        this.moveSingleSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n        this.moveSingleSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetX) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumns() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newX = extent.x + offsetX;\n        //var newY = grid.getRowCount();\n\n        newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX));\n\n        grid.clearMostRecentColumnSelection();\n        grid.selectColumn(origin.x, origin.x + newX);\n\n        grid.setDragExtent(grid.newPoint(newX, 0));\n\n        if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetX) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumnsCount() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newX = mouseCorner.x + offsetX;\n        //var newY = grid.getRowCount();\n\n        newX = Math.min(maxColumns, Math.max(0, newX));\n\n        grid.clearSelections();\n        grid.selectColumn(newX);\n        grid.setMouseDown(grid.newPoint(newX, 0));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelColIsVisible(newX, offsetX)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    isColumnDragging: function(grid) {\n        var dragger = grid.lookupFeature('ColumnMoving');\n        if (!dragger) {\n            return false;\n        }\n        var isActivated = dragger.dragging && !this.dragging;\n        return isActivated;\n    }\n\n});\n\nmodule.exports = ColumnSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnSorting = Feature.extend('ColumnSorting', {\n\n    alias: 'ColumnSorting',\n\n    /**\n     * @memberOf ColumnSorting.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n\n    handleDoubleClick: function(grid, event) {\n        var gridCell = event.gridCell;\n        if (grid.isShowHeaderRow() && gridCell.y === 0 && gridCell.x !== -1) {\n            var keys = event.primitiveEvent.detail.keys;\n            grid.toggleSort(gridCell.x, keys);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSorting.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        var y = event.gridCell.y;\n        if (this.isFixedRow(grid, event) && y < 1) {\n            this.cursor = 'pointer';\n        } else {\n            this.cursor = null;\n        }\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = ColumnSorting;\n","'use strict';\n\nvar Base = require('extend-me').Base;\n\n/**\n * @constructor\n * @desc instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n */\nvar Feature = Base.extend('Feature', {\n\n    /**\n     * the next feature to be given a chance to handle incoming events\n     * @type {Feature}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    next: null,\n\n    /**\n     * a temporary holding field for my next feature when I'm in a disconnected state\n     * @type {Feature}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    detached: null,\n\n    /**\n     * the cursor I want to be displayed\n     * @type {string}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    cursor: null,\n\n    /**\n     * the cell location where the cursor is currently\n     * @type {Point}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    currentHoverCell: null,\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc set my next field, or if it's populated delegate to the feature in my next field\n     * @param {Feature} nextFeature - this is how we build the chain of responsibility\n     */\n    setNext: function(nextFeature) {\n        if (this.next) {\n            this.next.setNext(nextFeature);\n        } else {\n            this.next = nextFeature;\n            this.detached = nextFeature;\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc disconnect my child\n     */\n    detachChain: function() {\n        this.next = null;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc reattach my child from the detached reference\n     */\n    attachChain: function() {\n        this.next = this.detached;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle mouse move down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseExit: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseExit(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseEnter: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseEnter(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (this.next) {\n            this.next.handleKeyDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyUp: function(grid, event) {\n        if (this.next) {\n            this.next.handleKeyUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleWheelMoved: function(grid, event) {\n        if (this.next) {\n            this.next.handleWheelMoved(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDoubleClick: function(grid, event) {\n        if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleTap: function(grid, event) {\n        if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleContextMenu: function(grid, event) {\n        if (this.next) {\n            this.next.handleContextMenu(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc toggle the column picker\n     */\n\n    moveSingleSelect: function(grid, x, y) {\n        if (this.next) {\n            this.next.moveSingleSelect(grid, x, y);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFixedRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y < grid.getFixedRowCount();\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y < 1;\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFixedColumn: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.x < grid.getFixedColumnCount();\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedColumn: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var edge = grid.isShowRowNumbers() ? 0 : 1;\n        var isFixed = gridCell.x < edge;\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isTopLeft: function(grid, event) {\n        var isTopLeft = this.isFixedRow(grid, event) && this.isFixedColumn(grid, event);\n        return isTopLeft;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    setCursor: function(grid) {\n        if (this.next) {\n            this.next.setCursor(grid);\n        }\n        if (this.cursor) {\n            grid.beCursor(this.cursor);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    initializeOn: function(grid) {\n        if (this.next) {\n            this.next.initializeOn(grid);\n        }\n    }\n\n});\n\nmodule.exports = Feature;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar Filters = Feature.extend('Filters', {\n\n    alias: 'Filters',\n\n    handleDoubleClick: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    handleTap: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n           grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) {\n        var isFilterRow = grid.isFilterRow(event.gridCell.y);\n        var activateEditor = isDoubleClickEditorActivation && isFilterRow;\n        return activateEditor;\n    }\n\n});\n\nmodule.exports = Filters;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\nvar commands = {\n    PAGEDOWN: function(grid) { grid.pageDown(); },\n    PAGEUP: function(grid) { grid.pageUp(); },\n    PAGELEFT: function(grid) { grid.pageLeft(); },\n    PAGERIGHT: function(grid) { grid.pageRight(); }\n};\n\n/**\n * @constructor\n */\nvar KeyPaging = Feature.extend('KeyPaging', {\n\n    alias: 'KeyPaging',\n\n    /**\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf KeyPaging.prototype\n     */\n    handleKeyDown: function(grid, event) {\n        var detail = event.detail.char;\n        var func = commands[detail];\n        if (func) {\n            func(grid);\n        } else if (this.next) {\n            this.next.handleKeyDown(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = KeyPaging;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar OnHover = Feature.extend('OnHover', {\n\n    alias: 'OnHover',\n\n    /**\n     * @desc Hhandle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf OnHover.prototype\n     */\n    handleMouseMove: function(grid, event) {\n        var currentHoverCell = grid.getHoverCell();\n        if (!event.gridCell.equals(currentHoverCell)) {\n            if (currentHoverCell) {\n                this.handleMouseExit(grid, currentHoverCell);\n            }\n            this.handleMouseEnter(grid, event);\n            grid.setHoverCell(event.gridCell);\n        } else {\n            if (this.next) {\n                this.next.handleMouseMove(grid, event);\n            }\n        }\n    }\n\n});\n\nmodule.exports = OnHover;\n","'use strict';\n\nvar ColumnResizing = require('./ColumnResizing');\n\n/**\n * @constructor\n */\nvar RowResizing = ColumnResizing.extend('RowResizing', {\n\n    alias: 'RowResizing',\n\n    /**\n     * the index of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragArea: -1,\n\n    /**\n     * the pixel location of the where the drag was initiated\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragStart: -1,\n\n    /**\n     * the starting width/height of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragAreaStartingSize: -1,\n\n    /**\n     * @memberOf RowResizing.prototype\n     * @desc get the mouse x,y coordinate\n     * @returns {number}\n     * @param {MouseEvent} event - the mouse event to query\n     */\n    getMouseValue: function(event) {\n        return event.primitiveEvent.detail.mouse.y;\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc get the grid cell x,y coordinate\n     * @returns {number}\n     * @param {Point} gridCell\n     */\n    getGridCellValue: function(gridCell) {\n        return gridCell.x;\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the grids x,y scroll value\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getScrollValue: function(grid) {\n        return grid.getVScrollValue();\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the width/height of the row/column of interest\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getAreaSize: function(grid, index) {\n        return grid.getRowHeight(index);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc set the width/height of the row/column at index\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     * @param {number} value - the width/height to set to\n     */\n    setAreaSize: function(grid, index, value) {\n        grid.setRowHeight(index, value);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc returns the index of which divider I'm over\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    overAreaDivider: function(grid, event) {\n        return grid.overRowDivider(event);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc am I over the column/row area\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedOtherArea: function(grid, event) {\n        return this.isFirstFixedColumn(grid, event);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the cursor name\n     * @returns {string}\n     */\n    getCursorName: function() {\n        return 'row-resize';\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the recently rendered area's width/height\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getPreviousAbsoluteSize: function(grid, index) {\n        return grid.getRenderedHeight(index);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the fixed area rows/columns count\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getOtherFixedAreaCount: function(grid) {\n        return grid.getFixedColumnCount();\n    },\n\n    /**\n     *\n     * @param {Hypergrid} grid\n     * @returns {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    getFixedAreaCount: function(grid) {\n        return grid.getFixedRowCount() + grid.getHeaderRowCount();\n    },\n\n    /**\n     *\n     * @param {Hypergrid} grid\n     * @returns {boolean}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    isEnabled: function(grid) {\n        return grid.isRowResizeable();\n    }\n\n});\n\nmodule.exports = RowResizing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar RowSelection = Feature.extend('RowSelection', {\n\n    alias: 'RowSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {Point}\n     * @default null\n     * @memberOf RowSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * The cell coordinates of the where the mouse pointer is during a drag operation.\n     * @type {Object}\n     * @default null\n     * @memberOf RowSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf RowSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf RowSelection.prototype\n     */\n    sbAutoStart: 0,\n\n    dragArmed: false,\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragArmed) {\n            this.dragArmed = false;\n            //global row selection\n            if (event.gridCell.x === -1 && event.gridCell.y === 0) {\n                grid.toggleSelectAllRows();\n            }\n            grid.fireSyntheticRowSelectionChangedEvent();\n        } else if (this.dragging) {\n            this.dragging = false;\n            grid.fireSyntheticRowSelectionChangedEvent();\n        } else if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n\n\n        var isHeader = grid.isShowRowNumbers() && dx < 0;\n\n        if (!grid.isRowSelection() || isRightClick || !isHeader) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n\n            var numFixedRows = grid.getFixedRowCount();\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(0, dy);\n\n            var primEvent = event.primitiveEvent;\n            var keys = primEvent.detail.keys;\n            this.dragArmed = true;\n            this.extendSelection(grid, dCell, keys);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (!this.dragArmed || !grid.isRowSelection() || isRightClick) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n            this.dragging = true;\n            var numFixedRows = grid.getFixedRowCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            //var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(0, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (grid.getLastSelectionType() !== 'row') {\n            if (this.next) {\n                this.next.handleKeyDown(grid, event);\n            }\n            return;\n        }\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Handle a mousedrag selection\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n\n        //var behavior = grid.getBehavior();\n        var y = gridCell.y;\n        //            var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        var newY = y - mouseDown.y;\n        //var newY = y - mouseDown.y;\n\n        // if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n        //     return;\n        // }\n\n        grid.clearMostRecentRowSelection();\n\n        grid.selectRow(mouseDown.y, y);\n        grid.setDragExtent(grid.newPoint(0, newY));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (this.currentDrag.y < b.origin.y) {\n            yOffset = -1;\n        }\n\n        if (this.currentDrag.y > b.origin.y + b.extent.y) {\n            yOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        grid.stopEditing();\n        //var hasCTRL = keys.indexOf('CTRL') !== -1;\n        var hasSHIFT = keys.indexOf('SHIFT') !== -1;\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentRowSelection();\n            grid.selectRow(y, mousePoint.y);\n            grid.setDragExtent(grid.newPoint(0, y - mousePoint.y));\n        } else {\n            grid.toggleSelectRow(y, keys);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid) {\n        this.moveSingleSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid) {\n        this.moveSingleSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var newX = grid.getHeaderColumnCount() + grid.getHScrollValue();\n        var newY = mouseCorner.y;\n\n        newX = Math.min(maxColumns, newX);\n\n        grid.clearSelections();\n        grid.select(newX, newY, 0, 0);\n        grid.setMouseDown(grid.newPoint(newX, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetY) {\n\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableRows = grid.getVisibleRows() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newY = extent.y + offsetY;\n        //var newY = grid.getRowCount();\n\n        newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY));\n\n        grid.clearMostRecentRowSelection();\n        grid.selectRow(origin.y, origin.y + newY);\n\n        grid.setDragExtent(grid.newPoint(0, newY));\n\n        if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.fireSyntheticRowSelectionChangedEvent();\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetY) {\n\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableRows = grid.getVisibleRowsCount() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newY = mouseCorner.y + offsetY;\n        //var newY = grid.getRowCount();\n\n        newY = Math.min(maxRows, Math.max(0, newY));\n\n        grid.clearSelections();\n        grid.selectRow(newY);\n        grid.setMouseDown(grid.newPoint(0, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelRowIsVisible(newY, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.fireSyntheticRowSelectionChangedEvent();\n        grid.repaint();\n\n    },\n\n    isSingleRowSelection: function() {\n        return true;\n    }\n\n});\n\nmodule.exports = RowSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ThumbwheelScrolling = Feature.extend('ThumbwheelScrolling', {\n\n    alias: 'ThumbwheelScrolling',\n\n    /**\n     * @memberOf ThumbwheelScrolling.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleWheelMoved: function(grid, e) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var primEvent = e.primitiveEvent;\n        var deltaY = primEvent.wheelDeltaY || -primEvent.deltaY;\n        var deltaX = primEvent.wheelDeltaX || -primEvent.deltaX;\n        if (deltaY > 0) {\n            grid.scrollBy(0, -1);\n        } else if (deltaY < -0) {\n            grid.scrollBy(0, 1);\n        } else if (deltaX > 0) {\n            grid.scrollBy(-1, 0);\n        } else if (deltaX < -0) {\n            grid.scrollBy(1, 0);\n        }\n    }\n\n});\n\n\nmodule.exports = ThumbwheelScrolling;\n","'use strict';\n\nmodule.exports = {\n    Feature: require('./Feature'), // abstract base class\n    CellClick: require('./CellClick'),\n    CellEditing: require('./CellEditing'),\n    CellSelection: require('./CellSelection'),\n    ColumnAutosizing: require('./ColumnAutosizing'),\n    ColumnMoving: require('./ColumnMoving'),\n    ColumnResizing: require('./ColumnResizing'),\n    ColumnSelection: require('./ColumnSelection'),\n    ColumnSorting: require('./ColumnSorting'),\n    Filters: require('./Filters'),\n    KeyPaging: require('./KeyPaging'),\n    OnHover: require('./OnHover'),\n    ColumnPicker: require('./ColumnPicker'),\n    RowResizing: require('./RowResizing'),\n    RowSelection: require('./RowSelection'),\n    ThumbwheelScrolling: require('./ThumbwheelScrolling')\n};\n","'use strict';\n\nmodule.exports = (function() {\n\n    var depthString = '                                                                                ';\n\n    function DataNodeBase(key) {\n        this.label = key;\n        this.data = [''];\n        this.rowIndexes = [];\n        this.hasChildren = false;\n        this.depth = 0;\n        this.height = 1;\n        this.expanded = false;\n    }\n\n    DataNodeBase.prototype.isNullObject = false;\n\n    DataNodeBase.prototype.getValue = function(x) {\n        return this.data[x];\n    };\n\n    DataNodeBase.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeBase.prototype.computeDepthString = function() {\n        var string = depthString.substring(0, 2 + (this.depth * 3)) + this.label;\n        return string;\n    };\n\n    DataNodeBase.prototype.computeHeight = function() {\n        return 1;\n    };\n\n    DataNodeBase.prototype.getAllRowIndexes = function() {\n        return this.rowIndexes;\n    };\n\n    DataNodeBase.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n    };\n\n    DataNodeBase.prototype.applyAggregates = function(aggregator) {\n        var hasGroupsOffset = aggregator.hasGroups() ? 1 : 0;\n        var indexes = this.getAllRowIndexes();\n        if (indexes.length === 0) {\n            return; // no data to rollup on\n        }\n        var aggregates = aggregator.aggregates;\n        var data = this.data;\n        data.length = aggregates.length + hasGroupsOffset;\n\n        var sorter = aggregator.sorterInstance;\n        sorter.indexes = indexes;\n\n        for (var i = 0; i < aggregates.length; i++) {\n            var aggregate = aggregates[i];\n            data[i + hasGroupsOffset] = aggregate(sorter);\n        }\n\n        this.data = data;\n    };\n\n    DataNodeBase.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n    };\n\n    DataNodeBase.prototype.toggleExpansionState = function() { /* aggregator */\n        //do nothing by default\n    };\n\n    return DataNodeBase;\n\n})();\n","'use strict';\n\nvar Map = require('./Map');\nvar DataNodeBase = require('./DataNodeBase');\n\nmodule.exports = (function() {\n\n    var ExpandedMap = {\n        true: '▾',\n        false: '▸'\n    };\n    var depthString = '                                                                                ';\n\n    function DataNodeGroup(key) {\n        DataNodeBase.call(this, key);\n        this.children = new Map();\n    }\n\n    DataNodeGroup.prototype = Object.create(DataNodeBase.prototype);\n\n    DataNodeGroup.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.children = this.children.values;\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.prune(this.depth + 1);\n        }\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeGroup.prototype.computeDepthString = function() {\n        var icon = ExpandedMap[this.expanded + ''];\n        var string = depthString.substring(0, this.depth * 3) + icon + ' ' + this.label;\n        return string;\n    };\n\n    DataNodeGroup.prototype.getAllRowIndexes = function() {\n        if (this.rowIndexes.length === 0) {\n            this.rowIndexes = this.computeAllRowIndexes();\n        }\n        return this.rowIndexes;\n    };\n\n    DataNodeGroup.prototype.computeAllRowIndexes = function() {\n        var result = [];\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            var childIndexes = child.getAllRowIndexes();\n            Array.prototype.splice.apply(result, [result.length, 0].concat(childIndexes));\n        }\n        return result;\n    };\n\n    DataNodeGroup.prototype.toggleExpansionState = function(aggregator) { /* aggregator */\n        this.expanded = !this.expanded;\n        this.data[0] = this.computeDepthString();\n        if (this.expanded) {\n            this.computeAggregates(aggregator);\n        }\n    };\n\n    DataNodeGroup.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n        if (!this.expanded) {\n            return; // were not being viewed, don't have child nodes do computation;\n        }\n        for (var i = 0; i < this.children.length; i++) {\n            this.children[i].computeAggregates(aggregator);\n        }\n    };\n\n    DataNodeGroup.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n        if (this.expanded) {\n            for (var i = 0; i < this.children.length; i++) {\n                var child = this.children[i];\n                child.buildView(aggregator);\n            }\n        }\n    };\n\n    DataNodeGroup.prototype.computeHeight = function() {\n        var height = 1; //I'm 1 high\n        if (!this.expanded) {\n            this.height = 1;\n        } else {\n            for (var i = 0; i < this.children.length; i++) {\n                height = height + this.children[i].computeHeight();\n            }\n            this.height = height;\n        }\n        return this.height;\n    };\n\n    return DataNodeGroup;\n\n})();\n","'use strict';\n\nvar DataNodeBase = require('./DataNodeBase');\n\nmodule.exports = (function() {\n\n    function DataNodeLeaf(key) {\n        DataNodeBase.call(this, key);\n    }\n\n    DataNodeLeaf.prototype = Object.create(DataNodeBase.prototype);\n\n    DataNodeLeaf.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeLeaf.prototype.computeHeight = function() {\n        return 1;\n    };\n\n    DataNodeLeaf.prototype.getAllRowIndexes = function() {\n        return this.rowIndexes;\n    };\n\n    DataNodeLeaf.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n    };\n\n    DataNodeLeaf.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n    };\n\n    return DataNodeLeaf;\n\n})();\n","'use strict';\n\nvar DataNodeGroup = require('./DataNodeGroup');\n\nmodule.exports = (function() {\n\n    function DataNodeTree(key) {\n        DataNodeGroup.call(this, key);\n        this.height = 0;\n        this.expanded = true;\n    }\n\n    DataNodeTree.prototype = Object.create(DataNodeGroup.prototype);\n\n    DataNodeTree.prototype.prune = function() {\n        this.children = this.children.values;\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.prune(0);\n        }\n    };\n\n    DataNodeTree.prototype.buildView = function(aggregator) {\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.buildView(aggregator);\n        }\n    };\n\n    DataNodeTree.prototype.computeHeight = function() {\n        var height = 1;\n        for (var i = 0; i < this.children.length; i++) {\n            height = height + this.children[i].computeHeight();\n        }\n        this.height = height;\n\n        return this.height;\n    };\n\n\n    return DataNodeTree;\n\n})();\n","'use strict';\n\nvar DataSourceSorter = require('./DataSourceSorter');\nvar DataNodeTree = require('./DataNodeTree');\nvar DataNodeGroup = require('./DataNodeGroup');\nvar DataNodeLeaf = require('./DataNodeLeaf');\n\nmodule.exports = (function() {\n\n    var headerify = function(string) {\n        var pieces = string.replace(/[_-]/g, ' ').replace(/[A-Z]/g, ' $&').split(' ').map(function(s) {\n            return (s.charAt(0).toUpperCase() + s.slice(1)).trim();\n        });\n        pieces = pieces.filter(function(e) {\n            return e.length !== 0;\n        });\n        return pieces.join(' ').trim();\n    };\n\n    //?[t,c,b,a]\n    // t is a dataSource,\n    // a is a dicitionary of aggregates,  columnName:function\n    // b is a dicitionary of groupbys, columnName:sourceColumnName\n    // c is a list of constraints,\n\n    function DataSourceAggregator(dataSource) {\n        this.tree = new DataNodeTree('Totals');\n        this.indexes = [];\n        this.dataSource = dataSource;\n        this.aggregates = [];\n        this.headers = [];\n        this.groupBys = [];\n        this.view = [];\n        this.sorterInstance = {};\n        this.presortGroups = true;\n        this.lastAggregate = {};\n        this.setAggregates({});\n    }\n\n    DataSourceAggregator.prototype.isNullObject = false;\n\n    DataSourceAggregator.prototype.setAggregates = function(aggregations) {\n        this.lastAggregate = aggregations;\n        var props = [];\n        var i;\n        this.clearAggregations();\n        this.headers.length = 0;\n\n        for (var key in aggregations) {\n            props.push([key, aggregations[key]]);\n        }\n\n        // if (props.length === 0) {\n        //     var fields = [].concat(this.dataSource.getFields());\n        //     for (i = 0; i < fields.length; i++) {\n        //         props.push([fields[i], Aggregations.first(i)]); /* jshint ignore:line */\n        //     }\n        // }\n        if (this.hasGroups()) {\n            this.headers.push('Tree');\n        }\n\n        for (i = 0; i < props.length; i++) {\n            var agg = props[i];\n            this.addAggregate(agg[0], agg[1]);\n        }\n    };\n\n    DataSourceAggregator.prototype.addAggregate = function(label, func) {\n        this.headers.push(headerify(label));\n        this.aggregates.push(func);\n    };\n\n    DataSourceAggregator.prototype.setGroupBys = function(columnIndexArray) {\n        this.groupBys.length = 0;\n        for (var i = 0; i < columnIndexArray.length; i++) {\n            this.groupBys.push(columnIndexArray[i]);\n        }\n        this.setAggregates(this.lastAggregate);\n    };\n\n    DataSourceAggregator.prototype.addGroupBy = function(index) {\n        this.groupBys.push(index);\n    };\n\n    DataSourceAggregator.prototype.hasGroups = function() {\n        return this.groupBys.length > 0;\n    };\n\n    DataSourceAggregator.prototype.hasAggregates = function() {\n        return this.aggregates.length > 0;\n    };\n\n    DataSourceAggregator.prototype.apply = function() {\n        this.buildGroupTree();\n    };\n\n    DataSourceAggregator.prototype.clearGroups = function() {\n        this.groupBys.length = 0;\n    };\n\n    DataSourceAggregator.prototype.clearAggregations = function() {\n        this.aggregates.length = 0;\n        this.headers.length = 0;\n    };\n\n    DataSourceAggregator.prototype.buildGroupTree = function() {\n        var c, r, g, value, createFunc;\n        var createBranch = function(key, map) {\n            value = new DataNodeGroup(key);\n            map.set(key, value);\n            return value;\n        };\n        var createLeaf = function(key, map) {\n            value = new DataNodeLeaf(key);\n            map.set(key, value);\n            return value;\n        };\n        var groupBys = this.groupBys;\n        var source = this.dataSource;\n        var rowCount = source.getRowCount();\n\n        // lets sort our data first....\n        if (this.presortGroups) {\n            for (c = 0; c < groupBys.length; c++) {\n                g = groupBys[groupBys.length - c - 1];\n                source = new DataSourceSorter(source);\n                source.sortOn(g);\n            }\n        }\n\n        var tree = this.tree = new DataNodeTree('Totals');\n        var path = tree;\n        var leafDepth = groupBys.length - 1;\n        for (r = 0; r < rowCount; r++) {\n            for (c = 0; c < groupBys.length; c++) {\n                g = groupBys[c];\n                value = source.getValue(g, r);\n\n                //test that I'm not a leaf\n                createFunc = (c === leafDepth) ? createLeaf : createBranch;\n                path = path.children.getIfAbsent(value, createFunc);\n            }\n            path.rowIndexes.push(r);\n            path = tree;\n        }\n        this.sorterInstance = new DataSourceSorter(source);\n        tree.prune();\n        this.tree.computeAggregates(this);\n        this.buildView();\n    };\n\n    DataSourceAggregator.prototype.buildView = function() {\n        this.view.length = 0;\n        this.tree.computeHeight();\n        this.tree.buildView(this);\n    };\n\n    DataSourceAggregator.prototype.viewMakesSense = function() {\n        return this.hasAggregates();\n    };\n\n    DataSourceAggregator.prototype.getValue = function(x, y) {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getValue(x, y);\n        }\n        var row = this.view[y];\n        if (!row) {\n            return null;\n        }\n        return row.getValue(x);\n    };\n\n    DataSourceAggregator.prototype.setValue = function(x, y, value) {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.setValue(x, y, value);\n        }\n    };\n\n    DataSourceAggregator.prototype.getColumnCount = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getColumnCount();\n        }\n        var colCount = this.getHeaders().length;\n        return colCount;\n    };\n\n    DataSourceAggregator.prototype.getRowCount = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getRowCount();\n        }\n        return this.view.length; //header column\n    };\n\n    DataSourceAggregator.prototype.click = function(y) {\n        var group = this.view[y];\n        group.toggleExpansionState(this);\n        this.buildView();\n    };\n\n    DataSourceAggregator.prototype.getHeaders = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getHeaders();\n        }\n        return this.headers;\n    };\n\n    DataSourceAggregator.prototype.setHeaders = function(headers) {\n        this.dataSource.setHeaders(headers);\n    };\n\n    DataSourceAggregator.prototype.getFields = function() {\n        return this.dataSource.getFields();\n    };\n\n    DataSourceAggregator.prototype.setFields = function(fields) {\n        return this.dataSource.setFields(fields);\n    };\n\n    DataSourceAggregator.prototype.getGrandTotals = function() {\n        var view = this.tree;\n        return [view.data];\n    };\n\n    DataSourceAggregator.prototype.getRow = function(y) {\n\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getRow(y);\n        }\n\n        var rollups = this.view[y];\n        if (!rollups) {\n            return this.tree;\n        }\n\n        return rollups;\n    };\n\n    DataSourceAggregator.prototype.setData = function(arrayOfUniformObjects) {\n        this.dataSource.setData(arrayOfUniformObjects);\n        this.apply();\n    };\n\n    return DataSourceAggregator;\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    function DataSourceDecorator(dataSource) {\n        this.dataSource = dataSource;\n        this.indexes = [];\n    }\n\n    DataSourceDecorator.prototype.isNullObject = false;\n\n    DataSourceDecorator.prototype.transposeY = function(y) {\n        if (this.indexes.length !== 0) {\n            return this.indexes[y];\n        }\n        return y;\n    };\n\n    DataSourceDecorator.prototype.getValue = function(x, y) {\n        var value = this.dataSource.getValue(x, this.transposeY(y));\n        return value;\n    };\n\n    DataSourceDecorator.prototype.getRow = function(y) {\n\n        return this.dataSource.getRow(this.transposeY(y));\n    };\n\n    DataSourceDecorator.prototype.setValue = function(x, y, value) {\n\n        this.dataSource.setValue(x, this.transposeY(y), value);\n    };\n\n    DataSourceDecorator.prototype.getColumnCount = function() {\n\n        return this.dataSource.getColumnCount();\n    };\n\n    DataSourceDecorator.prototype.getFields = function() {\n\n        return this.dataSource.getFields();\n    };\n\n    DataSourceDecorator.prototype.setFields = function(fields) {\n\n        return this.dataSource.setFields(fields);\n    };\n\n    DataSourceDecorator.prototype.click = function(y) {\n\n        return this.dataSource.click(this.transposeY(y));\n    };\n\n    DataSourceDecorator.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    DataSourceDecorator.prototype.setHeaders = function(headers) {\n        return this.dataSource.setHeaders(headers);\n    };\n\n    DataSourceDecorator.prototype.getHeaders = function() {\n\n        return this.dataSource.getHeaders();\n    };\n\n    DataSourceDecorator.prototype.getGrandTotals = function() {\n        return this.dataSource.getGrandTotals();\n    };\n\n    DataSourceDecorator.prototype.initializeIndexVector = function() {\n        var rowCount = this.dataSource.getRowCount();\n        var indexVector = new Array(rowCount);\n        for (var r = 0; r < rowCount; r++) {\n            indexVector[r] = r;\n        }\n        this.indexes = indexVector;\n    };\n\n    DataSourceDecorator.prototype.setData = function(arrayOfUniformObjects) {\n        this.dataSource.setData(arrayOfUniformObjects);\n    };\n\n    return DataSourceDecorator;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\n\nmodule.exports = (function() {\n\n    function DataSourceFilter(dataSource) {\n        DataSourceDecorator.call(this, dataSource, false);\n        this.filters = [];\n    }\n\n    DataSourceFilter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceFilter.prototype.add = function(columnIndex, filter) {\n        filter.columnIndex = columnIndex;\n        this.filters.push(filter);\n    };\n    DataSourceFilter.prototype.set = function(columnIndex, filter) {\n        filter.columnIndex = columnIndex;\n        this.filters.push(filter);\n    };\n\n    DataSourceFilter.prototype.clearAll = function() { /* filter */\n        this.filters.length = 0;\n        this.indexes.length = 0;\n    };\n\n    DataSourceFilter.prototype.applyAll = function() {\n        if (this.filters.length === 0) {\n            this.indexes.length = 0;\n            return;\n        }\n        var indexes = this.indexes;\n        indexes.length = 0;\n        var count = this.dataSource.getRowCount();\n        for (var r = 0; r < count; r++) {\n            if (this.applyFiltersTo(r)) {\n                indexes.push(r);\n            }\n        }\n    };\n\n    DataSourceFilter.prototype.applyFiltersTo = function(r) {\n        var filters = this.filters;\n        var isFiltered = true;\n        for (var f = 0; f < filters.length; f++) {\n            var filter = filters[f];\n            var rowObject = this.dataSource.getRow(r);\n            isFiltered = isFiltered && filter(this.dataSource.getValue(filter.columnIndex, r), rowObject, r);\n        }\n        return isFiltered;\n    };\n\n    DataSourceFilter.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        //our filter matched nothing....\n        if (this.filters.length !== 0) {\n            return 0;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    return DataSourceFilter;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\n\nmodule.exports = (function() {\n\n    function DataSourceGlobalFilter(dataSource) {\n        DataSourceDecorator.call(this, dataSource, false);\n        this.filter = null;\n    }\n\n    DataSourceGlobalFilter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceGlobalFilter.prototype.set = function(filter) {\n        this.filter = filter;\n    };\n\n    DataSourceGlobalFilter.prototype.clear = function() { /* filter */\n        this.filter = null;\n        this.indexes.length = 0;\n    };\n\n    DataSourceGlobalFilter.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        //our filter matched nothing....\n        if (this.filter) {\n            return 0;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    DataSourceGlobalFilter.prototype.apply = function(visibleColumns) {\n        if (!this.filter) {\n            this.indexes.length = 0;\n            return;\n        }\n        var visibleColumnMap = this.visibleColumnMap = [];\n        visibleColumns.forEach(function(column) {\n            visibleColumnMap.push(column.index);\n        });\n        var indexes = this.indexes;\n        indexes.length = 0;\n        var count = this.dataSource.getRowCount();\n        for (var r = 0; r < count; r++) {\n            if (this.applyFilterTo(r)) {\n                indexes.push(r);\n            }\n        }\n    };\n\n    DataSourceGlobalFilter.prototype.applyFilterTo = function(r) {\n        var isFiltered = false;\n        var filter = this.filter;\n        var visColCount = this.visibleColumnMap.length;\n        var rowObject = this.dataSource.getRow(r);\n        for (var v = 0; v < visColCount; v++) {\n            var i = this.visibleColumnMap[v];\n            isFiltered = isFiltered || filter(this.dataSource.getValue(i, r), rowObject, r);\n            if (isFiltered) {\n                return true;\n            }\n        }\n        return false;\n    };\n\n    return DataSourceGlobalFilter;\n\n})();\n","'use strict';\n\nvar Utils = require('./Utils.js');\nvar DataSourceDecorator = require('./DataSourceDecorator');\nvar valueOrFunctionExecute = function(valueOrFunction) {\n    var isFunction = (((typeof valueOrFunction)[0]) === 'f');\n    var result = isFunction ? valueOrFunction() : valueOrFunction;\n    return result;\n};\n\nmodule.exports = (function() {\n\n    function DataSourceSorter(dataSource) {\n        DataSourceDecorator.call(this, dataSource);\n        this.descendingSort = false;\n    }\n\n    DataSourceSorter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceSorter.prototype.sortOn = function(columnIndex, sortType) {\n        if (sortType === 0) {\n            this.indexes.length = 0;\n            return;\n        }\n        this.initializeIndexVector();\n        var self = this;\n        Utils.stableSort(this.indexes, function(index) {\n            var val = self.dataSource.getValue(columnIndex, index);\n            val = valueOrFunctionExecute(val);\n            return val;\n        }, sortType);\n    };\n\n    return DataSourceSorter;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\nvar DataSourceSorter = require('./DataSourceSorter');\n\nmodule.exports = (function() {\n\n    function DataSourceSorterComposite(dataSource) {\n        DataSourceDecorator.call(this, dataSource);\n        this.sorts = [];\n        this.last = this.dataSource;\n    }\n\n    DataSourceSorterComposite.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceSorterComposite.prototype.getRow = function(y) {\n        return this.last.getRow(y);\n    };\n\n    DataSourceSorterComposite.prototype.sortOn = function(columnIndex, sortType) {\n        this.sorts.push([columnIndex, sortType]);\n    };\n\n    DataSourceSorterComposite.prototype.applySorts = function() {\n        var sorts = this.sorts;\n        var each = this.dataSource;\n        for (var i = 0; i < sorts.length; i++) {\n            var sort = sorts[i];\n            each = new DataSourceSorter(each);\n            each.sortOn(sort[0], sort[1]);\n        }\n        this.last = each;\n    };\n\n    DataSourceSorterComposite.prototype.clearSorts = function() {\n        this.sorts.length = 0;\n        this.last = this.dataSource;\n    };\n\n    DataSourceSorterComposite.prototype.getValue = function(x, y) {\n        return this.last.getValue(x, y);\n    };\n\n    DataSourceSorterComposite.prototype.setValue = function(x, y, value) {\n        this.last.setValue(x, y, value);\n    };\n\n    return DataSourceSorterComposite;\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    var headerify = function(string) {\n        var pieces = string.replace(/[_-]/g, ' ').replace(/[A-Z]/g, ' $&').split(' ').map(function(s) {\n            return s.charAt(0).toUpperCase() + s.slice(1);\n        });\n        return pieces.join(' ');\n    };\n\n    var computeFieldNames = function(object) {\n        if (!object) {\n            return [];\n        }\n        var fields = [].concat(Object.getOwnPropertyNames(object).filter(function(e) {\n            return e.substr(0, 2) !== '__';\n        }));\n        return fields;\n    };\n\n    function JSDataSource(data, fields) {\n        this.fields = fields || computeFieldNames(data[0]);\n        this.headers = [];\n        this.data = data;\n    }\n\n    JSDataSource.prototype.isNullObject = false;\n\n    JSDataSource.prototype.getValue = function(x, y) {\n        var row = this.data[y];\n        if (!row) {\n            return null;\n        }\n        var value = row[this.fields[x]];\n        return value;\n    };\n\n    JSDataSource.prototype.getRow = function(y) {\n\n        return this.data[y];\n    };\n\n    JSDataSource.prototype.setValue = function(x, y, value) {\n\n        this.data[y][this.fields[x]] = value;\n    };\n\n    JSDataSource.prototype.getColumnCount = function() {\n\n        return this.getFields().length;\n    };\n\n    JSDataSource.prototype.getRowCount = function() {\n\n        return this.data.length;\n    };\n\n    JSDataSource.prototype.getFields = function() {\n\n        return this.fields;\n    };\n\n    JSDataSource.prototype.getHeaders = function() {\n        if (!this.headers || this.headers.length === 0) {\n            this.headers = this.getDefaultHeaders().map(function(each) {\n                return headerify(each);\n            });\n        }\n        return this.headers;\n    };\n\n    JSDataSource.prototype.getDefaultHeaders = function() {\n\n        return this.getFields();\n    };\n\n    JSDataSource.prototype.setFields = function(fields) {\n\n        this.fields = fields;\n    };\n\n    JSDataSource.prototype.setHeaders = function(headers) {\n\n        this.headers = headers;\n    };\n\n    JSDataSource.prototype.getGrandTotals = function() {\n        //nothing here\n        return;\n    };\n\n    JSDataSource.prototype.setData = function(arrayOfUniformObjects) {\n        this.data = arrayOfUniformObjects;\n    };\n\n    return JSDataSource;\n\n})();\n","'use strict';\n\nvar stableSort = require('./stableSort.js');\nvar Map = require('./Map.js');\n\nmodule.exports = (function() {\n\n    return {\n        stableSort: stableSort,\n        Map: Map\n    };\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    return {\n\n        count: function() { /* columIndex */\n            return function(group) {\n                var rows = group.getRowCount();\n                return rows;\n            };\n        },\n\n        sum: function(columIndex) {\n            return function(group) {\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                return sum;\n            };\n        },\n\n        min: function(columIndex) {\n            return function(group) {\n                var min = Infinity;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    min = Math.min(min, group.getValue(columIndex, r));\n                }\n                return min;\n            };\n        },\n\n\n        max: function(columIndex) {\n            return function(group) {\n                var max = -Infinity;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    max = Math.max(max, group.getValue(columIndex, r));\n                }\n                return max;\n            };\n        },\n\n        avg: function(columIndex) {\n            return function(group) {\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                return sum / rows;\n            };\n        },\n\n        first: function(columIndex) {\n            return function(group) {\n                return group.getValue(columIndex, 0);\n            };\n        },\n\n        last: function(columIndex) {\n            return function(group) {\n                var rows = group.getRowCount();\n                return group.getValue(columIndex, rows - 1);\n            };\n        },\n\n        stddev: function(columIndex) {\n            return function(group) {\n                var r;\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                var mean = sum / rows;\n                var variance = 0;\n                for (r = 0; r < rows; r++) {\n                    var dev = (group.getValue(columIndex, r) - mean);\n                    variance = variance + (dev * dev);\n                }\n                var stddev = Math.sqrt(variance / rows);\n                return stddev;\n            };\n        }\n    };\n\n})();\n","'use strict';\n\nvar JSDataSource = require('./JSDataSource');\nvar DataSourceSorter = require('./DataSourceSorter');\nvar DataSourceSorterComposite = require('./DataSourceSorterComposite');\nvar DataSourceFilter = require('./DataSourceFilter');\nvar DataSourceGlobalFilter = require('./DataSourceGlobalFilter');\nvar DataSourceAggregator = require('./DataSourceAggregator');\nvar aggregations = require('./aggregations');\n\nmodule.exports = (function() {\n\n    return {\n        JSDataSource: JSDataSource,\n        DataSourceSorter: DataSourceSorter,\n        DataSourceSorterComposite: DataSourceSorterComposite,\n        DataSourceFilter: DataSourceFilter,\n        DataSourceGlobalFilter: DataSourceGlobalFilter,\n        DataSourceAggregator: DataSourceAggregator,\n        aggregations: aggregations\n    };\n\n})();\n","'use strict';\n\nvar stabilize = function(comparator, descending) {\n    return function(arr1, arr2) {\n        var x = arr1[0];\n        var y = arr2[0];\n        if (x === y) {\n            x = descending ? arr2[1] : arr1[1];\n            y = descending ? arr1[1] : arr2[1];\n        } else {\n            if (y === null) {\n                return -1;\n            }\n            if (x === null) {\n                return 1;\n            }\n        }\n        return comparator(x, y);\n    };\n};\n\n\nvar ascendingNumbers = function(x, y) {\n    return x - y;\n};\n\nvar descendingNumbers = function(x, y) {\n    return y - x;\n};\n\nvar ascendingAllOthers = function(x, y) {\n    return x < y ? -1 : 1;\n};\n\nvar descendingAllOthers = function(x, y) {\n    return y < x ? -1 : 1;\n};\n\nvar ascending = function(typeOfData) {\n    if (typeOfData === 'number') {\n        return stabilize(ascendingNumbers, false);\n    }\n    return stabilize(ascendingAllOthers, false);\n};\n\nvar descending = function(typeOfData) {\n    if (typeOfData === 'number') {\n        return stabilize(descendingNumbers, true);\n    }\n    return stabilize(descendingAllOthers, true);\n};\n\nmodule.exports = (function() {\n\n    function sort(indexVector, dataSource, sortType) {\n\n        var compare, i;\n\n        if (indexVector.length === 0) {\n            return; //nothing to do;\n        }\n\n        if (sortType === undefined) {\n            sortType = 1;\n        }\n\n        if (sortType === 0) {\n            return; // nothing to sort here;\n        }\n\n        var typeOfData = typeof dataSource(0);\n\n        compare = (sortType === 1) ? ascending(typeOfData) : descending(typeOfData);\n\n        //start the actually sorting.....\n        var tmp = new Array(indexVector.length);\n\n        //lets add the index for stability\n        for (i = 0; i < indexVector.length; i++) {\n            tmp[i] = [dataSource(i), i];\n        }\n\n        tmp.sort(compare);\n\n        //copy the sorted values into our index vector\n        for (i = 0; i < indexVector.length; i++) {\n            indexVector[i] = tmp[i][1];\n        }\n    }\n\n    return sort;\n})();\n","'use strict';\n\nvar images = require('../images');\n\n/**\n * @summary Writes error message into cell.\n *\n * @desc This funciton is guaranteed to be called as follows:\n *\n * ```javascript\n * gc.save();\n * gc.beginPath();\n * gc.rect(x, y, width, height);\n * gc.clip();\n * renderCellError(gc, message, x, y, width, height);\n * gc.restore();\n * ```\n *\n * Before doing anything else, this function should clear the cell by setting `gc.fillStyle` and calling `gc.fill()`.\n *\n * @param {CanvasRenderingContext2D} gc\n * @param {string} message\n * @param {number} x\n * @param {number} y\n * @param {number} width\n * @param {number} height\n */\nfunction renderCellError(gc, message, x, y, width, height) {\n\n    // clear the cell\n    // (this makes use of the rect path defined by the caller)\n    gc.fillStyle = '#FFD500';\n    gc.fill();\n\n    // render cell border\n    gc.strokeStyle = gc.createPattern(images.caution, 'repeat');\n    gc.lineWidth = 5;\n    gc.beginPath();\n    gc.moveTo(x, y); // caution: do not use rect() here because Chrome does not clip its stroke properly\n    gc.lineTo(x + width, y);\n    gc.lineTo(x + width, y + height);\n    gc.lineTo(x, y + height);\n    gc.lineTo(x, y);\n    gc.stroke();\n\n    // adjust clip region to prevent text from rendering over right border should it overflow\n    gc.beginPath();\n    gc.rect(x, y, width - 2, height);\n    gc.clip();\n\n    // render message text\n    gc.fillStyle = '#A00';\n    gc.textAlign = 'start';\n    gc.textBaseline = 'middle';\n    gc.font = 'bold 6pt \"arial narrow\", verdana, geneva';\n    gc.fillText(message, x + 4, y + height / 2 + 0.5);\n\n}\n\nmodule.exports = renderCellError;\n","'use strict';\n\nvar cssInjector = require('css-injector');\n\nvar stylesheets = {\n    grid: [\n        'div#grid-container {',\n        '    position: relative;',\n        '    display: inline-block;',\n        '    -webkit-user-select: none;',\n        '    -moz-user-select: none;',\n        '    -ms-user-select: none;',\n        '    -o-user-select: none;',\n        '    user-select: none;',\n        '    overflow: hidden; }',\n        'visible { opacity: 0.75; }',\n        'hidden { opacity: 0.0; }',\n        'editor {',\n        '    position: absolute;',\n        '    display: none;',\n        '    border: solid 2px black;',\n        '    outline: 0;',\n        '    padding: 0;',\n        '    z-index: 1000; }'\n    ],\n    'list-dragon': [\n        'div.dragon-list, li.dragon-pop {',\n        '    font-family: Roboto, sans-serif;',\n        '    text-transform: capitalize; }',\n        'div.dragon-list {',\n        '    position: absolute;',\n        '    top: 4%;',\n        '    left: 4%;',\n        '    height: 92%;',\n        '    width: 20%; }',\n        'div.dragon-list:nth-child(2) { left: 28%; }',\n        'div.dragon-list:nth-child(3) { left: 52%; }',\n        'div.dragon-list:nth-child(4) { left: 76%; }',\n        'div.dragon-list > div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }',\n        'div.dragon-list > ul { top: 46px; }',\n        'div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {',\n        '    content: \\'\\\\2b24\\';', // BLACK LARGE CIRCLE\n        '    color: #b6b6b6;',\n        '    font-size: 30px;',\n        '    margin: 8px 14px 8px 8px; }',\n        'li.dragon-pop { opacity:.8; }'\n    ]\n};\n\nfunction addStylesheet(key, referenceElement) {\n    cssInjector(stylesheets[key], key, referenceElement);\n}\n\nmodule.exports = addStylesheet;\n"]} +},{}]},{},[48]) +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/browser-pack/_prelude.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/css/stylesheets.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/images/images.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/images/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/css-injector/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/extend-me/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/FilterLeaf.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/FilterNode.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/css.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/leaf-operators.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/template.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/filter-tree/js/tree-operators.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/finbars/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/fincanvas/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/fincanvas/js/GraphicsContext.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/fincanvas/js/gc-console-logger.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/fincanvas/js/polymergestures.dev.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/list-dragon/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/lru-cache/lib/lru-cache.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/mustache/mustache.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/object-iterators/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/rectangular/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/regexp-like/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/sparse-boolean-array/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/templex/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/node_modules/unstrungify/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/Hypergrid.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/Behavior.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/Column.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/DataModelDecorator.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/JSON.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/Local.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/Null.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/behaviors/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/CellEditor.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Choice.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Color.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Date.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Filter.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Simple.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Slider.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Spinner.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/Textfield.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/cellEditors/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/dataModels/DataModel.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/dataModels/JSON.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/defaults.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/fake_f7e8d5a2.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/CellClick.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/CellEditing.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/CellSelection.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnAutosizing.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnMoving.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnPicker.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnResizing.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnSelection.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ColumnSorting.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/Feature.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/Filters.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/KeyPaging.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/OnHover.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/RowResizing.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/RowSelection.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/ThumbwheelScrolling.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/features/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/Base.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/CellProvider.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/CustomFilter.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/Formatters.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/Mappy.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/Renderer.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/SelectionModel.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/TableDialog.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/deprecated.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/lib/renderCellError.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeBase.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeGroup.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeLeaf.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataNodeTree.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceAggregator.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceDecorator.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceFilter.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceGlobalFilter.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceSorter.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/DataSourceSorterComposite.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/JSDataSource.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/Utils.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/aggregations.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/index.js","/Users/jonathan/repos/hypergrid/fin-hypergrid/src/local_node_modules/finanalytics/stableSort.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/xBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACt1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxrEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7kBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxiBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtxGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACv3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9uBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACziBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5qBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/iBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9fA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5pBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9lCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACphBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error(\"Cannot find module '\"+o+\"'\")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","'use strict';\n\nvar cssInjector = require('css-injector');\n\nvar stylesheets = {\n    grid: [\n        'div#grid-container {',\n        '    position: relative;',\n        '    display: inline-block;',\n        '    -webkit-user-select: none;',\n        '    -moz-user-select: none;',\n        '    -ms-user-select: none;',\n        '    -o-user-select: none;',\n        '    user-select: none;',\n        '    overflow: hidden; }',\n        'visible { opacity: 0.75; }',\n        'hidden { opacity: 0.0; }',\n        'editor {',\n        '    position: absolute;',\n        '    display: none;',\n        '    border: solid 2px black;',\n        '    outline: 0;',\n        '    padding: 0;',\n        '    z-index: 1000; }'\n    ],\n    'list-dragon': [\n        'div.dragon-list, li.dragon-pop {',\n        '    font-family: Roboto, sans-serif;',\n        '    text-transform: capitalize; }',\n        'div.dragon-list {',\n        '    position: absolute;',\n        '    top: 4%;',\n        '    left: 4%;',\n        '    height: 92%;',\n        '    width: 20%; }',\n        'div.dragon-list:nth-child(2) { left: 28%; }',\n        'div.dragon-list:nth-child(3) { left: 52%; }',\n        'div.dragon-list:nth-child(4) { left: 76%; }',\n        'div.dragon-list > div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }',\n        'div.dragon-list > ul { top: 46px; }',\n        'div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {',\n        '    content: \\'\\\\2b24\\';', // BLACK LARGE CIRCLE\n        '    color: #b6b6b6;',\n        '    font-size: 30px;',\n        '    margin: 8px 14px 8px 8px; }',\n        'li.dragon-pop { opacity:.8; }'\n    ]\n};\n\nfunction addStylesheet(key, referenceElement) {\n    cssInjector(stylesheets[key], key, referenceElement);\n}\n\nmodule.exports = addStylesheet;\n","module.exports = { // This file generated by gulp-imagine-64 at 6:03:02 AM on 2/1/2016\n\t\"calendar\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAAc0lEQVR4nIXQwQkCMRSE4U9ZLMCT9Xjaq2AfNhfYU5oQLMAOtoN48EWei5iBIRPe/yYQ3qrhf1lFG7iKcEaJxSfukUvMWgdHavt0uWHtg2QwxXnAnJZ2uOLyVZtybzzhgWNmfoFl0/YB87NbzR1cjP9xeQHSDC6mcL1xFQAAAABJRU5ErkJggg==\"\n\t},\n\t\"checked\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAYJJREFUOE+NkstLglEQxf0fahG0iFrUxm2ElFDYLohCqCDaCAkWPaxIRbFFEJEaGEKLDCoMETRFUAMLyaIHBUG6sSKIMtKFqEhLT818ZUgmDhzu3DPn9z0uV1RrmUwmyGQyqNVqfFvViwBxu5RFPZuLSyGMKhz/qlEsRV19K8xm6y+w7bpBPFnAferjj3bdQX6DpHcAUwavAHUN2RGIZxBJZHH2mC/TUeydwwTZvBegLENNgw7sX6Wh1FswNmPEmjPCDyGRRwCtW9E3tMgdAtQw7GZjYcNX+gza2wJ3ZXsSZUuQ0vWCOV8SHfJJ/uluhbHUj1v8PKNMszIoQNRMHCShD6Wh8zyhrbOPwz8w+STKlCCJ7oRNUzQH63kBs5thBghePXxlj2aUoSxDPcuXPNiLAc5EEZ6HIkbmV2DYiXBPHs0o079+K0DTVj/s11mE00A0L+g4VcDp10qKZMAzytBhMaTRaPmYg885DlcSzSij0eoEiIouoUqlqqqaL2rlEok+Ad4vlfzPoVDsAAAAAElFTkSuQmCC\"\n\t},\n\t\"down-rectangle\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAAkAAAAECAYAAABcDxXOAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAadEVYdFNvZnR3YXJlAFBhaW50Lk5FVCB2My41LjExR/NCNwAAABpJREFUGFdjgIL/eDAKIKgABggqgAE0BQwMAPTlD/Fpi0JfAAAAAElFTkSuQmCC\"\n\t},\n\t\"filter-off\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAAChSURBVChTzZHBCoUgFET9TqEiskgyWoutQvRLRIr+cR7XQAjiJW/1BgZmMUevXsY5xy9OoDEGMcYiUzeB67qibVuwQjVNA6311V+WBeM4vsLDMEApde/1fY9pmtI453neHEKAlBJd1z0fXtc16PbjODK07zvmeUZVVd8nooc75zJIOX3Gm6i0bVsGKf8xKIRIuyJTLgJJ3nvQzsjW2geIsQ/pr9hMVrSncAAAAABJRU5ErkJggg==\"\n\t},\n\t\"filter-on\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAACoSURBVChTY3BqfP2fHAzWmDbj7f8p294RhVOBasEa02e+/e/VBmQQCTxaX/9PnvYGoj5ywpv/Qd2ENft3vv4f1gfVBAP+nW/+h/a+ATtn1q73KHjytvdgg3070DTBgHvL6/8g22fsQGiaDmSHA21xaybgIpDHixa8hWssnA8NDEIApCh3LkIjiD2INYJCL2X6W3B8gdhEaQQBUOCA4gyE8+e9xaKJgQEA/74BNE3cElkAAAAASUVORK5CYII=\"\n\t},\n\t\"unchecked\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAARBJREFUOE+9krtug1AQRPldSio7FQ1tZImOkoKOBomGT0EURC5ino54yTw90WywQhTkIkVWGoF2zuxdrlD+t0zThKZpT0Vmxb8CQRCg6zr0fb8rer7vfwcPxxdcrx+YpgnzPGNZlh9ibxxHlGUJshLSdV0at9tNpg7DIBrX5+OkPM9BVkKGYSBJEtR1jbZrBdiqbVtUVYU0TUFWQq+nE+I4xvvlImGaW7FHjwxZCVmWhbfzGVmWoSgKWXUr9uiRISshx3FkEldomubXauzRI0NWQp7nyUR+NG/rfr/jUXxnjx5vmKyEbNuWox9Xvid6ZMhK6HA4wnVdhGGIKIp2RY8MWQmx+JuoqvpUZFb8L6UonyYL3uOtrFH+AAAAAElFTkSuQmCC\"\n\t},\n\t\"up-down-spin\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGJJREFUOE+lkwEKACEIBH2Zb/PnHsoGeaVJDUjGOgRRpKpkiIj+y4MME3eDR7kaKOVNsJyMNjIHzGy9YnW6J7qIcrriQimeCqORNABd0fpRTkt8uVUj7EsxC6vs/q3e/Q6iD2bwnByjPXHNAAAAAElFTkSuQmCC\"\n\t},\n\t\"up-down\": {\n\t\ttype: \"image/png\",\n\t\tdata: \"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGFJREFUOE+lkkEKQCEIRD2ZJ3Ph3iN4WD9GflpYhj0YYowpGgJmbikd3gjMDFokwbuT1iAiurG5nomgqo5QaPo9ERQRI6Jf7sfGjudy2je23+i0Wl2oQ85TOdlfrJQOazF8br+rqTXQKn0AAAAASUVORK5CYII=\"\n\t},\n};\n","/* eslint-env browser */\n\n'use strict';\n\nvar _ = require('object-iterators');\n\nvar images = require('./images'); // this is the file generated by gulpfile.js (and ignored by git)\n\n_(images).each(function(image, key) {\n    var element = new Image();\n    element.src = 'data:' + image.type + ';base64,' + image.data;\n    images[key] = element;\n});\n\nimages.checkbox = function(state) {\n    return images[state ? 'checked' : 'unchecked'];\n};\n\nimages.filter = function(state) {\n    return images[state ? 'filter-on' : 'filter-off'];\n};\n\nmodule.exports = images;\n","'use strict';\n\n/* eslint-env browser */\n\n/** @namespace cssInjector */\n\n/**\n * @summary Insert base stylesheet into DOM\n *\n * @desc Creates a new `<style>...</style>` element from the named text string(s) and inserts it but only if it does not already exist in the specified container as per `referenceElement`.\n *\n * > Caveat: If stylesheet is for use in a shadow DOM, you must specify a local `referenceElement`.\n *\n * @returns A reference to the newly created `<style>...</style>` element.\n *\n * @param {string|string[]} cssRules\n * @param {string} [ID]\n * @param {undefined|null|Element|string} [referenceElement] - Container for insertion. Overloads:\n * * `undefined` type (or omitted): injects stylesheet at top of `<head>...</head>` element\n * * `null` value: injects stylesheet at bottom of `<head>...</head>` element\n * * `Element` type: injects stylesheet immediately before given element, wherever it is found.\n * * `string` type: injects stylesheet immediately before given first element found that matches the given css selector.\n *\n * @memberOf cssInjector\n */\nfunction cssInjector(cssRules, ID, referenceElement) {\n    if (typeof referenceElement === 'string') {\n        referenceElement = document.querySelector(referenceElement);\n        if (!referenceElement) {\n            throw 'Cannot find reference element for CSS injection.';\n        }\n    } else if (referenceElement && !(referenceElement instanceof Element)) {\n        throw 'Given value not a reference element.';\n    }\n\n    var container = referenceElement && referenceElement.parentNode || document.head || document.getElementsByTagName('head')[0];\n\n    if (ID) {\n        ID = cssInjector.idPrefix + ID;\n\n        if (container.querySelector('#' + ID)) {\n            return; // stylesheet already in DOM\n        }\n    }\n\n    var style = document.createElement('style');\n    style.type = 'text/css';\n    if (ID) {\n        style.id = ID;\n    }\n    if (cssRules instanceof Array) {\n        cssRules = cssRules.join('\\n');\n    }\n    cssRules = '\\n' + cssRules + '\\n';\n    if (style.styleSheet) {\n        style.styleSheet.cssText = cssRules;\n    } else {\n        style.appendChild(document.createTextNode(cssRules));\n    }\n\n    if (referenceElement === undefined) {\n        referenceElement = container.firstChild;\n    }\n\n    container.insertBefore(style, referenceElement);\n\n    return style;\n}\n\n/**\n * @summary Optional prefix for `<style>` tag IDs.\n * @desc Defaults to `'injected-stylesheet-'`.\n * @type {string}\n * @memberOf cssInjector\n */\ncssInjector.idPrefix = 'injected-stylesheet-';\n\n// Interface\nmodule.exports = cssInjector;\n","'use strict';\n\n/** @namespace extend-me **/\n\n/** @summary Extends an existing constructor into a new constructor.\n *\n * @returns {ChildConstructor} A new constructor, extended from the given context, possibly with some prototype additions.\n *\n * @desc Extends \"objects\" (constructors), with optional additional code, optional prototype additions, and optional prototype member aliases.\n *\n * > CAVEAT: Not to be confused with Underscore-style .extend() which is something else entirely. I've used the name \"extend\" here because other packages (like Backbone.js) use it this way. You are free to call it whatever you want when you \"require\" it, such as `var inherits = require('extend')`.\n *\n * Provide a constructor as the context and any prototype additions you require in the first argument.\n *\n * For example, if you wish to be able to extend `BaseConstructor` to a new constructor with prototype overrides and/or additions, basic usage is:\n *\n * ```javascript\n * var Base = require('extend-me').Base;\n * var BaseConstructor = Base.extend(basePrototype); // mixes in .extend\n * var ChildConstructor = BaseConstructor.extend(childPrototypeOverridesAndAdditions);\n * var GrandchildConstructor = ChildConstructor.extend(grandchildPrototypeOverridesAndAdditions);\n * ```\n *\n * This function (`extend()`) is added to the new extended object constructor as a property `.extend`, essentially making the object constructor itself easily \"extendable.\" (Note: This is a property of each constructor and not a method of its prototype!)\n *\n * @param {string} [extendedClassName] - This is simply added to the prototype as $$CLASS_NAME. Useful for debugging because all derived constructors appear to have the same name (\"Constructor\") in the debugger. This property is ignored unless `extend.debug` is explicitly set to a truthy value.\n *\n * @param {extendedPrototypeAdditionsObject} [prototypeAdditions] - Object with members to copy to new constructor's prototype. Most members will be copied to the prototype. Some members, however, have special meanings as explained in the {@link extendedPrototypeAdditionsObject|type definition} (and may or may not be copied to the prototype).\n *\n * @property {boolean} [debug] - See parameter `extendedClassName` _(above)_.\n *\n * @property {object} Base - A convenient base class from which all other classes can be extended.\n *\n * @memberOf extend-me\n */\nfunction extend(extendedClassName, prototypeAdditions) {\n    switch (arguments.length) {\n        case 0:\n            prototypeAdditions = {};\n            break;\n        case 1:\n            prototypeAdditions = extendedClassName;\n            if (typeof prototypeAdditions !== 'object') {\n                throw 'Single parameter overload must be object.';\n            }\n            extendedClassName = undefined;\n            break;\n        case 2:\n            if (typeof extendedClassName !== 'string' || typeof prototypeAdditions !== 'object') {\n                throw 'Two parameter overload must be string, object.';\n            }\n            break;\n        default:\n            throw 'Too many parameters';\n    }\n\n    function Constructor() {\n        if (prototypeAdditions.preInitialize) {\n            prototypeAdditions.preInitialize.apply(this, arguments);\n        }\n\n        initializePrototypeChain.apply(this, arguments);\n\n        if (prototypeAdditions.postInitialize) {\n            prototypeAdditions.postInitialize.apply(this, arguments);\n        }\n    }\n\n    Constructor.extend = extend;\n\n    var prototype = Constructor.prototype = Object.create(this.prototype);\n    prototype.constructor = Constructor;\n\n    if (extendedClassName && extend.debug) {\n        prototype.$$CLASS_NAME = extendedClassName;\n    }\n\n    for (var key in prototypeAdditions) {\n        if (prototypeAdditions.hasOwnProperty(key)) {\n            var value = prototypeAdditions[key];\n            switch (key) {\n                case 'initializeOwn':\n                    // already called above; not needed in prototype\n                    break;\n                case 'aliases':\n                    for (var alias in value) {\n                        if (value.hasOwnProperty(alias)) {\n                            makeAlias(value[alias], alias);\n                        }\n                    }\n                    break;\n                default:\n                    if (typeof value === 'string' && value[0] === '#') {\n                        makeAlias(value, key.substr(1));\n                    } else {\n                        prototype[key] = value;\n                    }\n            }\n        }\n    }\n\n    return Constructor;\n\n    function makeAlias(value, key) { // eslint-disable-line no-shadow\n        prototype[key] = prototypeAdditions[value];\n    }\n}\n\nextend.Base = function () {};\nextend.Base.extend = extend;\n\n/** @typedef {function} extendedConstructor\n * @property prototype.super - A reference to the prototype this constructor was extended from.\n * @property [extend] - If `prototypeAdditions.extendable` was truthy, this will be a reference to {@link extend.extend|extend}.\n */\n\n/** @typedef {object} extendedPrototypeAdditionsObject\n * @property {function} [initialize] - Additional constructor code for new object. This method is added to the new constructor's prototype. Gets passed new object as context + same args as constructor itself. Called on instantiation after similar function in all ancestors called with same signature.\n * @property {function} [initializeOwn] - Additional constructor code for new object. This method is added to the new constructor's prototype. Gets passed new object as context + same args as constructor itself. Called on instantiation after (all) the `initialize` function(s).\n * @property {object} [aliases] - Hash of aliases for prototype members in form `{ key: 'member', ... }` where `key` is the name of an alieas and `'member'` is the name of an existing member in the prototype. Each such key is added to the prototype as a reference to the named member. (The `aliases` object itself is *not* added to prototype.) Alternatively:\n * @property {string} [keys] - Arbitrary property names defined here with string values starting with a `#` character will alias the actual properties named in the strings (following the `#`). This is an alternative to providing an `aliases` hash, perhaps simpler (though subtler). (Use arbitrary identifiers here; don't use the name `keys`!)\n * @property {*} [arbitraryProperties] - Any additional arbitrary properties defined here will be added to the new constructor's prototype. (Use arbitrary identifiers here; don't use the name `aribitraryProperties`!)\n */\n\n/** @summary Call all `initialize` methods found in prototype chain.\n * @desc This recursive routine is called by the constructor.\n * 1. Walks back the prototype chain to `Object`'s prototype\n * 2. Walks forward to new object, calling any `initialize` methods it finds along the way with the same context and arguments with which the constructor was called.\n * @private\n * @memberOf extend-me\n */\nfunction initializePrototypeChain() {\n    var term = this,\n        args = arguments;\n    recur(term);\n\n    function recur(obj) {\n        var proto = Object.getPrototypeOf(obj);\n        if (proto.constructor !== Object) {\n            recur(proto);\n            if (proto.hasOwnProperty('initialize')) {\n                proto.initialize.apply(term, args);\n            }\n        }\n    }\n}\n\nmodule.exports = extend;\n","/* eslint-env browser */\n\n// This is the main file, usable as is, such as by /test/index.js.\n// For npm: gulpfile.js copies this file to ../index.js, adjusting the require paths and defining the `css` local.\n// For CDN: gulpfile.js then browserifies ../index.js with sourcemap to /build/filter-tree.js and uglified without sourcemap to /build/filter-tree.min.js. The CDN is https://joneit.github.io/filter-tree.\n\n'use strict';\n\nvar unstrungify = require('unstrungify');\n\nvar cssInjector = require('./js/css');\nvar FilterNode = require('./js/FilterNode');\nvar DefaultFilter = require('./js/FilterLeaf');\nvar template = require('./js/template');\nvar operators = require('./js/tree-operators');\n\nvar ordinal = 0;\nvar reFilterTreeErrorString = /^filter-tree: /;\n\n/** @constructor\n *\n * @summary A node in a filter tree (including the root node), representing a complex filter expression.\n *\n * @desc A `FilterTree` is an n-ary tree with a single `operator` to be applied to all its `children`.\n *\n * Also known as a \"subtree\" or a \"subexpression\".\n *\n * Each of the `children` can be either:\n *\n * * a terminal node `Filter` (or an object inheriting from `Filter`) representing a simple conditional expression; or\n * * a nested `FilterTree` representing a complex subexpression.\n *\n * The `operator` must be one of the {@link operators|tree operators} or may be left undefined iff there is only one child node.\n *\n * Notes:\n * 1. A `FilterTree` may consist of a single leaf, in which case the `operator` is not used and may be left undefined. However, if a second child is added and the operator is still undefined, it will be set to the default (`'op-and'`).\n * 2. The order of the children is undefined as all operators are commutative. For the '`op-or`' operator, evaluation ceases on the first positive result and for efficiency, all simple conditional expressions will be evaluated before any complex subexpressions.\n * 3. A nested `FilterTree` is distinguished in the JSON object from a `Filter` by the presence of a `children` member.\n * 4. Nesting a `FilterTree` containing a single child is valid (albeit pointless).\n *\n * See {@link FilterNode} for additional `options` properties.\n *\n * @param {object} [options.editors] - Editor hash to override prototype's. These are constructors for objects that extend from `FilterTree.prototype.editors.Default`. Typically, you would include the default editor itself: `{ Default: FilterTree.prototype.editors.Default, ... }`. Alternatively, before instantiating, you might add your additional editors to `FilterTree.prototype.editors` for use by all filter tree objects.\n *\n * @property {FilterTree} parent\n * @property {number} ordinal\n * @property {string} operator\n * @property {FilterNode[]} children - Each one is either a `Filter` (or an object inheriting from `Filter`) or another `FilterTree`..\n * @property {Element} el - The root element of this (sub)tree.\n */\nvar FilterTree = FilterNode.extend('FilterTree', {\n\n    preInitialize: function(options) {\n        cssInjector('filter-tree-base', options && options.cssStylesheetReferenceElement);\n\n        if (options.editors) {\n            this.editors = options.editors;\n        }\n    },\n\n    destroy: function() {\n        detachChooser.call(this);\n    },\n\n    editors: {\n        Default: DefaultFilter\n    },\n\n    addEditor: function(key, overrides) {\n        if (overrides) {\n            this.editors[key] = DefaultFilter.extend(overrides);\n        } else {\n            delete this.editors[key];\n        }\n    },\n\n    newView: function() {\n        this.el = template('tree', ++ordinal);\n        this.el.addEventListener('click', catchClick.bind(this));\n    },\n\n    getState: unstrungify,\n\n    getJSON: function() {\n        var ready = JSON.stringify(this, null, this.JSONspace);\n        return ready ? ready : '';\n    },\n\n    setJSON: function(json) {\n        this.setState(JSON.parse(json));\n    },\n\n    load: function(state) {\n        if (!state) {\n            var filterEditorNames = Object.keys(this.editors),\n                onlyOneFilterEditor = filterEditorNames.length === 1;\n            this.children = onlyOneFilterEditor ? [new this.editors[filterEditorNames[0]]({\n                parent: this\n            })] : [];\n            this.operator = 'op-and';\n        } else {\n            throwIfJSON(state);\n\n            // Validate `state.operator`\n            if (!(operators[state.operator] || state.operator === undefined && state.children.length === 1)) {\n                throw FilterNode.Error('Expected `operator` property to be one of: ' + Object.keys(operators));\n            }\n            this.operator = state.operator;\n\n            // Validate `state.children`\n            if (!(state.children instanceof Array && state.children.length)) {\n                throw FilterNode.Error('Expected `children` property to be a non-empty array.');\n            }\n            this.children = [];\n            var self = this;\n            state.children.forEach(function(state) { // eslint-disable-line no-shadow\n                var Constructor;\n                if (typeof state !== 'object') {\n                    throw self.Error('Expected child to be an object containing either `children`, `editor`, or neither.');\n                }\n                if (state.children) {\n                    Constructor = FilterTree;\n                } else {\n                    Constructor = self.editors[state.editor || 'Default'];\n                }\n                self.children.push(new Constructor({\n                    state: state,\n                    parent: self\n                }));\n            });\n        }\n    },\n\n    render: function() {\n        // simulate click on the operator to display strike-through and operator between filters\n        var radioButton = this.el.querySelector('input[value=' + this.operator + ']');\n        radioButton.checked = true;\n        this['filter-tree-op-choice']({\n            target: radioButton\n        });\n\n        // when multiple filter editors available, simulate click on the new \"add conditional\" link\n        if (!this.children.length && Object.keys(this.editors).length > 1) {\n            var addFilterLink = this.el.querySelector('.filter-tree-add-filter');\n            this['filter-tree-add-filter']({\n                target: addFilterLink\n            });\n        }\n\n        // proceed with render\n        FilterNode.prototype.render.call(this);\n    },\n\n    'filter-tree-op-choice': function(evt) {\n        var radioButton = evt.target;\n\n        this.operator = radioButton.value;\n\n        // display strike-through\n        var radioButtons = this.el.querySelectorAll('label>input.filter-tree-op-choice[name=' + radioButton.name + ']');\n        Array.prototype.slice.call(radioButtons).forEach(function(radioButton) { // eslint-disable-line no-shadow\n            radioButton.parentElement.style.textDecoration = radioButton.checked ? 'none' : 'line-through';\n        });\n\n        // display operator between filters by adding operator string as a CSS class of this tree\n        for (var key in operators) {\n            this.el.classList.remove(key);\n        }\n        this.el.classList.add(this.operator);\n    },\n\n    'filter-tree-add-filter': function(evt) {\n        var filterEditorNames = Object.keys(this.editors);\n        if (filterEditorNames.length === 1) {\n            this.children.push(new this.editors[filterEditorNames[0]]({\n                parent: this\n            }));\n        } else {\n            attachChooser.call(this, evt);\n        }\n    },\n\n    'filter-tree-add': function() {\n        this.children.push(new FilterTree({\n            parent: this\n        }));\n    },\n\n    'filter-tree-remove': function(evt) {\n        var deleteButton = evt.target,\n            listItem = deleteButton.parentElement,\n            children = this.children,\n            el = deleteButton.nextElementSibling;\n\n        children.forEach(function(child, idx) {\n            if (child.el === el) {\n                delete children[idx];\n                listItem.remove();\n            }\n        });\n    },\n\n    /**\n     * @param {boolean} [object.rethrow=false] - Catch (do not throw) the error.\n     * @param {boolean} [object.alert=true] - Announce error via window.alert() before returning.\n     * @param {boolean} [object.focus=true] - Place the focus on the offending control and give it error color.\n     * @returns {undefined|string} `undefined` means valid or string containing error message.\n     */\n    validate: function(options) {\n        options = options || {};\n\n        var focus = options.focus === undefined || options.focus,\n            alert = options.alert === undefined || options.alert,\n            rethrow = options.rethrow === true,\n            result;\n\n        try {\n            validate.call(this, focus);\n        } catch (err) {\n            result = err.message;\n\n            // Throw when not a filter tree error\n            if (rethrow || !reFilterTreeErrorString.test(result)) {\n                throw err;\n            }\n\n            if (alert) {\n                result = result.replace(reFilterTreeErrorString, '');\n                window.alert(result); // eslint-disable-line no-alert\n            }\n        }\n\n        return result;\n    },\n\n    test: function test(dataRow) {\n        var operator = operators[this.operator],\n            result = operator.seed,\n            noChildrenDefined = true;\n\n        this.children.find(function(child) {\n            if (child) {\n                noChildrenDefined = false;\n                if (child instanceof DefaultFilter) {\n                    result = operator.reduce(result, child.test(dataRow));\n                } else if (child.children.length) {\n                    result = operator.reduce(result, test.call(child, dataRow));\n                }\n                return result === operator.abort;\n            }\n\n            return false;\n        });\n\n        return noChildrenDefined || (operator.negate ? !result : result);\n    },\n\n    toJSON: function toJSON() {\n        var state = {\n            operator: this.operator,\n            children: []\n        };\n\n        this.children.forEach(function(child) {\n            if (child) {\n                if (child instanceof DefaultFilter) {\n                    state.children.push(child);\n                } else if (child.children.length) {\n                    var ready = toJSON.call(child);\n                    if (ready) {\n                        state.children.push(ready);\n                    }\n                }\n            }\n        });\n\n        var metadata = FilterNode.prototype.toJSON.call(this);\n        Object.keys(metadata).forEach(function(key) {\n            state[key] = metadata[key];\n        });\n\n        return state.children.length ? state : undefined;\n    },\n\n    getSqlWhereClause: function getSqlWhereClause() {\n        var lexeme = operators[this.operator].SQL,\n            where = '';\n\n        this.children.forEach(function(child, idx) {\n            var op = idx ? ' ' + lexeme.op + ' ' : '';\n            if (child) {\n                if (child instanceof DefaultFilter) {\n                    where += op + child.getSqlWhereClause();\n                } else if (child.children.length) {\n                    where += op + getSqlWhereClause.call(child);\n                }\n            }\n        });\n\n        if (!where) {\n            where = 'NULL IS NULL';\n        }\n\n        return lexeme.beg + where + lexeme.end;\n    }\n\n});\n\n/**\n * Checks to make sure `state` is defined as a plain object and not a JSON string.\n * If not, throws error and does not return.\n * @param {object} state\n * @private\n */\nfunction throwIfJSON(state) {\n    if (typeof state !== 'object') {\n        var errMsg = 'Expected `state` parameter to be an object.';\n        if (typeof state === 'string') {\n            errMsg += ' See `JSON.parse()`.';\n        }\n        throw FilterNode.Error(errMsg);\n    }\n}\n\nfunction catchClick(evt) { // must be called with context\n    var elt = evt.target;\n\n    var handler = this[elt.className] || this[elt.parentNode.className];\n    if (handler) {\n        if (this.detachChooser) {\n            this.detachChooser();\n        }\n        handler.call(this, evt);\n        evt.stopPropagation();\n    }\n\n    if (this.eventHandler) {\n        this.eventHandler(evt);\n    }\n}\n\n/**\n * Throws error if invalid expression tree.\n * Caught by {@link FilterTree#validate|FilterTree.prototype.validate()}.\n * @param {boolean} focus - Move focus to offending control.\n * @returns {undefined} if valid\n * @private\n */\nfunction validate(focus) { // must be called with context\n    if (this instanceof FilterTree && !this.children.length) {\n        throw new FilterNode.Error('Empty subexpression (no filters).');\n    }\n\n    this.children.forEach(function(child) {\n        if (child instanceof DefaultFilter) {\n            child.validate(focus);\n        } else if (child.children.length) {\n            validate.call(child, focus);\n        }\n    });\n}\n\nfunction attachChooser(evt) { // must be called with context\n    var tree = this,\n        rect = evt.target.getBoundingClientRect();\n\n    if (!rect.width) {\n        // not in DOM yet so try again later\n        setTimeout(function() {\n            attachChooser.call(tree, evt);\n        }, 50);\n        return;\n    }\n\n    // Create it\n    var editors = Object.keys(FilterTree.prototype.editors),\n        chooser = this.chooser = document.createElement('select');\n\n    chooser.className = 'filter-tree-chooser';\n    chooser.size = editors.length;\n\n    editors.forEach(function(key) {\n        var name = tree.editors[key].prototype.name || key;\n        name = name.replace('?', '\\u225F'); // make question mark into \"? over equals\" UNICODE char\n        chooser.add(new Option(name, key));\n    });\n\n    chooser.onmouseover = function(evt) { // eslint-disable-line no-shadow\n        evt.target.selected = true;\n    };\n\n    // Position it\n    chooser.style.left = rect.left + 19 + 'px';\n    chooser.style.top = rect.bottom + 'px';\n\n    this.detachChooser = detachChooser.bind(this);\n    window.addEventListener('click', this.detachChooser); // detach chooser if click outside\n\n    chooser.onclick = function() {\n        tree.children.push(new tree.editors[chooser.value]({\n            parent: tree\n        }));\n        // click bubbles up to window where it detaches chooser\n    };\n\n    chooser.onmouseout = function() {\n        chooser.selectedIndex = -1;\n    };\n\n    // Add it to the DOM\n    this.el.appendChild(chooser);\n\n    // Color the link similarly\n    this.chooserTarget = evt.target;\n    this.chooserTarget.classList.add('as-menu-header');\n}\n\nfunction detachChooser() { // must be called with context\n    var chooser = this.chooser;\n    if (chooser) {\n        this.el.removeChild(chooser);\n        this.chooserTarget.classList.remove('as-menu-header');\n\n        chooser.onclick = chooser.onmouseout = null;\n        window.removeEventListener('click', this.detachChooser);\n\n        delete this.detachChooser;\n        delete this.chooser;\n    }\n}\n\nmodule.exports = FilterTree;\n","/* eslint-env browser */\n/* eslint-disable key-spacing */\n\n'use strict';\n\nvar FilterNode = require('./FilterNode');\nvar template = require('./template');\nvar operators = require('./leaf-operators');\n\n\n/** @typedef {object} converter\n * @property {function} to - Returns input value converted to type. Fails silently.\n * @property {function} not - Tests input value against type, returning `false if type or `true` if not type.\n */\n/** @type {converter} */\nvar numberConverter = { to: Number, not: isNaN };\n\n/** @type {converter} */\nvar dateConverter = { to: function(s) { return new Date(s); }, not: isNaN };\n\n/** @constructor\n * @summary A terminal node in a filter tree, representing a conditional expression.\n * @desc Also known as a \"filter.\"\n */\nvar FilterLeaf = FilterNode.extend('FilterLeaf', {\n\n    name: 'column : value',\n\n    preInitialize: function() {\n        this.onChange = cleanUpAndMoveOn.bind(this);\n    },\n\n    operators: operators,\n    operatorsOptions: operators.options,\n\n    destroy: function() {\n        if (this.controls) {\n            for (var key in this.controls) {\n                this.controls[key].removeEventListener('change', this.onChange);\n            }\n        }\n    },\n\n    newView: function() {\n        var fields = this.parent.nodeFields || this.fields;\n\n        if (!fields) {\n            throw FilterNode.Error('Terminal node requires a fields list.');\n        }\n\n        var root = this.el = document.createElement('span');\n        root.className = 'filter-tree-default';\n\n        this.controls = {\n            column: this.makeElement(root, fields, 'column', true),\n            operator: this.makeElement(root, this.operatorsOptions, 'operator'),\n            value: this.makeElement(root)\n        };\n\n        root.appendChild(document.createElement('br'));\n    },\n\n    /** @typedef {object} valueOption\n     * You should supply both `name` and `alias` but you could omit one or the other and whichever you provide will be used for both. (In such case you might as well just give a string for {@link fieldOption} rather than this object.)\n     * @property {string} [name]\n     * @property {string} [alias]\n     * @property {string} [type] One of the keys of `this.converters`. If not one of these (including `undefined`), field values will be tested with a string comparison.\n     * @property {boolean} [hidden=false]\n     */\n    /** @typedef {object} optionGroup\n     * @property {string} label\n     * @property {fieldOption[]} options\n     */\n    /** @typedef {string|valueOption|optionGroup} fieldOption\n     * The three possible types specify either an `<option>....</option>` element or an `<optgroup>....</optgroup>` element as follows:\n     * * `string` - specifies only the text of an `<option>....</option>` element (the value naturally defaults to the text)\n     * * {@link valueOption} - specifies both the text (`.name`) and the value (`.alias`) of an `<option....</option>` element\n     * * {@link optionGroup} - specifies an `<optgroup>....</optgroup>` element\n     */\n    /**\n     * @summary HTML form controls factory.\n     * @desc Creates and appends a text box or a drop-down.\n     * @returns The new element.\n     * @param {Element} container - An element to which to append the new element.\n     * @param {fieldOption[]} [options] - Overloads:\n     * * If omitted, will create an `<input/>` (text box) element.\n     * * If contains only a single option, will create a `<span>...</span>` element containing the string and a `<input type=hidden>` containing the value.\n     * * Otherwise, creates a `<select>...</select>` element with these options.\n     * @param {null|string} [prompt=''] - Adds an initial `<option>...</option>` element to the drop-down with this value, parenthesized, as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses.\n     */\n    makeElement: function(container, options, prompt, sort) {\n        var el, option, span,\n            tagName = options ? 'select' : 'input';\n\n        if (options && options.length === 1) {\n            option = options[0];\n\n            el = document.createElement('input');\n            el.type = 'hidden';\n            el.value = option.name || option.alias || option;\n\n            span = document.createElement('span');\n            span.innerHTML = option.alias || option.name || option;\n            span.appendChild(el);\n\n            container.appendChild(span);\n        } else {\n            el = addOptions(tagName, options, prompt, sort);\n            if (el.type === 'text' && this.eventHandler) {\n                this.el.addEventListener('keyup', this.eventHandler);\n            }\n            this.el.addEventListener('change', this.onChange);\n            FilterNode.setWarningClass(el);\n            container.appendChild(el);\n        }\n\n        return el;\n    },\n\n    load: function(state) {\n        if (state) {\n            var value, el, i, b, selected, notes = [];\n            for (var key in state) {\n                if (key !== 'fields' && key !== 'editor') {\n                    value = state[key];\n                    el = this.controls[key];\n                    switch (el.type) {\n                        case 'checkbox':\n                        case 'radio':\n                            el = document.querySelectorAll('input[name=\\'' + el.name + '\\']');\n                            for (i = 0; i < el.length; i++) {\n                                el[i].checked = value.indexOf(el[i].value) >= 0;\n                            }\n                            break;\n                        case 'select-multiple':\n                            el = el.options;\n                            for (i = 0, b = false; i < el.length; i++, b = b || selected) {\n                                selected = value.indexOf(el[i].value) >= 0;\n                                el[i].selected = selected;\n                            }\n                            FilterNode.setWarningClass(el, b);\n                            break;\n                        default:\n                            el.value = value;\n                            if (!FilterNode.setWarningClass(el) && el.value !== value) {\n                                notes.push({ key: key, value: value });\n                            }\n                    }\n                }\n            }\n            if (notes.length) {\n                var multiple = notes.length > 1,\n                    footnotes = template(multiple ? 'notes' : 'note'),\n                    inner = footnotes.lastElementChild;\n                notes.forEach(function(note) {\n                    var footnote = multiple ? document.createElement('li') : inner;\n                    note = template('optionMissing', note.key, note.value);\n                    while (note.length) { footnote.appendChild(note[0]); }\n                    if (multiple) { inner.appendChild(footnote); }\n                });\n                el.parentNode.replaceChild(footnotes, el.parentNode.lastElementChild);\n            }\n        }\n    },\n\n    /**\n     * @property {converter} number\n     * @property {converter} date\n     */\n    converters: {\n        number: numberConverter,\n        int: numberConverter, // pseudo-type: really just a Number\n        float: numberConverter, // pseudo-type: really just a Number\n        date: dateConverter\n    },\n\n    /**\n     * Throws error if invalid expression.\n     * Caught by {@link FilterTree#validate|FilterTree.prototype.validate()}.\n     *\n     * Also performs the following compilation actions:\n     * * Copies all `this.controls`' values from the DOM to similarly named properties of `this`.\n     * * Pre-sets `this.op` and `this.converter` for use in `test`'s tree walk.\n     *\n     * @param {boolean} focus - Move focus to offending control.\n     * @returns {undefined} if valid\n     */\n    validate: function(focus) {\n        var elementName, fields, field;\n\n        for (elementName in this.controls) {\n            var el = this.controls[elementName],\n                value = controlValue(el).trim();\n\n            if (value === '') {\n                if (focus) { clickIn(el); }\n                throw new FilterNode.Error('Blank ' + elementName + ' control.\\nComplete the filter or delete it.');\n            } else {\n                // Copy each controls's value to property of this object.\n                this[elementName] = value;\n            }\n        }\n\n        this.op = this.operators[this.operator];\n\n        this.converter = undefined; // remains undefined when neither operator nor column is typed\n        if (this.op.type) {\n            this.converter = this.converters[this.op.type];\n        } else {\n            for (elementName in this.controls) {\n                if (/^column/.test(elementName)) {\n                    fields = this.parent.nodeFields || this.fields;\n                    field = findField(fields, this[elementName]);\n                    if (field && field.type) {\n                        this.converter = this.converters[field.type];\n                    }\n                }\n            }\n        }\n    },\n\n    p: function(dataRow) { return dataRow[this.column]; },\n    q: function() { return this.value; },\n\n    test: function(dataRow) {\n        var p, q, // untyped versions of args\n            P, Q, // typed versions of p and q\n            convert;\n\n        return (p = this.p(dataRow)) === undefined || (q = this.q(dataRow)) === undefined\n            ? false\n            : (\n                (convert = this.converter) &&\n                !convert.not(P = convert.to(p)) &&\n                !convert.not(Q = convert.to(q))\n            )\n                ? this.op.test(P, Q)\n                : this.op.test(p, q);\n    },\n\n    toJSON: function(options) { // eslint-disable-line no-unused-vars\n        var state = {};\n        if (this.editor) {\n            state.editor = this.editor;\n        }\n        for (var key in this.controls) {\n            state[key] = this[key];\n        }\n        if (!this.parent.nodeFields && this.fields !== this.parent.fields) {\n            state.fields = this.fields;\n        }\n        return state;\n    },\n\n    getSqlWhereClause: function() {\n        return this.SQL_QUOTED_IDENTIFIER + this.column + this.SQL_QUOTED_IDENTIFIER + ' ' + (\n            typeof this.op.sql === 'function'\n                ? this.op.sql(this.value)\n                : (this.op.sql || this.operator) + operators.sq(this.value)\n        );\n    }\n});\n\nfunction findField(fields, name) {\n    var complex, simple;\n\n    simple = fields.find(function(field) {\n        if ((field.options || field) instanceof Array) {\n            return (complex = findField(field.options || field, name));\n        } else {\n            return field.name === name;\n        }\n    });\n\n    return complex || simple;\n}\n\n/** `change` or `click` event handler for all form controls.\n * Removes error CSS class from control.\n * Adds warning CSS class from control if blank; removes if not blank.\n * Moves focus to next non-blank sibling control.\n */\nfunction cleanUpAndMoveOn(evt) {\n    var el = evt.target;\n\n    // remove `error` CSS class, which may have been added by `FilterLeaf.prototype.validate`\n    el.classList.remove('filter-tree-error');\n\n    // set or remove 'warning' CSS class, as per el.value\n    FilterNode.setWarningClass(el);\n\n    // find next sibling control, if any\n    if (!el.multiple && el.value) {\n        while ((el = el.nextElementSibling) && (!('name' in el) || el.value.trim() !== '')); // eslint-disable-line curly\n    }\n\n    // and click in it (opens select list)\n    if (el && el.value.trim() === '') {\n        el.value = ''; // rid of any white space\n        FilterNode.clickIn(el);\n    }\n\n    if (this.eventHandler) {\n        this.eventHandler(evt);\n    }\n}\n\nfunction clickIn(el) {\n    setTimeout(function() {\n        el.classList.add('filter-tree-error');\n        FilterNode.clickIn(el);\n    }, 0);\n}\n\nfunction controlValue(el) {\n    var value, i;\n\n    switch (el.type) {\n        case 'checkbox':\n        case 'radio':\n            el = document.querySelectorAll('input[name=\\'' + el.name + '\\']:enabled:checked');\n            for (value = [], i = 0; i < el.length; i++) {\n                value.push(el[i].value);\n            }\n            break;\n\n        case 'select-multiple':\n            el = el.options;\n            for (value = [], i = 0; i < el.length; i++) {\n                if (!el.disabled && el.selected) {\n                    value.push(el[i].value);\n                }\n            }\n            break;\n\n        default:\n            value = el.value;\n    }\n\n    return value;\n}\n\n/**\n * @summary Creates a new element and adds options to it.\n * @param {string} tagName - Must be one of:\n * * `'input'` for a text box\n * * `'select'` for a drop-down\n * * `'optgroup'` (for internal use only)\n * @param {fieldOption[]} [options] - Strings to add as `<option>...</option>` elements. Omit to create a text box.\n * @param {null|string} [prompt=''] - Adds an initial `<option>...</option>` element to the drop-down with this value in parentheses as its `text`; and empty string as its `value`. Omitting creates a blank prompt; `null` suppresses.\n * @returns {Element} Either a `<select>` or `<optgroup>` element.\n */\nfunction addOptions(tagName, options, prompt, sort) {\n    var el = document.createElement(tagName);\n\n    if (options) {\n        var add, newOption;\n        if (tagName === 'select') {\n            add = el.add;\n            if (prompt) {\n                newOption = new Option('(' + prompt, '');\n                newOption.innerHTML += '&hellip;)';\n                el.add(newOption);\n            } else if (prompt !== null) {\n                el.add(new Option());\n            }\n        } else {\n            add = el.appendChild;\n            el.label = prompt;\n        }\n\n        if (sort) {\n            options = options.slice().sort(fieldComparator); // clone it and sort the clone\n        }\n\n        options.forEach(function(option) {\n            if ((option.options || option) instanceof Array) {\n                var optgroup = addOptions('optgroup', option.options || option, option.label);\n                el.add(optgroup);\n            } else {\n                var newElement = typeof option !== 'object'\n                    ? new Option(option)\n                    : new Option(\n                        option.alias || option.name,\n                        option.name || option.alias\n                    );\n                add.call(el, newElement);\n            }\n        });\n    } else {\n        el.type = 'text';\n    }\n\n    return el;\n}\n\nfunction fieldComparator(a, b) {\n    a = a.alias || a.name || a.label || a;\n    b = b.alias || b.name || b.label || b;\n    return a < b ? -1 : a > b ? 1 : 0;\n}\n\nmodule.exports = FilterLeaf;\n","/* eslint-env browser */\n\n'use strict';\n\nvar extend = require('extend-me');\nvar Base = extend.Base;\n\nvar template = require('./template');\n\nextend.debug = true;\n\nvar CHILDREN_TAG = 'OL',\n    CHILD_TAG = 'LI';\n\n/**\n * @constructor\n *\n * @description A filter tree represents a _complex conditional expression_ and consists of a single `FilterNode` object serving as the _root_ of an _n_-ary tree.\n *\n * Each `FilterNode` represents a node in tree. Each node is one of two types of objects extended from `FilterNode`:\n *\n * * The non-terminal (@link FilterTree} nodes represent _complex subexpressions_, each consisting of two or more _conditional_ (boolean expressions), all concatenated together with one of the _tree operators_.\n * * The terminal {@link FilterLeaf} nodes represent _simple expressions_.\n *\n * Tree operators currently include **_AND_** (labeled \"all\" in the UI; and \"op-and\" internally), **_OR_** (\"any\"; \"op-or\"), and **_NOR_** (\"none\"; \"op-nor\").\n *\n * Each conditional in a _subexpression_ (non-terminal node) is represented by a child node which may be either a _simple expression_ (terminal node) or another (\"nested\") subexpression non-terminal node.\n *\n * The `FilterLeaf` object is the default type of simple expression, which is in the form _field-property operator-property argument-property_ where:\n *\n * * _field-property_ - the name of a column, selected from a drop-down;\n * * _operator-property_ - an equality (=), inequality (<, ≤, ≠, ≥, >), or pattern operator (LIKE, NOT LIKE), also selected from a drop-down; and\n * * _argument-property_ is a constant typed into a text box.\n *\n * The `FilterTree` object has polymorphic methods that operate on the entire tree using recursion. When the recursion reaches a terminal node, it calls the methods on the `FilterLeaf` object instead. Calling `test()` on the root tree therefore returns a boolean that determines if the row passes through the entire filter expression (`true`) or is blocked by it (`false`).\n *\n * The programmer may define a new type of simple expression by extending from `FilterLeaf`. An example is the `FilterField` object. Such an implementation must include methods:\n *\n * * Save and subsequently reload the state of the conditional as entered by the user (`toJSON()` and `setState()`, respectively).\n * * Create the DOM objects that represent the UI filter editor and render them to the UI (`newView()` and `render()`, respectively).\n * * Filter a table by implementing one or more of the following:\n *   * Apply the conditional logic to available table row data (`test()`).\n *   * Apply the conditional logic to a remote data-store by generating a **SQL** or **Q** _WHERE_ clause (`toSQL()` and `toQ()`, respectively).\n *\n * Some of the above-named methods as already implemented in `FilterLeaf` and/or `FilterNode` may be sufficient to handle your needs as is (without further code).\n *\n * @param {string[]} [options.fields] - A default list of column names for field drop-downs of all descendant terminal nodes. Overrides `options.state.fields` (see). May be defined for any node and pertains to all descendants of that node (including terminal nodes). If omitted (and no `nodeFields`), will use the nearest ancestor `fields` definition. However, descendants with their own definition of `types` will override any ancestor definition.\n *\n * > Typically only used by the caller for the top-level (root) tree.\n *\n * @param {string[]} [options.nodeFields] - A default list of column names for field drop-downs of immediate descendant terminal nodes _only_. Overrides `options.state.nodeFields` (see).\n *\n * Although both `options.fields` and `options.nodeFields` are notated as optional herein, by the time a terminal node tries to render a fields drop-down, a `fields` list _must_ be defined through (in order of priority):\n *\n * * Terminal node's own `options.fields` (or `options.state.fields`) definition.\n * * Terminal node's parent node's `option.nodeFields` (or `option.state.nodesFields`) definition.\n * * Any of terminal node's ancestor's `options.fields` (or `options.state.fields`) definition.\n *\n * @param {object} [options.state] - A data structure that describes a tree, subtree, or leaf:\n *\n * * May describe a terminal node with properties:\n *   * `fields` - Overridden on instantiation by `options.fields`. If both unspecified, uses parent's definition.\n *   * `editor` - A string identifying the type of conditional. Must be in the tree's (see {@link FilterTree#editors|editors}) hash. If omitted, defaults to `'Default'`.\n *   * misc. - Other properties peculiar to this filter type (but typically including at least a `field` property).\n * * May describe a non-terminal node with properties:\n *   * `fields` - Overridden on instantiation by `options.fields`. If both unspecified, uses parent's definition.\n *   * `operator` - One of {@link treeOperators}.\n *   * `children` -  Array containing additional terminal and non-terminal nodes.\n *\n * If this `options.state` object is omitted altogether, loads an empty filter, which is a `FilterTree` node consisting the default `operator` value (`'op-and'`).\n *\n * > Note that this is a JSON object; not a JSON string (_i.e.,_ \"parsed\"; not \"stringified\").\n *\n * @param {function} [options.editor='Default'] - Type of simple expression.\n *\n * @param {FilterTree} [options.parent] - Used internally to insert element when creating nested subtrees. For the top level tree, you don't give a value for `parent`; you are responsible for inserting the top-level `.el` into the DOM.\n */\nvar FilterNode = Base.extend({\n\n    initialize: function(options) {\n        var parent = options && options.parent,\n            state = options && (\n                options.state ||\n                options.json && JSON.parse(options.json)\n            );\n\n        this.parent = parent;\n\n        /** Default list of fields only for direct child terminal-node drop-downs.\n         * @type {string[]}\n         * @memberOf FilterNode.prototype\n         */\n        this.nodeFields = setOption('nodeFields', options, state);\n\n        /** Default list of fields for all descending terminal-node drop-downs.\n         * @type {string[]}\n         * @memberOf FilterNode.prototype\n         */\n        this.fields = setOption('fields', options, state, parent);\n\n        /** Type of filter editor.\n         * @type {string}\n         * @memberOf FilterNode.prototype\n         */\n        this.editor = setOption('editor', options, state, parent);\n\n        /** Event handler for UI events.\n         * @type {string}\n         * @memberOf FilterNode.prototype\n         */\n        this.eventHandler = setOption('eventHandler', options, state, parent);\n\n        this.setState(state);\n    },\n\n    /** Insert each subtree into its parent node along with a \"delete\" button.\n     * > The root tree is has no parent and is inserted into the DOM by the instantiating code (without a delete button).\n     */\n    render: function() {\n        if (this.parent) {\n            var newListItem = document.createElement(CHILD_TAG);\n            newListItem.appendChild(template('removeButton'));\n            newListItem.appendChild(this.el);\n            this.parent.el.querySelector(CHILDREN_TAG).appendChild(newListItem);\n        }\n    },\n\n    setState: function(state) {\n        var oldEl = this.el;\n        this.newView();\n        this.load(state);\n        this.render();\n        if (oldEl && !this.parent) {\n            oldEl.parentNode.replaceChild(this.el, oldEl);\n        }\n    },\n\n    toJSON: function toJSON() {\n        var state = {};\n\n        if (this.toJsonOptions) {\n            var tree = this, metadata = [];\n            if (this.toJsonOptions.fields) {\n                metadata.push('fields');\n                metadata.push('nodeFields');\n            }\n            if (this.toJsonOptions.editor) {\n                metadata.push('editor');\n            }\n            metadata.forEach(function(prop) {\n                if (!tree.parent || tree[prop] && tree[prop] !== tree.parent[prop]) {\n                    state[prop] = tree[prop];\n                }\n            });\n        }\n\n        return state;\n    },\n\n    SQL_QUOTED_IDENTIFIER: '\"'\n\n});\n\nfunction setOption(key, options, state, parent) {\n    return (\n        options && options[key] ||\n        state && state[key] ||\n        parent && parent[key] // reference parent value now so we don't have to search up the tree later\n    );\n}\n\nFilterNode.setWarningClass = function(el, value) {\n    if (arguments.length < 2) {\n        value = el.value;\n    }\n    el.classList[value ? 'remove' : 'add']('filter-tree-warning');\n    return value;\n\n};\n\nFilterNode.Error = function(msg) {\n    return new Error('filter-tree: ' + msg);\n};\n\nFilterNode.clickIn = function(el) {\n    if (el) {\n        if (el.tagName === 'SELECT') {\n            setTimeout(function() { el.dispatchEvent(new MouseEvent('mousedown')); }, 0);\n        } else {\n            el.focus();\n        }\n    }\n};\n\nmodule.exports = FilterNode;\n","'use strict';\n\nvar cssInjector = require('css-injector');\n\nvar css; // defined by code inserted by gulpfile between following comments\n/* inject:css */\ncss = '.filter-tree{font-family:sans-serif;font-size:10pt;line-height:1.5em}.filter-tree label{font-weight:400}.filter-tree input[type=checkbox],.filter-tree input[type=radio]{left:3px;margin-right:3px}.filter-tree ol{margin-top:0}.filter-tree-add,.filter-tree-add-filter,.filter-tree-remove{cursor:pointer}.filter-tree-add,.filter-tree-add-filter{font-style:italic;color:#444;font-size:90%}.filter-tree-add-filter{margin:3px 0 3px 3em;width:120px;display:inline-block}.filter-tree-add-filter:hover,.filter-tree-add:hover{text-decoration:underline}.filter-tree-add-filter.as-menu-header,.filter-tree-add.as-menu-header{background-color:#fff;font-weight:700;font-style:normal}.filter-tree-add-filter.as-menu-header:hover{text-decoration:inherit}.filter-tree-add-filter>div,.filter-tree-add>div,.filter-tree-remove{display:inline-block;width:15px;height:15px;border-radius:8px;background-color:#8c8;font-size:11.5px;font-weight:700;color:#fff;text-align:center;line-height:normal;font-style:normal;font-family:sans-serif;text-shadow:0 0 1.5px grey;margin-right:4px}.filter-tree-add-filter>div:before,.filter-tree-add>div:before{content:\\'\\\\ff0b\\'}.filter-tree-remove{background-color:#e88;border:0}.filter-tree-remove:before{content:\\'\\\\2212\\'}.filter-tree li::after{font-size:70%;font-style:italic;font-weight:700;color:#080}.filter-tree>ol>li:last-child::after{display:none}.op-or>ol>li::after{margin-left:2.5em;content:\\'— OR —\\'}.op-and>ol>li::after{margin-left:2.5em;content:\\'— AND —\\'}.op-nor>ol>li::after{margin-left:2.5em;content:\\'— NOR —\\'}.filter-tree-default>:enabled{margin:0 .4em;background-color:#ddd;border:0}.filter-tree-default>input[type=text]{width:8em;padding:0 5px}.filter-tree-default>select{border:0}.filter-tree-default>.filter-tree-warning{background-color:#ffc}.filter-tree-default>.filter-tree-error{background-color:#Fcc}.filter-tree .footnotes{font-size:6pt;margin:2px 0 0;line-height:normal;white-space:normal;color:#999}.filter-tree .footnotes>ol{margin:0;padding-left:2em}.filter-tree .footnotes>ol>li{margin:2px 0}.filter-tree .footnotes .field-name,.filter-tree .footnotes .field-value{font-weight:700;color:#777}.filter-tree .footnotes .field-value:after,.filter-tree .footnotes .field-value:before{content:\\'\\\"\\'}.filter-tree .footnotes .field-value{font-family:monospace}.filter-tree-chooser{position:absolute;font-size:9pt;outline:0;box-shadow:5px 5px 10px grey}';\n/* endinject */\n\nmodule.exports = cssInjector.bind(this, css);\n","'use strict';\n\nvar regExpLIKE = require('regexp-like');\n\nvar LIKE = 'LIKE ',\n    NOT_LIKE = 'NOT ' + LIKE,\n    LIKE_WILD_CARD = '%';\n\nvar leafOperators = {\n    '<': {\n        test: function(a, b) { return a < b; }\n    },\n    '\\u2264': {\n        test: function(a, b) { return a <= b; },\n        sql: '<='\n    },\n    '=': {\n        test: function(a, b) { return a === b; }\n    },\n    '\\u2265': {\n        test: function(a, b) { return a >= b; },\n        sql: '>='\n    },\n    '>': {\n        test: function(a, b) { return a > b; }\n    },\n    '\\u2260': {\n        test: function(a, b) { return a !== b; },\n        sql: '<>'\n    },\n    LIKE: {\n        test: function(a, b) { return regExpLIKE.cached(b, true).test(a); },\n        type: 'string'\n    },\n    'NOT LIKE': {\n        test: function(a, b) { return !regExpLIKE.cached(b, true).test(a); },\n        type: 'string'\n    },\n    IN: { // TODO: currently forcing string typing; rework calling code to respect column type\n        test: function(a, b) { return inOp(a, b) >= 0; },\n        sql: sqlIN,\n        type: 'string'\n    },\n    'NOT IN': { // TODO: currently forcing string typing; rework calling code to respect column type\n        test: function(a, b) { return inOp(a, b) < 0; },\n        sql: sqlNotIN,\n        type: 'string'\n    },\n    CONTAINS: {\n        test: function(a, b) { return containsOp(a, b) >= 0; },\n        sql: sqlLkeExpression.bind(this, LIKE_WILD_CARD, LIKE_WILD_CARD, LIKE),\n        type: 'string'\n    },\n    'NOT CONTAINS': {\n        test: function(a, b) { return containsOp(a, b) < 0; },\n        sql: sqlLkeExpression.bind(this, LIKE_WILD_CARD, LIKE_WILD_CARD, NOT_LIKE),\n        type: 'string'\n    },\n    BEGINS: {\n        test: function(a, b) { b = b.toString().toLowerCase(); return beginsOp(a, b.length) === b; },\n        sql: sqlLkeExpression.bind(this, '', LIKE_WILD_CARD, LIKE),\n        type: 'string'\n    },\n    'NOT BEGINS': {\n        test: function(a, b) { b = b.toString().toLowerCase(); return beginsOp(a, b.length) !== b; },\n        sql: sqlLkeExpression.bind(this, '', LIKE_WILD_CARD, NOT_LIKE),\n        type: 'string'\n    },\n    ENDS: {\n        test: function(a, b) { b = b.toString().toLowerCase(); return endsOp(a, b.length) === b; },\n        sql: sqlLkeExpression.bind(this, LIKE_WILD_CARD, '', LIKE),\n        type: 'string'\n    },\n    'NOT ENDS': {\n        test: function(a, b) { b = b.toString().toLowerCase(); return endsOp(a, b.length) !== b; },\n        sql: sqlLkeExpression.bind(this, LIKE_WILD_CARD, '', NOT_LIKE),\n        type: 'string'\n    }\n};\n\nfunction inOp(a, b) {\n    return b\n        .trim() // remove leading and trailing space chars\n        .replace(/\\s*,\\s*/g, ',') // remove any white-space chars from around commas\n        .split(',') // put in an array\n        .indexOf(a.toString()); // search array whole matches\n}\n\nfunction containsOp(a, b) {\n    return a.toString().toLowerCase().indexOf(b.toString().toLowerCase());\n}\n\nfunction beginsOp(a, length) {\n    return a.toString().toLowerCase().substr(0, length);\n}\n\nfunction endsOp(a, length) {\n    return a.toString().toLowerCase().substr(-length, length);\n}\n\n// List the operators as drop-down options in an hierarchical array (rendered as option groups):\n\nvar equality, inequalities, sets, strings, patterns;\n\nequality = ['='];\nequality.label = 'Equality';\n\ninequalities = ['<', '\\u2264', '\\u2260', '\\u2265', '>'];\ninequalities.label = 'Inquality';\n\nsets = ['IN', 'NOT IN'];\nsets.label = 'Set scan';\n\nstrings = [\n    'CONTAINS', 'NOT CONTAINS',\n    'BEGINS', 'NOT BEGINS',\n    'ENDS', 'NOT ENDS'\n];\nstrings.label = 'String scan';\n\n// Alternatively, option groups can also be set up as an object with .options and .label properties:\n\npatterns = { options: ['LIKE', 'NOT LIKE'], label: 'Pattern matching' };\n\nleafOperators.options = [\n    equality,\n    inequalities,\n    sets,\n    strings,\n    patterns\n];\n\nfunction sqlLkeExpression(beg, end, LIKE_OR_NOT_LIKE, likePattern) {\n    var escaped = likePattern.replace(/([\\[_%\\]])/g, '[$1]'); // escape all LIKE reserved chars\n    return LIKE_OR_NOT_LIKE + sq(beg + escaped + end);\n}\n\nfunction sqlIN(b) {\n    return 'IN (\\'' + sqEsc(b).replace(/\\s*,\\s*/g, '\\', \\'') + '\\')';\n}\n\nfunction sqlNotIN(b) {\n    return 'NOT ' + sqlIN(b);\n}\n\nfunction sqEsc(sqlString) {\n    return sqlString.replace(/'/g, '\\'\\'');\n}\n\nfunction sq(sqlString) {\n    return ' \\'' + sqEsc(sqlString) + '\\'';\n}\n\nleafOperators.sq = sq;\n\nmodule.exports = leafOperators;\n","/* eslint-env browser */\n\n'use strict';\n\nvar templex = require('templex');\n\nvar templates = {\n\n    tree: function() {\n        /*\n        <span class=\"filter-tree\">\n            Match\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-or\">any</label>\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-and\">all</label>\n            <label><input type=\"radio\" class=\"filter-tree-op-choice\" name=\"treeOp{1}\" value=\"op-nor\">none</label>\n            of:<br/>\n            <span class=\"filter-tree-add-filter\" title=\"Add a new conditional to this match.\">\n                <div></div>conditional\n            </span>\n            <span class=\"filter-tree-add\" title=\"Add a new sub-match under this match.\">\n                <div></div>subexpression\n            </span>\n            <ol></ol>\n        </span>\n        */\n    },\n\n    removeButton: function() {\n        /*\n        <div class=\"filter-tree-remove\" title=\"delete conditional\"></div>\n        */\n    },\n\n    note: function() {\n        /*\n        <div class=\"footnotes\">\n            <em>Note regarding the above expression:</em>\n            <span></span>\n            Select a new value or delete the expression altogether.\n        </div>\n        */\n    },\n\n    notes: function() {\n        /*\n         <div class=\"footnotes\">\n            <em>Notes regarding the above expression:</em>\n            <ol></ol>\n            Select new values or delete the expression altogether.\n         </div>\n         */\n    },\n\n    optionMissing: function() {\n        /*\n        The previous <span class=\"field-name\">{1:encode}</span>\n        value <span class=\"field-value\">{2:encode}</span>\n        is no longer valid.\n        */\n    }\n\n};\n\nvar extract = /\\/\\*\\s*([^]+?)\\s+\\*\\//; // finds the string inside the /* ... */; the group excludes the whitespace\nvar encoders = /\\{(\\d+)\\:encode\\}/g;\n\nfunction get(templateName) {\n    var temp = document.createElement('div');\n    var text = templates[templateName].toString().match(extract)[1];\n    var templexArgs = [text].concat(Array.prototype.slice.call(arguments, 1));\n    var keys, encoder = {};\n\n    encoders.lastIndex = 0;\n    while ((keys = encoders.exec(text))) {\n        encoder[keys[1]] = true;\n    }\n    keys = Object.keys(encoder);\n    if (keys.length) {\n        keys.forEach(function(key) {\n            temp.textContent = templexArgs[key];\n            templexArgs[key] = temp.innerHTML;\n        });\n        templexArgs[0] = text.replace(encoders, '{$1}');\n    }\n\n    temp.innerHTML = templex.apply(this, templexArgs);\n\n    // if only one HTMLElement, return it; otherwise entire list of nodes\n    return temp.children.length === 1 && temp.childNodes.length === 1 ? temp.firstChild : temp.childNodes;\n}\n\nmodule.exports = get;\n","'use strict';\n\n/** @typedef {function} operationReducer\n * @param {boolean} p\n * @param {boolean} q\n * @returns {boolean} The result of applying the operator to the two parameters.\n */\n\n/**\n * @private\n * @type {operationReducer}\n */\nfunction AND(p, q) {\n    return p && q;\n}\n\n/**\n * @private\n * @type {operationReducer}\n */\nfunction OR(p, q) {\n    return p || q;\n}\n\n/** @typedef {obejct} treeOperator\n * @desc Each `treeOperator` object describes two things:\n *\n * 1. How to take the test results of _n_ child nodes by applying the operator to all the results to \"reduce\" it down to a single result.\n * 2. How to generate SQL WHERE clause syntax that applies the operator to _n_ child nodes.\n *\n * @property {operationReducer} reduce\n * @property {boolean} seed -\n * @property {boolean} abort -\n * @property {boolean} negate -\n * @property {string} SQL.op -\n * @property {string} SQL.beg -\n * @property {string} SQL.end -\n */\n\n/** A hash of {@link treeOperator} objects.\n * @type {object}\n */\nvar treeOperators = {\n    'op-and': {\n        reduce: AND,\n        seed: true,\n        abort: false,\n        negate: false,\n        SQL: {\n            op: 'AND',\n            beg: '(',\n            end: ')'\n        }\n    },\n    'op-or': {\n        reduce: OR,\n        seed: false,\n        abort: true,\n        negate: false,\n        SQL: {\n            op: 'OR',\n            beg: '(',\n            end: ')'\n        }\n    },\n    'op-nor': {\n        reduce: OR,\n        seed: false,\n        abort: true,\n        negate: true,\n        SQL: {\n            op: 'OR',\n            beg: 'NOT (',\n            end: ')'\n        }\n    }\n};\n\nmodule.exports = treeOperators;\n","'use strict';\n\n/* eslint-env node, browser */\n\nvar cssInjector = require('css-injector');\n\n/**\n * @constructor FinBar\n * @summary Create a scrollbar object.\n * @desc Creating a scrollbar is a three-step process:\n *\n * 1. Instantiate the scrollbar object by calling this constructor function. Upon instantiation, the DOM element for the scrollbar (with a single child element for the scrollbar \"thumb\") is created but is not insert it into the DOM.\n * 2. After instantiation, it is the caller's responsibility to insert the scrollbar, {@link FinBar#bar|this.bar}, into the DOM.\n * 3. After insertion, the caller must call {@link FinBar#resize|resize()} at least once to size and position the scrollbar and its thumb. After that, `resize()` should also be called repeatedly on resize events (as the content element is being resized).\n *\n * Suggested configurations:\n * * _**Unbound**_<br/>\n * The scrollbar serves merely as a simple range (slider) control. Omit both `options.onchange` and `options.content`.\n * * _**Bound to virtual content element**_<br/>\n * Virtual content is projected into the element using a custom event handler supplied by the programmer in `options.onchange`. A typical use case would be to handle scrolling of the virtual content. Other use cases include data transformations, graphics transformations, _etc._\n * * _**Bound to real content**_<br/>\n * Set `options.content` to the \"real\" content element but omit `options.onchange`. This will cause the scrollbar to use the built-in event handler (`this.scrollRealContent`) which implements smooth scrolling of the content element within the container.\n *\n * @param {finbarOptions} [options={}] - Options object. See the type definition for member details.\n */\nfunction FinBar(options) {\n\n    // make bound versions of all the mouse event handler\n    var bound = this._bound = {};\n    for (key in handlersToBeBound) {\n        bound[key] = handlersToBeBound[key].bind(this);\n    }\n\n    /**\n     * @name thumb\n     * @summary The generated scrollbar thumb element.\n     * @desc The thumb element's parent element is always the {@link FinBar#bar|bar} element.\n     *\n     * This property is typically referenced internally only. The size and position of the thumb element is maintained by `_calcThumb()`.\n     * @type {Element}\n     * @memberOf FinBar.prototype\n     */\n    var thumb = document.createElement('div');\n    thumb.classList.add('thumb');\n    thumb.onclick = bound.shortStop;\n    thumb.onmouseover = bound.onmouseover;\n    this.thumb = thumb;\n\n    /**\n     * @name bar\n     * @summary The generated scrollbar element.\n     * @desc The caller inserts this element into the DOM (typically into the content container) and then calls its {@link FinBar#resize|resize()} method.\n     *\n     * Thus the node tree is typically:\n     * * A **content container** element, which contains:\n     *    * The content element(s)\n     *    * This **scrollbar element**, which in turn contains:\n     *        * The **thumb element**\n     *\n     * @type {Element}\n     * @memberOf FinBar.prototype\n     */\n    var bar = document.createElement('div');\n\n    bar.classList.add('finbar-vertical');\n\n    bar.appendChild(thumb);\n    if (this.paging) {\n        bar.onclick = bound.onclick;\n    }\n    this.bar = bar;\n\n    options = options || {};\n\n    // presets\n    this.orientation = 'vertical';\n    this._min = this._index = 0;\n    this._max = 100;\n\n    // options\n    for (var key in options) {\n        if (options.hasOwnProperty(key)) {\n            var option = options[key];\n            switch (key) {\n\n            case 'index':\n                this._index = option;\n                break;\n\n            case 'range':\n                validRange(option);\n                this._min = option.min;\n                this._max = option.max;\n                this.contentSize = option.max - option.min + 1;\n                break;\n\n            default:\n                if (\n                    key.charAt(0) !== '_' &&\n                    typeof FinBar.prototype[key] !== 'function'\n                ) {\n                    // override prototype defaults for standard ;\n                    // extend with additional properties (for use in onchange event handlers)\n                    this[key] = option;\n                }\n                break;\n\n            }\n        }\n    }\n\n    cssInjector(cssFinBars, 'finbar-base', options.cssStylesheetReferenceElement);\n}\n\nFinBar.prototype = {\n\n    /**\n     * @summary The scrollbar orientation.\n     * @desc Set by the constructor to either `'vertical'` or `'horizontal'`. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * Useful values are `'vertical'` (the default) or `'horizontal'`.\n     *\n     * Setting this property resets `this.oh` and `this.deltaProp` and changes the class names so as to reposition the scrollbar as per the CSS rules for the new orientation.\n     * @default 'vertical'\n     * @type {string}\n     * @memberOf FinBar.prototype\n     */\n    set orientation(orientation) {\n        if (orientation === this._orientation) {\n            return;\n        }\n\n        this._orientation = orientation;\n\n        /**\n         * @readonly\n         * @name oh\n         * @summary <u>O</u>rientation <u>h</u>ash for this scrollbar.\n         * @desc Set by the `orientation` setter to either the vertical or the horizontal orientation hash. The property should always be synchronized with `orientation`; do not update directly!\n         *\n         * This object is used internally to access scrollbars' DOM element properties in a generalized way without needing to constantly query the scrollbar orientation. For example, instead of explicitly coding `this.bar.top` for a vertical scrollbar and `this.bar.left` for a horizontal scrollbar, simply code `this.bar[this.oh.leading]` instead. See the {@link orientationHashType} definition for details.\n         *\n         * This object is useful externally for coding generalized {@link finbarOnChange} event handler functions that serve both horizontal and vertical scrollbars.\n         * @type {orientationHashType}\n         * @memberOf FinBar.prototype\n         */\n        this.oh = orientationHashes[this._orientation];\n\n        if (!this.oh) {\n            error('Invalid value for `options._orientation.');\n        }\n\n        /**\n         * @name deltaProp\n         * @summary The name of the `WheelEvent` property this scrollbar should listen to.\n         * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n         *\n         * Useful values are `'deltaX'`, `'deltaY'`, or `'deltaZ'`. A value of `null` means to ignore mouse wheel events entirely.\n         *\n         * The mouse wheel is one-dimensional and only emits events with `deltaY` data. This property is provided so that you can override the default of `'deltaX'` with a value of `'deltaY'` on your horizontal scrollbar primarily to accommodate certain \"panoramic\" interface designs where the mouse wheel should control horizontal rather than vertical scrolling. Just give `{ deltaProp: 'deltaY' }` in your horizontal scrollbar instantiation.\n         *\n         * Caveat: Note that a 2-finger drag on an Apple trackpad emits events with _both_ `deltaX ` and `deltaY` data so you might want to delay making the above adjustment until you can determine that you are getting Y data only with no X data at all (which is a sure bet you on a mouse wheel rather than a trackpad).\n\n         * @type {object|null}\n         * @memberOf FinBar.prototype\n         */\n        this.deltaProp = this.oh.delta;\n\n        this.bar.className = this.bar.className.replace(/(vertical|horizontal)/g, orientation);\n\n        if (this.bar.style.cssText || this.thumb.style.cssText) {\n            this.bar.removeAttribute('style');\n            this.thumb.removeAttribute('style');\n            this.resize();\n        }\n    },\n    get orientation() {\n        return this._orientation;\n    },\n\n    /**\n     * @summary Callback for scroll events.\n     * @desc Set by the constructor via the similarly named property in the {@link finbarOptions} object. After instantiation, `this.onchange` may be updated directly.\n     *\n     * This event handler is called whenever the value of the scrollbar is changed through user interaction. The typical use case is when the content is scrolled. It is called with the `FinBar` object as its context and the current value of the scrollbar (its index, rounded) as the only parameter.\n     *\n     * Set this property to `null` to stop emitting such events.\n     * @type {function(number)|null}\n     * @memberOf FinBar.prototype\n     */\n    onchange: null,\n\n    /**\n     * @summary Add a CSS class name to the bar element's class list.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * The bar element's class list will always include `finbar-vertical` (or `finbar-horizontal` based on the current orientation). Whenever this property is set to some value, first the old prefix+orientation is removed from the bar element's class list; then the new prefix+orientation is added to the bar element's class list. This property causes _an additional_ class name to be added to the bar element's class list. Therefore, this property will only add at most one additional class name to the list.\n     *\n     * To remove _classname-orientation_ from the bar element's class list, set this property to a falsy value, such as `null`.\n     *\n     * > NOTE: You only need to specify an additional class name when you need to have mulltiple different styles of scrollbars on the same page. If this is not a requirement, then you don't need to make a new class; you would just create some additional rules using the same selectors in the built-in stylesheet (../css/finbars.css):\n     * *`div.finbar-vertical` (or `div.finbar-horizontal`) for the scrollbar\n     * *`div.finbar-vertical > div` (or `div.finbar-horizontal > div`) for the \"thumb.\"\n     *\n     * Of course, your rules should come after the built-ins.\n     * @type {string}\n     * @memberOf FinBar.prototype\n     */\n    set classPrefix(prefix) {\n        if (this._classPrefix) {\n            this.bar.classList.remove(this._classPrefix + this.orientation);\n        }\n\n        this._classPrefix = prefix;\n\n        if (prefix) {\n            this.bar.classList.add(prefix + '-' + this.orientation);\n        }\n    },\n    get classPrefix() {\n        return this._classPrefix;\n    },\n\n    /**\n     * @name increment\n     * @summary Number of scrollbar index units representing a pageful. Used exclusively for paging up and down and for setting thumb size relative to content size.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * Can also be given as a parameter to the {@link FinBar#resize|resize} method, which is pertinent because content area size changes affect the definition of a \"pageful.\" However, you only need to do this if this value is being used. It not used when:\n     * * you define `paging.up` and `paging.down`\n     * * your scrollbar is using `scrollRealContent`\n     * @type {number}\n     * @memberOf FinBar.prototype\n     */\n    increment: 1,\n\n    /**\n     * @name barStyles\n     * @summary Scrollbar styles to be applied by {@link FinBar#resize|resize()}.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * This is a value to be assigned to {@link FinBar#styles|styles} on each call to {@link FinBar#resize|resize()}. That is, a hash of values to be copied to the scrollbar element's style object on resize; or `null` for none.\n     *\n     * @see {@link FinBar#style|style}\n     * @type {finbarStyles|null}\n     * @memberOf FinBar.prototype\n     */\n    barStyles: null,\n\n    /**\n     * @name style\n     * @summary Additional scrollbar styles.\n     * @desc See type definition for more details. These styles are applied directly to the scrollbar's `bar` element.\n     *\n     * Values are adjusted as follows before being applied to the element:\n     * 1. Included \"pseudo-property\" names from the scrollbar's orientation hash, {@link FinBar#oh|oh}, are translated to actual property names before being applied.\n     * 2. When there are margins, percentages are translated to absolute pixel values because CSS ignores margins in its percentage calculations.\n     * 3. If you give a value without a unit (a raw number), \"px\" unit is appended.\n     *\n     * General notes:\n     * 1. It is always preferable to specify styles via a stylesheet. Only set this property when you need to specifically override (a) stylesheet value(s).\n     * 2. Can be set directly or via calls to the {@link FinBar#resize|resize} method.\n     * 3. Should only be set after the scrollbar has been inserted into the DOM.\n     * 4. Before applying these new values to the element, _all_ in-line style values are reset (by removing the element's `style` attribute), exposing inherited values (from stylesheets).\n     * 5. Empty object has no effect.\n     * 6. Falsey value in place of object has no effect.\n     *\n     * > CAVEAT: Do not attempt to treat the object you assign to this property as if it were `this.bar.style`. Specifically, changing this object after assigning it will have no effect on the scrollbar. You must assign it again if you want it to have an effect.\n     *\n     * @see {@link FinBar#barStyles|barStyles}\n     * @type {finbarStyles}\n     * @memberOf FinBar.prototype\n     */\n    set style(styles) {\n        var keys = Object.keys(styles = extend({}, styles, this._auxStyles));\n\n        if (keys.length) {\n            var bar = this.bar,\n                barRect = bar.getBoundingClientRect(),\n                container = this.container || bar.parentElement,\n                containerRect = container.getBoundingClientRect(),\n                oh = this.oh;\n\n            // Before applying new styles, revert all styles to values inherited from stylesheets\n            bar.removeAttribute('style');\n\n            keys.forEach(function (key) {\n                var val = styles[key];\n\n                if (key in oh) {\n                    key = oh[key];\n                }\n\n                if (!isNaN(Number(val))) {\n                    val = (val || 0) + 'px';\n                } else if (/%$/.test(val)) {\n                    // When bar size given as percentage of container, if bar has margins, restate size in pixels less margins.\n                    // (If left as percentage, CSS's calculation will not exclude margins.)\n                    var oriented = axis[key],\n                        margins = barRect[oriented.marginLeading] + barRect[oriented.marginTrailing];\n                    if (margins) {\n                        val = parseInt(val, 10) / 100 * containerRect[oriented.size] - margins + 'px';\n                    }\n                }\n\n                bar.style[key] = val;\n            });\n        }\n    },\n\n    /**\n     * @readonly\n     * @name paging\n     * @summary Enable page up/dn clicks.\n     * @desc Set by the constructor. See the similarly named property in the {@link finbarOptions} object.\n     *\n     * If truthy, listen for clicks in page-up and page-down regions of scrollbar.\n     *\n     * If an object, call `.paging.up()` on page-up clicks and `.paging.down()` will be called on page-down clicks.\n     *\n     * Changing the truthiness of this value after instantiation currently has no effect.\n     * @type {boolean|object}\n     * @memberOf FinBar.prototype\n     */\n    paging: true,\n\n    /**\n     * @name range\n     * @summary Setter for the minimum and maximum scroll values.\n     * @desc Set by the constructor. These values are the limits for {@link FooBar#index|index}.\n     *\n     * The setter accepts an object with exactly two numeric properties: `.min` which must be less than `.max`. The values are extracted and the object is discarded.\n     *\n     * The getter returns a new object with `.min` and '.max`.\n     *\n     * @type {rangeType}\n     * @memberOf FinBar.prototype\n     */\n    set range(range) {\n        validRange(range);\n        this._min = range.min;\n        this._max = range.max;\n        this.contentSize = range.max - range.min + 1;\n        this.index = this.index; // re-clamp\n    },\n    get range() {\n        return {\n            min: this._min,\n            max: this._max\n        };\n    },\n\n    /**\n     * @summary Index value of the scrollbar.\n     * @desc This is the position of the scroll thumb.\n     *\n     * Setting this value clamps it to {@link FinBar#min|min}..{@link FinBar#max|max}, scroll the content, and moves thumb.\n     *\n     * Getting this value returns the current index. The returned value will be in the range `min`..`max`. It is intentionally not rounded.\n     *\n     * Use this value as an alternative to (or in addition to) using the {@link FinBar#onchange|onchange} callback function.\n     *\n     * @see {@link FinBar#_setScroll|_setScroll}\n     * @type {number}\n     * @memberOf FinBar.prototype\n     */\n    set index(idx) {\n        idx = Math.min(this._max, Math.max(this._min, idx)); // clamp it\n        this._setScroll(idx);\n        // this._setThumbSize();\n    },\n    get index() {\n        return this._index;\n    },\n\n    /**\n     * @private\n     * @summary Move the thumb.\n     * @desc Also displays the index value in the test panel and invokes the callback.\n     * @param idx - The new scroll index, a value in the range `min`..`max`.\n     * @param [scaled=f(idx)] - The new thumb position in pixels and scaled relative to the containing {@link FinBar#bar|bar} element, i.e., a proportional number in the range `0`..`thumbMax`. When omitted, a function of `idx` is used.\n     * @memberOf FinBar.prototype\n     */\n    _setScroll: function (idx, scaled) {\n        this._index = idx;\n\n        // Display the index value in the test panel\n        if (this.testPanelItem && this.testPanelItem.index instanceof Element) {\n            this.testPanelItem.index.innerHTML = Math.round(idx);\n        }\n\n        // Call the callback\n        if (this.onchange) {\n            this.onchange.call(this, Math.round(idx));\n        }\n\n        // Move the thumb\n        if (scaled === undefined) {\n            scaled = (idx - this._min) / (this._max - this._min) * this._thumbMax;\n        }\n        this.thumb.style[this.oh.leading] = scaled + 'px';\n    },\n\n    scrollRealContent: function (idx) {\n        var containerRect = this.content.parentElement.getBoundingClientRect(),\n            sizeProp = this.oh.size,\n            maxScroll = Math.max(0, this.content[sizeProp] - containerRect[sizeProp]),\n            //scroll = Math.min(idx, maxScroll);\n            scroll = (idx - this._min) / (this._max - this._min) * maxScroll;\n        //console.log('scroll: ' + scroll);\n        this.content.style[this.oh.leading] = -scroll + 'px';\n    },\n\n    /**\n     * @summary Recalculate thumb position.\n     *\n     * @desc This method recalculates the thumb size and position. Call it once after inserting your scrollbar into the DOM, and repeatedly while resizing the scrollbar (which typically happens when the scrollbar's parent is resized by user.\n     *\n     * > This function shifts args if first arg omitted.\n     *\n     * @param {number} [increment=this.increment] - Resets {@link FooBar#increment|increment} (see).\n     *\n     * @param {finbarStyles} [barStyles=this.barStyles] - (See type definition for details.) Scrollbar styles to be applied to the bar element.\n     *\n     * Only specify a `barStyles` object when you need to override stylesheet values. If provided, becomes the new default (`this.barStyles`), for use as a default on subsequent calls.\n     *\n     * It is generally the case that the scrollbar's new position is sufficiently described by the current styles. Therefore, it is unusual to need to provide a `barStyles` object on every call to `resize`.\n     *\n     * @returns {FinBar} Self for chaining.\n     * @memberOf FinBar.prototype\n     */\n    resize: function (increment, barStyles) {\n        var bar = this.bar;\n\n        if (!bar.parentNode) {\n            return; // not in DOM yet so nothing to do\n        }\n\n        var container = this.container || bar.parentElement,\n            containerRect = container.getBoundingClientRect();\n\n        // shift args if if 1st arg omitted\n        if (typeof increment === 'object') {\n            barStyles = increment;\n            increment = undefined;\n        }\n\n        this.style = this.barStyles = barStyles || this.barStyles;\n\n        // Bound to real content: Content was given but no onchange handler.\n        // Set up .onchange, .containerSize, and .increment.\n        // Note this only makes sense if your index unit is pixels.\n        if (this.content) {\n            if (!this.onchange) {\n                this.onchange = this.scrollRealContent;\n                this.contentSize = this.content[this.oh.size];\n                this._min = 0;\n                this._max = this.contentSize - 1;\n            }\n        }\n        if (this.onchange === this.scrollRealContent) {\n            this.containerSize = containerRect[this.oh.size];\n            this.increment = this.containerSize / (this.contentSize - this.containerSize) * (this._max - this._min);\n        } else {\n            this.containerSize = 1;\n            this.increment = increment || this.increment;\n        }\n\n        var index = this.index;\n        this.testPanelItem = this.testPanelItem || this._addTestPanelItem();\n        this._setThumbSize();\n        this.index = index;\n\n        if (this.deltaProp !== null) {\n            container.addEventListener('wheel', this._bound.onwheel);\n        }\n\n        return this;\n    },\n\n    /**\n     * @summary Shorten trailing end of scrollbar by thickness of some other scrollbar.\n     * @desc In the \"classical\" scenario where vertical scroll bar is on the right and horizontal scrollbar is on the bottom, you want to shorten the \"trailing end\" (bottom and right ends, respectively) of at least one of them so they don't overlay.\n     *\n     * This convenience function is an programmatic alternative to hardcoding the correct style with the correct value in your stylesheet; or setting the correct style with the correct value in the {@link FinBar#barStyles|barStyles} object.\n     *\n     * @see {@link FinBar#foreshortenBy|foreshortenBy}.\n     *\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    shortenBy: function (otherFinBar) { return this.shortenEndBy('trailing', otherFinBar); },\n\n    /**\n     * @summary Shorten leading end of scrollbar by thickness of some other scrollbar.\n     * @desc Supports non-classical scrollbar scenarios where vertical scroll bar may be on left and horizontal scrollbar may be on top, in which case you want to shorten the \"leading end\" rather than the trailing end.\n     * @see {@link FinBar#shortenBy|shortenBy}.\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    foreshortenBy: function (otherFinBar) { return this.shortenEndBy('leading', otherFinBar); },\n\n    /**\n     * @summary Generalized shortening function.\n     * @see {@link FinBar#shortenBy|shortenBy}.\n     * @see {@link FinBar#foreshortenBy|foreshortenBy}.\n     * @param {string} whichEnd - a CSS style property name or an orientation hash name that translates to a CSS style property name.\n     * @param {FinBar|null} otherFinBar - Other scrollbar to avoid by shortening this one; `null` removes the trailing space\n     * @returns {FinBar} For chaining\n     */\n    shortenEndBy: function (whichEnd, otherFinBar) {\n        if (!otherFinBar) {\n            delete this._auxStyles;\n        } else if (otherFinBar instanceof FinBar && otherFinBar.orientation !== this.orientation) {\n            var otherStyle = window.getComputedStyle(otherFinBar.bar),\n                ooh = orientationHashes[otherFinBar.orientation];\n            this._auxStyles = {};\n            this._auxStyles[whichEnd] = otherStyle[ooh.thickness];\n        }\n        return this; // for chaining\n    },\n\n    /**\n     * @private\n     * @summary Sets the proportional thumb size and hides thumb when 100%.\n     * @desc The thumb size has an absolute minimum of 20 (pixels).\n     * @memberOf FinBar.prototype\n     */\n    _setThumbSize: function () {\n        var oh = this.oh,\n            thumbComp = window.getComputedStyle(this.thumb),\n            thumbMarginLeading = parseInt(thumbComp[oh.marginLeading]),\n            thumbMarginTrailing = parseInt(thumbComp[oh.marginTrailing]),\n            thumbMargins = thumbMarginLeading + thumbMarginTrailing,\n            barSize = this.bar.getBoundingClientRect()[oh.size],\n            thumbSize = Math.max(20, barSize * this.containerSize / this.contentSize);\n\n        if (this.containerSize < this.contentSize) {\n            this.bar.style.visibility = 'visible';\n            this.thumb.style[oh.size] = thumbSize + 'px';\n        } else {\n            this.bar.style.visibility = 'hidden';\n        }\n\n        /**\n         * @private\n         * @name _thumbMax\n         * @summary Maximum offset of thumb's leading edge.\n         * @desc This is the pixel offset within the scrollbar of the thumb when it is at its maximum position at the extreme end of its range.\n         *\n         * This value takes into account the newly calculated size of the thumb element (including its margins) and the inner size of the scrollbar (the thumb's containing element, including _its_ margins).\n         *\n         * NOTE: Scrollbar padding is not taken into account and assumed to be 0 in the current implementation and is assumed to be `0`; use thumb margins in place of scrollbar padding.\n         * @type {number}\n         * @memberOf FinBar.prototype\n         */\n        this._thumbMax = barSize - thumbSize - thumbMargins;\n\n        this._thumbMarginLeading = thumbMarginLeading; // used in mousedown\n    },\n\n    /**\n     * @summary Remove the scrollbar.\n     * @desc Unhooks all the event handlers and then removes the element from the DOM. Always call this method prior to disposing of the scrollbar object.\n     * @memberOf FinBar.prototype\n     */\n    remove: function () {\n        this._removeEvt('mousedown');\n        this._removeEvt('mousemove');\n        this._removeEvt('mouseup');\n\n        (this.container || this.bar.parentElement)._removeEvt('wheel', this._bound.onwheel);\n\n        this.bar.onclick =\n            this.thumb.onclick =\n                this.thumb.onmouseover =\n                    this.thumb.transitionend =\n                        this.thumb.onmouseout = null;\n\n        this.bar.remove();\n    },\n\n    /**\n     * @private\n     * @function _addTestPanelItem\n     * @summary Append a test panel element.\n     * @desc If there is a test panel in the DOM (typically an `<ol>...</ol>` element) with class names of both `this.classPrefix` and `'test-panel'` (or, barring that, any element with class name `'test-panel'`), an `<li>...</li>` element will be created and appended to it. This new element will contain a span for each class name given.\n     *\n     * You should define a CSS selector `.listening` for these spans. This class will be added to the spans to alter their appearance when a listener is added with that class name (prefixed with 'on').\n     *\n     * (This is an internal function that is called once by the constructor on every instantiation.)\n     * @returns {Element|undefined} The appended `<li>...</li>` element or `undefined` if there is no test panel.\n     * @memberOf FinBar.prototype\n     */\n    _addTestPanelItem: function () {\n        var testPanelItem,\n            testPanelElement = document.querySelector('.' + this._classPrefix + '.test-panel') || document.querySelector('.test-panel');\n\n        if (testPanelElement) {\n            var testPanelItemPartNames = [ 'mousedown', 'mousemove', 'mouseup', 'index' ],\n                item = document.createElement('li');\n\n            testPanelItemPartNames.forEach(function (partName) {\n                item.innerHTML += '<span class=\"' + partName + '\">' + partName.replace('mouse', '') + '</span>';\n            });\n\n            testPanelElement.appendChild(item);\n\n            testPanelItem = {};\n            testPanelItemPartNames.forEach(function (partName) {\n                testPanelItem[partName] = item.getElementsByClassName(partName)[0];\n            });\n        }\n\n        return testPanelItem;\n    },\n\n    _addEvt: function (evtName) {\n        var spy = this.testPanelItem && this.testPanelItem[evtName];\n        if (spy) { spy.classList.add('listening'); }\n        window.addEventListener(evtName, this._bound['on' + evtName]);\n    },\n\n    _removeEvt: function (evtName) {\n        var spy = this.testPanelItem && this.testPanelItem[evtName];\n        if (spy) { spy.classList.remove('listening'); }\n        window.removeEventListener(evtName, this._bound['on' + evtName]);\n    }\n};\n\nfunction extend(obj) {\n    for (var i = 1; i < arguments.length; ++i) {\n        var objn = arguments[i];\n        if (objn) {\n            for (var key in objn) {\n                obj[key] = objn[key];\n            }\n        }\n    }\n    return obj;\n}\n\nfunction validRange(range) {\n    var keys = Object.keys(range),\n        valid =  keys.length === 2 &&\n            typeof range.min === 'number' &&\n            typeof range.max === 'number' &&\n            range.min <= range.max;\n\n    if (!valid) {\n        error('Invalid .range object.');\n    }\n}\n\n/**\n * @private\n * @name handlersToBeBound\n * @type {object}\n * @desc The functions defined in this object are all DOM event handlers that are bound by the FinBar constructor to each new instance. In other words, the `this` value of these handlers, once bound, refer to the FinBar object and not to the event emitter. \"Do not consume raw.\"\n */\nvar handlersToBeBound = {\n    shortStop: function (evt) {\n        evt.stopPropagation();\n    },\n\n    onwheel: function (evt) {\n        this.index += evt[this.deltaProp];\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onclick: function (evt) {\n        var thumbBox = this.thumb.getBoundingClientRect(),\n            goingUp = evt[this.oh.coordinate] < thumbBox[this.oh.leading];\n\n        if (typeof this.paging === 'object') {\n            this.index = this.paging[goingUp ? 'up' : 'down'](Math.round(this.index));\n        } else {\n            this.index += goingUp ? -this.increment : this.increment;\n        }\n\n        // make the thumb glow momentarily\n        this.thumb.classList.add('hover');\n        var self = this;\n        this.thumb.addEventListener('transitionend', function waitForIt() {\n            this.removeEventListener('transitionend', waitForIt);\n            self._bound.onmouseup(evt);\n        });\n\n        evt.stopPropagation();\n    },\n\n    onmouseover: function () {\n        this.thumb.classList.add('hover');\n        this.thumb.onmouseout = this._bound.onmouseout;\n        this._addEvt('mousedown');\n    },\n\n    onmouseout: function () {\n        this._removeEvt('mousedown');\n        this.thumb.onmouseover = this._bound.onmouseover;\n        this.thumb.classList.remove('hover');\n    },\n\n    onmousedown: function (evt) {\n        this._removeEvt('mousedown');\n        this.thumb.onmouseover = this.thumb.onmouseout = null;\n\n        var thumbBox = this.thumb.getBoundingClientRect();\n        this.pinOffset = evt[this.oh.axis] - thumbBox[this.oh.leading] + this.bar.getBoundingClientRect()[this.oh.leading] + this._thumbMarginLeading;\n        document.documentElement.style.cursor = 'default';\n\n        this._addEvt('mousemove');\n        this._addEvt('mouseup');\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onmousemove: function (evt) {\n        var scaled = Math.min(this._thumbMax, Math.max(0, evt[this.oh.axis] - this.pinOffset));\n        var idx = scaled / this._thumbMax * (this._max - this._min) + this._min;\n\n        this._setScroll(idx, scaled);\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    },\n\n    onmouseup: function (evt) {\n        this._removeEvt('mousemove');\n        this._removeEvt('mouseup');\n\n        document.documentElement.style.cursor = 'auto';\n\n        var thumbBox = this.thumb.getBoundingClientRect();\n        if (\n            thumbBox.left <= evt.clientX && evt.clientX <= thumbBox.right &&\n            thumbBox.top <= evt.clientY && evt.clientY <= thumbBox.bottom\n        ) {\n            this._bound.onmouseover(evt);\n        } else {\n            this._bound.onmouseout(evt);\n        }\n\n        evt.stopPropagation();\n        evt.preventDefault();\n    }\n};\n\nvar orientationHashes = {\n    vertical: {\n        coordinate:     'clientY',\n        axis:           'pageY',\n        size:           'height',\n        outside:        'right',\n        inside:         'left',\n        leading:        'top',\n        trailing:       'bottom',\n        marginLeading:  'marginTop',\n        marginTrailing: 'marginBottom',\n        thickness:      'width',\n        delta:          'deltaY'\n    },\n    horizontal: {\n        coordinate:     'clientX',\n        axis:           'pageX',\n        size:           'width',\n        outside:        'bottom',\n        inside:         'top',\n        leading:        'left',\n        trailing:       'right',\n        marginLeading:  'marginLeft',\n        marginTrailing: 'marginRight',\n        thickness:      'height',\n        delta:          'deltaX'\n    }\n};\n\nvar axis = {\n    top:    'vertical',\n    bottom: 'vertical',\n    height: 'vertical',\n    left:   'horizontal',\n    right:  'horizontal',\n    width:  'horizontal'\n};\n\nvar cssFinBars; // definition inserted by gulpfile between following comments\n/* inject:css */\ncssFinBars = 'div.finbar-horizontal,div.finbar-vertical{position:absolute;margin:3px}div.finbar-horizontal>.thumb,div.finbar-vertical>.thumb{position:absolute;background-color:#d3d3d3;-webkit-box-shadow:0 0 1px #000;-moz-box-shadow:0 0 1px #000;box-shadow:0 0 1px #000;border-radius:4px;margin:2px;opacity:.4;transition:opacity .5s}div.finbar-horizontal>.thumb.hover,div.finbar-vertical>.thumb.hover{opacity:1;transition:opacity .5s}div.finbar-vertical{top:0;bottom:0;right:0;width:11px}div.finbar-vertical>.thumb{top:0;right:0;width:7px}div.finbar-horizontal{left:0;right:0;bottom:0;height:11px}div.finbar-horizontal>.thumb{left:0;bottom:0;height:7px}';\n/* endinject */\n\nfunction error(msg) {\n    throw 'finbars: ' + msg;\n}\n\n// Interface\nmodule.exports = FinBar;\n","/* eslint-env browser */\n\n'use strict';\n\nvar rectangular = require('rectangular');\n\nvar gestures = require('./js/polymergestures.dev.js');\nvar GraphicsContext = require('./js/GraphicsContext.js');\n\nvar RESIZE_POLLING_INTERVAL = 200,\n    paintables = [],\n    resizables = [],\n    paintLoopRunning = true,\n    resizeLoopRunning = true,\n    charMap = makeCharMap();\n\nfunction Canvas(div, component, options) {\n    var self = this;\n\n    this.div = div;\n    this.component = component;\n\n    options = options || {};\n    this.doubleClickDelay = options.doubleClickDelay || 325;\n\n    this.dragEndtime = Date.now();\n\n    this.canvas = document.createElement('canvas');\n    this.div.appendChild(this.canvas);\n\n    this.canvas.style.outline = 'none';\n\n    // this.focuser = document.createElement('button');\n    // this.focuser.style.position = 'absolute';\n    // this.focuser.style.top = '0px';\n    // this.focuser.style.left = '0px';\n    // this.focuser.style.zIndex = '-1';\n    // this.focuser.style.outline = 'none';\n    // this.div.appendChild(this.focuser);\n\n    this.canvasCTX = this.canvas.getContext('2d');\n    this.gc = new GraphicsContext(this.canvasCTX);\n\n    this.buffer = document.createElement('canvas');\n    this.bufferCTX = this.buffer.getContext('2d');\n    this.bufferGC = new GraphicsContext(this.bufferCTX);\n\n    this.mouseLocation = new rectangular.Point(-1, -1);\n    this.dragstart = new rectangular.Point(-1, -1);\n    //this.origin = new rectangular.Point(0, 0);\n    this.bounds = new rectangular.Rectangle(0, 0, 0, 0);\n    this.hasMouse = false;\n\n    document.addEventListener('mousemove', function(e) {\n        if (self.hasMouse || self.isDragging()) {\n            self.finmousemove(e);\n        }\n    });\n    document.addEventListener('mouseup', function(e) {\n        self.finmouseup(e);\n    });\n    document.addEventListener('wheel', function(e) {\n        self.finwheelmoved(e);\n    });\n    document.addEventListener('keydown', function(e) {\n        self.finkeydown(e);\n    });\n    document.addEventListener('keyup', function(e) {\n        self.finkeyup(e);\n    });\n\n    this.canvas.onmouseover = function() {\n        self.hasMouse = true;\n    };\n    this.canvas.addEventListener('focus', function(e) {\n        self.finfocusgained(e);\n    });\n    this.canvas.addEventListener('blur', function(e) {\n        self.finfocuslost(e);\n    });\n    this.canvas.addEventListener('mousedown', function(e) {\n        self.finmousedown(e);\n    });\n    this.canvas.addEventListener('mouseout', function(e) {\n        self.hasMouse = false;\n        self.finmouseout(e);\n    });\n    this.canvas.addEventListener('click', function(e) {\n        self.finclick(e);\n    });\n    this.canvas.addEventListener('contextmenu', function(e) {\n        self.fincontextmenu(e);\n        e.preventDefault();\n        return false;\n    });\n\n    gestures.addEventListener(this.canvas, 'tap', function(e) {\n        self.fintap(e);\n    });\n    gestures.addEventListener(this.canvas, 'holdpulse', function(e) {\n        self.finholdpulse(e);\n    });\n    gestures.addEventListener(this.canvas, 'flick', function(e) {\n        self.finflick(e);\n    });\n    gestures.addEventListener(this.canvas, 'release', function(e) {\n        self.finrelease(e);\n    });\n    gestures.addEventListener(this.canvas, 'trackstart', function(e) {\n        self.fintrackstart(e);\n    });\n    gestures.addEventListener(this.canvas, 'track', function(e) {\n        self.fintrack(e);\n    });\n    gestures.addEventListener(this.canvas, 'trackend', function(e) {\n        self.fintrackend(e);\n    });\n\n    this.canvas.setAttribute('tabindex', 0);\n    this.canvas.contentEditable = true;\n\n    this.resize();\n\n    this.beginResizing();\n    this.beginPainting();\n}\n\nCanvas.prototype = {\n    constructor: Canvas.prototype.constructor,\n    div: null,\n    component: null,\n    gestures: gestures, // TODO: why do we need this? (was previously at bottom of file)\n    canvas: null,\n    canvasCTX: null,\n    focuser: null,\n    buffer: null,\n    ctx: null,\n    mouseLocation: null,\n    holdPulseCount: -1,\n    dragstart: null,\n    origin: null,\n    bounds: null,\n    dirty: false,\n    size: null,\n    mousedown: false,\n    dragging: false,\n    repeatKeyCount: 0,\n    repeatKey: null,\n    repeatKeyStartTime: 0,\n    currentKeys: [],\n    hasMouse: false,\n    lastDoubleClickTime: 0,\n    dragEndTime: 0,\n    lastRepaintTime: 0,\n\n    addEventListener: function(name, callback) {\n        this.canvas.addEventListener(name, callback);\n    },\n\n    stopPaintLoop: function() {\n        paintLoopRunning = false;\n    },\n\n    restartPaintLoop: function() {\n        if (paintLoopRunning) {\n            return; // already running\n        }\n        paintLoopRunning = true;\n        requestAnimationFrame(paintLoopFunction);\n    },\n\n    stopResizeLoop: function() {\n        resizeLoopRunning = false;\n    },\n\n    restartResizeLoop: function() {\n        if (resizeLoopRunning) {\n            return; // already running\n        }\n        resizeLoopRunning = true;\n        setInterval(resizablesLoopFunction, 200);\n    },\n\n    detached: function() {\n        this.stopPainting();\n        this.stopResizing();\n    },\n\n    useHiDPI: function() {\n        return this.component.resolveProperty('useHiDPI');\n    },\n\n    useBitBlit: function() {\n        return this.component.resolveProperty('useBitBlit');\n    },\n\n    getFPS: function() {\n        var fps = this.component.resolveProperty('repaintIntervalRate');\n        return fps ? parseInt(fps) : 0;\n    },\n\n    tickPaint: function(now) {\n        var fps = this.getFPS();\n        if (fps === 0) {\n            return;\n        }\n        var interval = 1000 / fps;\n\n        var elapsed = now - this.lastRepaintTime;\n        if (elapsed > interval && this.dirty) {\n            this.lastRepaintTime = now - (elapsed % interval);\n            this.paintNow();\n        }\n    },\n\n    beginPainting: function() {\n        var self = this;\n        this.dirty = true;\n        this.tickPainter = function(now) {\n            self.tickPaint(now);\n        };\n        paintables.push(this);\n    },\n\n    stopPainting: function() {\n        paintables.splice(paintables.indexOf(this), 1);\n    },\n\n    beginResizing: function() {\n        var self = this;\n        this.tickResizer = function() {\n            self.checksize();\n        };\n        resizables.push(this);\n    },\n\n    stopResizing: function() {\n        resizables.splice(resizables.indexOf(this), 1);\n    },\n\n    start: function() {\n        this.beginPainting();\n        this.beginResizing();\n    },\n\n    stop: function() {\n        this.stopPainting();\n        this.stopResizing();\n    },\n\n    checksize: function() {\n        //this is expensive lets do it at some modulo\n        var sizeNow = this.div.getBoundingClientRect();\n        if (sizeNow.width !== this.size.width || sizeNow.height !== this.size.height) {\n            this.sizeChangedNotification();\n        }\n    },\n\n    sizeChangedNotification: function() {\n        this.resize();\n    },\n\n    resize: function() {\n        var box = this.size = this.div.getBoundingClientRect();\n\n        this.canvas.width = this.buffer.width = box.width;\n        this.canvas.height = this.buffer.height = box.height;\n\n        //fix ala sir spinka, see\n        //http://www.html5rocks.com/en/tutorials/canvas/hidpi/\n        //just add 'hdpi' as an attribute to the fin-canvas tag\n        var ratio = 1;\n        var useBitBlit = this.useBitBlit();\n        var isHIDPI = window.devicePixelRatio && this.useHiDPI();\n        if (isHIDPI) {\n            var devicePixelRatio = window.devicePixelRatio || 1;\n            var backingStoreRatio = this.canvasCTX.webkitBackingStorePixelRatio ||\n                this.canvasCTX.mozBackingStorePixelRatio ||\n                this.canvasCTX.msBackingStorePixelRatio ||\n                this.canvasCTX.oBackingStorePixelRatio ||\n                this.canvasCTX.backingStorePixelRatio || 1;\n\n            ratio = devicePixelRatio / backingStoreRatio;\n            //this.canvasCTX.scale(ratio, ratio);\n        }\n        var width = this.canvas.getAttribute('width');\n        var height = this.canvas.getAttribute('height');\n        this.canvas.width = width * ratio;\n        this.canvas.height = height * ratio;\n        this.buffer.width = width * ratio;\n        this.buffer.height = height * ratio;\n\n        this.canvas.style.width = width + 'px';\n        this.canvas.style.height = height + 'px';\n        this.buffer.style.width = width + 'px';\n        this.buffer.style.height = height + 'px';\n\n        this.bufferCTX.scale(ratio, ratio);\n        if (isHIDPI && !useBitBlit) {\n            this.canvasCTX.scale(ratio, ratio);\n        }\n\n        //this.origin = new rectangular.Point(Math.round(this.size.left), Math.round(this.size.top));\n        this.bounds = new rectangular.Rectangle(0, 0, box.width, box.height);\n        //setTimeout(function() {\n        var comp = this.component;\n        if (comp) {\n            comp.setBounds(this.bounds);\n        }\n        this.resizeNotification();\n        this.paintNow();\n        //});\n    },\n\n    resizeNotification: function() {\n        //to be overridden\n    },\n\n    getBounds: function() {\n        return this.bounds;\n    },\n\n    paintNow: function() {\n        var self = this;\n        this.safePaintImmediately(function(gc) {\n            gc.clearRect(0, 0, self.canvas.width, self.canvas.height);\n\n            var comp = self.component;\n            if (comp) {\n                comp._paint(gc);\n            }\n\n            self.dirty = false;\n        });\n    },\n\n    safePaintImmediately: function(paintFunction) {\n        var useBitBlit = this.useBitBlit(),\n            gc = useBitBlit ? this.bufferGC : this.gc;\n        try {\n            gc.save();\n            paintFunction(gc);\n        } catch (e) {\n            console.error(e);\n        } finally {\n            gc.restore();\n        }\n        if (useBitBlit) {\n            this.flushBuffer();\n        }\n    },\n\n    flushBuffer: function() {\n        if (this.buffer.width > 0 && this.buffer.height > 0) {\n            this.canvasCTX.drawImage(this.buffer, 0, 0);\n        }\n    },\n\n    dispatchNewEvent: function(event, name, detail) {\n        detail = {\n            detail: detail || {}\n        };\n        detail.detail.primitiveEvent = event;\n        return this.canvas.dispatchEvent(new CustomEvent(name, detail));\n    },\n\n    dispatchNewMouseKeysEvent: function(event, name, detail) {\n        detail = detail || {};\n        detail.mouse = this.mouseLocation;\n        detail.keys = this.currentKeys;\n        return this.dispatchNewEvent(event, name, detail);\n    },\n\n    finmousemove: function(e) {\n        if (!this.isDragging() && this.mousedown) {\n            this.beDragging();\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dragstart', {\n                isRightClick: this.isRightClick(e)\n            });\n            this.dragstart = new rectangular.Point(this.mouseLocation.x, this.mouseLocation.y);\n        }\n        this.mouseLocation = this.getLocal(e);\n        //console.log(this.mouseLocation);\n        if (this.isDragging()) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-drag', {\n                dragstart: this.dragstart,\n                isRightClick: this.isRightClick(e)\n            });\n        }\n        if (this.bounds.contains(this.mouseLocation)) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mousemove');\n        }\n    },\n\n    finmousedown: function(e) {\n        this.mouseLocation = this.mouseDownLocation = this.getLocal(e);\n        this.mousedown = true;\n\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mousedown', {\n            isRightClick: this.isRightClick(e)\n        });\n        this.takeFocus();\n    },\n\n    finmouseup: function(e) {\n        if (this.isDragging()) {\n            this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dragend', {\n                dragstart: this.dragstart,\n                isRightClick: this.isRightClick(e)\n            });\n            this.beNotDragging();\n            this.dragEndtime = Date.now();\n        }\n        this.mousedown = false;\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mouseup', {\n            isRightClick: this.isRightClick(e)\n        });\n        //this.mouseLocation = new rectangular.Point(-1, -1);\n    },\n\n    finmouseout: function(e) {\n        if (!this.mousedown) {\n            this.mouseLocation = new rectangular.Point(-1, -1);\n        }\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-mouseout');\n    },\n\n    finwheelmoved: function(e) {\n        if (this.isDragging() || !this.hasFocus()) {\n            return;\n        }\n        e.preventDefault();\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-wheelmoved', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    finclick: function(e) {\n        if (this.doubleClickTimer && Date.now() - this.lastClickTime < this.doubleClickDelay) {\n            //this is a double click...\n            clearTimeout(this.doubleClickTimer); // prevent click event\n            this.doubleClickTimer = undefined;\n            this.findblclick(e);\n        } else {\n            this.lastClickTime = Date.now();\n\n            this.doubleClickTimer = setTimeout(function() {\n                this.doubleClickTimer = undefined;\n                this.mouseLocation = this.getLocal(e);\n                this.dispatchNewMouseKeysEvent(e, 'fin-canvas-click', {\n                    isRightClick: this.isRightClick(e)\n                });\n            }.bind(this), this.doubleClickDelay);\n        }\n    },\n\n    finrelease: function(e) {\n        this.holdPulseCount = 0;\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-release');\n    },\n\n    finflick: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-flick', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    fintrackstart: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-trackstart');\n    },\n\n    fintrack: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-track');\n    },\n\n    fintrackend: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-trackend');\n    },\n\n    finhold: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-hold', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    finholdpulse: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-holdpulse', {\n            count: this.holdPulseCount++\n        });\n    },\n\n    fintap: function(e) {\n        //this nonsense is to hold a tap if it's really a double click\n        var self = this;\n        var now = Date.now();\n        var dif = now - this.lastDoubleClickTime;\n        if (dif < 300) {\n            return;\n        }\n        //dragend is also causing a tap\n        //lets fix this here\n        if (now - this.dragEndtime < 100) {\n            return;\n        }\n        setTimeout(function() {\n            self._fintap(e);\n        }, 180);\n    },\n\n    _fintap: function(e) {\n        //this nonsense is to hold a tap if it's really a double click\n        var now = Date.now();\n        var dif = now - this.lastDoubleClickTime;\n        if (dif < 300) {\n            return;\n        }\n\n        if (this.mouseDownLocation) { // maybe no mousedown on a phone?\n            this.mouseLocation = this.mouseDownLocation; // mouse may have moved since mousedown\n            this.mouseDownLocation = undefined; // consume it (maybe not needed; once a mousedown always a mousedown)\n        }\n\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-tap', {\n            isRightClick: this.isRightClick(e)\n        });\n    },\n\n    findblclick: function(e) {\n        this.mouseLocation = this.getLocal(e);\n        this.lastDoubleClickTime = Date.now();\n        this.dispatchNewMouseKeysEvent(e, 'fin-canvas-dblclick', {\n            isRightClick: this.isRightClick(e)\n        });\n        //console.log('dblclick', this.currentKeys);\n    },\n\n    getCharMap: function() { //TODO: This is static. Make it a property of the constructor.\n        return charMap;\n    },\n\n    finkeydown: function(e) {\n        if (!this.hasFocus()) {\n            return;\n        }\n\n        //e.preventDefault();\n        var keyChar = e.shiftKey ? charMap[e.keyCode][1] : charMap[e.keyCode][0];\n        if (e.repeat) {\n            if (this.repeatKey === keyChar) {\n                this.repeatKeyCount++;\n            } else {\n                this.repeatKey = keyChar;\n                this.repeatKeyStartTime = Date.now();\n            }\n        } else {\n            this.repeatKey = null;\n            this.repeatKeyCount = 0;\n            this.repeatKeyStartTime = 0;\n        }\n        if (this.currentKeys.indexOf(keyChar) === -1) {\n            this.currentKeys.push(keyChar);\n        }\n        //console.log(keyChar, e.keyCode);\n        this.dispatchNewEvent(e, 'fin-canvas-keydown', {\n            alt: e.altKey,\n            ctrl: e.ctrlKey,\n            char: keyChar,\n            code: e.charCode,\n            key: e.keyCode,\n            meta: e.metaKey,\n            repeatCount: this.repeatKeyCount,\n            repeatStartTime: this.repeatKeyStartTime,\n            shift: e.shiftKey,\n            identifier: e.keyIdentifier,\n            currentKeys: this.currentKeys.slice(0)\n        });\n    },\n\n    finkeyup: function(e) {\n        var keyChar = e.shiftKey ? charMap[e.keyCode][1] : charMap[e.keyCode][0];\n        this.currentKeys.splice(this.currentKeys.indexOf(keyChar), 1);\n        if (!this.hasFocus()) {\n            return;\n        }\n        this.repeatKeyCount = 0;\n        this.repeatKey = null;\n        this.repeatKeyStartTime = 0;\n        this.dispatchNewEvent(e, 'fin-canvas-keyup', {\n            alt: e.altKey,\n            ctrl: e.ctrlKey,\n            char: keyChar,\n            code: e.charCode,\n            key: e.keyCode,\n            meta: e.metaKey,\n            repeat: e.repeat,\n            shift: e.shiftKey,\n            identifier: e.keyIdentifier,\n            currentKeys: this.currentKeys.slice(0)\n        });\n    },\n\n    finfocusgained: function(e) {\n        this.dispatchNewEvent(e, 'fin-canvas-focus-gained');\n    },\n\n    finfocuslost: function(e) {\n        this.dispatchNewEvent(e, 'fin-canvas-focus-lost');\n    },\n\n    fincontextmenu: function(e) {\n        if (e.ctrlKey && this.currentKeys.indexOf('CTRL') === -1) {\n            this.currentKeys.push('CTRL');\n        }\n        if (this.doubleRightClickTimer && Date.now() - this.lastClickTime < this.doubleClickDelay) {\n            //this is a double click...\n            clearTimeout(this.doubleRightClickTimer); // prevent context menu event\n            this.doubleRightClickTimer = undefined;\n            this.findblclick(e);\n        } else {\n            this.lastClickTime = Date.now();\n\n            this.doubleRightClickTimer = setTimeout(function() {\n                this.doubleRightClickTimer = undefined;\n                this.dispatchNewMouseKeysEvent(e, 'fin-canvas-context-menu', {\n                    isRightClick: this.isRightClick(e)\n                });\n            }.bind(this), this.doubleClickDelay);\n        }\n    },\n\n    repaint: function() {\n        var fps = this.getFPS();\n        this.dirty = true;\n        if (!paintLoopRunning || fps === 0) {\n            this.paintNow();\n        }\n    },\n\n    getMouseLocation: function() {\n        return this.mouseLocation;\n    },\n\n    getOrigin: function() {\n        var rect = this.canvas.getBoundingClientRect();\n        var p = new rectangular.Point(rect.left, rect.top);\n        return p;\n    },\n\n    getLocal: function(e) {\n        var rect = this.canvas.getBoundingClientRect();\n        var p = new rectangular.Point(e.clientX - rect.left, e.clientY - rect.top);\n        return p;\n    },\n\n    hasFocus: function() {\n        return document.activeElement === this.canvas;\n    },\n\n    takeFocus: function() {\n        var self = this;\n        if (!this.hasFocus()) {\n            setTimeout(function() {\n                self.canvas.focus();\n            }, 10);\n        }\n    },\n\n    beDragging: function() {\n        this.dragging = true;\n        this.disableDocumentElementSelection();\n    },\n\n    beNotDragging: function() {\n        this.dragging = false;\n        this.enableDocumentElementSelection();\n    },\n\n    isDragging: function() {\n        return this.dragging;\n    },\n\n    disableDocumentElementSelection: function() {\n        var style = document.body.style;\n        style.cssText = style.cssText + '-webkit-user-select: none';\n    },\n\n    enableDocumentElementSelection: function() {\n        var style = document.body.style;\n        style.cssText = style.cssText.replace('-webkit-user-select: none', '');\n    },\n\n    setFocusable: function(truthy) {\n        this.focuser.style.display = truthy ? '' : 'none';\n    },\n\n    isRightClick: function(e) {\n        var isRightMB;\n        e = e || window.event;\n\n        if ('which' in e) { // Gecko (Firefox), WebKit (Safari/Chrome) & Opera\n            isRightMB = e.which === 3;\n        } else if ('button' in e) { // IE, Opera\n            isRightMB = e.button === 2;\n        }\n        return isRightMB;\n    },\n\n    dispatchEvent: function(e) {\n        return this.canvas.dispatchEvent(e);\n    }\n};\n\nfunction paintLoopFunction(now) {\n    if (!paintLoopRunning) {\n        return;\n    }\n    for (var i = 0; i < paintables.length; i++) {\n        try {\n            paintables[i].tickPainter(now);\n        } catch (e) {\n            console.error(e);\n        }\n    }\n    requestAnimationFrame(paintLoopFunction);\n}\nrequestAnimationFrame(paintLoopFunction);\n\nfunction resizablesLoopFunction(now) {\n    if (!resizeLoopRunning) {\n        return;\n    }\n    for (var i = 0; i < resizables.length; i++) {\n        try {\n            resizables[i].tickResizer(now);\n        } catch (e) {\n            console.error(e);\n        }\n    }\n}\nsetInterval(resizablesLoopFunction, RESIZE_POLLING_INTERVAL);\n\nfunction makeCharMap() {\n    var map = [];\n\n    var empty = ['', ''];\n\n    for (var i = 0; i < 256; i++) {\n        map[i] = empty;\n    }\n\n    map[27] = ['ESC', 'ESCSHIFT'];\n    map[192] = ['`', '~'];\n    map[49] = ['1', '!'];\n    map[50] = ['2', '@'];\n    map[51] = ['3', '#'];\n    map[52] = ['4', '$'];\n    map[53] = ['5', '%'];\n    map[54] = ['6', '^'];\n    map[55] = ['7', '&'];\n    map[56] = ['8', '*'];\n    map[57] = ['9', '('];\n    map[48] = ['0', ')'];\n    map[189] = ['-', '_'];\n    map[187] = ['=', '+'];\n    map[8] = ['BACKSPACE', 'BACKSPACESHIFT'];\n    map[46] = ['DELETE', 'DELETESHIFT'];\n    map[9] = ['TAB', 'TABSHIFT'];\n    map[81] = ['q', 'Q'];\n    map[87] = ['w', 'W'];\n    map[69] = ['e', 'E'];\n    map[82] = ['r', 'R'];\n    map[84] = ['t', 'T'];\n    map[89] = ['y', 'Y'];\n    map[85] = ['u', 'U'];\n    map[73] = ['i', 'I'];\n    map[79] = ['o', 'O'];\n    map[80] = ['p', 'P'];\n    map[219] = ['[', '{'];\n    map[221] = [']', '}'];\n    map[220] = ['\\\\', '|'];\n    map[220] = ['CAPSLOCK', 'CAPSLOCKSHIFT'];\n    map[65] = ['a', 'A'];\n    map[83] = ['s', 'S'];\n    map[68] = ['d', 'D'];\n    map[70] = ['f', 'F'];\n    map[71] = ['g', 'G'];\n    map[72] = ['h', 'H'];\n    map[74] = ['j', 'J'];\n    map[75] = ['k', 'K'];\n    map[76] = ['l', 'L'];\n    map[186] = [';', ':'];\n    map[222] = ['\\'', '|'];\n    map[13] = ['RETURN', 'RETURNSHIFT'];\n    map[16] = ['SHIFT', 'SHIFT'];\n    map[90] = ['z', 'Z'];\n    map[88] = ['x', 'X'];\n    map[67] = ['c', 'C'];\n    map[86] = ['v', 'V'];\n    map[66] = ['b', 'B'];\n    map[78] = ['n', 'N'];\n    map[77] = ['m', 'M'];\n    map[188] = [',', '<'];\n    map[190] = ['.', '>'];\n    map[191] = ['/', '?'];\n    map[16] = ['SHIFT', 'SHIFT'];\n    map[17] = ['CTRL', 'CTRLSHIFT'];\n    map[18] = ['ALT', 'ALTSHIFT'];\n    map[91] = ['COMMANDLEFT', 'COMMANDLEFTSHIFT'];\n    map[32] = ['SPACE', 'SPACESHIFT'];\n    map[93] = ['COMMANDRIGHT', 'COMMANDRIGHTSHIFT'];\n    map[18] = ['ALT', 'ALTSHIFT'];\n    map[38] = ['UP', 'UPSHIFT'];\n    map[37] = ['LEFT', 'LEFTSHIFT'];\n    map[40] = ['DOWN', 'DOWNSHIFT'];\n    map[39] = ['RIGHT', 'RIGHTSHIFT'];\n\n    map[33] = ['PAGEUP', 'PAGEUPSHIFT'];\n    map[34] = ['PAGEDOWN', 'PAGEDOWNSHIFT'];\n    map[35] = ['PAGERIGHT', 'PAGERIGHTSHIFT']; // END\n    map[36] = ['PAGELEFT', 'PAGELEFTSHIFT']; // HOME\n\n    map[112] = ['F1', 'F1SHIFT'];\n    map[113] = ['F2', 'F2SHIFT'];\n    map[114] = ['F3', 'F3SHIFT'];\n    map[115] = ['F4', 'F4SHIFT'];\n    map[116] = ['F5', 'F5SHIFT'];\n    map[117] = ['F6', 'F6SHIFT'];\n    map[118] = ['F7', 'F7SHIFT'];\n    map[119] = ['F8', 'F8SHIFT'];\n    map[120] = ['F9', 'F9SHIFT'];\n    map[121] = ['F10', 'F10SHIFT'];\n    map[122] = ['F11', 'F1S1HIFT'];\n    map[123] = ['F12', 'F121HIFT'];\n\n    return map;\n}\n\nmodule.exports = Canvas;\n","'use strict';\n\nvar consoleLogger = require('./gc-console-logger');\n\n/**\n * @constructor\n * @param gc - The 2-D graphics context from your canvas\n * @param {boolean|apiLogger} [logger=true]\n * * `true` uses `gc-console-logger` function bound to 'gc.' as prefix\n * * string uses `gc-console-logger` function bound to string\n * * function used as is\n */\nfunction GraphicsContext(gc, logger) {\n    this.gc = gc;\n\n    var self = this;\n    var reWEBKIT = /^webkit/;\n\n    switch (typeof logger) {\n\n        case 'string':\n            logger =  consoleLogger.bind(undefined, logger + '.');\n            break;\n\n        case 'boolean':\n            if (logger === true) {\n                logger = consoleLogger.bind(undefined, 'gc.');\n            }\n            break;\n\n        case 'function':\n            if (logger.length !== 3) {\n                throw 'GraphicsContext: User-supplied API logger function does not accept three parameters.';\n            }\n            break;\n\n        default:\n            logger = false;\n    }\n\n    // Stub out all the prototype members of the canvas 2D graphics context:\n    Object.keys(Object.getPrototypeOf(gc)).forEach(MakeStub);\n\n    // Some older browsers (e.g., Chrome 40) did not have all members of canvas\n    // 2D graphics context in the prototype so we make this additional call:\n    Object.keys(gc).forEach(MakeStub);\n\n    function MakeStub(key) {\n        if (key in GraphicsContext.prototype || reWEBKIT.test(key)) {\n            return;\n        }\n        if (typeof gc[key] === 'function') {\n            self[key] = !logger ? gc[key].bind(gc) : function() {\n                return logger(key, arguments, gc[key].apply(gc, arguments));\n            };\n        } else {\n            Object.defineProperty(self, key, {\n                get: function() {\n                    var result = gc[key];\n                    return logger ? logger(key, 'getter', result) : result;\n                },\n                set: function(value) {\n                    gc[key] = logger ? logger(key, 'setter', value) : value;\n                }\n            });\n        }\n    }\n}\n\nmodule.exports = GraphicsContext;\n","'use strict';\n\nvar YIELDS = '\\u27F9'; // LONG RIGHTWARDS DOUBLE ARROW\n\nfunction consoleLogger(prefix, name, args, value) {\n    var result = value;\n\n    if (typeof value === 'string') {\n        result = '\"' + result + '\"';\n    }\n\n    name = prefix + name;\n\n    switch (args) {\n        case 'getter':\n            console.log(name, '=', result);\n            break;\n\n        case 'setter':\n            console.log(name, YIELDS, result);\n            break;\n\n        default: // method call\n            name += '(' + Array.prototype.slice.call(args).join(', ') + ')';\n            if (result === undefined) {\n                console.log(name);\n            } else {\n                console.log(name, YIELDS, result);\n            }\n    }\n\n    return value;\n}\n\nmodule.exports = consoleLogger;\n","/* eslint-disable */\n\n/**\n * @license\n * Copyright (c) 2014 The Polymer Project Authors. All rights reserved.\n * This code may only be used under the BSD style license found at http://polymer.github.io/LICENSE.txt\n * The complete set of authors may be found at http://polymer.github.io/AUTHORS.txt\n * The complete set of contributors may be found at http://polymer.github.io/CONTRIBUTORS.txt\n * Code distributed by Google as part of the polymer project is also\n * subject to an additional IP rights grant found at http://polymer.github.io/PATENTS.txt\n */\n//module.exports = {};\n\n(function(scope) {\n    var hasFullPath = false;\n\n    // test for full event path support\n    var pathTest = document.createElement('meta');\n    if (pathTest.createShadowRoot) {\n        var sr = pathTest.createShadowRoot();\n        var s = document.createElement('span');\n        sr.appendChild(s);\n        pathTest.addEventListener('testpath', function(ev) {\n            if (ev.path) {\n                // if the span is in the event path, then path[0] is the real source for all events\n                hasFullPath = ev.path[0] === s;\n            }\n            ev.stopPropagation();\n        });\n        var ev = new CustomEvent('testpath', {\n            bubbles: true\n        });\n        // must add node to DOM to trigger event listener\n        document.head.appendChild(pathTest);\n        s.dispatchEvent(ev);\n        pathTest.parentNode.removeChild(pathTest);\n        sr = s = null;\n    }\n    pathTest = null;\n\n    var target = {\n        shadow: function(inEl) {\n            if (inEl) {\n                return inEl.shadowRoot || inEl.webkitShadowRoot;\n            }\n        },\n        canTarget: function(shadow) {\n            return shadow && Boolean(shadow.elementFromPoint);\n        },\n        targetingShadow: function(inEl) {\n            var s = this.shadow(inEl);\n            if (this.canTarget(s)) {\n                return s;\n            }\n        },\n        olderShadow: function(shadow) {\n            var os = shadow.olderShadowRoot;\n            if (!os) {\n                var se = shadow.querySelector('shadow');\n                if (se) {\n                    os = se.olderShadowRoot;\n                }\n            }\n            return os;\n        },\n        allShadows: function(element) {\n            var shadows = [],\n                s = this.shadow(element);\n            while (s) {\n                shadows.push(s);\n                s = this.olderShadow(s);\n            }\n            return shadows;\n        },\n        searchRoot: function(inRoot, x, y) {\n            var t, st, sr, os;\n            if (inRoot) {\n                t = inRoot.elementFromPoint(x, y);\n                if (t) {\n                    // found element, check if it has a ShadowRoot\n                    sr = this.targetingShadow(t);\n                } else if (inRoot !== document) {\n                    // check for sibling roots\n                    sr = this.olderShadow(inRoot);\n                }\n                // search other roots, fall back to light dom element\n                return this.searchRoot(sr, x, y) || t;\n            }\n        },\n        owner: function(element) {\n            if (!element) {\n                return document;\n            }\n            var s = element;\n            // walk up until you hit the shadow root or document\n            while (s.parentNode) {\n                s = s.parentNode;\n            }\n            // the owner element is expected to be a Document or ShadowRoot\n            if (s.nodeType != Node.DOCUMENT_NODE && s.nodeType != Node.DOCUMENT_FRAGMENT_NODE) {\n                s = document;\n            }\n            return s;\n        },\n        findTarget: function(inEvent) {\n            if (hasFullPath && inEvent.path && inEvent.path.length) {\n                return inEvent.path[0];\n            }\n            var x = inEvent.clientX,\n                y = inEvent.clientY;\n            // if the listener is in the shadow root, it is much faster to start there\n            var s = this.owner(inEvent.target);\n            // if x, y is not in this root, fall back to document search\n            if (!s.elementFromPoint(x, y)) {\n                s = document;\n            }\n            return this.searchRoot(s, x, y);\n        },\n        findTouchAction: function(inEvent) {\n            var n;\n            if (hasFullPath && inEvent.path && inEvent.path.length) {\n                var path = inEvent.path;\n                for (var i = 0; i < path.length; i++) {\n                    n = path[i];\n                    if (n.nodeType === Node.ELEMENT_NODE && n.hasAttribute('touch-action')) {\n                        return n.getAttribute('touch-action');\n                    }\n                }\n            } else {\n                n = inEvent.target;\n                while (n) {\n                    if (n.nodeType === Node.ELEMENT_NODE && n.hasAttribute('touch-action')) {\n                        return n.getAttribute('touch-action');\n                    }\n                    n = n.parentNode || n.host;\n                }\n            }\n            // auto is default\n            return \"auto\";\n        },\n        LCA: function(a, b) {\n            if (a === b) {\n                return a;\n            }\n            if (a && !b) {\n                return a;\n            }\n            if (b && !a) {\n                return b;\n            }\n            if (!b && !a) {\n                return document;\n            }\n            // fast case, a is a direct descendant of b or vice versa\n            if (a.contains && a.contains(b)) {\n                return a;\n            }\n            if (b.contains && b.contains(a)) {\n                return b;\n            }\n            var adepth = this.depth(a);\n            var bdepth = this.depth(b);\n            var d = adepth - bdepth;\n            if (d >= 0) {\n                a = this.walk(a, d);\n            } else {\n                b = this.walk(b, -d);\n            }\n            while (a && b && a !== b) {\n                a = a.parentNode || a.host;\n                b = b.parentNode || b.host;\n            }\n            return a;\n        },\n        walk: function(n, u) {\n            for (var i = 0; n && (i < u); i++) {\n                n = n.parentNode || n.host;\n            }\n            return n;\n        },\n        depth: function(n) {\n            var d = 0;\n            while (n) {\n                d++;\n                n = n.parentNode || n.host;\n            }\n            return d;\n        },\n        deepContains: function(a, b) {\n            var common = this.LCA(a, b);\n            // if a is the common ancestor, it must \"deeply\" contain b\n            return common === a;\n        },\n        insideNode: function(node, x, y) {\n            var rect = node.getBoundingClientRect();\n            return (rect.left <= x) && (x <= rect.right) && (rect.top <= y) && (y <= rect.bottom);\n        },\n        path: function(event) {\n            var p;\n            if (hasFullPath && event.path && event.path.length) {\n                p = event.path;\n            } else {\n                p = [];\n                var n = this.findTarget(event);\n                while (n) {\n                    p.push(n);\n                    n = n.parentNode || n.host;\n                }\n            }\n            return p;\n        }\n    };\n    scope.targetFinding = target;\n    /**\n     * Given an event, finds the \"deepest\" node that could have been the original target before ShadowDOM retargetting\n     *\n     * @param {Event} Event An event object with clientX and clientY properties\n     * @return {Element} The probable event origninator\n     */\n    scope.findTarget = target.findTarget.bind(target);\n    /**\n     * Determines if the \"container\" node deeply contains the \"containee\" node, including situations where the \"containee\" is contained by one or more ShadowDOM\n     * roots.\n     *\n     * @param {Node} container\n     * @param {Node} containee\n     * @return {Boolean}\n     */\n    scope.deepContains = target.deepContains.bind(target);\n\n    /**\n     * Determines if the x/y position is inside the given node.\n     *\n     * Example:\n     *\n     *     function upHandler(event) {\n     *       var innode = PolymerGestures.insideNode(event.target, event.clientX, event.clientY);\n     *       if (innode) {\n     *         // wait for tap?\n     *       } else {\n     *         // tap will never happen\n     *       }\n     *     }\n     *\n     * @param {Node} node\n     * @param {Number} x Screen X position\n     * @param {Number} y screen Y position\n     * @return {Boolean}\n     */\n    scope.insideNode = target.insideNode;\n\n})(exports);\n\n(function() {\n    function shadowSelector(v) {\n        return 'html /deep/ ' + selector(v);\n    }\n\n    function selector(v) {\n        return '[touch-action=\"' + v + '\"]';\n    }\n\n    function rule(v) {\n        return '{ -ms-touch-action: ' + v + '; touch-action: ' + v + ';}';\n    }\n    var attrib2css = [\n        'none',\n        'auto',\n        'pan-x',\n        'pan-y', {\n            rule: 'pan-x pan-y',\n            selectors: [\n                'pan-x pan-y',\n                'pan-y pan-x'\n            ]\n        },\n        'manipulation'\n    ];\n    var styles = '';\n    // only install stylesheet if the browser has touch action support\n    var hasTouchAction = typeof document.head.style.touchAction === 'string';\n    // only add shadow selectors if shadowdom is supported\n    var hasShadowRoot = !window.ShadowDOMPolyfill && document.head.createShadowRoot;\n\n    if (hasTouchAction) {\n        attrib2css.forEach(function(r) {\n            if (String(r) === r) {\n                styles += selector(r) + rule(r) + '\\n';\n                if (hasShadowRoot) {\n                    styles += shadowSelector(r) + rule(r) + '\\n';\n                }\n            } else {\n                styles += r.selectors.map(selector) + rule(r.rule) + '\\n';\n                if (hasShadowRoot) {\n                    styles += r.selectors.map(shadowSelector) + rule(r.rule) + '\\n';\n                }\n            }\n        });\n\n        var el = document.createElement('style');\n        el.textContent = styles;\n        document.head.appendChild(el);\n    }\n})();\n\n/**\n * This is the constructor for new PointerEvents.\n *\n * New Pointer Events must be given a type, and an optional dictionary of\n * initialization properties.\n *\n * Due to certain platform requirements, events returned from the constructor\n * identify as MouseEvents.\n *\n * @constructor\n * @param {String} inType The type of the event to create.\n * @param {Object} [inDict] An optional dictionary of initial event properties.\n * @return {Event} A new PointerEvent of type `inType` and initialized with properties from `inDict`.\n */\n(function(scope) {\n\n    var MOUSE_PROPS = [\n        'bubbles',\n        'cancelable',\n        'view',\n        'detail',\n        'screenX',\n        'screenY',\n        'clientX',\n        'clientY',\n        'ctrlKey',\n        'altKey',\n        'shiftKey',\n        'metaKey',\n        'button',\n        'relatedTarget',\n        'pageX',\n        'pageY'\n    ];\n\n    var MOUSE_DEFAULTS = [\n        false,\n        false,\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        false,\n        false,\n        false,\n        false,\n        0,\n        null,\n        0,\n        0\n    ];\n\n    var NOP_FACTORY = function() {\n        return function() {};\n    };\n\n    var eventFactory = {\n        // TODO(dfreedm): this is overridden by tap recognizer, needs review\n        preventTap: NOP_FACTORY,\n        makeBaseEvent: function(inType, inDict) {\n            var e = document.createEvent('Event');\n            e.initEvent(inType, inDict.bubbles || false, inDict.cancelable || false);\n            e.preventTap = eventFactory.preventTap(e);\n            return e;\n        },\n        makeGestureEvent: function(inType, inDict) {\n            inDict = inDict || Object.create(null);\n\n            var e = this.makeBaseEvent(inType, inDict);\n            for (var i = 0, keys = Object.keys(inDict), k; i < keys.length; i++) {\n                k = keys[i];\n                if (k !== 'bubbles' && k !== 'cancelable') {\n                    e[k] = inDict[k];\n                }\n            }\n            return e;\n        },\n        makePointerEvent: function(inType, inDict) {\n            inDict = inDict || Object.create(null);\n\n            var e = this.makeBaseEvent(inType, inDict);\n            // define inherited MouseEvent properties\n            for (var i = 2, p; i < MOUSE_PROPS.length; i++) {\n                p = MOUSE_PROPS[i];\n                e[p] = inDict[p] || MOUSE_DEFAULTS[i];\n            }\n            e.buttons = inDict.buttons || 0;\n\n            // Spec requires that pointers without pressure specified use 0.5 for down\n            // state and 0 for up state.\n            var pressure = 0;\n            if (inDict.pressure) {\n                pressure = inDict.pressure;\n            } else {\n                pressure = e.buttons ? 0.5 : 0;\n            }\n\n            // add x/y properties aliased to clientX/Y\n            e.x = e.clientX;\n            e.y = e.clientY;\n\n            // define the properties of the PointerEvent interface\n            e.pointerId = inDict.pointerId || 0;\n            e.width = inDict.width || 0;\n            e.height = inDict.height || 0;\n            e.pressure = pressure;\n            e.tiltX = inDict.tiltX || 0;\n            e.tiltY = inDict.tiltY || 0;\n            e.pointerType = inDict.pointerType || '';\n            e.hwTimestamp = inDict.hwTimestamp || 0;\n            e.isPrimary = inDict.isPrimary || false;\n            e._source = inDict._source || '';\n            return e;\n        }\n    };\n\n    scope.eventFactory = eventFactory;\n})(exports);\n\n/**\n * This module implements an map of pointer states\n */\n(function(scope) {\n    var USE_MAP = window.Map && window.Map.prototype.forEach;\n    var POINTERS_FN = function() {\n        return this.size;\n    };\n\n    function PointerMap() {\n        if (USE_MAP) {\n            var m = new Map();\n            m.pointers = POINTERS_FN;\n            return m;\n        } else {\n            this.keys = [];\n            this.values = [];\n        }\n    }\n\n    PointerMap.prototype = {\n        set: function(inId, inEvent) {\n            var i = this.keys.indexOf(inId);\n            if (i > -1) {\n                this.values[i] = inEvent;\n            } else {\n                this.keys.push(inId);\n                this.values.push(inEvent);\n            }\n        },\n        has: function(inId) {\n            return this.keys.indexOf(inId) > -1;\n        },\n        'delete': function(inId) {\n            var i = this.keys.indexOf(inId);\n            if (i > -1) {\n                this.keys.splice(i, 1);\n                this.values.splice(i, 1);\n            }\n        },\n        get: function(inId) {\n            var i = this.keys.indexOf(inId);\n            return this.values[i];\n        },\n        clear: function() {\n            this.keys.length = 0;\n            this.values.length = 0;\n        },\n        // return value, key, map\n        forEach: function(callback, thisArg) {\n            this.values.forEach(function(v, i) {\n                callback.call(thisArg, v, this.keys[i], this);\n            }, this);\n        },\n        pointers: function() {\n            return this.keys.length;\n        }\n    };\n\n    scope.PointerMap = PointerMap;\n})(exports);\n\n(function(scope) {\n    var CLONE_PROPS = [\n        // MouseEvent\n        'bubbles',\n        'cancelable',\n        'view',\n        'detail',\n        'screenX',\n        'screenY',\n        'clientX',\n        'clientY',\n        'ctrlKey',\n        'altKey',\n        'shiftKey',\n        'metaKey',\n        'button',\n        'relatedTarget',\n        // DOM Level 3\n        'buttons',\n        // PointerEvent\n        'pointerId',\n        'width',\n        'height',\n        'pressure',\n        'tiltX',\n        'tiltY',\n        'pointerType',\n        'hwTimestamp',\n        'isPrimary',\n        // event instance\n        'type',\n        'target',\n        'currentTarget',\n        'which',\n        'pageX',\n        'pageY',\n        'timeStamp',\n        // gesture addons\n        'preventTap',\n        'tapPrevented',\n        '_source'\n    ];\n\n    var CLONE_DEFAULTS = [\n        // MouseEvent\n        false,\n        false,\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        false,\n        false,\n        false,\n        false,\n        0,\n        null,\n        // DOM Level 3\n        0,\n        // PointerEvent\n        0,\n        0,\n        0,\n        0,\n        0,\n        0,\n        '',\n        0,\n        false,\n        // event instance\n        '',\n        null,\n        null,\n        0,\n        0,\n        0,\n        0,\n        function() {},\n        false\n    ];\n\n    var HAS_SVG_INSTANCE = (typeof SVGElementInstance !== 'undefined');\n\n    var eventFactory = scope.eventFactory;\n\n    // set of recognizers to run for the currently handled event\n    var currentGestures;\n\n    /**\n     * This module is for normalizing events. Mouse and Touch events will be\n     * collected here, and fire PointerEvents that have the same semantics, no\n     * matter the source.\n     * Events fired:\n     *   - pointerdown: a pointing is added\n     *   - pointerup: a pointer is removed\n     *   - pointermove: a pointer is moved\n     *   - pointerover: a pointer crosses into an element\n     *   - pointerout: a pointer leaves an element\n     *   - pointercancel: a pointer will no longer generate events\n     */\n    var dispatcher = {\n        IS_IOS: false,\n        pointermap: new scope.PointerMap(),\n        requiredGestures: new scope.PointerMap(),\n        eventMap: Object.create(null),\n        // Scope objects for native events.\n        // This exists for ease of testing.\n        eventSources: Object.create(null),\n        eventSourceList: [],\n        gestures: [],\n        // map gesture event -> {listeners: int, index: gestures[int]}\n        dependencyMap: {\n            // make sure down and up are in the map to trigger \"register\"\n            down: {\n                listeners: 0,\n                index: -1\n            },\n            up: {\n                listeners: 0,\n                index: -1\n            }\n        },\n        gestureQueue: [],\n        /**\n         * Add a new event source that will generate pointer events.\n         *\n         * `inSource` must contain an array of event names named `events`, and\n         * functions with the names specified in the `events` array.\n         * @param {string} name A name for the event source\n         * @param {Object} source A new source of platform events.\n         */\n        registerSource: function(name, source) {\n            var s = source;\n            var newEvents = s.events;\n            if (newEvents) {\n                newEvents.forEach(function(e) {\n                    if (s[e]) {\n                        this.eventMap[e] = s[e].bind(s);\n                    }\n                }, this);\n                this.eventSources[name] = s;\n                this.eventSourceList.push(s);\n            }\n        },\n        registerGesture: function(name, source) {\n            var obj = Object.create(null);\n            obj.listeners = 0;\n            obj.index = this.gestures.length;\n            for (var i = 0, g; i < source.exposes.length; i++) {\n                g = source.exposes[i].toLowerCase();\n                this.dependencyMap[g] = obj;\n            }\n            this.gestures.push(source);\n        },\n        register: function(element, initial) {\n            var l = this.eventSourceList.length;\n            for (var i = 0, es;\n                (i < l) && (es = this.eventSourceList[i]); i++) {\n                // call eventsource register\n                es.register.call(es, element, initial);\n            }\n        },\n        unregister: function(element) {\n            var l = this.eventSourceList.length;\n            for (var i = 0, es;\n                (i < l) && (es = this.eventSourceList[i]); i++) {\n                // call eventsource register\n                es.unregister.call(es, element);\n            }\n        },\n        // EVENTS\n        down: function(inEvent) {\n            this.requiredGestures.set(inEvent.pointerId, currentGestures);\n            this.fireEvent('down', inEvent);\n        },\n        move: function(inEvent) {\n            // pipe move events into gesture queue directly\n            inEvent.type = 'move';\n            this.fillGestureQueue(inEvent);\n        },\n        up: function(inEvent) {\n            this.fireEvent('up', inEvent);\n            this.requiredGestures.delete(inEvent.pointerId);\n        },\n        cancel: function(inEvent) {\n            inEvent.tapPrevented = true;\n            this.fireEvent('up', inEvent);\n            this.requiredGestures.delete(inEvent.pointerId);\n        },\n        addGestureDependency: function(node, currentGestures) {\n            var gesturesWanted = node._pgEvents;\n            if (gesturesWanted && currentGestures) {\n                var gk = Object.keys(gesturesWanted);\n                for (var i = 0, r, ri, g; i < gk.length; i++) {\n                    // gesture\n                    g = gk[i];\n                    if (gesturesWanted[g] > 0) {\n                        // lookup gesture recognizer\n                        r = this.dependencyMap[g];\n                        // recognizer index\n                        ri = r ? r.index : -1;\n                        currentGestures[ri] = true;\n                    }\n                }\n            }\n        },\n        // LISTENER LOGIC\n        eventHandler: function(inEvent) {\n            // This is used to prevent multiple dispatch of events from\n            // platform events. This can happen when two elements in different scopes\n            // are set up to create pointer events, which is relevant to Shadow DOM.\n\n            var type = inEvent.type;\n\n            // only generate the list of desired events on \"down\"\n            if (type === 'touchstart' || type === 'mousedown' || type === 'pointerdown' || type === 'MSPointerDown') {\n                if (!inEvent._handledByPG) {\n                    currentGestures = {};\n                }\n\n                // in IOS mode, there is only a listener on the document, so this is not re-entrant\n                if (this.IS_IOS) {\n                    var ev = inEvent;\n                    if (type === 'touchstart') {\n                        var ct = inEvent.changedTouches[0];\n                        // set up a fake event to give to the path builder\n                        ev = {\n                            target: inEvent.target,\n                            clientX: ct.clientX,\n                            clientY: ct.clientY,\n                            path: inEvent.path\n                        };\n                    }\n                    // use event path if available, otherwise build a path from target finding\n                    var nodes = inEvent.path || scope.targetFinding.path(ev);\n                    for (var i = 0, n; i < nodes.length; i++) {\n                        n = nodes[i];\n                        this.addGestureDependency(n, currentGestures);\n                    }\n                } else {\n                    this.addGestureDependency(inEvent.currentTarget, currentGestures);\n                }\n            }\n\n            if (inEvent._handledByPG) {\n                return;\n            }\n            var fn = this.eventMap && this.eventMap[type];\n            if (fn) {\n                fn(inEvent);\n            }\n            inEvent._handledByPG = true;\n        },\n        // set up event listeners\n        listen: function(target, events) {\n            for (var i = 0, l = events.length, e;\n                (i < l) && (e = events[i]); i++) {\n                this.addEvent(target, e);\n            }\n        },\n        // remove event listeners\n        unlisten: function(target, events) {\n            for (var i = 0, l = events.length, e;\n                (i < l) && (e = events[i]); i++) {\n                this.removeEvent(target, e);\n            }\n        },\n        addEvent: function(target, eventName) {\n            target.addEventListener(eventName, this.boundHandler);\n        },\n        removeEvent: function(target, eventName) {\n            target.removeEventListener(eventName, this.boundHandler);\n        },\n        // EVENT CREATION AND TRACKING\n        /**\n         * Creates a new Event of type `inType`, based on the information in\n         * `inEvent`.\n         *\n         * @param {string} inType A string representing the type of event to create\n         * @param {Event} inEvent A platform event with a target\n         * @return {Event} A PointerEvent of type `inType`\n         */\n        makeEvent: function(inType, inEvent) {\n            var e = eventFactory.makePointerEvent(inType, inEvent);\n            e.preventDefault = inEvent.preventDefault;\n            e.tapPrevented = inEvent.tapPrevented;\n            e._target = e._target || inEvent.target;\n            return e;\n        },\n        // make and dispatch an event in one call\n        fireEvent: function(inType, inEvent) {\n            var e = this.makeEvent(inType, inEvent);\n            return this.dispatchEvent(e);\n        },\n        /**\n         * Returns a snapshot of inEvent, with writable properties.\n         *\n         * @param {Event} inEvent An event that contains properties to copy.\n         * @return {Object} An object containing shallow copies of `inEvent`'s\n         *    properties.\n         */\n        cloneEvent: function(inEvent) {\n            var eventCopy = Object.create(null),\n                p;\n            for (var i = 0; i < CLONE_PROPS.length; i++) {\n                p = CLONE_PROPS[i];\n                eventCopy[p] = inEvent[p] || CLONE_DEFAULTS[i];\n                // Work around SVGInstanceElement shadow tree\n                // Return the <use> element that is represented by the instance for Safari, Chrome, IE.\n                // This is the behavior implemented by Firefox.\n                if (p === 'target' || p === 'relatedTarget') {\n                    if (HAS_SVG_INSTANCE && eventCopy[p] instanceof SVGElementInstance) {\n                        eventCopy[p] = eventCopy[p].correspondingUseElement;\n                    }\n                }\n            }\n            // keep the semantics of preventDefault\n            eventCopy.preventDefault = function() {\n                inEvent.preventDefault();\n            };\n            return eventCopy;\n        },\n        /**\n         * Dispatches the event to its target.\n         *\n         * @param {Event} inEvent The event to be dispatched.\n         * @return {Boolean} True if an event handler returns true, false otherwise.\n         */\n        dispatchEvent: function(inEvent) {\n            var t = inEvent._target;\n            if (t) {\n                t.dispatchEvent(inEvent);\n                // clone the event for the gesture system to process\n                // clone after dispatch to pick up gesture prevention code\n                var clone = this.cloneEvent(inEvent);\n                clone.target = t;\n                this.fillGestureQueue(clone);\n            }\n        },\n        gestureTrigger: function() {\n            // process the gesture queue\n            for (var i = 0, e, rg; i < this.gestureQueue.length; i++) {\n                e = this.gestureQueue[i];\n                rg = e._requiredGestures;\n                if (rg) {\n                    for (var j = 0, g, fn; j < this.gestures.length; j++) {\n                        // only run recognizer if an element in the source event's path is listening for those gestures\n                        if (rg[j]) {\n                            g = this.gestures[j];\n                            fn = g[e.type];\n                            if (fn) {\n                                fn.call(g, e);\n                            }\n                        }\n                    }\n                }\n            }\n            this.gestureQueue.length = 0;\n        },\n        fillGestureQueue: function(ev) {\n            // only trigger the gesture queue once\n            if (!this.gestureQueue.length) {\n                requestAnimationFrame(this.boundGestureTrigger);\n            }\n            ev._requiredGestures = this.requiredGestures.get(ev.pointerId);\n            this.gestureQueue.push(ev);\n        }\n    };\n    dispatcher.boundHandler = dispatcher.eventHandler.bind(dispatcher);\n    dispatcher.boundGestureTrigger = dispatcher.gestureTrigger.bind(dispatcher);\n    scope.dispatcher = dispatcher;\n\n    /**\n     * Listen for `gesture` on `node` with the `handler` function\n     *\n     * If `handler` is the first listener for `gesture`, the underlying gesture recognizer is then enabled.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @return Boolean `gesture` is a valid gesture\n     */\n    scope.activateGesture = function(node, gesture) {\n        var g = gesture.toLowerCase();\n        var dep = dispatcher.dependencyMap[g];\n        if (dep) {\n            var recognizer = dispatcher.gestures[dep.index];\n            if (!node._pgListeners) {\n                dispatcher.register(node);\n                node._pgListeners = 0;\n            }\n            // TODO(dfreedm): re-evaluate bookkeeping to avoid using attributes\n            if (recognizer) {\n                var touchAction = recognizer.defaultActions && recognizer.defaultActions[g];\n                var actionNode;\n                switch (node.nodeType) {\n                    case Node.ELEMENT_NODE:\n                        actionNode = node;\n                        break;\n                    case Node.DOCUMENT_FRAGMENT_NODE:\n                        actionNode = node.host;\n                        break;\n                    default:\n                        actionNode = null;\n                        break;\n                }\n                if (touchAction && actionNode && !actionNode.hasAttribute('touch-action')) {\n                    actionNode.setAttribute('touch-action', touchAction);\n                }\n            }\n            if (!node._pgEvents) {\n                node._pgEvents = {};\n            }\n            node._pgEvents[g] = (node._pgEvents[g] || 0) + 1;\n            node._pgListeners++;\n        }\n        return Boolean(dep);\n    };\n\n    /**\n     *\n     * Listen for `gesture` from `node` with `handler` function.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @param {Function} handler\n     * @param {Boolean} capture\n     */\n    scope.addEventListener = function(node, gesture, handler, capture) {\n        if (handler) {\n            scope.activateGesture(node, gesture);\n            node.addEventListener(gesture, handler, capture);\n        }\n    };\n\n    /**\n     * Tears down the gesture configuration for `node`\n     *\n     * If `handler` is the last listener for `gesture`, the underlying gesture recognizer is disabled.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @return Boolean `gesture` is a valid gesture\n     */\n    scope.deactivateGesture = function(node, gesture) {\n        var g = gesture.toLowerCase();\n        var dep = dispatcher.dependencyMap[g];\n        if (dep) {\n            if (node._pgListeners > 0) {\n                node._pgListeners--;\n            }\n            if (node._pgListeners === 0) {\n                dispatcher.unregister(node);\n            }\n            if (node._pgEvents) {\n                if (node._pgEvents[g] > 0) {\n                    node._pgEvents[g]--;\n                } else {\n                    node._pgEvents[g] = 0;\n                }\n            }\n        }\n        return Boolean(dep);\n    };\n\n    /**\n     * Stop listening for `gesture` from `node` with `handler` function.\n     *\n     * @param {Element} node\n     * @param {string} gesture\n     * @param {Function} handler\n     * @param {Boolean} capture\n     */\n    scope.removeEventListener = function(node, gesture, handler, capture) {\n        if (handler) {\n            scope.deactivateGesture(node, gesture);\n            node.removeEventListener(gesture, handler, capture);\n        }\n    };\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    // radius around touchend that swallows mouse events\n    var DEDUP_DIST = 25;\n\n    var WHICH_TO_BUTTONS = [0, 1, 4, 2];\n\n    var currentButtons = 0;\n\n    var FIREFOX_LINUX = /Linux.*Firefox\\//i;\n\n    var HAS_BUTTONS = (function() {\n        // firefox on linux returns spec-incorrect values for mouseup.buttons\n        // https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent.buttons#See_also\n        // https://codereview.chromium.org/727593003/#msg16\n        if (FIREFOX_LINUX.test(navigator.userAgent)) {\n            return false;\n        }\n        try {\n            return new MouseEvent('test', {\n                buttons: 1\n            }).buttons === 1;\n        } catch (e) {\n            return false;\n        }\n    })();\n\n    // handler block for native mouse events\n    var mouseEvents = {\n        POINTER_ID: 1,\n        POINTER_TYPE: 'mouse',\n        events: [\n            'mousedown',\n            'mousemove',\n            'mouseup'\n        ],\n        exposes: [\n            'down',\n            'up',\n            'move'\n        ],\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        lastTouches: [],\n        // collide with the global mouse listener\n        isEventSimulatedFromTouch: function(inEvent) {\n            var lts = this.lastTouches;\n            var x = inEvent.clientX,\n                y = inEvent.clientY;\n            for (var i = 0, l = lts.length, t; i < l && (t = lts[i]); i++) {\n                // simulated mouse events will be swallowed near a primary touchend\n                var dx = Math.abs(x - t.x),\n                    dy = Math.abs(y - t.y);\n                if (dx <= DEDUP_DIST && dy <= DEDUP_DIST) {\n                    return true;\n                }\n            }\n        },\n        prepareEvent: function(inEvent) {\n            var e = dispatcher.cloneEvent(inEvent);\n            e.pointerId = this.POINTER_ID;\n            e.isPrimary = true;\n            e.pointerType = this.POINTER_TYPE;\n            e._source = 'mouse';\n            if (!HAS_BUTTONS) {\n                var type = inEvent.type;\n                var bit = WHICH_TO_BUTTONS[inEvent.which] || 0;\n                if (type === 'mousedown') {\n                    currentButtons |= bit;\n                } else if (type === 'mouseup') {\n                    currentButtons &= ~bit;\n                }\n                e.buttons = currentButtons;\n            }\n            return e;\n        },\n        mousedown: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var p = pointermap.has(this.POINTER_ID);\n                var e = this.prepareEvent(inEvent);\n                e.target = scope.findTarget(inEvent);\n                pointermap.set(this.POINTER_ID, e.target);\n                dispatcher.down(e);\n            }\n        },\n        mousemove: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var target = pointermap.get(this.POINTER_ID);\n                if (target) {\n                    var e = this.prepareEvent(inEvent);\n                    e.target = target;\n                    // handle case where we missed a mouseup\n                    if ((HAS_BUTTONS ? e.buttons : e.which) === 0) {\n                        if (!HAS_BUTTONS) {\n                            currentButtons = e.buttons = 0;\n                        }\n                        dispatcher.cancel(e);\n                        this.cleanupMouse(e.buttons);\n                    } else {\n                        dispatcher.move(e);\n                    }\n                }\n            }\n        },\n        mouseup: function(inEvent) {\n            if (!this.isEventSimulatedFromTouch(inEvent)) {\n                var e = this.prepareEvent(inEvent);\n                e.relatedTarget = scope.findTarget(inEvent);\n                e.target = pointermap.get(this.POINTER_ID);\n                dispatcher.up(e);\n                this.cleanupMouse(e.buttons);\n            }\n        },\n        cleanupMouse: function(buttons) {\n            if (buttons === 0) {\n                pointermap.delete(this.POINTER_ID);\n            }\n        }\n    };\n\n    scope.mouseEvents = mouseEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var allShadows = scope.targetFinding.allShadows.bind(scope.targetFinding);\n    var pointermap = dispatcher.pointermap;\n    var touchMap = Array.prototype.map.call.bind(Array.prototype.map);\n    // This should be long enough to ignore compat mouse events made by touch\n    var DEDUP_TIMEOUT = 2500;\n    var DEDUP_DIST = 25;\n    var CLICK_COUNT_TIMEOUT = 200;\n    var HYSTERESIS = 20;\n    var ATTRIB = 'touch-action';\n    // TODO(dfreedm): disable until http://crbug.com/399765 is resolved\n    // var HAS_TOUCH_ACTION = ATTRIB in document.head.style;\n    var HAS_TOUCH_ACTION = false;\n\n    // handler block for native touch events\n    var touchEvents = {\n        IS_IOS: false,\n        events: [\n            'touchstart',\n            'touchmove',\n            'touchend',\n            'touchcancel'\n        ],\n        exposes: [\n            'down',\n            'up',\n            'move'\n        ],\n        register: function(target, initial) {\n            if (this.IS_IOS ? initial : !initial) {\n                dispatcher.listen(target, this.events);\n            }\n        },\n        unregister: function(target) {\n            if (!this.IS_IOS) {\n                dispatcher.unlisten(target, this.events);\n            }\n        },\n        scrollTypes: {\n            EMITTER: 'none',\n            XSCROLLER: 'pan-x',\n            YSCROLLER: 'pan-y',\n        },\n        touchActionToScrollType: function(touchAction) {\n            var t = touchAction;\n            var st = this.scrollTypes;\n            if (t === st.EMITTER) {\n                return 'none';\n            } else if (t === st.XSCROLLER) {\n                return 'X';\n            } else if (t === st.YSCROLLER) {\n                return 'Y';\n            } else {\n                return 'XY';\n            }\n        },\n        POINTER_TYPE: 'touch',\n        firstTouch: null,\n        isPrimaryTouch: function(inTouch) {\n            return this.firstTouch === inTouch.identifier;\n        },\n        setPrimaryTouch: function(inTouch) {\n            // set primary touch if there no pointers, or the only pointer is the mouse\n            if (pointermap.pointers() === 0 || (pointermap.pointers() === 1 && pointermap.has(1))) {\n                this.firstTouch = inTouch.identifier;\n                this.firstXY = {\n                    X: inTouch.clientX,\n                    Y: inTouch.clientY\n                };\n                this.firstTarget = inTouch.target;\n                this.scrolling = null;\n                this.cancelResetClickCount();\n            }\n        },\n        removePrimaryPointer: function(inPointer) {\n            if (inPointer.isPrimary) {\n                this.firstTouch = null;\n                this.firstXY = null;\n                this.resetClickCount();\n            }\n        },\n        clickCount: 0,\n        resetId: null,\n        resetClickCount: function() {\n            var fn = function() {\n                this.clickCount = 0;\n                this.resetId = null;\n            }.bind(this);\n            this.resetId = setTimeout(fn, CLICK_COUNT_TIMEOUT);\n        },\n        cancelResetClickCount: function() {\n            if (this.resetId) {\n                clearTimeout(this.resetId);\n            }\n        },\n        typeToButtons: function(type) {\n            var ret = 0;\n            if (type === 'touchstart' || type === 'touchmove') {\n                ret = 1;\n            }\n            return ret;\n        },\n        findTarget: function(touch, id) {\n            if (this.currentTouchEvent.type === 'touchstart') {\n                if (this.isPrimaryTouch(touch)) {\n                    var fastPath = {\n                        clientX: touch.clientX,\n                        clientY: touch.clientY,\n                        path: this.currentTouchEvent.path,\n                        target: this.currentTouchEvent.target\n                    };\n                    return scope.findTarget(fastPath);\n                } else {\n                    return scope.findTarget(touch);\n                }\n            }\n            // reuse target we found in touchstart\n            return pointermap.get(id);\n        },\n        touchToPointer: function(inTouch) {\n            var cte = this.currentTouchEvent;\n            var e = dispatcher.cloneEvent(inTouch);\n            // Spec specifies that pointerId 1 is reserved for Mouse.\n            // Touch identifiers can start at 0.\n            // Add 2 to the touch identifier for compatibility.\n            var id = e.pointerId = inTouch.identifier + 2;\n            e.target = this.findTarget(inTouch, id);\n            e.bubbles = true;\n            e.cancelable = true;\n            e.detail = this.clickCount;\n            e.buttons = this.typeToButtons(cte.type);\n            e.width = inTouch.webkitRadiusX || inTouch.radiusX || 0;\n            e.height = inTouch.webkitRadiusY || inTouch.radiusY || 0;\n            e.pressure = inTouch.webkitForce || inTouch.force || 0.5;\n            e.isPrimary = this.isPrimaryTouch(inTouch);\n            e.pointerType = this.POINTER_TYPE;\n            e._source = 'touch';\n            // forward touch preventDefaults\n            var self = this;\n            e.preventDefault = function() {\n                self.scrolling = false;\n                self.firstXY = null;\n                cte.preventDefault();\n            };\n            return e;\n        },\n        processTouches: function(inEvent, inFunction) {\n            var tl = inEvent.changedTouches;\n            this.currentTouchEvent = inEvent;\n            for (var i = 0, t, p; i < tl.length; i++) {\n                t = tl[i];\n                p = this.touchToPointer(t);\n                if (inEvent.type === 'touchstart') {\n                    pointermap.set(p.pointerId, p.target);\n                }\n                if (pointermap.has(p.pointerId)) {\n                    inFunction.call(this, p);\n                }\n                if (inEvent.type === 'touchend' || inEvent._cancel) {\n                    this.cleanUpPointer(p);\n                }\n            }\n        },\n        // For single axis scrollers, determines whether the element should emit\n        // pointer events or behave as a scroller\n        shouldScroll: function(inEvent) {\n            if (this.firstXY) {\n                var ret;\n                var touchAction = scope.targetFinding.findTouchAction(inEvent);\n                var scrollAxis = this.touchActionToScrollType(touchAction);\n                if (scrollAxis === 'none') {\n                    // this element is a touch-action: none, should never scroll\n                    ret = false;\n                } else if (scrollAxis === 'XY') {\n                    // this element should always scroll\n                    ret = true;\n                } else {\n                    var t = inEvent.changedTouches[0];\n                    // check the intended scroll axis, and other axis\n                    var a = scrollAxis;\n                    var oa = scrollAxis === 'Y' ? 'X' : 'Y';\n                    var da = Math.abs(t['client' + a] - this.firstXY[a]);\n                    var doa = Math.abs(t['client' + oa] - this.firstXY[oa]);\n                    // if delta in the scroll axis > delta other axis, scroll instead of\n                    // making events\n                    ret = da >= doa;\n                }\n                return ret;\n            }\n        },\n        findTouch: function(inTL, inId) {\n            for (var i = 0, l = inTL.length, t; i < l && (t = inTL[i]); i++) {\n                if (t.identifier === inId) {\n                    return true;\n                }\n            }\n        },\n        // In some instances, a touchstart can happen without a touchend. This\n        // leaves the pointermap in a broken state.\n        // Therefore, on every touchstart, we remove the touches that did not fire a\n        // touchend event.\n        // To keep state globally consistent, we fire a\n        // pointercancel for this \"abandoned\" touch\n        vacuumTouches: function(inEvent) {\n            var tl = inEvent.touches;\n            // pointermap.pointers() should be < tl.length here, as the touchstart has not\n            // been processed yet.\n            if (pointermap.pointers() >= tl.length) {\n                var d = [];\n                pointermap.forEach(function(value, key) {\n                    // Never remove pointerId == 1, which is mouse.\n                    // Touch identifiers are 2 smaller than their pointerId, which is the\n                    // index in pointermap.\n                    if (key !== 1 && !this.findTouch(tl, key - 2)) {\n                        var p = value;\n                        d.push(p);\n                    }\n                }, this);\n                d.forEach(function(p) {\n                    this.cancel(p);\n                    pointermap.delete(p.pointerId);\n                }, this);\n            }\n        },\n        touchstart: function(inEvent) {\n            this.vacuumTouches(inEvent);\n            this.setPrimaryTouch(inEvent.changedTouches[0]);\n            this.dedupSynthMouse(inEvent);\n            if (!this.scrolling) {\n                this.clickCount++;\n                this.processTouches(inEvent, this.down);\n            }\n        },\n        down: function(inPointer) {\n            dispatcher.down(inPointer);\n        },\n        touchmove: function(inEvent) {\n            if (HAS_TOUCH_ACTION) {\n                // touchevent.cancelable == false is sent when the page is scrolling under native Touch Action in Chrome 36\n                // https://groups.google.com/a/chromium.org/d/msg/input-dev/wHnyukcYBcA/b9kmtwM1jJQJ\n                if (inEvent.cancelable) {\n                    this.processTouches(inEvent, this.move);\n                }\n            } else {\n                if (!this.scrolling) {\n                    if (this.scrolling === null && this.shouldScroll(inEvent)) {\n                        this.scrolling = true;\n                    } else {\n                        this.scrolling = false;\n                        inEvent.preventDefault();\n                        this.processTouches(inEvent, this.move);\n                    }\n                } else if (this.firstXY) {\n                    var t = inEvent.changedTouches[0];\n                    var dx = t.clientX - this.firstXY.X;\n                    var dy = t.clientY - this.firstXY.Y;\n                    var dd = Math.sqrt(dx * dx + dy * dy);\n                    if (dd >= HYSTERESIS) {\n                        this.touchcancel(inEvent);\n                        this.scrolling = true;\n                        this.firstXY = null;\n                    }\n                }\n            }\n        },\n        move: function(inPointer) {\n            dispatcher.move(inPointer);\n        },\n        touchend: function(inEvent) {\n            this.dedupSynthMouse(inEvent);\n            this.processTouches(inEvent, this.up);\n        },\n        up: function(inPointer) {\n            inPointer.relatedTarget = scope.findTarget(inPointer);\n            dispatcher.up(inPointer);\n        },\n        cancel: function(inPointer) {\n            dispatcher.cancel(inPointer);\n        },\n        touchcancel: function(inEvent) {\n            inEvent._cancel = true;\n            this.processTouches(inEvent, this.cancel);\n        },\n        cleanUpPointer: function(inPointer) {\n            pointermap['delete'](inPointer.pointerId);\n            this.removePrimaryPointer(inPointer);\n        },\n        // prevent synth mouse events from creating pointer events\n        dedupSynthMouse: function(inEvent) {\n            var lts = scope.mouseEvents.lastTouches;\n            var t = inEvent.changedTouches[0];\n            // only the primary finger will synth mouse events\n            if (this.isPrimaryTouch(t)) {\n                // remember x/y of last touch\n                var lt = {\n                    x: t.clientX,\n                    y: t.clientY\n                };\n                lts.push(lt);\n                var fn = (function(lts, lt) {\n                    var i = lts.indexOf(lt);\n                    if (i > -1) {\n                        lts.splice(i, 1);\n                    }\n                }).bind(null, lts, lt);\n                setTimeout(fn, DEDUP_TIMEOUT);\n            }\n        }\n    };\n\n    // prevent \"ghost clicks\" that come from elements that were removed in a touch handler\n    var STOP_PROP_FN = Event.prototype.stopImmediatePropagation || Event.prototype.stopPropagation;\n    document.addEventListener('click', function(ev) {\n        var x = ev.clientX,\n            y = ev.clientY;\n        // check if a click is within DEDUP_DIST px radius of the touchstart\n        var closeTo = function(touch) {\n            var dx = Math.abs(x - touch.x),\n                dy = Math.abs(y - touch.y);\n            return (dx <= DEDUP_DIST && dy <= DEDUP_DIST);\n        };\n        // if click coordinates are close to touch coordinates, assume the click came from a touch\n        var wasTouched = scope.mouseEvents.lastTouches.some(closeTo);\n        // if the click came from touch, and the touchstart target is not in the path of the click event,\n        // then the touchstart target was probably removed, and the click should be \"busted\"\n        var path = scope.targetFinding.path(ev);\n        if (wasTouched) {\n            for (var i = 0; i < path.length; i++) {\n                if (path[i] === touchEvents.firstTarget) {\n                    return;\n                }\n            }\n            ev.preventDefault();\n            STOP_PROP_FN.call(ev);\n        }\n    }, true);\n\n    scope.touchEvents = touchEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    var HAS_BITMAP_TYPE = window.MSPointerEvent && typeof window.MSPointerEvent.MSPOINTER_TYPE_MOUSE === 'number';\n    var msEvents = {\n        events: [\n            'MSPointerDown',\n            'MSPointerMove',\n            'MSPointerUp',\n            'MSPointerCancel',\n        ],\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        POINTER_TYPES: [\n            '',\n            'unavailable',\n            'touch',\n            'pen',\n            'mouse'\n        ],\n        prepareEvent: function(inEvent) {\n            var e = inEvent;\n            e = dispatcher.cloneEvent(inEvent);\n            if (HAS_BITMAP_TYPE) {\n                e.pointerType = this.POINTER_TYPES[inEvent.pointerType];\n            }\n            e._source = 'ms';\n            return e;\n        },\n        cleanup: function(id) {\n            pointermap['delete'](id);\n        },\n        MSPointerDown: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.target = scope.findTarget(inEvent);\n            pointermap.set(inEvent.pointerId, e.target);\n            dispatcher.down(e);\n        },\n        MSPointerMove: function(inEvent) {\n            var target = pointermap.get(inEvent.pointerId);\n            if (target) {\n                var e = this.prepareEvent(inEvent);\n                e.target = target;\n                dispatcher.move(e);\n            }\n        },\n        MSPointerUp: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.up(e);\n            this.cleanup(inEvent.pointerId);\n        },\n        MSPointerCancel: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.cancel(e);\n            this.cleanup(inEvent.pointerId);\n        }\n    };\n\n    scope.msEvents = msEvents;\n})(exports);\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var pointermap = dispatcher.pointermap;\n    var pointerEvents = {\n        events: [\n            'pointerdown',\n            'pointermove',\n            'pointerup',\n            'pointercancel'\n        ],\n        prepareEvent: function(inEvent) {\n            var e = dispatcher.cloneEvent(inEvent);\n            e._source = 'pointer';\n            return e;\n        },\n        register: function(target) {\n            dispatcher.listen(target, this.events);\n        },\n        unregister: function(target) {\n            if (target.nodeType === Node.DOCUMENT_NODE) {\n                return;\n            }\n            dispatcher.unlisten(target, this.events);\n        },\n        cleanup: function(id) {\n            pointermap['delete'](id);\n        },\n        pointerdown: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.target = scope.findTarget(inEvent);\n            pointermap.set(e.pointerId, e.target);\n            dispatcher.down(e);\n        },\n        pointermove: function(inEvent) {\n            var target = pointermap.get(inEvent.pointerId);\n            if (target) {\n                var e = this.prepareEvent(inEvent);\n                e.target = target;\n                dispatcher.move(e);\n            }\n        },\n        pointerup: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.up(e);\n            this.cleanup(inEvent.pointerId);\n        },\n        pointercancel: function(inEvent) {\n            var e = this.prepareEvent(inEvent);\n            e.relatedTarget = scope.findTarget(inEvent);\n            e.target = pointermap.get(e.pointerId);\n            dispatcher.cancel(e);\n            this.cleanup(inEvent.pointerId);\n        }\n    };\n\n    scope.pointerEvents = pointerEvents;\n})(exports);\n\n/**\n * This module contains the handlers for native platform events.\n * From here, the dispatcher is called to create unified pointer events.\n * Included are touch events (v1), mouse events, and MSPointerEvents.\n */\n(function(scope) {\n\n    var dispatcher = scope.dispatcher;\n    var nav = window.navigator;\n\n    if (window.PointerEvent) {\n        dispatcher.registerSource('pointer', scope.pointerEvents);\n    } else if (nav.msPointerEnabled) {\n        dispatcher.registerSource('ms', scope.msEvents);\n    } else {\n        dispatcher.registerSource('mouse', scope.mouseEvents);\n        if (window.ontouchstart !== undefined) {\n            dispatcher.registerSource('touch', scope.touchEvents);\n        }\n    }\n\n    // Work around iOS bugs https://bugs.webkit.org/show_bug.cgi?id=135628 and https://bugs.webkit.org/show_bug.cgi?id=136506\n    var ua = navigator.userAgent;\n    var IS_IOS = ua.match(/iPad|iPhone|iPod/) && 'ontouchstart' in window;\n\n    dispatcher.IS_IOS = IS_IOS;\n    scope.touchEvents.IS_IOS = IS_IOS;\n\n    dispatcher.register(document, true);\n})(exports);\n\n/**\n * This event denotes the beginning of a series of tracking events.\n *\n * @module PointerGestures\n * @submodule Events\n * @class trackstart\n */\n/**\n * Pixels moved in the x direction since trackstart.\n * @type Number\n * @property dx\n */\n/**\n * Pixes moved in the y direction since trackstart.\n * @type Number\n * @property dy\n */\n/**\n * Pixels moved in the x direction since the last track.\n * @type Number\n * @property ddx\n */\n/**\n * Pixles moved in the y direction since the last track.\n * @type Number\n * @property ddy\n */\n/**\n * The clientX position of the track gesture.\n * @type Number\n * @property clientX\n */\n/**\n * The clientY position of the track gesture.\n * @type Number\n * @property clientY\n */\n/**\n * The pageX position of the track gesture.\n * @type Number\n * @property pageX\n */\n/**\n * The pageY position of the track gesture.\n * @type Number\n * @property pageY\n */\n/**\n * The screenX position of the track gesture.\n * @type Number\n * @property screenX\n */\n/**\n * The screenY position of the track gesture.\n * @type Number\n * @property screenY\n */\n/**\n * The last x axis direction of the pointer.\n * @type Number\n * @property xDirection\n */\n/**\n * The last y axis direction of the pointer.\n * @type Number\n * @property yDirection\n */\n/**\n * A shared object between all tracking events.\n * @type Object\n * @property trackInfo\n */\n/**\n * The element currently under the pointer.\n * @type Element\n * @property relatedTarget\n */\n/**\n * The type of pointer that make the track gesture.\n * @type String\n * @property pointerType\n */\n/**\n *\n * This event fires for all pointer movement being tracked.\n *\n * @class track\n * @extends trackstart\n */\n/**\n * This event fires when the pointer is no longer being tracked.\n *\n * @class trackend\n * @extends trackstart\n */\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var track = {\n        events: [\n            'down',\n            'move',\n            'up',\n        ],\n        exposes: [\n            'trackstart',\n            'track',\n            'trackx',\n            'tracky',\n            'trackend'\n        ],\n        defaultActions: {\n            'track': 'none',\n            'trackx': 'pan-y',\n            'tracky': 'pan-x'\n        },\n        WIGGLE_THRESHOLD: 4,\n        clampDir: function(inDelta) {\n            return inDelta > 0 ? 1 : -1;\n        },\n        calcPositionDelta: function(inA, inB) {\n            var x = 0,\n                y = 0;\n            if (inA && inB) {\n                x = inB.pageX - inA.pageX;\n                y = inB.pageY - inA.pageY;\n            }\n            return {\n                x: x,\n                y: y\n            };\n        },\n        fireTrack: function(inType, inEvent, inTrackingData) {\n            var t = inTrackingData;\n            var d = this.calcPositionDelta(t.downEvent, inEvent);\n            var dd = this.calcPositionDelta(t.lastMoveEvent, inEvent);\n            if (dd.x) {\n                t.xDirection = this.clampDir(dd.x);\n            } else if (inType === 'trackx') {\n                return;\n            }\n            if (dd.y) {\n                t.yDirection = this.clampDir(dd.y);\n            } else if (inType === 'tracky') {\n                return;\n            }\n            var gestureProto = {\n                bubbles: true,\n                cancelable: true,\n                trackInfo: t.trackInfo,\n                relatedTarget: inEvent.relatedTarget,\n                pointerType: inEvent.pointerType,\n                pointerId: inEvent.pointerId,\n                _source: 'track'\n            };\n            if (inType !== 'tracky') {\n                gestureProto.x = inEvent.x;\n                gestureProto.dx = d.x;\n                gestureProto.ddx = dd.x;\n                gestureProto.clientX = inEvent.clientX;\n                gestureProto.pageX = inEvent.pageX;\n                gestureProto.screenX = inEvent.screenX;\n                gestureProto.xDirection = t.xDirection;\n            }\n            if (inType !== 'trackx') {\n                gestureProto.dy = d.y;\n                gestureProto.ddy = dd.y;\n                gestureProto.y = inEvent.y;\n                gestureProto.clientY = inEvent.clientY;\n                gestureProto.pageY = inEvent.pageY;\n                gestureProto.screenY = inEvent.screenY;\n                gestureProto.yDirection = t.yDirection;\n            }\n            var e = eventFactory.makeGestureEvent(inType, gestureProto);\n            t.downTarget.dispatchEvent(e);\n        },\n        down: function(inEvent) {\n            if (inEvent.isPrimary && (inEvent.pointerType === 'mouse' ? inEvent.buttons === 1 : true)) {\n                var p = {\n                    downEvent: inEvent,\n                    downTarget: inEvent.target,\n                    trackInfo: {},\n                    lastMoveEvent: null,\n                    xDirection: 0,\n                    yDirection: 0,\n                    tracking: false\n                };\n                pointermap.set(inEvent.pointerId, p);\n            }\n        },\n        move: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            if (p) {\n                if (!p.tracking) {\n                    var d = this.calcPositionDelta(p.downEvent, inEvent);\n                    var move = d.x * d.x + d.y * d.y;\n                    // start tracking only if finger moves more than WIGGLE_THRESHOLD\n                    if (move > this.WIGGLE_THRESHOLD) {\n                        p.tracking = true;\n                        p.lastMoveEvent = p.downEvent;\n                        this.fireTrack('trackstart', inEvent, p);\n                    }\n                }\n                if (p.tracking) {\n                    this.fireTrack('track', inEvent, p);\n                    this.fireTrack('trackx', inEvent, p);\n                    this.fireTrack('tracky', inEvent, p);\n                }\n                p.lastMoveEvent = inEvent;\n            }\n        },\n        up: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            if (p) {\n                if (p.tracking) {\n                    this.fireTrack('trackend', inEvent, p);\n                }\n                pointermap.delete(inEvent.pointerId);\n            }\n        }\n    };\n    dispatcher.registerGesture('track', track);\n})(exports);\n\n/**\n * This event is fired when a pointer is held down for 200ms.\n *\n * @module PointerGestures\n * @submodule Events\n * @class hold\n */\n/**\n * Type of pointer that made the holding event.\n * @type String\n * @property pointerType\n */\n/**\n * Screen X axis position of the held pointer\n * @type Number\n * @property clientX\n */\n/**\n * Screen Y axis position of the held pointer\n * @type Number\n * @property clientY\n */\n/**\n * Type of pointer that made the holding event.\n * @type String\n * @property pointerType\n */\n/**\n * This event is fired every 200ms while a pointer is held down.\n *\n * @class holdpulse\n * @extends hold\n */\n/**\n * Milliseconds pointer has been held down.\n * @type Number\n * @property holdTime\n */\n/**\n * This event is fired when a held pointer is released or moved.\n *\n * @class release\n */\n\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var hold = {\n        // wait at least HOLD_DELAY ms between hold and pulse events\n        HOLD_DELAY: 200,\n        // pointer can move WIGGLE_THRESHOLD pixels before not counting as a hold\n        WIGGLE_THRESHOLD: 16,\n        events: [\n            'down',\n            'move',\n            'up',\n        ],\n        exposes: [\n            'hold',\n            'holdpulse',\n            'release'\n        ],\n        heldPointer: null,\n        holdJob: null,\n        pulse: function() {\n            var hold = Date.now() - this.heldPointer.timeStamp;\n            var type = this.held ? 'holdpulse' : 'hold';\n            this.fireHold(type, hold);\n            this.held = true;\n        },\n        cancel: function() {\n            clearInterval(this.holdJob);\n            if (this.held) {\n                this.fireHold('release');\n            }\n            this.held = false;\n            this.heldPointer = null;\n            this.target = null;\n            this.holdJob = null;\n        },\n        down: function(inEvent) {\n            if (inEvent.isPrimary && !this.heldPointer) {\n                this.heldPointer = inEvent;\n                this.target = inEvent.target;\n                this.holdJob = setInterval(this.pulse.bind(this), this.HOLD_DELAY);\n            }\n        },\n        up: function(inEvent) {\n            if (this.heldPointer && this.heldPointer.pointerId === inEvent.pointerId) {\n                this.cancel();\n            }\n        },\n        move: function(inEvent) {\n            if (this.heldPointer && this.heldPointer.pointerId === inEvent.pointerId) {\n                var x = inEvent.clientX - this.heldPointer.clientX;\n                var y = inEvent.clientY - this.heldPointer.clientY;\n                if ((x * x + y * y) > this.WIGGLE_THRESHOLD) {\n                    this.cancel();\n                }\n            }\n        },\n        fireHold: function(inType, inHoldTime) {\n            var p = {\n                bubbles: true,\n                cancelable: true,\n                pointerType: this.heldPointer.pointerType,\n                pointerId: this.heldPointer.pointerId,\n                x: this.heldPointer.clientX,\n                y: this.heldPointer.clientY,\n                _source: 'hold'\n            };\n            if (inHoldTime) {\n                p.holdTime = inHoldTime;\n            }\n            var e = eventFactory.makeGestureEvent(inType, p);\n            this.target.dispatchEvent(e);\n        }\n    };\n    dispatcher.registerGesture('hold', hold);\n})(exports);\n\n/**\n * This event is fired when a pointer quickly goes down and up, and is used to\n * denote activation.\n *\n * Any gesture event can prevent the tap event from being created by calling\n * `event.preventTap`.\n *\n * Any pointer event can prevent the tap by setting the `tapPrevented` property\n * on itself.\n *\n * @module PointerGestures\n * @submodule Events\n * @class tap\n */\n/**\n * X axis position of the tap.\n * @property x\n * @type Number\n */\n/**\n * Y axis position of the tap.\n * @property y\n * @type Number\n */\n/**\n * Type of the pointer that made the tap.\n * @property pointerType\n * @type String\n */\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var tap = {\n        events: [\n            'down',\n            'up'\n        ],\n        exposes: [\n            'tap'\n        ],\n        down: function(inEvent) {\n            if (inEvent.isPrimary && !inEvent.tapPrevented) {\n                pointermap.set(inEvent.pointerId, {\n                    target: inEvent.target,\n                    buttons: inEvent.buttons,\n                    x: inEvent.clientX,\n                    y: inEvent.clientY\n                });\n            }\n        },\n        shouldTap: function(e, downState) {\n            var tap = true;\n            if (e.pointerType === 'mouse') {\n                // only allow left click to tap for mouse\n                tap = (e.buttons ^ 1) && (downState.buttons & 1);\n            }\n            return tap && !e.tapPrevented;\n        },\n        up: function(inEvent) {\n            var start = pointermap.get(inEvent.pointerId);\n            if (start && this.shouldTap(inEvent, start)) {\n                // up.relatedTarget is target currently under finger\n                var t = scope.targetFinding.LCA(start.target, inEvent.relatedTarget);\n                if (t) {\n                    var e = eventFactory.makeGestureEvent('tap', {\n                        bubbles: true,\n                        cancelable: true,\n                        x: inEvent.clientX,\n                        y: inEvent.clientY,\n                        detail: inEvent.detail,\n                        pointerType: inEvent.pointerType,\n                        pointerId: inEvent.pointerId,\n                        altKey: inEvent.altKey,\n                        ctrlKey: inEvent.ctrlKey,\n                        metaKey: inEvent.metaKey,\n                        shiftKey: inEvent.shiftKey,\n                        _source: 'tap'\n                    });\n                    t.dispatchEvent(e);\n                }\n            }\n            pointermap.delete(inEvent.pointerId);\n        }\n    };\n    // patch eventFactory to remove id from tap's pointermap for preventTap calls\n    eventFactory.preventTap = function(e) {\n        return function() {\n            e.tapPrevented = true;\n            pointermap.delete(e.pointerId);\n        };\n    };\n    dispatcher.registerGesture('tap', tap);\n})(exports);\n\n/*\n * Basic strategy: find the farthest apart points, use as diameter of circle\n * react to size change and rotation of the chord\n */\n\n/**\n * @module pointer-gestures\n * @submodule Events\n * @class pinch\n */\n/**\n * Scale of the pinch zoom gesture\n * @property scale\n * @type Number\n */\n/**\n * Center X position of pointers causing pinch\n * @property centerX\n * @type Number\n */\n/**\n * Center Y position of pointers causing pinch\n * @property centerY\n * @type Number\n */\n\n/**\n * @module pointer-gestures\n * @submodule Events\n * @class rotate\n */\n/**\n * Angle (in degrees) of rotation. Measured from starting positions of pointers.\n * @property angle\n * @type Number\n */\n/**\n * Center X position of pointers causing rotation\n * @property centerX\n * @type Number\n */\n/**\n * Center Y position of pointers causing rotation\n * @property centerY\n * @type Number\n */\n(function(scope) {\n    var dispatcher = scope.dispatcher;\n    var eventFactory = scope.eventFactory;\n    var pointermap = new scope.PointerMap();\n    var RAD_TO_DEG = 180 / Math.PI;\n    var pinch = {\n        events: [\n            'down',\n            'up',\n            'move',\n            'cancel'\n        ],\n        exposes: [\n            'pinchstart',\n            'pinch',\n            'pinchend',\n            'rotate'\n        ],\n        defaultActions: {\n            'pinch': 'none',\n            'rotate': 'none'\n        },\n        reference: {},\n        down: function(inEvent) {\n            pointermap.set(inEvent.pointerId, inEvent);\n            if (pointermap.pointers() == 2) {\n                var points = this.calcChord();\n                var angle = this.calcAngle(points);\n                this.reference = {\n                    angle: angle,\n                    diameter: points.diameter,\n                    target: scope.targetFinding.LCA(points.a.target, points.b.target)\n                };\n\n                this.firePinch('pinchstart', points.diameter, points);\n            }\n        },\n        up: function(inEvent) {\n            var p = pointermap.get(inEvent.pointerId);\n            var num = pointermap.pointers();\n            if (p) {\n                if (num === 2) {\n                    // fire 'pinchend' before deleting pointer\n                    var points = this.calcChord();\n                    this.firePinch('pinchend', points.diameter, points);\n                }\n                pointermap.delete(inEvent.pointerId);\n            }\n        },\n        move: function(inEvent) {\n            if (pointermap.has(inEvent.pointerId)) {\n                pointermap.set(inEvent.pointerId, inEvent);\n                if (pointermap.pointers() > 1) {\n                    this.calcPinchRotate();\n                }\n            }\n        },\n        cancel: function(inEvent) {\n            this.up(inEvent);\n        },\n        firePinch: function(type, diameter, points) {\n            var zoom = diameter / this.reference.diameter;\n            var e = eventFactory.makeGestureEvent(type, {\n                bubbles: true,\n                cancelable: true,\n                scale: zoom,\n                centerX: points.center.x,\n                centerY: points.center.y,\n                _source: 'pinch'\n            });\n            this.reference.target.dispatchEvent(e);\n        },\n        fireRotate: function(angle, points) {\n            var diff = Math.round((angle - this.reference.angle) % 360);\n            var e = eventFactory.makeGestureEvent('rotate', {\n                bubbles: true,\n                cancelable: true,\n                angle: diff,\n                centerX: points.center.x,\n                centerY: points.center.y,\n                _source: 'pinch'\n            });\n            this.reference.target.dispatchEvent(e);\n        },\n        calcPinchRotate: function() {\n            var points = this.calcChord();\n            var diameter = points.diameter;\n            var angle = this.calcAngle(points);\n            if (diameter != this.reference.diameter) {\n                this.firePinch('pinch', diameter, points);\n            }\n            if (angle != this.reference.angle) {\n                this.fireRotate(angle, points);\n            }\n        },\n        calcChord: function() {\n            var pointers = [];\n            pointermap.forEach(function(p) {\n                pointers.push(p);\n            });\n            var dist = 0;\n            // start with at least two pointers\n            var points = {\n                a: pointers[0],\n                b: pointers[1]\n            };\n            var x, y, d;\n            for (var i = 0; i < pointers.length; i++) {\n                var a = pointers[i];\n                for (var j = i + 1; j < pointers.length; j++) {\n                    var b = pointers[j];\n                    x = Math.abs(a.clientX - b.clientX);\n                    y = Math.abs(a.clientY - b.clientY);\n                    d = x + y;\n                    if (d > dist) {\n                        dist = d;\n                        points = {\n                            a: a,\n                            b: b\n                        };\n                    }\n                }\n            }\n            x = Math.abs(points.a.clientX + points.b.clientX) / 2;\n            y = Math.abs(points.a.clientY + points.b.clientY) / 2;\n            points.center = {\n                x: x,\n                y: y\n            };\n            points.diameter = dist;\n            return points;\n        },\n        calcAngle: function(points) {\n            var x = points.a.clientX - points.b.clientX;\n            var y = points.a.clientY - points.b.clientY;\n            return (360 + Math.atan2(y, x) * RAD_TO_DEG) % 360;\n        }\n    };\n    dispatcher.registerGesture('pinch', pinch);\n})(exports);","// list-dragon node module\n// https://github.com/openfin/list-dragon\n\n/* eslint-env node, browser */\n\n'use strict';\n\nvar cssInjector = require('css-injector');\nvar format = require('templex');\n\nvar REVERT_TO_STYLESHEET_VALUE = null;  // null removes the style\n\nvar transform, timer, scrollVelocity, cssListDragon;\n\n/* inject:css */\ncssListDragon = 'div.dragon-list{position:relative;background-color:#fff}div.dragon-list>div,div.dragon-list>ul{position:absolute;left:0;right:0}div.dragon-list>div{text-align:center;background-color:#00796b;color:#fff;box-shadow:0 3px 6px rgba(0,0,0,.16),0 3px 6px rgba(0,0,0,.23);overflow:hidden;white-space:nowrap}div.dragon-list>ul{overflow-y:auto;bottom:0;margin:0;padding:0;box-shadow:0 1px 3px rgba(0,0,0,.12),0 1px 2px rgba(0,0,0,.24)}div.dragon-list>ul>li,li.dragon-pop{white-space:nowrap;list-style-type:none;border:0 solid #f4f4f4;border-bottom:1px solid #e0e0e0;cursor:move;transition:border-top-width .2s}div.dragon-list>ul>li:last-child{height:0;border-bottom:none}li.dragon-pop{position:fixed;background-color:#fff;border:1px solid #e0e0e0;left:0;top:0;overflow-x:hidden;box-shadow:rgba(0,0,0,.188235) 0 10px 20px,rgba(0,0,0,.227451) 0 6px 6px}';\n/* endinject */\n\n/**\n * @constructor ListDragon\n *\n * @desc This object services a set of item lists that allow dragging and dropping items within and between lists in a set.\n *\n * Two strategies are supported:\n *\n * 1. Supply your own HTML markup and let the API build the item models for you.\n *    To use this strategy, script your HTML and provide one of these:\n *    * an array of all the list item (`<li>`) tags\n *    * a CSS selector that points to all the list item tags\n * 2. Supply your own item models and let the API build the HTML markup for you.\n *    To use this strategy, provide an array of model lists.\n *\n * The new ListDragon object's `modelLists` property references the array of model lists the API constructed for you in strategy #1 or the array of model lists you supplied for strategy #2.\n *\n * After the user performs a successful drag-and-drop operation, the position of the model references within the `modelLists` array is rearranged. (The models themselves are the original objects as supplied in the model lists; they are not rebuilt or altered in any way. Just the references to them are moved around.)\n *\n * @param {string|Element[]|modelListType[]} selectorOrModelLists - You must supply one of the items in **bold** below:\n *\n * 1. _For strategy #1 above (API creates models from supplied elements):_ All the list item (`<li>`) DOM elements of all the lists you want the new object to manage, as either:\n *    1. **A CSS selector;** _or_\n *    2. **An array of DOM elements**\n * 2. _For strategy #2 above (API creates elements from supplied models):_ **An array of model lists,** each of which is in one of the following forms:\n *    1. An array of item models (with various option properties hanging off of it); _and/or_\n *    2. A {@link modelListType} object with those same various option properties including the required `models` property containing that same array of item models.\n *\n * In either case (2.1 or 2.2), each element of such arrays of item models may take the form of:\n * * A string primitive; _or_\n * * A {@link itemModelType} object with a various option properties including the required `label` property containing a string primitive.\n *\n * Regarding these string primitives, each is either:\n * * A string to be displayed in the list item; _or_\n * * A format string with other property values merged in, the result of which is to be displayed in the list item.\n *\n * @param {object} [options={}] - You may supply \"global\" template variables here, representing the \"outer scope,\" after first searching each model and then each model list.\n * @param {undefined|null|Element|string} [cssStylesheetReferenceElement] - Determines where to insert the stylesheet. (This is the only formal option.) Passed to css-injector, the overloads are (from css-injector docs):\n * * `undefined` type (or omitted): injects stylesheet at top of `<head>...</head>` element\n * * `null` value: injects stylesheet at bottom of `<head>...</head>` element\n * * `Element` type: injects stylesheet immediately before given element, wherever it is found.\n * * `string` type: injects stylesheet immediately before given first element found that matches the given css selector.\n */\nfunction ListDragon(selectorOrModelLists, options) {\n\n    if (!(this instanceof ListDragon)) {\n        throw error('Not called with \"new\" keyword.');\n    }\n\n    var self = this, modelLists, items;\n\n    options = options || {};\n\n    if (typeof selectorOrModelLists === 'string') {\n        items = toArray(document.querySelectorAll(selectorOrModelLists));\n        modelLists = createModelListsFromListElements(items);\n    } else if (selectorOrModelLists[0] instanceof Element) {\n        items = toArray(selectorOrModelLists);\n        modelLists = createModelListsFromListElements(items);\n    } else {\n        // param is array of model lists\n        // build new <ul> element(s) for each list and put in `.modelLists`;\n        // fill `.items` array with <li> elements from these new <ul> elements\n        items = [];\n        modelLists = createListElementsFromModelLists(selectorOrModelLists, options);\n        modelLists.forEach(function (list) {\n            items = items.concat(toArray(list.element.querySelectorAll('li')));\n        });\n    }\n\n    // grab wheel events and don't let 'em bubble\n    modelLists.forEach(function (modelList) {\n        modelList.element.addEventListener('wheel', captureEvent);\n    });\n\n    items.forEach(function (itemElement, index) {\n        var item = (itemElement !== itemElement.parentElement.lastElementChild)\n            ? self.addEvt(itemElement, 'mousedown', itemElement, true)\n            : { element: itemElement };\n\n        /* `item.model` not currently needed so commented out here.\n         * (Originally used for rebuilding modelLists for final\n         * reporting, modelLists are now spliced on every successful\n         * drag-and-drop operation so they're always up to date.)\n\n         var origin = this.itemCoordinates(itemElement);\n         item.model = this.modelLists[origin.list].models[origin.item];\n\n         */\n\n        items[index] = item;\n    });\n\n    transform = 'transform' in items[0].element.style\n        ? 'transform' // Chrome 45 and Firefox 40\n        : '-webkit-transform'; // Safari 8\n\n    // set up the new object\n    this.modelLists = modelLists;\n    this.items = items;\n    this.bindings = {};\n    this.callback = {};\n\n    cssInjector(cssListDragon, 'list-dragon-base', options.cssStylesheetReferenceElement);\n\n}\n\nListDragon.prototype = {\n\n    addEvt: function (target, type, listener, doNotBind) {\n        var binding = {\n            handler: handlers[type].bind(target, this),\n            element: listener || window\n        };\n\n        if (!doNotBind) {\n            this.bindings[type] = binding;\n        }\n\n        binding.element.addEventListener(type, binding.handler);\n\n        return binding;\n    },\n\n    removeEvt: function (type) {\n        var binding = this.bindings[type];\n        delete this.bindings[type];\n        binding.element.removeEventListener(type, binding.handler);\n    },\n\n    removeAllEventListeners: function () {\n        // remove drag & drop events (mousemove, mouseup, and transitionend)\n        for (var type in this.bindings) {\n            var binding = this.bindings[type];\n            binding.element.removeEventListener(type, binding.handler);\n        }\n        // remove the mousedown events from all list items\n        this.items.forEach(function (item) {\n            if (item.handler) {\n                item.element.removeEventListener('mousedown', item.handler);\n            }\n        });\n        // wheel events on the list elements\n        this.modelLists.forEach(function (modelList) {\n            modelList.element.removeEventListener('wheel', captureEvent);\n        });\n    },\n\n    pointInListRects: function (point) {\n        return this.modelLists.find(function (modelList) {\n            var rect = modelList.element.getBoundingClientRect();\n\n            rect = {\n                left:   window.scrollX + rect.left,\n                top:    window.scrollY + rect.top,\n                right:  window.scrollX + rect.right,\n                bottom: window.scrollY + rect.bottom,\n                width:  rect.width,\n                height: rect.height\n            };\n\n            modelList.rect = rect;\n\n            if (pointInRect(point, rect)) {\n                modelList.rect = rect;\n                return true; // found\n            } else {\n                return false;\n            }\n        });\n    },\n\n    pointInItemRects: function (point, except1, except2) {\n        return this.items.find(function (item) {\n            var element = item.element;\n            return (\n                element !== except1 &&\n                element !== except2 &&\n                pointInRect(point, item.rect)\n            );\n        });\n    },\n\n    // get positions of all list items in page coords (normalized for window and list scrolling)\n    getAllItemBoundingRects: function () {\n        var modelLists = this.modelLists, height;\n        this.items.forEach(function (item) {\n            var itemElement = item.element,\n                listElement = itemElement.parentElement,\n                list = modelLists.find(function (list) { return list.element === listElement; });\n\n            if (\n                // omitted: default to true\n                list.isDropTarget === undefined ||\n\n                // function: use return value\n                typeof list.isDropTarget === 'function' && list.isDropTarget() ||\n\n                // otherwise: use truthiness of given value\n                list.isDropTarget\n            ) {\n                var rect = itemElement.getBoundingClientRect(),\n                    bottom = rect.bottom;\n\n                if (itemElement === listElement.lastElementChild) {\n                    bottom = listElement.getBoundingClientRect().bottom;\n                    if (bottom < rect.top) {\n                        bottom = rect.top + (height || 50);\n                    }\n                } else {\n                    height = rect.height;\n                }\n\n                rect = {\n                    left:   window.scrollX + rect.left,\n                    right:  window.scrollX + rect.right,\n                    top:    window.scrollY + rect.top    + listElement.scrollTop,\n                    bottom: window.scrollY + bottom + listElement.scrollTop\n                };\n\n                item.rect = rect;\n            }\n        });\n    },\n\n    reinsert: function (target) {\n        var style = target.style;\n        style.width = style[transform] = style.transition = REVERT_TO_STYLESHEET_VALUE;\n\n        target.classList.remove('dragon-pop');\n\n        this.drop.style.transitionDuration = '0s';\n        this.drop.style.borderTopWidth = REVERT_TO_STYLESHEET_VALUE;\n        this.drop.parentElement.insertBefore(target, this.drop);\n\n        delete this.drop;\n    },\n\n    // return an object { item: <item index within list>, list: <list index within list of lists> }\n    itemCoordinates: function (item) {\n        var listElement = item.parentElement,\n            coords = { item: 0 };\n\n        while ((item = item.previousElementSibling)) {\n            ++coords.item;\n        }\n\n        this.modelLists.find(function (list, index) {\n            coords.list = index;\n            return list.element === listElement; // stop when we find the one we belong to\n        });\n\n        return coords;\n    }\n\n};\n\nvar handlers = {\n    mousedown: function (dragon, evt) {\n\n        evt.stopPropagation();\n        evt.preventDefault();  //prevents user selection of rendered nodes during drag\n\n        if (dragon.drop) {\n            return;\n        }\n\n        var rect = this.getBoundingClientRect();\n\n        dragon.rect = rect = {\n            left:   Math.round(rect.left - 1),\n            top:    Math.round(rect.top - 1),\n            right:  Math.round(rect.right),\n            bottom: Math.round(rect.bottom),\n            width:  Math.round(rect.width),\n            height: Math.round(rect.height)\n        };\n\n        dragon.pin = {\n            x: window.scrollX + evt.clientX,\n            y: window.scrollY + evt.clientY\n        };\n\n        dragon.origin = dragon.itemCoordinates(this);\n\n        if (dragon.callback.grabbed) {\n            dragon.callback.grabbed.call(this, dragon);\n        }\n\n        dragon.getAllItemBoundingRects();\n\n        dragon.drop = this.nextElementSibling;\n        dragon.drop.style.transitionDuration = '0s';\n        dragon.drop.style.borderTopWidth = rect.height + 'px';\n\n        this.style.width = rect.width + 'px';\n        this.style.transitionDuration = '0s';\n        this.style[transform] = translate(\n            rect.left - window.scrollX,\n            rect.top  - window.scrollY\n        );\n        this.classList.add('dragon-pop');\n        this.style.zIndex = window.getComputedStyle(dragon.modelLists[0].container.parentElement).zIndex;\n\n        if (!dragon.container) {\n            // walk back to closest shadow root OR body tag OR root tag\n            var container = this;\n            while (container.parentNode) {\n                container = container.parentNode;\n                if (\n                    typeof ShadowRoot !== 'undefined' && container instanceof ShadowRoot ||\n                    container.tagName === 'BODY'\n                ){\n                    break;\n                }\n            }\n            dragon.container = container;\n        }\n\n        dragon.container.appendChild(this);\n\n        rect.left   += window.scrollX;\n        rect.top    += window.scrollY;\n        rect.right  += window.scrollX;\n        rect.bottom += window.scrollY;\n\n        dragon.addEvt(this, 'mousemove');\n        dragon.addEvt(this, 'mouseup');\n    },\n\n    mousemove: function (dragon, evt) {\n        dragon.drop.style.transition = REVERT_TO_STYLESHEET_VALUE;\n\n        var hoverList = dragon.pointInListRects({ x: evt.clientX, y: evt.clientY }) || dragon.mostRecentHoverList;\n\n        if (hoverList) {\n            var dx = evt.clientX - dragon.pin.x,\n                dy = evt.clientY - dragon.pin.y;\n\n            dragon.mostRecentHoverList = hoverList;\n\n            var maxScrollY = hoverList.element.scrollHeight - hoverList.rect.height,\n                y = evt.clientY + window.scrollY,\n                magnitude;\n\n            if (maxScrollY > 0) {\n                // list is scrollable (is taller than rect)\n                if (hoverList.element.scrollTop > 0 && (magnitude = y - (hoverList.rect.top + 5)) < 0) {\n                    // mouse near or above top and list is not scrolled to top yet\n                    resetAutoScrollTimer(magnitude, 0, hoverList.element);\n                } else if (hoverList.element.scrollTop < maxScrollY && (magnitude = y - (hoverList.rect.bottom - 1 - 5)) > 0) {\n                    // mouse near or below bottom and list not scrolled to bottom yet\n                    resetAutoScrollTimer(magnitude, maxScrollY, hoverList.element);\n                } else {\n                    // mouse inside\n                    resetAutoScrollTimer();\n                }\n            }\n\n            var other = dragon.pointInItemRects({\n                x: evt.clientX,\n                y: dragon.rect.bottom + window.scrollY + dy + hoverList.element.scrollTop\n            }, this, dragon.drop);\n\n            this.style[transform] = translate(\n                dragon.rect.left - window.scrollX + dx,\n                dragon.rect.top - window.scrollY + dy\n            );\n\n            if (other) {\n                var element = other.element;\n                element.style.transition = REVERT_TO_STYLESHEET_VALUE;\n                element.style.borderTopWidth = dragon.drop.style.borderTopWidth;\n                dragon.drop.style.borderTopWidth = null;\n                dragon.drop = element;\n            }\n        }\n    },\n\n    mouseup: function (dragon, evt) {\n        resetAutoScrollTimer();\n        dragon.removeEvt('mousemove');\n        dragon.removeEvt('mouseup');\n\n        evt.stopPropagation();\n\n        var newRect = this.getBoundingClientRect();\n\n        if (\n            window.scrollX + newRect.left === dragon.rect.left &&\n            window.scrollY + newRect.top === dragon.rect.top\n        ) {\n            dragon.reinsert(this);\n        } else {\n            var dropRect = dragon.drop.getBoundingClientRect();\n\n            dragon.addEvt(this, 'transitionend', this);\n            this.style.transitionDuration = REVERT_TO_STYLESHEET_VALUE; //reverts to 200ms\n            this.style.transitionProperty = transform;\n            this.style[transform] = translate(\n                dropRect.left - window.scrollX,\n                dropRect.top - window.scrollY\n            );\n        }\n    },\n\n    transitionend: function (dragon, evt) {\n        if (evt.propertyName === transform) {\n            dragon.removeEvt('transitionend');\n            dragon.reinsert(this);\n\n            this.style.transitionProperty = REVERT_TO_STYLESHEET_VALUE; //reverts to border-top-width\n\n            var model = dragon.modelLists[dragon.origin.list].splice(dragon.origin.item, 1)[0];\n            var destination = dragon.itemCoordinates(this);\n            dragon.modelLists[destination.list].splice(destination.item, 0, model);\n\n            if (dragon.callback.dropped) {\n                dragon.callback.dropped.call(this, dragon);\n            }\n        }\n    }\n};\n\nfunction resetAutoScrollTimer(magnitude, limit, element) {\n    if (!magnitude) {\n        clearInterval(timer);\n        scrollVelocity = 0;\n    } else {\n        var changeDirection =\n            scrollVelocity  <  0 && magnitude  >= 0 ||\n            scrollVelocity === 0 && magnitude !== 0 ||\n            scrollVelocity  >  0 && magnitude  <= 0;\n        scrollVelocity = magnitude > 0 ? Math.min(50, magnitude) : Math.max(-50, magnitude);\n        if (changeDirection) {\n            clearInterval(timer);\n            timer = setInterval(function (limit) {\n                var scrollTop = element.scrollTop + scrollVelocity;\n                if (scrollVelocity < 0 && scrollTop < limit || scrollVelocity > 0 && scrollTop > limit) {\n                    element.scrollTop = limit;\n                    clearInterval(timer);\n                } else {\n                    element.scrollTop = scrollTop;\n                }\n            }, 125);\n        }\n    }\n}\n\nfunction toArray(arrayLikeObject) {\n    return Array.prototype.slice.call(arrayLikeObject);\n}\n\nfunction pointInRect(point, rect) {\n    return rect.top <= point.y && point.y <= rect.bottom\n        && rect.left <= point.x && point.x <= rect.right;\n}\n\nfunction translate(left, top) {\n    return 'translate('\n        + Math.floor(left + window.scrollX) + 'px,'\n        + Math.floor(top + window.scrollY) + 'px)';\n}\n\nfunction htmlEncode(string) {\n    var textNode = document.createTextNode(string);\n\n    return document\n        .createElement('a')\n        .appendChild(textNode)\n        .parentNode\n        .innerHTML;\n}\n\n/**\n * Creates `<ul>...</ul>` elements and inserts them into an `element` property on each model.\n * @param {object} modelLists\n * @returns `modelLists`\n */\nfunction createListElementsFromModelLists(modelLists, options) {\n    var templateLabel = options.label || '{label}';\n\n    modelLists.forEach(function (modelList, listIndex) {\n        var listLabel = modelList.label || templateLabel,\n            listHtmlEncode = modelList.htmlEncode !== undefined && modelList.htmlEncode || options.htmlEncode,\n            container = document.createElement('div'),\n            listElement = document.createElement('ul');\n\n        if (modelList.models) {\n            Object.keys(modelList).forEach(function (key) {\n                if (key !== 'models') {\n                    modelList.models[key] = modelList[key];\n                }\n            });\n            modelLists[listIndex] = modelList = modelList.models;\n        } else if (modelList instanceof Array) {\n            modelList.models = modelList; // point to self\n        } else {\n            throw error('List [{1}] not an array of models (with or without additional properties) OR ' +\n                'an object (with a `models` property containing an array of models).', listIndex);\n        }\n\n        modelList.forEach(function (model) {\n            var modelLabel = model.label || listLabel,\n                modelHtmlEncode = model.htmlEncode !== undefined && model.htmlEncode || listHtmlEncode,\n                modelObject = typeof model === 'object' ? model : { label: model},\n                label = format.call([modelObject, modelList, options], modelLabel),\n                itemElement = document.createElement('li');\n\n            itemElement.innerHTML = modelHtmlEncode ? htmlEncode(label) : label;\n\n            listElement.appendChild(itemElement);\n        });\n\n        // append the final \"fencepost\" item -- drop target at bottom of list after all items\n        var itemElement = document.createElement('li');\n        itemElement.innerHTML = '&nbsp;';\n        listElement.appendChild(itemElement);\n\n        // append header to container\n        if (modelList.title) {\n            var header = document.createElement('div');\n            header.innerHTML = listHtmlEncode ? htmlEncode(modelList.title) : modelList.title;\n            container.appendChild(header);\n        }\n\n        container.appendChild(listElement);\n        container.className = modelList.cssClassNames || options.cssClassNames || 'dragon-list';\n        modelList.element = listElement;\n        modelList.container = container;\n    });\n\n    return modelLists;\n}\n\n/**\n * Create a `.modelLists` array with these <li> elements' parent <ul> elements\n * @param {Element[]} listItemElements\n * @returns {Array}\n */\nfunction createModelListsFromListElements(listItemElements) {\n    var modelLists = [];\n\n    listItemElements.forEach(function (itemElement) {\n        var listElement = itemElement.parentElement,\n            container = listElement.parentElement,\n            models = [];\n        if (!modelLists.find(function (list) { return list.element === listElement; })) {\n            toArray(listElement.querySelectorAll('li')).forEach(function (itemElement) {\n                if (itemElement !== listElement.lastElementChild) {\n                    models.push(itemElement.innerHTML);\n                }\n            });\n            models.element = listElement;\n            models.container = container;\n            modelLists.push(models);\n        }\n    });\n\n    return modelLists;\n}\n\nfunction captureEvent(evt) {\n    evt.stopPropagation();\n}\n\nfunction error() {\n    return 'list-dragon: ' + format.apply(this, Array.prototype.slice.call(arguments));\n}\n\n// this interface consists solely of the prototypal object constructor\nmodule.exports = ListDragon;\n",";(function () { // closure for web browsers\n\nif (typeof module === 'object' && module.exports) {\n  module.exports = LRUCache\n} else {\n  // just set the global for non-node platforms.\n  this.LRUCache = LRUCache\n}\n\nfunction hOP (obj, key) {\n  return Object.prototype.hasOwnProperty.call(obj, key)\n}\n\nfunction naiveLength () { return 1 }\n\nvar didTypeWarning = false\nfunction typeCheckKey(key) {\n  if (!didTypeWarning && typeof key !== 'string' && typeof key !== 'number') {\n    didTypeWarning = true\n    console.error(new TypeError(\"LRU: key must be a string or number. Almost certainly a bug! \" + typeof key).stack)\n  }\n}\n\nfunction LRUCache (options) {\n  if (!(this instanceof LRUCache))\n    return new LRUCache(options)\n\n  if (typeof options === 'number')\n    options = { max: options }\n\n  if (!options)\n    options = {}\n\n  this._max = options.max\n  // Kind of weird to have a default max of Infinity, but oh well.\n  if (!this._max || !(typeof this._max === \"number\") || this._max <= 0 )\n    this._max = Infinity\n\n  this._lengthCalculator = options.length || naiveLength\n  if (typeof this._lengthCalculator !== \"function\")\n    this._lengthCalculator = naiveLength\n\n  this._allowStale = options.stale || false\n  this._maxAge = options.maxAge || null\n  this._dispose = options.dispose\n  this.reset()\n}\n\n// resize the cache when the max changes.\nObject.defineProperty(LRUCache.prototype, \"max\",\n  { set : function (mL) {\n      if (!mL || !(typeof mL === \"number\") || mL <= 0 ) mL = Infinity\n      this._max = mL\n      if (this._length > this._max) trim(this)\n    }\n  , get : function () { return this._max }\n  , enumerable : true\n  })\n\n// resize the cache when the lengthCalculator changes.\nObject.defineProperty(LRUCache.prototype, \"lengthCalculator\",\n  { set : function (lC) {\n      if (typeof lC !== \"function\") {\n        this._lengthCalculator = naiveLength\n        this._length = this._itemCount\n        for (var key in this._cache) {\n          this._cache[key].length = 1\n        }\n      } else {\n        this._lengthCalculator = lC\n        this._length = 0\n        for (var key in this._cache) {\n          this._cache[key].length = this._lengthCalculator(this._cache[key].value)\n          this._length += this._cache[key].length\n        }\n      }\n\n      if (this._length > this._max) trim(this)\n    }\n  , get : function () { return this._lengthCalculator }\n  , enumerable : true\n  })\n\nObject.defineProperty(LRUCache.prototype, \"length\",\n  { get : function () { return this._length }\n  , enumerable : true\n  })\n\n\nObject.defineProperty(LRUCache.prototype, \"itemCount\",\n  { get : function () { return this._itemCount }\n  , enumerable : true\n  })\n\nLRUCache.prototype.forEach = function (fn, thisp) {\n  thisp = thisp || this\n  var i = 0\n  var itemCount = this._itemCount\n\n  for (var k = this._mru - 1; k >= 0 && i < itemCount; k--) if (this._lruList[k]) {\n    i++\n    var hit = this._lruList[k]\n    if (isStale(this, hit)) {\n      del(this, hit)\n      if (!this._allowStale) hit = undefined\n    }\n    if (hit) {\n      fn.call(thisp, hit.value, hit.key, this)\n    }\n  }\n}\n\nLRUCache.prototype.keys = function () {\n  var keys = new Array(this._itemCount)\n  var i = 0\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    keys[i++] = hit.key\n  }\n  return keys\n}\n\nLRUCache.prototype.values = function () {\n  var values = new Array(this._itemCount)\n  var i = 0\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    values[i++] = hit.value\n  }\n  return values\n}\n\nLRUCache.prototype.reset = function () {\n  if (this._dispose && this._cache) {\n    for (var k in this._cache) {\n      this._dispose(k, this._cache[k].value)\n    }\n  }\n\n  this._cache = Object.create(null) // hash of items by key\n  this._lruList = Object.create(null) // list of items in order of use recency\n  this._mru = 0 // most recently used\n  this._lru = 0 // least recently used\n  this._length = 0 // number of items in the list\n  this._itemCount = 0\n}\n\nLRUCache.prototype.dump = function () {\n  var arr = []\n  var i = 0\n\n  for (var k = this._mru - 1; k >= 0 && i < this._itemCount; k--) if (this._lruList[k]) {\n    var hit = this._lruList[k]\n    if (!isStale(this, hit)) {\n      //Do not store staled hits\n      ++i\n      arr.push({\n        k: hit.key,\n        v: hit.value,\n        e: hit.now + (hit.maxAge || 0)\n      });\n    }\n  }\n  //arr has the most read first\n  return arr\n}\n\nLRUCache.prototype.dumpLru = function () {\n  return this._lruList\n}\n\nLRUCache.prototype.set = function (key, value, maxAge) {\n  maxAge = maxAge || this._maxAge\n  typeCheckKey(key)\n\n  var now = maxAge ? Date.now() : 0\n  var len = this._lengthCalculator(value)\n\n  if (hOP(this._cache, key)) {\n    if (len > this._max) {\n      del(this, this._cache[key])\n      return false\n    }\n    // dispose of the old one before overwriting\n    if (this._dispose)\n      this._dispose(key, this._cache[key].value)\n\n    this._cache[key].now = now\n    this._cache[key].maxAge = maxAge\n    this._cache[key].value = value\n    this._length += (len - this._cache[key].length)\n    this._cache[key].length = len\n    this.get(key)\n\n    if (this._length > this._max)\n      trim(this)\n\n    return true\n  }\n\n  var hit = new Entry(key, value, this._mru++, len, now, maxAge)\n\n  // oversized objects fall out of cache automatically.\n  if (hit.length > this._max) {\n    if (this._dispose) this._dispose(key, value)\n    return false\n  }\n\n  this._length += hit.length\n  this._lruList[hit.lu] = this._cache[key] = hit\n  this._itemCount ++\n\n  if (this._length > this._max)\n    trim(this)\n\n  return true\n}\n\nLRUCache.prototype.has = function (key) {\n  typeCheckKey(key)\n  if (!hOP(this._cache, key)) return false\n  var hit = this._cache[key]\n  if (isStale(this, hit)) {\n    return false\n  }\n  return true\n}\n\nLRUCache.prototype.get = function (key) {\n  typeCheckKey(key)\n  return get(this, key, true)\n}\n\nLRUCache.prototype.peek = function (key) {\n  typeCheckKey(key)\n  return get(this, key, false)\n}\n\nLRUCache.prototype.pop = function () {\n  var hit = this._lruList[this._lru]\n  del(this, hit)\n  return hit || null\n}\n\nLRUCache.prototype.del = function (key) {\n  typeCheckKey(key)\n  del(this, this._cache[key])\n}\n\nLRUCache.prototype.load = function (arr) {\n  //reset the cache\n  this.reset();\n\n  var now = Date.now()\n  //A previous serialized cache has the most recent items first\n  for (var l = arr.length - 1; l >= 0; l-- ) {\n    var hit = arr[l]\n    typeCheckKey(hit.k)\n    var expiresAt = hit.e || 0\n    if (expiresAt === 0) {\n      //the item was created without expiration in a non aged cache\n      this.set(hit.k, hit.v)\n    } else {\n      var maxAge = expiresAt - now\n      //dont add already expired items\n      if (maxAge > 0) this.set(hit.k, hit.v, maxAge)\n    }\n  }\n}\n\nfunction get (self, key, doUse) {\n  typeCheckKey(key)\n  var hit = self._cache[key]\n  if (hit) {\n    if (isStale(self, hit)) {\n      del(self, hit)\n      if (!self._allowStale) hit = undefined\n    } else {\n      if (doUse) use(self, hit)\n    }\n    if (hit) hit = hit.value\n  }\n  return hit\n}\n\nfunction isStale(self, hit) {\n  if (!hit || (!hit.maxAge && !self._maxAge)) return false\n  var stale = false;\n  var diff = Date.now() - hit.now\n  if (hit.maxAge) {\n    stale = diff > hit.maxAge\n  } else {\n    stale = self._maxAge && (diff > self._maxAge)\n  }\n  return stale;\n}\n\nfunction use (self, hit) {\n  shiftLU(self, hit)\n  hit.lu = self._mru ++\n  self._lruList[hit.lu] = hit\n}\n\nfunction trim (self) {\n  while (self._lru < self._mru && self._length > self._max)\n    del(self, self._lruList[self._lru])\n}\n\nfunction shiftLU (self, hit) {\n  delete self._lruList[ hit.lu ]\n  while (self._lru < self._mru && !self._lruList[self._lru]) self._lru ++\n}\n\nfunction del (self, hit) {\n  if (hit) {\n    if (self._dispose) self._dispose(hit.key, hit.value)\n    self._length -= hit.length\n    self._itemCount --\n    delete self._cache[ hit.key ]\n    shiftLU(self, hit)\n  }\n}\n\n// classy, since V8 prefers predictable objects.\nfunction Entry (key, value, lu, length, now, maxAge) {\n  this.key = key\n  this.value = value\n  this.lu = lu\n  this.length = length\n  this.now = now\n  if (maxAge) this.maxAge = maxAge\n}\n\n})()\n","/*!\n * mustache.js - Logic-less {{mustache}} templates with JavaScript\n * http://github.com/janl/mustache.js\n */\n\n/*global define: false Mustache: true*/\n\n(function defineMustache (global, factory) {\n  if (typeof exports === 'object' && exports && typeof exports.nodeName !== 'string') {\n    factory(exports); // CommonJS\n  } else if (typeof define === 'function' && define.amd) {\n    define(['exports'], factory); // AMD\n  } else {\n    global.Mustache = {};\n    factory(global.Mustache); // script, wsh, asp\n  }\n}(this, function mustacheFactory (mustache) {\n\n  var objectToString = Object.prototype.toString;\n  var isArray = Array.isArray || function isArrayPolyfill (object) {\n    return objectToString.call(object) === '[object Array]';\n  };\n\n  function isFunction (object) {\n    return typeof object === 'function';\n  }\n\n  /**\n   * More correct typeof string handling array\n   * which normally returns typeof 'object'\n   */\n  function typeStr (obj) {\n    return isArray(obj) ? 'array' : typeof obj;\n  }\n\n  function escapeRegExp (string) {\n    return string.replace(/[\\-\\[\\]{}()*+?.,\\\\\\^$|#\\s]/g, '\\\\$&');\n  }\n\n  /**\n   * Null safe way of checking whether or not an object,\n   * including its prototype, has a given property\n   */\n  function hasProperty (obj, propName) {\n    return obj != null && typeof obj === 'object' && (propName in obj);\n  }\n\n  // Workaround for https://issues.apache.org/jira/browse/COUCHDB-577\n  // See https://github.com/janl/mustache.js/issues/189\n  var regExpTest = RegExp.prototype.test;\n  function testRegExp (re, string) {\n    return regExpTest.call(re, string);\n  }\n\n  var nonSpaceRe = /\\S/;\n  function isWhitespace (string) {\n    return !testRegExp(nonSpaceRe, string);\n  }\n\n  var entityMap = {\n    '&': '&amp;',\n    '<': '&lt;',\n    '>': '&gt;',\n    '\"': '&quot;',\n    \"'\": '&#39;',\n    '/': '&#x2F;',\n    '`': '&#x60;',\n    '=': '&#x3D;'\n  };\n\n  function escapeHtml (string) {\n    return String(string).replace(/[&<>\"'`=\\/]/g, function fromEntityMap (s) {\n      return entityMap[s];\n    });\n  }\n\n  var whiteRe = /\\s*/;\n  var spaceRe = /\\s+/;\n  var equalsRe = /\\s*=/;\n  var curlyRe = /\\s*\\}/;\n  var tagRe = /#|\\^|\\/|>|\\{|&|=|!/;\n\n  /**\n   * Breaks up the given `template` string into a tree of tokens. If the `tags`\n   * argument is given here it must be an array with two string values: the\n   * opening and closing tags used in the template (e.g. [ \"<%\", \"%>\" ]). Of\n   * course, the default is to use mustaches (i.e. mustache.tags).\n   *\n   * A token is an array with at least 4 elements. The first element is the\n   * mustache symbol that was used inside the tag, e.g. \"#\" or \"&\". If the tag\n   * did not contain a symbol (i.e. {{myValue}}) this element is \"name\". For\n   * all text that appears outside a symbol this element is \"text\".\n   *\n   * The second element of a token is its \"value\". For mustache tags this is\n   * whatever else was inside the tag besides the opening symbol. For text tokens\n   * this is the text itself.\n   *\n   * The third and fourth elements of the token are the start and end indices,\n   * respectively, of the token in the original template.\n   *\n   * Tokens that are the root node of a subtree contain two more elements: 1) an\n   * array of tokens in the subtree and 2) the index in the original template at\n   * which the closing tag for that section begins.\n   */\n  function parseTemplate (template, tags) {\n    if (!template)\n      return [];\n\n    var sections = [];     // Stack to hold section tokens\n    var tokens = [];       // Buffer to hold the tokens\n    var spaces = [];       // Indices of whitespace tokens on the current line\n    var hasTag = false;    // Is there a {{tag}} on the current line?\n    var nonSpace = false;  // Is there a non-space char on the current line?\n\n    // Strips all whitespace tokens array for the current line\n    // if there was a {{#tag}} on it and otherwise only space.\n    function stripSpace () {\n      if (hasTag && !nonSpace) {\n        while (spaces.length)\n          delete tokens[spaces.pop()];\n      } else {\n        spaces = [];\n      }\n\n      hasTag = false;\n      nonSpace = false;\n    }\n\n    var openingTagRe, closingTagRe, closingCurlyRe;\n    function compileTags (tagsToCompile) {\n      if (typeof tagsToCompile === 'string')\n        tagsToCompile = tagsToCompile.split(spaceRe, 2);\n\n      if (!isArray(tagsToCompile) || tagsToCompile.length !== 2)\n        throw new Error('Invalid tags: ' + tagsToCompile);\n\n      openingTagRe = new RegExp(escapeRegExp(tagsToCompile[0]) + '\\\\s*');\n      closingTagRe = new RegExp('\\\\s*' + escapeRegExp(tagsToCompile[1]));\n      closingCurlyRe = new RegExp('\\\\s*' + escapeRegExp('}' + tagsToCompile[1]));\n    }\n\n    compileTags(tags || mustache.tags);\n\n    var scanner = new Scanner(template);\n\n    var start, type, value, chr, token, openSection;\n    while (!scanner.eos()) {\n      start = scanner.pos;\n\n      // Match any text between tags.\n      value = scanner.scanUntil(openingTagRe);\n\n      if (value) {\n        for (var i = 0, valueLength = value.length; i < valueLength; ++i) {\n          chr = value.charAt(i);\n\n          if (isWhitespace(chr)) {\n            spaces.push(tokens.length);\n          } else {\n            nonSpace = true;\n          }\n\n          tokens.push([ 'text', chr, start, start + 1 ]);\n          start += 1;\n\n          // Check for whitespace on the current line.\n          if (chr === '\\n')\n            stripSpace();\n        }\n      }\n\n      // Match the opening tag.\n      if (!scanner.scan(openingTagRe))\n        break;\n\n      hasTag = true;\n\n      // Get the tag type.\n      type = scanner.scan(tagRe) || 'name';\n      scanner.scan(whiteRe);\n\n      // Get the tag value.\n      if (type === '=') {\n        value = scanner.scanUntil(equalsRe);\n        scanner.scan(equalsRe);\n        scanner.scanUntil(closingTagRe);\n      } else if (type === '{') {\n        value = scanner.scanUntil(closingCurlyRe);\n        scanner.scan(curlyRe);\n        scanner.scanUntil(closingTagRe);\n        type = '&';\n      } else {\n        value = scanner.scanUntil(closingTagRe);\n      }\n\n      // Match the closing tag.\n      if (!scanner.scan(closingTagRe))\n        throw new Error('Unclosed tag at ' + scanner.pos);\n\n      token = [ type, value, start, scanner.pos ];\n      tokens.push(token);\n\n      if (type === '#' || type === '^') {\n        sections.push(token);\n      } else if (type === '/') {\n        // Check section nesting.\n        openSection = sections.pop();\n\n        if (!openSection)\n          throw new Error('Unopened section \"' + value + '\" at ' + start);\n\n        if (openSection[1] !== value)\n          throw new Error('Unclosed section \"' + openSection[1] + '\" at ' + start);\n      } else if (type === 'name' || type === '{' || type === '&') {\n        nonSpace = true;\n      } else if (type === '=') {\n        // Set the tags for the next time around.\n        compileTags(value);\n      }\n    }\n\n    // Make sure there are no open sections when we're done.\n    openSection = sections.pop();\n\n    if (openSection)\n      throw new Error('Unclosed section \"' + openSection[1] + '\" at ' + scanner.pos);\n\n    return nestTokens(squashTokens(tokens));\n  }\n\n  /**\n   * Combines the values of consecutive text tokens in the given `tokens` array\n   * to a single token.\n   */\n  function squashTokens (tokens) {\n    var squashedTokens = [];\n\n    var token, lastToken;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      token = tokens[i];\n\n      if (token) {\n        if (token[0] === 'text' && lastToken && lastToken[0] === 'text') {\n          lastToken[1] += token[1];\n          lastToken[3] = token[3];\n        } else {\n          squashedTokens.push(token);\n          lastToken = token;\n        }\n      }\n    }\n\n    return squashedTokens;\n  }\n\n  /**\n   * Forms the given array of `tokens` into a nested tree structure where\n   * tokens that represent a section have two additional items: 1) an array of\n   * all tokens that appear in that section and 2) the index in the original\n   * template that represents the end of that section.\n   */\n  function nestTokens (tokens) {\n    var nestedTokens = [];\n    var collector = nestedTokens;\n    var sections = [];\n\n    var token, section;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      token = tokens[i];\n\n      switch (token[0]) {\n        case '#':\n        case '^':\n          collector.push(token);\n          sections.push(token);\n          collector = token[4] = [];\n          break;\n        case '/':\n          section = sections.pop();\n          section[5] = token[2];\n          collector = sections.length > 0 ? sections[sections.length - 1][4] : nestedTokens;\n          break;\n        default:\n          collector.push(token);\n      }\n    }\n\n    return nestedTokens;\n  }\n\n  /**\n   * A simple string scanner that is used by the template parser to find\n   * tokens in template strings.\n   */\n  function Scanner (string) {\n    this.string = string;\n    this.tail = string;\n    this.pos = 0;\n  }\n\n  /**\n   * Returns `true` if the tail is empty (end of string).\n   */\n  Scanner.prototype.eos = function eos () {\n    return this.tail === '';\n  };\n\n  /**\n   * Tries to match the given regular expression at the current position.\n   * Returns the matched text if it can match, the empty string otherwise.\n   */\n  Scanner.prototype.scan = function scan (re) {\n    var match = this.tail.match(re);\n\n    if (!match || match.index !== 0)\n      return '';\n\n    var string = match[0];\n\n    this.tail = this.tail.substring(string.length);\n    this.pos += string.length;\n\n    return string;\n  };\n\n  /**\n   * Skips all text until the given regular expression can be matched. Returns\n   * the skipped string, which is the entire tail if no match can be made.\n   */\n  Scanner.prototype.scanUntil = function scanUntil (re) {\n    var index = this.tail.search(re), match;\n\n    switch (index) {\n      case -1:\n        match = this.tail;\n        this.tail = '';\n        break;\n      case 0:\n        match = '';\n        break;\n      default:\n        match = this.tail.substring(0, index);\n        this.tail = this.tail.substring(index);\n    }\n\n    this.pos += match.length;\n\n    return match;\n  };\n\n  /**\n   * Represents a rendering context by wrapping a view object and\n   * maintaining a reference to the parent context.\n   */\n  function Context (view, parentContext) {\n    this.view = view;\n    this.cache = { '.': this.view };\n    this.parent = parentContext;\n  }\n\n  /**\n   * Creates a new context using the given view with this context\n   * as the parent.\n   */\n  Context.prototype.push = function push (view) {\n    return new Context(view, this);\n  };\n\n  /**\n   * Returns the value of the given name in this context, traversing\n   * up the context hierarchy if the value is absent in this context's view.\n   */\n  Context.prototype.lookup = function lookup (name) {\n    var cache = this.cache;\n\n    var value;\n    if (cache.hasOwnProperty(name)) {\n      value = cache[name];\n    } else {\n      var context = this, names, index, lookupHit = false;\n\n      while (context) {\n        if (name.indexOf('.') > 0) {\n          value = context.view;\n          names = name.split('.');\n          index = 0;\n\n          /**\n           * Using the dot notion path in `name`, we descend through the\n           * nested objects.\n           *\n           * To be certain that the lookup has been successful, we have to\n           * check if the last object in the path actually has the property\n           * we are looking for. We store the result in `lookupHit`.\n           *\n           * This is specially necessary for when the value has been set to\n           * `undefined` and we want to avoid looking up parent contexts.\n           **/\n          while (value != null && index < names.length) {\n            if (index === names.length - 1)\n              lookupHit = hasProperty(value, names[index]);\n\n            value = value[names[index++]];\n          }\n        } else {\n          value = context.view[name];\n          lookupHit = hasProperty(context.view, name);\n        }\n\n        if (lookupHit)\n          break;\n\n        context = context.parent;\n      }\n\n      cache[name] = value;\n    }\n\n    if (isFunction(value))\n      value = value.call(this.view);\n\n    return value;\n  };\n\n  /**\n   * A Writer knows how to take a stream of tokens and render them to a\n   * string, given a context. It also maintains a cache of templates to\n   * avoid the need to parse the same template twice.\n   */\n  function Writer () {\n    this.cache = {};\n  }\n\n  /**\n   * Clears all cached templates in this writer.\n   */\n  Writer.prototype.clearCache = function clearCache () {\n    this.cache = {};\n  };\n\n  /**\n   * Parses and caches the given `template` and returns the array of tokens\n   * that is generated from the parse.\n   */\n  Writer.prototype.parse = function parse (template, tags) {\n    var cache = this.cache;\n    var tokens = cache[template];\n\n    if (tokens == null)\n      tokens = cache[template] = parseTemplate(template, tags);\n\n    return tokens;\n  };\n\n  /**\n   * High-level method that is used to render the given `template` with\n   * the given `view`.\n   *\n   * The optional `partials` argument may be an object that contains the\n   * names and templates of partials that are used in the template. It may\n   * also be a function that is used to load partial templates on the fly\n   * that takes a single argument: the name of the partial.\n   */\n  Writer.prototype.render = function render (template, view, partials) {\n    var tokens = this.parse(template);\n    var context = (view instanceof Context) ? view : new Context(view);\n    return this.renderTokens(tokens, context, partials, template);\n  };\n\n  /**\n   * Low-level method that renders the given array of `tokens` using\n   * the given `context` and `partials`.\n   *\n   * Note: The `originalTemplate` is only ever used to extract the portion\n   * of the original template that was contained in a higher-order section.\n   * If the template doesn't use higher-order sections, this argument may\n   * be omitted.\n   */\n  Writer.prototype.renderTokens = function renderTokens (tokens, context, partials, originalTemplate) {\n    var buffer = '';\n\n    var token, symbol, value;\n    for (var i = 0, numTokens = tokens.length; i < numTokens; ++i) {\n      value = undefined;\n      token = tokens[i];\n      symbol = token[0];\n\n      if (symbol === '#') value = this.renderSection(token, context, partials, originalTemplate);\n      else if (symbol === '^') value = this.renderInverted(token, context, partials, originalTemplate);\n      else if (symbol === '>') value = this.renderPartial(token, context, partials, originalTemplate);\n      else if (symbol === '&') value = this.unescapedValue(token, context);\n      else if (symbol === 'name') value = this.escapedValue(token, context);\n      else if (symbol === 'text') value = this.rawValue(token);\n\n      if (value !== undefined)\n        buffer += value;\n    }\n\n    return buffer;\n  };\n\n  Writer.prototype.renderSection = function renderSection (token, context, partials, originalTemplate) {\n    var self = this;\n    var buffer = '';\n    var value = context.lookup(token[1]);\n\n    // This function is used to render an arbitrary template\n    // in the current context by higher-order sections.\n    function subRender (template) {\n      return self.render(template, context, partials);\n    }\n\n    if (!value) return;\n\n    if (isArray(value)) {\n      for (var j = 0, valueLength = value.length; j < valueLength; ++j) {\n        buffer += this.renderTokens(token[4], context.push(value[j]), partials, originalTemplate);\n      }\n    } else if (typeof value === 'object' || typeof value === 'string' || typeof value === 'number') {\n      buffer += this.renderTokens(token[4], context.push(value), partials, originalTemplate);\n    } else if (isFunction(value)) {\n      if (typeof originalTemplate !== 'string')\n        throw new Error('Cannot use higher-order sections without the original template');\n\n      // Extract the portion of the original template that the section contains.\n      value = value.call(context.view, originalTemplate.slice(token[3], token[5]), subRender);\n\n      if (value != null)\n        buffer += value;\n    } else {\n      buffer += this.renderTokens(token[4], context, partials, originalTemplate);\n    }\n    return buffer;\n  };\n\n  Writer.prototype.renderInverted = function renderInverted (token, context, partials, originalTemplate) {\n    var value = context.lookup(token[1]);\n\n    // Use JavaScript's definition of falsy. Include empty arrays.\n    // See https://github.com/janl/mustache.js/issues/186\n    if (!value || (isArray(value) && value.length === 0))\n      return this.renderTokens(token[4], context, partials, originalTemplate);\n  };\n\n  Writer.prototype.renderPartial = function renderPartial (token, context, partials) {\n    if (!partials) return;\n\n    var value = isFunction(partials) ? partials(token[1]) : partials[token[1]];\n    if (value != null)\n      return this.renderTokens(this.parse(value), context, partials, value);\n  };\n\n  Writer.prototype.unescapedValue = function unescapedValue (token, context) {\n    var value = context.lookup(token[1]);\n    if (value != null)\n      return value;\n  };\n\n  Writer.prototype.escapedValue = function escapedValue (token, context) {\n    var value = context.lookup(token[1]);\n    if (value != null)\n      return mustache.escape(value);\n  };\n\n  Writer.prototype.rawValue = function rawValue (token) {\n    return token[1];\n  };\n\n  mustache.name = 'mustache.js';\n  mustache.version = '2.2.1';\n  mustache.tags = [ '{{', '}}' ];\n\n  // All high-level mustache.* functions use this writer.\n  var defaultWriter = new Writer();\n\n  /**\n   * Clears all cached templates in the default writer.\n   */\n  mustache.clearCache = function clearCache () {\n    return defaultWriter.clearCache();\n  };\n\n  /**\n   * Parses and caches the given template in the default writer and returns the\n   * array of tokens it contains. Doing this ahead of time avoids the need to\n   * parse templates on the fly as they are rendered.\n   */\n  mustache.parse = function parse (template, tags) {\n    return defaultWriter.parse(template, tags);\n  };\n\n  /**\n   * Renders the `template` with the given `view` and `partials` using the\n   * default writer.\n   */\n  mustache.render = function render (template, view, partials) {\n    if (typeof template !== 'string') {\n      throw new TypeError('Invalid template! Template should be a \"string\" ' +\n                          'but \"' + typeStr(template) + '\" was given as the first ' +\n                          'argument for mustache#render(template, view, partials)');\n    }\n\n    return defaultWriter.render(template, view, partials);\n  };\n\n  // This is here for backwards compatibility with 0.4.x.,\n  /*eslint-disable */ // eslint wants camel cased function name\n  mustache.to_html = function to_html (template, view, partials, send) {\n    /*eslint-enable*/\n\n    var result = mustache.render(template, view, partials);\n\n    if (isFunction(send)) {\n      send(result);\n    } else {\n      return result;\n    }\n  };\n\n  // Export the escaping function so that the user may override it.\n  // See https://github.com/janl/mustache.js/issues/244\n  mustache.escape = escapeHtml;\n\n  // Export these mainly for testing, but also for advanced usage.\n  mustache.Scanner = Scanner;\n  mustache.Context = Context;\n  mustache.Writer = Writer;\n\n}));\n","/* object-iterators.js - Mini Underscore library\n * by Jonathan Eiten\n *\n * The methods below operate on objects (but not arrays) similarly\n * to Underscore (http://underscorejs.org/#collections).\n *\n * For more information:\n * https://github.com/joneit/object-iterators\n */\n\n'use strict';\n\n/**\n * @constructor\n * @summary Wrap an object for one method call.\n * @Desc Note that the `new` keyword is not necessary.\n * @param {object|null|undefined} object - `null` or `undefined` is treated as an empty plain object.\n * @return {Wrapper} The wrapped object.\n */\nfunction Wrapper(object) {\n    if (object instanceof Wrapper) {\n        return object;\n    }\n    if (!(this instanceof Wrapper)) {\n        return new Wrapper(object);\n    }\n    this.originalValue = object;\n    this.o = object || {};\n}\n\n/**\n * @name Wrapper.chain\n * @summary Wrap an object for a chain of method calls.\n * @Desc Calls the constructor `Wrapper()` and modifies the wrapper for chaining.\n * @param {object} object\n * @return {Wrapper} The wrapped object.\n */\nWrapper.chain = function (object) {\n    var wrapped = Wrapper(object); // eslint-disable-line new-cap\n    wrapped.chaining = true;\n    return wrapped;\n};\n\nWrapper.prototype = {\n    /**\n     * Unwrap an object wrapped with {@link Wrapper.chain|Wrapper.chain()}.\n     * @return {object|null|undefined} The value originally wrapped by the constructor.\n     * @memberOf Wrapper.prototype\n     */\n    value: function () {\n        return this.originalValue;\n    },\n\n    /**\n     * @desc Mimics Underscore's [each](http://underscorejs.org/#each) method: Iterate over the members of the wrapped object, calling `iteratee()` with each.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function is undefined; an `.each` loop cannot be broken out of (use {@link Wrapper#find|.find} instead).\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `iteratee`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {Wrapper} The wrapped object for chaining.\n     * @memberOf Wrapper.prototype\n     */\n    each: function (iteratee, context) {\n        var o = this.o;\n        Object.keys(o).forEach(function (key) {\n            iteratee.call(this, o[key], key, o);\n        }, context || o);\n        return this;\n    },\n\n    /**\n     * @desc Mimics Underscore's [find](http://underscorejs.org/#find) method: Look through each member of the wrapped object, returning the first one that passes a truth test (`predicate`), or `undefined` if no value passes the test. The function returns the value of the first acceptable member, and doesn't necessarily traverse the entire object.\n     * @param {function} predicate - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function should be truthy if the member passes the test and falsy otherwise.\n     * @param {object} [context] - If given, `predicate` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} The found property's value, or undefined if not found.\n     * @memberOf Wrapper.prototype\n     */\n    find: function (predicate, context) {\n        var o = this.o;\n        var result;\n        if (o) {\n            result = Object.keys(o).find(function (key) {\n                return predicate.call(this, o[key], key, o);\n            }, context || o);\n            if (result !== undefined) {\n                result = o[result];\n            }\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [filter](http://underscorejs.org/#filter) method: Look through each member of the wrapped object, returning the values of all members that pass a truth test (`predicate`), or empty array if no value passes the test. The function always traverses the entire object.\n     * @param {function} predicate - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function should be truthy if the member passes the test and falsy otherwise.\n     * @param {object} [context] - If given, `predicate` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} An array containing the filtered values.\n     * @memberOf Wrapper.prototype\n     */\n    filter: function (predicate, context) {\n        var o = this.o;\n        var result = [];\n        if (o) {\n            Object.keys(o).forEach(function (key) {\n                if (predicate.call(this, o[key], key, o)) {\n                    result.push(o[key]);\n                }\n            }, context || o);\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [map](http://underscorejs.org/#map) method: Produces a new array of values by mapping each value in list through a transformation function (`iteratee`). The function always traverses the entire object.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with three arguments: `(value, key, object)`. The return value of this function is concatenated to the end of the new array.\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `predicate`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} An array containing the filtered values.\n     * @memberOf Wrapper.prototype\n     */\n    map: function (iteratee, context) {\n        var o = this.o;\n        var result = [];\n        if (o) {\n            Object.keys(o).forEach(function (key) {\n                result.push(iteratee.call(this, o[key], key, o));\n            }, context || o);\n        }\n        return result;\n    },\n\n    /**\n     * @desc Mimics Underscore's [reduce](http://underscorejs.org/#reduce) method: Boil down the values of all the members of the wrapped object into a single value. `memo` is the initial state of the reduction, and each successive step of it should be returned by `iteratee()`.\n     * @param {function} iteratee - For each member of the wrapped object, this function is called with four arguments: `(memo, value, key, object)`. The return value of this function becomes the new value of `memo` for the next iteration.\n     * @param {*} [memo] - If no memo is passed to the initial invocation of reduce, the iteratee is not invoked on the first element of the list. The first element is instead passed as the memo in the invocation of the iteratee on the next element in the list.\n     * @param {object} [context] - If given, `iteratee` is bound to this object. In other words, this object becomes the `this` value in the calls to `iteratee`. (Otherwise, the `this` value will be the unwrapped object.)\n     * @return {*} The value of `memo` \"reduced\" as per `iteratee`.\n     * @memberOf Wrapper.prototype\n     */\n    reduce: function (iteratee, memo, context) {\n        var o = this.o;\n        if (o) {\n            Object.keys(o).forEach(function (key, idx) {\n                memo = (!idx && memo === undefined) ? o[key] : iteratee(memo, o[key], key, o);\n            }, context || o);\n        }\n        return memo;\n    },\n\n    /**\n     * @desc Mimics Underscore's [extend](http://underscorejs.org/#extend) method: Copy all of the properties in each of the `source` object parameter(s) over to the (wrapped) destination object (thus mutating it). It's in-order, so the properties of the last `source` object will override properties with the same name in previous arguments or in the destination object.\n     * > This method copies own members as well as members inherited from prototype chain.\n     * @param {...object|null|undefined} source - Values of `null` or `undefined` are treated as empty plain objects.\n     * @return {Wrapper|object} The wrapped destination object if chaining is in effect; otherwise the unwrapped destination object.\n     * @memberOf Wrapper.prototype\n     */\n    extend: function (source) {\n        var o = this.o;\n        Array.prototype.slice.call(arguments).forEach(function (object) {\n            if (object) {\n                for (var key in object) {\n                    o[key] = object[key];\n                }\n            }\n        });\n        return this.chaining ? this : o;\n    },\n\n    /**\n     * @desc Mimics Underscore's [extendOwn](http://underscorejs.org/#extendOwn) method: Like {@link Wrapper#extend|extend}, but only copies its \"own\" properties over to the destination object.\n     * @param {...object|null|undefined} source - Values of `null` or `undefined` are treated as empty plain objects.\n     * @return {Wrapper|object} The wrapped destination object if chaining is in effect; otherwise the unwrapped destination object.\n     * @memberOf Wrapper.prototype\n     */\n    extendOwn: function (source) {\n        var o = this.o;\n        Array.prototype.slice.call(arguments).forEach(function (object) {\n            Wrapper(object).each(function (val, key) { // eslint-disable-line new-cap\n                o[key] = val;\n            });\n        });\n        return this.chaining ? this : o;\n    }\n};\n\n// https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/find\nif (!Array.prototype.find) {\n    Array.prototype.find = function (predicate) { // eslint-disable-line no-extend-native\n        if (this === null) {\n            throw new TypeError('Array.prototype.find called on null or undefined');\n        }\n        if (typeof predicate !== 'function') {\n            throw new TypeError('predicate must be a function');\n        }\n        var list = Object(this);\n        var length = list.length >>> 0;\n        var thisArg = arguments[1];\n        var value;\n\n        for (var i = 0; i < length; i++) {\n            value = list[i];\n            if (predicate.call(thisArg, value, i, list)) {\n                return value;\n            }\n        }\n        return undefined;\n    };\n}\n\nmodule.exports = Wrapper;\n","'use strict';\n\n/* eslint-env node, browser */\n\n/**\n * Creates a new read-only property and attaches it to the provided context.\n * @private\n * @param {string} name - Name for new property.\n * @param {*} [value] - Value of new property.\n */\nfunction addReadOnlyProperty(name, value) {\n    Object.defineProperty(this, name, {\n        value: value,\n        writable: false,\n        enumerable: true,\n        configurable: false\n    });\n}\n\n/**\n * @constructor Point\n *\n * @desc This object represents a single point in an abstract 2-dimensional matrix.\n *\n * The unit of measure is typically pixels.\n * (If used to model computer graphics, vertical coordinates are typically measured downwards\n * from the top of the window. This convention however is not inherent in this object.)\n *\n * Note: This object should be instantiated with the `new` keyword.\n *\n * @param {number} x - the new point's `x` property\n * @param {number} y - the new point's `y` property\n */\nfunction Point(x, y) {\n\n    /**\n     * @name x\n     * @type {number}\n     * @summary This point's horizontal coordinate.\n     * @desc Created upon instantiation by the {@link Point|constructor}.\n     * @memberOf Point.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'x', Number(x) || 0);\n\n    /**\n     * @name y\n     * @type {number}\n     * @summary This point's vertical coordinate.\n     * @desc Created upon instantiation by the {@link Point|constructor}.\n     * @memberOf Point.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'y', Number(y) || 0);\n\n}\n\nPoint.prototype = {\n\n    /**\n     * @returns {Point} A new point which is this point's position increased by coordinates of given `offset`.\n     * @param {Point} offset - Horizontal and vertical values to add to this point's coordinates.\n     * @memberOf Point.prototype\n     */\n    plus: function(offset) {\n        return new Point(\n            this.x + offset.x,\n            this.y + offset.y\n        );\n    },\n\n    /**\n     * @returns {Point} A new point which is this point's position increased by given offsets.\n     * @param {number} [offsetX=0] - Value to add to this point's horizontal coordinate.\n     * @param {number} [offsetY=0] - Value to add to this point's horizontal coordinate.\n     * @memberOf Point.prototype\n     */\n    plusXY: function(offsetX, offsetY) {\n        return new Point(\n            this.x + (offsetX || 0),\n            this.y + (offsetY || 0)\n        );\n    },\n\n    /**\n     * @returns {Point} A new point which is this point's position decreased by coordinates of given `offset`.\n     * @param {Point} offset - Horizontal and vertical values to subtract from this point's coordinates.\n     * @memberOf Point.prototype\n     */\n    minus: function(offset) {\n        return new Point(\n            this.x - offset.x,\n            this.y - offset.y\n        );\n    },\n\n    /**\n     * @returns {Point} A new `Point` positioned to least x and least y of this point and given `offset`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    min: function(point) {\n        return new Point(\n            Math.min(this.x, point.x),\n            Math.min(this.y, point.y)\n        );\n    },\n\n    /**\n     * @returns {Point} A new `Point` positioned to greatest x and greatest y of this point and given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    max: function(point) {\n        return new Point(\n            Math.max(this.x, point.x),\n            Math.max(this.y, point.y)\n        );\n    },\n\n    /**\n     * @returns {number} Distance between given `point` and this point using Pythagorean Theorem formula.\n     * @param {Point} point - A point from which to compute the distance to this point.\n     * @memberOf Point.prototype\n     */\n    distance: function(point) {\n        var deltaX = point.x - this.x,\n            deltaY = point.y - this.y;\n\n        return Math.sqrt(\n            deltaX * deltaX +\n            deltaY * deltaY\n        );\n    },\n\n    /**\n     * _(Formerly: `equal`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are exactly equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    equals: function(point) {\n        var result = false;\n\n        if (point) {\n            result =\n                this.x === point.x &&\n                this.y === point.y;\n        }\n\n        return result;\n    },\n\n    /**\n     * @returns {boolean} `true` iff _both_ coordinates of this point are greater than those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    greaterThan: function(point) {\n        return (\n            this.x > point.x &&\n            this.y > point.y\n        );\n    },\n\n    /**\n     * @returns {boolean} `true` iff _both_ coordinates of this point are less than those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    lessThan: function(point) {\n        return (\n            this.x < point.x &&\n            this.y < point.y\n        );\n    },\n\n    /**\n     * _(Formerly `greaterThanEqualTo`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are greater than or equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point\n     * @memberOf Point.prototype\n     */\n    greaterThanOrEqualTo: function(point) {\n        return (\n            this.x >= point.x &&\n            this.y >= point.y\n        );\n    },\n\n    /**\n     * _(Formerly `lessThanEqualTo`.)_\n     * @returns {boolean} `true` iff _both_ coordinates of this point are less than or equal to those of given `point`.\n     * @param {Point} point - A point to compare to this point.\n     * @memberOf Point.prototype\n     */\n    lessThanOrEqualTo: function(point) {\n        return (\n            this.x <= point.x &&\n            this.y <= point.y\n        );\n    },\n\n    /**\n     * _(Formerly `isContainedWithinRectangle`.)_\n     * @param rect {Rectangle} - Rectangle to test this point against.\n     * @returns {boolean} `true` iff this point is within given `rect`.\n     * @memberOf Point.prototype\n     */\n    within: function(rect) {\n        var minX = rect.origin.x,\n            maxX = minX + rect.extent.x;\n        var minY = rect.origin.y,\n            maxY = minY + rect.extent.y;\n\n        if (rect.extent.x < 0) {\n            minX = maxX;\n            maxX = rect.origin.x;\n        }\n\n        if (rect.extent.y < 0) {\n            minY = maxY;\n            maxY = rect.origin.y;\n        }\n\n        return (\n            minX <= this.x && this.x < maxX &&\n            minY <= this.y && this.y < maxY\n        );\n    }\n};\n\nPoint.prototype.EQ = Point.prototype.equals;\nPoint.prototype.GT = Point.prototype.greaterThan;\nPoint.prototype.LT = Point.prototype.lessThan;\nPoint.prototype.GE = Point.prototype.greaterThanOrEqualTo;\nPoint.prototype.LE = Point.prototype.lessThanOrEqualTo;\n\n\n/**\n * @constructor Rectangle\n *\n * @desc This object represents a rectangular area within an abstract 2-dimensional matrix.\n *\n * The unit of measure is typically pixels.\n * (If used to model computer graphics, vertical coordinates are typically measured downwards\n * from the top of the window. This convention however is not inherent in this object.)\n *\n * Normally, the `x` and `y` parameters to the constructor describe the upper left corner of the rect.\n * However, negative values of `width` and `height` will be added to the given `x` and `y`. That is,\n * a negative value of the `width` parameter will extend the rect to the left of the given `x` and\n * a negative value of the `height` parameter will extend the rect above the given `y`.\n * In any case, after instantiation the following are guaranteed to always be true:\n * * The `extent`, `width`, and `height` properties _always_ give positive values.\n * * The `origin`, `top`, and `left` properties _always_ reflect the upper left corner.\n * * The `corner`, `bottom`, and `right` properties _always_ reflect the lower right corner.\n *\n * Note: This object should be instantiated with the `new` keyword.\n *\n * @param {number} [x=0] - Horizontal coordinate of some corner of the rect.\n * @param {number} [y=0] - Vertical coordinate of some corner of the rect.\n * @param {number} [width=0] - Width of the new rect. May be negative (see above).\n * @param {number} [height=0] - Height of the new rect. May be negative (see above).\n */\nfunction Rectangle(x, y, width, height) {\n\n    x = Number(x) || 0;\n    y = Number(y) || 0;\n    width = Number(width) || 0;\n    height = Number(height) || 0;\n\n    if (width < 0) {\n        x += width;\n        width = -width;\n    }\n\n    if (height < 0) {\n        y += height;\n        height = -height;\n    }\n\n    /**\n     * @name origin\n     * @type {Point}\n     * @summary Upper left corner of this rect.\n     * @desc Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'origin', new Point(x, y));\n\n    /**\n     * @name extent\n     * @type {Point}\n     * @summary this rect's width and height.\n     * @desc Unlike the other `Point` properties, `extent` is not a global coordinate pair; rather it consists of a _width_ (`x`, always positive) and a _height_ (`y`, always positive).\n     *\n     * This object might be more legitimately typed as something like `Area` with properties `width` and `height`; however we wanted it to be able to use it efficiently with a point's `plus` and `minus` methods (that is, without those methods having to check and branch on the type of its parameter).\n     *\n     * Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @see The {@link Rectangle#corner|corner} method.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'extent', new Point(width, height));\n\n    /**\n     * @name corner\n     * @type {Point}\n     * @summary Lower right corner of this rect.\n     * @desc This is a calculated value created upon instantiation by the {@linkplain Rectangle|constructor}. It is `origin` offset by `extent`.\n     *\n     * **Note:** These coordinates actually point to the pixel one below and one to the right of the rect's actual lower right pixel.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'corner', new Point(x + width, y + height));\n\n    /**\n     * @name center\n     * @type {Point}\n     * @summary Center of this rect.\n     * @desc Created upon instantiation by the {@linkplain Rectangle|constructor}.\n     * @memberOf Rectangle.prototype\n     * @abstract\n     */\n    addReadOnlyProperty.call(this, 'center', new Point(x + (width / 2), y + (height / 2)));\n\n}\n\nRectangle.prototype = {\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Minimum vertical coordinate of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get top() {\n        return this.origin.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Minimum horizontal coordinate of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get left() {\n        return this.origin.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Maximum vertical coordinate of this rect + 1.\n     * @memberOf Rectangle.prototype\n     */\n    get bottom() {\n        return this.corner.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Maximum horizontal coordinate of this rect + 1.\n     * @memberOf Rectangle.prototype\n     */\n    get right() {\n        return this.corner.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Width of this rect (always positive).\n     * @memberOf Rectangle.prototype\n     */\n    get width() {\n        return this.extent.x;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Height of this rect (always positive).\n     * @memberOf Rectangle.prototype\n     */\n    get height() {\n        return this.extent.y;\n    },\n\n    /**\n     * @type {number}\n     * @desc _(Formerly a function; now a getter.)_\n     * @summary Area of this rect.\n     * @memberOf Rectangle.prototype\n     */\n    get area() {\n        return this.width * this.height;\n    },\n\n    /**\n     * @returns {Rectangle} A copy of this rect but with horizontal position reset to given `x` and no width.\n     * @param {number} x - Horizontal coordinate of the new rect.\n     * @memberOf Rectangle.prototype\n     */\n    flattenXAt: function(x) {\n        return new Rectangle(x, this.origin.y, 0, this.extent.y);\n    },\n\n    /**\n     * @returns {Rectangle} A copy of this rect but with vertical position reset to given `y` and no height.\n     * @param {number} y - Vertical coordinate of the new rect.\n     * @memberOf Rectangle.prototype\n     */\n    flattenYAt: function(y) {\n        return new Rectangle(this.origin.x, y, this.extent.x, 0);\n    },\n\n    /**\n     * @returns {boolean} `true` iff given `point` entirely contained within this rect.\n     * @param {Point} pointOrRect - The point or rect to test for containment.\n     * @memberOf Rectangle.prototype\n     */\n    contains: function(pointOrRect) {\n        return pointOrRect.within(this);\n    },\n\n    /**\n     * _(Formerly `isContainedWithinRectangle`.)_\n     * @returns {boolean} `true` iff `this` rect is entirely contained within given `rect`.\n     * @param {Rectangle} rect - Rectangle to test against this rect.\n     * @memberOf Rectangle.prototype\n     */\n    within: function(rect) {\n        return (\n            rect.origin.lessThanOrEqualTo(this.origin) &&\n            rect.corner.greaterThanOrEqualTo(this.corner)\n        );\n    },\n\n    /**\n     * _(Formerly: `insetBy`.)_\n     * @returns {Rectangle} That is enlarged/shrunk by given `padding`.\n     * @param {number} padding - Amount by which to increase (+) or decrease (-) this rect\n     * @see The {@link Rectangle#shrinkBy|shrinkBy} method.\n     * @memberOf Rectangle.prototype\n     */\n    growBy: function(padding) {\n        return new Rectangle(\n            this.origin.x + padding,\n            this.origin.y + padding,\n            this.extent.x - padding - padding,\n            this.extent.y - padding - padding);\n    },\n\n    /**\n     * @returns {Rectangle} That is enlarged/shrunk by given `padding`.\n     * @param {number} padding - Amount by which to decrease (+) or increase (-) this rect.\n     * @see The {@link Rectangle#growBy|growBy} method.\n     * @memberOf Rectangle.prototype\n     */\n    shrinkBy: function(padding) {\n        return this.growBy(-padding);\n    },\n\n    /**\n     * @returns {Rectangle} Bounding rect that contains both this rect and the given `rect`.\n     * @param {Rectangle} rect - The rectangle to union with this rect.\n     * @memberOf Rectangle.prototype\n     */\n    union: function(rect) {\n        var origin = this.origin.min(rect.origin),\n            corner = this.corner.max(rect.corner),\n            extent = corner.minus(origin);\n\n        return new Rectangle(\n            origin.x, origin.y,\n            extent.x, extent.y\n        );\n    },\n\n    /**\n     * iterate over all points within this rect, invoking `iteratee` for each.\n     * @param {function(number,number)} iteratee - Function to call for each point.\n     * Bound to `context` when given; otherwise it is bound to this rect.\n     * Each invocation of `iteratee` is called with two arguments:\n     * the horizontal and vertical coordinates of the point.\n     * @param {object} [context=this] - Context to bind to `iteratee` (when not `this`).\n     * @memberOf Rectangle.prototype\n     */\n    forEach: function(iteratee, context) {\n        context = context || this;\n        for (var x = this.origin.x, x2 = this.corner.x; x < x2; x++) {\n            for (var y = this.origin.y, y2 = this.corner.y; y < y2; y++) {\n                iteratee.call(context, x, y);\n            }\n        }\n    },\n\n    /**\n     * @returns {Rectangle} One of:\n     * * _If this rect intersects with the given `rect`:_\n     *      a new rect representing that intersection.\n     * * _If it doesn't intersect and `ifNoneAction` defined:_\n     *      result of calling `ifNoneAction`.\n     * * _If it doesn't intersect and `ifNoneAction` undefined:_\n     *      `null`.\n     * @param {Rectangle} rect - The rectangle to intersect with this rect.\n     * @param {function(Rectangle)} [ifNoneAction] - When no intersection, invoke and return result.\n     * Bound to `context` when given; otherwise bound to this rect.\n     * Invoked with `rect` as sole parameter.\n     * @param {object} [context=this] - Context to bind to `ifNoneAction` (when not `this`).\n     * @memberOf Rectangle.prototype\n     */\n    intersect: function(rect, ifNoneAction, context) {\n        var result = null,\n            origin = this.origin.max(rect.origin),\n            corner = this.corner.min(rect.corner),\n            extent = corner.minus(origin);\n\n        if (extent.x > 0 && extent.y > 0) {\n            result = new Rectangle(\n                origin.x, origin.y,\n                extent.x, extent.y\n            );\n        } else if (typeof ifNoneAction === 'function') {\n            result = ifNoneAction.call(context || this, rect);\n        }\n\n        return result;\n    },\n\n    /**\n     * @returns {boolean} `true` iff this rect overlaps with given `rect`.\n     * @param {Rectangle} rect - The rectangle to intersect with this rect.\n     * @memberOf Rectangle.prototype\n     */\n    intersects: function(rect) {\n        return (\n            rect.corner.x > this.origin.x &&\n            rect.corner.y > this.origin.y &&\n            rect.origin.x < this.corner.x &&\n            rect.origin.y < this.corner.y\n        );\n    }\n};\n\n// Interface\nexports.Point = Point;\nexports.Rectangle = Rectangle;\n","'use strict';\n\nvar // a regex search pattern that matches all the reserved chars of a regex search pattern\n    reserved = /([\\.\\\\\\+\\*\\?\\^\\$\\(\\)\\{\\}\\=\\!\\<\\>\\|\\:\\[\\]])/g,\n\n    // regex wildcard search patterns\n    REGEXP_WILDCARD = '.*',\n    REGEXP_WILDCHAR = '.',\n    REGEXP_WILDCARD_MATCHER = '(' + REGEXP_WILDCARD + ')',\n\n    // LIKE search patterns\n    LIKE_WILDCHAR = '_',\n    LIKE_WILDCARD = '%',\n\n    // regex search patterns that match LIKE search patterns\n    REGEXP_LIKE_PATTERN_MATCHER = new RegExp('(' + [\n        LIKE_WILDCHAR,\n        LIKE_WILDCARD,\n        '\\\\[\\\\^?[^-\\\\]]+]', // matches a LIKE set (same syntax as a RegExp set)\n        '\\\\[\\\\^?[^-\\\\]]\\\\-[^\\\\]]]' // matches a LIKE range (same syntax as a RegExp range)\n    ].join('|') + ')', 'g');\n\nfunction regExpLIKE(pattern, ignoreCase) {\n    var i, parts;\n\n    // Find all LIKE patterns\n    parts = pattern.match(REGEXP_LIKE_PATTERN_MATCHER);\n\n    if (parts) {\n        // Translate found LIKE patterns to regex patterns, escaped intervening non-patterns, and interleave the two\n\n        for (i = 0; i < parts.length; ++i) {\n            // Escape left brackets (unpaired right brackets are OK)\n            if (parts[i][0] === '[') {\n                parts[i] = regExpLIKE.reserve(parts[i]);\n            }\n\n            // Make each found pattern matchable by enclosing in parentheses\n            parts[i] = '(' + parts[i] + ')';\n        }\n\n        // Match these precise patterns again with their intervening non-patterns (i.e., text)\n        parts = pattern.match(new RegExp(\n            REGEXP_WILDCARD_MATCHER +\n            parts.join(REGEXP_WILDCARD_MATCHER)  +\n            REGEXP_WILDCARD_MATCHER\n        ));\n\n        // Discard first match of non-global search (which is the whole string)\n        parts.shift();\n\n        // For each re-found pattern part, translate % and _ to regex equivalent\n        for (i = 1; i < parts.length; i += 2) {\n            var part = parts[i];\n            switch (part) {\n                case LIKE_WILDCARD: part = REGEXP_WILDCARD; break;\n                case LIKE_WILDCHAR: part = REGEXP_WILDCHAR; break;\n                default:\n                    var j = part[1] === '^' ? 2 : 1;\n                    part = '[' + regExpLIKE.reserve(part.substr(j, part.length - (j + 1))) + ']';\n            }\n            parts[i] = part;\n        }\n    } else {\n        parts = [pattern];\n    }\n\n    // For each surrounding text part, escape reserved regex chars\n    for (i = 0; i < parts.length; i += 2) {\n        parts[i] = regExpLIKE.reserve(parts[i]);\n    }\n\n    // Join all the interleaved parts\n    parts = parts.join('');\n\n    // Optimize or anchor the pattern at each end as needed\n    if (parts.substr(0, 2) === REGEXP_WILDCARD) { parts = parts.substr(2); } else { parts = '^' + parts; }\n    if (parts.substr(-2, 2) === REGEXP_WILDCARD) { parts = parts.substr(0, parts.length - 2); } else { parts += '$'; }\n\n    // Return the new regex\n    return new RegExp(parts, ignoreCase ? 'i' : undefined);\n}\n\nregExpLIKE.reserve = function (s) {\n    return s.replace(reserved, '\\\\$1');\n};\n\nvar cache, size;\n\n/**\n * @summary Delete a pattern from the cache; or clear the whole cache.\n * @param {string} [pattern] - The LIKE pattern to remove from the cache. Fails silently if not found in the cache. If pattern omitted, clears whole cache.\n */\n(regExpLIKE.clearCache = function (pattern) {\n    if (!pattern) {\n        cache = {};\n        size = 0;\n    } else if (cache[pattern]) {\n        delete cache[pattern];\n        size--;\n    }\n    return size;\n})(); // init the cache\n\nregExpLIKE.getCacheSize = function () { return size; };\n\n/**\n * @summary Cached version of `regExpLIKE()`.\n * @desc Cached entries are subject to garbage collection if `keep` is `undefined` or `false` on insertion or `false` on most recent reference. Garbage collection will occur iff `regExpLIKE.cacheMax` is defined and it equals the number of cached patterns. The garbage collector sorts the patterns based on most recent reference; the oldest 10% of the entries are deleted. Alternatively, you can manage the cache yourself to a limited extent (see {@link regeExpLIKE.clearCache|clearCache}).\n * @param pattern - the LIKE pattern (to be) converted to a RegExp\n * @param [keep] - If given, changes the keep status for this pattern as follows:\n * * `true` permanently caches the pattern (not subject to garbage collection) until `false` is given on a subsequent call\n * * `false` allows garbage collection on the cached pattern\n * * `undefined` no change to keep status\n * @returns {RegExp}\n */\nregExpLIKE.cached = function (keep, pattern, ignoreCase) {\n    if (typeof keep === 'string') {\n        ignoreCase = pattern;\n        pattern = keep;\n        keep = false;\n    }\n    var patternAndCase = pattern + (ignoreCase ? 'i' : 'c'),\n        item = cache[patternAndCase];\n    if (item) {\n        item.when = new Date().getTime();\n        if (keep !== undefined) {\n            item.keep = keep;\n        }\n    } else {\n        if (size === regExpLIKE.cacheMax) {\n            var age = [], ages = 0, key, i;\n            for (key in cache) {\n                item = cache[key];\n                if (!item.keep) {\n                    for (i = 0; i < ages; ++i) {\n                        if (item.when < age[i].item.when) {\n                            break;\n                        }\n                    }\n                    age.splice(i, 0, { key: key, item: item });\n                    ages++;\n                }\n            }\n            if (!age.length) {\n                return regExpLIKE(pattern, ignoreCase); // cache is full!\n            }\n            i = Math.ceil(age.length / 10); // will always be at least 1\n            size -= i;\n            while (i--) {\n                delete cache[age[i].key];\n            }\n        }\n        item = cache[patternAndCase] = {\n            regex: regExpLIKE(pattern, ignoreCase),\n            keep: keep,\n            when: new Date().getTime()\n        };\n        size++;\n    }\n    return item.regex;\n};\n\nmodule.exports = regExpLIKE;\n","'use strict';\n\n/* eslint-env node, browser */\n\n(function (module) {  // eslint-disable-line no-unused-expressions\n\n    // This closure supports NodeJS-less client side includes with <script> tags. See https://github.com/joneit/mnm.\n\n    /**\n     * @constructor RangeSelectionModel\n     *\n     * @desc This object models selection of \"cells\" within an abstract single-dimensional matrix.\n     *\n     * Disjoint selections can be built with calls to the following methods:\n     * * {@link RangeSelectionModel#select|select(start, stop)} - Add a range to the matrix.\n     * * {@link RangeSelectionModel#deselect|deselect(start, stop)} - Remove a range from the matrix.\n     *\n     * Two more methods are available:\n     * * Test a cell to see if it {@link RangeSelectionModel#isSelected|isSelected(cell)}\n     * * {@link RangeSelectionModel#clear|clear()} the matrix\n     *\n     * Internally, the selection is run-length-encoded. It is therefore a \"sparse\" matrix\n     * with undefined bounds. A single data property called `selection` is an array that\n     * contains all the \"runs\" (ranges) of selected cells albeit in no particular order.\n     * This property should not normally need to be accessed directly.\n     *\n     * Note: This object should be instantiated with the `new` keyword.\n     *\n     * @returns {RangeSelectionModel} Self (i.e., `this` object).\n     */\n    function RangeSelectionModel() {\n        /**\n         * @name selection\n         * @type {Array.Array.number}\n         * @summary Unordered list of runs.\n         * @desc A \"run\" is defined as an Array(2) where:\n         * * element [0] is the beginning of the run\n         * * element [1] is the end of the run (inclusive) and is always >= element [0]\n         * The order of the runs within is undefined.\n         * @memberOf RangeSelectionModel.prototype\n         * @abstract\n         */\n        this.selection = [];\n\n        //we need to be able to go back in time\n        //the states field\n        this.states = [];\n\n        //clone and store my current state\n        //so we can unwind changes if need be\n        this.storeState = function () {\n            var sels = this.selection;\n            var state = [];\n            var copy;\n            for (var i = 0; i < sels.length; i++) {\n                copy = [].concat(sels[i]);\n                state.push(copy);\n            }\n            this.states.push(state);\n        };\n    }\n\n    RangeSelectionModel.prototype = {\n\n        /**\n         * @summary Add a contiguous run of points to the selection.\n         * @desc Insert a new run into `this.selection`.\n         * The new run will be merged with overlapping and adjacent runs.\n         *\n         * The two parameters may be given in either order.\n         * The start and stop elements in the resulting run will however always be ordered.\n         * (However, note that the order of the runs within `this.selection` is itself always unordered.)\n         *\n         * Note that `this.selection` is updated in place, preserving validity of any external references.\n         * @param {number} start - Start of run. May be greater than `stop`.\n         * @param {number} [stop=stop] - End of run (inclusive). May be less than `start`.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        select: function (start, stop) {\n            this.storeState();\n            var run = makeRun(start, stop);\n            var splicer = [0, 1];\n            this.selection.forEach(function (each) {\n                if (overlaps(each, run) || abuts(each, run)) {\n                    run = merge(each, run);\n                } else {\n                    splicer.push(each);\n                }\n            });\n            splicer.push(run);\n            splicer[1] = this.selection.length;\n            this.selection.splice.apply(this.selection, splicer); // update in place to preserve external references\n            return this;\n        },\n\n        /**\n         * @summary Remove a contiguous run of points from the selection.\n         * @desc Truncate and/or remove run(s) from `this.selection`.\n         * Removing part of existing runs will (correctly) shorten them or break them into two fragments.\n         *\n         * The two parameters may be given in either order.\n         *\n         * Note that `this.selection` is updated in place, preserving validity of any external references.\n         * @param {number} start - Start of run. May be greater than `stop`.\n         * @param {number} [stop=stop] - End of run (inclusive). May be less than `start`.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        deselect: function (start, stop) {\n            var run = makeRun(start, stop);\n            var splicer = [0, 0];\n            this.selection.forEach(function (each) {\n                if (overlaps(each, run)) {\n                    var pieces = subtract(each, run);\n                    splicer = splicer.concat(pieces);\n                } else {\n                    splicer.push(each);\n                }\n            });\n            splicer[1] = this.selection.length;\n            this.selection.splice.apply(this.selection, splicer); // update in place to preserve external references\n            return this;\n        },\n\n        /**\n         * @summary Empties `this.selection`, effectively removing all runs.\n         * @returns {RangeSelectionModel} Self (i.e., `this`), for chaining.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        clear: function () {\n            this.states.length = 0;\n            this.selection.length = 0;\n            return this;\n        },\n\n        clearMostRecentSelection: function () {\n            if (this.states.length === 0) {\n                return;\n            }\n            this.selection = this.states.pop();\n        },\n\n        /**\n         * @summary Determines if the given `cell` is selected.\n         * @returns {boolean} `true` iff given `cell` is within any of the runs in `this.selection`.\n         * @param {number} cell - The cell to test for inclusion in the selection.\n         * @memberOf RangeSelectionModel.prototype\n         */\n        isSelected: function (cell) {\n            return this.selection.some(function (each) {\n                return each[0] <= cell && cell <= each[1];\n            });\n        },\n\n        isEmpty: function (){\n            return this.selection.length === 0;\n        },\n\n        /**\n         * @summary Return the indexes that are selected.\n         * @desc Return the indexes that are selected.\n         * @returns {Array.Array.number}\n         * @memberOf RangeSelectionModel.prototype\n         */\n        getSelections: function (){\n            var result = [];\n            this.selection.forEach(function (each) {\n                for (var i = each[0]; i <= each[1]; i++) {\n                    result.push(i);\n                }\n            });\n            result.sort(function (a, b){\n                return a - b;\n            });\n            return result;\n        }\n\n    };\n\n    /**\n     * @private\n     * @summary Preps `start` and `stop` params into order array\n     * @function makeRun\n     * @desc Utility function called by both `select()` and `deselect()`.\n     * @param {number|number[]} start - Start of run. if array, `start` and `stop` are taken from first two elements.\n     * @param {number} [stop=start] - End of run (inclusive).\n     */\n    function makeRun(start, stop) {\n        return (\n            start instanceof Array\n                ? makeRun.apply(this, start) // extract params from given array\n                : stop === undefined\n                ? [ start, start ] // single param is a run that stops where it starts\n                : start <= stop\n                ? [ start, stop ]\n                : [ stop, start ] // reverse descending params into ascending order\n        );\n    }\n\n    /**\n     * @private\n     * @function overlaps\n     * @returns {boolean} `true` iff `run1` overlaps `run2`\n     * @summary Comparison operator that determines if given runs overlap with one another.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * Overlap is defined to include the case where one run completely contains the other.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - first run\n     * @param {number[]} run2 - second run\n     */\n    function overlaps(run1, run2) {\n        return (\n            run1[0] <= run2[0] && run2[0] <= run1[1] || // run2's start is within run1 OR...\n            run1[0] <= run2[1] && run2[1] <= run1[1] || // run2's stop is within run1 OR...\n            run2[0] <  run1[0] && run1[1] <  run2[1]    // run2 completely contains run1\n        );\n    }\n\n    /**\n     * @private\n     * @function abuts\n     * @summary Comparison operator that determines if given runs are consecutive with one another.\n     * @returns {boolean} `true` iff `run1` is consecutive with `run2`\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - first run\n     * @param {number[]} run2 - second run\n     */\n    function abuts(run1, run2) {\n        return (\n            run1[1] === run2[0] - 1 || // run1's top immediately precedes run2's start OR...\n            run2[1] === run1[0] - 1    // run2's top immediately precedes run1's start\n        );\n    }\n\n    /**\n     * @private\n     * @function subtract\n     * @summary Operator that subtracts one run from another.\n     * @returns {Array.Array.number} The remaining pieces of `minuend` after removing `subtrahend`.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * This function _does not assumes_ that `overlap()` has already been called with the same runs and has returned `true`.\n     *\n     * Returned array contains 0, 1, or 2 runs which are the portion(s) of `minuend` that do _not_ include `subtrahend`.\n     *\n     * Caveat: This operator is *not* commutative.\n     * @param {number[]} minuend - a run from which to \"subtract\" `subtrahend`\n     * @param {number[]} subtrahend - a run to \"subtracted\" from `minuend`\n     */\n    function subtract(minuend, subtrahend) {\n        var m0 = minuend[0];\n        var m1 = minuend[1];\n        var s0 = subtrahend[0];\n        var s1 = subtrahend[1];\n        var result = [];\n\n        if (s0 <= m0 && s1 < m1) {\n            //subtrahend extends before minuend: return remaining piece of `minuend`\n            result.push([s1 + 1, m1]);\n        } else if (s0 > m0 && s1 >= m1) {\n            //subtrahend extends after minuend: return remaining piece of `minuend`\n            result.push([m0, s0 - 1]);\n        } else if (m0 < s0 && s1 < m1) {\n            //completely inside: return 2 smaller pieces resulting from the hole\n            result.push([m0, s0 - 1]);\n            result.push([s1 + 1, m1]);\n        } else if (s1 < m0 || s0 > m1) {\n            // completely outside: return `minuend` untouched\n            result.push(minuend);\n        }\n\n        //else subtrahend must completely overlap minuend so return no pieces\n\n        return result;\n    }\n\n\n    // Local utility functions\n\n    /**\n     * @private\n     * @function merge\n     * @summary Operator that merges given runs.\n     * @returns {number[]} A single merged run.\n     * @desc Both parameters are assumed to be _ordered_ arrays.\n     *\n     * The runs are assumed to be overlapping or adjacent to one another.\n     *\n     * Note: This operator is commutative.\n     * @param {number[]} run1 - a run to merge with `run2`\n     * @param {number[]} run2 - a run to merge with `run1`\n     */\n    function merge(run1, run2) {\n        var min = Math.min(Math.min.apply(Math, run1), Math.min.apply(Math, run2));\n        var max = Math.max(Math.max.apply(Math, run1), Math.max.apply(Math, run2));\n        return [min, max];\n    }\n\n    // Interface\n    module.exports = RangeSelectionModel;\n})(\n    typeof module === 'object' && module || (window.RangeSelectionModel = {}),\n    typeof module === 'object' && module.exports || (window.RangeSelectionModel.exports = {})\n) || (\n    typeof module === 'object' || (window.RangeSelectionModel = window.RangeSelectionModel.exports)\n);\n\n/* About the above IIFE:\n * This file is a \"modified node module.\" It functions as usual in Node.js *and* is also usable directly in the browser.\n * 1. Node.js: The IIFE is superfluous but innocuous.\n * 2. In the browser: The IIFE closure serves to keep internal declarations private.\n * 2.a. In the browser as a global: The logic in the actual parameter expressions + the post-invocation expression\n * will put your API in `window.RangeSelectionModel`.\n * 2.b. In the browser as a module: If you predefine a `window.module` object, the results will be in `module.exports`.\n * The bower component `mnm` makes this easy and also provides a global `require()` function for referencing your module\n * from other closures. In either case, this works with both NodeJs-style export mechanisms -- a single API assignment,\n * `module.exports = yourAPI` *or* a series of individual property assignments, `module.exports.property = property`.\n *\n * Before the IIFE runs, the actual parameter expressions are executed:\n * 1. If `window` object undefined, we're in NodeJs so assume there is a `module` object with an `exports` property\n * 2. If `window` object defined, we're in browser\n * 2.a. If `module` object predefined, use it\n * 2.b. If `module` object undefined, create a `RangeSelectionModel` object\n *\n * After the IIFE returns:\n * Because it always returns undefined, the expression after the || will execute:\n * 1. If `window` object undefined, then we're in NodeJs so we're done\n * 2. If `window` object defined, then we're in browser\n * 2.a. If `module` object predefined, we're done; results are in `moudule.exports`\n * 2.b. If `module` object undefined, redefine`RangeSelectionModel` to be the `RangeSelectionModel.exports` object\n */\n","// templex node module\n// https://github.com/joneit/templex\n\n/* eslint-env node */\n\n/**\n * Merges values of execution context properties named in template by {prop1},\n * {prop2}, etc., or any javascript expression incorporating such prop names.\n * The context always includes the global object. In addition you can specify a single\n * context or an array of contexts to search (in the order given) before finally\n * searching the global context.\n *\n * Merge expressions consisting of simple numeric terms, such as {0}, {1}, etc., deref\n * the first context given, which is assumed to be an array. As a convenience feature,\n * if additional args are given after `template`, `arguments` is unshifted onto the context\n * array, thus making first additional arg available as {1}, second as {2}, etc., as in\n * `templex('Hello, {1}!', 'World')`. ({0} is the template so consider this to be 1-based.)\n *\n * If you prefer something other than braces, redefine `templex.regexp`.\n *\n * See tests for examples.\n *\n * @param {string} template\n * @param {...string} [args]\n */\nfunction templex(template) {\n    var contexts = this instanceof Array ? this : [this];\n    if (arguments.length > 1) { contexts.unshift(arguments); }\n    return template.replace(templex.regexp, templex.merger.bind(contexts));\n}\n\ntemplex.regexp = /\\{(.*?)\\}/g;\n\ntemplex.with = function (i, s) {\n    return 'with(this[' + i + ']){' + s + '}';\n};\n\ntemplex.cache = [];\n\ntemplex.deref = function (key) {\n    if (!(this.length in templex.cache)) {\n        var code = 'return eval(expr)';\n\n        for (var i = 0; i < this.length; ++i) {\n            code = templex.with(i, code);\n        }\n\n        templex.cache[this.length] = eval('(function(expr){' + code + '})'); // eslint-disable-line no-eval\n    }\n    return templex.cache[this.length].call(this, key);\n};\n\ntemplex.merger = function (match, key) {\n    // Advanced features: Context can be a list of contexts which are searched in order.\n    var replacement;\n\n    try {\n        replacement = isNaN(key) ? templex.deref.call(this, key) : this[0][key];\n    } catch (e) {\n        replacement = '{' + key + '}';\n    }\n\n    return replacement;\n};\n\n// this interface consists solely of the templex function (and it's properties)\nmodule.exports = templex;\n","// Created by Jonathan Eiten on 1/7/16.\n\n'use strict';\n\n\n/**\n * @summary Walk a hierarchical object as JSON.stringify does but without serializing.\n *\n * @desc Usage:\n * * var myDistilledObject = unstrungify.call(myObject);\n * * var myDistilledObject = myApi.getState(); // where myApi.prototype.getState = unstrungify\n *\n * Result equivalent to `JSON.parse(JSON.stringify(this))`.\n *\n * > Do not use this function to get a JSON string; use `JSON.stringify(this)` instead.\n *\n * @this {*|object|*[]} - Object to walk; typically an object or array.\n *\n * @param {boolean} [options.preserve=false] - Preserve undefined array elements as `null`s.\n * Use this when precise index matters (not merely the order of the elements).\n *\n * @returns {object} - Distilled object.\n */\nfunction unstrungify(options) {\n    var clone, value,\n        object = (typeof this.toJSON === 'function') ? this.toJSON() : this,\n        preserve = options && options.preserve;\n\n    if (unstrungify.isArray(object)) {\n        clone = [];\n        object.forEach(function(obj) {\n            value = unstrungify.call(obj);\n            if (value !== undefined) {\n                clone.push(value);\n            } else if (preserve) {\n                clone.push(null); // undefined not a valid JSON value\n            }\n        });\n    } else  if (typeof object === 'object') {\n        clone = {};\n        Object.keys(object).forEach(function(key) {\n            value = unstrungify.call(object[key]);\n            if (value !== undefined) {\n                clone[key] = value;\n            }\n        });\n    } else {\n        clone = object;\n    }\n\n    return clone;\n}\n\n/**\n * Very fast array test.\n * For cross-frame scripting; use `crossFramesIsArray` instead.\n * @param {*} arr - The object to test.\n * @returns {boolean}\n */\nfunction isArray(arr) { return arr.constructor === Array; }\nunstrungify.isArray = isArray;\n\nvar toString = Object.prototype.toString, arrString = '[object Array]';\n\n/**\n * Very slow array test. Suitable for cross-frame scripting.\n *\n * Suggestion: If you need this and have jQuery loaded, use `jQuery.isArray` instead which is reasonably fast.\n *\n * @param {*} arr - The object to test.\n * @returns {boolean}\n */\nfunction crossFramesIsArray(arr) { return toString.call(arr) === arrString; } // eslint-disable-line no-unused-vars\n\nmodule.exports = unstrungify;\n","/* eslint-env browser */\n\n'use strict';\n\nvar extend = require('extend-me');\nvar deprecated = require('./lib/deprecated');\nextend.debug = true;\n\nvar FinBar = require('finbars');\nvar Canvas = require('fincanvas');\nvar Point = require('rectangular').Point;\nvar Rectangle = require('rectangular').Rectangle;\nvar _ = require('object-iterators');\n\nvar defaults = require('./defaults');\nvar Renderer = require('./lib/Renderer');\nvar SelectionModel = require('./lib/SelectionModel');\nvar addStylesheet = require('../css/stylesheets');\nvar TableDialog = require('./lib/TableDialog');\nvar Formatters = require('./lib/Formatters');\nvar CustomFilter = require('./lib/CustomFilter');\n\nvar themeInitialized = false,\n    polymerTheme = Object.create(defaults),\n    globalProperties = Object.create(polymerTheme);\n\n/**\n * @constructor\n * @param {string|Element} div - CSS selector or Element\n * @param {string} behaviorName - name of a behavior constructor from ./behaviors\n * @param {object} [margin] - optional canvas margins\n * @param {string} [margin.top]\n * @param {string} [margin.right='-200px']\n * @param {string} [margin.bottom]\n * @param {string} [margin.left]\n */\nfunction Hypergrid(div, behaviorFactory, margin) {\n    var self = this;\n\n    this.div = (typeof div === 'string') ? document.querySelector(div) : div;\n\n    addStylesheet('grid');\n\n    this.lastEdgeSelection = [0, 0];\n\n    this.lnfProperties = Object.create(globalProperties);\n\n    this.isWebkit = navigator.userAgent.toLowerCase().indexOf('webkit') > -1;\n    this.selectionModel = new SelectionModel(this);\n    this.localCellEditors = {};\n    this.cellEditors = Object.create(this.localCellEditors);\n    this.renderOverridesCache = {};\n    this.behavior = behaviorFactory(this);\n\n    //prevent the default context menu for appearing\n    this.div.oncontextmenu = function(event) {\n        event.preventDefault();\n        return false;\n    };\n\n    this.clearMouseDown();\n    this.dragExtent = new Point(0, 0);\n    this.numRows = 0;\n    this.numColumns = 0;\n\n    //install any plugins\n    this.pluginsDo(function(each) {\n        if (each.installOn) {\n            each.installOn(self);\n        }\n    });\n\n    margin = margin || {};\n    margin.top = margin.top || 0;\n    margin.right = margin.right || '-200px';\n    margin.bottom = margin.bottom || 0;\n    margin.left = margin.left || 0;\n\n    //initialize our various pieces\n    if (!themeInitialized) {\n        themeInitialized = true;\n        buildPolymerTheme();\n    }\n    this.initRenderer();\n    this.initCanvas(margin);\n    this.initScrollbars();\n    this.initLocalCellEditors();\n\n    //Register a listener for the copy event so we can copy our selected region to the pastebuffer if conditions are right.\n    document.body.addEventListener('copy', function(evt) {\n        self.checkClipboardCopy(evt);\n    });\n    this.getCanvas().resize();\n\n    this.dialog = new TableDialog(this);\n    //this.computeCellsBounds();\n\n    this.filter = new CustomFilter();\n}\n\nHypergrid.prototype = {\n    constructor: Hypergrid.prototype.constructor,\n\n    deprecated: deprecated,\n\n    /**\n     *\n     * A null object behavior serves as a place holder.\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n    behavior: null,\n\n    /**\n     * Cached resulan}\n     * @memberOf Hypergrid.prototype\n     */\n    isWebkit: true,\n\n    /**\n     * The pixel location of an initial mousedown click, either for editing a cell or for dragging a selection.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n    mouseDown: [],\n\n    /**\n     * The extent from the mousedown point during a drag operation.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n\n    dragExtent: null,\n\n    /**\n     * A float value between 0.0 - 1.0 of the vertical scroll position.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    vScrollValue: 0,\n\n    /**\n     * A float value between 0.0 - 1.0 of the horizontal scroll position.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    hScrollValue: 0,\n\n    /**\n     * @property {window.fin.rectangular} rectangular - Namespace for Point and Rectangle \"classes\" (constructors).\n     * @memberOf Hypergrid.prototype\n     */\n    rectangular: null,\n\n    /**\n     * @property {fin-hypergrid-selection-model} selectionModel - A [fin-hypergrid-selection-model](module-._selection-model.html) instance.\n     * @memberOf Hypergrid.prototype\n     */\n    selectionModel: null,\n\n    /**\n     * @property {fin-hypergrid-cell-editor} cellEditor - The current instance of [fin-hypergrid-cell-editor](module-cell-editors_base.html).\n     * @memberOf Hypergrid.prototype\n     */\n    cellEditor: null,\n\n    /**\n     * @property {fin-vampire-bar} sbHScroller - An instance of [fin-vampire-bar](http://datamadic.github.io/fin-vampire-bar/components/fin-vampire-bar/).\n     * @memberOf Hypergrid.prototype\n     */\n    sbHScroller: null,\n\n    /**\n     * @property {fin-vampire-bar} sbVScroller - An instance of [fin-vampire-bar](http://datamadic.github.io/fin-vampire-bar/components/fin-vampire-bar/).\n     * @memberOf Hypergrid.prototype\n     */\n    sbVScroller: null,\n\n    /**\n     * The previous value of sbVScrollVal.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    sbPrevVScrollValue: null,\n\n    /**\n     * The previous value of sbHScrollValue.\n     * @type {number}\n     * @memberOf Hypergrid.prototype\n     */\n    sbPrevHScrollValue: null,\n\n    /**\n     * The cache of singleton cellEditors.\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n    cellEditors: null,\n\n    /**\n     * is the short term memory of what column I might be dragging around\n     * @type {object}\n     * @memberOf Hypergrid.prototype\n     */\n\n    renderOverridesCache: {},\n\n    /**\n     * The pixel location of the current hovered cell.\n     * @type {Point}\n     * @memberOf Hypergrid.prototype\n     */\n    hoverCell: null,\n\n    scrollingNow: false,\n\n    lastEdgeSelection: null,\n\n    /**\n     * @memberOf Hypergrid.prototype\n    clear out the LRU cache of text widths\n     */\n    setAttribute: function(attribute, value) {\n        this.div.setAttribute(attribute, value);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n    clear out all state and data of the grid\n     */\n    reset: function() {\n        this.lastEdgeSelection = [0, 0];\n        this.lnfProperties = Object.create(globalProperties);\n        this.selectionModel = new SelectionModel(this);\n        this.cellEditors = Object.create(this.localCellEditors);\n        this.renderOverridesCache = {};\n        this.clearMouseDown();\n        this.dragExtent = new Point(0, 0);\n\n        this.numRows = 0;\n        this.numColumns = 0;\n\n        this.vScrollValue = 0;\n        this.hScrollValue = 0;\n\n        this.cellEditor = null;\n\n        this.sbPrevVScrollValue = null;\n        this.sbPrevHScrollValue = null;\n\n        this.hoverCell = null;\n        this.scrollingNow = false;\n        this.lastEdgeSelection = [0, 0];\n\n        this.behavior.reset();\n        this.getRenderer().reset();\n        this.getCanvas().resize();\n        this.behaviorChanged();\n    },\n\n    //resetTextWidthCache: function() {\n    //    textWidthCache = new LRUCache(2000);\n    //},\n\n    getProperties: function() {\n        return this.getPrivateState();\n    },\n\n    _getProperties: function() {\n        return this.lnfProperties;\n    },\n\n    computeCellsBounds: function() {\n        var renderer = this.getRenderer();\n        if (renderer) {\n            renderer.computeCellsBounds();\n        }\n    },\n\n    initCellEditor: function(cellEditor) {\n        this.localCellEditors[cellEditor.alias] = cellEditor;\n        cellEditor.grid = this;\n    },\n\n    initLocalCellEditors: function() {\n\n        var cellEditors = [\n            'Textfield',\n            'Choice',\n            //'Combo',\n            'Color',\n            'Date',\n            'Slider',\n            'Spinner',\n            'Filter'\n        ];\n\n        var self = this;\n        cellEditors.forEach(function(name) {\n            self.initCellEditor(new Hypergrid.cellEditors[name]);\n        });\n\n        this.localCellEditors.int = this.localCellEditors.spinner;\n        this.localCellEditors.float = this.localCellEditors.spinner;\n        this.localCellEditors.date = this.localCellEditors.date;\n        this.localCellEditors.string = this.localCellEditors.extfield;\n    },\n\n    toggleColumnPicker: function() {\n        this.behavior.toggleColumnPicker();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The pointer is over the given cell.\n     * @param {number} x - The x cell coordinate.\n     * @param {number} y - The y cell coordinate.\n     */\n    isHovered: function(x, y) {\n        var p = this.getHoverCell();\n        return p && p.x === x && p.y === y;\n    },\n\n    registerFormatter: function(name, func) {\n        Formatters[name] = func;\n    },\n\n    getFormatter: function(type) {\n        var formatter = Formatters[type];\n        if (formatter) {\n            return formatter;\n        }\n        return Formatters.default;\n    },\n\n    formatValue: function(type, value) {\n        var formatter = this.getFormatter(type);\n        return formatter(value);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns boolean} The pointer is hovering over the given column.\n     * @param {number} x - The horizontal cell coordinate.\n     */\n    isColumnHovered: function(x) {\n        var p = this.getHoverCell();\n        return p && p.x === x;\n    },\n\n    isRowResizeable: function() {\n        return this.resolveProperty('rowResize');\n    },\n\n    isCheckboxOnlyRowSelections: function() {\n        return this.resolveProperty('checkboxOnlyRowSelections');\n    },\n\n    /**\n     *\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The pointer is hovering over the row `y`.\n     * @param {number} y - The vertical cell coordinate.\n     */\n    isRowHovered: function(y) {\n        var p = this.getHoverCell();\n        return p && p.y === y;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Point} The cell over which the cursor is hovering.\n     */\n    getHoverCell: function() {\n        return this.hoverCell;\n    },\n\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the cell under the cursor.\n     * @param {Point} point\n     */\n    setHoverCell: function(point) {\n        var me = this.hoverCell;\n        var newPoint = new Point(point.x, point.y);\n        if (me && me.equals(newPoint)) {\n            return;\n        }\n        this.hoverCell = newPoint;\n        this.fireSyntheticOnCellEnterEvent(newPoint);\n        this.repaint();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for all hypergrids in this process.\n     * @param {object} properties - A simple properties hash.\n     */\n    addGlobalProperties: function(properties) {\n        //we check for existence to avoid race condition in initialization\n        if (!globalProperties) {\n            var self = this;\n            setTimeout(function() {\n                self.addGlobalProperties(properties);\n            }, 10);\n        } else {\n            this._addGlobalProperties(properties);\n        }\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for all hypergrids in this process.\n     * @param {object} properties - A simple properties hash.\n     * @private\n     */\n    _addGlobalProperties: function(properties) {\n        _(properties).each(function(property, key) {\n            globalProperties[key] = property;\n        });\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Utility function to push out properties if we change them.\n     * @param {object} properties - An object of various key value pairs.\n     */\n\n    refreshProperties: function() {\n        // this.canvas = this.shadowRoot.querySelector('fin-canvas');\n        //this.canvas = new Canvas(this.divCanvas, this.renderer); //TODO: Do we really need to be recreating it here?\n        this.checkScrollbarVisibility();\n        this.behavior.defaultRowHeight = null;\n        if (this.isColumnAutosizing()) {\n            this.behavior.autosizeAllColumns();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Ammend properties for this hypergrid only.\n     * @param {object} moreProperties - A simple properties hash.\n     */\n    addProperties: function(moreProperties) {\n        var properties = this.getProperties();\n        addDeepProperties(properties, moreProperties);\n        this.refreshProperties();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} The state object for remembering our state.\n     * @see [Memento pattern](http://en.wikipedia.org/wiki/Memento_pattern)\n     */\n    getPrivateState: function() {\n        return this.behavior.getPrivateState();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the state object to return to the given user configuration.\n     * @param {object} state - A memento object.\n     * @see [Memento pattern](http://en.wikipedia.org/wiki/Memento_pattern)\n     */\n    setState: function(state) {\n        var self = this;\n        this.behavior.setState(state);\n        setTimeout(function() {\n            self.behaviorChanged();\n            self.synchronizeScrollingBoundries();\n        }, 100);\n    },\n\n    getState: function() {\n        return this.behavior.getState();\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} The initial mouse position on a mouse down event for cell editing or a drag operation.\n     * @memberOf Hypergrid.prototype\n     */\n    getMouseDown: function() {\n        var last = this.mouseDown.length - 1;\n        if (last < 0) {\n            return null;\n        }\n        return this.mouseDown[last];\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Remove the last item from the mouse down stack.\n     */\n    popMouseDown: function() {\n        if (this.mouseDown.length !== 0) {\n            this.mouseDown.length = this.mouseDown.length - 1;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Empty out the mouse down stack.\n     */\n    clearMouseDown: function() {\n        this.mouseDown = [new Point(-1, -1)];\n        this.dragExtent = null;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     set the mouse point that initated a cell edit or drag operation\n     *\n     * @param {Point} point\n     */\n    setMouseDown: function(point) {\n        this.mouseDown.push(point);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Point} The extent point of the current drag selection rectangle.\n     */\n    getDragExtent: function() {\n        return this.dragExtent;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Sets the extent point of the current drag selection operation.\n     * @param {Point} point\n     */\n    setDragExtent: function(point) {\n        this.dragExtent = point;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Iterate over the plugins invoking the given function with each.\n     * @todo We need a new plugin mechanism!\n     * @param {function} func - The function to invoke on all the plugins.\n     */\n    pluginsDo: function(func) {\n        //TODO: We need a new plugin mechanism!\n        //var userPlugins = this.children.array();\n        //var pluginsTag = this.shadowRoot.querySelector('fin-plugins');\n        //\n        //var plugins = userPlugins;\n        //if (pluginsTag) {\n        //    var systemPlugins = pluginsTag.children.array();\n        //    plugins = systemPlugins.concat(plugins);\n        //}\n        //\n        //plugins.forEach(function(plugin) {\n        //    func(plugin);\n        //});\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The CellProvider is accessed through Hypergrid because Hypergrid is the mediator and should have ultimate control on where it comes from. The default is to delegate through the behavior object.\n     * @returns {fin-hypergrid-cell-provider}\n     */\n    getCellProvider: function() {\n        var provider = this.behavior.getCellProvider();\n        return provider;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc This function is a callback from the HypergridRenderer sub-component. It is called after each paint of the canvas.\n     */\n    gridRenderedNotification: function() {\n        this.updateRenderedSizes();\n        if (this.cellEditor) {\n            this.cellEditor.gridRenderedNotification();\n        }\n        this.checkColumnAutosizing();\n        this.fireSyntheticGridRenderedEvent();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The grid has just been rendered, make sure the column widths are optimal.\n     */\n    checkColumnAutosizing: function() {\n        var behavior = this.behavior;\n        behavior.autoSizeRowNumberColumn();\n        if (this.isColumnAutosizing()) {\n            behavior.checkColumnAutosizing(false);\n        }\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Notify the GridBehavior how many rows and columns we just rendered.\n     */\n    updateRenderedSizes: function() {\n        //add one to each of these values as we want also to include\n        //the columns and rows that are partially visible\n        this.behavior.setRenderedColumnCount(this.getVisibleColumns() + 1);\n        this.behavior.setRenderedRowCount(this.getVisibleRows() + 1);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Conditionally copy to clipboard.\n     * @desc If we have focus, copy our current selection data to the system clipboard.\n     * @param {event} event - The copy system event.\n     */\n    checkClipboardCopy: function(event) {\n        if (this.hasFocus()) {\n            event.preventDefault();\n            var csvData = this.getSelectionAsTSV();\n            event.clipboardData.setData('text/plain', csvData);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have any selections.\n     */\n    hasSelections: function() {\n        if (!this.getSelectionModel) {\n            return; // were not fully initialized yet\n        }\n        return this.selectionModel.hasSelections();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {string} Tab separated value string from the selection and our data.\n     */\n    getSelectionAsTSV: function() {\n        var sm = this.selectionModel;\n        if (sm.hasSelections()) {\n            var selections = this.getSelectionMatrix();\n            selections = selections[selections.length - 1];\n            return this.getMatrixSelectionAsTSV(selections);\n        } else if (sm.hasRowSelections()) {\n            return this.getMatrixSelectionAsTSV(this.getRowSelectionMatrix());\n        } else if (sm.hasColumnSelections()) {\n            return this.getMatrixSelectionAsTSV(this.getColumnSelectionMatrix());\n        }\n    },\n\n    getMatrixSelectionAsTSV: function(selections) {\n        //only use the data from the last selection\n        if (selections.length) {\n            var width = selections.length,\n                height = selections[0].length,\n                area = width * height,\n                collector = [];\n\n            //disallow if selection is too big\n            if (area > 20000) {\n                alert('selection size is too big to copy to the paste buffer'); // eslint-disable-line no-alert\n                return '';\n            }\n\n            for (var h = 0; h < height; h++) {\n                for (var w = 0; w < width; w++) {\n                    collector.push(selections[w][h]);\n                    if (w < width) {\n                        collector.push('\\t');\n                    }\n                }\n                if (h < height) {\n                    collector.push('\\n');\n                }\n            }\n\n            var result = collector.join('');\n\n            return result;\n        }\n        return '';\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have focus.\n     */\n    hasFocus: function() {\n        return this.getCanvas().hasFocus();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear all the selections.\n     */\n    clearSelections: function() {\n        var dontClearRows = this.isCheckboxOnlyRowSelections();\n        this.selectionModel.clear(dontClearRows);\n        this.clearMouseDown();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent selection.\n     */\n    clearMostRecentSelection: function() {\n        var dontClearRows = this.isCheckboxOnlyRowSelections();\n        this.selectionModel.clearMostRecentSelection(dontClearRows);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent column selection.\n     */\n    clearMostRecentColumnSelection: function() {\n        this.selectionModel.clearMostRecentColumnSelection();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Clear the most recent row selection.\n     */\n    clearMostRecentRowSelection: function() {\n        //this.selectionModel.clearMostRecentRowSelection(); // commented off as per GRID-112\n    },\n\n    clearRowSelection: function() {\n        this.selectionModel.clearRowSelection();\n        this.behavior.getDataModel().getComponent().clearSelectedData();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Select given region.\n     * @param {number} ox - origin x\n     * @param {number} oy - origin y\n     * @param {number} ex - extent x\n     * @param {number} ex - extent y\n     */\n    select: function(ox, oy, ex, ey) {\n        if (ox < 0 || oy < 0) {\n            //we don't select negative area\n            //also this means there is no origin mouse down for a selection rect\n            return;\n        }\n        this.selectionModel.select(ox, oy, ex, ey);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} Given point is selected.\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     */\n    isSelected: function(x, y) {\n        return this.selectionModel.isSelected(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given column is selected anywhere in the entire table.\n     * @param {number} col - The column index.\n     */\n    isCellSelectedInRow: function(col) {\n        var selectionModel = this.selectionModel;\n        var isSelected = selectionModel.isCellSelectedInRow(col);\n        return isSelected;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given row is selected anywhere in the entire table.\n     * @param {number} row - The row index.\n     */\n    isCellSelectedInColumn: function(row) {\n        var selectionModel = this.selectionModel;\n        var isSelected = selectionModel.isCellSelectedInColumn(row);\n        return isSelected;\n    },\n\n    /** @deprecated Use `.selectionModel` property instead.\n     * @memberOf Hypergrid.prototype\n     * @returns {SelectionModel} The selection model.\n     */\n    getSelectionModel: function() {\n        return this.deprecated('selectionModel', { since: '0.2' });\n    },\n\n    /** @deprecated Use `.behavior` property instead.\n     * @memberOf Hypergrid.prototype\n     * @returns {Behavior} The behavior (model).\n     */\n    getBehavior: function() {\n        return this.deprecated('behavior', { since: '0.2' });\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Set the Behavior (model) object for this grid control.\n     * @desc This can be done dynamically.\n     * @param {Behavior} The behavior (model).\n     */\n    setBehavior: function(newBehavior) {\n\n        this.behavior = newBehavior;\n        this.behavior.setGrid(this);\n\n        this.behavior.changed = this.behaviorChanged.bind(this);\n        this.behavior.shapeChanged = this.behaviorShapeChanged.bind(this);\n        this.behavior.stateChanged = this.behaviorStateChanged.bind(this);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc I've been notified that the behavior has changed.\n     */\n    behaviorChanged: function() {\n        if (this.numColumns !== this.getColumnCount() || this.numRows !== this.getRowCount()) {\n            this.numColumns = this.getColumnCount();\n            this.numRows = this.getRowCount();\n            this.behaviorShapeChanged();\n        }\n        this.computeCellsBounds();\n        this.repaint();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Rectangle} My bounds.\n     */\n    getBounds: function() {\n        var renderer = this.getRenderer();\n        return renderer && renderer.getBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {string} The value of a lnf property.\n     * @param {string} key - A look-and-feel key.\n     */\n    resolveProperty: function(key) {\n        var keys = key.split('.');\n        var prop = this.getProperties();\n        while (keys.length) { prop = prop[keys.shift()]; }\n        return prop;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The dimensions of the grid data have changed. You've been notified.\n     */\n    behaviorShapeChanged: function() {\n        this.synchronizeScrollingBoundries();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The dimensions of the grid data have changed. You've been notified.\n     */\n    behaviorStateChanged: function() {\n        this.getRenderer().computeCellsBounds();\n        this.repaint();\n    },\n\n    repaint: function() {\n        var now = this.resolveProperty('repaintImmediately');\n        var canvas = this.getCanvas();\n        if (canvas) {\n            if (now === true) {\n                canvas.paintNow();\n            } else {\n                canvas.repaint();\n            }\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Paint immediately in this microtask.\n     */\n    paintNow: function() {\n        var canvas = this.getCanvas();\n        canvas.paintNow();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} In HiDPI mode (has an attribute as such).\n     */\n    useHiDPI: function() {\n        return this.resolveProperty('useHiDPI') !== false;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Initialize drawing surface.\n     * @private\n     */\n    initCanvas: function(margin) {\n\n        var self = this;\n\n        var divCanvas = this.divCanvas = document.createElement('div');\n        this.div.appendChild(divCanvas);\n        this.canvas = new Canvas(divCanvas, this.renderer);\n\n        var style = divCanvas.style;\n        style.position = 'absolute';\n        style.top = margin.top;\n        style.right = margin.right;\n        style.bottom = margin.bottom;\n        style.left = margin.left;\n\n        this.canvas.resizeNotification = function() {\n            self.resized();\n        };\n\n        this.addFinEventListener('fin-canvas-mousemove', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseMove(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-mousedown', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.keys = e.detail.keys;\n            mouseEvent.primitiveEvent = e;\n            self.mouseDownState = mouseEvent;\n            self.delegateMouseDown(mouseEvent);\n            self.fireSyntheticMouseDownEvent(mouseEvent);\n            self.repaint();\n        });\n\n\n        // this.addFinEventListener('fin-canvas-click', function(e) {\n        //     if (self.resolveProperty('readOnly')) {\n        //         return;\n        //     }\n        //     //self.stopEditing();\n        //     var mouse = e.detail.mouse;\n        //     var mouseEvent = self.getGridCellFromMousePoint(mouse);\n        //     mouseEvent.primitiveEvent = e;\n        //     self.fireSyntheticClickEvent(mouseEvent);\n        // });\n\n        this.addFinEventListener('fin-canvas-mouseup', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.dragging = false;\n            if (self.isScrollingNow()) {\n                self.setScrollingNow(false);\n            }\n            if (self.columnDragAutoScrolling) {\n                self.columnDragAutoScrolling = false;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseUp(mouseEvent);\n            if (self.mouseDownState) {\n                self.fireSyntheticButtonPressedEvent(self.mouseDownState);\n            }\n            self.mouseDownState = null;\n            self.fireSyntheticMouseUpEvent(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-dblclick', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.fireSyntheticDoubleClickEvent(mouseEvent, e);\n            self.delegateDoubleClick(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-tap', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            //self.stopEditing();\n            var mouse = e.detail.mouse;\n            var tapEvent = self.getGridCellFromMousePoint(mouse);\n            tapEvent.primitiveEvent = e;\n            tapEvent.keys = e.detail.keys;\n            self.fireSyntheticClickEvent(tapEvent);\n            self.delegateTap(tapEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-drag', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.dragging = true;\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e;\n            self.delegateMouseDrag(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-keydown', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            if (self.resolveProperty('editOnKeydown')) {\n                var char = e.detail.char,\n                    isVisibleChar, isDeleteChar, currentCell, editor;\n\n                if (\n                    !self.isEditing() &&\n                    (\n                        char === 'F2' ||\n                        (isVisibleChar = char.length === 1) ||\n                        (isDeleteChar = char === 'DELETE' || char === 'BACKSPACE')\n                    )\n                ) {\n                    currentCell = self.selectionModel.getLastSelection();\n                    if (currentCell) {\n                        editor = self.activateEditor(currentCell.origin.x, currentCell.origin.y);\n                        if (editor instanceof Hypergrid.cellEditors.Simple) {\n                            if (isVisibleChar) {\n                                editor.input.value = char;\n                            } else if (isDeleteChar) {\n                                editor.input.value = '';\n                            }\n                            e.detail.primitiveEvent.preventDefault();\n                        }\n                    }\n                }\n            }\n            self.fireSyntheticKeydownEvent(e);\n            self.delegateKeyDown(e);\n        });\n\n        this.addFinEventListener('fin-canvas-keyup', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            self.fireSyntheticKeyupEvent(e);\n            self.delegateKeyUp(e);\n        });\n\n        this.addFinEventListener('fin-canvas-track', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            if (self.dragging) {\n                return;\n            }\n            var primEvent = e.detail.primitiveEvent;\n            if (Math.abs(primEvent.dy) > Math.abs(primEvent.dx)) {\n                if (primEvent.yDirection > 0) {\n                    self.scrollVBy(-2);\n                } else if (primEvent.yDirection < -0) {\n                    self.scrollVBy(2);\n                }\n            } else {\n                if (primEvent.xDirection > 0) {\n                    self.scrollHBy(-1);\n                } else if (primEvent.xDirection < -0) {\n                    self.scrollHBy(1);\n                }\n            }\n        });\n\n        // this.addFinEventListener('fin-canvas-holdpulse', function(e) {\n        //     console.log('holdpulse');\n        //     if (self.resolveProperty('readOnly')) {\n        //         return;\n        //     }\n        //     var mouse = e.detail.mouse;\n        //     var mouseEvent = self.getGridCellFromMousePoint(mouse);\n        //     mouseEvent.primitiveEvent = e;\n        //     self.delegateHoldPulse(mouseEvent);\n        // });\n\n        this.addFinEventListener('fin-canvas-wheelmoved', function(e) {\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateWheelMoved(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-mouseout', function(e) {\n            if (self.resolveProperty('readOnly')) {\n                return;\n            }\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateMouseExit(mouseEvent);\n        });\n\n        this.addFinEventListener('fin-canvas-context-menu', function(e) {\n            var mouse = e.detail.mouse;\n            var mouseEvent = self.getGridCellFromMousePoint(mouse);\n            mouseEvent.primitiveEvent = e.detail.primitiveEvent;\n            self.delegateContextMenu(mouseEvent);\n        });\n\n        this.div.removeAttribute('tabindex');\n\n    },\n\n    convertViewPointToDataPoint: function(viewPoint) {\n        return this.behavior.convertViewPointToDataPoint(viewPoint);\n    },\n\n    convertDataPointToViewPoint: function(dataPoint) {\n        return this.behavior.convertDataPointToViewPoint(dataPoint);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Add an event listener to me.\n     * @param {string} eventName - The type of event we are interested in.\n     * @param {function} callback - The event handler.\n     */\n    addFinEventListener: function(eventName, callback) {\n        this.canvas.addEventListener(eventName, callback);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Set for `scrollingNow` field.\n     * @param {boolean} isItNow - The type of event we are interested in.\n     */\n    setScrollingNow: function(isItNow) {\n        this.scrollingNow = isItNow;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The `scrollingNow` field.\n     */\n    isScrollingNow: function() {\n        return this.scrollingNow;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The index of the column divider under the mouse coordinates.\n     * @param {MouseEvent} mouseEvent - The event to interogate.\n     */\n    overColumnDivider: function(mouseEvent) {\n        var x = mouseEvent.primitiveEvent.detail.mouse.x;\n        var whichCol = this.getRenderer().overColumnDivider(x);\n        return whichCol;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The index of the row divider under the mouse coordinates.\n     * @param {MouseEvent} mouseEvent - The event to interogate.\n     */\n    overRowDivider: function(mouseEvent) {\n        var y = mouseEvent.primitiveEvent.detail.mouse.y;\n        var which = this.getRenderer().overRowDivider(y);\n        return which;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Switch the cursor for the grid.\n     * @param {string} cursorName - A well know cursor name.\n     * @see [cursor names](http://www.javascripter.net/faq/stylesc.htm)\n     */\n    beCursor: function(cursorName) {\n        if (!cursorName) {\n            cursorName = 'default';\n        }\n        this.div.style.cursor = cursorName;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate the wheel moved event to the behavior.\n     * @param {Event} event - The pertinent event.\n     */\n    delegateWheelMoved: function(event) {\n        this.behavior.onWheelMoved(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseExit to the behavior (model).\n     * @param {Event} event - The pertinent event.\n     */\n    delegateMouseExit: function(event) {\n        this.behavior.handleMouseExit(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseExit to the behavior (model).\n     * @param {Event} event - The pertinent event.\n     */\n    delegateContextMenu: function(event) {\n       this. behavior.onContextMenu(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate MouseMove to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseMove: function(mouseDetails) {\n        this.behavior.onMouseMove(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mousedown to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseDown: function(mouseDetails) {\n        this.behavior.handleMouseDown(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mouseup to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseUp: function(mouseDetails) {\n        this.behavior.onMouseUp(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate tap to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateTap: function(mouseDetails) {\n        this.behavior.onTap(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate mouseDrag to the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateMouseDrag: function(mouseDetails) {\n        this.behavior.onMouseDrag(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc We've been doubleclicked on. Delegate through the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateDoubleClick: function(mouseDetails) {\n        this.behavior.onDoubleClick(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Delegate holdpulse through the behavior (model).\n     * @param {mouseDetails} mouseDetails - An enriched mouse event from fin-canvas.\n     */\n    delegateHoldPulse: function(mouseDetails) {\n        this.behavior.onHoldPulse(this, mouseDetails);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Generate a function name and call it on self.\n     * @desc This should also be delegated through Behavior keeping the default implementation here though.\n     * @param {event} event - The pertinent event.\n     */\n    delegateKeyDown: function(event) {\n        this.behavior.onKeyDown(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Generate a function name and call it on self.\n     * @desc This should also be delegated through Behavior keeping the default implementation here though.\n     * @param {event} event - The pertinent event.\n     */\n    delegateKeyUp: function(event) {\n        this.behavior.onKeyUp(this, event);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Shut down the current cell editor.\n     */\n    stopEditing: function() {\n        if (this.cellEditor && this.isEditing()) {\n            if (this.cellEditor.stopEditing) {\n                this.cellEditor.stopEditing();\n            }\n            this.cellEditor = null;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Register a cell editor.\n     * @desc This is typically called from within a cell-editor's `installOn` method, when it is being initialized as a plugin.\n     * @param {string} alias - The name/id of the cell editor.\n     * @param {fin-hypergrid-cell-editor-base} cellEditor - see [fin-hypergrid-cell-editor-base](module-cell-editors_base.html)\n     */\n    registerCellEditor: function(alias, cellEditor) {\n        this.cellEditors[alias] = cellEditor;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Rectangle} The pixel coordinates of just the center 'main\" data area.\n     */\n    getDataBounds: function() {\n        var colDNDHackWidth = 200; //this was a hack to help with column dnd, need to factor this into a shared variable\n        var b = this.canvas.bounds;\n\n        //var x = this.getRowNumbersWidth();\n        // var y = behavior.getFixedRowsHeight() + 2;\n\n        var result = new Rectangle(0, 0, b.origin.x + b.extent.x - colDNDHackWidth, b.origin.y + b.extent.y);\n        return result;\n    },\n\n    getRowNumbersWidth: function() {\n        if (this.isShowRowNumbers()) {\n            return this.getRenderer().getRowNumbersWidth();\n        } else {\n            return 0;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Canvas} Our fin-canvas instance.\n     */\n    getCanvas: function() {\n        return this.canvas;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Open the given cell-editor at the provided model coordinates.\n     * @param {string} cellEditor - The specific cell editor to use.\n     * @param {Point} coordinates - The pixel locaiton of the cell to edit at.\n     */\n    editAt: function(cellEditor, coordinates) {\n\n        this.cellEditor = cellEditor;\n\n        var cell = coordinates.gridCell;\n\n        var x = cell.x;\n        var y = cell.y;\n\n        if (x >= 0 && y >= 0) {\n            var editPoint = new Point(x, y);\n            this.setMouseDown(editPoint);\n            this.setDragExtent(new Point(0, 0));\n            cellEditor.beginEditAt(editPoint);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given column is fully visible.\n     * @param {number} columnIndex - The column index in question.\n     * @return {boolan} Visible.\n     */\n    isColumnVisible: function(columnIndex) {\n        return this.getRenderer().isColumnVisible(columnIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given row is fully visible.\n     * @param {number} rowIndex - The row index in question.\n     * @return {boolan} Visible.\n     */\n    isDataRowVisible: function(rowIndex) {\n        return this.getRenderer().isRowVisible(rowIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The given cell is fully is visible.\n     * @param {number} columnIndex - The column index in question.\n     * @param {number} rowIndex - The row index in question.\n     * @return {boolean} Data is visible.\n     */\n    isDataVisible: function(columnIndex, rowIndex) {\n        return this.isDataRowVisible(rowIndex) && this.isColumnVisible(columnIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll in the `offsetX` direction if column index `colIndex` is not visible.\n     * @param {number} colIndex - The column index in question.\n     * @param {number} offsetX - The direction and magnitude to scroll if we need to.\n     * @return {boolean} Column is visible.\n     */\n    insureModelColIsVisible: function(colIndex, offsetX) {\n        var maxCols = this.getColumnCount() - 1, // -1 excludes partially visible columns\n            indexToCheck = colIndex + (offsetX > 0),\n            visible = !this.isColumnVisible(indexToCheck) || colIndex === maxCols;\n\n        if (visible) {\n            //the scroll position is the leftmost column\n            this.scrollBy(offsetX, 0);\n        }\n\n        return visible;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll in the offsetY direction if column index c is not visible.\n     * @param {number} rowIndex - The column index in question.\n     * @param {number} offsetX - The direction and magnitude to scroll if we need to.\n     * @return {boolean} Row is visible.\n     */\n    insureModelRowIsVisible: function(rowIndex, offsetY) {\n        var maxRows = this.getRowCount() - 1, // -1 excludes partially visible rows\n            indexToCheck = rowIndex + (offsetY > 0),\n            visible = !this.isDataRowVisible(indexToCheck) || rowIndex === maxRows;\n\n        if (visible) {\n            //the scroll position is the topmost row\n            this.scrollBy(0, offsetY);\n        }\n\n        return visible;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll horizontal and vertically by the provided offsets.\n     * @param {number} offsetX - Scroll in the x direction this much.\n     * @param {number} offsetY - Scroll in the y direction this much.\n     */\n    scrollBy: function(offsetX, offsetY) {\n        this.scrollHBy(offsetX);\n        this.scrollVBy(offsetY);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll vertically by the provided offset.\n     * @param {number} offsetY - Scroll in the y direction this much.\n     */\n    scrollVBy: function(offsetY) {\n        var max = this.sbVScroller.range.max;\n        var oldValue = this.getVScrollValue();\n        var newValue = Math.min(max, Math.max(0, oldValue + offsetY));\n        if (newValue !== oldValue) {\n            this.setVScrollValue(newValue);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Scroll horizontally by the provided offset.\n     * @param {number} offsetX - Scroll in the x direction this much.\n     */\n    scrollHBy: function(offsetX) {\n        var max = this.sbHScroller.range.max;\n        var oldValue = this.getHScrollValue();\n        var newValue = Math.min(max, Math.max(0, oldValue + offsetX));\n        if (newValue !== oldValue) {\n            this.setHScrollValue(newValue);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Answer which data cell is under a pixel value mouse point.\n     * @param {mousePoint} mouse - The mouse point to interrogate.\n     */\n\n    getGridCellFromMousePoint: function(mouse) {\n        var cell = this.getRenderer().getGridCellFromMousePoint(mouse);\n        return cell;\n    },\n\n    /**\n     * @returns {Rectangle} The pixel based bounds rectangle given a data cell point.\n     * @param {Point} cell - The pixel location of the mouse.\n     * @memberOf Hypergrid.prototype\n     */\n    getBoundsOfCell: function(cell) {\n        var b = this.getRenderer().getBoundsOfCell(cell);\n\n        //we need to convert this to a proper rectangle\n        var newBounds = new Rectangle(b.x, b.y, b.width, b.height);\n        return newBounds;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc This is called by the fin-canvas when a resize occurs.\n     */\n    resized: function() {\n        this.synchronizeScrollingBoundries();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A click event occured.\n     * @desc Determine the cell and delegate to the behavior (model).\n     * @param {MouseEvent} event - The mouse event to interrogate.\n     */\n    cellClicked: function(event) {\n        var cell = event.gridCell;\n\n        //click occurred in background area\n        if (\n            cell.x <= this.getColumnCount() &&\n            cell.y <= this.getRowCount()\n        ) {\n            var hovered = this.getHoverCell(),\n                x = hovered.x,\n                y = hovered.y;\n\n            // if (x >= 0) {\n            //     x = behavior.translateColumnIndex(x + this.getHScrollValue());\n            // }\n\n            if (y >= 0) {\n                y += this.getVScrollValue();\n            }\n\n            this.behavior.cellClicked(new Point(x, y), event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} atBottom - this value is in the \"bottom\" totals area\n     */\n    setTotalsValueNotification: function(x, y, value, atBottom) {\n        this.fireSyntheticSetTotalsValue(x, y, value, atBottom);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} atBottom - this value is in the \"bottom\" totals area\n     */\n    fireSyntheticSetTotalsValue: function(x, y, value, atBottom) {\n        var clickEvent = new CustomEvent('fin-set-totals-value', {\n            detail: {\n                x: x,\n                y: y,\n                value: value,\n                area: atBottom ? 'bottom' : 'top'\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyUpEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-keyup', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyDownEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-keydown', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorKeyPressEvent: function(inputControl, keyEvent) {\n        var clickEvent = new CustomEvent('fin-editor-keypress', {\n            detail: {\n                input: inputControl,\n                keyEvent: keyEvent\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticEditorDataChangeEvent: function(inputControl, oldValue, newValue) {\n        var clickEvent = new CustomEvent('fin-editor-data-change', {\n            detail: {\n                input: inputControl,\n                oldValue: oldValue,\n                newValue: newValue\n            },\n            cancelable: true\n        });\n        return this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-row-selection-changed` event.\n     */\n    fireSyntheticRowSelectionChangedEvent: function() {\n        var selectionEvent = new CustomEvent('fin-row-selection-changed', {\n            detail: {\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections(),\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n\n    fireSyntheticColumnSelectionChangedEvent: function() {\n        var selectionEvent = new CustomEvent('fin-column-selection-changed', {\n            detail: {\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and dispatch a `fin-selection-changed` event.\n     */\n    selectionChanged: function() {\n        var selectedRows = this.getSelectedRows();\n        var selectionEvent = new CustomEvent('fin-selection-changed', {\n            detail: {\n                rows: selectedRows,\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections(),\n            }\n        });\n        this.canvas.dispatchEvent(selectionEvent);\n    },\n\n\n    getRowSelection: function() {\n        var c, column, self = this,\n            selectedRowIndexes = this.selectionModel.getSelectedRows(),\n            numCols = this.getColumnCount(),\n            result = {};\n\n        function setValue(selectedRowIndex, r) {\n            column[r] = valueOrFunctionExecute(self.getValue(c, selectedRowIndex));\n        }\n\n        for (c = 0; c < numCols; c++) {\n            column = new Array(selectedRowIndexes.length);\n            result[this.getField(c)] = column;\n            selectedRowIndexes.forEach(setValue);\n        }\n\n        return result;\n    },\n\n    getRowSelectionMatrix: function() {\n        var c, self = this,\n            selectedRowIndexes = this.selectionModel.getSelectedRows(),\n            numCols = this.getColumnCount(),\n            result = new Array(numCols);\n\n        function getValue(selectedRowIndex, r) {\n            result[c][r] = valueOrFunctionExecute(self.getValue(c, selectedRowIndex));\n        }\n\n        for (c = 0; c < numCols; c++) {\n            result[c] = new Array(selectedRowIndexes.length);\n            selectedRowIndexes.forEach(getValue);\n        }\n\n        return result;\n    },\n\n    getColumnSelectionMatrix: function() {\n        var selectedColumnIndexes = this.getSelectedColumns();\n        var numRows = this.getRowCount();\n        var result = new Array(selectedColumnIndexes.length);\n        var self = this;\n        selectedColumnIndexes.forEach(function(selectedColumnIndex, c) {\n            result[c] = new Array(numRows);\n            for (var r = 0; r < numRows; r++) {\n                result[c][r] = valueOrFunctionExecute(self.getValue(selectedColumnIndex, r));\n            }\n        });\n        return result;\n    },\n\n    getColumnSelection: function() {\n        var selectedColumnIndexes = this.getSelectedColumns();\n        var result = {};\n        var rowCount = this.getRowCount();\n        var self = this;\n        selectedColumnIndexes.forEach(function(selectedColumnIndex) {\n            var column = new Array(rowCount);\n            result[self.getField(selectedColumnIndex)] = column;\n            for (var r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(self.getValue(selectedColumnIndex, r));\n            }\n        });\n        return result;\n    },\n\n    getSelection: function() {\n        var self = this;\n        var selections = this.getSelections();\n        var result = new Array(selections.length);\n        selections.forEach(function(selectionRect, i) {\n            result[i] = self._getSelection(selectionRect);\n        });\n        return result;\n    },\n\n    _getSelection: function(rect) {\n        rect = normalizeRect(rect);\n        var colCount = rect.extent.x + 1;\n        var rowCount = rect.extent.y + 1;\n        var ox = rect.origin.x;\n        var oy = rect.origin.y;\n        var result = {};\n        var r;\n        for (var c = 0; c < colCount; c++) {\n            var column = new Array(rowCount);\n            result[this.getField(c + ox)] = column;\n            for (r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(this.getValue(ox + c, oy + r));\n            }\n        }\n        return result;\n    },\n\n    getSelectionMatrix: function() {\n        var self = this;\n        var selections = this.getSelections();\n        var result = new Array(selections.length);\n        selections.forEach(function(selectionRect, i) {\n            result[i] = self._getSelectionMatrix(selectionRect);\n        });\n        return result;\n    },\n\n    _getSelectionMatrix: function(rect) {\n        rect = normalizeRect(rect);\n        var colCount = rect.extent.x + 1;\n        var rowCount = rect.extent.y + 1;\n        var ox = rect.origin.x;\n        var oy = rect.origin.y;\n        var result = [];\n        for (var c = 0; c < colCount; c++) {\n            var column = new Array(rowCount);\n            result[c] = column;\n            for (var r = 0; r < rowCount; r++) {\n                column[r] = valueOrFunctionExecute(this.getValue(ox + c, oy + r));\n            }\n        }\n        return result;\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-context-menu` event\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticContextMenuEvent: function(e) {\n        e.gridCell = this.convertViewPointToDataPoint(e.gridCell);\n        var event = new CustomEvent('fin-context-menu', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    fireSyntheticMouseUpEvent: function(e) {\n        var event = new CustomEvent('fin-mouseup', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    fireSyntheticMouseDownEvent: function(e) {\n        this.stopEditing();\n        var event = new CustomEvent('fin-mousedown', {\n            detail: {\n                gridCell: e.gridCell,\n                mousePoint: e.mousePoint,\n                viewPoint: e.viewPoint,\n                primitiveEvent: e.primitiveEvent,\n                rows: this.getSelectedRows(),\n                columns: this.getSelectedColumns(),\n                selections: this.selectionModel.getSelections()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n    },\n\n    isViewableButton: function(c, r) {\n        return this.getRenderer().isViewableButton(c, r);\n    },\n\n    fireSyntheticButtonPressedEvent: function(evt) {\n        var dataCell = evt.dataCell;\n        var gridCell = evt.gridCell;\n        if (this.isViewableButton(dataCell.x, dataCell.y)) {\n            var event = new CustomEvent('fin-button-pressed', {\n                detail: {\n                    gridCell: gridCell\n                }\n            });\n            this.canvas.dispatchEvent(event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-keydown` event.\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticKeydownEvent: function(keyEvent) {\n        var clickEvent = new CustomEvent('fin-keydown', {\n            detail: keyEvent.detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-keyup` event.\n     * @param {keyEvent} event - The canvas event.\n     */\n    fireSyntheticKeyupEvent: function(keyEvent) {\n        var clickEvent = new CustomEvent('fin-keyup', {\n            detail: keyEvent.detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticFilterAppliedEvent: function(details) {\n        var event = new CustomEvent('fin-filter-applied', {\n            detail: details\n        });\n        if (this.canvas) {\n            this.canvas.dispatchEvent(event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-enter` event\n     * @param {Point} cell - The pixel location of the cell in which the click event occurred.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticOnCellEnterEvent: function(cell) {\n        var detail = {\n            gridCell: cell,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-cell-enter', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    fireSyntheticGroupsChangedEvent: function(groups) {\n        var detail = {\n            groups: groups,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-groups-changed', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-exit` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticOnCellExitEvent: function(cell) {\n        var detail = {\n            gridCell: cell,\n            time: Date.now(),\n            grid: this\n        };\n        var clickEvent = new CustomEvent('fin-cell-exit', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-cell-click` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticClickEvent: function(mouseEvent) {\n        var cell = mouseEvent.gridCell;\n        var detail = {\n            gridCell: cell,\n            mousePoint: mouseEvent.mousePoint,\n            keys: mouseEvent.keys,\n            primitiveEvent: mouseEvent,\n            time: Date.now(),\n            grid: this\n        };\n        this.behavior.enhanceDoubleClickEvent(detail);\n        var clickEvent = new CustomEvent('fin-click', {\n            detail: detail\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a `fin-double-click` event.\n     * @param {Point} cell - The pixel location of the cell in which the click event occured.\n     * @param {MouseEvent} event - The system mouse event.\n     */\n    fireSyntheticDoubleClickEvent: function(mouseEvent) {\n        this.stopEditing();\n        var cell = mouseEvent.gridCell;\n        var detail = {\n            gridCell: cell,\n            mousePoint: mouseEvent.mousePoint,\n            time: Date.now(),\n            grid: this\n        };\n        this.behavior.enhanceDoubleClickEvent(mouseEvent);\n        var clickEvent = new CustomEvent('fin-double-click', {\n            detail: detail\n        });\n        this.behavior.cellDoubleClicked(cell, mouseEvent);\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a rendered event.\n     */\n    fireSyntheticGridRenderedEvent: function() {\n        var event = new CustomEvent('fin-grid-rendered', {\n            detail: {\n                source: this,\n                time: Date.now()\n            }\n        });\n        if (this.canvas) {\n            this.canvas.dispatchEvent(event);\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a scroll event.\n     * @param {string} type - Should be either `fin-scroll-x` or `fin-scroll-y`.\n     * @param {number} oldValue - The old scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    fireScrollEvent: function(type, oldValue, newValue) {\n        var event = new CustomEvent(type, {\n            detail: {\n                oldValue: oldValue,\n                value: newValue,\n                time: Date.now()\n            }\n        });\n        this.canvas.dispatchEvent(event);\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the vertical scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    setVScrollValue: function(y) {\n        y = Math.round(y);\n        var max = this.sbVScroller.range.max;\n        y = Math.min(max, Math.max(0, y));\n        var self = this;\n        if (y !== this.vScrollValue) {\n            this.behavior._setScrollPositionY(y);\n            var oldY = this.vScrollValue;\n            this.vScrollValue = y;\n            this.scrollValueChangedNotification();\n            setTimeout(function() {\n                // self.sbVRangeAdapter.subjectChanged();\n                self.fireScrollEvent('fin-scroll-y', oldY, y);\n            });\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @return {number} The vertical scroll value.\n     */\n    getVScrollValue: function() {\n        return this.vScrollValue;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the horizontal scroll value.\n     * @param {number} newValue - The new scroll value.\n     */\n    setHScrollValue: function(x) {\n        x = Math.round(x);\n        var max = this.sbHScroller.range.max;\n        x = Math.min(max, Math.max(0, x));\n        var self = this;\n        if (x !== this.hScrollValue) {\n            this.behavior._setScrollPositionX(x);\n            var oldX = this.hScrollValue;\n            this.hScrollValue = x;\n            this.scrollValueChangedNotification();\n            setTimeout(function() {\n                //self.sbHRangeAdapter.subjectChanged();\n                self.fireScrollEvent('fin-scroll-x', oldX, x);\n                self.synchronizeScrollingBoundries();\n            });\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns The vertical scroll value.\n     */\n    getHScrollValue: function() {\n        return this.hScrollValue;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Request input focus.\n     */\n    takeFocus: function() {\n        if (this.isEditing()) {\n            this.stopEditing();\n        } else {\n            this.getCanvas().takeFocus();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Request focus for our cell editor.\n     */\n    editorTakeFocus: function() {\n        if (this.cellEditor) {\n            return this.cellEditor.takeFocus();\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} We have a currently active cell editor.\n     */\n    isEditing: function() {\n        return this.cellEditor && this.cellEditor.isEditing;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Initialize the scroll bars.\n     */\n    initScrollbars: function() {\n\n        var self = this;\n\n        var horzBar = new FinBar({\n            orientation: 'horizontal',\n            onchange: self.setHScrollValue.bind(self),\n            cssStylesheetReferenceElement: this.div\n        });\n\n        var vertBar = new FinBar({\n            orientation: 'vertical',\n            onchange: self.setVScrollValue.bind(self),\n            paging: {\n                up: self.pageUp.bind(self),\n                down: self.pageDown.bind(self)\n            }\n        });\n\n        this.sbHScroller = horzBar;\n        this.sbVScroller = vertBar;\n\n        var hPrefix = this.resolveProperty('hScrollbarClassPrefix');\n        var vPrefix = this.resolveProperty('vScrollbarClassPrefix');\n\n        if (hPrefix && hPrefix !== '') {\n            this.sbHScroller.classPrefix = hPrefix;\n        }\n\n        if (vPrefix && vPrefix !== '') {\n            this.sbVScroller.classPrefix = vPrefix;\n        }\n\n        this.div.appendChild(horzBar.bar);\n        this.div.appendChild(vertBar.bar);\n\n        this.resizeScrollbars();\n\n    },\n\n    resizeScrollbars: function() {\n        this.sbHScroller.shortenBy(this.sbVScroller).resize();\n        //this.sbVScroller.shortenBy(this.sbHScroller);\n        this.sbVScroller.resize();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll values have changed, we've been notified.\n     */\n    setVScrollbarValues: function(max) {\n        this.sbVScroller.range = {\n            min: 0,\n            max: max\n        };\n    },\n\n    setHScrollbarValues: function(max) {\n        this.sbHScroller.range = {\n            min: 0,\n            max: max\n        };\n    },\n\n    scrollValueChangedNotification: function() {\n\n        if (this.hScrollValue === this.sbPrevHScrollValue && this.vScrollValue === this.sbPrevVScrollValue) {\n            return;\n        }\n\n        this.sbPrevHScrollValue = this.hScrollValue;\n        this.sbPrevVScrollValue = this.vScrollValue;\n\n        if (this.cellEditor) {\n            this.cellEditor.scrollValueChangedNotification();\n        }\n\n        this.computeCellsBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Get data value at given cell.\n     * @desc Delegates to the behavior.\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     * @param {*} value\n     */\n    getValue: function(x, y) {\n        return this.behavior.getValue(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set a data value into the behavior (model) at the given point\n     * @param {number} x - The horizontal coordinate.\n     * @param {number} y - The vertical coordinate.\n     */\n    setValue: function(x, y, value) {\n        this.behavior.setValue(x, y, value);\n    },\n\n    getColumnAlignment: function(c) {\n        return this.behavior.getColumnAlignment(c);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc The data dimensions have changed, or our pixel boundries have changed.\n     * Adjust the scrollbar properties as necessary.\n     */\n    synchronizeScrollingBoundries: function() {\n        var numFixedColumns = this.getFixedColumnCount();\n        var numFixedRows = this.getFixedRowCount();\n\n        var numColumns = this.getColumnCount();\n        var numRows = this.getRowCount();\n\n        var bounds = this.getBounds();\n        if (!bounds) {\n            return;\n        }\n        var scrollableHeight = bounds.height - this.behavior.getFixedRowsMaxHeight() - 15; //5px padding at bottom and right side\n        var scrollableWidth = (bounds.width - 200) - this.behavior.getFixedColumnsMaxWidth() - 15;\n\n        var lastPageColumnCount = 0;\n        var columnsWidth = 0;\n        for (; lastPageColumnCount < numColumns; lastPageColumnCount++) {\n            var eachWidth = this.getColumnWidth(numColumns - lastPageColumnCount - 1);\n            columnsWidth = columnsWidth + eachWidth;\n            if (columnsWidth > scrollableWidth) {\n                break;\n            }\n        }\n\n        var lastPageRowCount = 0;\n        var rowsHeight = 0;\n        for (; lastPageRowCount < numRows; lastPageRowCount++) {\n            var eachHeight = this.getRowHeight(numRows - lastPageRowCount - 1);\n            rowsHeight = rowsHeight + eachHeight;\n            if (rowsHeight > scrollableHeight) {\n                break;\n            }\n        }\n\n        var hMax = Math.max(0, numColumns - numFixedColumns - lastPageColumnCount);\n        this.setHScrollbarValues(hMax);\n\n        var vMax = 1 + Math.max(0, numRows - numFixedRows - lastPageRowCount);\n        this.setVScrollbarValues(vMax);\n\n        this.setHScrollValue(Math.min(this.getHScrollValue(), hMax));\n        this.setVScrollValue(Math.min(this.getVScrollValue(), vMax));\n\n        //this.getCanvas().resize();\n        this.computeCellsBounds();\n        this.repaint();\n\n        this.resizeScrollbars();\n\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Note that \"viewable rows\" includes any partially viewable rows.\n     * @returns {number} The number of viewable rows.\n     */\n    getVisibleRows: function() {\n        return this.getRenderer().getVisibleRows();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Note that \"viewable columns\" includes any partially viewable columns.\n     * @returns {number} The number of viewable columns.\n     */\n    getVisibleColumns: function() {\n        return this.getRenderer().getVisibleColumns();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Initialize the renderer sub-component.\n     */\n    initRenderer: function() {\n        this.renderer = new Renderer(this);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Renderer} sub-component\n     */\n    getRenderer: function() {\n        return this.renderer;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The width of the given column.\n     * @param {number} columnIndex - The untranslated column index.\n     */\n    getColumnWidth: function(columnIndex) {\n        return this.behavior.getColumnWidth(columnIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the width of the given column.\n     * @param {number} columnIndex - The untranslated column index.\n     * @param {number} columnWidth - The width in pixels.\n     */\n    setColumnWidth: function(columnIndex, columnWidth) {\n        this.stopEditing();\n        this.behavior.setColumnWidth(columnIndex, columnWidth);\n    },\n\n    getColumnEdge: function(c) {\n        return this.behavior.getColumnEdge(c, this.getRenderer());\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The total width of all the fixed columns.\n     */\n    getFixedColumnsWidth: function() {\n        return this.behavior.getFixedColumnsWidth();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The height of the given row\n     * @param {number} rowIndex - The untranslated fixed column index.\n     */\n    getRowHeight: function(rowIndex) {\n        return this.behavior.getRowHeight(rowIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Set the height of the given row.\n     * @param {number} rowIndex - The row index.\n     * @param {number} rowHeight - The width in pixels.\n     */\n    setRowHeight: function(rowIndex, rowHeight) {\n        this.stopEditing();\n        this.behavior.setRowHeight(rowIndex, rowHeight);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The total fixed rows height\n     */\n    getFixedRowsHeight: function() {\n        return this.behavior.getFixedRowsHeight();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of columns.\n     */\n    getColumnCount: function() {\n        return this.behavior.getColumnCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of fixed rows.\n     */\n    getRowCount: function() {\n        return this.behavior.getRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of unfiltered rows.\n     */\n    getUnfilteredRowCount: function() {\n        return this.behavior.getUnfilteredRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        return this.behavior.getFixedColumnCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        return this.behavior.getFixedRowCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary The top left area has been clicked on\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    topLeftClicked: function(mouse) {\n        this.behavior.topLeftClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A fixed row has been clicked.\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    rowHeaderClicked: function(mouse) {\n        this.behavior.rowHeaderClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary A fixed column has been clicked.\n     * @desc Delegates to the behavior.\n     * @param {event} event - The event details.\n     */\n    columnHeaderClicked: function(mouse) {\n        this.behavior.columnHeaderClicked(this, mouse);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc An edit event has occurred. Activate the editor.\n     * @param {event} event - The event details.\n     */\n    _activateEditor: function(event) {\n        var gridCell = event.gridCell;\n        this.activateEditor(gridCell.x, gridCell.y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Activate the editor at the given coordinates.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     * @returns {CellEditor} (or objected extended from same) The editor object.\n     */\n    activateEditor: function(x, y) {\n        var editor;\n\n        if (this.isEditable() || this.isFilterRow(y)) {\n            editor = this.getCellEditorAt(x, y);\n\n            if (editor) {\n                var point = editor.getEditorPoint();\n                if (editor) {\n                    if (point.x === x && point.y === y && editor.isEditing) {\n                        return; //we're already open at this location\n                    }\n\n                    if (this.isEditing()) {\n                        this.stopEditing(); //other editor is open, close it first\n                    }\n                    event.gridCell = {\n                        x: x,\n                        y: y\n                    };\n                    this.editAt(editor, event);\n                }\n            }\n        }\n\n        return editor;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Get the cell editor.\n     * @desc Delegates to the behavior.\n     * @returns The cell editor at the given coordinates.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     */\n    getCellEditorAt: function(x, y) {\n        return this.behavior._getCellEditorAt(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @summary Toggle HiDPI support.\n     * @desc HiDPI support is now *on* by default.\n     * > There used to be a bug in Chrome that caused severe slow down on bit blit of large images, so this HiDPI needed to be optional.\n     */\n    toggleHiDPI: function() {\n        if (this.useHiDPI()) {\n            this.removeAttribute('hidpi');\n        } else {\n            this.setAttribute('hidpi', null);\n        }\n        this.canvas.resize();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} Te HiDPI ratio.\n     */\n    getHiDPI: function(ctx) {\n        if (window.devicePixelRatio && this.useHiDPI()) {\n            var devicePixelRatio = window.devicePixelRatio || 1;\n            var backingStoreRatio = ctx.webkitBackingStorePixelRatio ||\n                ctx.mozBackingStorePixelRatio ||\n                ctx.msBackingStorePixelRatio ||\n                ctx.oBackingStorePixelRatio ||\n                ctx.backingStorePixelRatio || 1;\n\n            var ratio = devicePixelRatio / backingStoreRatio;\n            return ratio;\n        } else {\n            return 1;\n        }\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The width of the given (recently rendered) column.\n     * @param {number} colIndex - The column index.\n     */\n    getRenderedWidth: function(colIndex) {\n        return this.renderer.getRenderedWidth(colIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The height of the given (recently rendered) row.\n     * @param {number} rowIndex - Tthe row index.\n     */\n    getRenderedHeight: function(rowIndex) {\n        return this.renderer.getRenderedHeight(rowIndex);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {CellEditor} The cell editor at alias \"name\" (a sub-component).\n     * @param {string} name\n     */\n    resolveCellEditor: function(name) {\n        return this.cellEditors[name];\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Update the cursor under the hover cell.\n     */\n    updateCursor: function() {\n        var cursor = this.behavior.getCursorAt(-1, -1);\n        var hoverCell = this.getHoverCell();\n        if (\n            hoverCell &&\n            hoverCell.x > -1 &&\n            hoverCell.y > -1\n        ) {\n            var x = hoverCell.x + this.getHScrollValue();\n            cursor = this.behavior.getCursorAt(x, hoverCell.y + this.getVScrollValue());\n        }\n        this.beCursor(cursor);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Repaint the given cell.\n     * @param {x} x - The horizontal coordinate.\n     * @param {y} y - The vertical coordinate.\n     */\n    repaintCell: function(x, y) {\n        this.getRenderer().repaintCell(x, y);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {boolean} The user is currently dragging a column to reorder it.\n     */\n    isDraggingColumn: function() {\n        return !!this.renderOverridesCache.dragger;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll up one full page.\n     * @returns {number}\n     */\n    pageUp: function() {\n        var rowNum = this.getRenderer().getPageUpRow();\n        this.setVScrollValue(rowNum);\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Scroll down one full page.\n     * @returns {number}\n     */\n    pageDown: function() {\n        var rowNum = this.getRenderer().getPageDownRow();\n        this.setVScrollValue(rowNum);\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Not yet implemented.\n     */\n    pageLeft: function() {\n        console.log('page left');\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Not yet implemented.\n     */\n    pageRight: function() {\n        console.log('page right');\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object[]} Objects with the values that were just rendered.\n     */\n    getRenderedData: function() {\n        // assumes one row of headers\n        var behavior = this.behavior,\n            renderer = this.getRenderer(),\n            colCount = this.getColumnCount(),\n            rowCount = renderer.getVisibleRows(),\n            headers = new Array(colCount),\n            results = new Array(rowCount),\n            row;\n\n        headers.forEach(function(header, c) {\n            headers[c] = behavior.getColumnId(c, 0);\n        });\n\n        results.forEach(function(result, r) {\n            row = results[r] = {\n                hierarchy: behavior.getFixedColumnValue(0, r)\n            };\n            headers.forEach(function(field, c) {\n                row[field] = behavior.getValue(c, r);\n            });\n        });\n\n        return results;\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {object} An object that represents the currently selection row.\n     */\n    getSelectedRow: function() {\n        var sels = this.selectionModel.getSelections();\n        if (sels.length) {\n            var behavior = this.behavior,\n                colCount = this.getColumnCount(),\n                topRow = sels[0].origin.y,\n                row = {\n                    //hierarchy: behavior.getFixedColumnValue(0, topRow)\n                };\n\n            for (var c = 0; c < colCount; c++) {\n                row[behavior.getColumnId(c, 0)] = behavior.getValue(c, topRow);\n            }\n\n            return row;\n        }\n    },\n\n    fireRequestCellEdit: function(cell, value) {\n        var clickEvent = new CustomEvent('fin-request-cell-edit', {\n            cancelable: true,\n            detail: {\n                value: value,\n                gridCell: cell,\n                time: Date.now()\n            }\n        });\n        return this.canvas.dispatchEvent(clickEvent); //I wasn't cancelled\n    },\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Synthesize and fire a fin-before-cell-edit event.\n     * @param {Point} cell - The x,y coordinates.\n     * @param {Object} value - The current value.\n     */\n    fireBeforeCellEdit: function(cell, oldValue, newValue, control) {\n        var clickEvent = new CustomEvent('fin-before-cell-edit', {\n            cancelable: true,\n            detail: {\n                oldValue: oldValue,\n                newValue: newValue,\n                gridCell: cell,\n                time: Date.now(),\n                input: control,\n                row: this.getRow(cell.y)\n            }\n        });\n        var proceed = this.canvas.dispatchEvent(clickEvent);\n        return proceed; //I wasn't cancelled\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {Renderer} sub-component\n     * @param {Point} cell - The x,y coordinates.\n     * @param {Object} oldValue - The old value.\n     * @param {Object} newValue - The new value.\n     */\n    fireAfterCellEdit: function(cell, oldValue, newValue, control) {\n        var clickEvent = new CustomEvent('fin-after-cell-edit', {\n            detail: {\n                newValue: newValue,\n                oldValue: oldValue,\n                gridCell: cell,\n                time: Date.now(),\n                input: control,\n                row: this.getRow(cell.y)\n            }\n        });\n        this.canvas.dispatchEvent(clickEvent);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Autosize the column at colIndex for best fit.\n     * @param {number} colIndex - The column index to modify at\n     */\n    autosizeColumn: function(colIndex) {\n        var column = this.behavior.getColumn(colIndex);\n        column.checkColumnAutosizing(true);\n        this.computeCellsBounds();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Enable/disable if this component can receive the focus.\n     * @param {boolean} - canReceiveFocus\n     */\n    setFocusable: function(canReceiveFocus) {\n        this.getCanvas().setFocusable(canReceiveFocus);\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of columns that were just rendered\n     */\n    getVisibleColumnsCount: function() {\n        return this.getRenderer().getVisibleColumnsCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @returns {number} The number of rows that were just rendered\n     */\n    getVisibleRowsCount: function() {\n        return this.getRenderer().getVisibleRowsCount();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n    update the size of the grid\n     *\n     * #### returns: integer\n     */\n    updateSize: function() {\n        this.canvas.checksize();\n    },\n\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Stop the global repainting flag thread.\n     */\n    stopPaintThread: function() {\n        this.canvas.stopPaintThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Stop the global resize check flag thread.\n     */\n    stopResizeThread: function() {\n        this.canvas.stopResizeThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Restart the global resize check flag thread.\n     */\n    restartResizeThread: function() {\n        this.canvas.restartResizeThread();\n    },\n\n    /**\n     * @memberOf Hypergrid.prototype\n     * @desc Restart the global repainting check flag thread.\n     */\n    restartPaintThread: function() {\n        this.canvas.restartPaintThread();\n    },\n\n    swapColumns: function(source, target) {\n        this.behavior.swapColumns(source, target);\n    },\n\n    endDragColumnNotification: function() {\n        this.behavior.endDragColumnNotification();\n    },\n\n    getFixedColumnsMaxWidth: function() {\n        return this.behavior.getFixedColumnsMaxWidth();\n    },\n\n    isMouseDownInHeaderArea: function() {\n        var numHeaderColumns = this.getHeaderColumnCount();\n        var numHeaderRows = this.getHeaderRowCount();\n        var mouseDown = this.getMouseDown();\n        return mouseDown.x < numHeaderColumns || mouseDown.y < numHeaderRows;\n    },\n\n    isHeaderWrapping: function() {\n        return this.resolveProperty('headerTextWrapping');\n    },\n\n    _getBoundsOfCell: function(x, y) {\n        return this.getRenderer()._getBoundsOfCell(x, y);\n    },\n\n    getColumnProperties: function(columnIndex) {\n        return this.behavior.getColumnProperties(columnIndex);\n    },\n\n    setColumnProperties: function(columnIndex, properties) {\n        this.behavior.setColumnProperties(columnIndex, properties);\n    },\n\n    moveSingleSelect: function(x, y) {\n        this.behavior.moveSingleSelect(this, x, y);\n    },\n\n    selectCell: function(x, y) {\n        this.selectionModel.clear();\n        this.selectionModel.select(x, y, 0, 0);\n    },\n\n    getHeaderColumnCount: function() {\n        return this.behavior.getHeaderColumnCount();\n    },\n\n    toggleSort: function(x, keys) {\n        this.stopEditing();\n        var behavior = this.behavior;\n        var self = this;\n        behavior.toggleSort(x, keys);\n\n        setTimeout(function() {\n            self.synchronizeScrollingBoundries();\n            //self.behaviorChanged();\n            if (self.isColumnAutosizing()) {\n                behavior.autosizeAllColumns();\n            }\n            self.repaint();\n        }, 10);\n    },\n\n    toggleSelectColumn: function(x, keys) {\n        keys = keys || [];\n        var model = this.selectionModel;\n        var alreadySelected = model.isColumnSelected(x);\n        var hasCTRL = keys.indexOf('CTRL') > -1;\n        var hasSHIFT = keys.indexOf('SHIFT') > -1;\n        if (!hasCTRL && !hasSHIFT) {\n            model.clear();\n            if (!alreadySelected) {\n                model.selectColumn(x);\n            }\n        } else {\n            if (hasCTRL) {\n                if (alreadySelected) {\n                    model.deselectColumn(x);\n                } else {\n                    model.selectColumn(x);\n                }\n            }\n            if (hasSHIFT) {\n                model.clear();\n                model.selectColumn(this.lastEdgeSelection[0], x);\n            }\n        }\n        if (!alreadySelected && !hasSHIFT) {\n            this.lastEdgeSelection[0] = x;\n        }\n        this.repaint();\n        this.fireSyntheticColumnSelectionChangedEvent();\n    },\n\n    toggleSelectRow: function(y, keys) {\n        //we can select the totals rows if they exist, but not rows above that\n        if (y > this.getFilterRowIndex()) {\n            keys = keys || [];\n\n            var sm = this.selectionModel;\n            var alreadySelected = sm.isRowSelected(y);\n            var hasSHIFT = keys.indexOf('SHIFT') >= 0;\n\n            if (alreadySelected) {\n                sm.deselectRow(y);\n            } else {\n                this.singleSelect();\n                sm.selectRow(y);\n            }\n\n            if (hasSHIFT) {\n                sm.clear();\n                sm.selectRow(this.lastEdgeSelection[1], y);\n            }\n\n            if (!alreadySelected && !hasSHIFT) {\n                this.lastEdgeSelection[1] = y;\n            }\n            this.repaint();\n        }\n    },\n\n    singleSelect: function() {\n        var isCheckboxOnlyRowSelections = this.isCheckboxOnlyRowSelections(),\n            isSingleRowSelectionMode = this.isSingleRowSelectionMode(),\n            hasCTRL = this.mouseDownState.primitiveEvent.detail.primitiveEvent.ctrlKey,\n            result = (\n                isCheckboxOnlyRowSelections && isSingleRowSelectionMode ||\n                !isCheckboxOnlyRowSelections && (!hasCTRL || isSingleRowSelectionMode)\n            );\n\n        if (result) {\n            this.selectionModel.clearRowSelection();\n        }\n\n        return result;\n    },\n\n    selectViewportCell: function(x, y) {\n        var headerRowCount = this.getHeaderRowCount();\n        var renderer = this.getRenderer();\n        var realX = renderer.getVisibleColumns()[x];\n        var realY = renderer.getVisibleRows()[y];\n        this.clearSelections();\n        this.select(realX, realY + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(realX, realY + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToViewportCell: function(x, y) {\n        var selections = this.getSelections();\n        if (selections && selections.length) {\n            var headerRowCount = this.getHeaderRowCount();\n            var renderer = this.getRenderer();\n            var realX = renderer.getVisibleColumns()[x];\n            var realY = renderer.getVisibleRows()[y] + headerRowCount;\n            var selection = selections[0];\n            var origin = selection.origin;\n            this.setDragExtent(this.newPoint(realX - origin.x, realY - origin.y));\n            this.select(origin.x, origin.y, realX - origin.x, realY - origin.y);\n            this.repaint();\n        }\n    },\n\n    selectFinalCellOfCurrentRow: function() {\n        var x = this.getColumnCount() - 1;\n        var y = this.getSelectedRows()[0];\n        var headerRowCount = this.getHeaderRowCount();\n        this.clearSelections();\n        this.scrollBy(this.getColumnCount(), 0);\n        this.select(x, y + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(x, y + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToFinalCellOfCurrentRow: function() {\n        var selections = this.getSelections();\n        if (selections && selections.length) {\n            var selection = selections[0];\n            var origin = selection.origin;\n            var extent = selection.extent;\n            var columnCount = this.getColumnCount();\n            this.scrollBy(columnCount, 0);\n\n            this.clearSelections();\n            this.select(origin.x, origin.y, columnCount - origin.x - 1, extent.y);\n\n            this.repaint();\n        }\n    },\n\n    selectFirstCellOfCurrentRow: function() {\n        var x = 0;\n        var y = this.getSelectedRows()[0];\n        var headerRowCount = this.getHeaderRowCount();\n        this.clearSelections();\n        this.setHScrollValue(0);\n        this.select(x, y + headerRowCount, 0, 0);\n        this.setMouseDown(this.newPoint(x, y + headerRowCount));\n        this.setDragExtent(this.newPoint(0, 0));\n        this.repaint();\n    },\n\n    selectToFirstCellOfCurrentRow: function() {\n        var selections = this.getSelections();\n        if (selections && selections.length) {\n            var selection = selections[0];\n            var origin = selection.origin;\n            var extent = selection.extent;\n            this.clearSelections();\n            this.select(origin.x, origin.y, -origin.x, extent.y);\n            this.setHScrollValue(0);\n            this.repaint();\n        }\n    },\n\n    selectFinalCell: function() {\n        this.selectCell(this.getColumnCount() - 1, this.getRowCount() - 1);\n        this.scrollBy(this.getColumnCount(), this.getRowCount());\n        this.repaint();\n    },\n\n    selectToFinalCell: function() {\n        var selections = this.getSelections();\n        if (selections && selections.length) {\n            var selection = selections[0];\n            var origin = selection.origin;\n            var columnCount = this.getColumnCount();\n            var rowCount = this.getRowCount();\n\n            this.clearSelections();\n            this.select(origin.x, origin.y, columnCount - origin.x - 1, rowCount - origin.y - 1);\n            this.scrollBy(columnCount, rowCount);\n            this.repaint();\n        }\n    },\n\n    isShowRowNumbers: function() {\n        return this.resolveProperty('showRowNumbers');\n    },\n    isEditable: function() {\n        return this.resolveProperty('editable') === true;\n    },\n    isShowFilterRow: function() {\n        return this.resolveProperty('showFilterRow');\n    },\n    isShowHeaderRow: function() {\n        return this.resolveProperty('showHeaderRow');\n    },\n    getHeaderRowCount: function() {\n        return this.behavior.getHeaderRowCount();\n    },\n    isFilterRow: function(y) {\n        return y === this.getFilterRowIndex();\n    },\n    getFilterRowIndex: function() {\n        return !this.isShowFilterRow() ? -1 : this.isShowHeaderRow() ? 1 : 0;\n    },\n    setGroups: function(arrayOfColumnIndexes) {\n        this.behavior.setGroups(arrayOfColumnIndexes);\n    },\n    filterClicked: function(event) {\n        this.activateEditor(event.gridCell.x, event.gridCell.y);\n    },\n    hasHierarchyColumn: function() {\n        return this.behavior.hasHierarchyColumn();\n    },\n    isHierarchyColumn: function(x) {\n        return this.hasHierarchyColumn() && x === 0;\n    },\n    checkScrollbarVisibility: function() {\n        // var hoverClassOver = this.resolveProperty('scrollbarHoverOver');\n        // var hoverClassOff = this.resolveProperty('scrollbarHoverOff');\n\n        // if (hoverClassOff === 'visible') {\n        //     this.sbHScroller.classList.remove(hoverClassOver);\n        //     this.sbVScroller.classList.remove(hoverClassOff);\n        //     this.sbHScroller.classList.add('visible');\n        //     this.sbVScroller.classList.add('visible');\n        // }\n    },\n    isColumnOrRowSelected: function() {\n        return this.selectionModel.isColumnOrRowSelected();\n    },\n    selectColumn: function(x1, x2) {\n        this.selectionModel.selectColumn(x1, x2);\n    },\n    selectRow: function(y1, y2) {\n        var sm = this.selectionModel;\n        var selectionEdge = this.getFilterRowIndex() + 1;\n\n        if (this.singleSelect()) {\n            y1 = y2;\n        } else {\n            // multiple row selection\n            y2 = y2 || y1;\n        }\n        var min = Math.min(y1, y2);\n        if (min >= selectionEdge) {\n            var max = Math.max(y1, y2);\n            sm.selectRow(min, max);\n        }\n    },\n    isRowNumberAutosizing: function() {\n        return this.resolveProperty('rowNumberAutosizing');\n    },\n    isRowSelected: function(r) {\n        return this.selectionModel.isRowSelected(r);\n    },\n    isColumnSelected: function(c) {\n        return this.selectionModel.isColumnSelected(c);\n    },\n    lookupFeature: function(key) {\n        return this.behavior.lookupFeature(key);\n    },\n    getRow: function(y) {\n        return this.behavior.getRow(y);\n    },\n    getFieldName: function(index) {\n        return this.behavior.getFieldName(index);\n    },\n\n    getColumnIndex: function(fieldName) {\n        return this.behavior.getColumnIndex(fieldName);\n    },\n    isCellSelection: function() {\n        return this.resolveProperty('cellSelection') === true;\n    },\n    isRowSelection: function() {\n        return this.resolveProperty('rowSelection') === true;\n    },\n    isColumnSelection: function() {\n        return this.resolveProperty('columnSelection') === true;\n    },\n    getComputedRow: function(y) {\n        return this.behavior.getComputedRow(y);\n    },\n    isColumnAutosizing: function() {\n        return this.resolveProperty('columnAutosizing') === true;\n    },\n    setGlobalFilter: function(string) {\n        this.behavior.setGlobalFilter(string);\n    },\n    selectRowsFromCells: function() {\n        if (!this.isCheckboxOnlyRowSelections()) {\n            var last,\n                hasCTRL = this.mouseDownState.primitiveEvent.detail.primitiveEvent.ctrlKey;\n\n            if (hasCTRL && !this.isSingleRowSelectionMode()) {\n                this.selectionModel.selectRowsFromCells(0, hasCTRL);\n            } else if ((last = this.selectionModel.getLastSelection())) {\n                this.selectRow(null, last.corner.y);\n            } else {\n                this.clearRowSelection();\n            }\n        }\n    },\n    selectColumnsFromCells: function() {\n        this.selectionModel.selectColumnsFromCells();\n    },\n    getSelectedRows: function() {\n        return this.behavior.getSelectedRows();\n    },\n    getSelectedColumns: function() {\n        return this.behavior.getSelectedColumns();\n    },\n    getSelections: function() {\n        return this.behavior.getSelections();\n    },\n    getLastSelectionType: function() {\n        return this.selectionModel.getLastSelectionType();\n    },\n    isCellSelected: function(x, y) {\n        return this.selectionModel.isCellSelected(x, y);\n    },\n    isInCurrentSelectionRectangle: function(x, y) {\n        return this.selectionModel.isInCurrentSelectionRectangle(x, y);\n    },\n    selectAllRows: function() {\n        this.selectionModel.selectAllRows();\n    },\n    areAllRowsSelected: function() {\n        return this.selectionModel.areAllRowsSelected();\n    },\n    toggleSelectAllRows: function() {\n        if (this.areAllRowsSelected()) {\n            this.selectionModel.clear();\n        } else {\n            this.selectAllRows();\n        }\n        this.repaint();\n    },\n    getField: function(x) {\n        return this.behavior.getField(x);\n    },\n    isSingleRowSelectionMode: function() {\n        return this.resolveProperty('singleRowSelectionMode');\n    },\n    newPoint: function(x, y) {\n        return new Point(x, y);\n    },\n    newRectangle: function(x, y, width, height) {\n        return new Rectangle(x, y, width, height);\n    },\n    getFormattedValue: function(x, y) {\n        y = y + this.getHeaderRowCount();\n        var formatType = this.getColumnProperties(x).format;\n        var value = this.getValue(x, y);\n        var formatter = this.getFormatter(formatType);\n        var string = formatter(value);\n        return string;\n    }\n};\n\n/**\n * @summary Update deep properties with new values.\n * @desc This function is a recursive property setter which updates a deep property in a destination object with the value of a congruent property in a source object.\n *\n * > Terminology: A deep property is a \"terminal node\" (primitive value) nested at some depth (i.e., depth > 1) inside a complex object (an object containing nested objects). A congruent property is a property in another object with the same name and at the same level of nesting.\n *\n * This function is simple and elegant. I recommend you study the code, which nonetheless implies all of the following:\n *\n * * If the deep property is _not_ found in `destination`, it will be created.\n * * If the deep property is found in `destination` _and_ is a primitive type, it will be modified (overwritten with the value from `source`).\n * * If the deep property is found in `destination` _but_ is not a primitive type (i.e., is a nested object), it will _also_ be overwritten with the (primitive) value from `source`.\n * * If the nested object the deep property inhabits in `source` is not found in `destination`, it will be created.\n * * If the nested object the deep property inhabits in `source` is found in `destination` but is not in fact an object (i.e., it is a primitive value), it will be overwritten with a reference to that object.\n * * If the primitive value is `undefined`, the destination property is deleted.\n * * `source` may contain multiple properties to update.\n *\n * That one rule is simply this: If both the source _and_ the destination properties are objects, then recurse; else overwrite the destination property with the source property.\n *\n * > Caveat: This is _not_ equivalent to a deep extend function. While both a deep extend and this function will recurse over a complex object, they are fundamentally different: A deep extend clones the nested objects as it finds them; this function merely updates them (or creates them where they don't exist).\n *\n * @param {object} destination - An object to update with new or modified property values\n * @param {object} source - A congruent object continaly (only) the new or modified property values.\n * @returns {object} Always returns `destination`.\n */\nfunction addDeepProperties(destination, source) {\n    _(source).each(function(property, key) {\n        if (typeof destination[key] === 'object' && typeof property === 'object') {\n            addDeepProperties(destination[key], property);\n        } else if (property === undefined) {\n            delete destination[key];\n        } else {\n            destination[key] = property;\n        }\n    });\n    return destination;\n}\n\nfunction normalizeRect(rect) {\n    var o = rect.origin;\n    var c = rect.corner;\n\n    var ox = Math.min(o.x, c.x);\n    var oy = Math.min(o.y, c.y);\n\n    var cx = Math.max(o.x, c.x);\n    var cy = Math.max(o.y, c.y);\n\n    var result = new Rectangle(ox, oy, cx - ox, cy - oy);\n\n    return result;\n}\n\nfunction buildPolymerTheme() {\n    clearObjectProperties(polymerTheme);\n    var pb = document.createElement('paper-button');\n\n    pb.style.display = 'none';\n    pb.setAttribute('disabled', true);\n    document.body.appendChild(pb);\n    var p = window.getComputedStyle(pb);\n\n    var section = document.createElement('section');\n    section.style.display = 'none';\n    section.setAttribute('hero', true);\n    document.body.appendChild(section);\n\n    var h = window.getComputedStyle(document.querySelector('html'));\n    var hb = window.getComputedStyle(document.querySelector('html, body'));\n    var s = window.getComputedStyle(section);\n\n    polymerTheme.columnHeaderBackgroundColor = p.color;\n    polymerTheme.rowHeaderBackgroundColor = p.color;\n    polymerTheme.topLeftBackgroundColor = p.color;\n    polymerTheme.lineColor = p.backgroundColor;\n\n    polymerTheme.backgroundColor2 = hb.backgroundColor;\n\n    polymerTheme.color = h.color;\n    polymerTheme.fontFamily = h.fontFamily;\n    polymerTheme.backgroundColor = s.backgroundColor;\n\n    pb.setAttribute('disabled', false);\n    pb.setAttribute('secondary', true);\n    pb.setAttribute('raised', true);\n    p = window.getComputedStyle(pb);\n\n    polymerTheme.columnHeaderColor = p.color;\n    polymerTheme.rowHeaderColor = p.color;\n    polymerTheme.topLeftColor = p.color;\n\n\n    polymerTheme.backgroundSelectionColor = p.backgroundColor;\n    polymerTheme.foregroundSelectionColor = p.color;\n\n    pb.setAttribute('secondary', false);\n    pb.setAttribute('warning', true);\n\n    polymerTheme.columnHeaderForegroundSelectionColor = p.color;\n    polymerTheme.columnHeaderBackgroundSelectionColor = p.backgroundColor;\n    polymerTheme.rowHeaderForegroundSelectionColor = p.color;\n    polymerTheme.fixedColumnBackgroundSelectionColor = p.backgroundColor;\n\n    //check if there is actually a theme loaded if not, clear out all bogus values\n    //from my cache\n    if (polymerTheme.columnHeaderBackgroundSelectionColor === 'rgba(0, 0, 0, 0)' ||\n        polymerTheme.lineColor === 'transparent') {\n        clearObjectProperties(polymerTheme);\n    }\n\n    document.body.removeChild(pb);\n    document.body.removeChild(section);\n}\n\nfunction clearObjectProperties(obj) {\n    for (var prop in obj) {\n        if (obj.hasOwnProperty(prop)) {\n            delete obj[prop];\n        }\n    }\n}\n\nfunction valueOrFunctionExecute(valueOrFunction) {\n    var result = typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction;\n    return result || result === 0 ? result : '';\n}\n\nmodule.exports = Hypergrid;\n","/* eslint-env browser */\n'use strict';\n\nvar _ = require('object-iterators');\nvar Base = require('../lib/Base');\n\nvar Column = require('./Column');\nvar CellProvider = require('../lib/CellProvider');\n\nvar noExportProperties = [\n    'columnHeader',\n    'columnHeaderColumnSelection',\n    'filterProperties',\n    'rowHeader',\n    'rowHeaderRowSelection',\n    'rowNumbersProperties',\n    'treeColumnProperties',\n    'treeColumnPropertiesColumnSelection',\n];\n\nvar isNull = {\n    isNull: true\n};\n\n/**\n * @constructor\n * @desc This is the base class for creating behaviors.  a behavior can be thought of as a model++.\nit contains all code/data that's necessary for easily implementing a virtual data source and it's manipulation/analytics\n */\nvar Behavior = Base.extend('Behavior', {\n\n    /**\n     * @desc this is the callback for the plugin pattern of nested tags\n     * @param {Hypergrid} grid\n     * @memberOf Behavior.prototype\n     */\n    initialize: function(grid) { //formerly installOn\n        grid.setBehavior(this);\n        this.initializeFeatureChain(grid);\n\n        this.getDataModel();\n        this.cellProvider = this.createCellProvider();\n        this.renderedColumnCount = 30;\n        this.renderedRowCount = 60;\n        this.dataUpdates = {}; //for overriding with edit values;\n    },\n\n    /**\n     * @desc create the feature chain - this is the [chain of responsibility](http://c2.com/cgi/wiki?ChainOfResponsibilityPattern) pattern.\n     * @param {Hypergrid} grid\n     * @memberOf Behavior.prototype\n     */\n    initializeFeatureChain: function(grid) {\n        var self = this;\n        this.features.forEach(function(FeatureConstructor) {\n            self.setNextFeature(new FeatureConstructor);\n        });\n\n        this.featureChain.initializeOn(grid);\n    },\n\n    features: [], // in case implementing class has no features TODO: Will this ever happen?\n\n    /**\n     * memento for the user configured visual properties of the table\n     * @type {object}\n     * @memberOf Behavior.prototype\n     */\n    tableState: null,\n\n    /**\n     * @type {Hypergrid}\n     * @memberOf Behavior.prototype\n     */\n    grid: null,\n\n    /**\n     * list of default cell editor names\n     * @type {string[]}\n     * @memberOf Behavior.prototype\n     */\n    editorTypes: [\n        'choice',\n        'textfield',\n        'color',\n        'slider',\n        'spinner',\n        'date'\n    ],\n\n    /**\n     * controller chain of command\n     * @type {object}\n     * @memberOf Behavior.prototype\n     */\n    featureChain: null,\n\n    dataModel: null,\n    baseModel: null,\n\n    scrollPositionX: 0,\n    scrollPositionY: 0,\n\n    featureMap: {},\n    allColumns: [],\n    columns: [],\n\n    reset: function() {\n\n        this.cellProvider = this.createCellProvider();\n        this.renderedColumnCount = 30;\n        this.renderedRowCount = 60;\n        this.dataUpdates = {}; //for overriding with edit values;\n        this.clearColumns();\n        this.clearState();\n        this.getDataModel().reset();\n        this.createColumns();\n    },\n\n    clearColumns: function() {\n        this.columns = [];\n        this.allColumns = [];\n        this.columns[-1] = this.newColumn(-1, '');\n        this.columns[-2] = this.newColumn(-2, 'Tree');\n        this.allColumns[-1] = this.columns[-1];\n        this.allColumns[-2] = this.columns[-2];\n    },\n\n    getColumn: function(x) {\n        return this.columns[x];\n    },\n\n    getColumnId: function(x) {\n        return this.getColumn(x).label;\n    },\n\n    newColumn: function(index, label) {\n        var properties = this.createColumnProperties();\n        this.getPrivateState().columnProperties[index] = properties;\n        return new Column(this, index, label);\n    },\n\n    addColumn: function(index, label) {\n        var column = this.newColumn(index, label);\n        this.columns.push(column);\n        this.allColumns.push(column);\n        return column;\n    },\n\n    createColumns: function() {\n        //concrete implementation here\n    },\n\n    createColumnProperties: function() {\n        var tableState = this.getPrivateState();\n        var properties = Object.create(tableState);\n\n        properties.rowNumbersProperties = Object.create(properties, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.rowHeader = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderFont;\n                },\n                set: function(value) {\n                    this.rowHeaderFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderColor;\n                },\n                set: function(value) {\n                    this.rowHeaderColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.columnHeader = Object.create(properties, {\n            format: {\n                value: 'default'\n            },\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderFont;\n                },\n                set: function(value) {\n                    this.columnHeaderFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderColor;\n                },\n                set: function(value) {\n                    this.columnHeaderColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.columnHeaderColumnSelection = Object.create(properties.columnHeader, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderForegroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderForegroundColumnSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.columnHeaderBackgroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.columnHeaderBackgroundColumnSelectionColor = value;\n                }\n            }\n        });\n\n        properties.rowHeaderRowSelection = Object.create(properties.rowHeader, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderForegroundRowSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderForegroundRowSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.rowHeaderBackgroundRowSelectionColor;\n                },\n                set: function(value) {\n                    this.rowHeaderBackgroundRowSelectionColor = value;\n                }\n            }\n        });\n\n        properties.filterProperties = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.filterFont;\n                },\n                set: function(value) {\n                    this.filterFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.filterColor;\n                },\n                set: function(value) {\n                    this.filterColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterBackgroundColor;\n                },\n                set: function(value) {\n                    this.filterBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.filterForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.filterBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.filterBackgroundSelectionColor = value;\n                }\n            },\n            cellBorderStyle: {\n                configurable: true,\n                get: function() {\n                    return this.filterCellBorderStyle;\n                },\n                set: function(value) {\n                    this.filterCellBorderStyle = value;\n                }\n            },\n            cellBorderThickness: {\n                configurable: true,\n                get: function() {\n                    return this.filterCellBorderThickness;\n                },\n                set: function(value) {\n                    this.filterCellBorderThickness = value;\n                }\n            }\n        });\n\n        properties.treeColumnProperties = Object.create(properties, {\n            font: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnFont;\n                },\n                set: function(value) {\n                    this.treeColumnFont = value;\n                }\n            },\n            color: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnColor;\n                },\n                set: function(value) {\n                    this.treeColumnColor = value;\n                }\n            },\n            backgroundColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundColor = value;\n                }\n            },\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnForegroundSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnForegroundSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundSelectionColor = value;\n                }\n            }\n        });\n\n        properties.treeColumnPropertiesColumnSelection = Object.create(properties.treeColumnProperties, {\n            foregroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnForegroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnForegroundColumnSelectionColor = value;\n                }\n            },\n            backgroundSelectionColor: {\n                configurable: true,\n                get: function() {\n                    return this.treeColumnBackgroundColumnSelectionColor;\n                },\n                set: function(value) {\n                    this.treeColumnBackgroundColumnSelectionColor = value;\n                }\n            }\n        });\n\n        return properties;\n    },\n\n    getColumnWidth: function(x) {\n        var col = this.getColumn(x);\n        if (!col) {\n            return this.resolveProperty('defaultColumnWidth');\n        }\n        var width = col.getWidth();\n        return width;\n    },\n\n    setColumnWidth: function(x, width) {\n        this.getColumn(x).setWidth(width);\n        this.stateChanged();\n    },\n\n    getDataModel: function() {\n        if (this.dataModel === null) {\n            var dataModel = this.getDefaultDataModel();\n            this.setDataModel(dataModel);\n        }\n        return this.dataModel;\n    },\n\n    getCellRenderer: function(config, x, y) {\n        return this.getColumn(x).getCellRenderer(config, y);\n    },\n\n    setDataModel: function(newDataModel) {\n        this.dataModel = newDataModel;\n    },\n\n    setComplexFilter: function(columnIndex, complexFilter) {\n        var col = this.getColumn(columnIndex);\n        if (col) {\n            col.setComplexFilter(complexFilter);\n        }\n    },\n\n    getComplexFilter: function(columnIndex) {\n        var col = this.getColumn(columnIndex);\n        return col && col.getComplexFilter();\n    },\n\n    applyAnalytics: function() {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc utility function to empty an object of its members\n     * @param {object} obj - the object to empty\n     * @param {boolean} [exportProps]\n     * * `undefined` (omitted) - delete *all* properties\n     * * **falsy** - delete *only* the export properties\n     * * **truthy** - delete all properties *except* the export properties\n     */\n    clearObjectProperties: function(obj, exportProps) {\n        for (var key in obj) {\n            if (\n                obj.hasOwnProperty(key) && (\n                    exportProps === undefined ||\n                    !exportProps && noExportProperties.indexOf(key) >= 0 ||\n                    exportProps && noExportProperties.indexOf(key) < 0\n                )\n            ) {\n                delete obj[key];\n            }\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc getter for a [Memento](http://c2.com/cgi/wiki?MementoPattern) Object\n     * @returns {object}\n     */\n    getPrivateState: function() {\n        if (!this.tableState) {\n            this.tableState = this.getDefaultState();\n        }\n        return this.tableState;\n    },\n\n    //this is effectively a clone, with certain things removed....\n    getState: function() {\n        var copy = JSON.parse(JSON.stringify(this.getPrivateState()));\n        this.clearObjectProperties(copy.columnProperties, false);\n        return copy;\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc clear all table state\n     */\n    clearState: function() {\n        this.tableState = null;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {object} Newly created default empty tablestate.\n     */\n    getDefaultState: function() {\n        var tableProperties = this.grid._getProperties();\n        var state = Object.create(tableProperties);\n\n        _(state).extendOwn({\n            rowHeights: {},\n            cellProperties: {},\n            columnProperties: []\n        });\n\n        return state;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Restore this table to a previous state.\n     * See the [memento pattern](http://c2.com/cgi/wiki?MementoPattern).\n     * @param {Object} memento - an encapsulated representation of table state\n     */\n    setState: function(memento) {\n\n        //we don't want to clobber the column properties completely\n        if (!memento.columnIndexes) {\n            var fields = this.getFields();\n            memento.columnIndexes = [];\n            for (var i = 0; i < fields.length; i++) {\n                memento.columnIndexes[i] = i;\n            }\n        }\n        var colProperties = memento.columnProperties;\n        delete memento.columnProperties;\n        this.tableState = null;\n        var state = this.getPrivateState();\n        this.createColumns();\n        this.setColumnOrder(memento.columnIndexes);\n        _(state).extendOwn(memento);\n        this.setAllColumnProperties(colProperties);\n        memento.columnProperties = colProperties;\n        //memento.columnProperties = colProperties;\n\n        // this.getDataModel().setState(memento);\n        // var self = this;\n        // requestAnimationFrame(function() {\n        //     self.applySorts();\n        //     self.changed();\n        //     self.stateChanged();\n        // });\n\n        //just to be close/ it's easier on the eyes\n        this.setColumnWidth(-1, 24.193359375);\n        this.getDataModel().applyState();\n    },\n\n    setAllColumnProperties: function(properties) {\n        properties = properties || [];\n        for (var i = 0; i < properties.length; i++) {\n            var current = this.getPrivateState().columnProperties[i];\n            this.clearObjectProperties(current, false);\n            _(current).extendOwn(properties[i]);\n        }\n    },\n\n    setColumnOrder: function(indexes) {\n        if (!indexes) {\n            this.columns.length = 0;\n            return;\n        }\n        this.columns.length = indexes.length;\n        for (var i = 0; i < indexes.length; i++) {\n            this.columns[i] = this.allColumns[indexes[i]];\n        }\n    },\n\n    applySorts: function() {\n        //if I have sorts, apply them now//\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc fetch the value for a property key\n     * @returns {*} The value of the given property.\n     * @param {string} key - a property name\n     */\n    resolveProperty: function(key) {\n        return this.grid.resolveProperty(key);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc A specific cell was clicked; you've been notified.\n     * @param {Point} cell - point of cell coordinates\n     * @param {Object} event - all event information\n     */\n    cellClicked: function(cell, event) {\n        this.getDataModel().cellClicked(cell, event);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc A specific cell was le double-clicked; you've been notified.\n     * @param {Point} cell - point of cell coordinates\n     * @param {Object} event - all event information\n     */\n    cellDoubleClicked: function(cell, event) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc add nextFeature to me If I don't have a next node, otherwise pass it along\n     * @param {Feature}\n     */\n    setNextFeature: function(nextFeature) {\n        this.featureMap[nextFeature.alias] = nextFeature;\n        if (this.featureChain) {\n            this.featureChain.setNext(nextFeature);\n        } else {\n            this.featureChain = nextFeature;\n        }\n    },\n\n    lookupFeature: function(key) {\n        return this.featureMap[key];\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc getter for the cell provider\n     * @return {CellProvider}\n     */\n    getCellProvider: function() {\n        return this.cellProvider;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc setter for the hypergrid\n     * @param {Hypergrid} grid\n     */\n    setGrid: function(finGrid) {\n        this.grid = finGrid;\n        this.getDataModel().setGrid(finGrid);\n        this.clearColumns();\n    },\n\n    /** @deprecated Use `.grid` property instead.\n     * @memberOf Behavior.prototype\n     * @returns {Hypergrid} The hypergrid to which this behavior is attached.\n     * @param {type} varname - descripton\n     */\n    getGrid: function() {\n        return this.deprecated('grid', { since: '0.2' });\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc You can override this function and substitute your own cell provider.\n     * @return {CellProvider}\n     */\n    createCellProvider: function() {\n        return new CellProvider();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc First check to see if something was overridden.\n     * @return {*} The value at `x,y` for the top left section of the hypergrid.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     */\n    getValue: function(x, y) {\n        var column = this.getColumn(x);\n        return column && column.getValue(y);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        var column = this.getColumn(x);\n        return column && column.getUnfilteredValue(y);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc update the data at point x, y with value\n     * @return The data.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     * @param {Object} value - the value to use\n     */\n    setValue: function(x, y, value) {\n        var column = this.getColumn(x);\n        return column && column.setValue(y, value);\n    },\n\n    getDataValue: function(x, y) {\n        return this.getDataModel().getValue(x, y);\n    },\n\n    setDataValue: function(x, y, value) {\n        this.getDataModel().setValue(x, y, value);\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc First checks to see if something was overridden.\n     * @return {*} The value at x,y for the top left section of the hypergrid.\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     */\n    getCellProperties: function(x, y) {\n        return this.getColumn(x).getCellProperties(y);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc update the data at point x, y with value\n     * @param {number} x - x coordinate\n     * @param {number} y - y coordinate\n     * @param {Object} value - the value to use\n     */\n    setCellProperties: function(x, y, value) {\n        var col = this.getColumn(x);\n        if (col) {\n            col.setCellProperties(y, value);\n        }\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of rows in the hypergrid.\n     */\n    getRowCount: function() {\n        return this.getDataModel().getRowCount();\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.getDataModel().getUnfilteredRowCount();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The height in pixels of the fixed rows area  of the hypergrid.\n     */\n    getFixedRowsHeight: function() {\n        var count = this.getFixedRowCount();\n        var total = 0;\n        for (var i = 0; i < count; i++) {\n            total = total + this.getRowHeight(i);\n        }\n        //var footerHeight = this.getDefaultRowHeight();\n        //total = total + (footerHeight * this.getFooterRowCount());\n        return total;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The height in pixels of a specific row in the hypergrid.\n     * @param {number} rowNum - row index of interest\n     */\n    getRowHeight: function(rowNum) {\n        var rowHeights = this.getPrivateState().rowHeights;\n        return rowHeights && rowHeights[rowNum] || this.getDefaultRowHeight();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The value is lazily initialized and comes from the properties mechanism for '`defaultRowHeight`', which should be ~20px.\n     * @returns {number} The row height in pixels.\n     */\n    getDefaultRowHeight: function() {\n        if (!this.defaultRowHeight) {\n            this.defaultRowHeight = this.resolveProperty('defaultRowHeight');\n        }\n        return this.defaultRowHeight;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc set the pixel height of a specific row\n     * @param {number} rowNum - the row index of interest\n     * @param {number} height - pixel height\n     */\n    setRowHeight: function(rowNum, height) {\n        var tableState = this.getPrivateState();\n        tableState.rowHeights[rowNum] = Math.max(5, height);\n        this.stateChanged();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc This will allow 'floating' fixed rows.\n     * @return {number} The maximum height of the fixed rows area in the hypergrid.\n     */\n    getFixedRowsMaxHeight: function() {\n        return this.getFixedRowsHeight();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The width of the fixed column area in the hypergrid.\n     */\n    getFixedColumnsWidth: function() {\n        var count = this.getFixedColumnCount();\n        var total = 0;\n        if (this.grid.isShowRowNumbers()) {\n            total = this.getColumnWidth(-1);\n        }\n        for (var i = 0; i < count; i++) {\n            total = total + this.getColumnWidth(i);\n        }\n        return total;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc This exists to support \"floating\" columns.\n     * @return {number} The total width of the fixed columns area.\n     */\n    getFixedColumnsMaxWidth: function() {\n        return this.getFixedColumnsWidth();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the scroll position in vertical dimension and notify listeners.\n     * @param {number} y - the new y value\n     */\n    _setScrollPositionY: function(y) {\n        this.setScrollPositionY(y);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the scroll position in horizontal dimension and notify listeners.\n     * @param {number} x - the new x value\n     */\n    _setScrollPositionX: function(x) {\n        this.setScrollPositionX(x);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of columns just rendered, including partially rendered columns.\n     * @param {number} count - how many columns were just rendered\n     */\n    setRenderedColumnCount: function(count) {\n        this.renderedColumnCount = count;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of rows just rendered, including partially rendered rows.\n     * @param {number} count - how many rows were just rendered\n     */\n    setRenderedRowCount: function(count) {\n        this.renderedRowCount = count;\n    },\n\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The fixed row area has been clicked, massage the details and call the real function.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - event details\n     */\n    _fixedRowClicked: function(grid, mouse) {\n        var x = this.translateColumnIndex(this.getScrollPositionX() + mouse.gridCell.x - this.getFixedColumnCount());\n        var translatedPoint = this.grid.newPoint(x, mouse.gridCell.y);\n        mouse.gridCell = translatedPoint;\n        this.fixedRowClicked(grid, mouse);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc The fixed column area has been clicked, massage the details and call the real function.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - event details\n     */\n    _fixedColumnClicked: function(grid, mouse) {\n        var translatedPoint = this.grid.newPoint(mouse.gridCell.x, this.getScrollPositionY() + mouse.gridCell.y - this.getFixedRowCount());\n        mouse.gridCell = translatedPoint;\n        this.fixedColumnClicked(grid, mouse);\n    },\n\n    moveSingleSelect: function(grid, x, y) {\n        if (this.featureChain) {\n            this.featureChain.moveSingleSelect(grid, x, y);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate setting the cursor up the feature chain of responsibility\n     * @param {Hypergrid} grid\n     */\n    setCursor: function(grid) {\n        grid.updateCursor();\n        this.featureChain.setCursor(grid);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse move to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseMove: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseMove(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling tap to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onTap: function(grid, event) {\n\n        if (this.featureChain) {\n            this.featureChain.handleTap(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling tap to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onContextMenu: function(grid, event) {\n        var proceed = grid.fireSyntheticContextMenuEvent(event);\n        if (proceed && this.featureChain) {\n            this.featureChain.handleContextMenu(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling wheel moved to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onWheelMoved: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleWheelMoved(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse up to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseUp: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseUp(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse drag to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onMouseDrag: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseDrag(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling key down to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onKeyDown: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleKeyDown(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling key up to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onKeyUp: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleKeyUp(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling double click to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onDoubleClick: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleDoubleClick(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling hold pulse to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    onHoldPulse: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleHoldPulse(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling double click to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    toggleColumnPicker: function() {\n        var dialog = this.grid.dialog;\n        var self = this;\n        if (dialog.isOpen()) {\n            dialog.close();\n        } else {\n            this.buildColumnPicker(dialog.overlay);\n            dialog.onClose = function() {\n                self.updateFromColumnPicker(dialog.overlay);\n            };\n            dialog.open();\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse down to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseDown(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc delegate handling mouse exit to the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseExit: function(grid, event) {\n        if (this.featureChain) {\n            this.featureChain.handleMouseExit(grid, event);\n            this.setCursor(grid);\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is replaced by the grid on initialization and serves as the callback\n     */\n    changed: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is replaced by the grid on initialization and serves as the callback\n     */\n    shapeChanged: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {boolean} Can re-order columns.\n     */\n    isColumnReorderable: function() {\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {Object} The properties for a specific column. These are used if no cell properties are specified.\n     * @param {index} columnIndex - the column index of interest\n     */\n    getColumnProperties: function(columnIndex) {\n        var col = this.columns[columnIndex];\n        if (!col) {\n            return isNull;\n        }\n        var properties = col.getProperties(); //TODO: returns `null` on Hypergrid.reset();\n        if (!properties) {\n            return isNull;\n        }\n        return properties;\n    },\n    setColumnProperties: function(columnIndex, properties) {\n        var columnProperties = this.allColumns[columnIndex].getProperties();\n        _(columnProperties).extendOwn(properties);\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The field at `colIndex`.\n     * @param {number} colIndex - the column index of interest\n     */\n    getField: function(colIndex) {\n        return colIndex === -1 ? 'tree' : this.getColumn(colIndex).getField();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The column heading at `colIndex'.\n     * @param {number} colIndex - the column index of interest\n     */\n    getHeader: function(colIndex) {\n        return colIndex === -1 ? 'Tree' : this.getColumn(colIndex).getHeader();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this is called by the column editor post closing; rebuild the column order indexes\n     * @param {Array} list - list of column objects from the column editor\n     */\n    setColumnDescriptors: function(lists) {\n        //assumes there is one row....\n        var visible = lists.visible;\n        var tableState = this.getPrivateState();\n\n        var columnCount = visible.length;\n        var indexes = [];\n        var i;\n        for (i = 0; i < columnCount; i++) {\n            indexes.push(visible[i].id);\n        }\n        tableState.columnIndexes = indexes;\n        this.changed();\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string[]} All the currently hidden column header labels.\n     */\n    getHiddenColumnDescriptors: function() {\n        var tableState = this.getPrivateState();\n        var indexes = tableState.columnIndexes;\n        var labels = [];\n        var columnCount = this.getColumnCount();\n        for (var i = 0; i < columnCount; i++) {\n            if (indexes.indexOf(i) === -1) {\n                labels.push({\n                    id: i,\n                    label: this.getHeader(i),\n                    field: this.getField(i)\n                });\n            }\n        }\n        return labels;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc hide columns that are specified by their indexes\n     * @param {Array} arrayOfIndexes - an array of column indexes to hide\n     */\n    hideColumns: function(arrayOfIndexes) {\n        var tableState = this.getPrivateState();\n        var order = tableState.columnIndexes;\n        for (var i = 0; i < arrayOfIndexes.length; i++) {\n            var each = arrayOfIndexes[i];\n            if (order.indexOf(each) !== -1) {\n                order.splice(order.indexOf(each), 1);\n            }\n        }\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {integer} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        return this.getPrivateState().fixedColumnCount || 0;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc set the number of fixed columns\n     * @param {number} n - the integer count of how many columns to be fixed\n     */\n    setFixedColumnCount: function(n) {\n        this.getPrivateState().fixedColumnCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {integer} The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        if (!this.tableState) {\n            return 0;\n        }\n        var headers = this.grid.getHeaderRowCount();\n        var usersSize = this.tableState.fixedRowCount || 0;\n        return headers + usersSize;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Set the number of fixed rows, which includes (top to bottom order):\n     * 1. The header rows\n     *    1. The header labels row (optional)\n     *    2. The filter row (optional)\n     *    3. The top total rows (0 or more)\n     * 2. The non-scrolling rows (externally called \"the fixed rows\")\n     *\n     * @returns {number} Sum of the above or 0 if none of the above are in use.\n     *\n     * @param {number} n - The number of rows.\n     */\n    setFixedRowCount: function(n) {\n        this.tableState.fixedRowCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of header rows.\n     * A portion of the number returned by {@link Behavior#getFixedRowCount()|getFixedRowCount()}.\n     * (The remaining _fixed rows_ are the _top totals_ rows.)\n     */\n    getHeaderRowCount: function() {\n        var header = this.grid.isShowHeaderRow() ? 1 : 0;\n        var filter = this.grid.isShowFilterRow() ? 1 : 0;\n        var totals = this.getTopTotals().length;\n        return header + filter + totals;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of footer rows, consisting entirely of 0 or more _bottom totals_ rows.\n     */\n    getFooterRowCount: function() {\n        return this.getBottomTotals().length;\n    },\n\n    getTopTotals: function() {\n        return this.getDataModel().getTopTotals();\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @summary Set the number of header rows.\n     * @param {number} n - The number of _fixed rows_ to reserve as header rows.\n     * (The remaining _fixed rows_ are the _top totals_ rows.)\n     */\n    setHeaderRowCount: function(n) {\n        this.tableState.headerRowCount = n;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The number of fixed rows.\n     */\n    getHeaderColumnCount: function() {\n        return this.grid.resolveProperty('headerColumnCount');\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} The number of fixed rows.\n     */\n    setHeaderColumnCount: function(numberOfHeaderColumns) {\n        this.tableState.headerColumnCount = numberOfHeaderColumns;\n    },\n    /**\n     * @memberOf Behavior.prototype\n     * @desc build and open the editor within the container div argument\n     * @return {boolean} `false` prevents editor from opening\n     * @param {HTMLDivElement} div - the containing div element\n     */\n    buildColumnPicker: function(div) {\n        var container = document.createElement('div');\n\n        var hidden = document.createElement('fin-hypergrid-dnd-list');\n        var visible = document.createElement('fin-hypergrid-dnd-list');\n\n        container.appendChild(hidden);\n        container.appendChild(visible);\n\n        this.beColumnStyle(hidden.style);\n        hidden.title = 'hidden columns';\n        hidden.list = this.getHiddenColumnDescriptors();\n\n        this.beColumnStyle(visible.style);\n        visible.style.left = '50%';\n        visible.title = 'visible columns';\n        visible.list = this.getColumnDescriptors();\n\n        div.lists = {\n            hidden: hidden.list,\n            visible: visible.list\n        };\n        div.appendChild(container);\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc the editor is requesting close; deal with the edits\n     * @return `true`\n     * @param {HTMLDivElement} div - the containing div element\n     */\n    updateFromColumnPicker: function(div) {\n        var lists = div.lists;\n        this.setColumnDescriptors(lists);\n        return true;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc a dnd column has just been dropped, we've been notified\n     */\n    endDragColumnNotification: function() {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc bind column editor appropriate css values to arg style\n     * @param {HTMLStyleElement} style - the style object to enhance\n     */\n    beColumnStyle: function(style) {\n        style.top = '5%';\n        style.position = 'absolute';\n        style.width = '50%';\n        style.height = '100%';\n        style.whiteSpace = 'nowrap';\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {null} the cursor at a specific x,y coordinate\n     * @param {number} x - the x coordinate\n     * @param {number} y - the y coordinate\n     */\n    getCursorAt: function(x, y) {\n        return null;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {number} The total number of columns.\n     */\n    getColumnCount: function() {\n        return this.columns.length;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {string} The column alignment at column `x`: `'left'`, `'center'` , or `'right'`\n     * @param {number} x - The column index of interest.\n     */\n    getColumnAlignment: function(x) {\n        return 'center';\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Quietly set the horizontal scroll position.\n     * @param {number} x - The new position in pixels.\n     */\n    setScrollPositionX: function(x) {\n        this.scrollPositionX = x;\n    },\n\n    getScrollPositionX: function() {\n        return this.scrollPositionX;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc Quietly set the vertical scroll position.\n     * @param {number} y - The new position in pixels.\n     */\n    setScrollPositionY: function(y) {\n        this.scrollPositionY = y;\n    },\n\n    getScrollPositionY: function() {\n        return this.scrollPositionY;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {cellEditor} The cell editor for the cell at cell coordinates `x,y`\n     * @param {number} x - The horizontal cell coordinate.\n     * @param {number} y - The vertical cell coordinate.\n     */\n    _getCellEditorAt: function(x, y) {\n        var editor = this.getColumn(x).getCellEditorAt(x, y);\n\n        if (!editor) {\n            var column = this.getColumn(x);\n            var type = this.grid.isFilterRow(y) ? column.getFilterType() : column.getType();\n            editor = this.grid.resolveCellEditor(type);\n        }\n\n        return editor;\n    },\n\n    getCellEditorAt: function(x, y) {\n        if (this.grid.isFilterRow(y)) {\n            return this.grid.cellEditors.textfield;\n        }\n        return this.getDataModel().getCellEditorAt(x, y);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} x - The column index.\n     * @param {string[]} keys\n     */\n    toggleSort: function(x, keys) {\n        this.getColumn(x).toggleSort(keys);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {boolean} `true` if we should highlight on hover\n     * @param {boolean} isColumnHovered - the column is hovered or not\n     * @param {boolean} isRowHovered - the row is hovered or not\n     */\n    highlightCellOnHover: function(isColumnHovered, isRowHovered) {\n        return isColumnHovered && isRowHovered;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellPropertiesPrePaintNotification: function(cellProperties) {\n        var row = this.getRow(cellProperties.y);\n        var columnId = this.getHeader(cellProperties.x);\n        cellProperties.row = row;\n        cellProperties.columnId = columnId;\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a fixed row cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellFixedRowPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a fixed column cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellFixedColumnPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function is a hook and is called just before the painting of a top left cell occurs\n     * @param {window.fin.rectangular.Point} cell\n     */\n    cellTopLeftPrePaintNotification: function(cell) {\n\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc this function enhance the double click event just before it's broadcast to listeners\n     * @param {Object} event - event to enhance\n     */\n    enhanceDoubleClickEvent: function(event) {},\n\n    /**\n     * @memberOf Behavior.prototype\n     * @desc swap src and tar columns\n     * @param {number} src - column index\n     * @param {number} tar - column index\n     */\n    swapColumns: function(source, target) {\n        var columns = this.columns;\n        var tmp = columns[source];\n        columns[source] = columns[target];\n        columns[target] = tmp;\n        this.changed();\n    },\n\n    getColumnEdge: function(c, renderer) {\n        return this.getDataModel().getColumnEdge(c, renderer);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @param {number} x - column index\n     * @param {number} y - totals row index local to the totals area\n     * @param value\n     * @param {boolean} [atBottom=false] - this value is in the \"bottom\" totals area\n     */\n    setTotalsValue: function(x, y, value, atBottom) {\n        this.grid.setTotalsValueNotification(x, y, value, !!atBottom);\n    },\n\n    /**\n     * @memberOf Behavior.prototype\n     * @return {object} The object at y index.\n     * @param {number} y - the row index of interest\n     */\n    getRow: function(y) {\n        return this.getDataModel().getRow(y);\n    },\n\n    convertViewPointToDataPoint: function(viewPoint) {\n        var newX = this.getColumn(viewPoint.x).index;\n        var newPoint = this.grid.newPoint(newX, viewPoint.y);\n        return newPoint;\n    },\n\n    setGroups: function(arrayOfColumnIndexes) {\n        this.getDataModel().setGroups(arrayOfColumnIndexes);\n        this.createColumns();\n        this.changed();\n    },\n\n    setAggregates: function(mapOfKeysToFunctions) {\n        var self = this;\n        this.getDataModel().setAggregates(mapOfKeysToFunctions);\n        this.createColumns();\n        setTimeout(function() {\n            self.changed();\n        }, 100);\n    },\n\n    hasHierarchyColumn: function() {\n        return false;\n    },\n\n    getRowContextFunction: function(selectedRows) {\n        return function() {\n            return null;\n        };\n    },\n\n    getSelectionMatrixFunction: function(selectedRows) {\n        return function() {\n            return null;\n        };\n    },\n\n    getFieldName: function(index) {\n        return this.getFields()[index];\n    },\n\n    getColumnIndex: function(fieldName) {\n        return this.getFields().indexOf(fieldName);\n    },\n\n    getComputedRow: function(y) {\n        return this.getDataModel().getComputedRow(y);\n    },\n\n    autosizeAllColumns: function() {\n        this.checkColumnAutosizing(true);\n        this.changed();\n    },\n\n    checkColumnAutosizing: function(force) {\n        force = force === true;\n        this.autoSizeRowNumberColumn();\n        this.allColumns[-2].checkColumnAutosizing(force);\n        this.allColumns.forEach(function(column) {\n            column.checkColumnAutosizing(force);\n        });\n    },\n\n    autoSizeRowNumberColumn: function() {\n        if (this.grid.isRowNumberAutosizing()) {\n            this.allColumns[-1].checkColumnAutosizing(true);\n        }\n    },\n\n    setGlobalFilter: function(string) {\n        this.getDataModel().setGlobalFilter(string);\n    },\n\n    getSelectedRows: function() {\n        return this.grid.selectionModel.getSelectedRows();\n    },\n\n    getSelectedColumns: function() {\n        return this.grid.selectionModel.getSelectedColumns();\n    },\n\n    getSelections: function() {\n        return this.grid.selectionModel.getSelections();\n    },\n\n    getData: function() {\n        return this.getDataModel().getData();\n    },\n\n    getFilteredData: function() {\n        return this.getDataModel().getFilteredData();\n    },\n});\n\nmodule.exports = Behavior;\n","/* eslint-env browser */\n\n'use strict';\n\nvar _ = require('object-iterators');\n\nfunction Column(behavior, index, label) {\n    this.behavior = behavior;\n    this.dataModel = behavior.getDataModel();\n    this.index = index;\n    this.label = label;\n}\n\nColumn.prototype = {\n    constructor: Column.prototype.constructor,\n\n    getUnfilteredValue: function(y) {\n        return this.dataModel.getUnfilteredValue(this.index, y);\n    },\n\n    getValue: function(y) {\n        return this.dataModel.getValue(this.index, y);\n    },\n\n    setValue: function(y, value) {\n        return this.dataModel.setValue(this.index, y, value);\n    },\n\n    getWidth: function() {\n        var properties = this.getProperties();\n        return properties && properties.width || this.behavior.resolveProperty('defaultColumnWidth');\n    },\n\n    setWidth: function(width) {\n        this.getProperties().width = Math.max(5, width);\n    },\n\n    getCellRenderer: function(config, y) {\n        return this.dataModel.getCellRenderer(config, this.index, y);\n    },\n\n    getCellProperties: function(y) {\n        return this.behavior.getPrivateState().cellProperties[this.index + ',' + y];\n    },\n\n    setCellProperties: function(y, value) {\n        this.behavior.getPrivateState().cellProperties[this.index + ',' + y] = value;\n    },\n\n    setComplexFilter: function(data) {\n        this.getProperties().complexFilter = data;\n    },\n\n    getComplexFilter: function() {\n        return this.getProperties().complexFilter;\n    },\n\n    checkColumnAutosizing: function(force) {\n        var properties = this.getProperties();\n        var a, b, d;\n        if (properties) {\n            a = properties.width;\n            b = properties.preferredWidth || a;\n            d = properties.columnAutosized && !force;\n            if (a !== b || !d) {\n                properties.width = !d ? b : Math.max(a, b);\n                properties.columnAutosized = !isNaN(properties.width);\n            }\n        }\n    },\n\n    getCellType: function(y) {\n        var value = this.getValue(y);\n        var type = this.typeOf(value);\n        return type;\n    },\n\n    getFilterType: function() {\n        // var props = this.getProperties();\n        // var type = props.filterType;\n        // if (!type) {\n        //     type = this.getType();\n        //     if (type !== 'unkknown') {\n        //         props.type = type;\n        //     }\n        // }\n        // return type;\n        return 'filter';\n    },\n\n    getType: function() {\n        var props = this.getProperties();\n        var type = props.type;\n        if (!type) {\n            type = this.computeColumnType();\n            if (type !== 'unkknown') {\n                props.type = type;\n            }\n        }\n        return type;\n    },\n\n    computeColumnType: function() {\n        var headerRowCount = this.behavior.getHeaderRowCount();\n        var height = this.behavior.getRowCount();\n        var value = this.getValue(headerRowCount);\n        var eachType = this.typeOf(value);\n        if (!eachType) {\n            return 'unknown';\n        }\n        var type = this.typeOf(value);\n        var isNumber = ((typeof value) === 'number');\n        for (var y = headerRowCount; y < height; y++) {\n            value = this.getValue(y);\n            eachType = this.typeOf(value);\n            if (type !== eachType) {\n                if (isNumber && (typeof value === 'number')) {\n                    type = 'float';\n                } else {\n                    return 'mixed';\n                }\n            }\n        }\n        return type;\n    },\n\n    typeOf: function(something) {\n        var typeOf = typeof something;\n        switch (typeOf) {\n            case 'object':\n                return something.constructor.name.toLowerCase();\n            case 'number':\n                return parseInt(something) === something ? 'int' : 'float';\n            default:\n                return typeOf;\n        }\n    },\n\n    getProperties: function() {\n        return this.behavior.getPrivateState().columnProperties[this.index];\n    },\n\n    setProperties: function(properties) {\n        var current = this.behavior.getPrivateState().columnProperties[this.index];\n        this.clearObjectProperties(current, false);\n        _(current).extendOwn(properties);\n    },\n\n    toggleSort: function(keys) {\n        this.dataModel.toggleSort(this.index, keys);\n    },\n\n    getCellEditorAt: function(x, y) {\n        return this.dataModel.getCellEditorAt(this.index, y);\n    },\n\n    getHeader: function() {\n        return this.label;\n    },\n\n    getField: function() {\n        return this.dataModel.getFields()[this.index];\n    }\n};\n\nmodule.exports = Column;\n","'use strict';\n\nvar deprecated = require('../lib/deprecated');\n\nfunction DataModelDecorator(grid, component) {\n    this.setComponent(component);\n    this.setGrid(grid);\n}\n\nDataModelDecorator.prototype = {\n    constructor: DataModelDecorator.prototype.constructor,\n\n    deprecated: deprecated,\n\n    component: null,\n    grid: null,\n\n    /** @deprecated Use `.grid` property instead. */\n    getGrid: function() {\n        return this.deprecated('grid', { since: '0.2' });\n    },\n\n    setGrid: function(newGrid) {\n        this.grid = newGrid;\n        this.getComponent().setGrid(newGrid);\n    },\n\n    /** @deprecated Use `.grid.behavior` property instead. */\n    getBehavior: function() {\n        return this.deprecated('grid.behavior', { since: '0.2' });\n    },\n\n    changed: function() {\n        this.grid.behavior.changed();\n    },\n\n    getPrivateState: function() {\n        return this.grid.getPrivateState();\n    },\n\n    applyState: function() {\n\n    },\n\n    setComponent: function(newComponent) {\n        this.component = newComponent;\n    },\n\n    getComponent: function() {\n        return this.component;\n    },\n\n    setGlobalFilter: function(string) {\n        return this.getComponent().setGlobalFilter(string);\n    },\n\n    getData: function() {\n        return this.getComponent().getData();\n    },\n\n    getFilteredData: function() {\n        return this.getComponent().getFilteredData();\n    },\n\n    getValue: function(x, y) {\n        return this.getComponent().getValue(x, y);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        return this.getComponent().getUnfilteredValue(x, y);\n    },\n\n    setValue: function(x, y, value) {\n        this.getComponent().setValue(x, y, value);\n    },\n\n    getColumnCount: function() {\n        return this.getComponent().getColumnCount();\n    },\n\n    applyAnalytics: function() {\n        return this.getComponent().applyAnalytics();\n    },\n\n    getRowCount: function() {\n        return this.getComponent().getRowCount();\n    },\n\n    getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) {\n        return this.getComponent().getCellRenderer(config, x, y, untranslatedX, untranslatedY);\n    },\n\n    getRowHeight: function(y) {\n        return this.getComponent().getRowHeight(y);\n    },\n\n    getColumnEdge: function(x, renderer) {\n        return this.getComponent().getColumnEdge(x, renderer);\n    },\n\n    getColumnWidth: function(x) {\n        return this.getComponent().getColumnWidth(x);\n    },\n\n    setColumnWidth: function(x, width) {\n        this.getComponent().setColumnWidth(x, width);\n    },\n\n    toggleSort: function(x, keys) {\n        this.getComponent().toggleSort(x, keys);\n    },\n\n    getColumnProperties: function(columnIndex) {\n        return this.getComponent().getColumnProperties(columnIndex);\n    },\n\n    setColumnProperties: function(columnIndex, properties) {\n        this.getComponent().setColumnProperties(columnIndex, properties);\n    },\n\n    getHeaders: function() {\n        return this.getComponent().getHeaders();\n    },\n\n    getFields: function() {\n        return this.getComponent().getFields();\n    },\n\n    setFields: function(fields) {\n        this.getComponent().setFields(fields);\n    },\n\n    getCellProperties: function(x, y) {\n        return this.getComponent().getCellProperties(x, y);\n    },\n\n    setCellProperties: function(x, y, value) {\n        this.getComponent().setCellProperties(x, y, value);\n    },\n\n    getRow: function(y) {\n        return this.getComponent().getRow(y);\n    },\n\n    getTopTotals: function() {\n        return this.getComponent().getTopTotals();\n    },\n\n    setTopTotals: function(totalRows) {\n        this.getComponent().setTopTotals(totalRows);\n    },\n\n    getBottomTotals: function() {\n        return this.getComponent().getBottomTotals();\n    },\n\n    setBottomTotals: function(totalRows) {\n        this.getComponent().setBottomTotals(totalRows);\n    },\n\n    setData: function(y) {\n        return this.getComponent().setData(y);\n    },\n\n    hasHierarchyColumn: function() {\n        return this.getComponent().hasHierarchyColumn();\n    },\n\n    setHeaders: function(headerLabels) {\n        return this.getComponent().setHeaders(headerLabels);\n    },\n\n    cellClicked: function(cell, event) {\n        return this.getComponent().cellClicked(cell, event);\n    },\n\n    getAvailableGroups: function() {\n        return this.getComponent().getAvailableGroups();\n    },\n\n    getGroups: function() {\n        return this.getComponent().getGroups();\n    },\n\n    setGroups: function(groups) {\n        this.getComponent().setGroups(groups);\n    },\n\n    getHiddenColumns: function() {\n        return this.getComponent().getHiddenColumns();\n    },\n\n    getVisibleColumns: function() {\n        return this.getComponent().getVisibleColumns();\n    },\n\n    setAggregates: function(aggregates) {\n        return this.getComponent().setAggregates(aggregates);\n    },\n\n    reset: function() {\n        this.getComponent().reset();\n    },\n\n    getCellEditorAt: function(x, y) {\n        return this.getComponent().getCellEditorAt(x, y);\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.getComponent().getUnfilteredRowCount();\n    }\n};\n\nmodule.exports = DataModelDecorator;\n","'use strict';\n\nvar ListDragon = require('list-dragon');\n\nvar Local = require('./Local');\nvar DataModelDecorator = require('./DataModelDecorator');\nvar DataModelJSON = require('../dataModels/JSON');\nvar features = require('../features/index');\nvar addStylesheet = require('../../css/stylesheets');\n//var aggregations = require('hyper-analytics').util.aggregations;\n//var aggregations = require('../local_node_modules/hyper-analytics').util.aggregations;\nvar aggregations = require('../local_node_modules/finanalytics').aggregations;\n\n/**\n * @name behaviors.JSON\n * @desc > Same parameters as {@link behaviors.Behavior#initialize|initialize}, which is called by this constructor.\n * @constructor\n */\nvar JSON = Local.extend('behaviors.JSON', {\n\n    /**\n     * @summary Constructor logic, called _after_{@link Behavior#initialize|Behavior.initialize()}.\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     *\n     * @param grid - the hypergrid\n     * @param {object[]} dataRows - array of uniform data objects\n     * @memberOf behaviors.JSON.prototype\n     */\n    initialize: function(grid, dataRows) {\n        this.setData(dataRows);\n    },\n\n    features: [\n        features.CellSelection,\n        features.KeyPaging,\n        features.ColumnPicker,\n        features.ColumnResizing,\n        features.RowResizing,\n        features.Filters,\n        features.RowSelection,\n        features.ColumnSelection,\n        features.ColumnMoving,\n        features.ColumnSorting,\n        features.CellEditing,\n        features.CellClick,\n        features.OnHover\n    ],\n\n    aggregations: aggregations,\n\n    createColumns: function() {\n        var dataModel = this.getDataModel();\n        var columnCount = dataModel.getColumnCount();\n        var headers = dataModel.getHeaders();\n        var fields = dataModel.getFields();\n        this.clearColumns();\n        for (var i = 0; i < columnCount; i++) {\n            var header = headers[i];\n            var column = this.addColumn(i, header);\n            var properties = column.getProperties();\n            properties.field = fields[i];\n            properties.header = header;\n            properties.complexFilter = null;\n        }\n    },\n\n    getDefaultDataModel: function() {\n        var model = new DataModelJSON();\n        var wrapper = new DataModelDecorator(this.grid, model);\n        wrapper.setComponent(model); // TODO: Redundant? Already performed in DataModelDecorator()?\n        return wrapper;\n    },\n\n    applyAnalytics: function() {\n        this.dataModel.applyAnalytics();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the header labels.\n     * @param {string[]} headerLabels - The header labels.\n     */\n    setHeaders: function(headerLabels) {\n        this.getDataModel().setHeaders(headerLabels);\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @desc * @returns {string[]} The header labels.\n     */\n    getHeaders: function() {\n        return this.getDataModel().getHeaders();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the fields array.\n     * @param {string[]} fieldNames - The field names.\n     */\n    setFields: function(fieldNames) {\n        //were defining the columns based on field names....\n        //we must rebuild the column definitions\n        this.getDataModel().setFields(fieldNames);\n        this.createColumns();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Get the field names.\n     * @returns {string[]}\n     */\n    getFields: function() {\n        return this.getDataModel().getFields();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Set the data field.\n     * @param {object[]} objects - An array of uniform objects, each being a row in the grid.\n     */\n    setData: function(dataRows) {\n        this.getDataModel().setData(dataRows);\n        this.createColumns();\n        var self = this;\n        if (this.grid.isColumnAutosizing()) {\n            setTimeout(function() {\n                self.autosizeAllColumns();\n            }, 100);\n            self.changed();\n        } else {\n            setTimeout(function() {\n                self.allColumns[-1].checkColumnAutosizing(true);\n                self.changed();\n            });\n        }\n    },\n\n    /**\n     * @summary Set the top totals.\n     * @memberOf behaviors.JSON.prototype\n     * @param {Array<Array>} totalRows - array of rows (arrays) of totals\n     */\n    setTopTotals: function(totalRows) {\n        this.getDataModel().setTopTotals(totalRows);\n    },\n\n    /**\n     * @summary Get the top totals.\n     * @memberOf behaviors.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getTopTotals: function() {\n        return this.getDataModel().getTopTotals();\n    },\n\n    /**\n     * @summary Set the bottom totals.\n     * @memberOf behaviors.JSON.prototype\n     * @param {Array<Array>} totalRows - array of rows (arrays) of totals\n     */\n    setBottomTotals: function(totalRows) {\n        this.getDataModel().setBottomTotals(totalRows);\n    },\n\n    /**\n     * @summary Get the bottom totals.\n     * @memberOf behaviors.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getBottomTotals: function() {\n        return this.getDataModel().getBottomTotals();\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Build the fields and headers from the supplied column definitions.\n     * ```javascript\n     * myJsonBehavior.setColumns([\n     *     { title: 'Stock Name', field: 'short_description' },\n     *     { title: 'Status', field: 'trading_phase' },\n     *     { title: 'Reference Price', field: 'reference_price' }\n     * ]);\n     * ```\n     * @param {Array} columnDefinitions - an array of objects with fields 'title', and 'field'\n     */\n    setColumns: function(columnDefinitions) {\n        this.getDataModel().setColumns(columnDefinitions);\n    },\n\n    /**\n     * @memberOf behaviors.JSON.prototype\n     * @description Enhance the double-click event just before it's broadcast to listeners.\n     * @param {Point} event\n     */\n    enhanceDoubleClickEvent: function(event) {\n        event.row = this.getRow(event.gridCell.y);\n    },\n\n    setDataProvider: function(dataProvider) {\n        this.getDataModel().setDataProvider(dataProvider);\n    },\n\n    hasHierarchyColumn: function() {\n        return this.getDataModel().hasHierarchyColumn();\n    },\n\n    getColumnAlignment: function(x) {\n        if (x === 0 && this.hasHierarchyColumn()) {\n            return 'left';\n        } else {\n            return 'center';\n        }\n    },\n\n    getRowSelectionMatrix: function(selectedRows) {\n        return this.getDataModel().getRowSelectionMatrix(selectedRows);\n    },\n\n    getColumnSelectionMatrix: function(selectedColumns) {\n        return this.getDataModel().getColumnSelectionMatrix(selectedColumns);\n    },\n\n    getSelectionMatrix: function(selections) {\n        return this.getDataModel().getSelectionMatrix(selections);\n    },\n\n    getRowSelection: function() {\n        var selectedRows = this.getSelectedRows();\n        return this.getDataModel().getRowSelection(selectedRows);\n    },\n\n    getColumnSelection: function() {\n        var selectedColumns = this.getSelectedColumns();\n        return this.getDataModel().getColumnSelection(selectedColumns);\n    },\n\n    getSelection: function() {\n        var selections = this.getSelections();\n        return this.getDataModel().getSelection(selections);\n    },\n\n    buildColumnPicker: function(div) {\n        if (!this.isColumnReorderable()) {\n            return false;\n        }\n\n        var listOptions = {\n            cssStylesheetReferenceElement: div\n        };\n\n        var groups = { models: this.getGroups(), title: 'Groups' },\n            availableGroups = { models: this.getAvailableGroups(), title: 'Available Groups' },\n            hiddenColumns = { models: this.getHiddenColumns(), title: 'Hidden Columns' },\n            visibleColumns = { models: this.getVisibleColumns(), title: 'Visible Columns'},\n            groupLists = new ListDragon([groups, availableGroups], listOptions),\n            columnLists = new ListDragon([hiddenColumns, visibleColumns], listOptions),\n            listSets = [groupLists, columnLists];\n\n        addStylesheet('list-dragon', div);\n\n        listSets.forEach(function(listSet) {\n            listSet.modelLists.forEach(function(list) {\n                div.appendChild(list.container);\n            });\n        });\n\n        //attach for later retrieval\n        div.lists = {\n            group: groups.models,\n            availableGroups: availableGroups.models,\n            hidden: hiddenColumns.models,\n            visible: visibleColumns.models\n        };\n\n        return true;\n    },\n    getGroups: function() {\n        return this.getDataModel().getGroups();\n    },\n    getAvailableGroups: function() {\n        return this.getDataModel().getAvailableGroups();\n    },\n    getHiddenColumns: function() {\n        return this.getDataModel().getHiddenColumns();\n    },\n    getVisibleColumns: function() {\n        return this.getDataModel().getVisibleColumns();\n    },\n    setColumnDescriptors: function(lists) {\n        //assumes there is one row....\n        var tree = this.columns[0];\n        this.columns.length = 0;\n        if (tree && tree.label === 'Tree') {\n            this.columns.push(tree);\n        }\n        for (var i = 0; i < lists.visible.length; i++) {\n            this.columns.push(lists.visible[i]);\n        }\n\n        var groupBys = lists.group.map(function(e) {\n            return e.id;\n        });\n        this.getDataModel().setGroups(groupBys);\n\n        this.changed();\n    },\n\n    getSelectedRows: function() {\n        var offset = -this.grid.getHeaderRowCount();\n        var selections = this.grid.selectionModel.getSelectedRows();\n        var result = selections.map(function(each) {\n            return each + offset;\n        });\n        return result;\n    },\n\n    getSelectedColumns: function() {\n        return this.grid.selectionModel.getSelectedColumns();\n    },\n\n    getSelections: function() {\n        return this.grid.selectionModel.getSelections();\n    }\n\n});\n\nmodule.exports = JSON;\n","'use strict';\n\n//var ListDragon = require('list-dragon');\n\nvar Behavior = require('./Behavior');\n//var DataModelDecorator = require('./DataModelDecorator');\n//var DataModelJSON = require('../dataModels/JSON');\n//var features = require('../features/index');\n//var addStylesheet = require('../stylesheets');\n////var aggregations = require('hyper-analytics').util.aggregations;\n////var aggregations = require('../local_node_modules/hyper-analytics').util.aggregations;\n//var aggregations = require('../local_node_modules/finanalytics').aggregations;\n\n/**\n * @name behaviors.Local\n * @desc > Same parameters as {@link behaviors.Behavior#initialize|initialize}, which is called by this constructor.\n * @constructor\n */\nvar Local = Behavior.extend('Local', {\n\n});\n\nmodule.exports = Local;\n","'use strict';\n\nvar Behavior = require('./Behavior');\n\nvar noop = function() {},\n    n00p = function() { return 0; };\n\n/**\n * @constructor\n */\nvar Null = Behavior.extend('Null', {\n\n    //initalize: function(grid, component) {},\n\n    setScrollPositionY: noop,\n    setScrollPositionX: noop,\n    getColumnCount: n00p,\n    getFixedColumnCount: n00p,\n    getFixedColumnsWidth: n00p,\n    getFixedColumnsMaxWidth: n00p,\n    setRenderedWidth: n00p,\n    getRowCount: n00p,\n    getFixedRowCount: n00p,\n    getFixedRowsHeight: n00p,\n    getFixedRowsMaxHeight: n00p,\n    setRenderedHeight: n00p,\n    getCellProvider: noop,\n    click: noop,\n    doubleClick: noop\n});\n\nmodule.exports = Null;\n","'use strict';\n\nmodule.exports = {\n    Behavior: require('./Behavior'), // abstract base class\n    JSON: require('./JSON'),\n    Null: require('./Null')\n};","/* eslint-env browser */\n\n'use strict';\n\nvar mustache = require('mustache');\nvar Base = require('../lib/Base');\n\nvar extract = /\\/\\*\\s*([^]+?)\\s+\\*\\//; // finds the string inside the /* ... */; the (group) excludes the whitespace\n\n/**\n * @constructor\n */\nvar CellEditor = Base.extend('CellEditor', {\n\n    alias: 'base',\n\n    /**\n     * am I currently editing (i.e., between calls to `beginEditAt` and either `stopEditing` or `cancelEditing`)\n     * @type {boolean}\n     * @default false\n     * @memberOf CellEditor.prototype\n     */\n    isEditing: false,\n\n    /**\n     * the point that I am editing at right now\n     * @type {Point}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    editorPoint: {\n        x: -1,\n        y: -1\n    },\n\n    /**\n     * if true, check that the editor is in the right location\n     * @type {boolean}\n     * @default false\n     * @memberOf CellEditor.prototype\n     */\n    checkEditorPositionFlag: false,\n\n    /**\n     * my instance of hypergrid\n     * @type {Hypergrid}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    grid: null,\n\n    /**\n     * the value before editing\n     * @type {type}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    initialValue: null,\n\n    /** @deprecated Use `.grid.behavior` property instead.\n     * @memberOf CellEditor.prototype\n     * @returns {Behavior} The behavior (model).\n     */\n    getBehavior: function() {\n        return this.deprecated('grid.behavior', { since: '0.2' });\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc This function is a callback from the fin-hypergrid.   It is called after each paint of the canvas.\n     */\n    gridRenderedNotification: function() {\n        this.checkEditor();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc scroll values have changed, we've been notified\n     */\n    scrollValueChangedNotification: function() {\n        this.setCheckEditorPositionFlag();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc turn on checkEditorPositionFlag boolean field\n     */\n    setCheckEditorPositionFlag: function() {\n        this.checkEditorPositionFlag = true;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc begin editing at location point\n     * @param {Point} point - the location to start editing at\n     */\n    beginEditAt: function(point) {\n        this.editorPoint = point;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc put value into our editor\n     * @param {object} value - whatever value we want to edit\n     */\n    setEditorValue: function(value) {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc returns the point at which we are currently editing\n     * @returns {Point}\n     */\n    getEditorPoint: function() {\n        return this.editorPoint;\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc set the current editor location\n     * @param {Point} point - the data location of the current editor\n     */\n    setEditorPoint: function(point) {\n        this.editorPoint = point;\n        this.modelPoint = this.grid.convertViewPointToDataPoint(point);\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc display the editor\n     */\n    showEditor: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc hide the editor\n     */\n    hideEditor: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc stop editing\n     */\n    stopEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        var proceed = this.grid.fireSyntheticEditorDataChangeEvent(this, this.initialValue, this.getEditorValue, this);\n        if (!proceed) {\n            return;\n        }\n        this.saveEditorValue();\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    cancelEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc save the new value into the behavior(model)\n     */\n    saveEditorValue: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc return the current editor's value\n     */\n    getEditorValue: function() {},\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc request focus\n     */\n    takeFocus: function() {},\n\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc check that the editor is in the correct location, and is showing/hidden appropriately\n     */\n    checkEditor: function() {\n    },\n\n    /** @deprecated Use `.grid` property instead. */\n    getGrid: function() {\n        return this.deprecated('grid', { since: '0.2' });\n    },\n\n    template: function() {\n        /*\n\n         */\n    },\n\n    getHTML: function() {\n        var template = this.template.toString().match(extract)[1];\n        return mustache.render(template, this);\n    },\n\n});\n\nmodule.exports = CellEditor;\n","'use strict';\n\nvar Simple = require('./Simple');\nvar Map = require('../lib/Mappy');\n\n/**\n * @constructor\n */\nvar Choice = Simple.extend('Choice', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Choice.prototype\n     */\n    alias: 'choice',\n\n    /**\n     * the list of items to pick from\n     * @type {Array}\n     * @memberOf Choice.prototype\n     */\n    items: ['a', 'b', 'c'],\n\n    template: function() {\n        /*\n                <select id=\"editor\">\n                    {{#items}}\n                        <option value=\"{{.}}\">{{.}}</option>\n                    {{/items}}\n                </select>\n            */\n    },\n\n    autopopulate: function() {\n        var behavior = this.grid.behavior;\n        var point = this.getEditorPoint();\n        var colProps = this.grid.getColumnProperties(point.x);\n        if (!colProps.autopopulateEditor) {\n            return;\n        }\n        var headerCount = this.grid.getHeaderRowCount();\n        var rowCount = this.grid.getUnfilteredRowCount() - headerCount;\n        var column = point.x;\n        var map = new Map();\n        for (var r = 0; r < rowCount; r++) {\n            var each = behavior.getUnfilteredValue(column, r);\n            map.set(each, each);\n        }\n        var values = map.values;\n        values.sort();\n\n        if (values.length > 0 && values[0].length > 0) {\n            values.unshift('');\n        }\n\n        this.setItems(values);\n    },\n\n    //no events are fired while the dropdown is open\n    //see http://jsfiddle.net/m4tndtu4/6/\n\n    /**\n     * @memberOf Choice.prototype\n     */\n    showEditor: function() {\n        var self = this;\n        this.input.style.display = 'inline';\n        setTimeout(function() {\n            self.showDropdown(self.input);\n        }, 50);\n    },\n\n    preShowEditorNotification: function() {\n        this.autopopulate();\n        this.setEditorValue(this.initialValue);\n    },\n\n    /**\n     * @memberOf Choice.prototype\n     * @param items\n     */\n    setItems: function(items) {\n        this.items = items;\n        this.updateView();\n    },\n\n    /**\n     * @memberOf Choice.prototype\n     * @param input\n     */\n    initializeInput: function(input) {\n        var self = this;\n        Simple.prototype.initializeInput.apply(this, [input]);\n        input.onchange = function() {\n            self.stopEditing();\n        };\n    }\n\n});\n\nmodule.exports = Choice;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Color = Simple.extend('Color', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Color.prototype\n     */\n    alias: 'color',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"color\">\n        */\n    }\n\n});\n\nmodule.exports = Color;\n","/* eslint-env browser */\n\n'use strict';\n\nvar Simple = require('./Simple');\nvar Formatters = require('../lib/Formatters');\n\nfunction parseDate(input) {\n  var parts = input.match(/(\\d+)/g);\n  // new Date(year, month [, date [, hours[, minutes[, seconds[, ms]]]]])\n  return new window.Date(parts[0], parts[1] - 1, parts[2]); // months are 0-based\n}\n\n/**\n * @constructor\n */\nvar Date = Simple.extend('Date', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Date.prototype\n     */\n    alias: 'date',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"date\">\n        */\n    },\n\n    setEditorValue: function(value) {\n        if (value != null && value.constructor.name === 'Date') {\n            value = Formatters.date(value);\n        }\n        this.getInput().value = value + '';\n    },\n\n    getEditorValue: function() {\n        var value = this.getInput().value;\n        value = parseDate(value);\n        return value;\n    },\n\n});\n\nmodule.exports = Date;\n","/* eslint-env browser */\n'use strict';\n\nvar CellEditor = require('./CellEditor');\n\n/**\n * @constructor\n */\nvar Filter = CellEditor.extend('Filter', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Textfield.prototype\n     */\n    alias: 'filter',\n\n    initialize: function() {\n        var data = document.createElement('div');\n        var style = data.style;\n        style.position = 'absolute';\n        style.top = style.bottom = '44px';\n        style.right = style.left = '1em';\n        style.overflowY = 'scroll';\n\n        var table = document.createElement('table');\n        data.appendChild(table);\n\n        style = table.style;\n        style.width = style.height = '100%';\n\n        var tr = document.createElement('tr');\n        var td = document.createElement('td');\n        table.appendChild(tr);\n        tr.appendChild(td);\n\n\n        this.title = document.createElement('div');\n        this.title.innerHTML = 'Filter Editor';\n\n        this.dialog = document.createElement('div');\n\n        this.content = td;\n        this.buttons = document.createElement('div');\n\n        style = this.dialog.style;\n        style.position = 'absolute';\n        style.top = style.left = style.right = style.bottom = 0;\n        style.whiteSpace = 'nowrap';\n\n        style = this.title.style;\n        style.position = 'absolute';\n        style.top = style.left = style.right = 0;\n        style.height = '44px';\n        style.whiteSpace = 'nowrap';\n        style.textAlign = 'center';\n        style.padding = '11px';\n\n        style = this.buttons.style;\n        style.position = 'absolute';\n        style.left = style.right = style.bottom = 0;\n        style.height = '44px';\n        style.whiteSpace = 'nowrap';\n        style.textAlign = 'center';\n        style.padding = '8px';\n\n        this.dialog.appendChild(this.title);\n        this.dialog.appendChild(data);\n        this.dialog.appendChild(this.buttons);\n\n        this.ok = document.createElement('button');\n        this.ok.style.borderRadius = '8px';\n        this.ok.style.width = '5.5em';\n\n        this.cancel = document.createElement('button');\n        this.cancel.style.marginLeft = '2em';\n        this.cancel.style.borderRadius = '8px';\n        this.cancel.style.width = '5.5em';\n\n        this.delete = document.createElement('button');\n        this.delete.style.marginLeft = '2em';\n        this.delete.style.borderRadius = '8px';\n        this.delete.style.width = '5.5em';\n\n        this.reset = document.createElement('button');\n        this.reset.style.marginLeft = '2em';\n        this.reset.style.borderRadius = '8px';\n        this.reset.style.width = '5.5em';\n\n        this.ok.innerHTML = 'ok';\n        this.cancel.innerHTML = 'cancel';\n        this.delete.innerHTML = 'delete';\n        this.reset.innerHTML = 'reset';\n\n        this.buttons.appendChild(this.ok);\n        this.buttons.appendChild(this.reset);\n        this.buttons.appendChild(this.delete);\n        this.buttons.appendChild(this.cancel);\n\n        var self = this;\n        this.ok.onclick = function() {\n            self.okPressed();\n        };\n        this.cancel.onclick = function() {\n            self.cancelPressed();\n        };\n        this.delete.onclick = function() {\n            self.deletePressed();\n        };\n        this.reset.onclick = function() {\n            self.resetPressed();\n        };\n    },\n\n    tearDown: function() {\n        this.content.innerHTML = '';\n    },\n\n    okPressed: function() {\n        var dialog = this.grid.dialog;\n        dialog.onOkPressed();\n    },\n\n    cancelPressed: function() {\n        var dialog = this.grid.dialog;\n        dialog.onCancelPressed();\n    },\n\n    deletePressed: function() {\n        var dialog = this.grid.dialog;\n        dialog.onDeletePressed();\n    },\n\n    resetPressed: function() {\n        var dialog = this.grid.dialog;\n        dialog.onResetPressed();\n    },\n\n    beginEditAt: function(editorPoint) {\n        var behavior = this.grid.behavior;\n        var dialog = this.grid.dialog;\n\n        var columnIndex = editorPoint.x,\n            title = behavior.getColumnId(columnIndex),\n            field = behavior.getField(columnIndex),\n            type = behavior.getColumn(columnIndex).getType();\n\n        var fieldsProvider = function() {\n            return [{\n                name: field,\n                alias: title,\n                type: type\n            }];\n        };\n        this.title.innerHTML = 'Manage Filters';\n        var filter = this.grid.filter;\n        //var self = this;\n        if (dialog.isOpen()) {\n            dialog.close();\n        } else {\n            var self = this;\n\n            dialog.clear();\n            dialog.overlay.appendChild(this.dialog);\n\n            filter.initialize(fieldsProvider);\n\n            dialog.onOkPressed = function() {\n                if (filter.onOk && filter.onOk()) { // onOK() truthy result means abort; falsy means proceed\n                    return;\n                }\n                self.tearDown();\n                behavior.setComplexFilter(columnIndex, {\n                    //type: filter.alias,\n                    state: filter.getState()\n                });\n                dialog.close();\n                behavior.applyAnalytics();\n                behavior.changed();\n            };\n\n            dialog.onCancelPressed = function() {\n                if (filter.onCancel && filter.onCancel()) {\n                    return;\n                }\n                self.tearDown();\n                dialog.close();\n                filter = undefined;\n            };\n\n            dialog.onDeletePressed = function() {\n                if (filter.onDelete && filter.onDelete()) {\n                    return;\n                }\n                self.tearDown();\n                behavior.setComplexFilter(columnIndex, undefined);\n                dialog.close();\n                behavior.applyAnalytics();\n                behavior.changed();\n            };\n\n            dialog.onResetPressed = function() {\n                if (filter.onReset && filter.onReset()) {\n                    return;\n                }\n                self.tearDown();\n                filter.initialize(dialog);\n                if (filter.onShow) {\n                    filter.onShow(self.content);\n                }\n            };\n\n            var cellBounds = this.grid._getBoundsOfCell(columnIndex, editorPoint.y);\n\n            //hack to accomodate bootstrap margin issues...\n            var xOffset =\n                this.grid.div.getBoundingClientRect().left -\n                this.grid.divCanvas.getBoundingClientRect().left;\n            cellBounds.x = cellBounds.x - xOffset;\n            dialog.openFrom(cellBounds);\n            var previousState = behavior.getComplexFilter(columnIndex);\n            if (previousState) {\n                filter.setState(previousState.state);\n            }\n            setTimeout(function() {\n                if (filter.onShow) {\n                    filter.onShow(self.content);\n                }\n            }, dialog.getAnimationTime() + 10);\n        }\n    },\n\n});\n\nmodule.exports = Filter;\n","/* eslint-env browser */\n\n'use strict';\n\nvar CellEditor = require('./CellEditor.js');\n\n/**\n * @constructor\n */\nvar Simple = CellEditor.extend('Simple', {\n\n    /**\n     * my main input control\n     * @type {Element}\n     * @default null\n     * @memberOf CellEditor.prototype\n     */\n    input: null,\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Simple.prototype\n     */\n    alias: 'simple',\n\n    /**\n     * @memberOf Simple.prototype\n     */\n    initialize: function() {\n        this.editorPoint = {\n            x: 0,\n            y: 0\n        };\n    },\n\n    specialKeyups: {\n        //0x08: 'clearStopEditing', // backspace\n        0x09: 'stopEditing', // tab\n        0x0d: 'stopEditing', // return/enter\n        0x1b: 'cancelEditing' // escape\n    },\n\n    keyup: function(e) {\n        if (e) {\n            var specialKeyup = this.specialKeyups[e.keyCode];\n\n            if (specialKeyup) {\n                e.preventDefault();\n                this[specialKeyup]();\n                this.grid.repaint();\n                this.grid.takeFocus();\n            }\n\n            this.grid.fireSyntheticEditorKeyUpEvent(this, e);\n        }\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc  the function to override for initialization\n     */\n    initializeInput: function(input) {\n        var self = this;\n        input.addEventListener('keyup', this.keyup.bind(this));\n        input.addEventListener('keydown', function(e) {\n            self.grid.fireSyntheticEditorKeyDownEvent(self, e);\n        });\n        input.addEventListener('keypress', function(e) {\n            self.grid.fireSyntheticEditorKeyPressEvent(self, e);\n        });\n        input.onblur = function(e) {\n            self.cancelEditing();\n        };\n\n        // input.addEventListener('focusout', function() {\n        //     self.stopEditing();\n        // });\n        // input.addEventListener('blur', function() {\n        //     self.stopEditing();\n        // });\n\n        input.style.position = 'absolute';\n        input.style.display = 'none';\n        input.style.border = 'solid 2px black';\n        input.style.outline = 0;\n        input.style.padding = 0;\n        input.style.boxShadow = 'white 0px 0px 1px 1px';\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @returns {object} the current editor's value\n     */\n    getEditorValue: function() {\n        var value = this.getInput().value;\n        return value;\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc save the new value into the behavior(model)\n     */\n    setEditorValue: function(value) {\n        this.getInput().value = value;\n    },\n\n    clearStopEditing: function() {\n        this.setEditorValue('');\n        this.stopEditing();\n    },\n\n    cancelEditing: function() {\n        if (!this.isEditing) {\n            return;\n        }\n        this.getInput().value = null;\n        this.isEditing = false;\n        this.hideEditor();\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc display the editor\n     */\n    showEditor: function() {\n        this.getInput().style.display = 'inline';\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc hide the editor\n     */\n    hideEditor: function() {\n        this.getInput().style.display = 'none';\n    },\n\n    /**\n     * @summary Request focus for my input control.\n     * @desc See GRID-95 \"Scrollbar moves inward\" for issue and work-around explanation.\n     * @memberOf Simple.prototype\n     */\n    takeFocus: function() {\n        var self = this;\n        setTimeout(function() {\n            var transformWas = self.input.style.transform;\n            self.input.style.transform = 'translate(0,0)'; // work-around: move to upper left\n\n            self.input.focus();\n            self.selectAll();\n\n            self.input.style.transform = transformWas;\n        });\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc select everything\n     */\n    selectAll: function() {\n\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc how much should I offset my bounds from 0,0\n     */\n    originOffset: function() {\n        return [0, 0];\n    },\n\n    /**\n     * @memberOf Simple.prototype\n     * @desc set the bounds of my input control\n     * @param {rectangle} rectangle - the bounds to move to\n     */\n    setBounds: function(cellBounds) {\n        var originOffset = this.originOffset();\n        var translation = 'translate('\n            + (cellBounds.x - 1 + originOffset[0]) + 'px,'\n            + (cellBounds.y - 1 + originOffset[1]) + 'px)';\n\n        var input = this.getInput();\n\n        input.style.boxSizing = 'border-box';\n\n        input.style.webkitTransform = translation;\n        input.style.MozTransform = translation;\n        input.style.msTransform = translation;\n        input.style.OTransform = translation;\n\n        // TODO: Obviously this was changed at some point from left,top to trnasform:translation. Wondering why this was necessary...?\n\n        // input.style.left = cellBounds.x + originOffset[0] + 'px';\n        // input.style.top = cellBounds.y + originOffset[1] + 'px';\n\n        input.style.width = (cellBounds.width + 2) + 'px';\n        input.style.height = (cellBounds.height + 2) + 'px';\n        //var xOffset = this.grid.canvas.getBoundingClientRect().left;\n    },\n\n    saveEditorValue: function() {\n        var point = this.getEditorPoint();\n        var value = this.getEditorValue();\n        if (value === this.initialValue) {\n            return; //data didn't change do nothing\n        }\n        if (parseFloat(this.initialValue) === this.initialValue) { // I'm a number\n            value = parseFloat(value);\n        }\n        var continued = this.grid.fireBeforeCellEdit(point, this.initialValue, value, this);\n        if (!continued) {\n            return;\n        }\n        this.grid.behavior.setValue(point.x, point.y, value);\n        this.grid.fireAfterCellEdit(point, this.initialValue, value, this);\n    },\n\n    /**\n     * @memberOf CellEditor.prototype\n     * @desc move the editor to the current editor point\n     */\n    _moveEditor: function() {\n        var editorPoint = this.getEditorPoint();\n        var cellBounds = this.grid._getBoundsOfCell(editorPoint.x, editorPoint.y);\n\n        //hack to accommodate bootstrap margin issues...\n        var xOffset =\n            this.grid.div.getBoundingClientRect().left -\n            this.grid.divCanvas.getBoundingClientRect().left;\n        cellBounds.x = cellBounds.x - xOffset;\n\n        this.setBounds(cellBounds);\n    },\n\n    moveEditor: function() {\n        this._moveEditor();\n        this.takeFocus();\n    },\n\n    beginEditAt: function(point) {\n\n        if (!this.isAdded) {\n            this.isAdded = true;\n            this.attachEditor();\n        }\n\n        this.setEditorPoint(point);\n        var value = this.grid.behavior.getValue(point.x, point.y);\n        if (value.constructor.name === 'Array') {\n            value = value[1]; //it's a nested object\n        }\n        var proceed = this.grid.fireRequestCellEdit(point, value);\n        if (!proceed) {\n            //we were cancelled\n            return;\n        }\n        this.initialValue = value;\n        this.isEditing = true;\n        this.setCheckEditorPositionFlag();\n        this.checkEditor();\n    },\n\n    checkEditor: function() {\n        if (!this.checkEditorPositionFlag) {\n            return;\n        } else {\n            this.checkEditorPositionFlag = false;\n        }\n        if (!this.isEditing) {\n            return;\n        }\n        var editorPoint = this.getEditorPoint();\n        if (this.grid.isDataVisible(editorPoint.x, editorPoint.y)) {\n            this.preShowEditorNotification();\n            this.attachEditor();\n            this.moveEditor();\n            this.showEditor();\n        } else {\n            this.hideEditor();\n        }\n    },\n\n    attachEditor: function() {\n        var input = this.getInput(),\n            div = this.grid.div,\n            referenceNode = div.querySelectorAll('.finbar-horizontal, .finbar-vertical');\n\n        div.insertBefore(input, referenceNode.length ? referenceNode[0] : null);\n    },\n\n    preShowEditorNotification: function() {\n        this.setEditorValue(this.initialValue);\n    },\n\n    getInput: function() {\n        if (!this.input) {\n            this.input = this.getDefaultInput();\n        }\n        return this.input;\n    },\n\n    getDefaultInput: function() {\n        var div = document.createElement('DIV');\n        div.innerHTML = this.getHTML();\n        var input = div.firstChild;\n        this.initializeInput(input);\n        return input;\n    },\n\n    updateView: function() {\n        var oldGuy = this.getInput();\n        var parent = oldGuy.parentNode;\n        var newGuy = this.getDefaultInput();\n        this.input = newGuy;\n        parent.replaceChild(newGuy, oldGuy);\n    },\n\n    showDropdown: function(element) {\n        var event;\n        event = document.createEvent('MouseEvents');\n        event.initMouseEvent('mousedown', true, true, window);\n        element.dispatchEvent(event);\n    }\n});\n\nmodule.exports = Simple;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Slider = Simple.extend('Slider', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Slider.prototype\n     */\n    alias: 'slider',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"range\">\n        */\n    }\n\n});\n\nmodule.exports = Slider;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Spinner = Simple.extend('Spinner', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Spinner.prototype\n     */\n    alias: 'spinner',\n\n    template: function() {\n        /*\n            <input id=\"editor\" type=\"number\">\n        */\n    }\n\n});\n\nmodule.exports = Spinner;\n","'use strict';\n\nvar Simple = require('./Simple');\n\n/**\n * @constructor\n */\nvar Textfield = Simple.extend('Textfield', {\n\n    /**\n     * my lookup alias\n     * @type {string}\n     * @memberOf Textfield.prototype\n     */\n    alias: 'textfield',\n\n    template: function() {\n        /*\n            <input id=\"editor\">\n        */\n    },\n\n    selectAll: function() {\n        this.input.setSelectionRange(0, this.input.value.length);\n    },\n\n    specialKeyups: {\n        0x09: 'stopEditing', // tab\n        0x0d: 'stopEditing', // return/enter\n        0x1b: 'cancelEditing' // escape\n    },\n\n    keyup: function(e) {\n        if (e) {\n            Simple.prototype.keyup.call(this, e);\n\n            if (this.grid.isFilterRow(this.getEditorPoint().y)) {\n                setTimeout(keyup.bind(this));\n            }\n        }\n    }\n});\n\nfunction keyup() {\n    this.saveEditorValue();\n    this._moveEditor();\n}\n\nmodule.exports = Textfield;\n","'use strict';\n\nmodule.exports = {\n    CellEditor: require('./CellEditor'), // abstract base class\n    Textfield: require('./Textfield'),\n    Choice: require('./Choice'),\n    //Combo: require('./Combo'),\n    Color: require('./Color'),\n    Date: require('./Date'),\n    Simple: require('./Simple'),\n    Slider: require('./Slider'),\n    Spinner: require('./Spinner'),\n    Filter: require('./Filter')\n};\n","'use strict';\n\nvar Base = require('../lib/Base');\n\nvar A = 'A'.charCodeAt(0);\n\n/**\n * @constructor\n */\nvar DataModel = Base.extend('DataModel', {\n\n    next: null,\n\n    grid: null,\n\n    setGrid: function(newGrid) {\n        this.grid = newGrid;\n    },\n\n    /** @deprecated Use `.grid` property instead. */\n    getGrid: function() {\n        return this.deprecated('grid', { since: '0.2' });\n    },\n\n    /** @deprecated Use `.grid.behavior` property instead. */\n    getBehavior: function() {\n        return this.deprecated('grid.behavior', { since: '0.2' });\n    },\n\n    changed: function() {\n        this.grid.behavior.changed();\n    },\n\n    getPrivateState: function() {\n        return this.grid.getPrivateState();\n    },\n\n    applyState: function() {\n\n    },\n\n    alphaFor: function(i) {\n        // Name the column headers in A, .., AA, AB, AC, .., AZ format\n        // quotient/remainder\n        //var quo = Math.floor(col/27);\n        var quo = Math.floor(i / 26);\n        var rem = i % 26;\n        var code = '';\n        if (quo > 0) {\n            code += this.alpha(quo - 1);\n        }\n        code += this.alpha(rem);\n        return code;\n    },\n\n    alpha: function(i) {\n        return String.fromCharCode(A + i);\n    },\n\n    getCellEditorAt: function(x, y) {\n    },\n\n});\n\nmodule.exports = DataModel;\n","'use strict';\n\n//var analytics = require('hyper-analytics');\n//var analytics = require('../local_node_modules/hyper-analytics');\nvar analytics = require('../local_node_modules/finanalytics');\nvar DataModel = require('./DataModel');\nvar images = require('../../images');\n\nvar UPWARDS_BLACK_ARROW = '\\u25b2', // aka '▲'\n    DOWNWARDS_BLACK_ARROW = '\\u25bc'; // aka '▼'\n\nvar nullDataSource = {\n    isNullObject: function() {\n        return true;\n    },\n    getFields: function() {\n        return [];\n    },\n    getHeaders: function() {\n        return [];\n    },\n    getColumnCount: function() {\n        return 0;\n    },\n    getRowCount: function() {\n        return 0;\n    },\n    getAggregateTotals: function() {\n        return [];\n    },\n    hasAggregates: function() {\n        return false;\n    },\n    hasGroups: function() {\n        return false;\n    },\n    getRow: function() {\n        return null;\n    }\n};\n\n/**\n * @name dataModels.JSON\n * @constructor\n */\nvar JSON = DataModel.extend('dataModels.JSON', {\n\n    //null object pattern for the source object\n    source: nullDataSource,\n\n    preglobalfilter: nullDataSource,\n    prefilter: nullDataSource,\n\n    presorter: nullDataSource,\n    analytics: nullDataSource,\n    postglobalfilter: nullDataSource,\n    postfilter: nullDataSource,\n    postsorter: nullDataSource,\n\n    topTotals: [],\n    bottomTotals: [],\n\n    initialize: function() {\n        this.selectedData = [];\n    },\n\n    clearSelectedData: function() {\n        this.selectedData.length = 0;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasAggregates: function() {\n        return this.analytics.hasAggregates();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasGroups: function() {\n        return this.analytics.hasGroups();\n    },\n\n    getDataSource: function() {\n        return this.postsorter; //this.hasAggregates() ? this.analytics : this.presorter;\n    },\n\n    getFilterSource: function() {\n        return this.postfilter; //this.hasAggregates() ? this.postfilter : this.prefilter;\n    },\n\n    getGlobalFilterSource: function() {\n        return this.postglobalfilter; //this.hasAggregates() ? this.postfilter : this.prefilter;\n    },\n\n    getSortingSource: function() {\n        return this.postsorter; //this.hasAggregates() ? this.postsorter : this.presorter;\n    },\n\n    getData: function() {\n        return this.source.data;\n    },\n\n    getFilteredData: function() {\n        var ds = this.getDataSource();\n        var count = ds.getRowCount();\n        var result = new Array(count);\n        for (var y = 0; y < count; y++) {\n            result[y] = ds.getRow(y);\n        }\n        return result;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @returns {*}\n     */\n    getValue: function(x, y) {\n        var hasHierarchyColumn = this.hasHierarchyColumn();\n        var headerRowCount = this.grid.getHeaderRowCount();\n        var value;\n        if (hasHierarchyColumn) {\n            if (x === -2) {\n                x = 0;\n            }\n        } else if (this.hasAggregates()) {\n            x += 1;\n        }\n        if (y < headerRowCount) {\n            value = this.getHeaderRowValue(x, y);\n            return value;\n        }\n        // if (hasHierarchyColumn) {\n        //     y += 1;\n        // }\n        value = this.getDataSource().getValue(x, y - headerRowCount);\n        return value;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y - negative values refer to _bottom totals_ rows\n     * @returns {*}\n     */\n    getHeaderRowValue: function(x, y) {\n        var value;\n        if (y === undefined) {\n            value = this.getHeaders()[Math.max(x, 0)];\n        } else if (y < 0) { // bottom totals rows\n            var bottomTotals = this.getBottomTotals();\n            value = bottomTotals[bottomTotals.length + y][x];\n        } else {\n            var isFilterRow = this.grid.isShowFilterRow(),\n                isHeaderRow = this.grid.isShowHeaderRow(),\n                topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0);\n            if (y >= topTotalsOffset) { // top totals rows\n                value = this.getTopTotals()[y - topTotalsOffset][x];\n            } else if (isHeaderRow && y === 0) {\n                value = this.getHeaders()[x];\n                var sortString = this.getSortImageForColumn(x);\n                if (sortString) { value = sortString + value; }\n            } else { // must be filter row\n                value = this.getFilter(x);\n                var icon = images.filter(value.length);\n                return [null, value, icon];\n            }\n        }\n        return value;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @param value\n     */\n    setValue: function(x, y, value) {\n        var hasHierarchyColumn = this.hasHierarchyColumn();\n        var headerRowCount = this.grid.getHeaderRowCount();\n        if (hasHierarchyColumn) {\n            if (x === -2) {\n                x = 0;\n            }\n        } else if (this.hasAggregates()) {\n            x += 1;\n        }\n        if (y < headerRowCount) {\n            this.setHeaderRowValue(x, y, value);\n        } else {\n            this.getDataSource().setValue(x, y - headerRowCount, value);\n        }\n        this.changed();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} x\n     * @param {number} y\n     * @param value\n     * @returns {*}\n     */\n    setHeaderRowValue: function(x, y, value) {\n        if (value === undefined) {\n            return this._setHeader(x, y); // y is really the value\n        }\n        var isFilterRow = this.grid.isShowFilterRow();\n        var isHeaderRow = this.grid.isShowHeaderRow();\n        var isBoth = isFilterRow && isHeaderRow;\n        var topTotalsOffset = (isFilterRow ? 1 : 0) + (isHeaderRow ? 1 : 0);\n        if (y >= topTotalsOffset) {\n            this.getTopTotals()[y - topTotalsOffset][x] = value;\n        } else if (x === -1) {\n            return; // can't change the row numbers\n        } else if (isBoth) {\n            if (y === 0) {\n                return this._setHeader(x, value);\n            } else {\n                this.setFilter(x, value);\n            }\n        } else if (isFilterRow) {\n            this.setFilter(x, value);\n        } else {\n            return this._setHeader(x, value);\n        }\n        return '';\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @returns {*}\n     */\n    getColumnProperties: function(colIndex) {\n        //access directly because we want it ordered\n        var column = this.grid.behavior.allColumns[colIndex];\n        if (column) {\n            return column.getProperties();\n        }\n        return undefined;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @returns {string} The text to filter on for this column.\n     */\n    getFilter: function(colIndex) {\n        var filter, columnProperties;\n\n        if ((columnProperties = this.getColumnProperties(colIndex))) {\n            filter = columnProperties.filter;\n        }\n\n        return filter || '';\n    },\n\n    /** @typedef {function} rowFilterFunction\n     * @param {function|*} data - Data to test (or function to call to get data to test) to see if it qualifies for the result set.\n     * @returns {boolean} Row qualifies for the result set (passes through filter).\n     */\n    /**\n     * @param {number} colIndex\n     * @returns {undefined|rowFilterFunction} row filtering function\n     */\n    getComplexFilter: function(colIndex) {\n        var rowFilter, columnProperties, filter, filterObject, newFilter;\n\n        if (\n            (columnProperties = this.getColumnProperties(colIndex)) &&\n            (filter = columnProperties.complexFilter) &&\n            (filterObject = this.grid.filter) &&\n            (newFilter = filterObject.create(filter.state))\n        ) {\n            rowFilter = function(data) {\n                var transformed = valueOrFunctionExecute(data);\n                return newFilter(transformed);\n            };\n        }\n\n        return rowFilter;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param value\n     */\n    setFilter: function(colIndex, value) {\n        var columnProperties = this.getColumnProperties(colIndex);\n        columnProperties.filter = value;\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {number}\n     */\n    getColumnCount: function() {\n        var showTree = this.grid.resolveProperty('showTreeColumn') === true;\n        var hasAggregates = this.hasAggregates();\n        var offset = (hasAggregates && !showTree) ? -1 : 0;\n        return this.analytics.getColumnCount() + offset;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {number}\n     */\n    getRowCount: function() {\n        var count = this.getDataSource().getRowCount();\n        count += this.grid.getHeaderRowCount();\n        return count;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {string[]}\n     */\n    getHeaders: function() {\n        return this.analytics.getHeaders();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string[]} headers\n     */\n    setHeaders: function(headers) {\n        this.getDataSource().setHeaders(headers);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string[]} fields\n     */\n    setFields: function(fields) {\n        this.getDataSource().setFields(fields);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {string[]}\n     */\n    getFields: function() {\n        return this.getDataSource().getFields();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {object[]} dataRows\n     */\n    setData: function(dataRows) {\n        this.source = new analytics.JSDataSource(dataRows);\n        //this.preglobalfilter = new analytics.DataSourceGlobalFilter(this.source);\n        //this.prefilter = new analytics.DataSourceFilter(this.preglobalfilter);\n        //this.presorter = new analytics.DataSourceSorterComposite(this.prefilter);\n\n        this.analytics = new analytics.DataSourceAggregator(this.source);\n\n        this.postglobalfilter = new analytics.DataSourceGlobalFilter(this.analytics);\n        this.postfilter = new analytics.DataSourceFilter(this.postglobalfilter);\n        this.postsorter = new analytics.DataSourceSorterComposite(this.postfilter);\n\n        this.applyAnalytics();\n\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {Array<Array>} totalRows\n     */\n    setTopTotals: function(totalRows) {\n        this.topTotals = totalRows;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getTopTotals: function() {\n        if (!this.hasAggregates()) {\n            return this.topTotals;\n        }\n        return this.getDataSource().getGrandTotals();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {Array<Array>} totalRows\n     */\n    setBottomTotals: function(totalRows) {\n        this.bottomTotals = totalRows;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {Array<Array>}\n     */\n    getBottomTotals: function() {\n        if (!this.hasAggregates()) {\n            return this.bottomTotals;\n        }\n        return this.getDataSource().getGrandTotals();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param groups\n     */\n    setGroups: function(groups) {\n        this.analytics.setGroupBys(groups);\n        this.applyAnalytics();\n        this.grid.fireSyntheticGroupsChangedEvent(this.getGroups());\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getGroups: function() {\n        var headers = this.getHeaders().slice(0);\n        var fields = this.getFields().slice(0);\n        var groupBys = this.analytics.groupBys;\n        var groups = [];\n        for (var i = 0; i < groupBys.length; i++) {\n            var field = headers[groupBys[i]];\n            groups.push({\n                id: groupBys[i],\n                label: field,\n                field: fields\n            });\n        }\n        return groups;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getAvailableGroups: function() {\n        var headers = this.source.getHeaders().slice(0);\n        var groupBys = this.analytics.groupBys;\n        var groups = [];\n        for (var i = 0; i < headers.length; i++) {\n            if (groupBys.indexOf(i) === -1) {\n                var field = headers[i];\n                groups.push({\n                    id: i,\n                    label: field,\n                    field: field\n                });\n            }\n        }\n        return groups;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getVisibleColumns: function() {\n        var items = this.grid.behavior.columns;\n        items = items.filter(function(each) {\n            return each.label !== 'Tree';\n        });\n        return items;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {object[]}\n     */\n    getHiddenColumns: function() {\n        var visible = this.grid.behavior.columns;\n        var all = this.grid.behavior.allColumns;\n        var hidden = [];\n        for (var i = 0; i < all.length; i++) {\n            if (visible.indexOf(all[i]) === -1) {\n                hidden.push(all[i]);\n            }\n        }\n        hidden.sort(function(a, b) {\n            return a.label < b.label;\n        });\n        return hidden;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param aggregations\n     */\n    setAggregates: function(aggregations) {\n        this.quietlySetAggregates(aggregations);\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param aggregations\n     */\n    quietlySetAggregates: function(aggregations) {\n        this.analytics.setAggregates(aggregations);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @returns {boolean}\n     */\n    hasHierarchyColumn: function() {\n        var showTree = this.grid.resolveProperty('showTreeColumn') === true;\n        return this.hasAggregates() && this.hasGroups() && showTree;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyAnalytics: function(dontApplyGroupBysAndAggregations) {\n        selectedDataRowsBackingSelectedGridRows.call(this);\n\n        if (!dontApplyGroupBysAndAggregations) {\n            applyGroupBysAndAggregations.call(this);\n        }\n        applyFilters.call(this);\n        applySorts.call(this);\n\n        reselectGridRowsBackedBySelectedDataRows.call(this);\n    },\n\n    createFormattedFilter: function(formatter, filter) {\n        return function(value) {\n            var formattedValue = formatter(value);\n            return filter(formattedValue);\n        };\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param keys\n     */\n    toggleSort: function(colIndex, keys) {\n        this.incrementSortState(colIndex, keys);\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} colIndex\n     * @param {string[]} keys\n     */\n    incrementSortState: function(colIndex, keys) {\n        colIndex++; //hack to get around 0 index\n        var state = this.getPrivateState();\n        var hasCTRL = keys.indexOf('CTRL') > -1;\n        state.sorts = state.sorts || [];\n        var already = state.sorts.indexOf(colIndex);\n        if (already === -1) {\n            already = state.sorts.indexOf(-1 * colIndex);\n        }\n        if (already > -1) {\n            if (state.sorts[already] > 0) {\n                state.sorts[already] = -1 * state.sorts[already];\n            } else {\n                state.sorts.splice(already, 1);\n            }\n        } else if (hasCTRL || state.sorts.length === 0) {\n            state.sorts.unshift(colIndex);\n        } else {\n            state.sorts.length = 0;\n            state.sorts.unshift(colIndex);\n        }\n        if (state.sorts.length > 3) {\n            state.sorts.length = 3;\n        }\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param index\n     * @param returnAsString\n     * @returns {*}\n     */\n    getSortImageForColumn: function(index) {\n        index++;\n        var up = true;\n        var sorts = this.getPrivateState().sorts;\n        if (!sorts) {\n            return null;\n        }\n        var position = sorts.indexOf(index);\n        if (position < 0) {\n            position = sorts.indexOf(-1 * index);\n            up = false;\n        }\n        if (position < 0) {\n            return null;\n        }\n        var rank = sorts.length - position;\n        var arrow = up ? UPWARDS_BLACK_ARROW : DOWNWARDS_BLACK_ARROW;\n        return rank + arrow + ' ';\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param cell\n     * @param event\n     */\n    cellClicked: function(cell, event) {\n        if (!this.hasAggregates()) {\n            return;\n        }\n        if (event.gridCell.x !== 0) {\n            return; // this wasn't a click on the hierarchy column\n        }\n        var headerRowCount = this.grid.getHeaderRowCount();\n        var y = event.gridCell.y - headerRowCount;\n        this.getDataSource().click(y);\n        this.applyAnalytics(true);\n        this.changed();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    getRow: function(y) {\n        var headerRowCount = this.grid.getHeaderRowCount();\n        if (y < headerRowCount && !this.hasAggregates()) {\n            var topTotals = this.getTopTotals();\n            return topTotals[y - (headerRowCount - topTotals.length)];\n        }\n        return this.getDataSource().getRow(y - headerRowCount);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    buildRow: function(y) {\n        var colCount = this.getColumnCount();\n        var fields = [].concat(this.getFields());\n        var result = {};\n        if (this.hasAggregates()) {\n            result.tree = this.getValue(-2, y);\n            fields.shift();\n        }\n        for (var i = 0; i < colCount; i++) {\n            result[fields[i]] = this.getValue(i, y);\n        }\n        return result;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {number} y\n     * @returns {object}\n     */\n    getComputedRow: function(y) {\n        var rcf = this.getRowContextFunction([y]);\n        var fields = this.getFields();\n        var row = {};\n        for (var i = 0; i < fields.length; i++) {\n            var field = fields[i];\n            row[field] = rcf(field)[0];\n        }\n        return row;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {string} fieldName\n     * @param {number} y\n     * @returns {*}\n     */\n    getValueByField: function(fieldName, y) {\n        var index = this.getFields().indexOf(fieldName);\n        if (this.hasAggregates()) {\n            y += 1;\n        }\n        return this.getDataSource().getValue(index, y);\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {sring} string\n     */\n    setGlobalFilter: function(string) {\n        var globalFilterSource = this.getGlobalFilterSource();\n        if (!string || string.length === 0) {\n            globalFilterSource.clear();\n        } else {\n            globalFilterSource.set(textMatchFilter(string));\n        }\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     * @param {object} config\n     * @param {number} x\n     * @param {number} y\n     * @param {number} untranslatedX\n     * @param {number} untranslatedY\n     * @returns {object}\n     */\n    getCellRenderer: function(config, x, y, untranslatedX, untranslatedY) {\n        var renderer;\n        var provider = this.grid.getCellProvider();\n\n        config.x = x;\n        config.y = y;\n        config.untranslatedX = untranslatedX;\n        config.untranslatedY = untranslatedY;\n\n        renderer = provider.getCell(config);\n        renderer.config = config;\n\n        return renderer;\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    applyState: function() {\n        this.applyAnalytics();\n    },\n\n    /**\n     * @memberOf dataModels.JSON.prototype\n     */\n    reset: function() {\n        this.setData([]);\n    },\n\n    getUnfilteredValue: function(x, y) {\n        return this.source.getValue(x, y);\n    },\n\n    getUnfilteredRowCount: function() {\n        return this.source.getRowCount();\n    },\n\n});\n\nfunction valueOrFunctionExecute(valueOrFunction) {\n    return typeof valueOrFunction === 'function' ? valueOrFunction() : valueOrFunction;\n}\n\nfunction textMatchFilter(string) {\n    string = string.toLowerCase();\n    return function(each) {\n        each = valueOrFunctionExecute(each);\n        return (each + '').toLowerCase().indexOf(string) > -1;\n    };\n}\n\n// LOCAL METHODS -- to be called with `.call(this`\n\n/**\n * Accumulate actual data row objects backing current grid row selections.\n * This call should be paired with a subsequent call to `reselectGridRowsBackedBySelectedDataRows`.\n * @private\n * @memberOf dataModels.JSON.prototype\n */\nfunction selectedDataRowsBackingSelectedGridRows() {\n    //// STEP 1: Accumulate actual data row objects backing current grid row selections.\n    if (this.grid.selectionModel.hasRowSelections()) { // any current grid row selections?\n        var filteredData = this.getFilteredData(),\n            selectedGridRows = this.grid.getSelectedRows(),\n            selectedData = this.selectedData;\n\n        // 1.a. Remove any filtered data rows from the recently selected list.\n        selectedData.forEach(function(dataRow, idx) {\n            if (filteredData.indexOf(dataRow) > 0) {\n                delete selectedData[idx];\n            }\n        });\n\n        // 1.b. Accumulate the data rows backing any currently selected grid rows in `this.selectedData`.\n        selectedGridRows.forEach(function(selectedRowIndex) {\n            var dataRow = filteredData[selectedRowIndex];\n            if (selectedData.indexOf(dataRow) < 0) {\n                selectedData.push(dataRow);\n            }\n        });\n    }\n}\n\n/**\n * Re-establish grid row selections based on actual data row objects accumulated by `selectedDataRowsBackingSelectedGridRows` which should be called first.\n * @private\n * @memberOf dataModels.JSON.prototype\n */\nfunction reselectGridRowsBackedBySelectedDataRows() {\n    if (this.selectedData.length) { // any data row objects added from previous grid row selections?\n        var selectionModel = this.grid.selectionModel,\n            offset = this.grid.getHeaderRowCount(),\n            filteredData = this.getFilteredData();\n\n        selectionModel.clearRowSelection();\n\n        this.selectedData.forEach(function(dataRow) {\n            var index = filteredData.indexOf(dataRow);\n            if (index >= 0) {\n                selectionModel.selectRow(offset + index);\n            }\n        });\n    }\n}\n\n/**\n * @private\n * @memberOf dataModels.JSON.prototype\n */\nfunction applyGroupBysAndAggregations() {\n    if (this.analytics.aggregates.length === 0) {\n        this.quietlySetAggregates({});\n    }\n    this.analytics.apply();\n}\n\n/**\n * @private\n * @memberOf dataModels.JSON.prototype\n */\nfunction applyFilters() {\n    var visibleColumns = this.getVisibleColumns();\n    this.getGlobalFilterSource().apply(visibleColumns);\n    var details = [];\n    var filterSource = this.getFilterSource();\n    var groupOffset = 0; //this.hasHierarchyColumn() ? 0 : 1;\n\n    // apply column filters\n    filterSource.clearAll();\n\n    visibleColumns.forEach(function(column) {\n        var columnIndex = column.index,\n            filterText = this.getFilter(columnIndex),\n            formatterType = column.getProperties().format,\n            formatter = this.grid.getFormatter(formatterType),\n            complexFilter = this.getComplexFilter(columnIndex),\n            filter = complexFilter || filterText.length > 0 && textMatchFilter(filterText);\n\n        if (filter) {\n            filterSource.add(columnIndex - groupOffset, this.createFormattedFilter(formatter, filter));\n            details.push({\n                column: column.label,\n                format: complexFilter ? 'complex' : formatterType\n            });\n        }\n    }.bind(this));\n\n    filterSource.applyAll();\n\n    this.grid.fireSyntheticFilterAppliedEvent({\n        details: details\n    });\n}\n\n/**\n * @private\n * @memberOf dataModels.JSON.prototype\n */\nfunction applySorts() {\n    var sortingSource = this.getSortingSource();\n    var sorts = this.getPrivateState().sorts;\n    var groupOffset = this.hasAggregates() ? 1 : 0;\n    if (!sorts || sorts.length === 0) {\n        sortingSource.clearSorts();\n    } else {\n        for (var i = 0; i < sorts.length; i++) {\n            var colIndex = Math.abs(sorts[i]) - 1;\n            var type = sorts[i] < 0 ? -1 : 1;\n            sortingSource.sortOn(colIndex - groupOffset, type);\n        }\n    }\n    sortingSource.applySorts();\n}\n\nmodule.exports = JSON;\n","/* eslint-env browser */\n\n'use strict';\n\nvar LRUCache = require('lru-cache');\n\nvar renderCellError = require('./lib/renderCellError');\n\n/**\n * This module lists the properties that can be set on a {@link Hypergrid} along with their default values.\n * Edit this file to override the defaults.\n * @module defaults\n */\n\nmodule.exports = {\n\n    //these are for the theme\n\n\n    /**\n     * The font for data cells.\n     * @default '13px Tahoma, Geneva, sans-serif'\n     * @type {cssFont}\n     * @instance\n     */\n    font: '13px Tahoma, Geneva, sans-serif',\n\n    /**\n     * Font color for data cells.\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    color: 'rgb(25, 25, 25)',\n\n    /**\n     * Background color for data cells.\n     * @default 'rgb(241, 241, 241)'\n     * @type {string}\n     * @instance\n     */\n    backgroundColor: 'rgb(241, 241, 241)',\n\n    /**\n     * Font color for selected cell(s).\n     * @default 'rgb(25, 25, 25)'\n     * @type {string}\n     * @instance\n     */\n    foregroundSelectionColor: 'black',\n\n    /**\n     * Background color for selected cell(s).\n     * @default 'rgba(147, 185, 255, 0.45)'\n     * @type {string}\n     * @instance\n     */\n    backgroundSelectionColor: 'rgba(147, 185, 255, 0.45)',\n\n\n    /********** SECTION: COLUMN HEADER COLORS **********/\n\n    // IMPORTANT CAVEAT: The code is inconsistent regarding the terminology. Is the \"column header\" section _the row_ of cells at the top (that act as headers for each column) or is it _the column_ of cells (that act as headers for each row)? Oh my.\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {cssFont}\n     * @instance\n     */\n    columnHeaderFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgba(255, 220, 97, 0.45)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderForegroundColumnSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {cssColor}\n     * @instance\n     */\n    columnHeaderBackgroundColumnSelectionColor: 'rgb(255, 180, 0)',\n\n\n    /********** SECTION: ROW HEADER COLORS **********/\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {cssFont}\n     * @instance\n     */\n    rowHeaderFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgba(255, 220, 97, 0.45)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderForegroundRowSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {cssColor}\n     * @instance\n     */\n    rowHeaderBackgroundRowSelectionColor: 'rgb(255, 180, 0)',\n\n\n    /********** SECTION: FILTER ROW COLORS **********/\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {cssFont}\n     * @instance\n     */\n    filterFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    filterColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'white'\n     * @type {cssColor}\n     * @instance\n     */\n    filterBackgroundColor: 'white',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    filterForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 220, 97)'\n     * @type {cssColor}\n     * @instance\n     */\n    filterBackgroundSelectionColor: 'rgb(255, 220, 97)',\n\n    /**\n     * @default 'rgba(0,0,0,0.8)'\n     * @type {cssColor}\n     * @instance\n     */\n    filterCellBorderStyle: 'rgba(0,0,0,0.8)',\n\n    /**\n     * @default 0.4\n     * @type {number}\n     * @instance\n     */\n    filterCellBorderThickness: 0.4,\n\n\n    /********** SECTION: TREE COLUMN COLORS **********/\n    // The \"tree column\" contains the hierarchical drill-down controls.\n\n    /**\n     * @default '12px Tahoma, Geneva, sans-serif'\n     * @type {cssFont}\n     * @instance\n     */\n    treeColumnFont: '12px Tahoma, Geneva, sans-serif',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(223, 227, 232)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnBackgroundColor: 'rgb(223, 227, 232)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnForegroundSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgba(255, 220, 97, 0.45)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnBackgroundSelectionColor: 'rgba(255, 220, 97, 0.45)',\n\n    /**\n     * @default 'rgb(25, 25, 25)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnForegroundColumnSelectionColor: 'rgb(25, 25, 25)',\n\n    /**\n     * @default 'rgb(255, 180, 0)'\n     * @type {cssColor}\n     * @instance\n     */\n    treeColumnBackgroundColumnSelectionColor: 'rgb(255, 180, 0)',\n\n    /**\n     * @default 'rgb(201, 201, 201)'\n     * @type {cssColor}\n     * @instance\n     */\n    backgroundColor2: 'rgb(201, 201, 201)',\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    voffset: 0,\n\n    /**\n     * @default 'visible'\n     * @type {string}\n     * @instance\n     */\n    scrollbarHoverOver: 'visible',\n\n    /**\n     * @default 'hidden'\n     * @type {string}\n     * @instance\n     */\n    scrollbarHoverOff: 'hidden',\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    scrollingEnabled: true,\n\n    /**\n     * @default ''\n     * @type {string}\n     * @instance\n     */\n    vScrollbarClassPrefix: '',\n\n    /**\n     * @default ''\n     * @type {string}\n     * @instance\n     */\n    hScrollbarClassPrefix: '',\n\n    //these used to be in the constants element\n\n    /**\n     * @default 'center'\n     * @type {string}\n     * @instance\n     */\n    fixedRowAlign: 'center',\n\n    /**\n     * @default 'center'\n     * @type {string}\n     * @instance\n     */\n    fixedColAlign: 'center',\n\n    /**\n     * @default 5\n     * @type {number}\n     * @instance\n     */\n    cellPadding: 5,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    gridLinesH: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    gridLinesV: true,\n\n    /**\n     * @default 'rgb(199, 199 199)'\n     * @type {cssColor}\n     * @instance\n     */\n    lineColor: 'rgb(199, 199, 199)',\n\n    /**\n     * @default 0.4\n     * @type {number}\n     * @instance\n     */\n    lineWidth: 0.4,\n\n\n    /**\n     * @default 15\n     * @type {number}\n     * @instance\n     */\n    defaultRowHeight: 15,\n\n    /**\n     * @default 100\n     * @type {number}\n     * @instance\n     */\n    defaultColumnWidth: 100,\n\n    //for immediate painting, set these values to 0, true respectively\n\n    /**\n     * @default 60\n     * @type {number}\n     * @instance\n     */\n    repaintIntervalRate: 60,\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     * @instance\n     */\n    repaintImmediately: false,\n\n    //enable or disable double buffering\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     * @instance\n     */\n    useBitBlit: false,\n\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    useHiDPI: true,\n\n    /**\n     * @default ['alt', 'esc']\n     * @type {string}\n     * @instance\n     */\n    editorActivationKeys: ['alt', 'esc'],\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     * @instance\n     */\n    readOnly: false,\n\n    // inherited by cell renderers\n\n    /**\n     * @default getTextWidth\n     * @type {function}\n     * @instance\n     */\n    getTextWidth: getTextWidth,\n\n    /**\n     * @default getTextHeight\n     * @type {function}\n     * @instance\n     */\n    getTextHeight: getTextHeight,\n\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    fixedColumnCount: 0,\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    fixedRowCount: 0,\n\n    /**\n     * @default 0\n     * @type {number}\n     * @instance\n     */\n    headerColumnCount: 0,\n\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    showRowNumbers: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    showTreeColumn: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    showHeaderRow: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    showFilterRow: true,\n\n\n    /** Clicking in a cell \"selects\" it; it is added to the select region and repainted with \"cell selection\" colors.\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    cellSelection: true,\n\n    /** Clicking in a row header (leftmost column) \"selects\" the row; the entire row is added to the select region and repainted with \"row selection\" colors.\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    columnSelection: true,\n\n    /** Clicking in a column header (top row) \"selects\" the column; the entire column is added to the select region and repainted with \"column selection\" colors.\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    rowSelection: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    singleRowSelectionMode: true,\n\n    /**\n     * @default 'rgba(0, 0, 0, 0.2)'\n     * @type {cssColor}\n     * @instance\n     */\n    selectionRegionOverlayColor: 'rgba(0, 0, 0, 0.2)',\n\n    /**\n     * @default 'black'\n     * @type {string}\n     * @instance\n     */\n    selectionRegionOutlineColor: 'black',\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    columnAutosizing: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    rowNumberAutosizing: true,\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     * @instance\n     */\n    headerTextWrapping: false,\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     * @instance\n     */\n    rowResize: false,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    editable: true,\n\n    /**\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    editOnDoubleClick: true,\n\n    /**\n     * @default 325\n     * @type {number}\n     * @instance\n     */\n    doubleClickDelay: 325,\n\n    /**\n     * Grid-level property.\n     * When user presses a printable character key _or_ BACKSPACE _or_ DELETE:\n     * 1. Activate cell editor on current cell (i.e., origin of most recent selection).\n     * 2. If cell editor is a text editor:\n     *    1. Replace current value with the character the user typed; or\n     *    2. Clear it on BACKSPACE, DELETE, or other invalid character (_e.g._ when user types a letter but the cell editor only accepts digits).\n     *\n     * > In invoked, user has the option to back out by pressing the ESCAPE key.\n     *\n     * @default `true`\n     * @type {boolean}\n     * @instance\n     */\n    editOnKeydown: true,\n\n    /**\n     * @default renderCellError\n     * @type {function}\n     */\n    renderCellError: renderCellError,\n\n    /**\n     * @default `false`\n     * @type {boolean}\n     */\n    checkboxOnlyRowSelections: false,\n\n    /** Name of a formatters for cell text.\n     * @see /src/Formatters.js\n     */\n    format: 'default',\n\n    /********** HOVER COLORS **********/\n\n    /** @typedef hoverColors\n     * @property {boolean} [enable=false] - `false` means not hilite on hover\n     * @property {cssColor} backgroundColor - cell, row, or colummn background color. Alpha channel will be respected and if given will be painted over the cells predetermined color.\n     * @property {cssColor} [header.backgroundColor=backgroundColor] - for columns and rows, this is the background color of the column or row \"handle\" (header rows or columns, respectively). (Not used for cells.)\n     */\n\n    /** On mouse hover, whether to repaint the cell background and how.\n     * @type {hoverColors}\n     * @default '{ enabled: true, background: rgba(160, 160, 40, 0.30) }'\n     */\n    hoverCellHighlight: {\n        enabled: true,\n        backgroundColor: 'rgba(160, 160, 40, 0.45)'\n    },\n\n    /** On mouse hover, whether to repaint the row background and how.\n     * @type {hoverColors}\n     * @default '{ enabled: true, background: rgba(100, 100, 25, 0.15) }'\n     */\n    hoverRowHighlight: {\n        enabled: true,\n        backgroundColor: 'rgba(100, 100, 25, 0.30)'\n\n    },\n\n    /** On mouse hover, whether to repaint the column background and how.\n     * @type {hoverColors}\n     * @default '{ enabled: true, background: rgba(60, 60, 15, 0.15) }'\n     */\n    hoverColumnHighlight: {\n        enabled: true,\n        backgroundColor: 'rgba(60, 60, 15, 0.15)'\n    },\n\n\n    /** Display cell font with under-score line drawn over it.\n     * > Implementation of links right now is not automatic; you must attach a 'fin-click' listener to the hypergrid object, etc.\n     * @type {boolean}\n     * @default `false`\n     */\n    link: false,\n\n    /** Display cell font with strike-through line drawn over it.\n     * @type {boolean}\n     * @default `false`\n     */\n    strikeThrough: false,\n\n};\n\n/** @typedef {string} cssColor\n * @see https://developer.mozilla.org/en-US/docs/Web/CSS/color_value\n */\n/** @typedef {string} cssFont\n * @see https://developer.mozilla.org/en-US/docs/Web/CSS/font\n */\n\nvar textWidthCache = new LRUCache(2000);\n\nfunction getTextWidth(gc, string) {\n    if (string === null || string === undefined) {\n        return 0;\n    }\n    string = string + '';\n    if (string.length === 0) {\n        return 0;\n    }\n    var key = gc.font + string;\n    var width = textWidthCache.get(key);\n    if (!width) {\n        width = gc.measureText(string).width;\n        textWidthCache.set(key, width);\n    }\n    return width;\n}\n\nvar fontData = {};\n\nfunction getTextHeight(font) {\n    var result = fontData[font];\n\n    if (!result) {\n        result = {};\n\n        var text = document.createElement('span');\n        text.textContent = 'Hg';\n        text.style.font = font;\n\n        var block = document.createElement('div');\n        block.style.display = 'inline-block';\n        block.style.width = '1px';\n        block.style.height = '0px';\n\n        var div = document.createElement('div');\n        div.appendChild(text);\n        div.appendChild(block);\n\n        div.style.position = 'absolute';\n        document.body.appendChild(div);\n\n        try {\n\n            block.style.verticalAlign = 'baseline';\n\n            var blockRect = block.getBoundingClientRect();\n            var textRect = text.getBoundingClientRect();\n\n            result.ascent = blockRect.top - textRect.top;\n\n            block.style.verticalAlign = 'bottom';\n            result.height = blockRect.top - textRect.top;\n\n            result.descent = result.height - result.ascent;\n\n        } finally {\n            document.body.removeChild(div);\n        }\n        if (result.height !== 0) {\n            fontData[font] = result;\n        }\n    }\n\n    return result;\n}\n","/* eslint-env browser */\n\n'use strict';\n\nrequire('object-iterators'); // Installs the Array.find polyfill, as needed\n\nvar Hypergrid = require('./Hypergrid');\n\nHypergrid.images = require('../images');\nHypergrid.behaviors = require('./behaviors/index');\nHypergrid.cellEditors = require('./cellEditors/index');\nHypergrid.features = require('./features/index');\n\n(window.fin = window.fin || {}).Hypergrid = Hypergrid;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellClick = Feature.extend('CellClick', {\n\n    alias: 'CellClick',\n\n    /**\n     * @memberOf CellClick.prototype\n     * @desc Handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleTap: function(grid, event) {\n        if (\n            event.gridCell.y >= grid.behavior.getHeaderRowCount() &&\n            event.gridCell.x >= grid.behavior.getHeaderColumnCount()\n        ) {\n            grid.cellClicked(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    }\n});\n\nmodule.exports = CellClick;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellEditing = Feature.extend('CellEditing', {\n\n    alias: 'CellEditing',\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDoubleClick: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    handleTap: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n           grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) {\n        var headerRowCount = grid.behavior.getHeaderRowCount();\n        var headerColumnCount = grid.behavior.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        var isFilterRow = grid.isFilterRow(gridCell.y);\n        var activateEditor = isDoubleClickEditorActivation && gridCell.x >= headerColumnCount && (isFilterRow || gridCell.y >= headerRowCount);\n        return activateEditor;\n    }\n\n});\n\nmodule.exports = CellEditing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar CellSelection = Feature.extend('CellSelection', {\n\n    alias: 'CellSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {window.fin.rectangular.Point}\n     * @memberOf CellSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * the cell coordinates of the where the mouse pointer is during a drag operation\n     * @type {Object}\n     * @memberOf CellSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf CellSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf CellSelection.prototype\n     */\n    sbAutoStart: 0,\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragging) {\n            this.dragging = false;\n        }\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n        var headerRowCount = grid.behavior.getHeaderRowCount();\n        var headerColumnCount = grid.behavior.getHeaderColumnCount();\n        var columnCount = grid.behavior.getColumnCount();\n        var isOutside = viewCell.x >= columnCount;\n\n        var isHeader = dy < headerRowCount || dx < headerColumnCount;\n\n        if (!grid.isCellSelection() || isRightClick || isHeader || isOutside) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n            var numFixedColumns = grid.getFixedColumnCount();\n            var numFixedRows = grid.getFixedRowCount();\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            var keys = primEvent.detail.keys;\n            this.dragging = true;\n            this.extendSelection(grid, dCell, keys);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (!grid.isCellSelection() || isRightClick || !this.dragging) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n            var numFixedRows = grid.getFixedRowCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Handle a mousedrag selection.\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n\n        var headerRowCount = grid.behavior.getHeaderRowCount();\n        var headerColumnCount = grid.behavior.getHeaderColumnCount();\n        var x = gridCell.x;\n        var y = gridCell.y;\n        x = Math.max(headerColumnCount, x);\n        y = Math.max(headerRowCount, y);\n\n        var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        //var scrollingNow = grid.isScrollingNow();\n\n        var newX = x - mouseDown.x;\n        var newY = y - mouseDown.y;\n\n        if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n            return;\n        }\n\n        grid.clearMostRecentSelection();\n\n        grid.select(mouseDown.x, mouseDown.y, newX, newY);\n        grid.setDragExtent(grid.newPoint(newX, newY));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var dragStartedInHeaderArea = grid.isMouseDownInHeaderArea();\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (!dragStartedInHeaderArea) {\n            if (this.currentDrag.x < b.origin.x) {\n                xOffset = -1;\n            }\n            if (this.currentDrag.y < b.origin.y) {\n                yOffset = -1;\n            }\n        }\n        if (this.currentDrag.x > b.origin.x + b.extent.x) {\n            xOffset = 1;\n        }\n        if (this.currentDrag.y > b.origin.y + b.extent.y) {\n            yOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        var hasCTRL = keys.indexOf('CTRL') >= 0;\n        var hasSHIFT = keys.indexOf('SHIFT') >= 0;\n        // var scrollTop = grid.getVScrollValue();\n        // var scrollLeft = grid.getHScrollValue();\n\n        // var numFixedColumns = 0;//grid.getFixedColumnCount();\n        // var numFixedRows = 0;//grid.getFixedRowCount();\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        //we have repeated a click in the same spot deslect the value from last time\n        if (\n            hasCTRL &&\n            x === mousePoint.x &&\n            y === mousePoint.y\n        ) {\n            grid.clearMostRecentSelection();\n            grid.popMouseDown();\n            grid.repaint();\n            return;\n        }\n\n        if (!hasCTRL && !hasSHIFT) {\n            grid.clearSelections();\n        }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentSelection();\n            grid.select(mousePoint.x, mousePoint.y, x - mousePoint.x + 1, y - mousePoint.y + 1);\n            grid.setDragExtent(grid.newPoint(x - mousePoint.x + 1, y - mousePoint.y));\n        } else {\n            grid.select(x, y, 0, 0);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 0, 1);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 0, -1);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid, event) {\n        //keep the browser viewport from auto scrolling on key event\n        event.primitiveEvent.preventDefault();\n\n        var count = this.getAutoScrollAcceleration();\n        this.moveSingleSelect(grid, 0, count);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid, event) {\n        //keep the browser viewport from auto scrolling on key event\n        event.primitiveEvent.preventDefault();\n\n        var count = this.getAutoScrollAcceleration();\n        this.moveSingleSelect(grid, 0, -count);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {\n        this.moveSingleSelect(grid, -1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n        this.moveSingleSelect(grid, 1, 0);\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetX, offsetY) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumns() - 1;\n        var maxViewableRows = grid.getVisibleRows() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newX = extent.x + offsetX;\n        var newY = extent.y + offsetY;\n\n        newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX));\n        newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY));\n\n        grid.clearMostRecentSelection();\n        grid.select(origin.x, origin.y, newX, newY);\n\n        grid.setDragExtent(grid.newPoint(newX, newY));\n\n        if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) {\n            this.pingAutoScroll();\n        }\n        if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf CellSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetX, offsetY) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumnsCount() - 1;\n        var maxViewableRows = grid.getVisibleRowsCount() - 1;\n\n        var minRows = grid.getHeaderRowCount();\n        var minCols = grid.getHeaderColumnCount();\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newX = mouseCorner.x + offsetX;\n        var newY = mouseCorner.y + offsetY;\n\n        newX = Math.min(maxColumns, Math.max(minCols, newX));\n        newY = Math.min(maxRows, Math.max(minRows, newY));\n\n        grid.clearSelections();\n        grid.select(newX, newY, 0, 0);\n        grid.setMouseDown(grid.newPoint(newX, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelColIsVisible(newX, offsetX)) {\n            this.pingAutoScroll();\n        }\n        if (grid.insureModelRowIsVisible(newY, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    }\n\n});\n\nmodule.exports = CellSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnAutosizing = Feature.extend('ColumnAutosizing', {\n\n    alias: 'ColumnAutosizing',\n\n    /**\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf ColumnAutosizing.prototype\n     */\n    handleDoubleClick: function(grid, event) {\n        var headerRowCount = grid.getHeaderRowCount();\n        //var headerColCount = grid.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        if (gridCell.y <= headerRowCount) {\n            grid.autosizeColumn(gridCell.x);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = ColumnAutosizing;\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\n// This feature is responsible for column drag and drop reordering.\n// This object is a mess and desperately needs a complete rewrite.....\n\nvar Feature = require('./Feature.js');\n\nvar columnAnimationTime = 150;\nvar dragger;\nvar draggerCTX;\nvar floatColumn;\nvar floatColumnCTX;\n\n/**\n * @constructor\n */\nvar ColumnMoving = Feature.extend('ColumnMoving', {\n\n    alias: 'ColumnMoving',\n\n    /**\n     * queue up the animations that need to play so they are done synchronously\n     * @type {Array}\n     * @memberOf CellMoving.prototype\n     */\n    floaterAnimationQueue: [],\n\n    /**\n     * am I currently auto scrolling right\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    columnDragAutoScrollingRight: false,\n\n    /**\n     * am I currently auto scrolling left\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    columnDragAutoScrollingLeft: false,\n\n    /**\n     * is the drag mechanism currently enabled (\"armed\")\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    dragArmed: false,\n\n    /**\n     * am I dragging right now\n     * @type {boolean}\n     * @memberOf CellMoving.prototype\n     */\n    dragging: false,\n\n    /**\n     * the column index of the currently dragged column\n     * @type {number}\n     * @memberOf CellMoving.prototype\n     */\n    dragCol: -1,\n\n    /**\n     * an offset to position the dragged item from the cursor\n     * @type {number}\n     * @memberOf CellMoving.prototype\n     */\n    dragOffset: 0,\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc give me an opportunity to initialize stuff on the grid\n     * @param {Hypergrid} grid\n     */\n    initializeOn: function(grid) {\n        this.isFloatingNow = false;\n        this.initializeAnimationSupport(grid);\n        if (this.next) {\n            this.next.initializeOn(grid);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc initialize animation support on the grid\n     * @param {Hypergrid} grid\n     */\n    initializeAnimationSupport: function(grid) {\n        if (!dragger) {\n            dragger = document.createElement('canvas');\n            dragger.setAttribute('width', '0px');\n            dragger.setAttribute('height', '0px');\n\n            document.body.appendChild(dragger);\n            draggerCTX = dragger.getContext('2d');\n        }\n        if (!floatColumn) {\n            floatColumn = document.createElement('canvas');\n            floatColumn.setAttribute('width', '0px');\n            floatColumn.setAttribute('height', '0px');\n\n            document.body.appendChild(floatColumn);\n            floatColumnCTX = floatColumn.getContext('2d');\n        }\n\n    },\n\n    getCanDragCursorName: function() {\n        return '-webkit-grab';\n    },\n\n    getDraggingCursorName: function() {\n        return '-webkit-grabbing';\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n\n        var gridCell = event.gridCell;\n        var x;\n        //var y;\n\n        var distance = Math.abs(event.primitiveEvent.detail.dragstart.x - event.primitiveEvent.detail.mouse.x);\n\n        if (distance < 10) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n            return;\n        }\n\n        if (this.isHeaderRow(grid, event) && this.dragArmed && !this.dragging) {\n            this.dragging = true;\n            this.dragCol = gridCell.x;\n            this.dragOffset = event.mousePoint.x;\n            this.detachChain();\n            x = event.primitiveEvent.detail.mouse.x - this.dragOffset;\n            //y = event.primitiveEvent.detail.mouse.y;\n            this.createDragColumn(grid, x, this.dragCol);\n        } else if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n\n        if (this.dragging) {\n            x = event.primitiveEvent.detail.mouse.x - this.dragOffset;\n            //y = event.primitiveEvent.detail.mouse.y;\n            this.dragColumn(grid, x);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (grid.behavior.isColumnReorderable()) {\n            if (this.isHeaderRow(grid, event) && event.gridCell.x !== -1) {\n                this.dragArmed = true;\n                this.cursor = this.getDraggingCursorName();\n                grid.clearSelections();\n            }\n        }\n        if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        //var col = event.gridCell.x;\n        if (this.dragging) {\n            this.cursor = null;\n            //delay here to give other events a chance to be dropped\n            var self = this;\n            this.endDragColumn(grid);\n            setTimeout(function() {\n                self.attachChain();\n            }, 200);\n        }\n        this.dragCol = -1;\n        this.dragging = false;\n        this.dragArmed = false;\n        this.cursor = null;\n        grid.repaint();\n\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n\n        if (!this.dragging && event.mousePoint.y < 5 && event.viewPoint.y === 0) {\n            this.cursor = this.getCanDragCursorName();\n        } else {\n            this.cursor = null;\n        }\n\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n\n        if (this.isHeaderRow(grid, event) && this.dragging) {\n            this.cursor = this.getDraggingCursorName(); //move';\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc this is the main event handler that manages the dragging of the column\n     * @param {Hypergrid} grid\n     * @param {boolean} draggedToTheRight - are we moving to the right\n     */\n    floatColumnTo: function(grid, draggedToTheRight) {\n        this.floatingNow = true;\n\n        var renderer = grid.getRenderer();\n        var colEdges = renderer.getColumnEdges();\n        var scrollLeft = grid.getHScrollValue();\n        var floaterIndex = grid.renderOverridesCache.floater.columnIndex;\n        var draggerIndex = grid.renderOverridesCache.dragger.columnIndex;\n        var hdpiratio = grid.renderOverridesCache.dragger.hdpiratio;\n\n        var draggerStartX;\n        var floaterStartX;\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var draggerWidth = grid.getColumnWidth(draggerIndex);\n        var floaterWidth = grid.getColumnWidth(floaterIndex);\n\n        var max = grid.getVisibleColumnsCount();\n\n        var doffset = 0;\n        var foffset = 0;\n\n        if (draggerIndex >= fixedColumnCount) {\n            doffset = scrollLeft;\n        }\n        if (floaterIndex >= fixedColumnCount) {\n            foffset = scrollLeft;\n        }\n\n        if (draggedToTheRight) {\n            draggerStartX = colEdges[Math.min(max, draggerIndex - doffset)];\n            floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)];\n\n            grid.renderOverridesCache.dragger.startX = (draggerStartX + floaterWidth) * hdpiratio;\n            grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio;\n\n        } else {\n            floaterStartX = colEdges[Math.min(max, floaterIndex - foffset)];\n            draggerStartX = floaterStartX + draggerWidth;\n\n            grid.renderOverridesCache.dragger.startX = floaterStartX * hdpiratio;\n            grid.renderOverridesCache.floater.startX = draggerStartX * hdpiratio;\n        }\n        grid.swapColumns(draggerIndex, floaterIndex);\n        grid.renderOverridesCache.dragger.columnIndex = floaterIndex;\n        grid.renderOverridesCache.floater.columnIndex = draggerIndex;\n\n\n        this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(grid, floaterStartX, draggerStartX));\n\n        this.doFloaterAnimation(grid);\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc manifest the column drag and drop animation\n     * @param {Hypergrid} grid\n     * @param {number} floaterStartX - the x start coordinate of the column underneath that floats behind the dragged column\n     * @param {number} draggerStartX - the x start coordinate of the dragged column\n     */\n    doColumnMoveAnimation: function(grid, floaterStartX, draggerStartX) {\n        var self = this;\n        return function() {\n            var d = floatColumn;\n            d.style.display = 'inline';\n            self.setCrossBrowserProperty(d, 'transform', 'translate(' + floaterStartX + 'px, ' + 0 + 'px)');\n\n            //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)';\n            //d.style.webkit-webkit-Transform = 'translate(' + floaterStartX + 'px, ' + 0 + 'px)';\n\n            requestAnimationFrame(function() {\n                self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease');\n                self.setCrossBrowserProperty(d, 'transform', 'translate(' + draggerStartX + 'px, ' + -2 + 'px)');\n            });\n            grid.repaint();\n            //need to change this to key frames\n\n            setTimeout(function() {\n                self.setCrossBrowserProperty(d, 'transition', '');\n                grid.renderOverridesCache.floater = null;\n                grid.repaint();\n                self.doFloaterAnimation(grid);\n                requestAnimationFrame(function() {\n                    d.style.display = 'none';\n                    self.isFloatingNow = false;\n                });\n            }, columnAnimationTime + 50);\n        };\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc manifest the floater animation\n     * @param {Hypergrid} grid\n     */\n    doFloaterAnimation: function(grid) {\n        if (this.floaterAnimationQueue.length === 0) {\n            this.floatingNow = false;\n            grid.repaint();\n            return;\n        }\n        var animation = this.floaterAnimationQueue.pop();\n        animation();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc create the float column at columnIndex underneath the dragged column\n     * @param {Hypergrid} grid\n     * @param {number} columnIndex - the index of the column that will be floating\n     */\n    createFloatColumn: function(grid, columnIndex) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n\n        var columnWidth = grid.getColumnWidth(columnIndex);\n        var colHeight = grid.div.clientHeight;\n        var d = floatColumn;\n        var style = d.style;\n        var location = grid.div.getBoundingClientRect();\n\n        style.top = (location.top - 2) + 'px';\n        style.left = location.left + 'px';\n        style.position = 'fixed';\n\n        var hdpiRatio = grid.getHiDPI(floatColumnCTX);\n\n        d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px');\n        d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px');\n        style.boxShadow = '0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)';\n        style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px';\n        style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px';\n        style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor');\n        style.backgroundColor = renderer.resolveProperty('backgroundColor');\n\n        var startX = columnEdges[columnIndex - scrollLeft];\n        startX = startX * hdpiRatio;\n\n        floatColumnCTX.scale(hdpiRatio, hdpiRatio);\n\n        grid.renderOverridesCache.floater = {\n            columnIndex: columnIndex,\n            ctx: floatColumnCTX,\n            startX: startX,\n            width: columnWidth,\n            height: colHeight,\n            hdpiratio: hdpiRatio\n        };\n\n        style.zIndex = '4';\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -2 + 'px)');\n        style.cursor = this.getDraggingCursorName();\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc utility function for setting cross browser css properties\n     * @param {HTMLElement} element - descripton\n     * @param {string} property - the property\n     * @param {string} value - the value to assign\n     */\n    setCrossBrowserProperty: function(element, property, value) {\n        var uProperty = property[0].toUpperCase() + property.substr(1);\n        this.setProp(element, 'webkit' + uProperty, value);\n        this.setProp(element, 'Moz' + uProperty, value);\n        this.setProp(element, 'ms' + uProperty, value);\n        this.setProp(element, 'O' + uProperty, value);\n        this.setProp(element, property, value);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc utility function for setting properties on HTMLElements\n     * @param {HTMLElement} element - descripton\n     * @param {string} property - the property\n     * @param {string} value - the value to assign\n     */\n    setProp: function(element, property, value) {\n        if (property in element.style) {\n            element.style[property] = value;\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc create the dragged column at columnIndex above the floated column\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     * @param {number} columnIndex - the index of the column that will be floating\n     */\n    createDragColumn: function(grid, x, columnIndex) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n        var hdpiRatio = grid.getHiDPI(draggerCTX);\n        var columnWidth = grid.getColumnWidth(columnIndex);\n        var colHeight = grid.div.clientHeight;\n        var d = dragger;\n        var location = grid.div.getBoundingClientRect();\n        var style = d.style;\n\n        style.top = location.top + 'px';\n        style.left = location.left + 'px';\n        style.position = 'fixed';\n        style.opacity = 0.85;\n        style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)';\n        //style.zIndex = 100;\n        style.borderTop = '1px solid ' + renderer.resolveProperty('lineColor');\n        style.backgroundColor = grid.renderer.resolveProperty('backgroundColor');\n\n        d.setAttribute('width', Math.round(columnWidth * hdpiRatio) + 'px');\n        d.setAttribute('height', Math.round(colHeight * hdpiRatio) + 'px');\n\n        style.width = columnWidth + 'px'; //Math.round(columnWidth / hdpiRatio) + 'px';\n        style.height = colHeight + 'px'; //Math.round(colHeight / hdpiRatio) + 'px';\n\n        var startX = columnEdges[columnIndex - scrollLeft];\n        startX = startX * hdpiRatio;\n\n        draggerCTX.scale(hdpiRatio, hdpiRatio);\n\n        grid.renderOverridesCache.dragger = {\n            columnIndex: columnIndex,\n            ctx: draggerCTX,\n            startX: startX,\n            width: columnWidth,\n            height: colHeight,\n            hdpiratio: hdpiRatio\n        };\n\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, -5px)');\n        style.zIndex = '5';\n        style.cursor = this.getDraggingCursorName();\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc this function is the main dragging logic\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    dragColumn: function(grid, x) {\n\n        //TODO: this function is overly complex, refactor this in to something more reasonable\n        var self = this;\n        //var renderer = grid.getRenderer();\n        //var columnEdges = renderer.getColumnEdges();\n\n        var autoScrollingNow = this.columnDragAutoScrollingRight || this.columnDragAutoScrollingLeft;\n\n        var hdpiRatio = grid.getHiDPI(draggerCTX);\n\n        var dragColumnIndex = grid.renderOverridesCache.dragger.columnIndex;\n        var columnWidth = grid.renderOverridesCache.dragger.width;\n\n        var minX = 0; //grid.getFixedColumnsWidth();\n        var maxX = grid.renderer.getFinalVisableColumnBoundary() - columnWidth;\n        x = Math.min(x, maxX + 15);\n        x = Math.max(minX - 15, x);\n\n        //am I at my lower bound\n        var atMin = x < minX && dragColumnIndex !== 0;\n\n        //am I at my upper bound\n        var atMax = x > maxX;\n\n        var d = dragger;\n\n        this.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + 0 + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease');\n\n        this.setCrossBrowserProperty(d, 'transform', 'translate(' + x + 'px, ' + -10 + 'px)');\n        requestAnimationFrame(function() {\n            d.style.display = 'inline';\n        });\n\n        var overCol = grid.renderer.getColumnFromPixelX(x + (d.width / 2 / hdpiRatio));\n\n        if (atMin) {\n            overCol = 0;\n        }\n\n        if (atMax) {\n            overCol = grid.getColumnCount() - 1;\n        }\n\n        var doAFloat = dragColumnIndex > overCol;\n        doAFloat = doAFloat || (overCol - dragColumnIndex >= 1);\n\n        if (doAFloat && !atMax && !autoScrollingNow) {\n            var draggedToTheRight = dragColumnIndex < overCol;\n            // if (draggedToTheRight) {\n            //     overCol = overCol - 1;\n            // }\n            if (this.isFloatingNow) {\n                return;\n            }\n\n            this.isFloatingNow = true;\n            this.createFloatColumn(grid, overCol);\n            this.floatColumnTo(grid, draggedToTheRight);\n        } else {\n\n            if (x < minX - 10) {\n                this.checkAutoScrollToLeft(grid, x);\n            }\n            if (x > minX - 10) {\n                this.columnDragAutoScrollingLeft = false;\n            }\n            //lets check for autoscroll to right if were up against it\n            if (atMax || x > maxX + 10) {\n                this.checkAutoScrollToRight(grid, x);\n                return;\n            }\n            if (x < maxX + 10) {\n                this.columnDragAutoScrollingRight = false;\n            }\n        }\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc autoscroll to the right if necessary\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    checkAutoScrollToRight: function(grid, x) {\n        if (this.columnDragAutoScrollingRight) {\n            return;\n        }\n        this.columnDragAutoScrollingRight = true;\n        this._checkAutoScrollToRight(grid, x);\n    },\n\n    _checkAutoScrollToRight: function(grid, x) {\n        if (!this.columnDragAutoScrollingRight) {\n            return;\n        }\n        var scrollLeft = grid.getHScrollValue();\n        if (!grid.dragging || scrollLeft > (grid.sbHScrollConfig.rangeStop - 2)) {\n            return;\n        }\n        var draggedIndex = grid.renderOverridesCache.dragger.columnIndex;\n        grid.scrollBy(1, 0);\n        var newIndex = draggedIndex + 1;\n        console.log(newIndex, draggedIndex);\n        grid.swapColumns(newIndex, draggedIndex);\n        grid.renderOverridesCache.dragger.columnIndex = newIndex;\n\n        setTimeout(this._checkAutoScrollToRight.bind(this, grid, x), 250);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc autoscroll to the left if necessary\n     * @param {Hypergrid} grid\n     * @param {number} x - the start position\n     */\n    checkAutoScrollToLeft: function(grid, x) {\n        if (this.columnDragAutoScrollingLeft) {\n            return;\n        }\n        this.columnDragAutoScrollingLeft = true;\n        this._checkAutoScrollToLeft(grid, x);\n    },\n\n    _checkAutoScrollToLeft: function(grid, x) {\n        if (!this.columnDragAutoScrollingLeft) {\n            return;\n        }\n\n        var scrollLeft = grid.getHScrollValue();\n        if (!grid.dragging || scrollLeft < 1) {\n            return;\n        }\n        var draggedIndex = grid.renderOverridesCache.dragger.columnIndex;\n        grid.swapColumns(draggedIndex + scrollLeft, draggedIndex + scrollLeft - 1);\n        grid.scrollBy(-1, 0);\n        setTimeout(this._checkAutoScrollToLeft.bind(this, grid, x), 250);\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc a column drag has completed, update data and cleanup\n     * @param {Hypergrid} grid\n     */\n    endDragColumn: function(grid) {\n\n        var fixedColumnCount = grid.getFixedColumnCount();\n        var scrollLeft = grid.getHScrollValue();\n\n        var columnIndex = grid.renderOverridesCache.dragger.columnIndex;\n\n        if (columnIndex < fixedColumnCount) {\n            scrollLeft = 0;\n        }\n\n        var renderer = grid.getRenderer();\n        var columnEdges = renderer.getColumnEdges();\n        var self = this;\n        var startX = columnEdges[columnIndex - scrollLeft];\n        var d = dragger;\n\n        self.setCrossBrowserProperty(d, 'transition', (self.isWebkit ? '-webkit-' : '') + 'transform ' + columnAnimationTime + 'ms ease, box-shadow ' + columnAnimationTime + 'ms ease');\n        self.setCrossBrowserProperty(d, 'transform', 'translate(' + startX + 'px, ' + -1 + 'px)');\n        d.style.boxShadow = '0px 0px 0px #888888';\n\n        setTimeout(function() {\n            grid.renderOverridesCache.dragger = null;\n            grid.repaint();\n            requestAnimationFrame(function() {\n                d.style.display = 'none';\n                grid.endDragColumnNotification();\n            });\n        }, columnAnimationTime + 50);\n\n    },\n\n    /**\n     * @memberOf CellMoving.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isHeaderRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y === 0;\n        return isFixed;\n    }\n\n});\n\nmodule.exports = ColumnMoving;\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnPicker = Feature.extend('ColumnPicker', {\n\n    alias: 'ColumnPicker',\n\n    /**\n     * @memberOf ColumnPicker.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyUp: function(grid, event) {\n        var key = event.detail.char.toLowerCase();\n        var keys = grid.resolveProperty('editorActivationKeys');\n        if (keys.indexOf(key) > -1) {\n           grid.toggleColumnPicker();\n        }\n    },\n\n});\n\nmodule.exports = ColumnPicker;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnResizing = Feature.extend('ColumnResizing', {\n\n    alias: 'ColumnResizing',\n\n    /**\n     * the index of the column wall were currently dragging\n     * @type {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    dragIndex: -2,\n\n    /**\n     * the pixel location of the where the drag was initiated\n     * @type {number}\n     * @default -1\n     * @memberOf ColumnResizing.prototype\n     */\n    dragStart: -1,\n\n    /**\n     * the starting width/height of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf ColumnResizing.prototype\n     */\n    dragIndexStartingSize: -1,\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the mouse x,y coordinate\n     * @returns {number}\n     * @param {MouseEvent} event - the mouse event to query\n     */\n    getMouseValue: function(event) {\n        return event.primitiveEvent.detail.mouse.x;\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the grid cell x,y coordinate\n     * @returns {number}\n     * @param {window.fin.rectangular.Point} gridCell\n     */\n    getGridCellValue: function(gridCell) {\n        return gridCell.y;\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the grids x,y scroll value\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getScrollValue: function(grid) {\n        return grid.getHScrollValue();\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the width/height of the row/column of interest\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getAreaSize: function(grid, index) {\n        return grid.getColumnWidth(index);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc set the width/height of the row/column at index\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     * @param {number} value - the width/height to set to\n     */\n    setAreaSize: function(grid, index, value) {\n        grid.setColumnWidth(index, value);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the recently rendered area's width/height\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getPreviousAbsoluteSize: function(grid, index) {\n        return grid.getRenderedWidth(index);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc returns the index of which divider I'm over\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    overAreaDivider: function(grid, event) {\n        return grid.overColumnDivider(event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc am I over the column/row area\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedOtherArea: function(grid, event) {\n        return this.isFirstFixedRow(grid, event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the cursor name\n     * @returns {string}\n     */\n    getCursorName: function() {\n        return 'col-resize';\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        if (this.dragIndex > -2) {\n            //var fixedAreaCount = this.getFixedAreaCount(grid);\n            //var offset = this.getFixedAreaSize(grid, fixedAreaCount + areaIndex);\n            var mouse = this.getMouseValue(event);\n            var scrollValue = this.getScrollValue(grid);\n            if (this.dragIndex < this.getFixedAreaCount(grid)) {\n                scrollValue = 0;\n            }\n            var previous = this.getPreviousAbsoluteSize(grid, this.dragIndex - scrollValue);\n            var distance = mouse - previous;\n            this.setAreaSize(grid, this.dragIndex, distance);\n        } else if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc get the width/height of a specific row/column\n     * @param {Hypergrid} grid\n     * @param {number} areaIndex - the row/column index of interest\n     */\n    getSize: function(grid, areaIndex) {\n        return this.getAreaSize(grid, areaIndex);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc return the fixed area rows/columns count\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getOtherFixedAreaCount: function(grid) {\n        return grid.getFixedRowCount();\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        var overArea = this.overAreaDivider(grid, event);\n        if (isEnabled && overArea > -1 && this.isFirstFixedOtherArea(grid, event)) {\n            var scrollValue = this.getScrollValue(grid);\n            if (overArea < this.getFixedAreaCount(grid)) {\n                scrollValue = 0;\n            }\n            this.dragIndex = overArea - 1 + scrollValue;\n            this.dragStart = this.getMouseValue(event);\n            this.dragIndexStartingSize = 0;\n            this.detachChain();\n        } else if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        if (isEnabled && this.dragIndex > -2) {\n            this.cursor = null;\n            this.dragIndex = -2;\n\n            event.primitiveEvent.stopPropagation();\n            //delay here to give other events a chance to be dropped\n            var self = this;\n            grid.synchronizeScrollingBoundries();\n            setTimeout(function() {\n                self.attachChain();\n            }, 200);\n        } else if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        if (this.dragIndex > -2) {\n            return;\n        }\n        this.cursor = null;\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n        this.checkForAreaResizeCursorChange(grid, event);\n    },\n\n    /**\n     * @memberOf ColumnResizing.prototype\n     * @desc fill this in\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    checkForAreaResizeCursorChange: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        if (isEnabled && this.overAreaDivider(grid, event) > -1 && this.isFirstFixedOtherArea(grid, event)) {\n            this.cursor = this.getCursorName();\n        } else {\n            this.cursor = null;\n        }\n\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @returns {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    getFixedAreaCount: function(grid) {\n        var count = grid.getFixedColumnCount() + (grid.isShowRowNumbers() ? 1 : 0) + (grid.hasHierarchyColumn() ? 1 : 0);\n        return count;\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @param event\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    handleDoubleClick: function(grid, event) {\n        var isEnabled = this.isEnabled(grid);\n        var hasCursor = this.overAreaDivider(grid, event) > -1; //this.cursor !== null;\n        var headerRowCount = grid.getHeaderRowCount();\n        //var headerColCount = grid.getHeaderColumnCount();\n        var gridCell = event.gridCell;\n        if (isEnabled && hasCursor && (gridCell.y <= headerRowCount)) {\n            grid.autosizeColumn(gridCell.x - 1);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @param {Hypergrid} grid\n     * @returns {boolean}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    isEnabled: function(grid) {\n        return true;\n    }\n\n});\n\nmodule.exports = ColumnResizing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * Extra msecs to avoid race condition with fincanvas's double click timer.\n * @type {number}\n * @defaultvalue 50\n * NOTE: 50 msecs seems to work well. 10 and even 25 proved insufficient in Chrome.\n * @private\n */\nvar RACE_TIME = 50;\n\n/**\n * @constructor\n */\nvar ColumnSelection = Feature.extend('ColumnSelection', {\n\n    alias: 'ColumnSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {window.fin.rectangular.Point}\n     * @default null\n     * @memberOf ColumnSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * The cell coordinates of the where the mouse pointer is during a drag operation.\n     * @type {Object}\n     * @default null\n     * @memberOf ColumnSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf ColumnSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf ColumnSelection.prototype\n     */\n    sbAutoStart: 0,\n\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragging) {\n            this.dragging = false;\n        }\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    handleDoubleClick: function(grid, event) {\n        if (this.doubleClickTimer) {\n            clearTimeout(this.doubleClickTimer); // prevent mouseDown from continuing\n            this.doubleClickTimer = undefined;\n        }\n        if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (this.doubleClickTimer) {\n            return;\n        }\n\n        if ((!grid.isColumnSelection() || event.mousePoint.y < 5) && this.next) {\n            this.next.handleMouseDown(grid, event);\n            return;\n        }\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n\n        var isHeader = grid.isShowHeaderRow() && dy === 0 && dx !== -1;\n\n        if (isRightClick || !isHeader) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n            // HOLD OFF WHILE WAITING FOR DOUBLE-CLICK\n            this.doubleClickTimer = setTimeout(function() {\n                this.doubleClickTimer = undefined;\n                var numFixedColumns = grid.getFixedColumnCount();\n\n                //if we are in the fixed area do not apply the scroll values\n                //check both x and y values independently\n                if (viewCell.x < numFixedColumns) {\n                    dx = viewCell.x;\n                }\n\n                var dCell = grid.newPoint(dx, 0);\n\n                var primEvent = event.primitiveEvent;\n                var keys = primEvent.detail.keys;\n                this.dragging = true;\n                this.extendSelection(grid, dCell, keys);\n            }.bind(this), grid.resolveProperty('doubleClickDelay') + RACE_TIME);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n\n        if ((!grid.isColumnSelection() || this.isColumnDragging(grid)) && this.next) {\n            this.next.handleMouseDrag(grid, event);\n            return;\n        }\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (isRightClick || !this.dragging) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n\n            var numFixedColumns = grid.getFixedColumnCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.x < numFixedColumns) {\n                dx = viewCell.x;\n            }\n\n            var dCell = grid.newPoint(dx, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (grid.getLastSelectionType() !== 'column') {\n            if (this.next) {\n                this.next.handleKeyDown(grid, event);\n            }\n            return;\n        }\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Handle a mousedrag selection\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n        var x = gridCell.x;\n        //            var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        var newX = x - mouseDown.x;\n        //var newY = y - mouseDown.y;\n\n        // if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n        //     return;\n        // }\n\n        grid.clearMostRecentColumnSelection();\n\n        grid.selectColumn(mouseDown.x, x);\n        grid.setDragExtent(grid.newPoint(newX, 0));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (this.currentDrag.x < b.origin.x) {\n            xOffset = -1;\n        }\n\n        if (this.currentDrag.x > b.origin.x + b.extent.x) {\n            xOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        grid.stopEditing();\n        //var hasCTRL = keys.indexOf('CTRL') !== -1;\n        var hasSHIFT = keys.indexOf('SHIFT') !== -1;\n\n        // var scrollTop = grid.getVScrollValue();\n        // var scrollLeft = grid.getHScrollValue();\n\n        // var numFixedColumns = 0;//grid.getFixedColumnCount();\n        // var numFixedRows = 0;//grid.getFixedRowCount();\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        //we have repeated a click in the same spot deslect the value from last time\n        // if (mousePoint && x === mousePoint.x && y === mousePoint.y) {\n        //     grid.clearSelections();\n        //     grid.popMouseDown();\n        //     grid.repaint();\n        //     return;\n        // }\n\n        // if (!hasCTRL && !hasSHIFT) {\n        //     grid.clearSelections();\n        // }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentColumnSelection();\n            grid.selectColumn(x, mousePoint.x);\n            grid.setDragExtent(grid.newPoint(x - mousePoint.x, 0));\n        } else {\n            grid.toggleSelectColumn(x, keys);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid) {\n\n        // var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n        // var maxRows = grid.getRowCount() - 1;\n\n        // var newX = mouseCorner.x;\n        // var newY = grid.getHeaderRowCount() + grid.getVScrollValue();\n\n        // newY = Math.min(maxRows, newY);\n\n        // grid.clearSelections();\n        // grid.select(newX, newY, 0, 0);\n        // grid.setMouseDown(new grid.rectangular.Point(newX, newY));\n        // grid.setDragExtent(new grid.rectangular.Point(0, 0));\n\n        // grid.repaint();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid) {},\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {\n        this.moveSingleSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n        this.moveSingleSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetX) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumns() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newX = extent.x + offsetX;\n        //var newY = grid.getRowCount();\n\n        newX = Math.min(maxColumns - origin.x, Math.max(-origin.x, newX));\n\n        grid.clearMostRecentColumnSelection();\n        grid.selectColumn(origin.x, origin.x + newX);\n\n        grid.setDragExtent(grid.newPoint(newX, 0));\n\n        if (grid.insureModelColIsVisible(newX + origin.x, offsetX)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf ColumnSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetX) {\n\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var maxViewableColumns = grid.getVisibleColumnsCount() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxColumns = Math.min(maxColumns, maxViewableColumns);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newX = mouseCorner.x + offsetX;\n        //var newY = grid.getRowCount();\n\n        newX = Math.min(maxColumns, Math.max(0, newX));\n\n        grid.clearSelections();\n        grid.selectColumn(newX);\n        grid.setMouseDown(grid.newPoint(newX, 0));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelColIsVisible(newX, offsetX)) {\n            this.pingAutoScroll();\n        }\n\n        grid.repaint();\n\n    },\n\n    isColumnDragging: function(grid) {\n        var dragger = grid.lookupFeature('ColumnMoving');\n        if (!dragger) {\n            return false;\n        }\n        var isActivated = dragger.dragging && !this.dragging;\n        return isActivated;\n    }\n\n});\n\nmodule.exports = ColumnSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ColumnSorting = Feature.extend('ColumnSorting', {\n\n    alias: 'ColumnSorting',\n\n    /**\n     * @memberOf ColumnSorting.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n\n    handleDoubleClick: function(grid, event) {\n        var gridCell = event.gridCell;\n        if (grid.isShowHeaderRow() && gridCell.y === 0 && gridCell.x !== -1) {\n            var keys = event.primitiveEvent.detail.keys;\n            grid.toggleSort(gridCell.x, keys);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf ColumnSorting.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        var y = event.gridCell.y;\n        if (this.isFixedRow(grid, event) && y < 1) {\n            this.cursor = 'pointer';\n        } else {\n            this.cursor = null;\n        }\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = ColumnSorting;\n","'use strict';\n\nvar Base = require('../lib/Base');\n\n/**\n * @constructor\n * @desc instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n */\nvar Feature = Base.extend('Feature', {\n\n    /**\n     * the next feature to be given a chance to handle incoming events\n     * @type {Feature}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    next: null,\n\n    /**\n     * a temporary holding field for my next feature when I'm in a disconnected state\n     * @type {Feature}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    detached: null,\n\n    /**\n     * the cursor I want to be displayed\n     * @type {string}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    cursor: null,\n\n    /**\n     * the cell location where the cursor is currently\n     * @type {Point}\n     * @default null\n     * @memberOf Feature.prototype\n     */\n    currentHoverCell: null,\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc set my next field, or if it's populated delegate to the feature in my next field\n     * @param {Feature} nextFeature - this is how we build the chain of responsibility\n     */\n    setNext: function(nextFeature) {\n        if (this.next) {\n            this.next.setNext(nextFeature);\n        } else {\n            this.next = nextFeature;\n            this.detached = nextFeature;\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc disconnect my child\n     */\n    detachChain: function() {\n        this.next = null;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc reattach my child from the detached reference\n     */\n    attachChain: function() {\n        this.next = this.detached;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle mouse move down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseMove: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseMove(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseExit: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseExit(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseEnter: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseEnter(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (this.next) {\n            this.next.handleKeyDown(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyUp: function(grid, event) {\n        if (this.next) {\n            this.next.handleKeyUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleWheelMoved: function(grid, event) {\n        if (this.next) {\n            this.next.handleWheelMoved(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDoubleClick: function(grid, event) {\n        if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleTap: function(grid, event) {\n        if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        if (this.next) {\n            this.next.handleMouseDrag(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleContextMenu: function(grid, event) {\n        if (this.next) {\n            this.next.handleContextMenu(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc toggle the column picker\n     */\n\n    moveSingleSelect: function(grid, x, y) {\n        if (this.next) {\n            this.next.moveSingleSelect(grid, x, y);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFixedRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y < grid.getFixedRowCount();\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedRow: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.y < 1;\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFixedColumn: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var isFixed = gridCell.x < grid.getFixedColumnCount();\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedColumn: function(grid, event) {\n        var gridCell = event.viewPoint;\n        var edge = grid.isShowRowNumbers() ? 0 : 1;\n        var isFixed = gridCell.x < edge;\n        return isFixed;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isTopLeft: function(grid, event) {\n        var isTopLeft = this.isFixedRow(grid, event) && this.isFixedColumn(grid, event);\n        return isTopLeft;\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    setCursor: function(grid) {\n        if (this.next) {\n            this.next.setCursor(grid);\n        }\n        if (this.cursor) {\n            grid.beCursor(this.cursor);\n        }\n    },\n\n    /**\n     * @memberOf Feature.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    initializeOn: function(grid) {\n        if (this.next) {\n            this.next.initializeOn(grid);\n        }\n    }\n\n});\n\nmodule.exports = Feature;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar Filters = Feature.extend('Filters', {\n\n    alias: 'Filters',\n\n    handleDoubleClick: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleDoubleClick(grid, event);\n        }\n    },\n\n    handleTap: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n            grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleTap(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf CellEditing.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleHoldPulse: function(grid, event) {\n        var isDoubleClickEditorActivation = grid.resolveProperty('editOnDoubleClick');\n        if (this.checkActivateEditor(grid, event, !isDoubleClickEditorActivation)) {\n           grid._activateEditor(event);\n        } else if (this.next) {\n            this.next.handleHoldPulse(grid, event);\n        }\n    },\n\n    checkActivateEditor: function(grid, event, isDoubleClickEditorActivation) {\n        var isFilterRow = grid.isFilterRow(event.gridCell.y);\n        var activateEditor = isDoubleClickEditorActivation && isFilterRow;\n        return activateEditor;\n    }\n\n});\n\nmodule.exports = Filters;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\nvar commands = {\n    PAGEDOWN: function(grid) { grid.pageDown(); },\n    PAGEUP: function(grid) { grid.pageUp(); },\n    PAGELEFT: function(grid) { grid.pageLeft(); },\n    PAGERIGHT: function(grid) { grid.pageRight(); }\n};\n\n/**\n * @constructor\n */\nvar KeyPaging = Feature.extend('KeyPaging', {\n\n    alias: 'KeyPaging',\n\n    /**\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf KeyPaging.prototype\n     */\n    handleKeyDown: function(grid, event) {\n        var detail = event.detail.char;\n        var func = commands[detail];\n        if (func) {\n            func(grid);\n        } else if (this.next) {\n            this.next.handleKeyDown(grid, event);\n        }\n    }\n\n});\n\nmodule.exports = KeyPaging;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar OnHover = Feature.extend('OnHover', {\n\n    alias: 'OnHover',\n\n    /**\n     * @desc Hhandle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     * @memberOf OnHover.prototype\n     */\n    handleMouseMove: function(grid, event) {\n        var currentHoverCell = grid.getHoverCell();\n        if (!event.gridCell.equals(currentHoverCell)) {\n            if (currentHoverCell) {\n                this.handleMouseExit(grid, currentHoverCell);\n            }\n            this.handleMouseEnter(grid, event);\n            grid.setHoverCell(event.gridCell);\n        } else {\n            if (this.next) {\n                this.next.handleMouseMove(grid, event);\n            }\n        }\n    }\n\n});\n\nmodule.exports = OnHover;\n","'use strict';\n\nvar ColumnResizing = require('./ColumnResizing');\n\n/**\n * @constructor\n */\nvar RowResizing = ColumnResizing.extend('RowResizing', {\n\n    alias: 'RowResizing',\n\n    /**\n     * the index of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragArea: -1,\n\n    /**\n     * the pixel location of the where the drag was initiated\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragStart: -1,\n\n    /**\n     * the starting width/height of the row/column we are dragging\n     * @type {number}\n     * @default -1\n     * @memberOf RowResizing.prototype\n     */\n    dragAreaStartingSize: -1,\n\n    /**\n     * @memberOf RowResizing.prototype\n     * @desc get the mouse x,y coordinate\n     * @returns {number}\n     * @param {MouseEvent} event - the mouse event to query\n     */\n    getMouseValue: function(event) {\n        return event.primitiveEvent.detail.mouse.y;\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc get the grid cell x,y coordinate\n     * @returns {number}\n     * @param {Point} gridCell\n     */\n    getGridCellValue: function(gridCell) {\n        return gridCell.x;\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the grids x,y scroll value\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getScrollValue: function(grid) {\n        return grid.getVScrollValue();\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the width/height of the row/column of interest\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getAreaSize: function(grid, index) {\n        return grid.getRowHeight(index);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc set the width/height of the row/column at index\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     * @param {number} value - the width/height to set to\n     */\n    setAreaSize: function(grid, index, value) {\n        grid.setRowHeight(index, value);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc returns the index of which divider I'm over\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    overAreaDivider: function(grid, event) {\n        return grid.overRowDivider(event);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc am I over the column/row area\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    isFirstFixedOtherArea: function(grid, event) {\n        return this.isFirstFixedColumn(grid, event);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the cursor name\n     * @returns {string}\n     */\n    getCursorName: function() {\n        return 'row-resize';\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the recently rendered area's width/height\n     * @returns {number}\n     * @param {Hypergrid} grid\n     * @param {number} index - the row/column index of interest\n     */\n    getPreviousAbsoluteSize: function(grid, index) {\n        return grid.getRenderedHeight(index);\n    },\n\n    /**\n     * @function\n     * @memberOf RowResizing.prototype\n     * @desc return the fixed area rows/columns count\n     * @returns {number}\n     * @param {Hypergrid} grid\n     */\n    getOtherFixedAreaCount: function(grid) {\n        return grid.getFixedColumnCount();\n    },\n\n    /**\n     *\n     * @param {Hypergrid} grid\n     * @returns {number}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    getFixedAreaCount: function(grid) {\n        return grid.getFixedRowCount() + grid.getHeaderRowCount();\n    },\n\n    /**\n     *\n     * @param {Hypergrid} grid\n     * @returns {boolean}\n     * @default -2\n     * @memberOf ColumnResizing.prototype\n     */\n    isEnabled: function(grid) {\n        return grid.isRowResizeable();\n    }\n\n});\n\nmodule.exports = RowResizing;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar RowSelection = Feature.extend('RowSelection', {\n\n    alias: 'RowSelection',\n\n    /**\n     * The pixel location of the mouse pointer during a drag operation.\n     * @type {Point}\n     * @default null\n     * @memberOf RowSelection.prototype\n     */\n    currentDrag: null,\n\n    /**\n     * The cell coordinates of the where the mouse pointer is during a drag operation.\n     * @type {Object}\n     * @default null\n     * @memberOf RowSelection.prototype\n     */\n    lastDragCell: null,\n\n    /**\n     * a millisecond value representing the previous time an autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf RowSelection.prototype\n     */\n    sbLastAuto: 0,\n\n    /**\n     * a millisecond value representing the time the current autoscroll started\n     * @type {number}\n     * @default 0\n     * @memberOf RowSelection.prototype\n     */\n    sbAutoStart: 0,\n\n    dragArmed: false,\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseUp: function(grid, event) {\n        if (this.dragArmed) {\n            this.dragArmed = false;\n            //global row selection\n            if (event.gridCell.x === -1 && event.gridCell.y === 0) {\n                grid.toggleSelectAllRows();\n            }\n            grid.fireSyntheticRowSelectionChangedEvent();\n        } else if (this.dragging) {\n            this.dragging = false;\n            grid.fireSyntheticRowSelectionChangedEvent();\n        } else if (this.next) {\n            this.next.handleMouseUp(grid, event);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc * @desc Handle this event down the feature chain of responsibility.\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDown: function(grid, event) {\n\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n        var cell = event.gridCell;\n        var viewCell = event.viewPoint;\n        var dx = cell.x;\n        var dy = cell.y;\n\n\n        var isHeader = grid.isShowRowNumbers() && dx < 0;\n\n        if (!grid.isRowSelection() || isRightClick || !isHeader) {\n            if (this.next) {\n                this.next.handleMouseDown(grid, event);\n            }\n        } else {\n\n            var numFixedRows = grid.getFixedRowCount();\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(0, dy);\n\n            var primEvent = event.primitiveEvent;\n            var keys = primEvent.detail.keys;\n            this.dragArmed = true;\n            this.extendSelection(grid, dCell, keys);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleMouseDrag: function(grid, event) {\n        var isRightClick = event.primitiveEvent.detail.isRightClick;\n\n        if (!this.dragArmed || !grid.isRowSelection() || isRightClick) {\n            if (this.next) {\n                this.next.handleMouseDrag(grid, event);\n            }\n        } else {\n            this.dragging = true;\n            var numFixedRows = grid.getFixedRowCount();\n\n            var cell = event.gridCell;\n            var viewCell = event.viewPoint;\n            //var dx = cell.x;\n            var dy = cell.y;\n\n            //if we are in the fixed area do not apply the scroll values\n            //check both x and y values independently\n            if (viewCell.y < numFixedRows) {\n                dy = viewCell.y;\n            }\n\n            var dCell = grid.newPoint(0, dy);\n\n            var primEvent = event.primitiveEvent;\n            this.currentDrag = primEvent.detail.mouse;\n            this.lastDragCell = dCell;\n\n            this.checkDragScroll(grid, this.currentDrag);\n            this.handleMouseDragCellSelection(grid, dCell, primEvent.detail.keys);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleKeyDown: function(grid, event) {\n        if (grid.getLastSelectionType() !== 'row') {\n            if (this.next) {\n                this.next.handleKeyDown(grid, event);\n            }\n            return;\n        }\n        var command = 'handle' + event.detail.char;\n        if (this[command]) {\n            this[command].call(this, grid, event.detail);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Handle a mousedrag selection\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    handleMouseDragCellSelection: function(grid, gridCell, keys) {\n        var y = gridCell.y;\n        //            var previousDragExtent = grid.getDragExtent();\n        var mouseDown = grid.getMouseDown();\n\n        var newY = y - mouseDown.y;\n        //var newY = y - mouseDown.y;\n\n        // if (previousDragExtent.x === newX && previousDragExtent.y === newY) {\n        //     return;\n        // }\n\n        grid.clearMostRecentRowSelection();\n\n        grid.selectRow(mouseDown.y, y);\n        grid.setDragExtent(grid.newPoint(0, newY));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc this checks while were dragging if we go outside the visible bounds, if so, kick off the external autoscroll check function (above)\n     * @param {Hypergrid} grid\n     * @param {Object} mouse - the event details\n     */\n    checkDragScroll: function(grid, mouse) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var b = grid.getDataBounds();\n        var inside = b.contains(mouse);\n        if (inside) {\n            if (grid.isScrollingNow()) {\n                grid.setScrollingNow(false);\n            }\n        } else if (!grid.isScrollingNow()) {\n            grid.setScrollingNow(true);\n            this.scrollDrag(grid);\n        }\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc this function makes sure that while we are dragging outside of the grid visible bounds, we srcroll accordingly\n     * @param {Hypergrid} grid\n     */\n    scrollDrag: function(grid) {\n        if (!grid.isScrollingNow()) {\n            return;\n        }\n\n        var lastDragCell = this.lastDragCell;\n        var b = grid.getDataBounds();\n        var xOffset = 0;\n        var yOffset = 0;\n\n        var numFixedColumns = grid.getFixedColumnCount();\n        var numFixedRows = grid.getFixedRowCount();\n\n        var dragEndInFixedAreaX = lastDragCell.x < numFixedColumns;\n        var dragEndInFixedAreaY = lastDragCell.y < numFixedRows;\n\n        if (this.currentDrag.y < b.origin.y) {\n            yOffset = -1;\n        }\n\n        if (this.currentDrag.y > b.origin.y + b.extent.y) {\n            yOffset = 1;\n        }\n\n        var dragCellOffsetX = xOffset;\n        var dragCellOffsetY = yOffset;\n\n        if (dragEndInFixedAreaX) {\n            dragCellOffsetX = 0;\n        }\n\n        if (dragEndInFixedAreaY) {\n            dragCellOffsetY = 0;\n        }\n\n        this.lastDragCell = lastDragCell.plusXY(dragCellOffsetX, dragCellOffsetY);\n        grid.scrollBy(xOffset, yOffset);\n        this.handleMouseDragCellSelection(grid, lastDragCell, []); // update the selection\n        grid.repaint();\n        setTimeout(this.scrollDrag.bind(this, grid), 25);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc extend a selection or create one if there isnt yet\n     * @param {Hypergrid} grid\n     * @param {Object} gridCell - the event details\n     * @param {Array} keys - array of the keys that are currently pressed down\n     */\n    extendSelection: function(grid, gridCell, keys) {\n        grid.stopEditing();\n        //var hasCTRL = keys.indexOf('CTRL') !== -1;\n        var hasSHIFT = keys.indexOf('SHIFT') !== -1;\n\n        var mousePoint = grid.getMouseDown();\n        var x = gridCell.x; // - numFixedColumns + scrollLeft;\n        var y = gridCell.y; // - numFixedRows + scrollTop;\n\n        //were outside of the grid do nothing\n        if (x < 0 || y < 0) {\n            return;\n        }\n\n        if (hasSHIFT) {\n            grid.clearMostRecentRowSelection();\n            grid.selectRow(y, mousePoint.y);\n            grid.setDragExtent(grid.newPoint(0, y - mousePoint.y));\n        } else {\n            grid.toggleSelectRow(y, keys);\n            grid.setMouseDown(grid.newPoint(x, y));\n            grid.setDragExtent(grid.newPoint(0, 0));\n        }\n        grid.repaint();\n    },\n\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     */\n    handleDOWNSHIFT: function(grid) {\n        this.moveShiftSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUPSHIFT: function(grid) {\n        this.moveShiftSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFTSHIFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHTSHIFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleDOWN: function(grid) {\n        this.moveSingleSelect(grid, 1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleUP: function(grid) {\n        this.moveSingleSelect(grid, -1);\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleLEFT: function(grid) {},\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc handle this event\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleRIGHT: function(grid) {\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n        var maxColumns = grid.getColumnCount() - 1;\n\n        var newX = grid.getHeaderColumnCount() + grid.getHScrollValue();\n        var newY = mouseCorner.y;\n\n        newX = Math.min(maxColumns, newX);\n\n        grid.clearSelections();\n        grid.select(newX, newY, 0, 0);\n        grid.setMouseDown(grid.newPoint(newX, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        grid.repaint();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc If we are holding down the same navigation key, accelerate the increment we scroll\n     * #### returns: integer\n     */\n    getAutoScrollAcceleration: function() {\n        var count = 1;\n        var elapsed = this.getAutoScrollDuration() / 2000;\n        count = Math.max(1, Math.floor(elapsed * elapsed * elapsed * elapsed));\n        return count;\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc set the start time to right now when we initiate an auto scroll\n     */\n    setAutoScrollStartTime: function() {\n        this.sbAutoStart = Date.now();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc update the autoscroll start time if we haven't autoscrolled within the last 500ms otherwise update the current autoscroll time\n     */\n    pingAutoScroll: function() {\n        var now = Date.now();\n        if (now - this.sbLastAuto > 500) {\n            this.setAutoScrollStartTime();\n        }\n        this.sbLastAuto = Date.now();\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc answer how long we have been auto scrolling\n     * #### returns: integer\n     */\n    getAutoScrollDuration: function() {\n        if (Date.now() - this.sbLastAuto > 500) {\n            return 0;\n        }\n        return Date.now() - this.sbAutoStart;\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Augment the most recent selection extent by (offsetX,offsetY) and scroll if necessary.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveShiftSelect: function(grid, offsetY) {\n\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableRows = grid.getVisibleRows() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var origin = grid.getMouseDown();\n        var extent = grid.getDragExtent();\n\n        var newY = extent.y + offsetY;\n        //var newY = grid.getRowCount();\n\n        newY = Math.min(maxRows - origin.y, Math.max(-origin.y, newY));\n\n        grid.clearMostRecentRowSelection();\n        grid.selectRow(origin.y, origin.y + newY);\n\n        grid.setDragExtent(grid.newPoint(0, newY));\n\n        if (grid.insureModelRowIsVisible(newY + origin.y, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.fireSyntheticRowSelectionChangedEvent();\n        grid.repaint();\n\n    },\n\n    /**\n     * @memberOf RowSelection.prototype\n     * @desc Replace the most recent selection with a single cell selection that is moved (offsetX,offsetY) from the previous selection extent.\n     * @param {Hypergrid} grid\n     * @param {number} offsetX - x coordinate to start at\n     * @param {number} offsetY - y coordinate to start at\n     */\n    moveSingleSelect: function(grid, offsetY) {\n\n        var maxRows = grid.getRowCount() - 1;\n\n        var maxViewableRows = grid.getVisibleRowsCount() - 1;\n\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            maxRows = Math.min(maxRows, maxViewableRows);\n        }\n\n        var mouseCorner = grid.getMouseDown().plus(grid.getDragExtent());\n\n        var newY = mouseCorner.y + offsetY;\n        //var newY = grid.getRowCount();\n\n        newY = Math.min(maxRows, Math.max(0, newY));\n\n        grid.clearSelections();\n        grid.selectRow(newY);\n        grid.setMouseDown(grid.newPoint(0, newY));\n        grid.setDragExtent(grid.newPoint(0, 0));\n\n        if (grid.insureModelRowIsVisible(newY, offsetY)) {\n            this.pingAutoScroll();\n        }\n\n        grid.fireSyntheticRowSelectionChangedEvent();\n        grid.repaint();\n\n    },\n\n    isSingleRowSelection: function() {\n        return true;\n    }\n\n});\n\nmodule.exports = RowSelection;\n","'use strict';\n\nvar Feature = require('./Feature.js');\n\n/**\n * @constructor\n */\nvar ThumbwheelScrolling = Feature.extend('ThumbwheelScrolling', {\n\n    alias: 'ThumbwheelScrolling',\n\n    /**\n     * @memberOf ThumbwheelScrolling.prototype\n     * @desc handle this event down the feature chain of responsibility\n     * @param {Hypergrid} grid\n     * @param {Object} event - the event details\n     */\n    handleWheelMoved: function(grid, e) {\n        if (!grid.resolveProperty('scrollingEnabled')) {\n            return;\n        }\n        var primEvent = e.primitiveEvent;\n        var deltaY = primEvent.wheelDeltaY || -primEvent.deltaY;\n        var deltaX = primEvent.wheelDeltaX || -primEvent.deltaX;\n        if (deltaY > 0) {\n            grid.scrollBy(0, -1);\n        } else if (deltaY < -0) {\n            grid.scrollBy(0, 1);\n        } else if (deltaX > 0) {\n            grid.scrollBy(-1, 0);\n        } else if (deltaX < -0) {\n            grid.scrollBy(1, 0);\n        }\n    }\n\n});\n\n\nmodule.exports = ThumbwheelScrolling;\n","'use strict';\n\nmodule.exports = {\n    Feature: require('./Feature'), // abstract base class\n    CellClick: require('./CellClick'),\n    CellEditing: require('./CellEditing'),\n    CellSelection: require('./CellSelection'),\n    ColumnAutosizing: require('./ColumnAutosizing'),\n    ColumnMoving: require('./ColumnMoving'),\n    ColumnResizing: require('./ColumnResizing'),\n    ColumnSelection: require('./ColumnSelection'),\n    ColumnSorting: require('./ColumnSorting'),\n    Filters: require('./Filters'),\n    KeyPaging: require('./KeyPaging'),\n    OnHover: require('./OnHover'),\n    ColumnPicker: require('./ColumnPicker'),\n    RowResizing: require('./RowResizing'),\n    RowSelection: require('./RowSelection'),\n    ThumbwheelScrolling: require('./ThumbwheelScrolling')\n};\n","'use strict';\n\nvar deprecated = require('./deprecated');\nvar Base = require('extend-me').Base;\n\nBase.prototype.deprecated = deprecated;\n\nmodule.exports = Base;\n","'use strict';\n\nvar Base = require('./Base');\n\n/** @constructor\n * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n *\n * See {@link CellProvider#initialize|initialize} which is called by the constructor.\n */\nvar CellProvider = Base.extend('CellProvider', {\n\n    /**\n     * @summary Constructor logic\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     * @memberOf CellProvider.prototype\n     */\n    initialize: function() {\n        this.cellCache = {};\n        this.initializeCells();\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getColumnHeaderCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    /**\n     * @desc replace this function in on your instance of cellProvider\n     * @returns cell\n     * @param {object} config - an object with everything you might need for renderering a cell\n     * @memberOf CellProvider.prototype\n     */\n    getRowHeaderCell: function(config) {\n        var cell = this.cellCache.simpleCellRenderer;\n        cell.config = config;\n        return cell;\n    },\n\n    paintButton: function(gc, config) {\n        var val = config.value;\n        var c = config.x;\n        var r = config.y;\n        var bounds = config.bounds;\n        var x = bounds.x + 2;\n        var y = bounds.y + 2;\n        var width = bounds.width - 3;\n        var height = bounds.height - 3;\n        var radius = height / 2;\n        var arcGradient = gc.createLinearGradient(x, y, x, y + height);\n        if (config.mouseDown) {\n            arcGradient.addColorStop(0, '#B5CBED');\n            arcGradient.addColorStop(1, '#4d74ea');\n        } else {\n            arcGradient.addColorStop(0, '#ffffff');\n            arcGradient.addColorStop(1, '#aaaaaa');\n        }\n        gc.fillStyle = arcGradient;\n        gc.strokeStyle = '#000000';\n        roundRect(gc, x, y, width, height, radius, arcGradient, true);\n\n        var ox = (width - config.getTextWidth(gc, val)) / 2;\n        var oy = (height - config.getTextHeight(gc.font).descent) / 2;\n\n        if (gc.textBaseline !== 'middle') {\n            gc.textBaseline = 'middle';\n        }\n\n        gc.fillStyle = '#000000';\n\n        config.backgroundColor = 'rgba(0,0,0,0)';\n        gc.fillText(val, x + ox, y + oy);\n\n        //identify that we are a button\n        config.buttonCells[c + ',' + r] = true;\n    },\n\n    /**\n     * @summary The default cell rendering function for rendering a vanilla cell.\n     * @desc Great care has been taken in crafting this function as it needs to perform extremely fast. Reads on the gc object are expensive but not quite as expensive as writes to it. We do our best to avoid writes, then avoid reads. Clipping bounds are not set here as this is also an expensive operation. Instead, we truncate overflowing text and content by filling a rectangle with background color column by column instead of cell by cell.  This column by column fill happens higher up on the stack in a calling function from fin-hypergrid-renderer.  Take note we do not do cell by cell border renderering as that is expensive.  Instead we render many fewer gridlines after all cells are rendered.\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} config.bounds.x - the x screen coordinate of my origin\n     * @param {number} config.bounds.y - the y screen coordinate of my origin\n     * @param {number} config.bounds.width - the width I'm allowed to draw within\n     * @param {number} config.bounds.height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    defaultCellPaint: function(gc, config) {\n        var val = config.value,\n            x = config.bounds.x,\n            y = config.bounds.y,\n            width = config.bounds.width,\n            height = config.bounds.height,\n            wrapHeaders = config.headerTextWrapping,\n            leftPadding = 2, //TODO: fix this\n            isHeader = config.y === 0;\n\n        var leftIcon, rightIcon, centerIcon, ixoffset, iyoffset;\n\n        // setting gc properties are expensive, let's not do it needlessly\n\n        if (val && val.constructor === Array) {\n            leftIcon = val[0];\n            rightIcon = val[2];\n            val = val[1];\n            if (val && typeof val === 'object') {\n                if (val.constructor.name === 'HTMLImageElement') { // must be an image\n                    centerIcon = val;\n                    val = null;\n                }\n            }\n            if (leftIcon && leftIcon.nodeName !== 'IMG') {\n                leftIcon = null;\n            }\n            if (rightIcon && rightIcon.nodeName !== 'IMG') {\n                rightIcon = null;\n            }\n            if (centerIcon && centerIcon.nodeName !== 'IMG') {\n                centerIcon = null;\n            }\n        }\n\n        val = valueOrFunctionExecute(config, val);\n\n        val = config.formatter(val);\n\n        if (gc.font !== config.font) {\n            gc.font = config.font;\n        }\n        if (gc.textAlign !== 'left') {\n            gc.textAlign = 'left';\n        }\n        if (gc.textBaseline !== 'middle') {\n            gc.textBaseline = 'middle';\n        }\n\n        // fill background only if our bgColor is populated or we are a selected cell\n        var backgroundColor, hover, hoverColor, selectColor;\n        if (config.isCellHovered && config.hoverCellHighlight.enabled) {\n            hoverColor = config.hoverCellHighlight.backgroundColor;\n        } else if (config.isRowHovered && (hover = config.hoverRowHighlight).enabled) {\n            hoverColor = config.isGridColumn || !hover.header || hover.header.backgroundColor === undefined ? hover.backgroundColor : hover.header.backgroundColor;\n        } else if (config.isColumnHovered && (hover = config.hoverColumnHighlight).enabled) {\n            hoverColor = config.isGridRow || !hover.header || hover.header.backgroundColor === undefined ? hover.backgroundColor : hover.header.backgroundColor;\n        }\n        if (alpha(hoverColor) < 1) {\n            if (config.isSelected) {\n                selectColor = valueOrFunctionExecute(config, config.backgroundSelectionColor);\n            }\n            if (alpha(selectColor) < 1) {\n                backgroundColor = valueOrFunctionExecute(config, config.backgroundColor);\n                if (alpha(backgroundColor) > 0) {\n                    gc.fillStyle = backgroundColor;\n                    gc.fillRect(x, y, width, height);\n                }\n            }\n            if (selectColor !== undefined) {\n                gc.fillStyle = selectColor;\n                gc.fillRect(x, y, width, height);\n            }\n        }\n        if (hoverColor !== undefined) {\n            gc.fillStyle = hoverColor;\n            gc.fillRect(x, y, width, height);\n        }\n\n        // draw text\n        var theColor = valueOrFunctionExecute(config, config.isSelected ? config.foregroundSelectionColor : config.color);\n        if (gc.fillStyle !== theColor) {\n            gc.fillStyle = theColor;\n            gc.strokeStyle = theColor;\n        }\n\n        if (isHeader && wrapHeaders) {\n            this.renderMultiLineText(gc, x, y, height, width, config, val);\n        } else {\n            this.renderSingleLineText(gc, x, y, height, width, config, val);\n        }\n\n        var iconWidth = 0;\n        if (leftIcon) {\n            iyoffset = Math.round((height - leftIcon.height) / 2);\n            gc.drawImage(leftIcon, x + leftPadding, y + iyoffset);\n            iconWidth = Math.max(leftIcon.width + 2);\n        }\n        if (rightIcon && width > 1.75 * height) {\n            iyoffset = Math.round((height - rightIcon.height) / 2);\n            var rightX = x + width - rightIcon.width;\n            if (backgroundColor !== undefined) {\n                gc.fillStyle = backgroundColor;\n                gc.fillRect(rightX, y, rightIcon.width, height);\n            } else {\n                gc.clearRect(rightX, y, rightIcon.width, height);\n            }\n            gc.drawImage(rightIcon, rightX, y + iyoffset);\n            iconWidth = Math.max(rightIcon.width + 2);\n        }\n        if (centerIcon) {\n            iyoffset = Math.round((height - centerIcon.height) / 2);\n            ixoffset = Math.round((width - centerIcon.width) / 2);\n            gc.drawImage(centerIcon, x + width - ixoffset - centerIcon.width, y + iyoffset);\n            iconWidth = Math.max(centerIcon.width + 2);\n        }\n        if (config.cellBorderThickness) {\n            gc.beginPath();\n            gc.rect(x, y, width, height);\n            gc.lineWidth = config.cellBorderThickness;\n            gc.strokeStyle = config.cellBorderStyle;\n\n            // animate the dashed line a bit here for fun\n\n            gc.stroke();\n            gc.closePath();\n        }\n        config.minWidth = config.minWidth + 2 * (iconWidth);\n    },\n\n    renderMultiLineText: function(gc, x, y, height, width, config, val) {\n        var lines = fitText(gc, config, val, width);\n        if (lines.length === 1) {\n            return this.renderSingleLineText(gc, x, y, height, width, config, squeeze(val));\n        }\n\n        var colHEdgeOffset = config.cellPadding,\n            halignOffset = 0,\n            valignOffset = config.voffset,\n            halign = config.halign,\n            textHeight = config.getTextHeight(config.font).height;\n\n        switch (halign) {\n            case 'right':\n                halignOffset = width - colHEdgeOffset;\n                break;\n            case 'center':\n                halignOffset = width / 2;\n                break;\n            case 'left':\n                halignOffset = colHEdgeOffset;\n                break;\n        }\n\n        var hMin = 0, vMin = Math.ceil(textHeight / 2);\n\n        valignOffset += Math.ceil((height - (lines.length - 1) * textHeight) / 2);\n\n        halignOffset = Math.max(hMin, halignOffset);\n        valignOffset = Math.max(vMin, valignOffset);\n\n        gc.save(); // define a clipping region for cell\n        gc.rect(x, y, width, height);\n        gc.clip();\n\n        gc.textAlign = halign;\n\n        for (var i = 0; i < lines.length; i++) {\n            gc.fillText(lines[i], x + halignOffset, y + valignOffset + (i * textHeight));\n        }\n\n        gc.restore(); // discard clipping region\n    },\n\n    renderSingleLineText: function(gc, x, y, height, width, config, val) {\n        var colHEdgeOffset = config.cellPadding,\n            halignOffset = 0,\n            valignOffset = config.voffset,\n            halign = config.halign,\n            isCellHovered = config.isCellHovered,\n            isLink = config.link;\n\n        var fontMetrics = config.getTextHeight(config.font);\n        var textWidth = config.getTextWidth(gc, val);\n\n        //we must set this in order to compute the minimum width\n        //for column autosizing purposes\n        config.minWidth = textWidth + (2 * colHEdgeOffset);\n\n        switch (halign) {\n            case 'right':\n                //textWidth = config.getTextWidth(gc, config.value);\n                halignOffset = width - colHEdgeOffset - textWidth;\n                break;\n            case 'center':\n                //textWidth = config.getTextWidth(gc, config.value);\n                halignOffset = (width - textWidth) / 2;\n                break;\n            case 'left':\n                halignOffset = colHEdgeOffset;\n                break;\n        }\n\n        halignOffset = Math.max(0, halignOffset);\n        valignOffset = valignOffset + Math.ceil(height / 2);\n\n        if (val !== null) {\n            gc.fillText(val, x + halignOffset, y + valignOffset);\n        }\n\n        if (isCellHovered) {\n            gc.beginPath();\n            if (isLink) {\n                underline(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1);\n                gc.stroke();\n            }\n            gc.closePath();\n        }\n        if (config.strikeThrough === true) {\n            gc.beginPath();\n            strikeThrough(config, gc, val, x + halignOffset, y + valignOffset + Math.floor(fontMetrics.height / 2), 1);\n            gc.stroke();\n            gc.closePath();\n        }\n    },\n\n    /**\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     * @desc Emerson's paint function for a slider button. currently the user cannot interact with it\n     */\n    paintSlider: function(gc, x, y, width, height) {\n        // gc.strokeStyle = 'white';\n        // var val = this.config.value;\n        // var radius = height / 2;\n        // var offset = width * val;\n        // var bgColor = this.config.isSelected ? this.config.bgSelColor : '#333333';\n        // var btnGradient = gc.createLinearGradient(x, y, x, y + height);\n        // btnGradient.addColorStop(0, bgColor);\n        // btnGradient.addColorStop(1, '#666666');\n        // var arcGradient = gc.createLinearGradient(x, y, x, y + height);\n        // arcGradient.addColorStop(0, '#aaaaaa');\n        // arcGradient.addColorStop(1, '#777777');\n        // gc.fillStyle = btnGradient;\n        // roundRect(gc, x, y, width, height, radius, btnGradient);\n        // if (val < 1.0) {\n        //     gc.fillStyle = arcGradient;\n        // } else {\n        //     gc.fillStyle = '#eeeeee';\n        // }\n        // gc.beginPath();\n        // gc.arc(x + Math.max(offset - radius, radius), y + radius, radius, 0, 2 * Math.PI);\n        // gc.fill();\n        // gc.closePath();\n        // this.config.minWidth = 100;\n    },\n\n    /**\n     * @desc A simple implementation of a sparkline, because it's a barchart we've changed the name ;).\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    paintSparkbar: function(gc, x, y, width, height) {\n        gc.beginPath();\n        var val = this.config.value;\n        if (!val || !val.length) {\n            return;\n        }\n        var count = val.length;\n        var eWidth = width / count;\n        var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n        gc.fillStyle = fgColor;\n        for (var i = 0; i < val.length; i++) {\n            var barheight = val[i] / 110 * height;\n            gc.fillRect(x + 5, y + height - barheight, eWidth * 0.6666, barheight);\n            x = x + eWidth;\n        }\n        gc.closePath();\n        this.config.minWidth = count * 10;\n\n    },\n\n    /**\n     * @desc A simple implementation of a sparkline.  see [Edward Tufte sparkline](http://www.edwardtufte.com/bboard/q-and-a-fetch-msg?msg_id=0001OR)\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    paintSparkline: function(gc, x, y, width, height) {\n        gc.beginPath();\n        var val = this.config.value;\n        if (!val || !val.length) {\n            return;\n        }\n        var count = val.length;\n        var eWidth = width / count;\n\n        var fgColor = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n        gc.strokeStyle = fgColor;\n        gc.fillStyle = fgColor;\n        gc.beginPath();\n        var prev;\n        for (var i = 0; i < val.length; i++) {\n            var barheight = val[i] / 110 * height;\n            if (!prev) {\n                prev = barheight;\n            }\n            gc.lineTo(x + 5, y + height - barheight);\n            gc.arc(x + 5, y + height - barheight, 1, 0, 2 * Math.PI, false);\n            x = x + eWidth;\n        }\n        this.config.minWidth = count * 10;\n        gc.stroke();\n        gc.closePath();\n    },\n\n    /**\n     * @desc A simple implementation of a tree cell renderer for use mainly with the qtree.\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    treeCellRenderer: function(gc, x, y, width, height) {\n        var val = this.config.value.data;\n        var indent = this.config.value.indent;\n        var icon = this.config.value.icon;\n\n        //fill background only if our bgColor is populated or we are a selected cell\n        if (this.config.bgColor || this.config.isSelected) {\n            gc.fillStyle = this.config.isSelected ? this.config.bgSelColor : this.config.bgColor;\n            gc.fillRect(x, y, width, height);\n        }\n\n        if (!val || !val.length) {\n            return;\n        }\n        var valignOffset = Math.ceil(height / 2);\n\n        gc.fillStyle = this.config.isSelected ? this.config.fgSelColor : this.config.fgColor;\n        gc.fillText(icon + val, x + indent, y + valignOffset);\n\n        var textWidth = this.config.getTextWidth(gc, icon + val);\n        var minWidth = x + indent + textWidth + 10;\n        this.config.minWidth = minWidth;\n    },\n\n    /**\n     * @desc An empty implementation of a cell renderer, see [the null object pattern](http://c2.com/cgi/wiki?NullObject).\n     * @param {CanvasGraphicsContext} gc\n     * @param {number} x - the x screen coordinate of my origin\n     * @param {number} y - the y screen coordinate of my origin\n     * @param {number} width - the width I'm allowed to draw within\n     * @param {number} height - the height I'm allowed to draw within\n     * @memberOf CellProvider.prototype\n     */\n    emptyCellRenderer: function(gc, x, y, width, height) {},\n\n    /**\n     * @memberOf CellProvider.prototype\n     * @private\n     */\n    initializeCells: function() {\n        var self = this;\n        this.cellCache.simpleCellRenderer = {\n            paint: this.defaultCellPaint,\n            renderSingleLineText: this.renderSingleLineText,\n            renderMultiLineText: this.renderMultiLineText\n        };\n        this.cellCache.sliderCellRenderer = {\n            paint: this.paintSlider\n        };\n        this.cellCache.sparkbarCellRenderer = {\n            paint: this.paintSparkbar\n        };\n        this.cellCache.sparklineCellRenderer = {\n            paint: this.paintSparkline\n        };\n        this.cellCache.treeCellRenderer = {\n            paint: this.treeCellRenderer\n        };\n        this.cellCache.emptyCellRenderer = {\n            paint: this.emptyCellRenderer\n        };\n        this.cellCache.buttonRenderer = {\n            paint: this.paintButton,\n            //defaultCellPaint: this.defaultCellPaint\n        };\n        this.cellCache.linkCellRenderer = {\n            paint: function(gc, x, y, width, height) {\n                self.config = this.config;\n                self.defaultCellPaint(gc, x, y, width, height, true);\n            }\n        };\n    }\n});\n\nfunction valueOrFunctionExecute(config, valueOrFunction) {\n    var isFunction = (((typeof valueOrFunction)[0]) === 'f');\n    var result = isFunction ? valueOrFunction(config) : valueOrFunction;\n    if (!result && result !== 0) {\n        return '';\n    }\n    return result;\n}\n\nfunction underline(config, gc, text, x, y, thickness) {\n    var width = config.getTextWidth(gc, text);\n\n    switch (gc.textAlign) {\n        case 'center':\n            x -= (width / 2);\n            break;\n        case 'right':\n            x -= width;\n            break;\n    }\n\n    //gc.beginPath();\n    gc.lineWidth = thickness;\n    gc.moveTo(x + 0.5, y + 0.5);\n    gc.lineTo(x + width + 0.5, y + 0.5);\n}\n\nfunction strikeThrough(config, gc, text, x, y, thickness) {\n    var fontMetrics = config.getTextHeight(config.font);\n    var width = config.getTextWidth(gc, text);\n    y = y - (fontMetrics.height * 0.4);\n\n    switch (gc.textAlign) {\n        case 'center':\n            x -= (width / 2);\n            break;\n        case 'right':\n            x -= width;\n            break;\n    }\n\n    //gc.beginPath();\n    gc.lineWidth = thickness;\n    gc.moveTo(x + 0.5, y + 0.5);\n    gc.lineTo(x + width + 0.5, y + 0.5);\n}\n\nfunction findLines(gc, config, words, width) {\n\n    if (words.length === 1) {\n        return words;\n    }\n\n    // starting with just the first word…\n    var stillFits, line = [words.shift()];\n    while (\n        // so lone as line still fits within current column…\n    (stillFits = config.getTextWidth(gc, line.join(' ')) < width)\n        // …AND there are more words available…\n    && words.length\n        ) {\n        // …add another word to end of line and retest\n        line.push(words.shift());\n    }\n\n    if (\n        !stillFits // if line is now too long…\n        && line.length > 1 // …AND is multiple words…\n    ) {\n        words.unshift(line.pop()); // …back off by (i.e., remove) one word\n    }\n\n    line = [line.join(' ')];\n\n    if (words.length) { // if there's anything left…\n        line = line.concat(findLines(gc, config, words, width)); // …break it up as well\n    }\n\n    return line;\n}\n\nfunction fitText(gc, config, string, width) {\n    return findLines(gc, config, squeeze(string).split(' '), width);\n}\n\n// trim string; then reduce all runs of multiple spaces to a single space\nfunction squeeze(string) {\n    return string.toString().trim().replace(/\\s\\s+/g, ' ');\n}\n\nfunction roundRect(gc, x, y, width, height, radius, fill, stroke) {\n\n    if (!stroke) {\n        stroke = true;\n    }\n    if (!radius) {\n        radius = 5;\n    }\n    gc.beginPath();\n    gc.moveTo(x + radius, y);\n    gc.lineTo(x + width - radius, y);\n    gc.quadraticCurveTo(x + width, y, x + width, y + radius);\n    gc.lineTo(x + width, y + height - radius);\n    gc.quadraticCurveTo(x + width, y + height, x + width - radius, y + height);\n    gc.lineTo(x + radius, y + height);\n    gc.quadraticCurveTo(x, y + height, x, y + height - radius);\n    gc.lineTo(x, y + radius);\n    gc.quadraticCurveTo(x, y, x + radius, y);\n    gc.closePath();\n    if (stroke) {\n        gc.stroke();\n    }\n    if (fill) {\n        gc.fill();\n    }\n    gc.closePath();\n}\n\nfunction alpha(cssColorSpec) {\n    if (cssColorSpec === undefined) {\n        // undefined so not visible; treat as transparent\n        return 0;\n    }\n\n    var matches = cssColorSpec.match(alpha.regex);\n\n    if (matches === null) {\n        // an opaque color (a color spec with no alpha channel)\n        return 1;\n    }\n\n    var A = matches[4];\n\n    if (A === undefined) {\n        // cssColorSpec must have been 'transparent'\n        return 0;\n    }\n\n    return Number(A);\n}\n\nalpha.regex = /^(transparent|((RGB|HSL)A\\(.*,\\s*([\\d\\.]+)\\)))$/i;\n\nmodule.exports = CellProvider;\n","'use strict';\n\nvar FilterTree = require('filter-tree');\n\n/** @typedef {function} fieldsProviderFunc\n * @returns {fieldOption[]} see jsdoc typedef in filter-tree/js/FillerLeaf.js\n */\n\nvar validateQuietlyOptions = {\n    alert: false,\n    focus: false\n};\n\n/** @constructor\n */\nfunction CustomFilter() {\n    this.getDisplayString = function() { // TODO: What the heck is this? (not in use)\n        return '< ' + this.value;\n    };\n\n    // Following methods service dialog during which `this.filterTree` is valid\n    this.initialize = function(fieldsProvider) {\n        /** @type {fieldsProviderFunc}\n         */\n        this.fieldsProvider = fieldsProvider;\n        this.filterTree = new FilterTree({\n            fields: fieldsProvider()\n        });\n        delete this.filter; // forces this.create to recreate the filter function\n    };\n\n    this.onShow = function(container) {\n        container.appendChild(this.filterTree.el);\n    };\n\n    this.onOk = function() {\n        return this.filterTree.validate();\n    };\n\n    this.getState = function() {\n        var state = JSON.parse(JSON.stringify(this.filterTree)); // calls toJSON functions as needed\n        delete this.filterTree;\n        return state;\n    };\n\n    this.onReset = function() {\n\n    };\n\n    this.onDelete = function() {\n        delete this.filter;\n        delete this.filterTree;\n    };\n\n    this.onCancel = function() {\n        delete this.filterTree;\n    };\n\n    // Following methods called with `state` are independent of dialog; `this.filterTree` is undefined\n    this.create = function(state) {\n        if (!this.filter) {\n            var filterTree = this.setState(state);\n            var dataRow = {};\n            var fieldOption = this.fieldsProvider()[0],\n                fieldName = fieldOption.name || fieldOption;\n\n            if (!filterTree.validate(validateQuietlyOptions)) {\n                // `validate()` returned `undefined` meaning valid (returns error string when invalid)\n                this.filter = function(data) {\n                    dataRow[fieldName] = data;\n                    return filterTree.test(dataRow);\n                };\n            }\n        }\n        return this.filter;\n    };\n\n    this.setState = function(state) {\n        return (\n            this.filterTree = new FilterTree({\n                state: state,\n                fields: this.fieldsProvider()\n            })\n        );\n    };\n}\n\nmodule.exports = CustomFilter;\n","'use strict';\n\nfunction nn(number) {\n    return (number < 10 ? '-0' : '-') + number;\n}\n\nmodule.exports = {\n    date: function(value) {\n        return value instanceof Date\n            ? value.getFullYear() + nn(value.getMonth() + 1) + nn(value.getDate())\n            : value + '';\n    },\n    default: function(value) {\n        return value + '';\n    }\n};\n","'use strict';\n\nmodule.exports = (function() {\n\n    var oidPrefix = '.~.#%_'; //this should be something we never will see at the begining of a string\n    var counter = 0;\n\n    var hash = function(key) {\n        var typeOf = typeof key;\n        switch (typeOf) {\n            case 'number':\n                return oidPrefix + typeOf + '_' + key;\n            case 'string':\n                return oidPrefix + typeOf + '_' + key;\n            case 'boolean':\n                return oidPrefix + typeOf + '_' + key;\n            case 'symbol':\n                return oidPrefix + typeOf + '_' + key;\n            case 'undefined':\n                return oidPrefix + 'undefined';\n            case 'object':\n            case 'function':\n                /*eslint-disable */\n                if (!key.___finhash) {\n                    key.___finhash = oidPrefix + counter++;\n                }\n                return key.___finhash;\n                /*eslint-enable */\n        }\n    };\n\n    // Object.is polyfill, courtesy of @WebReflection\n    var is = Object.is ||\n        function(a, b) {\n            return a === b ? a !== 0 || 1 / a == 1 / b : a != a && b != b; // eslint-disable-line\n        };\n\n    // More reliable indexOf, courtesy of @WebReflection\n    var betterIndexOf = function(arr, value) {\n        if (value != value || value === 0) { // eslint-disable-line\n            for (var i = arr.length; i-- && !is(arr[i], value);) { // eslint-disable-line\n            }\n        } else {\n            i = [].indexOf.call(arr, value);\n        }\n        return i;\n    };\n\n    function Mappy() {\n        this.keys = [];\n        this.data = {};\n        this.values = [];\n    }\n\n    Mappy.prototype.set = function(key, value) {\n        var hashCode = hash(key);\n        if (this.data[hashCode] === undefined) {\n            this.keys.push(key);\n            this.values.push(value);\n        }\n        this.data[hashCode] = value;\n    };\n\n    Mappy.prototype.get = function(key) {\n        var hashCode = hash(key);\n        return this.data[hashCode];\n    };\n\n    Mappy.prototype.getIfAbsent = function(key, ifAbsentFunc) {\n        var value = this.get(key);\n        if (value === undefined) {\n            value = ifAbsentFunc(key, this);\n        }\n        return value;\n    };\n\n    Mappy.prototype.size = function() {\n        return this.keys.length;\n    };\n\n    Mappy.prototype.clear = function() {\n        this.keys.length = 0;\n        this.data = {};\n    };\n\n    Mappy.prototype.delete = function(key) {\n        var hashCode = hash(key);\n        if (this.data[hashCode] === undefined) {\n            return;\n        }\n        var index = betterIndexOf(this.keys, key);\n        this.keys.splice(index, 1);\n        this.values.splice(index, 1);\n        delete this.data[hashCode];\n    };\n\n    Mappy.prototype.forEach = function(func) {\n        var keys = this.keys;\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            func(value, key, this);\n        }\n    };\n\n    Mappy.prototype.map = function(func) {\n        var keys = this.keys;\n        var newMap = new Mappy();\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            var transformed = func(value, key, this);\n            newMap.set(key, transformed);\n        }\n        return newMap;\n    };\n\n    Mappy.prototype.copy = function() {\n        var keys = this.keys;\n        var newMap = new Mappy();\n        for (var i = 0; i < keys.length; i++) {\n            var key = keys[i];\n            var value = this.get(key);\n            newMap.set(key, value);\n        }\n        return newMap;\n    };\n\n    return Mappy;\n\n})();\n","/* eslint-env browser */\n/* global requestAnimationFrame */\n\n'use strict';\n\nvar _ = require('object-iterators');\nvar Base = require('./Base');\n\nvar images = require('../../images/index');\n\n/** @typedef {object} CanvasRenderingContext2D\n * @see [CanvasRenderingContext2D](https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D)\n */\n\n/**\n * @constructor\n * @desc fin-hypergrid-renderer is the canvas enabled top level sub component that handles the renderering of the Grid.\n *\n * It relies on two other external subprojects\n *\n * 1. fin-canvas: a wrapper to provide a simpler interface to the HTML5 canvas component\n * 2. rectangular: a small npm module providing Point and Rectangle objects\n *\n * The fin-hypergrid-renderer is in a unique position to provide critical functionality to the fin-hypergrid in a hightly performant manner.\n * Because it MUST iterate over all the visible cells it can store various bits of information that can be encapsulated as a service for consumption by the fin-hypergrid component.\n *\n * Instances of this object have basically four main functions.\n *\n * 1. render fixed row headers\n * 2. render fixed col headers\n * 3. render main data cells\n * 4. render grid lines\n *\n * Same parameters as {@link Renderer#initialize|initialize}, which is called by this constructor.\n *\n */\nvar Renderer = Base.extend('Renderer', {\n\n    //the shared single item \"pooled\" cell object for drawing each cell\n    cell: {\n        x: 0,\n        y: 0,\n        width: 0,\n        height: 0\n    },\n\n    scrollHeight: 0,\n\n    viewHeight: 0,\n\n    reset: function() {\n        this.bounds = {\n            width: 0,\n            height: 0\n        };\n        this.columnEdges = [];\n        this.columnEdgesIndexMap = [];\n        this.renderedColumnMinWidths = [];\n        this.rowEdges = [];\n        this.rowEdgesIndexMap = [];\n        this.visibleColumns = [];\n        this.visibleRows = [];\n        this.insertionBounds = [];\n    },\n\n    /**\n     * @summary Constructor logic\n     * @desc This method will be called upon instantiation of this class or of any class that extends from this class.\n     * > All `initialize()` methods in the inheritance chain are called, in turn, each with the same parameters that were passed to the constructor, beginning with that of the most \"senior\" class through that of the class of the new instance.\n     * @memberOf Renderer.prototype\n     */\n    initialize: function(grid) {\n        this.grid = grid;\n        this.reset();\n    },\n\n    //this function computes the grid coordinates used for extremely fast iteration over\n    //painting the grid cells. this function is very fast, for thousand rows X 100 columns\n    //on a modest machine taking usually 0ms and no more that 3 ms.\n    computeCellsBounds: function() {\n\n        //var startTime = Date.now();\n\n        var scrollTop = this.getScrollTop(),\n            scrollLeft = this.getScrollLeft(),\n\n            numColumns = this.getColumnCount(),\n            numFixedColumns = this.getFixedColumnCount(),\n\n            numRows = this.getRowCount(),\n            numFixedRows = this.getFixedRowCount(),\n\n            bounds = this.getBounds(),\n            grid = this.grid,\n            numberOfBottomTotalsRows = grid.behavior.getDataModel().getBottomTotals().length,\n            viewWidth = bounds.width || grid.canvas.width, // if 0, we must be in bootstrap\n            viewHeight = bounds.height - numberOfBottomTotalsRows * grid.behavior.getDefaultRowHeight(),\n\n            insertionBoundsCursor = 0,\n            previousInsertionBoundsCursorValue = 0,\n\n            start = 0,\n            x = 0, y = 0,\n            c, r,\n            vx, vy,\n            width, height,\n            firstVX, lastVX,\n            firstVY, lastVY;\n\n        this.getColumnEdges().length = 0;\n        this.rowEdges.length = 0;\n\n        this.columnEdges[0] = 0;\n        this.rowEdges[0] = 0;\n        this.scrollHeight = 0;\n\n        this.visibleColumns.length = 0;\n        this.visibleRows.length = 0;\n        this.columnEdgesIndexMap = [];\n        this.rowEdgesIndexMap = [];\n\n        this.insertionBounds = [];\n\n        if (this.grid.isShowRowNumbers()) {\n            start--;\n            this.columnEdges[-1] = -1;\n        }\n\n        for (c = start; c < numColumns; c++) {\n            vx = c;\n            if (c >= numFixedColumns) {\n                vx = vx + scrollLeft;\n                if (firstVX === undefined) {\n                    firstVX = vx;\n                }\n                lastVX = vx;\n            }\n            if (x > viewWidth || numColumns <= vx) {\n                break;\n            }\n            width = grid.getColumnWidth(vx);\n            x = x + width;\n            this.columnEdges[c + 1] = Math.round(x);\n            this.visibleColumns[c] = vx;\n            this.columnEdgesIndexMap[vx] = c;\n\n            insertionBoundsCursor = insertionBoundsCursor + Math.round(width / 2) + previousInsertionBoundsCursorValue;\n            this.insertionBounds.push(insertionBoundsCursor);\n            previousInsertionBoundsCursorValue = Math.round(width / 2);\n        }\n\n        for (r = 0; r < numRows; r++) {\n            vy = r;\n            if (r >= numFixedRows) {\n                vy = vy + scrollTop;\n                if (firstVY === undefined) {\n                    firstVY = vy;\n                }\n                lastVY = vy;\n            }\n            if (y > viewHeight || numRows <= vy) {\n                break;\n            }\n            height = grid.getRowHeight(vy);\n            y = y + height;\n            this.rowEdges[r + 1] = Math.round(y);\n            this.visibleRows[r] = vy;\n            this.rowEdgesIndexMap[vy] = r;\n        }\n        this.viewHeight = viewHeight;\n        this.dataWindow = this.grid.newRectangle(firstVX, firstVY, lastVX - firstVX, lastVY - firstVY);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {Object} a property value at a key, delegates to the grid\n     */\n    resolveProperty: function(key) {\n        return this.grid.resolveProperty(key);\n    },\n\n    /** @deprecated Use `.grid` property instead.\n     * @memberOf Renderer.prototype\n     * @returns {Hypergrid} grid\n     */\n    getGrid: function() {\n        return this.deprecated('grid', { since: '0.2' });\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Notify the fin-hypergrid everytime we've repainted.\n     * @desc This is the entry point from fin-canvas.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    _paint: function(gc) {\n        if (this.grid) {\n            this.renderGrid(gc);\n            this.grid.gridRenderedNotification();\n        }\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Answer how many rows we rendered\n     */\n    getVisibleRowsCount: function() {\n        return this.visibleRows.length - 1;\n    },\n\n    getVisibleScrollHeight: function() {\n        return this.viewHeight - this.grid.getFixedRowsHeight();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number[]} Rows we just rendered.\n     */\n    getVisibleRows: function() {\n        return this.visibleRows;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Numer of columns we just rendered.\n     */\n    getVisibleColumnsCount: function() {\n        return this.visibleColumns.length - 1;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Columns we just rendered.\n     */\n    getVisibleColumns: function() {\n        return this.visibleColumns;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The column index whne the mouseEvent coordinates are over a column divider.\n     */\n    overColumnDivider: function(x) {\n        x = Math.round(x);\n        var edges = this.getColumnEdges();\n        var whichCol = edges.indexOf(x - 1);\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x - 2);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x + 1);\n        }\n        if (whichCol < 0) {\n            whichCol = edges.indexOf(x - 3);\n        }\n\n        return whichCol;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row index when the mouseEvent coordinates are over a row divider.\n     */\n    overRowDivider: function(y) {\n        y = Math.round(y);\n        var which = this.rowEdges.indexOf(y + 1);\n        if (which < 0) {\n            which = this.rowEdges.indexOf(y);\n        }\n        if (which < 0) {\n            which = this.rowEdges.indexOf(y - 1);\n        }\n        return which;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {Point} cell\n     * @returns {Rectangle} Bounding rect of the given `cell`.\n     */\n    getBoundsOfCell: function(cell) {\n        return this._getBoundsOfCell(cell.x, cell.y);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {number} c - The horizontal coordinate.\n     * @param {number} r - The vertical coordinate.\n     * @returns {Rectangle} Bounding rect of cell with the given coordinates.\n     */\n    _getBoundsOfCell: function(c, r) {\n        var xOutside = false,\n            yOutside = false,\n            cell = this.cell;\n\n        var y, x = this.columnEdgesIndexMap[c];\n        if (x === undefined) {\n            x = this.columnEdgesIndexMap[c - 1];\n            xOutside = true;\n        }\n\n        var oy, ox = this.columnEdges[x],\n            cy, cx = this.columnEdges[x + 1],\n            ey, ex = cx - ox;\n\n        cell.x = xOutside ? cx : ox;\n        cell.width = xOutside ? 0 : ex;\n\n        if (r < 0) { // bottom totals rows\n            var behavior = this.grid.behavior,\n                bounds = this.getBounds();\n\n            ey = behavior.getDefaultRowHeight();\n            oy = bounds.height + r * ey;\n            cy = oy + ey;\n        } else {\n            y = this.rowEdgesIndexMap[r];\n            if (y === undefined) {\n                y = this.rowEdgesIndexMap[r - 1];\n                yOutside = true;\n            }\n\n            oy = this.rowEdges[y];\n            cy = this.rowEdges[y + 1];\n            ey = cy - oy;\n        }\n\n        cell.y = yOutside ? cy : oy;\n        cell.height = yOutside ? 0 : ey;\n\n        return cell;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc answer the column index under the coordinate at pixelX\n     * @param {number} pixelX - The horizontal coordinate.\n     * @returns {number} The column index under the coordinate at pixelX.\n     */\n    getColumnFromPixelX: function(pixelX) {\n        var width = 0,\n            fixedColumnCount = this.getFixedColumnCount(),\n            scrollLeft = this.grid.getHScrollValue(),\n            edges = this.getColumnEdges();\n\n        for (var c = 1; c < edges.length - 1; c++) {\n            width = edges[c] - (edges[c] - edges[c - 1]) / 2;\n            if (pixelX < width) {\n                if (c > fixedColumnCount) {\n                    c = c + scrollLeft;\n                }\n                return c - 1;\n            }\n        }\n        if (c > fixedColumnCount) {\n            c = c + scrollLeft;\n        }\n        return c - 1;\n    },\n\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc Answer specific data cell coordinates given mouse coordinates in pixels.\n     * @param {Point} point\n     * @returns {Point} Cell coordinates\n     */\n    getGridCellFromMousePoint: function(point) {\n\n        var behavior = this.grid.behavior;\n        var width = 0;\n        var height = 0;\n        var x, y, c, r;\n        var previous = 0;\n        var columnEdges = this.getColumnEdges();\n        var fixedColumnCount = this.getFixedColumnCount(); // + gridSize;\n        var fixedRowCount = this.getFixedRowCount();\n\n        // var fixedColumnCount = this.getFixedColumnCount();\n        // var fixedRowCount = this.getFixedRowCount();\n        var scrollX = this.getScrollLeft();\n        var scrollY = this.getScrollTop();\n\n        for (c = 0; c < columnEdges.length; c++) {\n            width = columnEdges[c];\n            if (point.x < width) {\n                x = Math.max(0, point.x - previous - 2);\n                break;\n            }\n            previous = width;\n        }\n        c--;\n        previous = 0;\n        for (r = 0; r < this.rowEdges.length; r++) {\n            height = this.rowEdges[r];\n            if (point.y < height) {\n                y = Math.max(0, point.y - previous - 2);\n                break;\n            }\n            previous = height;\n        }\n        r--;\n        if (point.x < 0) {\n            c = -1;\n        }\n        if (point.y < 0) {\n            r = -1;\n        }\n\n        var viewPoint = this.grid.newPoint(c, r);\n\n        //compensate if we are scrolled\n        if (c >= fixedColumnCount) {\n            c = c + scrollX;\n        }\n        if (r >= fixedRowCount) {\n            r = r + scrollY;\n        }\n\n        var translatedIndex = -1;\n\n        var column = behavior.getColumn(c);\n        if (column) {\n            translatedIndex = column.index;\n        }\n\n        return {\n            gridCell: this.grid.newPoint(c, r),\n            mousePoint: this.grid.newPoint(x, y),\n            viewPoint: viewPoint,\n            dataCell: this.grid.newPoint(translatedIndex, r),\n        };\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines if a column is visible.\n     * @param {number} colIndex - the column index*\n     * @returns {boolean} The given column is fully visible.\n     */\n    isColumnVisible: function(colIndex) {\n        var isVisible = this.visibleColumns.indexOf(colIndex) !== -1;\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The width x coordinate of the last rendered column\n     */\n    getFinalVisableColumnBoundary: function() {\n        var isMaxX = this.isLastColumnVisible();\n        var chop = isMaxX ? 2 : 1;\n        var colWall = this.getColumnEdges()[this.getColumnEdges().length - chop];\n        var result = Math.min(colWall, this.getBounds().width - 200);\n        return result;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines visibility of a row.\n     * @param {number} rowIndex - the row index\n     * @returns {boolean} The given row is fully visible.\n     */\n    isRowVisible: function(rowIndex) {\n        var isVisible = this.visibleRows.indexOf(rowIndex) !== -1;\n        return isVisible;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @summary Determines if a cell is selected.\n     * @param {number} x - the x cell coordinate\n     * @param {number} y - the y cell coordinate*\n     * @returns {boolean} The given cell is fully visible.\n     */\n    isSelected: function(x, y) {\n        return this.grid.isSelected(x, y);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc This is the main forking of the renderering task.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    renderGrid: function(gc) {\n        gc.beginPath();\n\n        this.paintCells(gc);\n        this.paintGridlines(gc);\n        this.renderOverrides(gc);\n        this.renderFocusCell(gc);\n        gc.closePath();\n    },\n\n    focusLineStep: [\n        [5, 5],\n        [0, 1, 5, 4],\n        [0, 2, 5, 3],\n        [0, 3, 5, 2],\n        [0, 4, 5, 1],\n        [0, 5, 5, 0],\n        [1, 5, 4, 0],\n        [2, 5, 3, 0],\n        [3, 5, 2, 0],\n        [4, 5, 1, 0]\n    ],\n\n    renderFocusCell: function(gc) {\n        gc.beginPath();\n        this._renderFocusCell(gc);\n        gc.closePath();\n    },\n\n    _renderFocusCell: function(gc) {\n\n        var selections = this.grid.selectionModel.getSelections();\n        if (!selections || selections.length === 0) {\n            return;\n        }\n        var selection = selections[selections.length - 1];\n        var mouseDown = selection.origin;\n        if (mouseDown.x === -1) {\n            //no selected area, lets exit\n            return;\n        }\n\n        var visibleColumns = this.getVisibleColumns();\n        var visibleRows = this.getVisibleRows();\n        var lastVisibleColumn = visibleColumns[visibleColumns.length - 1];\n        var lastVisibleRow = visibleRows[visibleRows.length - 1];\n\n        var extent = selection.extent;\n\n        var dpOX = Math.min(mouseDown.x, mouseDown.x + extent.x);\n        var dpOY = Math.min(mouseDown.y, mouseDown.y + extent.y);\n\n        //lets check if our selection rectangle is scrolled outside of the visible area\n        if (dpOX > lastVisibleColumn) {\n            return; //the top of our rectangle is below visible\n        }\n        if (dpOY > lastVisibleRow) {\n            return; //the left of our rectangle is to the right of being visible\n        }\n\n        var dpEX = Math.max(mouseDown.x, mouseDown.x + extent.x) + 1;\n        dpEX = Math.min(dpEX, 1 + lastVisibleColumn);\n\n        var dpEY = Math.max(mouseDown.y, mouseDown.y + extent.y) + 1;\n        dpEY = Math.min(dpEY, 1 + lastVisibleRow);\n\n        var o = this._getBoundsOfCell(dpOX, dpOY);\n        var ox = Math.round((o.x === undefined) ? this.grid.getFixedColumnsWidth() : o.x);\n        var oy = Math.round((o.y === undefined) ? this.grid.getFixedRowsHeight() : o.y);\n        // var ow = o.width;\n        // var oh = o.height;\n        var e = this._getBoundsOfCell(dpEX, dpEY);\n        var ex = Math.round((e.x === undefined) ? this.grid.getFixedColumnsWidth() : e.x);\n        var ey = Math.round((e.y === undefined) ? this.grid.getFixedRowsHeight() : e.y);\n        // var ew = e.width;\n        // var eh = e.height;\n        var x = Math.min(ox, ex);\n        var y = Math.min(oy, ey);\n        var width = 1 + ex - ox;\n        var height = 1 + ey - oy;\n        if (x === ex) {\n            width = ox - ex;\n        }\n        if (y === ey) {\n            height = oy - ey;\n        }\n        if (width * height < 1) {\n            //if we are only a skinny line, don't render anything\n            return;\n        }\n\n        gc.rect(x, y, width, height);\n        gc.fillStyle = this.resolveProperty('selectionRegionOverlayColor');\n        gc.fill();\n        gc.lineWidth = 1;\n        gc.strokeStyle = this.resolveProperty('selectionRegionOutlineColor');\n\n        // animate the dashed line a bit here for fun\n\n        gc.stroke();\n\n        //gc.rect(x, y, width, height);\n\n        //gc.strokeStyle = 'white';\n\n        // animate the dashed line a bit here for fun\n        //gc.setLineDash(this.focusLineStep[Math.floor(10 * (Date.now() / 300 % 1)) % this.focusLineStep.length]);\n\n        //gc.stroke();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc iterate the renderering overrides and manifest each\n     * @param {CanvasRenderingContext2D} gc\n     */\n    renderOverrides: function(gc) {\n        var cache = this.grid.renderOverridesCache;\n        for (var key in cache) {\n            if (cache.hasOwnProperty(key)) {\n                var override = cache[key];\n                if (override) {\n                    this.renderOverride(gc, override);\n                }\n            }\n        }\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc copy each overrides specified area to it's target and blank out the source area\n     * @param {CanvasRenderingContext2D} gc\n     * @param {OverrideObject} override - an object with details contain an area and a target context\n     */\n    renderOverride: function(gc, override) {\n        //lets blank out the drag row\n        var hdpiRatio = override.hdpiratio;\n        //var edges = this.getColumnEdges();\n        var startX = override.startX; //hdpiRatio * edges[override.columnIndex];\n        var width = override.width + 1;\n        var height = override.height;\n        var targetCTX = override.ctx;\n        var imgData = gc.getImageData(startX, 0, Math.round(width * hdpiRatio), Math.round(height * hdpiRatio));\n        targetCTX.putImageData(imgData, 0, 0);\n        gc.fillStyle = this.resolveProperty('backgroundColor2');\n        gc.fillRect(Math.round(startX / hdpiRatio), 0, width, height);\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Current vertical scroll value.\n     */\n    getScrollTop: function() {\n        return this.grid.getVScrollValue();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} Current horizontal scroll value.\n     */\n    getScrollLeft: function() {\n        return this.grid.getHScrollValue();\n    },\n\n    /** @deprecated Use `.grid.behavior` property instead.\n     * @memberOf Renderer.prototype\n     * @returns {Behavior} The behavior (model).\n     */\n    getBehavior: function() {\n        return this.deprecated('grid.behavior', { since: '0.2' });\n    },\n\n    getColumnEdges: function() {\n        return this.columnEdges;\n    },\n\n    getRowEdges: function() {\n        return this.rowEdges;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} The last col was rendered (is visible)\n     */\n    isLastColumnVisible: function() {\n        var lastColumnIndex = this.getColumnCount() - 1;\n        return this.visibleColumns.indexOf(lastColumnIndex) !== -1;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The rendered column width at index\n     */\n    getRenderedWidth: function(index) {\n        return this.getColumnEdges()[index];\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The rendered row height at index\n     */\n    getRenderedHeight: function(index) {\n        return this.rowEdges[index];\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {fin-canvas} my [fin-canvas](https://github.com/stevewirts/fin-canvas)\n     */\n    getCanvas: function() {\n        return this.grid.getCanvas();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {boolean} User is currently dragging a column for reordering.\n     */\n    isDraggingColumn: function() {\n        return this.grid.isDraggingColumn();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row to goto for a page up.\n     */\n    getPageUpRow: function() {\n        var grid = this.grid,\n            scrollHeight = this.getVisibleScrollHeight(),\n            headerRows = this.grid.getFixedRowCount(),\n            top = this.dataWindow.origin.y - headerRows,\n            scanHeight = 0;\n        while (scanHeight < scrollHeight && top > -1) {\n            scanHeight = scanHeight + grid.getRowHeight(top);\n            top--;\n        }\n        return top + 1;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The row to goto for a page down.\n     */\n    getPageDownRow: function() {\n        var headerRows = this.grid.getFixedRowCount();\n        var rowNum = this.dataWindow.corner.y - headerRows - 1;\n        return rowNum;\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of columns.\n     */\n    getColumnCount: function() {\n        return this.grid.getColumnCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of rows.\n     */\n    getRowCount: function() {\n        return this.grid.getRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of fixed columns.\n     */\n    getFixedColumnCount: function() {\n        return this.grid.getFixedColumnCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of fixed rows.\n     */\n    getFixedRowCount: function() {\n        return this.grid.getFixedRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of header rows.\n     */\n    getHeaderRowCount: function() {\n        return this.grid.getHeaderRowCount();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @returns {number} The number of header columns.\n     */\n    getHeaderColumnCount: function() {\n        return this.grid.getHeaderColumnCount();\n    },\n\n    /** @summary Smart render the grid.\n     * @desc Paint all the cells of a grid, including all \"fixed\" columns and rows.\n     * We snapshot the context to insure against its pollution.\n     * `try...catch` surrounds each cell paint in case a cell editor throws an error.\n     * The error message is error-logged to console AND displayed in cell.\n     * @memberOf Renderer.prototype\n     * @param {CanvasRenderingContext2D} gc\n     */\n    paintCells: function(gc) {\n        var renderCellError,\n            message,\n            x, y,\n            c, r,\n\n            columnEdges = this.getColumnEdges(),\n            rowEdges = this.rowEdges,\n\n            visibleCols = this.getVisibleColumns(),\n            visibleRows = this.getVisibleRows(),\n\n            behavior = this.grid.behavior,\n\n            clipX = 0,\n            clipY = 0,\n            clipWidth,\n            clipHeight = this.getBounds().height,\n\n            loopStart = this.grid.isShowRowNumbers() ? -1 : 0,\n            loopLength = visibleCols.length; // regardless of loopStart, due to definition of .length\n\n        this.buttonCells = {};\n\n        if (loopLength) { // this if prevents painting just the fixed columns when there are no visible columns\n\n            // For each column...\n            for (x = loopStart; x < loopLength; x++, clipX += clipWidth) {\n\n                c = visibleCols[x];\n                this.renderedColumnMinWidths[c] = 0;\n                renderCellError = behavior.getColumnProperties(c).renderCellError;\n\n                gc.save();\n\n                // Clip to visible portion of column to prevent overflow to right. Previously we clipped to entire visible grid and dealt with overflow by overpainting with next column. However, this strategy fails when transparent background (no background color).\n                // TODO: if extra clip() calls per column affect performance (not the clipping itself which was happening anyway, but the clip calls which set up the clipping), use previous strategy when there is a background color\n                clipWidth = columnEdges[x - loopStart] - clipX;\n                gc.beginPath();\n                gc.rect(clipX, clipY, clipWidth, clipHeight);\n                gc.clip();\n\n                // For each row (of each column)...\n                for (y = 0; y < visibleRows.length; y++) {\n\n                    r = visibleRows[y];\n\n                    try {\n\n                        this._paintCell(gc, c, r);\n\n                        //if (r === 9 && c === 2) { throw Error('She sells sea shells by the sea shore.'); }\n\n                    } catch (e) {\n\n                        message = e && (e.message || e) || 'Unknown error.';\n\n                        console.error(message);\n\n                        if (renderCellError) {\n                            var rawGc = gc.gc || gc, // Don't log these canvas calls\n                                errY = rowEdges[y],\n                                errHeight = rowEdges[y + 1] - errY;\n\n                            rawGc.save(); // define clipping region\n                            rawGc.beginPath();\n                            rawGc.rect(clipX, errY, clipWidth, errHeight);\n                            rawGc.clip();\n\n                            renderCellError(rawGc, message, clipX, errY, clipWidth, errHeight);\n\n                            rawGc.restore(); // discard clipping region\n                        }\n\n                    }\n                }\n\n                // Bottom totals rows...\n                for (y = -behavior.getDataModel().getBottomTotals().length; y; y++) {\n                    this._paintCell(gc, c, y);\n                }\n\n                gc.restore(); // Remove column's clip region (and anything else renderCellError() might have set)\n            }\n        }\n\n        setNumberColumnWidth(gc, behavior, this.grid.getRowCount());\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @desc We opted to not paint borders for each cell as that was extremely expensive. Instead we draw gridlines here. Also we record the widths and heights for later.\n     * @param {CanvasRenderingContext2D} gc\n     */\n    paintGridlines: function(gc) {\n        var x, y, c, r = 0;\n\n        var colWidths = this.getColumnEdges();\n        var rowHeights = this.rowEdges;\n\n        var viewWidth = colWidths[colWidths.length - 1];\n        var viewHeight = this.getBounds().height; //rowHeights[rowHeights.length - 1];\n\n        var drawThemH = this.resolveProperty('gridLinesH');\n        var drawThemV = this.resolveProperty('gridLinesV');\n        var lineColor = this.resolveProperty('lineColor');\n\n        gc.beginPath();\n\n        if (drawThemV) {\n            for (c = 0; c < colWidths.length + 1; c++) {\n                x = colWidths[c] + 0.5;\n                gc.moveTo(x, 0);\n                gc.lineTo(x, viewHeight);\n            }\n        }\n\n        if (drawThemH) {\n            for (r = 0; r < rowHeights.length - 1; r++) {\n                y = rowHeights[r] + 0.5;\n                gc.moveTo(0, y);\n                gc.lineTo(viewWidth, y);\n            }\n\n            // Bottom totals rows...\n            var behavior = this.grid.behavior,\n                rowHeight = behavior.getDefaultRowHeight();\n            for (r = -behavior.getDataModel().getBottomTotals().length, y = this.getBounds().height; r; r++) {\n                y -= rowHeight;\n                gc.moveTo(0, y);\n                gc.lineTo(viewWidth, y);\n            }\n        }\n\n        gc.closePath();\n\n        gc.strokeStyle = lineColor;\n        gc.lineWidth = this.resolveProperty('lineWidth');\n        gc.stroke();\n    },\n\n    /**\n     * @memberOf Renderer.prototype\n     * @param {CanvasRenderingContext2D} gc\n     * @param x\n     * @param y\n     */\n    paintCell: function(gc, x, y) {\n        gc.moveTo(0, 0);\n\n        var c = this.getVisibleColumns()[x],\n            r = this.getVisibleRows()[y];\n\n        if (c) { //something is being viewed at at the moment (otherwise returns undefined)\n            this._paintCell(gc, c, r);\n        }\n    },\n\n    _paintCell: function(gc, c, r) {\n\n        var grid = this.grid,\n            behavior = grid.behavior,\n            baseProperties = behavior.getColumnProperties(c);\n\n        if (baseProperties.isNull) {\n            return;\n        }\n\n        var columnProperties = baseProperties,\n\n            headerRowCount = behavior.getHeaderRowCount(),\n            isGridRow = r >= headerRowCount,\n            isFooterRow = r < 0,\n            isHeaderRow = !isGridRow && !isFooterRow,\n            isFilterRow = grid.isFilterRow(r),\n\n            headerColumnCount = behavior.getHeaderColumnCount(),\n            isGridColumn = c >= headerColumnCount,\n            isShowRowNumbers = grid.isShowRowNumbers(),\n            isHierarchyColumn = grid.isHierarchyColumn(c),\n\n            isRowSelected = grid.isRowSelected(r),\n            isColumnSelected = grid.isColumnSelected(c),\n            isCellSelected = grid.isCellSelected(c, r),\n            isCellSelectedInColumn = grid.isCellSelectedInColumn(c),\n            isCellSelectedInRow = grid.isCellSelectedInRow(r),\n            areAllRowsSelected = grid.areAllRowsSelected(),\n            cellProperties;\n\n        if ((isShowRowNumbers && c === -1) || isHierarchyColumn) {\n            if (isRowSelected) {\n                cellProperties = Object.create(baseProperties.rowHeaderRowSelection);\n                cellProperties.isSelected = true;\n            } else {\n                cellProperties = Object.create(baseProperties.rowHeader);\n                cellProperties.isSelected = isCellSelectedInRow;\n            }\n            cellProperties.isUserDataArea = false;\n        } else if (isHeaderRow || isFooterRow) {\n            if (isFilterRow) {\n                cellProperties = Object.create(baseProperties.filterProperties);\n                cellProperties.isSelected = false;\n            } else if (isColumnSelected) {\n                cellProperties = Object.create(baseProperties.columnHeaderColumnSelection);\n                cellProperties.isSelected = true;\n            } else {\n                cellProperties = Object.create(baseProperties.columnHeader);\n                cellProperties.isSelected = isCellSelectedInColumn;\n            }\n            cellProperties.isUserDataArea = false;\n        } else if (isHierarchyColumn) {\n            cellProperties = Object.create(baseProperties.rowHeader);\n            cellProperties.isSelected = isCellSelectedInRow;\n        } else {\n            cellProperties = Object.create(baseProperties);\n            cellProperties.isSelected = isCellSelected || isRowSelected || isColumnSelected;\n            cellProperties.isUserDataArea = true;\n        }\n\n        var rowNum = r - headerRowCount + 1;\n\n        if (c === -1) {\n            if (r === 0) { // header label row gets \"master\" checkbox\n                cellProperties.value = [images.checkbox(areAllRowsSelected), '', null];\n            } else if (isFilterRow) { // no checkbox but show filter icon\n                cellProperties.value = [images.filter(false), '', null];\n            } else if (isHeaderRow || isFooterRow) { // no checkbox on \"totals\" rows\n                cellProperties.value = '';\n            } else {\n                cellProperties.value = [images.checkbox(isRowSelected), rowNum, null];\n            }\n            cellProperties.halign = 'right';\n        } else {\n            cellProperties.value = grid.getValue(c, r);\n            cellProperties.halign = grid.getColumnAlignment(c);\n        }\n\n        cellProperties.isGridColumn = isGridColumn;\n        cellProperties.isGridRow = isGridRow;\n        cellProperties.isColumnHovered = grid.isColumnHovered(c) && isGridColumn;\n        cellProperties.isRowHovered = grid.isRowHovered(r) && isGridRow;\n        cellProperties.isCellHovered = grid.isHovered(c, r) && isGridColumn && isGridRow;\n        cellProperties.bounds = this._getBoundsOfCell(c, r);\n        cellProperties.isCellSelected = isCellSelected;\n        cellProperties.isRowSelected = isRowSelected;\n        cellProperties.isColumnSelected = isColumnSelected;\n        cellProperties.isInCurrentSelectionRectangle = grid.isInCurrentSelectionRectangle(c, r);\n\n        if (grid.mouseDownState) {\n            var point = grid.mouseDownState.gridCell;\n            cellProperties.mouseDown = point.x === c && point.y === r;\n        }\n\n        cellProperties.x = c;\n        cellProperties.y = r;\n\n        behavior.cellPropertiesPrePaintNotification(cellProperties);\n\n        var cell = behavior.getCellRenderer(cellProperties, c, r);\n        var overrides = behavior.getCellProperties(c, r);\n\n        //declarative cell properties\n        _(cellProperties).extendOwn(overrides);\n\n        //allow the renderer to identify itself if it's a button\n        cellProperties.buttonCells = this.buttonCells;\n        var formatType = cellProperties.isUserDataArea ? cellProperties.format : 'default';\n        cellProperties.formatter = grid.getFormatter(formatType);\n        cell.paint(gc, cellProperties);\n\n        this.renderedColumnMinWidths[c] = Math.max(cellProperties.minWidth || 0, this.renderedColumnMinWidths[c]);\n        columnProperties.preferredWidth = this.renderedColumnMinWidths[c];\n    },\n\n    isViewableButton: function(c, r) {\n        var key = c + ',' + r;\n        return this.buttonCells[key] === true;\n    },\n\n    getRowNumbersWidth: function() {\n        var colEdges = this.getColumnEdges();\n        if (colEdges.length === 0) {\n            return 0;\n        }\n        return colEdges[0];\n    },\n\n    startAnimator: function() {\n        var animate;\n        var self = this;\n        animate = function() {\n            self.animate();\n            requestAnimationFrame(animate);\n        };\n        requestAnimationFrame(animate);\n    },\n\n    animate: function() {\n        var ctx = this.getCanvas().canvasCTX;\n        ctx.beginPath();\n        ctx.save();\n        this.renderFocusCell(ctx);\n        ctx.restore();\n        ctx.closePath();\n    },\n\n    getBounds: function() {\n        return this.bounds;\n    },\n\n    setBounds: function(bounds) {\n        return (this.bounds = bounds);\n    }\n\n});\n\nfunction setNumberColumnWidth(gc, behavior, maxRow) {\n    var columnProperties = behavior.getColumnProperties(-1),\n        cellProperties = columnProperties.rowHeader,\n        icon = images.checked;\n\n    gc.font = cellProperties.font;\n\n    columnProperties.preferredWidth = icon.width + 7 + cellProperties.getTextWidth(gc, maxRow + 1);\n}\n\nmodule.exports = Renderer;\n","'use strict';\n\nvar RangeSelectionModel = require('sparse-boolean-array');\n\n/**\n *\n * @constructor\n * @desc We represent selections as a list of rectangles because large areas can be represented and tested against quickly with a minimal amount of memory usage. Also we need to maintain the selection rectangles flattened counter parts so we can test for single dimension contains. This is how we know to highlight the fixed regions on the edges of the grid.\n */\n\nfunction SelectionModel(grid) {\n\n    this.grid = grid;\n\n    /**\n     * @name selections\n     * @type {Rectangle[]}\n     * @summary The selection rectangles.\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.selections = [];\n\n    /**\n     * @name flattenedX\n     * @type {Rectangle[]}\n     * @summary The selection rectangles flattened in the horizontal direction (no width).\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.flattenedX = [];\n\n    /**\n     * @name flattenedY\n     * @type {Rectangle[]}\n     * @summary The selection rectangles flattened in the vertical direction (no height).\n     * @desc Created as an empty array upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.flattenedY = [];\n\n    /**\n     * @name rowSelectionModel\n     * @type {RangeSelectionModel}\n     * @summary The selection rectangles.\n     * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.rowSelectionModel = new RangeSelectionModel();\n\n    /**\n     * @name columnSelectionModel\n     * @type {RangeSelectionModel}\n     * @summary The selection rectangles.\n     * @desc Created as a new RangeSelectionModel upon instantiation by the {@link SelectionModel|constructor}.\n     * @memberOf SelectionModel.prototype\n     */\n    this.columnSelectionModel = new RangeSelectionModel();\n\n    this.setLastSelectionType('');\n}\n\nSelectionModel.prototype = {\n\n    constructor: SelectionModel.prototype.constructor,\n\n    /**\n     * @type {boolean}\n     * @memberOf SelectionModel.prototype\n     */\n    allRowsSelected: false,\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getLastSelection: function() {\n        var sels = this.selections;\n        var sel = sels[sels.length - 1];\n        return sel;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getLastSelectionType: function() {\n        return this.lastSelectionType;\n    },\n\n    /**\n     * @param type\n     * @memberOf SelectionModel.prototype\n     */\n    setLastSelectionType: function(type) {\n        this.lastSelectionType = type;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @description Select the region described by the given coordinates.\n     *\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     */\n    select: function(ox, oy, ex, ey) {\n        var newSelection = this.grid.newRectangle(ox, oy, ex, ey);\n        this.selections.push(newSelection);\n        this.flattenedX.push(newSelection.flattenXAt(0));\n        this.flattenedY.push(newSelection.flattenYAt(0));\n        this.setLastSelectionType('cell');\n        this.grid.selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     */\n    toggleSelect: function(ox, oy, ex, ey) {\n\n        var selected, index;\n\n        selected = this.selections.find(function(selection, idx) {\n            index = idx;\n            return (\n                selection.origin.x === ox && selection.origin.y === oy &&\n                selection.extent.x === ex && selection.extent.y === ey\n            );\n        });\n\n        if (selected) {\n            this.selections.splice(index, 1);\n            this.flattenedX.splice(index, 1);\n            this.flattenedY.splice(index, 1);\n            this.grid.selectionChanged();\n        } else {\n            this.select(ox, oy, ex, ey);\n        }\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @desc Remove the last selection that was created.\n     */\n    clearMostRecentSelection: function(dontClearRowSelections) {\n        dontClearRowSelections = dontClearRowSelections === true;\n        if (!dontClearRowSelections) {\n            this.setAllRowsSelected(false);\n        }\n        if (this.selections.length) { --this.selections.length; }\n        if (this.flattenedX.length) { --this.flattenedX.length; }\n        if (this.flattenedY.length) { --this.flattenedY.length; }\n        //this.getGrid().selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearMostRecentColumnSelection: function() {\n        this.columnSelectionModel.clearMostRecentSelection();\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearMostRecentRowSelection: function() {\n        this.rowSelectionModel.clearMostRecentSelection();\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    clearRowSelection: function() {\n        this.rowSelectionModel.clear();\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getSelections: function() {\n        return this.selections;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean} There are active selection(s).\n     */\n    hasSelections: function() {\n        return this.selections.length !== 0;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n    hasRowSelections: function() {\n        return !this.rowSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n    hasColumnSelections: function() {\n        return !this.columnSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @return {boolean} Selection covers a specific column.\n     * @param {number} y\n     */\n    isCellSelectedInRow: function(y) {\n        return this._isCellSelected(this.flattenedX, 0, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns Selection covers a specific row.\n     * @param {number} x\n     */\n    isCellSelectedInColumn: function(x) {\n        return this._isCellSelected(this.flattenedY, x, 0);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @summary Selection query function.\n     * @returns {boolean} The given cell is selected (part of an active selection).\n     * @param {Rectangle[]} selections - Selection rectangles to search through.\n     * @param {number} x\n     * @param {number} y\n     */\n    isSelected: function(x, y) {\n        return (\n            this.isColumnSelected(x) ||\n            this.isRowSelected(y) ||\n            this._isCellSelected(this.selections, x, y)\n        );\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @param y\n     * @returns {*}\n     */\n    isCellSelected: function(x, y) {\n        return this._isCellSelected(this.selections, x, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param selections\n     * @param x\n     * @param y\n     * @returns {boolean}\n     * @private\n     */\n    _isCellSelected: function(selections, x, y) {\n        var self = this;\n        return !!selections.find(function(selection) {\n            return self.rectangleContains(selection, x, y);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @desc empty out all our state\n     *\n     */\n    clear: function(dontClearRowSelections) {\n        dontClearRowSelections = dontClearRowSelections === true;\n        this.selections.length = 0;\n        this.flattenedX.length = 0;\n        this.flattenedY.length = 0;\n        this.columnSelectionModel.clear();\n        if (!dontClearRowSelections) {\n            this.setAllRowsSelected(false);\n            this.rowSelectionModel.clear();\n        }\n        //this.getGrid().selectionChanged();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param {number} ox - origin x coordinate\n     * @param {number} oy - origin y coordinate\n     * @param {number} ex - extent x coordinate\n     * @param {number} ey - extent y coordinate\n     * @returns {boolean}\n     */\n    isRectangleSelected: function(ox, oy, ex, ey) {\n        return !!this.selections.find(function(selection) {\n            return (\n                selection.origin.x === ox && selection.origin.y === oy &&\n                selection.extent.x === ex && selection.extent.y === ey\n            );\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @returns {*}\n     */\n    isColumnSelected: function(x) {\n        return this.columnSelectionModel.isSelected(x);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y\n     * @returns {boolean|*}\n     */\n    isRowSelected: function(y) {\n        return this.allRowsSelected || this.rowSelectionModel.isSelected(y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x1\n     * @param x2\n     */\n    selectColumn: function(x1, x2) {\n        this.columnSelectionModel.select(x1, x2);\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     */\n    selectAllRows: function() {\n        this.clear();\n        this.setAllRowsSelected(true);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n\n    setAllRowsSelected: function(isIt) {\n        this.allRowsSelected = isIt;\n    },\n\n    areAllRowsSelected: function() {\n        return this.allRowsSelected;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y1\n     * @param y2\n     */\n    selectRow: function(y1, y2) {\n        this.rowSelectionModel.select(y1, y2);\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x1\n     * @param x2\n     */\n    deselectColumn: function(x1, x2) {\n        this.columnSelectionModel.deselect(x1, x2);\n        this.setLastSelectionType('column');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param y1\n     * @param y2\n     */\n    deselectRow: function(y1, y2) {\n        this.rowSelectionModel.deselect(y1, y2);\n        this.setLastSelectionType('row');\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*}\n     */\n    getSelectedRows: function() {\n        if (this.areAllRowsSelected()) {\n            var headerRows = this.grid.getHeaderRowCount();\n            var rowCount = this.grid.getRowCount() - headerRows;\n            var result = new Array(rowCount);\n            for (var i = 0; i < rowCount; i++) {\n                result[i] = i + headerRows;\n            }\n            return result;\n        }\n        return this.rowSelectionModel.getSelections();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {*|Array.Array.number}\n     */\n    getSelectedColumns: function() {\n        return this.columnSelectionModel.getSelections();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {boolean}\n     */\n     isColumnOrRowSelected: function() {\n        return !this.columnSelectionModel.isEmpty() || !this.rowSelectionModel.isEmpty();\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @returns {Array}\n     */\n    getFlattenedYs: function() {\n        var result = [];\n        var set = {};\n        this.selections.forEach(function(selection) {\n            var top = selection.origin.y;\n            var size = selection.extent.y + 1;\n            for (var r = 0; r < size; r++) {\n                var ti = r + top;\n                if (!set[ti]) {\n                    result.push(ti);\n                    set[ti] = true;\n                }\n            }\n        });\n        result.sort(function(x, y) {\n            return x - y;\n        });\n        return result;\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param offset\n     */\n    selectRowsFromCells: function(offset, dontClearRowSelections) {\n        offset = offset || 0;\n        dontClearRowSelections = dontClearRowSelections === true;\n\n        var sm = this.rowSelectionModel;\n\n        if (!dontClearRowSelections) {\n            this.setAllRowsSelected(false);\n            sm.clear();\n        }\n\n        this.selections.forEach(function(selection) {\n            var top = selection.origin.y,\n                extent = selection.extent.y;\n            top += offset;\n            sm.select(top, top + extent);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param offset\n     */\n    selectColumnsFromCells: function(offset) {\n        offset = offset || 0;\n\n        var sm = this.columnSelectionModel;\n        sm.clear();\n\n        this.selections.forEach(function(selection) {\n            var left = selection.origin.x,\n                extent = selection.extent.x;\n            left += offset;\n            sm.select(left, left + extent);\n        });\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param x\n     * @param y\n     * @returns {*}\n     */\n    isInCurrentSelectionRectangle: function(x, y) {\n        var last = this.selections[this.selections.length - 1];\n        return last && this.rectangleContains(last, x, y);\n    },\n\n    /**\n     * @memberOf SelectionModel.prototype\n     * @param rect\n     * @param x\n     * @param y\n     * @returns {boolean}\n     */\n    rectangleContains: function(rect, x, y) { //TODO: explore why this works and contains on rectanglular does not\n        var minX = rect.origin.x;\n        var minY = rect.origin.y;\n        var maxX = minX + rect.extent.x;\n        var maxY = minY + rect.extent.y;\n\n        if (rect.extent.x < 0) {\n            minX = maxX;\n            maxX = rect.origin.x;\n        }\n\n        if (rect.extent.y < 0) {\n            minY = maxY;\n            maxY = rect.origin.y;\n        }\n\n        var result =\n            x >= minX &&\n            y >= minY &&\n            x <= maxX &&\n            y <= maxY;\n\n        return result;\n    }\n};\n\nmodule.exports = SelectionModel;\n","/* eslint-env browser */\n\n'use strict';\n\nvar Base = require('./Base');\n\n\nvar ANIMATION_TIME = 500,\n    TRANSITION = ANIMATION_TIME + 'ms ease-in';\n\n/** @constructor\n * @desc Instances of features are connected to one another to make a chain of responsibility for handling all the input to the hypergrid.\n *\n * See {@link TableDialog#initialize|initialize} which is called by the constructor.\n */\nvar TableDialog = Base.extend('TableDialog', {\n\n    initialize: function(grid) {\n        this.grid = grid;\n        this.initializeOverlaySurface();\n        this.openNow = false;\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc returns true if the overlay is open\n     * @returns {boolean}\n     * @param {Hypergrid} grid\n     */\n    isOpen: function() {\n        return this.openNow;\n    },\n\n    open: function() {\n        if (this.isOpen()) {\n            return;\n        }\n\n        this.openNow = true;\n        var self = this;\n        this.overlay.style.backgroundColor = this.grid.resolveProperty('backgroundColor');\n\n        this.overlay.style.top = this.overlay.style.bottom = this.overlay.style.right = this.overlay.style.left = 0;\n\n        self.overlay.style.webkitTransition = '';\n\n        this.overlay.style.margin = '15px 35px 35px 15px';\n        this.overlay.style.opacity = 0;\n        this.overlay.style.zIndex = 100;\n\n        this.closeTransition = function() {\n            this.overlay.style.opacity = 0;\n        };\n\n        if (!this._closer) {\n            this._closer = function(e) {\n                var key = self.getCharFor(e.keyCode).toLowerCase();\n                var keys = self.grid.resolveProperty('editorActivationKeys');\n                if (keys.indexOf(key) > -1 || e.keyCode === 27) {\n                    e.preventDefault();\n                    self.close();\n                }\n            };\n        }\n        requestAnimationFrame(function() {\n            self.overlay.style.webkitTransition = 'opacity ' + ANIMATION_TIME + 'ms ease-in';\n            requestAnimationFrame(function() {\n                document.addEventListener('keydown', self._closer, false);\n                self.overlay.style.opacity = 0.95;\n            });\n        });\n\n        setTimeout(function() {\n            self.overlay.focus();\n        }, 100);\n    },\n    /**\n     * @memberOf Overlay.prototype\n     * @desc open the overlay\n     * #### returns: type\n     * @param {Hypergrid} grid\n     */\n    openFrom: function(rect) {\n        if (this.isOpen()) {\n            return;\n        }\n        this.openNow = true;\n        var self = this;\n        var style = this.overlay.style;\n        style.backgroundColor = this.grid.resolveProperty('backgroundColor');\n\n        var bounds = this.grid.div.getBoundingClientRect(),\n            margins = rect.y + 'px ' +\n                (bounds.width - (rect.x + rect.width)) + 'px ' +\n                (bounds.height - (rect.y + rect.height)) + 'px ' +\n                rect.x + 'px';\n\n        style.webkitTransition = '';\n\n        style.top = style.right = style.bottom = style.left = 0;\n\n        style.margin = margins;\n        style.zIndex = 100;\n        style.opacity = 1;\n\n        this.closeTransition = function() {\n            style.margin = margins;\n        };\n\n        if (!this._closer) {\n            this._closer = function(e) {\n                var key = self.getCharFor(e.keyCode).toLowerCase();\n                var keys = self.grid.resolveProperty('editorActivationKeys');\n                if (keys.indexOf(key) > -1 || e.keyCode === 27) {\n                    e.preventDefault();\n                    self.close();\n                }\n            };\n        }\n\n        //grid.setFocusable(false);\n        requestAnimationFrame(function() {\n            document.addEventListener('keydown', self._closer, false);\n            requestAnimationFrame(function() {\n                requestAnimationFrame(function() {\n                    style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION;\n                    style.margin = '15px 35px 35px 15px';\n                });\n            });\n        });\n        setTimeout(function() {\n            self.overlay.focus();\n        }, 100);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc close the overlay\n     * @param {Hypergrid} grid\n     */\n    close: function() {\n        //grid.setFocusable(true);\n        this.openNow = false;\n        document.removeEventListener('keydown', this._closer, false);\n\n        var self = this;\n\n        requestAnimationFrame(function() {\n            self.closeTransition();\n        });\n\n        setTimeout(function() {\n            self.clear();\n            self.overlay.style.zIndex = -1000;\n            if (self.onClose) {\n                self.onClose();\n                self.onClose = undefined;\n            }\n            self.grid.takeFocus();\n        }, ANIMATION_TIME);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc initialize the overlay surface into the grid\n     * #### returns: type\n     * @param {Hypergrid} grid\n     */\n    initializeOverlaySurface: function() {\n        this.overlay = document.createElement('div');\n        this.overlay.setAttribute('tabindex', 0);\n        this.overlay.addEventListener('wheel', function(evt) { evt.stopPropagation(); });\n\n        var style = this.overlay.style;\n        style.outline = 'none';\n        style.boxShadow = '0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)';\n        style.position = 'absolute';\n\n        style.margin = 0;\n        style.overflow = 'hidden';\n\n        //style.display = 'none';\n\n        //style.webkitTransition = 'margin-top ' + TRANSITION + ', margin-right ' + TRANSITION + ', margin-bottom ' + TRANSITION + ', margin-left ' + TRANSITION;\n\n        style.opacity = 0;\n        style.zIndex = 10;\n        this.grid.div.appendChild(this.overlay);\n        //document.body.appendChild(this.overlay);\n    },\n\n    /**\n     * @memberOf Overlay.prototype\n     * @desc get a human readable description of the key pressed from it's integer representation\n     * @returns {string}\n     * @param {Hypergrid} grid\n     * @param {number} integer - the integer we want the char for\n     */\n    getCharFor: function(integer) {\n        var charMap = this.grid.getCanvas().getCharMap();\n        return charMap[integer][0];\n    },\n\n    clear: function() {\n        this.overlay.innerHTML = '';\n    },\n\n    querySelector: function(selector) {\n        var elements = this.overlay.querySelector(selector);\n        return elements;\n    },\n\n    getAnimationTime: function() {\n        return ANIMATION_TIME;\n    }\n});\n\nmodule.exports = TableDialog;\n","'use strict';\n\n// console.warn polyfill as needed\n// used for deprecation warnings\nif (!console.warn) {\n    console.warn = function() {\n        console.log.apply(console, ['WARNING:'].concat(Array.prototype.slice.call(arguments)));\n    };\n}\n\nvar deprecated = function(dotProps, options) {\n    var chain = dotProps.split('.'),\n        method = chain[chain.length - 1],\n        asOfVersion = options && options.asOfVersion,\n        result = this,\n        warning;\n\n    method = 'get' + method[0].toUpperCase() + method.substr(1);\n\n    warning = '.' + method + '() method is deprecated';\n\n    if (asOfVersion) {\n        warning += ' as of v' + options.asOfVersion;\n    }\n\n    warning += '. Use .' + dotProps;\n\n    if (dotProps[dotProps.length - 1] !== ')') {\n        warning += ' property';\n    }\n\n    warning += ' instead. (Will be removed in a future release.)';\n\n    console.warn(warning);\n\n    chain.forEach(function(link) {\n        result = result[link];\n    });\n\n    return result;\n};\n\nmodule.exports = deprecated;\n","'use strict';\n\nvar images = require('../../images/index');\n\n/**\n * @summary Writes error message into cell.\n *\n * @desc This funciton is guaranteed to be called as follows:\n *\n * ```javascript\n * gc.save();\n * gc.beginPath();\n * gc.rect(x, y, width, height);\n * gc.clip();\n * renderCellError(gc, message, x, y, width, height);\n * gc.restore();\n * ```\n *\n * Before doing anything else, this function should clear the cell by setting `gc.fillStyle` and calling `gc.fill()`.\n *\n * @param {CanvasRenderingContext2D} gc\n * @param {string} message\n * @param {number} x\n * @param {number} y\n * @param {number} width\n * @param {number} height\n */\nfunction renderCellError(gc, message, x, y, width, height) {\n\n    // clear the cell\n    // (this makes use of the rect path defined by the caller)\n    gc.fillStyle = '#FFD500';\n    gc.fill();\n\n    // render cell border\n    gc.strokeStyle = gc.createPattern(images.caution, 'repeat');\n    gc.lineWidth = 5;\n    gc.beginPath();\n    gc.moveTo(x, y); // caution: do not use rect() here because Chrome does not clip its stroke properly\n    gc.lineTo(x + width, y);\n    gc.lineTo(x + width, y + height);\n    gc.lineTo(x, y + height);\n    gc.lineTo(x, y);\n    gc.stroke();\n\n    // adjust clip region to prevent text from rendering over right border should it overflow\n    gc.beginPath();\n    gc.rect(x, y, width - 2, height);\n    gc.clip();\n\n    // render message text\n    gc.fillStyle = '#A00';\n    gc.textAlign = 'start';\n    gc.textBaseline = 'middle';\n    gc.font = 'bold 6pt \"arial narrow\", verdana, geneva';\n    gc.fillText(message, x + 4, y + height / 2 + 0.5);\n\n}\n\nmodule.exports = renderCellError;\n","'use strict';\n\nmodule.exports = (function() {\n\n    var depthString = '                                                                                ';\n\n    function DataNodeBase(key) {\n        this.label = key;\n        this.data = [''];\n        this.rowIndexes = [];\n        this.hasChildren = false;\n        this.depth = 0;\n        this.height = 1;\n        this.expanded = false;\n    }\n\n    DataNodeBase.prototype.isNullObject = false;\n\n    DataNodeBase.prototype.getValue = function(x) {\n        return this.data[x];\n    };\n\n    DataNodeBase.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeBase.prototype.computeDepthString = function() {\n        var string = depthString.substring(0, 2 + (this.depth * 3)) + this.label;\n        return string;\n    };\n\n    DataNodeBase.prototype.computeHeight = function() {\n        return 1;\n    };\n\n    DataNodeBase.prototype.getAllRowIndexes = function() {\n        return this.rowIndexes;\n    };\n\n    DataNodeBase.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n    };\n\n    DataNodeBase.prototype.applyAggregates = function(aggregator) {\n        var hasGroupsOffset = aggregator.hasGroups() ? 1 : 0;\n        var indexes = this.getAllRowIndexes();\n        if (indexes.length === 0) {\n            return; // no data to rollup on\n        }\n        var aggregates = aggregator.aggregates;\n        var data = this.data;\n        data.length = aggregates.length + hasGroupsOffset;\n\n        var sorter = aggregator.sorterInstance;\n        sorter.indexes = indexes;\n\n        for (var i = 0; i < aggregates.length; i++) {\n            var aggregate = aggregates[i];\n            data[i + hasGroupsOffset] = aggregate(sorter);\n        }\n\n        this.data = data;\n    };\n\n    DataNodeBase.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n    };\n\n    DataNodeBase.prototype.toggleExpansionState = function() { /* aggregator */\n        //do nothing by default\n    };\n\n    return DataNodeBase;\n\n})();\n","'use strict';\n\nvar Map = require('./Map');\nvar DataNodeBase = require('./DataNodeBase');\n\nmodule.exports = (function() {\n\n    var ExpandedMap = {\n        true: '▾',\n        false: '▸'\n    };\n    var depthString = '                                                                                ';\n\n    function DataNodeGroup(key) {\n        DataNodeBase.call(this, key);\n        this.children = new Map();\n    }\n\n    DataNodeGroup.prototype = Object.create(DataNodeBase.prototype);\n\n    DataNodeGroup.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.children = this.children.values;\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.prune(this.depth + 1);\n        }\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeGroup.prototype.computeDepthString = function() {\n        var icon = ExpandedMap[this.expanded + ''];\n        var string = depthString.substring(0, this.depth * 3) + icon + ' ' + this.label;\n        return string;\n    };\n\n    DataNodeGroup.prototype.getAllRowIndexes = function() {\n        if (this.rowIndexes.length === 0) {\n            this.rowIndexes = this.computeAllRowIndexes();\n        }\n        return this.rowIndexes;\n    };\n\n    DataNodeGroup.prototype.computeAllRowIndexes = function() {\n        var result = [];\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            var childIndexes = child.getAllRowIndexes();\n            Array.prototype.splice.apply(result, [result.length, 0].concat(childIndexes));\n        }\n        return result;\n    };\n\n    DataNodeGroup.prototype.toggleExpansionState = function(aggregator) { /* aggregator */\n        this.expanded = !this.expanded;\n        this.data[0] = this.computeDepthString();\n        if (this.expanded) {\n            this.computeAggregates(aggregator);\n        }\n    };\n\n    DataNodeGroup.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n        if (!this.expanded) {\n            return; // were not being viewed, don't have child nodes do computation;\n        }\n        for (var i = 0; i < this.children.length; i++) {\n            this.children[i].computeAggregates(aggregator);\n        }\n    };\n\n    DataNodeGroup.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n        if (this.expanded) {\n            for (var i = 0; i < this.children.length; i++) {\n                var child = this.children[i];\n                child.buildView(aggregator);\n            }\n        }\n    };\n\n    DataNodeGroup.prototype.computeHeight = function() {\n        var height = 1; //I'm 1 high\n        if (!this.expanded) {\n            this.height = 1;\n        } else {\n            for (var i = 0; i < this.children.length; i++) {\n                height = height + this.children[i].computeHeight();\n            }\n            this.height = height;\n        }\n        return this.height;\n    };\n\n    return DataNodeGroup;\n\n})();\n","'use strict';\n\nvar DataNodeBase = require('./DataNodeBase');\n\nmodule.exports = (function() {\n\n    function DataNodeLeaf(key) {\n        DataNodeBase.call(this, key);\n    }\n\n    DataNodeLeaf.prototype = Object.create(DataNodeBase.prototype);\n\n    DataNodeLeaf.prototype.prune = function(depth) {\n        this.depth = depth;\n        this.data[0] = this.computeDepthString();\n    };\n\n    DataNodeLeaf.prototype.computeHeight = function() {\n        return 1;\n    };\n\n    DataNodeLeaf.prototype.getAllRowIndexes = function() {\n        return this.rowIndexes;\n    };\n\n    DataNodeLeaf.prototype.computeAggregates = function(aggregator) {\n        this.applyAggregates(aggregator);\n    };\n\n    DataNodeLeaf.prototype.buildView = function(aggregator) {\n        aggregator.view.push(this);\n    };\n\n    return DataNodeLeaf;\n\n})();\n","'use strict';\n\nvar DataNodeGroup = require('./DataNodeGroup');\n\nmodule.exports = (function() {\n\n    function DataNodeTree(key) {\n        DataNodeGroup.call(this, key);\n        this.height = 0;\n        this.expanded = true;\n    }\n\n    DataNodeTree.prototype = Object.create(DataNodeGroup.prototype);\n\n    DataNodeTree.prototype.prune = function() {\n        this.children = this.children.values;\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.prune(0);\n        }\n    };\n\n    DataNodeTree.prototype.buildView = function(aggregator) {\n        for (var i = 0; i < this.children.length; i++) {\n            var child = this.children[i];\n            child.buildView(aggregator);\n        }\n    };\n\n    DataNodeTree.prototype.computeHeight = function() {\n        var height = 1;\n        for (var i = 0; i < this.children.length; i++) {\n            height = height + this.children[i].computeHeight();\n        }\n        this.height = height;\n\n        return this.height;\n    };\n\n\n    return DataNodeTree;\n\n})();\n","'use strict';\n\nvar DataSourceSorter = require('./DataSourceSorter');\nvar DataNodeTree = require('./DataNodeTree');\nvar DataNodeGroup = require('./DataNodeGroup');\nvar DataNodeLeaf = require('./DataNodeLeaf');\n\nmodule.exports = (function() {\n\n    var headerify = function(string) {\n        var pieces = string.replace(/[_-]/g, ' ').replace(/[A-Z]/g, ' $&').split(' ').map(function(s) {\n            return (s.charAt(0).toUpperCase() + s.slice(1)).trim();\n        });\n        pieces = pieces.filter(function(e) {\n            return e.length !== 0;\n        });\n        return pieces.join(' ').trim();\n    };\n\n    //?[t,c,b,a]\n    // t is a dataSource,\n    // a is a dicitionary of aggregates,  columnName:function\n    // b is a dicitionary of groupbys, columnName:sourceColumnName\n    // c is a list of constraints,\n\n    function DataSourceAggregator(dataSource) {\n        this.tree = new DataNodeTree('Totals');\n        this.indexes = [];\n        this.dataSource = dataSource;\n        this.aggregates = [];\n        this.headers = [];\n        this.groupBys = [];\n        this.view = [];\n        this.sorterInstance = {};\n        this.presortGroups = true;\n        this.lastAggregate = {};\n        this.setAggregates({});\n    }\n\n    DataSourceAggregator.prototype.isNullObject = false;\n\n    DataSourceAggregator.prototype.setAggregates = function(aggregations) {\n        this.lastAggregate = aggregations;\n        var props = [];\n        var i;\n        this.clearAggregations();\n        this.headers.length = 0;\n\n        for (var key in aggregations) {\n            props.push([key, aggregations[key]]);\n        }\n\n        // if (props.length === 0) {\n        //     var fields = [].concat(this.dataSource.getFields());\n        //     for (i = 0; i < fields.length; i++) {\n        //         props.push([fields[i], Aggregations.first(i)]); /* jshint ignore:line */\n        //     }\n        // }\n        if (this.hasGroups()) {\n            this.headers.push('Tree');\n        }\n\n        for (i = 0; i < props.length; i++) {\n            var agg = props[i];\n            this.addAggregate(agg[0], agg[1]);\n        }\n    };\n\n    DataSourceAggregator.prototype.addAggregate = function(label, func) {\n        this.headers.push(headerify(label));\n        this.aggregates.push(func);\n    };\n\n    DataSourceAggregator.prototype.setGroupBys = function(columnIndexArray) {\n        this.groupBys.length = 0;\n        for (var i = 0; i < columnIndexArray.length; i++) {\n            this.groupBys.push(columnIndexArray[i]);\n        }\n        this.setAggregates(this.lastAggregate);\n    };\n\n    DataSourceAggregator.prototype.addGroupBy = function(index) {\n        this.groupBys.push(index);\n    };\n\n    DataSourceAggregator.prototype.hasGroups = function() {\n        return this.groupBys.length > 0;\n    };\n\n    DataSourceAggregator.prototype.hasAggregates = function() {\n        return this.aggregates.length > 0;\n    };\n\n    DataSourceAggregator.prototype.apply = function() {\n        this.buildGroupTree();\n    };\n\n    DataSourceAggregator.prototype.clearGroups = function() {\n        this.groupBys.length = 0;\n    };\n\n    DataSourceAggregator.prototype.clearAggregations = function() {\n        this.aggregates.length = 0;\n        this.headers.length = 0;\n    };\n\n    DataSourceAggregator.prototype.buildGroupTree = function() {\n        var c, r, g, value, createFunc;\n        var createBranch = function(key, map) {\n            value = new DataNodeGroup(key);\n            map.set(key, value);\n            return value;\n        };\n        var createLeaf = function(key, map) {\n            value = new DataNodeLeaf(key);\n            map.set(key, value);\n            return value;\n        };\n        var groupBys = this.groupBys;\n        var source = this.dataSource;\n        var rowCount = source.getRowCount();\n\n        // lets sort our data first....\n        if (this.presortGroups) {\n            for (c = 0; c < groupBys.length; c++) {\n                g = groupBys[groupBys.length - c - 1];\n                source = new DataSourceSorter(source);\n                source.sortOn(g);\n            }\n        }\n\n        var tree = this.tree = new DataNodeTree('Totals');\n        var path = tree;\n        var leafDepth = groupBys.length - 1;\n        for (r = 0; r < rowCount; r++) {\n            for (c = 0; c < groupBys.length; c++) {\n                g = groupBys[c];\n                value = source.getValue(g, r);\n\n                //test that I'm not a leaf\n                createFunc = (c === leafDepth) ? createLeaf : createBranch;\n                path = path.children.getIfAbsent(value, createFunc);\n            }\n            path.rowIndexes.push(r);\n            path = tree;\n        }\n        this.sorterInstance = new DataSourceSorter(source);\n        tree.prune();\n        this.tree.computeAggregates(this);\n        this.buildView();\n    };\n\n    DataSourceAggregator.prototype.buildView = function() {\n        this.view.length = 0;\n        this.tree.computeHeight();\n        this.tree.buildView(this);\n    };\n\n    DataSourceAggregator.prototype.viewMakesSense = function() {\n        return this.hasAggregates();\n    };\n\n    DataSourceAggregator.prototype.getValue = function(x, y) {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getValue(x, y);\n        }\n        var row = this.view[y];\n        if (!row) {\n            return null;\n        }\n        return row.getValue(x);\n    };\n\n    DataSourceAggregator.prototype.setValue = function(x, y, value) {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.setValue(x, y, value);\n        }\n    };\n\n    DataSourceAggregator.prototype.getColumnCount = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getColumnCount();\n        }\n        var colCount = this.getHeaders().length;\n        return colCount;\n    };\n\n    DataSourceAggregator.prototype.getRowCount = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getRowCount();\n        }\n        return this.view.length; //header column\n    };\n\n    DataSourceAggregator.prototype.click = function(y) {\n        var group = this.view[y];\n        group.toggleExpansionState(this);\n        this.buildView();\n    };\n\n    DataSourceAggregator.prototype.getHeaders = function() {\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getHeaders();\n        }\n        return this.headers;\n    };\n\n    DataSourceAggregator.prototype.setHeaders = function(headers) {\n        this.dataSource.setHeaders(headers);\n    };\n\n    DataSourceAggregator.prototype.getFields = function() {\n        return this.dataSource.getFields();\n    };\n\n    DataSourceAggregator.prototype.setFields = function(fields) {\n        return this.dataSource.setFields(fields);\n    };\n\n    DataSourceAggregator.prototype.getGrandTotals = function() {\n        var view = this.tree;\n        return [view.data];\n    };\n\n    DataSourceAggregator.prototype.getRow = function(y) {\n\n        if (!this.viewMakesSense()) {\n            return this.dataSource.getRow(y);\n        }\n\n        var rollups = this.view[y];\n        if (!rollups) {\n            return this.tree;\n        }\n\n        return rollups;\n    };\n\n    DataSourceAggregator.prototype.setData = function(arrayOfUniformObjects) {\n        this.dataSource.setData(arrayOfUniformObjects);\n        this.apply();\n    };\n\n    return DataSourceAggregator;\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    function DataSourceDecorator(dataSource) {\n        this.dataSource = dataSource;\n        this.indexes = [];\n    }\n\n    DataSourceDecorator.prototype.isNullObject = false;\n\n    DataSourceDecorator.prototype.transposeY = function(y) {\n        if (this.indexes.length !== 0) {\n            return this.indexes[y];\n        }\n        return y;\n    };\n\n    DataSourceDecorator.prototype.getValue = function(x, y) {\n        var value = this.dataSource.getValue(x, this.transposeY(y));\n        return value;\n    };\n\n    DataSourceDecorator.prototype.getRow = function(y) {\n\n        return this.dataSource.getRow(this.transposeY(y));\n    };\n\n    DataSourceDecorator.prototype.setValue = function(x, y, value) {\n\n        this.dataSource.setValue(x, this.transposeY(y), value);\n    };\n\n    DataSourceDecorator.prototype.getColumnCount = function() {\n\n        return this.dataSource.getColumnCount();\n    };\n\n    DataSourceDecorator.prototype.getFields = function() {\n\n        return this.dataSource.getFields();\n    };\n\n    DataSourceDecorator.prototype.setFields = function(fields) {\n\n        return this.dataSource.setFields(fields);\n    };\n\n    DataSourceDecorator.prototype.click = function(y) {\n\n        return this.dataSource.click(this.transposeY(y));\n    };\n\n    DataSourceDecorator.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    DataSourceDecorator.prototype.setHeaders = function(headers) {\n        return this.dataSource.setHeaders(headers);\n    };\n\n    DataSourceDecorator.prototype.getHeaders = function() {\n\n        return this.dataSource.getHeaders();\n    };\n\n    DataSourceDecorator.prototype.getGrandTotals = function() {\n        return this.dataSource.getGrandTotals();\n    };\n\n    DataSourceDecorator.prototype.initializeIndexVector = function() {\n        var rowCount = this.dataSource.getRowCount();\n        var indexVector = new Array(rowCount);\n        for (var r = 0; r < rowCount; r++) {\n            indexVector[r] = r;\n        }\n        this.indexes = indexVector;\n    };\n\n    DataSourceDecorator.prototype.setData = function(arrayOfUniformObjects) {\n        this.dataSource.setData(arrayOfUniformObjects);\n    };\n\n    return DataSourceDecorator;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\n\nmodule.exports = (function() {\n\n    function DataSourceFilter(dataSource) {\n        DataSourceDecorator.call(this, dataSource, false);\n        this.filters = [];\n    }\n\n    DataSourceFilter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceFilter.prototype.add = function(columnIndex, filter) {\n        filter.columnIndex = columnIndex;\n        this.filters.push(filter);\n    };\n    DataSourceFilter.prototype.set = function(columnIndex, filter) {\n        filter.columnIndex = columnIndex;\n        this.filters.push(filter);\n    };\n\n    DataSourceFilter.prototype.clearAll = function() { /* filter */\n        this.filters.length = 0;\n        this.indexes.length = 0;\n    };\n\n    DataSourceFilter.prototype.applyAll = function() {\n        if (this.filters.length === 0) {\n            this.indexes.length = 0;\n            return;\n        }\n        var indexes = this.indexes;\n        indexes.length = 0;\n        var count = this.dataSource.getRowCount();\n        for (var r = 0; r < count; r++) {\n            if (this.applyFiltersTo(r)) {\n                indexes.push(r);\n            }\n        }\n    };\n\n    DataSourceFilter.prototype.applyFiltersTo = function(r) {\n        var filters = this.filters;\n        var isFiltered = true;\n        for (var f = 0; f < filters.length; f++) {\n            var filter = filters[f];\n            var rowObject = this.dataSource.getRow(r);\n            isFiltered = isFiltered && filter(this.dataSource.getValue(filter.columnIndex, r), rowObject, r);\n        }\n        return isFiltered;\n    };\n\n    DataSourceFilter.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        //our filter matched nothing....\n        if (this.filters.length !== 0) {\n            return 0;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    return DataSourceFilter;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\n\nmodule.exports = (function() {\n\n    function DataSourceGlobalFilter(dataSource) {\n        DataSourceDecorator.call(this, dataSource, false);\n        this.filter = null;\n    }\n\n    DataSourceGlobalFilter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceGlobalFilter.prototype.set = function(filter) {\n        this.filter = filter;\n    };\n\n    DataSourceGlobalFilter.prototype.clear = function() { /* filter */\n        this.filter = null;\n        this.indexes.length = 0;\n    };\n\n    DataSourceGlobalFilter.prototype.getRowCount = function() {\n        if (this.indexes.length !== 0) {\n            return this.indexes.length;\n        }\n        //our filter matched nothing....\n        if (this.filter) {\n            return 0;\n        }\n        return this.dataSource.getRowCount();\n    };\n\n    DataSourceGlobalFilter.prototype.apply = function(visibleColumns) {\n        if (!this.filter) {\n            this.indexes.length = 0;\n            return;\n        }\n        var visibleColumnMap = this.visibleColumnMap = [];\n        visibleColumns.forEach(function(column) {\n            visibleColumnMap.push(column.index);\n        });\n        var indexes = this.indexes;\n        indexes.length = 0;\n        var count = this.dataSource.getRowCount();\n        for (var r = 0; r < count; r++) {\n            if (this.applyFilterTo(r)) {\n                indexes.push(r);\n            }\n        }\n    };\n\n    DataSourceGlobalFilter.prototype.applyFilterTo = function(r) {\n        var isFiltered = false;\n        var filter = this.filter;\n        var visColCount = this.visibleColumnMap.length;\n        var rowObject = this.dataSource.getRow(r);\n        for (var v = 0; v < visColCount; v++) {\n            var i = this.visibleColumnMap[v];\n            isFiltered = isFiltered || filter(this.dataSource.getValue(i, r), rowObject, r);\n            if (isFiltered) {\n                return true;\n            }\n        }\n        return false;\n    };\n\n    return DataSourceGlobalFilter;\n\n})();\n","'use strict';\n\nvar Utils = require('./Utils.js');\nvar DataSourceDecorator = require('./DataSourceDecorator');\nvar valueOrFunctionExecute = function(valueOrFunction) {\n    var isFunction = (((typeof valueOrFunction)[0]) === 'f');\n    var result = isFunction ? valueOrFunction() : valueOrFunction;\n    return result;\n};\n\nmodule.exports = (function() {\n\n    function DataSourceSorter(dataSource) {\n        DataSourceDecorator.call(this, dataSource);\n        this.descendingSort = false;\n    }\n\n    DataSourceSorter.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceSorter.prototype.sortOn = function(columnIndex, sortType) {\n        if (sortType === 0) {\n            this.indexes.length = 0;\n            return;\n        }\n        this.initializeIndexVector();\n        var self = this;\n        Utils.stableSort(this.indexes, function(index) {\n            var val = self.dataSource.getValue(columnIndex, index);\n            val = valueOrFunctionExecute(val);\n            return val;\n        }, sortType);\n    };\n\n    return DataSourceSorter;\n\n})();\n","'use strict';\n\nvar DataSourceDecorator = require('./DataSourceDecorator');\nvar DataSourceSorter = require('./DataSourceSorter');\n\nmodule.exports = (function() {\n\n    function DataSourceSorterComposite(dataSource) {\n        DataSourceDecorator.call(this, dataSource);\n        this.sorts = [];\n        this.last = this.dataSource;\n    }\n\n    DataSourceSorterComposite.prototype = Object.create(DataSourceDecorator.prototype);\n\n    DataSourceSorterComposite.prototype.getRow = function(y) {\n        return this.last.getRow(y);\n    };\n\n    DataSourceSorterComposite.prototype.sortOn = function(columnIndex, sortType) {\n        this.sorts.push([columnIndex, sortType]);\n    };\n\n    DataSourceSorterComposite.prototype.applySorts = function() {\n        var sorts = this.sorts;\n        var each = this.dataSource;\n        for (var i = 0; i < sorts.length; i++) {\n            var sort = sorts[i];\n            each = new DataSourceSorter(each);\n            each.sortOn(sort[0], sort[1]);\n        }\n        this.last = each;\n    };\n\n    DataSourceSorterComposite.prototype.clearSorts = function() {\n        this.sorts.length = 0;\n        this.last = this.dataSource;\n    };\n\n    DataSourceSorterComposite.prototype.getValue = function(x, y) {\n        return this.last.getValue(x, y);\n    };\n\n    DataSourceSorterComposite.prototype.setValue = function(x, y, value) {\n        this.last.setValue(x, y, value);\n    };\n\n    return DataSourceSorterComposite;\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    var headerify = function(string) {\n        var pieces = string.replace(/[_-]/g, ' ').replace(/[A-Z]/g, ' $&').split(' ').map(function(s) {\n            return s.charAt(0).toUpperCase() + s.slice(1);\n        });\n        return pieces.join(' ');\n    };\n\n    var computeFieldNames = function(object) {\n        if (!object) {\n            return [];\n        }\n        var fields = [].concat(Object.getOwnPropertyNames(object).filter(function(e) {\n            return e.substr(0, 2) !== '__';\n        }));\n        return fields;\n    };\n\n    function JSDataSource(data, fields) {\n        this.fields = fields || computeFieldNames(data[0]);\n        this.headers = [];\n        this.data = data;\n    }\n\n    JSDataSource.prototype.isNullObject = false;\n\n    JSDataSource.prototype.getValue = function(x, y) {\n        var row = this.data[y];\n        if (!row) {\n            return null;\n        }\n        var value = row[this.fields[x]];\n        return value;\n    };\n\n    JSDataSource.prototype.getRow = function(y) {\n\n        return this.data[y];\n    };\n\n    JSDataSource.prototype.setValue = function(x, y, value) {\n\n        this.data[y][this.fields[x]] = value;\n    };\n\n    JSDataSource.prototype.getColumnCount = function() {\n\n        return this.getFields().length;\n    };\n\n    JSDataSource.prototype.getRowCount = function() {\n\n        return this.data.length;\n    };\n\n    JSDataSource.prototype.getFields = function() {\n\n        return this.fields;\n    };\n\n    JSDataSource.prototype.getHeaders = function() {\n        if (!this.headers || this.headers.length === 0) {\n            this.headers = this.getDefaultHeaders().map(function(each) {\n                return headerify(each);\n            });\n        }\n        return this.headers;\n    };\n\n    JSDataSource.prototype.getDefaultHeaders = function() {\n\n        return this.getFields();\n    };\n\n    JSDataSource.prototype.setFields = function(fields) {\n\n        this.fields = fields;\n    };\n\n    JSDataSource.prototype.setHeaders = function(headers) {\n\n        this.headers = headers;\n    };\n\n    JSDataSource.prototype.getGrandTotals = function() {\n        //nothing here\n        return;\n    };\n\n    JSDataSource.prototype.setData = function(arrayOfUniformObjects) {\n        this.data = arrayOfUniformObjects;\n    };\n\n    return JSDataSource;\n\n})();\n","'use strict';\n\nvar stableSort = require('./stableSort.js');\nvar Map = require('./Map.js');\n\nmodule.exports = (function() {\n\n    return {\n        stableSort: stableSort,\n        Map: Map\n    };\n\n})();\n","'use strict';\n\nmodule.exports = (function() {\n\n    return {\n\n        count: function() { /* columIndex */\n            return function(group) {\n                var rows = group.getRowCount();\n                return rows;\n            };\n        },\n\n        sum: function(columIndex) {\n            return function(group) {\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                return sum;\n            };\n        },\n\n        min: function(columIndex) {\n            return function(group) {\n                var min = Infinity;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    min = Math.min(min, group.getValue(columIndex, r));\n                }\n                return min;\n            };\n        },\n\n\n        max: function(columIndex) {\n            return function(group) {\n                var max = -Infinity;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    max = Math.max(max, group.getValue(columIndex, r));\n                }\n                return max;\n            };\n        },\n\n        avg: function(columIndex) {\n            return function(group) {\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (var r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                return sum / rows;\n            };\n        },\n\n        first: function(columIndex) {\n            return function(group) {\n                return group.getValue(columIndex, 0);\n            };\n        },\n\n        last: function(columIndex) {\n            return function(group) {\n                var rows = group.getRowCount();\n                return group.getValue(columIndex, rows - 1);\n            };\n        },\n\n        stddev: function(columIndex) {\n            return function(group) {\n                var r;\n                var sum = 0;\n                var rows = group.getRowCount();\n                for (r = 0; r < rows; r++) {\n                    sum = sum + group.getValue(columIndex, r);\n                }\n                var mean = sum / rows;\n                var variance = 0;\n                for (r = 0; r < rows; r++) {\n                    var dev = (group.getValue(columIndex, r) - mean);\n                    variance = variance + (dev * dev);\n                }\n                var stddev = Math.sqrt(variance / rows);\n                return stddev;\n            };\n        }\n    };\n\n})();\n","'use strict';\n\nvar JSDataSource = require('./JSDataSource');\nvar DataSourceSorter = require('./DataSourceSorter');\nvar DataSourceSorterComposite = require('./DataSourceSorterComposite');\nvar DataSourceFilter = require('./DataSourceFilter');\nvar DataSourceGlobalFilter = require('./DataSourceGlobalFilter');\nvar DataSourceAggregator = require('./DataSourceAggregator');\nvar aggregations = require('./aggregations');\n\nmodule.exports = (function() {\n\n    return {\n        JSDataSource: JSDataSource,\n        DataSourceSorter: DataSourceSorter,\n        DataSourceSorterComposite: DataSourceSorterComposite,\n        DataSourceFilter: DataSourceFilter,\n        DataSourceGlobalFilter: DataSourceGlobalFilter,\n        DataSourceAggregator: DataSourceAggregator,\n        aggregations: aggregations\n    };\n\n})();\n","'use strict';\n\nvar stabilize = function(comparator, descending) {\n    return function(arr1, arr2) {\n        var x = arr1[0];\n        var y = arr2[0];\n        if (x === y) {\n            x = descending ? arr2[1] : arr1[1];\n            y = descending ? arr1[1] : arr2[1];\n        } else {\n            if (y === null) {\n                return -1;\n            }\n            if (x === null) {\n                return 1;\n            }\n        }\n        return comparator(x, y);\n    };\n};\n\n\nvar ascendingNumbers = function(x, y) {\n    return x - y;\n};\n\nvar descendingNumbers = function(x, y) {\n    return y - x;\n};\n\nvar ascendingAllOthers = function(x, y) {\n    return x < y ? -1 : 1;\n};\n\nvar descendingAllOthers = function(x, y) {\n    return y < x ? -1 : 1;\n};\n\nvar ascending = function(typeOfData) {\n    if (typeOfData === 'number') {\n        return stabilize(ascendingNumbers, false);\n    }\n    return stabilize(ascendingAllOthers, false);\n};\n\nvar descending = function(typeOfData) {\n    if (typeOfData === 'number') {\n        return stabilize(descendingNumbers, true);\n    }\n    return stabilize(descendingAllOthers, true);\n};\n\nmodule.exports = (function() {\n\n    function sort(indexVector, dataSource, sortType) {\n\n        var compare, i;\n\n        if (indexVector.length === 0) {\n            return; //nothing to do;\n        }\n\n        if (sortType === undefined) {\n            sortType = 1;\n        }\n\n        if (sortType === 0) {\n            return; // nothing to sort here;\n        }\n\n        var typeOfData = typeof dataSource(0);\n\n        compare = (sortType === 1) ? ascending(typeOfData) : descending(typeOfData);\n\n        //start the actually sorting.....\n        var tmp = new Array(indexVector.length);\n\n        //lets add the index for stability\n        for (i = 0; i < indexVector.length; i++) {\n            tmp[i] = [dataSource(i), i];\n        }\n\n        tmp.sort(compare);\n\n        //copy the sorted values into our index vector\n        for (i = 0; i < indexVector.length; i++) {\n            indexVector[i] = tmp[i][1];\n        }\n    }\n\n    return sort;\n})();\n"]} diff --git a/examples/v0.2/index.min.js b/examples/v0.2/index.min.js index 936dc7745..e94f1f7a5 100644 --- a/examples/v0.2/index.min.js +++ b/examples/v0.2/index.min.js @@ -1,9 +1,9 @@ -!function t(e,i,n){function o(s,l){if(!i[s]){if(!e[s]){var a="function"==typeof require&&require;if(!l&&a)return a(s,!0);if(r)return r(s,!0);throw new Error("Cannot find module '"+s+"'")}var u=i[s]={exports:{}};e[s][0].call(u.exports,function(t){var i=e[s][1][t];return o(i?i:t)},u,u.exports,t,e,i,n)}return i[s].exports}for(var r="function"==typeof require&&require,s=0;sdiv,.filter-tree-add>div,.filter-tree-remove{display:inline-block;width:15px;height:15px;border-radius:8px;background-color:#8c8;font-size:11.5px;font-weight:700;color:#fff;text-align:center;line-height:normal;font-style:normal;font-family:sans-serif;text-shadow:0 0 1.5px grey;margin-right:4px}.filter-tree-add-filter>div:before,.filter-tree-add>div:before{content:'\\ff0b'}.filter-tree-remove{background-color:#e88;border:0}.filter-tree-remove:before{content:'\\2212'}.filter-tree li::after{font-size:70%;font-style:italic;font-weight:700;color:#080}.filter-tree>ol>li:last-child::after{display:none}.op-or>ol>li::after{margin-left:2.5em;content:'— OR —'}.op-and>ol>li::after{margin-left:2.5em;content:'— AND —'}.op-nor>ol>li::after{margin-left:2.5em;content:'— NOR —'}.filter-tree-default>:enabled{margin:0 .4em;background-color:#ddd;border:0}.filter-tree-default>input[type=text]{width:8em;padding:0 5px}.filter-tree-default>select{border:0}.filter-tree-default>.filter-tree-warning{background-color:#ffc}.filter-tree-default>.filter-tree-error{background-color:#Fcc}.filter-tree .footnotes{font-size:6pt;margin:2px 0 0;line-height:normal;white-space:normal;color:#999}.filter-tree .footnotes>ol{margin:0;padding-left:2em}.filter-tree .footnotes>ol>li{margin:2px 0}.filter-tree .footnotes .field-name,.filter-tree .footnotes .field-value{font-weight:700;color:#777}.filter-tree .footnotes .field-value:after,.filter-tree .footnotes .field-value:before{content:'\"'}.filter-tree .footnotes .field-value{font-family:monospace}.filter-tree-chooser{position:absolute;font-size:9pt;outline:0;box-shadow:5px 5px 10px grey}";var f=0,p=/^filter-tree: /,v=c.extend("FilterTree",{initialize:function(t){u(a,"filter-tree-base",t&&t.cssStylesheetReferenceElement),t.editors&&(this.editors=t.editors)},destroy:function(){l.call(this)},editors:{Default:h},addEditor:function(t,e){e?this.editors[t]=h.extend(e):delete this.editors[t]},newView:function(){this.el=d("tree",++f),this.el.addEventListener("click",o.bind(this))},load:function(t){if(t){if(n(t),!(g[t.operator]||void 0===t.operator&&1===t.children.length))throw this.Error("Expected `operator` property to be one of: "+Object.keys(g));if(this.operator=t.operator,!(t.children instanceof Array&&t.children.length))throw this.Error("Expected `children` property to be a non-empty array.");this.children=[];var e=this;t.children.forEach(function(t){var i;if("object"!=typeof t)throw e.Error("Expected child to be an object containing either `children`, `editor`, or neither.");i=t.children?v:e.editors[t.editor||"Default"],e.children.push(new i({state:t,parent:e}))})}else{var i=Object.keys(this.editors),o=1===i.length;this.children=o?[new this.editors[i[0]]({parent:this})]:[],this.operator="op-and"}},render:function(){var t=this.el.querySelector("input[value="+this.operator+"]");if(t.checked=!0,this["filter-tree-op-choice"]({target:t}),!this.children.length&&Object.keys(this.editors).length>1){var e=this.el.querySelector(".filter-tree-add-filter");this["filter-tree-add-filter"]({target:e})}c.prototype.render.call(this)},"filter-tree-op-choice":function(t){var e=t.target;this.operator=e.value;var i=this.el.querySelectorAll("label>input.filter-tree-op-choice[name="+e.name+"]");Array.prototype.slice.call(i).forEach(function(t){t.parentElement.style.textDecoration=t.checked?"none":"line-through"});for(var n in g)this.el.classList.remove(n);this.el.classList.add(this.operator)},"filter-tree-add-filter":function(t){var e=Object.keys(this.editors);1===e.length?this.children.push(new this.editors[e[0]]({parent:this})):s.call(this,t)},"filter-tree-add":function(){this.children.push(new v({parent:this}))},"filter-tree-remove":function(t){var e=t.target,i=e.parentElement,n=this.children,o=e.nextElementSibling;n.forEach(function(t,e){t.el===o&&(delete n[e],i.remove())})},validate:function(t){t=t||{};var e,i=void 0===t.focus||t.focus,n=void 0===t.alert||t.alert;try{r.call(this,i)}catch(o){if(e=o.message,!p.test(e))throw o;n&&(e=e.replace(p,""),window.alert(e))}return e},test:function m(t){var e=g[this.operator],i=e.seed;return this.children.find(function(n){return n?(n instanceof h?i=e.reduce(i,n.test(t)):n.children.length&&(i=e.reduce(i,m.call(n,t))),i===e.abort):!1}),e.negate?!i:i},toJSON:function C(){var t={operator:this.operator,children:[]};this.children.forEach(function(e){e&&(e instanceof h?t.children.push(e):e.children.length&&t.children.push(C.call(e)))});var e=c.prototype.toJSON.call(this);return Object.keys(e).forEach(function(i){t[i]=e[i]}),t},toSQL:function w(){var t=g[this.operator].SQL,e=t.beg;return this.children.forEach(function(i,n){var o=n?" "+t.op+" ":"";i&&(i instanceof h?e+=o+i.toSQL():i.children.length&&(e+=o+w.call(i)))}),e+=t.end}});e.exports=v},{"./js/FilterLeaf":6,"./js/FilterNode":7,"./js/template":8,"./js/tree-operators":9,"css-injector":3}],6:[function(t,e,i){"use strict";function n(t,e){var i,o;return o=t.find(function(t){return(t.options||t)instanceof Array?i=n(t.options||t,e):t.name===e}),i||o}function o(t){var e=t.target;if(e.classList.remove("filter-tree-error"),c.setWarningClass(e),!e.multiple&&e.value)for(;(e=e.nextElementSibling)&&(!("name"in e)||""!==e.value.trim()););e&&""===e.value.trim()&&(e.value="",c.clickIn(e))}function r(t){setTimeout(function(){t.classList.add("filter-tree-error"),c.clickIn(t)},0)}function s(t){var e,i;switch(t.type){case"checkbox":case"radio":for(t=document.querySelectorAll("input[name='"+t.name+"']:enabled:checked"),e=[],i=0;it?-1:t>e?1:0}var u=t("regexp-like").cached,c=t("./FilterNode"),h=t("./template"),d={to:Number,not:isNaN},g={to:function(t){return new Date(t)},not:isNaN},f=c.extend("FilterLeaf",{name:"Column ? Literal",operators:{"<":{test:function(t,e){return e>t}},"≤":{test:function(t,e){return e>=t},SQL:"<="},"=":{test:function(t,e){return t===e}},"≥":{test:function(t,e){return t>=e},SQL:">="},">":{test:function(t,e){return t>e}},"≠":{test:function(t,e){return t!==e},SQL:"<>"},LIKE:{test:function(t,e){return u(e).test(t)}},"NOT LIKE":{test:function(t,e){return!u(e).test(t)}}},destroy:function(){if(this.controls)for(var t in this.controls)this.controls[t].removeEventListener("change",o)},newView:function(){var t=this.parent.nodeFields||this.fields;if(!t)throw this.Error("Terminal node requires a fields list.");var e=this.el=document.createElement("span");e.className="filter-tree-default",this.controls={column:this.makeElement(e,t,"column"),operator:this.makeElement(e,Object.keys(this.operators),"operator"),argument:this.makeElement(e)},e.appendChild(document.createElement("br"))},makeElement:function(t,e,i){var n,r,s,a=e?"select":"input";return e&&1===e.length?(r=e[0],n=document.createElement("input"),n.type="hidden",n.value=r.name||r.alias||r,s=document.createElement("span"),s.innerHTML=r.alias||r.name||r,s.appendChild(n),t.appendChild(s)):(n=l(a,e,i),this.el.addEventListener("change",o),c.setWarningClass(n),t.appendChild(n)),n},load:function(t){if(t){var e,i,n,o,r,s=[];for(var l in t)if("fields"!==l&&"editor"!==l)switch(e=t[l],i=this.controls[l],i.type){case"checkbox":case"radio":for(i=document.querySelectorAll("input[name='"+i.name+"']"),n=0;n=0;break;case"select-multiple":for(i=i.options,n=0,o=!1;n=0,i[n].selected=r;c.setWarningClass(i,o);break;default:i.value=e,c.setWarningClass(i)||i.value===e||s.push({key:l,value:e})}if(s.length){var a=s.length>1,u=h(a?"notes":"note"),d=u.lastElementChild;s.forEach(function(t){var e=a?document.createElement("li"):d;for(t=h("optionMissing",t.key,t.value);t.length;)e.appendChild(t[0]);a&&d.appendChild(e)}),i.parentNode.replaceChild(u,i.parentNode.lastElementChild)}}},converters:{number:d,"int":d,"float":d,date:g},validate:function(t){for(var e in this.controls){var i=this.controls[e],o=s(i).trim();if(""===o)throw t&&r(i),new c.Error("Blank "+e+" control.\nComplete the filter or delete it.");switch(this[e]=o,e){case"operator":var l=this.operators[o];this.operation=l.test,this.sqlOperator=l.SQL||o;break;case"column":var a=this.parent.nodeFields||this.fields,u=n(a,o);u&&u.type&&(this.converter=this.converters[u.type])}}},p:function(t){return t[this.column]},q:function(){return this.argument},test:function(t){var e,i,n=this.p(t),o=this.q(t),r=this.converter;return!r||r.not(e=r.to(n))||r.not(i=r.to(o))?this.operation(n,o):this.operation(e,i)},toJSON:function(t){var e={};this.editor&&(e.editor=this.editor);for(var i in this.controls)e[i]=this[i];return this.parent.nodeFields||this.fields===this.parent.fields||(e.fields=this.fields),e},toSQL:function(){return[this.SQL_QUOTED_IDENTIFIER+this.column+this.SQL_QUOTED_IDENTIFIER,this.sqlOperator," '"+this.argument.replace(/'/g,"''")+"'"].join(" ")}});e.exports=f},{"./FilterNode":7,"./template":8,"regexp-like":20}],7:[function(t,e,i){"use strict";function n(t,e,i,n){return e&&e[t]||i&&i[t]||n&&n[t]}var o=t("extend-me"),r=o.Base,s=t("./template");o.debug=!0;var l="OL",a="LI",u=r.extend({initialize:function(t){var e=t&&t.parent,i=t&&t.state;this.parent=e,this.nodeFields=n("nodeFields",t,i),this.fields=n("fields",t,i,e),this.editor=n("editor",t,i,e),this.fromJSON(i)},render:function(){if(this.parent){var t=document.createElement(a);t.appendChild(s("removeButton")),t.appendChild(this.el),this.parent.el.querySelector(l).appendChild(t)}},toJSON:function(){var t={};if(this.toJsonOptions){var e=this,i=[];this.toJsonOptions.fields&&(i.push("fields"),i.push("nodeFields")),this.toJsonOptions.editor&&i.push("editor"),i.forEach(function(i){(!e.parent||e[i]&&e[i]!==e.parent[i])&&(t[i]=e[i])})}return t},fromJSON:function(t){var e=this.el;this.newView(),this.load(t),this.render(),e&&!this.parent&&e.parentNode.replaceChild(this.el,e)},SQL_QUOTED_IDENTIFIER:'"'});u.setWarningClass=function(t,e){return arguments.length<2&&(e=t.value),t.classList[e?"remove":"add"]("filter-tree-warning"),e},u.Error=function(t){return new Error("filter-tree: "+t)},u.clickIn=function(t){t&&("SELECT"===t.tagName?setTimeout(function(){t.dispatchEvent(new MouseEvent("mousedown"))},0):t.focus())},e.exports=u},{"./template":8,"extend-me":4}],8:[function(t,e,i){"use strict";function n(t){var e,i=document.createElement("div"),n=r[t].toString().match(s)[1],a=[n].concat(Array.prototype.slice.call(arguments,1)),u={};for(l.lastIndex=0;e=l.exec(n);)u[e[1]]=!0;return e=Object.keys(u),e.length&&(e.forEach(function(t){i.textContent=a[t],a[t]=i.innerHTML}),a[0]=n.replace(l,"{$1}")),i.innerHTML=o.apply(this,a),1===i.children.length&&1===i.childNodes.length?i.firstChild:i.childNodes}var o=t("templex"),r={tree:function(){},removeButton:function(){},note:function(){},notes:function(){},optionMissing:function(){}},s=/\/\*\s*([^]+?)\s+\*\//,l=/\{(\d+)\:encode\}/g;e.exports=n},{templex:22}],9:[function(t,e,i){"use strict";function n(t,e){return t&&e}function o(t,e){return t||e}var r={"op-and":{reduce:n,seed:!0,abort:!1,negate:!1,SQL:{op:"AND",beg:"(",end:")"}},"op-or":{reduce:o,seed:!1,abort:!0,negate:!1,SQL:{op:"OR",beg:"(",end:")"}},"op-nor":{reduce:o,seed:!1,abort:!0,negate:!0,SQL:{op:"OR",beg:"NOT (",end:")"}}};e.exports=r},{}],10:[function(t,e,i){"use strict";function n(t){var e=this._bound={};for(s in u)e[s]=u[s].bind(this);var i=document.createElement("div");i.classList.add("thumb"),i.onclick=e.shortStop,i.onmouseover=e.onmouseover,this.thumb=i;var o=document.createElement("div");o.classList.add("finbar-vertical"),o.appendChild(i),this.paging&&(o.onclick=e.onclick),this.bar=o,t=t||{},this.orientation="vertical",this._min=this._index=0,this._max=100;for(var s in t)if(t.hasOwnProperty(s)){var c=t[s];switch(s){case"index":this._index=c;break;case"range":r(c),this._min=c.min,this._max=c.max,this.contentSize=c.max-c.min+1;break;default:"_"!==s.charAt(0)&&"function"!=typeof n.prototype[s]&&(this[s]=c)}}l(a,"finbar-base",t.cssStylesheetReferenceElement)}function o(t){for(var e=1;e'+t.replace("mouse","")+""}),e.appendChild(n),t={},i.forEach(function(e){t[e]=n.getElementsByClassName(e)[0]})}return t},_addEvt:function(t){var e=this.testPanelItem&&this.testPanelItem[t];e&&e.classList.add("listening"),window.addEventListener(t,this._bound["on"+t])},_removeEvt:function(t){var e=this.testPanelItem&&this.testPanelItem[t];e&&e.classList.remove("listening"),window.removeEventListener(t,this._bound["on"+t])}};var a,u={shortStop:function(t){t.stopPropagation()},onwheel:function(t){this.index+=t[this.deltaProp],t.stopPropagation(),t.preventDefault()},onclick:function(t){var e=this.thumb.getBoundingClientRect(),i=t[this.oh.coordinate].thumb,div.finbar-vertical>.thumb{position:absolute;background-color:#d3d3d3;-webkit-box-shadow:0 0 1px #000;-moz-box-shadow:0 0 1px #000;box-shadow:0 0 1px #000;border-radius:4px;margin:2px;opacity:.4;transition:opacity .5s}div.finbar-horizontal>.thumb.hover,div.finbar-vertical>.thumb.hover{opacity:1;transition:opacity .5s}div.finbar-vertical{top:0;bottom:0;right:0;width:11px}div.finbar-vertical>.thumb{top:0;right:0;width:7px}div.finbar-horizontal{left:0;right:0;bottom:0;height:11px}div.finbar-horizontal>.thumb{left:0;bottom:0;height:7px}",e.exports=n},{"css-injector":3}],11:[function(t,e,i){"use strict";function n(t,e){var i=this;this.div=t,this._component=e,this.dragEndtime=Date.now(),this.canvas=document.createElement("canvas"),this.div.appendChild(this.canvas),this.canvas.style.outline="none",this.canvasCTX=this.canvas.getContext("2d"),this.gc=new u(this.canvasCTX),this.buffer=document.createElement("canvas"),this.bufferCTX=this.buffer.getContext("2d"),this.bufferGC=new u(this.bufferCTX),this.mouseLocation=new l.Point(-1,-1),this.dragstart=new l.Point(-1,-1),this.bounds=new l.Rectangle(0,0,0,0),this.hasMouse=!1,document.addEventListener("mousemove",function(t){(i.hasMouse||i.isDragging())&&i.finmousemove(t)}),document.addEventListener("mouseup",function(t){i.finmouseup(t)}),document.addEventListener("wheel",function(t){i.finwheelmoved(t)}),document.addEventListener("keydown",function(t){i.finkeydown(t)}),document.addEventListener("keyup",function(t){i.finkeyup(t)}),this.canvas.onmouseover=function(){i.hasMouse=!0},this.canvas.addEventListener("focus",function(t){i.finfocusgained(t)}),this.canvas.addEventListener("blur",function(t){i.finfocuslost(t)}),this.canvas.addEventListener("mousedown",function(t){i.finmousedown(t)}),this.canvas.addEventListener("mouseout",function(t){i.hasMouse=!1,i.finmouseout(t)}),this.canvas.addEventListener("click",function(t){i.finclick(t)}),this.canvas.addEventListener("contextmenu",function(t){return i.fincontextmenu(t),t.preventDefault(),!1}),a.addEventListener(this.canvas,"tap",function(t){i.fintap(t)}),a.addEventListener(this.canvas,"holdpulse",function(t){i.finholdpulse(t)}),a.addEventListener(this.canvas,"flick",function(t){i.finflick(t)}),a.addEventListener(this.canvas,"release",function(t){i.finrelease(t)}),a.addEventListener(this.canvas,"trackstart",function(t){i.fintrackstart(t)}),a.addEventListener(this.canvas,"track",function(t){i.fintrack(t)}),a.addEventListener(this.canvas,"trackend",function(t){i.fintrackend(t)}),this.canvas.setAttribute("tabindex",0),this.canvas.contentEditable=!0,this.resize(),this.beginResizing(),this.beginPainting()}function o(t){if(g){for(var e=0;ei;i++)t[i]=e;return t[27]=["ESC","ESCSHIFT"],t[192]=["`","~"],t[49]=["1","!"],t[50]=["2","@"],t[51]=["3","#"],t[52]=["4","$"],t[53]=["5","%"],t[54]=["6","^"],t[55]=["7","&"],t[56]=["8","*"],t[57]=["9","("],t[48]=["0",")"],t[189]=["-","_"],t[187]=["=","+"],t[8]=["DELETE","DELETESHIFT"],t[9]=["TAB","TABSHIFT"],t[81]=["q","Q"],t[87]=["w","W"],t[69]=["e","E"],t[82]=["r","R"],t[84]=["t","T"],t[89]=["y","Y"],t[85]=["u","U"],t[73]=["i","I"],t[79]=["o","O"],t[80]=["p","P"],t[219]=["[","{"],t[221]=["]","}"],t[220]=["\\","|"],t[220]=["CAPSLOCK","CAPSLOCKSHIFT"],t[65]=["a","A"],t[83]=["s","S"],t[68]=["d","D"],t[70]=["f","F"],t[71]=["g","G"],t[72]=["h","H"],t[74]=["j","J"],t[75]=["k","K"],t[76]=["l","L"],t[186]=[";",":"],t[222]=["'","|"],t[13]=["RETURN","RETURNSHIFT"],t[16]=["SHIFT","SHIFT"],t[90]=["z","Z"],t[88]=["x","X"],t[67]=["c","C"],t[86]=["v","V"], -t[66]=["b","B"],t[78]=["n","N"],t[77]=["m","M"],t[188]=[",","<"],t[190]=[".",">"],t[191]=["/","?"],t[16]=["SHIFT","SHIFT"],t[17]=["CTRL","CTRLSHIFT"],t[18]=["ALT","ALTSHIFT"],t[91]=["COMMANDLEFT","COMMANDLEFTSHIFT"],t[32]=["SPACE","SPACESHIFT"],t[93]=["COMMANDRIGHT","COMMANDRIGHTSHIFT"],t[18]=["ALT","ALTSHIFT"],t[38]=["UP","UPSHIFT"],t[37]=["LEFT","LEFTSHIFT"],t[40]=["DOWN","DOWNSHIFT"],t[39]=["RIGHT","RIGHTSHIFT"],t[33]=["PAGEUP","PAGEUPSHIFT"],t[34]=["PAGEDOWN","PAGEDOWNSHIFT"],t[35]=["PAGERIGHT","PAGERIGHTSHIFT"],t[36]=["PAGELEFT","PAGELEFTSHIFT"],t}var l=t("rectangular"),a=t("./js/polymergestures.dev.js"),u=t("./js/GraphicsContext.js"),c=200,h=[],d=[],g=!0,f=!0,p=s();n.prototype={constructor:n.prototype.constructor,div:null,_component:null,gestures:a,canvas:null,canvasCTX:null,focuser:null,buffer:null,ctx:null,mouseLocation:null,holdPulseCount:-1,dragstart:null,origin:null,bounds:null,dirty:!1,size:null,mousedown:!1,dragging:!1,repeatKeyCount:0,repeatKey:null,repeatKeyStartTime:0,currentKeys:[],hasMouse:!1,lastDoubleClickTime:0,dragEndTime:0,lastRepaintTime:0,addEventListener:function(t,e){this.canvas.addEventListener(t,e)},stopPaintLoop:function(){g=!1},restartPaintLoop:function(){g||(g=!0,requestAnimationFrame(o))},stopResizeLoop:function(){f=!1},restartResizeLoop:function(){f||(f=!0,setInterval(r,200))},detached:function(){this.stopPainting(),this.stopResizing()},useHiDPI:function(){return this._component.resolveProperty("useHiDPI")},useBitBlit:function(){return this._component.resolveProperty("useBitBlit")},getFPS:function(){var t=this._component.resolveProperty("repaintIntervalRate");return t?parseInt(t):0},tickPaint:function(t){var e=this.getFPS();if(0!==e){var i=1e3/e,n=t-this.lastRepaintTime;n>i&&this.dirty&&(this.lastRepaintTime=t-n%i,this.paintNow())}},beginPainting:function(){var t=this;this.dirty=!0,this.tickPainter=function(e){t.tickPaint(e)},h.push(this)},stopPainting:function(){h.splice(h.indexOf(this),1)},beginResizing:function(){var t=this;this.tickResizer=function(){t.checksize()},d.push(this)},stopResizing:function(){d.splice(d.indexOf(this),1)},start:function(){this.beginPainting(),this.beginResizing()},stop:function(){this.stopPainting(),this.stopResizing()},checksize:function(){var t=this.div.getBoundingClientRect();(t.width!==this.size.width||t.height!==this.size.height)&&this.sizeChangedNotification()},sizeChangedNotification:function(){this.resize()},resize:function(){var t=this.size=this.div.getBoundingClientRect();this.canvas.width=this.buffer.width=t.width,this.canvas.height=this.buffer.height=t.height;var e=1,i=this.useBitBlit(),n=window.devicePixelRatio&&this.useHiDPI();if(n){var o=window.devicePixelRatio||1,r=this.canvasCTX.webkitBackingStorePixelRatio||this.canvasCTX.mozBackingStorePixelRatio||this.canvasCTX.msBackingStorePixelRatio||this.canvasCTX.oBackingStorePixelRatio||this.canvasCTX.backingStorePixelRatio||1;e=o/r}var s=this.canvas.getAttribute("width"),a=this.canvas.getAttribute("height");this.canvas.width=s*e,this.canvas.height=a*e,this.buffer.width=s*e,this.buffer.height=a*e,this.canvas.style.width=s+"px",this.canvas.style.height=a+"px",this.buffer.style.width=s+"px",this.buffer.style.height=a+"px",this.bufferCTX.scale(e,e),n&&!i&&this.canvasCTX.scale(e,e),this.bounds=new l.Rectangle(0,0,t.width,t.height);var u=this._component;u&&u.setBounds(this.bounds),this.resizeNotification(),this.paintNow()},resizeNotification:function(){},getBounds:function(){return this.bounds},paintNow:function(){var t=this;this.safePaintImmediately(function(e){e.clearRect(0,0,t.canvas.width,t.canvas.height);var i=t._component;i&&i._paint(e),t.dirty=!1})},safePaintImmediately:function(t){var e=this.useBitBlit(),i=e?this.bufferGC:this.gc;try{i.save(),t(i)}catch(n){console.error(n)}finally{i.restore()}e&&this.flushBuffer()},flushBuffer:function(){this.buffer.width>0&&this.buffer.height>0&&this.canvasCTX.drawImage(this.buffer,0,0)},dispatchNewEvent:function(t,e,i){return i={detail:i||{}},i.detail.primitiveEvent=t,this.canvas.dispatchEvent(new CustomEvent(e,i))},dispatchNewMouseKeysEvent:function(t,e,i){return i=i||{},i.mouse=this.mouseLocation,i.keys=this.currentKeys,this.dispatchNewEvent(t,e,i)},finmousemove:function(t){!this.isDragging()&&this.mousedown&&(this.beDragging(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-dragstart",{isRightClick:this.isRightClick(t)}),this.dragstart=new l.Point(this.mouseLocation.x,this.mouseLocation.y)),this.mouseLocation=this.getLocal(t),this.isDragging()&&this.dispatchNewMouseKeysEvent(t,"fin-canvas-drag",{dragstart:this.dragstart,isRightClick:this.isRightClick(t)}),this.bounds.contains(this.mouseLocation)&&this.dispatchNewMouseKeysEvent(t,"fin-canvas-mousemove")},finmousedown:function(t){this.mouseLocation=this.getLocal(t),this.mousedown=!0,this.dispatchNewMouseKeysEvent(t,"fin-canvas-mousedown",{isRightClick:this.isRightClick(t)}),this.takeFocus()},finmouseup:function(t){this.isDragging()&&(this.dispatchNewMouseKeysEvent(t,"fin-canvas-dragend",{dragstart:this.dragstart,isRightClick:this.isRightClick(t)}),this.beNotDragging(),this.dragEndtime=Date.now()),this.mousedown=!1,this.dispatchNewMouseKeysEvent(t,"fin-canvas-mouseup",{isRightClick:this.isRightClick(t)})},finmouseout:function(t){this.mousedown||(this.mouseLocation=new l.Point(-1,-1)),this.dispatchNewMouseKeysEvent(t,"fin-canvas-mouseout")},finwheelmoved:function(t){!this.isDragging()&&this.hasFocus()&&(t.preventDefault(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-wheelmoved",{isRightClick:this.isRightClick(t)}))},finclick:function(t){return Date.now()-this.lastClickTime<250?void this.findblclick(t):(this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-click",{isRightClick:this.isRightClick(t)}),void(this.lastClickTime=Date.now()))},finrelease:function(t){this.holdPulseCount=0,this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-release")},finflick:function(t){this.hasFocus()&&(this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-flick",{isRightClick:this.isRightClick(t)}))},fintrackstart:function(t){this.hasFocus()&&(this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-trackstart"))},fintrack:function(t){this.hasFocus()&&(this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-track"))},fintrackend:function(t){this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-trackend")},finhold:function(t){this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-hold",{isRightClick:this.isRightClick(t)})},finholdpulse:function(t){this.mouseLocation=this.getLocal(t),this.dispatchNewMouseKeysEvent(t,"fin-canvas-holdpulse",{count:this.holdPulseCount++})},fintap:function(t){var e=this,i=Date.now(),n=i-this.lastDoubleClickTime;300>n||i-this.dragEndtime<100||setTimeout(function(){e._fintap(t)},180)},_fintap:function(t){var e=Date.now(),i=e-this.lastDoubleClickTime;300>i||this.dispatchNewMouseKeysEvent(t,"fin-canvas-tap",{isRightClick:this.isRightClick(t)})},findblclick:function(t){this.mouseLocation=this.getLocal(t),this.lastDoubleClickTime=Date.now(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-dblclick",{isRightClick:this.isRightClick(t)})},getCharMap:function(){return p},finkeydown:function(t){if(this.hasFocus()){var e=t.shiftKey?p[t.keyCode][1]:p[t.keyCode][0];t.repeat?this.repeatKey===e?this.repeatKeyCount++:(this.repeatKey=e,this.repeatKeyStartTime=Date.now()):(this.repeatKey=null,this.repeatKeyCount=0,this.repeatKeyStartTime=0),-1===this.currentKeys.indexOf(e)&&this.currentKeys.push(e),this.dispatchNewEvent(t,"fin-canvas-keydown",{alt:t.altKey,ctrl:t.ctrlKey,"char":e,code:t.charCode,key:t.keyCode,meta:t.metaKey,repeatCount:this.repeatKeyCount,repeatStartTime:this.repeatKeyStartTime,shift:t.shiftKey,identifier:t.keyIdentifier,currentKeys:this.currentKeys.slice(0)})}},finkeyup:function(t){var e=t.shiftKey?p[t.keyCode][1]:p[t.keyCode][0];this.currentKeys.splice(this.currentKeys.indexOf(e),1),this.hasFocus()&&(this.repeatKeyCount=0,this.repeatKey=null,this.repeatKeyStartTime=0,this.dispatchNewEvent(t,"fin-canvas-keyup",{alt:t.altKey,ctrl:t.ctrlKey,"char":e,code:t.charCode,key:t.keyCode,meta:t.metaKey,repeat:t.repeat,shift:t.shiftKey,identifier:t.keyIdentifier,currentKeys:this.currentKeys.slice(0)}))},finfocusgained:function(t){this.dispatchNewEvent(t,"fin-canvas-focus-gained")},finfocuslost:function(t){this.dispatchNewEvent(t,"fin-canvas-focus-lost")},fincontextmenu:function(t){return t.ctrlKey&&-1===this.currentKeys.indexOf("CTRL")&&this.currentKeys.push("CTRL"),Date.now()-this.lastClickTime<250?void this.findblclick(t):(this.dispatchNewMouseKeysEvent(t,"fin-canvas-context-menu",{isRightClick:this.isRightClick(t)}),void(this.lastClickTime=Date.now()))},repaint:function(){var t=this.getFPS();this.dirty=!0,g&&0!==t||this.paintNow()},getMouseLocation:function(){return this.mouseLocation},getOrigin:function(){var t=this.canvas.getBoundingClientRect(),e=new l.Point(t.left,t.top);return e},getLocal:function(t){var e=this.canvas.getBoundingClientRect(),i=new l.Point(t.clientX-e.left,t.clientY-e.top);return i},hasFocus:function(){return document.activeElement===this.canvas},takeFocus:function(){var t=this;this.hasFocus()||setTimeout(function(){t.canvas.focus()},10)},beDragging:function(){this.dragging=!0,this.disableDocumentElementSelection()},beNotDragging:function(){this.dragging=!1,this.enableDocumentElementSelection()},isDragging:function(){return this.dragging},disableDocumentElementSelection:function(){var t=document.body.style;t.cssText=t.cssText+"-webkit-user-select: none"},enableDocumentElementSelection:function(){var t=document.body.style;t.cssText=t.cssText.replace("-webkit-user-select: none","")},setFocusable:function(t){this.focuser.style.display=t?"":"none"},isRightClick:function(t){var e;return t=t||window.event,"which"in t?e=3===t.which:"button"in t&&(e=2===t.button),e},dispatchEvent:function(t){return this.canvas.dispatchEvent(t)}},requestAnimationFrame(o),setInterval(r,c),e.exports=n},{"./js/GraphicsContext.js":12,"./js/polymergestures.dev.js":14,rectangular:19}],12:[function(t,e,i){"use strict";function n(t,e){function i(i){i in n.prototype||s.test(i)||("function"==typeof t[i]?r[i]=e?function(){return e(i,arguments,t[i].apply(t,arguments))}:t[i].bind(t):Object.defineProperty(r,i,{get:function(){var n=t[i];return e?e(i,"getter",n):n},set:function(n){t[i]=e?e(i,"setter",n):n}}))}this.gc=t;var r=this,s=/^webkit/;switch(typeof e){case"string":e=o.bind(void 0,e+".");break;case"boolean":e===!0&&(e=o.bind(void 0,"gc."));break;case"function":if(3!==e.length)throw"GraphicsContext: User-supplied API logger function does not accept three parameters.";break;default:e=!1}Object.keys(Object.getPrototypeOf(t)).forEach(i),Object.keys(t).forEach(i)}var o=t("./gc-console-logger");e.exports=n},{"./gc-console-logger":13}],13:[function(t,e,i){"use strict";function n(t,e,i,n){var r=n;switch("string"==typeof n&&(r='"'+r+'"'),e=t+e,i){case"getter":console.log(e,"=",r);break;case"setter":console.log(e,o,r);break;default:e+="("+Array.prototype.slice.call(i).join(", ")+")",void 0===r?console.log(e):console.log(e,o,r)}return n}var o="⟹";e.exports=n},{}],14:[function(t,e,i){!function(t){var e=!1,i=document.createElement("meta");if(i.createShadowRoot){var n=i.createShadowRoot(),o=document.createElement("span");n.appendChild(o),i.addEventListener("testpath",function(t){t.path&&(e=t.path[0]===o),t.stopPropagation()});var r=new CustomEvent("testpath",{bubbles:!0});document.head.appendChild(i),o.dispatchEvent(r),i.parentNode.removeChild(i),n=o=null}i=null;var s={shadow:function(t){return t?t.shadowRoot||t.webkitShadowRoot:void 0},canTarget:function(t){return t&&Boolean(t.elementFromPoint)},targetingShadow:function(t){var e=this.shadow(t);return this.canTarget(e)?e:void 0},olderShadow:function(t){var e=t.olderShadowRoot;if(!e){var i=t.querySelector("shadow");i&&(e=i.olderShadowRoot)}return e},allShadows:function(t){for(var e=[],i=this.shadow(t);i;)e.push(i),i=this.olderShadow(i);return e},searchRoot:function(t,e,i){var n,o;return t?(n=t.elementFromPoint(e,i),n?o=this.targetingShadow(n):t!==document&&(o=this.olderShadow(t)),this.searchRoot(o,e,i)||n):void 0},owner:function(t){if(!t)return document;for(var e=t;e.parentNode;)e=e.parentNode;return e.nodeType!=Node.DOCUMENT_NODE&&e.nodeType!=Node.DOCUMENT_FRAGMENT_NODE&&(e=document),e},findTarget:function(t){if(e&&t.path&&t.path.length)return t.path[0];var i=t.clientX,n=t.clientY,o=this.owner(t.target);return o.elementFromPoint(i,n)||(o=document),this.searchRoot(o,i,n)},findTouchAction:function(t){var i;if(e&&t.path&&t.path.length){for(var n=t.path,o=0;o=0?t=this.walk(t,o):e=this.walk(e,-o);t&&e&&t!==e;)t=t.parentNode||t.host,e=e.parentNode||e.host;return t},walk:function(t,e){for(var i=0;t&&e>i;i++)t=t.parentNode||t.host;return t},depth:function(t){for(var e=0;t;)e++,t=t.parentNode||t.host;return e},deepContains:function(t,e){var i=this.LCA(t,e);return i===t},insideNode:function(t,e,i){var n=t.getBoundingClientRect();return n.left<=e&&e<=n.right&&n.top<=i&&i<=n.bottom},path:function(t){var i;if(e&&t.path&&t.path.length)i=t.path;else{i=[];for(var n=this.findTarget(t);n;)i.push(n),n=n.parentNode||n.host}return i}};t.targetFinding=s,t.findTarget=s.findTarget.bind(s),t.deepContains=s.deepContains.bind(s),t.insideNode=s.insideNode}(i),function(){function t(t){return"html /deep/ "+e(t)}function e(t){return'[touch-action="'+t+'"]'}function i(t){return"{ -ms-touch-action: "+t+"; touch-action: "+t+";}"}var n=["none","auto","pan-x","pan-y",{rule:"pan-x pan-y",selectors:["pan-x pan-y","pan-y pan-x"]},"manipulation"],o="",r="string"==typeof document.head.style.touchAction,s=!window.ShadowDOMPolyfill&&document.head.createShadowRoot;if(r){n.forEach(function(n){String(n)===n?(o+=e(n)+i(n)+"\n",s&&(o+=t(n)+i(n)+"\n")):(o+=n.selectors.map(e)+i(n.rule)+"\n",s&&(o+=n.selectors.map(t)+i(n.rule)+"\n"))});var l=document.createElement("style");l.textContent=o,document.head.appendChild(l)}}(),function(t){var e=["bubbles","cancelable","view","detail","screenX","screenY","clientX","clientY","ctrlKey","altKey","shiftKey","metaKey","button","relatedTarget","pageX","pageY"],i=[!1,!1,null,null,0,0,0,0,!1,!1,!1,!1,0,null,0,0],n=function(){return function(){}},o={preventTap:n,makeBaseEvent:function(t,e){var i=document.createEvent("Event");return i.initEvent(t,e.bubbles||!1,e.cancelable||!1),i.preventTap=o.preventTap(i),i},makeGestureEvent:function(t,e){e=e||Object.create(null);for(var i,n=this.makeBaseEvent(t,e),o=0,r=Object.keys(e);o-1?this.values[i]=e:(this.keys.push(t),this.values.push(e))},has:function(t){return this.keys.indexOf(t)>-1},"delete":function(t){var e=this.keys.indexOf(t);e>-1&&(this.keys.splice(e,1),this.values.splice(e,1))},get:function(t){var e=this.keys.indexOf(t);return this.values[e]},clear:function(){this.keys.length=0,this.values.length=0},forEach:function(t,e){this.values.forEach(function(i,n){t.call(e,i,this.keys[n],this)},this)},pointers:function(){return this.keys.length}},t.PointerMap=e}(i),function(t){var e,i=["bubbles","cancelable","view","detail","screenX","screenY","clientX","clientY","ctrlKey","altKey","shiftKey","metaKey","button","relatedTarget","buttons","pointerId","width","height","pressure","tiltX","tiltY","pointerType","hwTimestamp","isPrimary","type","target","currentTarget","which","pageX","pageY","timeStamp","preventTap","tapPrevented","_source"],n=[!1,!1,null,null,0,0,0,0,!1,!1,!1,!1,0,null,0,0,0,0,0,0,0,"",0,!1,"",null,null,0,0,0,0,function(){},!1],o="undefined"!=typeof SVGElementInstance,r=t.eventFactory,s={IS_IOS:!1,pointermap:new t.PointerMap,requiredGestures:new t.PointerMap,eventMap:Object.create(null),eventSources:Object.create(null),eventSourceList:[],gestures:[],dependencyMap:{down:{listeners:0,index:-1},up:{listeners:0,index:-1}},gestureQueue:[],registerSource:function(t,e){var i=e,n=i.events;n&&(n.forEach(function(t){i[t]&&(this.eventMap[t]=i[t].bind(i))},this),this.eventSources[t]=i,this.eventSourceList.push(i))},registerGesture:function(t,e){var i=Object.create(null);i.listeners=0,i.index=this.gestures.length;for(var n,o=0;oo&&(i=this.eventSourceList[o]);o++)i.register.call(i,t,e)},unregister:function(t){for(var e,i=this.eventSourceList.length,n=0;i>n&&(e=this.eventSourceList[n]);n++)e.unregister.call(e,t)},down:function(t){this.requiredGestures.set(t.pointerId,e),this.fireEvent("down",t)},move:function(t){t.type="move",this.fillGestureQueue(t)},up:function(t){this.fireEvent("up",t),this.requiredGestures["delete"](t.pointerId)},cancel:function(t){t.tapPrevented=!0,this.fireEvent("up",t),this.requiredGestures["delete"](t.pointerId)},addGestureDependency:function(t,e){var i=t._pgEvents;if(i&&e)for(var n,o,r,s=Object.keys(i),l=0;l0&&(n=this.dependencyMap[r],o=n?n.index:-1,e[o]=!0)},eventHandler:function(i){var n=i.type;if("touchstart"===n||"mousedown"===n||"pointerdown"===n||"MSPointerDown"===n)if(i._handledByPG||(e={}),this.IS_IOS){var o=i;if("touchstart"===n){var r=i.changedTouches[0];o={target:i.target,clientX:r.clientX,clientY:r.clientY,path:i.path}}for(var s,l=i.path||t.targetFinding.path(o),a=0;an&&(i=e[n]);n++)this.addEvent(t,i)},unlisten:function(t,e){for(var i,n=0,o=e.length;o>n&&(i=e[n]);n++)this.removeEvent(t,i)},addEvent:function(t,e){t.addEventListener(e,this.boundHandler)},removeEvent:function(t,e){t.removeEventListener(e,this.boundHandler)},makeEvent:function(t,e){var i=r.makePointerEvent(t,e);return i.preventDefault=e.preventDefault,i.tapPrevented=e.tapPrevented,i._target=i._target||e.target,i},fireEvent:function(t,e){var i=this.makeEvent(t,e);return this.dispatchEvent(i)},cloneEvent:function(t){for(var e,r=Object.create(null),s=0;s0&&t._pgListeners--,0===t._pgListeners&&s.unregister(t),t._pgEvents&&(t._pgEvents[i]>0?t._pgEvents[i]--:t._pgEvents[i]=0)),Boolean(n)},t.removeEventListener=function(e,i,n,o){n&&(t.deactivateGesture(e,i),e.removeEventListener(i,n,o))}}(i),function(t){var e=t.dispatcher,i=e.pointermap,n=25,o=[0,1,4,2],r=0,s=/Linux.*Firefox\//i,l=function(){if(s.test(navigator.userAgent))return!1;try{return 1===new MouseEvent("test",{buttons:1}).buttons}catch(t){return!1}}(),a={POINTER_ID:1,POINTER_TYPE:"mouse",events:["mousedown","mousemove","mouseup"],exposes:["down","up","move"],register:function(t){e.listen(t,this.events)},unregister:function(t){t.nodeType!==Node.DOCUMENT_NODE&&e.unlisten(t,this.events)},lastTouches:[],isEventSimulatedFromTouch:function(t){for(var e,i=this.lastTouches,o=t.clientX,r=t.clientY,s=0,l=i.length;l>s&&(e=i[s]);s++){var a=Math.abs(o-e.x),u=Math.abs(r-e.y);if(n>=a&&n>=u)return!0}},prepareEvent:function(t){var i=e.cloneEvent(t);if(i.pointerId=this.POINTER_ID,i.isPrimary=!0,i.pointerType=this.POINTER_TYPE,i._source="mouse",!l){var n=t.type,s=o[t.which]||0;"mousedown"===n?r|=s:"mouseup"===n&&(r&=~s),i.buttons=r}return i},mousedown:function(n){if(!this.isEventSimulatedFromTouch(n)){var o=(i.has(this.POINTER_ID),this.prepareEvent(n));o.target=t.findTarget(n),i.set(this.POINTER_ID,o.target),e.down(o)}},mousemove:function(t){if(!this.isEventSimulatedFromTouch(t)){var n=i.get(this.POINTER_ID);if(n){var o=this.prepareEvent(t);o.target=n,0===(l?o.buttons:o.which)?(l||(r=o.buttons=0),e.cancel(o),this.cleanupMouse(o.buttons)):e.move(o)}}},mouseup:function(n){if(!this.isEventSimulatedFromTouch(n)){var o=this.prepareEvent(n);o.relatedTarget=t.findTarget(n),o.target=i.get(this.POINTER_ID),e.up(o),this.cleanupMouse(o.buttons)}},cleanupMouse:function(t){0===t&&i["delete"](this.POINTER_ID)}};t.mouseEvents=a}(i),function(t){var e=t.dispatcher,i=(t.targetFinding.allShadows.bind(t.targetFinding),e.pointermap),n=(Array.prototype.map.call.bind(Array.prototype.map),2500),o=25,r=200,s=20,l=!1,a={IS_IOS:!1,events:["touchstart","touchmove","touchend","touchcancel"],exposes:["down","up","move"],register:function(t,i){(this.IS_IOS?i:!i)&&e.listen(t,this.events)},unregister:function(t){this.IS_IOS||e.unlisten(t,this.events)},scrollTypes:{EMITTER:"none",XSCROLLER:"pan-x",YSCROLLER:"pan-y"},touchActionToScrollType:function(t){var e=t,i=this.scrollTypes;return e===i.EMITTER?"none":e===i.XSCROLLER?"X":e===i.YSCROLLER?"Y":"XY"},POINTER_TYPE:"touch",firstTouch:null,isPrimaryTouch:function(t){return this.firstTouch===t.identifier},setPrimaryTouch:function(t){(0===i.pointers()||1===i.pointers()&&i.has(1))&&(this.firstTouch=t.identifier,this.firstXY={X:t.clientX,Y:t.clientY},this.firstTarget=t.target,this.scrolling=null,this.cancelResetClickCount())},removePrimaryPointer:function(t){t.isPrimary&&(this.firstTouch=null,this.firstXY=null,this.resetClickCount())},clickCount:0,resetId:null,resetClickCount:function(){var t=function(){this.clickCount=0,this.resetId=null}.bind(this);this.resetId=setTimeout(t,r)},cancelResetClickCount:function(){this.resetId&&clearTimeout(this.resetId)},typeToButtons:function(t){var e=0;return("touchstart"===t||"touchmove"===t)&&(e=1),e},findTarget:function(e,n){if("touchstart"===this.currentTouchEvent.type){if(this.isPrimaryTouch(e)){var o={clientX:e.clientX,clientY:e.clientY,path:this.currentTouchEvent.path,target:this.currentTouchEvent.target};return t.findTarget(o)}return t.findTarget(e)}return i.get(n)},touchToPointer:function(t){var i=this.currentTouchEvent,n=e.cloneEvent(t),o=n.pointerId=t.identifier+2;n.target=this.findTarget(t,o),n.bubbles=!0,n.cancelable=!0,n.detail=this.clickCount,n.buttons=this.typeToButtons(i.type),n.width=t.webkitRadiusX||t.radiusX||0,n.height=t.webkitRadiusY||t.radiusY||0,n.pressure=t.webkitForce||t.force||.5,n.isPrimary=this.isPrimaryTouch(t),n.pointerType=this.POINTER_TYPE,n._source="touch";var r=this;return n.preventDefault=function(){r.scrolling=!1,r.firstXY=null,i.preventDefault()},n},processTouches:function(t,e){var n=t.changedTouches;this.currentTouchEvent=t;for(var o,r,s=0;s=u}return i}},findTouch:function(t,e){for(var i,n=0,o=t.length;o>n&&(i=t[n]);n++)if(i.identifier===e)return!0},vacuumTouches:function(t){var e=t.touches;if(i.pointers()>=e.length){var n=[];i.forEach(function(t,i){if(1!==i&&!this.findTouch(e,i-2)){var o=t;n.push(o)}},this),n.forEach(function(t){this.cancel(t),i["delete"](t.pointerId)},this)}},touchstart:function(t){this.vacuumTouches(t),this.setPrimaryTouch(t.changedTouches[0]),this.dedupSynthMouse(t),this.scrolling||(this.clickCount++,this.processTouches(t,this.down))},down:function(t){e.down(t)},touchmove:function(t){if(l)t.cancelable&&this.processTouches(t,this.move);else if(this.scrolling){if(this.firstXY){var e=t.changedTouches[0],i=e.clientX-this.firstXY.X,n=e.clientY-this.firstXY.Y,o=Math.sqrt(i*i+n*n);o>=s&&(this.touchcancel(t),this.scrolling=!0,this.firstXY=null)}}else null===this.scrolling&&this.shouldScroll(t)?this.scrolling=!0:(this.scrolling=!1,t.preventDefault(),this.processTouches(t,this.move))},move:function(t){e.move(t)},touchend:function(t){this.dedupSynthMouse(t),this.processTouches(t,this.up)},up:function(i){i.relatedTarget=t.findTarget(i),e.up(i)},cancel:function(t){e.cancel(t)},touchcancel:function(t){t._cancel=!0,this.processTouches(t,this.cancel)},cleanUpPointer:function(t){i["delete"](t.pointerId),this.removePrimaryPointer(t)},dedupSynthMouse:function(e){var i=t.mouseEvents.lastTouches,o=e.changedTouches[0];if(this.isPrimaryTouch(o)){var r={x:o.clientX,y:o.clientY};i.push(r);var s=function(t,e){var i=t.indexOf(e);i>-1&&t.splice(i,1)}.bind(null,i,r);setTimeout(s,n)}}},u=Event.prototype.stopImmediatePropagation||Event.prototype.stopPropagation;document.addEventListener("click",function(e){var i=e.clientX,n=e.clientY,r=function(t){var e=Math.abs(i-t.x),r=Math.abs(n-t.y);return o>=e&&o>=r},s=t.mouseEvents.lastTouches.some(r),l=t.targetFinding.path(e);if(s){for(var c=0;c0?1:-1},calcPositionDelta:function(t,e){var i=0,n=0;return t&&e&&(i=e.pageX-t.pageX,n=e.pageY-t.pageY),{x:i,y:n}},fireTrack:function(t,e,n){var o=n,r=this.calcPositionDelta(o.downEvent,e),s=this.calcPositionDelta(o.lastMoveEvent,e);if(s.x)o.xDirection=this.clampDir(s.x);else if("trackx"===t)return;if(s.y)o.yDirection=this.clampDir(s.y);else if("tracky"===t)return;var l={bubbles:!0,cancelable:!0,trackInfo:o.trackInfo,relatedTarget:e.relatedTarget,pointerType:e.pointerType,pointerId:e.pointerId,_source:"track"};"tracky"!==t&&(l.x=e.x,l.dx=r.x,l.ddx=s.x,l.clientX=e.clientX,l.pageX=e.pageX,l.screenX=e.screenX,l.xDirection=o.xDirection),"trackx"!==t&&(l.dy=r.y,l.ddy=s.y,l.y=e.y,l.clientY=e.clientY,l.pageY=e.pageY,l.screenY=e.screenY,l.yDirection=o.yDirection);var a=i.makeGestureEvent(t,l);o.downTarget.dispatchEvent(a)},down:function(t){if(t.isPrimary&&("mouse"===t.pointerType?1===t.buttons:!0)){var e={downEvent:t,downTarget:t.target,trackInfo:{},lastMoveEvent:null,xDirection:0,yDirection:0,tracking:!1};n.set(t.pointerId,e)}},move:function(t){var e=n.get(t.pointerId);if(e){if(!e.tracking){var i=this.calcPositionDelta(e.downEvent,t),o=i.x*i.x+i.y*i.y;o>this.WIGGLE_THRESHOLD&&(e.tracking=!0,e.lastMoveEvent=e.downEvent,this.fireTrack("trackstart",t,e))}e.tracking&&(this.fireTrack("track",t,e),this.fireTrack("trackx",t,e),this.fireTrack("tracky",t,e)),e.lastMoveEvent=t}},up:function(t){var e=n.get(t.pointerId);e&&(e.tracking&&this.fireTrack("trackend",t,e),n["delete"](t.pointerId))}};e.registerGesture("track",o)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n={HOLD_DELAY:200,WIGGLE_THRESHOLD:16,events:["down","move","up"],exposes:["hold","holdpulse","release"],heldPointer:null,holdJob:null,pulse:function(){var t=Date.now()-this.heldPointer.timeStamp,e=this.held?"holdpulse":"hold"; -this.fireHold(e,t),this.held=!0},cancel:function(){clearInterval(this.holdJob),this.held&&this.fireHold("release"),this.held=!1,this.heldPointer=null,this.target=null,this.holdJob=null},down:function(t){t.isPrimary&&!this.heldPointer&&(this.heldPointer=t,this.target=t.target,this.holdJob=setInterval(this.pulse.bind(this),this.HOLD_DELAY))},up:function(t){this.heldPointer&&this.heldPointer.pointerId===t.pointerId&&this.cancel()},move:function(t){if(this.heldPointer&&this.heldPointer.pointerId===t.pointerId){var e=t.clientX-this.heldPointer.clientX,i=t.clientY-this.heldPointer.clientY;e*e+i*i>this.WIGGLE_THRESHOLD&&this.cancel()}},fireHold:function(t,e){var n={bubbles:!0,cancelable:!0,pointerType:this.heldPointer.pointerType,pointerId:this.heldPointer.pointerId,x:this.heldPointer.clientX,y:this.heldPointer.clientY,_source:"hold"};e&&(n.holdTime=e);var o=i.makeGestureEvent(t,n);this.target.dispatchEvent(o)}};e.registerGesture("hold",n)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n=new t.PointerMap,o={events:["down","up"],exposes:["tap"],down:function(t){t.isPrimary&&!t.tapPrevented&&n.set(t.pointerId,{target:t.target,buttons:t.buttons,x:t.clientX,y:t.clientY})},shouldTap:function(t,e){var i=!0;return"mouse"===t.pointerType&&(i=1^t.buttons&&1&e.buttons),i&&!t.tapPrevented},up:function(e){var o=n.get(e.pointerId);if(o&&this.shouldTap(e,o)){var r=t.targetFinding.LCA(o.target,e.relatedTarget);if(r){var s=i.makeGestureEvent("tap",{bubbles:!0,cancelable:!0,x:e.clientX,y:e.clientY,detail:e.detail,pointerType:e.pointerType,pointerId:e.pointerId,altKey:e.altKey,ctrlKey:e.ctrlKey,metaKey:e.metaKey,shiftKey:e.shiftKey,_source:"tap"});r.dispatchEvent(s)}}n["delete"](e.pointerId)}};i.preventTap=function(t){return function(){t.tapPrevented=!0,n["delete"](t.pointerId)}},e.registerGesture("tap",o)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n=new t.PointerMap,o=180/Math.PI,r={events:["down","up","move","cancel"],exposes:["pinchstart","pinch","pinchend","rotate"],defaultActions:{pinch:"none",rotate:"none"},reference:{},down:function(e){if(n.set(e.pointerId,e),2==n.pointers()){var i=this.calcChord(),o=this.calcAngle(i);this.reference={angle:o,diameter:i.diameter,target:t.targetFinding.LCA(i.a.target,i.b.target)},this.firePinch("pinchstart",i.diameter,i)}},up:function(t){var e=n.get(t.pointerId),i=n.pointers();if(e){if(2===i){var o=this.calcChord();this.firePinch("pinchend",o.diameter,o)}n["delete"](t.pointerId)}},move:function(t){n.has(t.pointerId)&&(n.set(t.pointerId,t),n.pointers()>1&&this.calcPinchRotate())},cancel:function(t){this.up(t)},firePinch:function(t,e,n){var o=e/this.reference.diameter,r=i.makeGestureEvent(t,{bubbles:!0,cancelable:!0,scale:o,centerX:n.center.x,centerY:n.center.y,_source:"pinch"});this.reference.target.dispatchEvent(r)},fireRotate:function(t,e){var n=Math.round((t-this.reference.angle)%360),o=i.makeGestureEvent("rotate",{bubbles:!0,cancelable:!0,angle:n,centerX:e.center.x,centerY:e.center.y,_source:"pinch"});this.reference.target.dispatchEvent(o)},calcPinchRotate:function(){var t=this.calcChord(),e=t.diameter,i=this.calcAngle(t);e!=this.reference.diameter&&this.firePinch("pinch",e,t),i!=this.reference.angle&&this.fireRotate(i,t)},calcChord:function(){var t=[];n.forEach(function(e){t.push(e)});for(var e,i,o,r=0,s={a:t[0],b:t[1]},l=0;lr&&(r=o,s={a:a,b:c})}return e=Math.abs(s.a.clientX+s.b.clientX)/2,i=Math.abs(s.a.clientY+s.b.clientY)/2,s.center={x:e,y:i},s.diameter=r,s},calcAngle:function(t){var e=t.a.clientX-t.b.clientX,i=t.a.clientY-t.b.clientY;return(360+Math.atan2(i,e)*o)%360}};e.registerGesture("pinch",r)}(i)},{}],15:[function(t,e,i){"use strict";function n(t,e){if(!(this instanceof n))throw d('Not called with "new" keyword.');var i,o,s=this;e=e||{},"string"==typeof t?(o=r(document.querySelectorAll(t)),i=c(o)):t[0]instanceof Element?(o=r(t),i=c(o)):(o=[],i=u(t,e),i.forEach(function(t){o=o.concat(r(t.element.querySelectorAll("li")))})),i.forEach(function(t){t.element.addEventListener("wheel",h)}),o.forEach(function(t,e){var i=t!==t.parentElement.lastElementChild?s.addEvt(t,"mousedown",t,!0):{element:t};o[e]=i}),g="transform"in o[0].element.style?"transform":"-webkit-transform",this.modelLists=i,this.items=o,this.bindings={},this.callback={},m(v,"list-dragon-base",e.cssStylesheetReferenceElement)}function o(t,e,i){if(t){var n=0>p&&t>=0||0===p&&0!==t||p>0&&0>=t;p=t>0?Math.min(50,t):Math.max(-50,t),n&&(clearInterval(f),f=setInterval(function(t){var e=i.scrollTop+p;0>p&&t>e||p>0&&e>t?(i.scrollTop=t,clearInterval(f)):i.scrollTop=e},125))}else clearInterval(f),p=0}function r(t){return Array.prototype.slice.call(t)}function s(t,e){return e.top<=t.y&&t.y<=e.bottom&&e.left<=t.x&&t.x<=e.right}function l(t,e){return"translate("+Math.floor(t+window.scrollX)+"px,"+Math.floor(e+window.scrollY)+"px)"}function a(t){var e=document.createTextNode(t);return document.createElement("a").appendChild(e).parentNode.innerHTML}function u(t,e){var i=e.label||"{label}";return t.forEach(function(n,o){var r=n.label||i,s=void 0!==n.htmlEncode&&n.htmlEncode||e.htmlEncode,l=document.createElement("div"),u=document.createElement("ul");if(n.models)Object.keys(n).forEach(function(t){"models"!==t&&(n.models[t]=n[t])}),t[o]=n=n.models;else{if(!(n instanceof Array))throw d("List [{1}] not an array of models (with or without additional properties) OR an object (with a `models` property containing an array of models).",o);n.models=n}n.forEach(function(t){var i=t.label||r,o=void 0!==t.htmlEncode&&t.htmlEncode||s,l="object"==typeof t?t:{label:t},c=C.call([l,n,e],i),h=document.createElement("li");h.innerHTML=o?a(c):c,u.appendChild(h)});var c=document.createElement("li");if(c.innerHTML=" ",u.appendChild(c),n.title){var h=document.createElement("div");h.innerHTML=s?a(n.title):n.title,l.appendChild(h)}l.appendChild(u),l.className=n.cssClassNames||e.cssClassNames||"dragon-list",n.element=u,n.container=l}),t}function c(t){var e=[];return t.forEach(function(t){var i=t.parentElement,n=i.parentElement,o=[];e.find(function(t){return t.element===i})||(r(i.querySelectorAll("li")).forEach(function(t){t!==i.lastElementChild&&o.push(t.innerHTML)}),o.element=i,o.container=n,e.push(o))}),e}function h(t){t.stopPropagation()}function d(){return"list-dragon: "+C.apply(this,Array.prototype.slice.call(arguments))}var g,f,p,v,m=t("css-injector"),C=t("templex"),w=null;v="div.dragon-list{position:relative;background-color:#fff}div.dragon-list>div,div.dragon-list>ul{position:absolute;left:0;right:0}div.dragon-list>div{text-align:center;background-color:#00796b;color:#fff;box-shadow:0 3px 6px rgba(0,0,0,.16),0 3px 6px rgba(0,0,0,.23);overflow:hidden;white-space:nowrap}div.dragon-list>ul{overflow-y:auto;bottom:0;margin:0;padding:0;box-shadow:0 1px 3px rgba(0,0,0,.12),0 1px 2px rgba(0,0,0,.24)}div.dragon-list>ul>li,li.dragon-pop{white-space:nowrap;list-style-type:none;border:0 solid #f4f4f4;border-bottom:1px solid #e0e0e0;cursor:move;transition:border-top-width .2s}div.dragon-list>ul>li:last-child{height:0;border-bottom:none}li.dragon-pop{position:fixed;background-color:#fff;border:1px solid #e0e0e0;left:0;top:0;overflow-x:hidden;box-shadow:rgba(0,0,0,.188235) 0 10px 20px,rgba(0,0,0,.227451) 0 6px 6px}",n.prototype={addEvt:function(t,e,i,n){var o={handler:y[e].bind(t,this),element:i||window};return n||(this.bindings[e]=o),o.element.addEventListener(e,o.handler),o},removeEvt:function(t){var e=this.bindings[t];delete this.bindings[t],e.element.removeEventListener(t,e.handler)},removeAllEventListeners:function(){for(var t in this.bindings){var e=this.bindings[t];e.element.removeEventListener(t,e.handler)}this.items.forEach(function(t){t.handler&&t.element.removeEventListener("mousedown",t.handler)}),this.modelLists.forEach(function(t){t.element.removeEventListener("wheel",h)})},pointInListRects:function(t){return this.modelLists.find(function(e){var i=e.element.getBoundingClientRect();return i={left:window.scrollX+i.left,top:window.scrollY+i.top,right:window.scrollX+i.right,bottom:window.scrollY+i.bottom,width:i.width,height:i.height},e.rect=i,s(t,i)?(e.rect=i,!0):!1})},pointInItemRects:function(t,e,i){return this.items.find(function(n){var o=n.element;return o!==e&&o!==i&&s(t,n.rect)})},getAllItemBoundingRects:function(){var t,e=this.modelLists;this.items.forEach(function(i){var n=i.element,o=n.parentElement,r=e.find(function(t){return t.element===o});if(void 0===r.isDropTarget||"function"==typeof r.isDropTarget&&r.isDropTarget()||r.isDropTarget){var s=n.getBoundingClientRect(),l=s.bottom;n===o.lastElementChild?(l=o.getBoundingClientRect().bottom,l0&&(i.element.scrollTop>0&&(s=u-(i.rect.top+5))<0?o(s,0,i.element):i.element.scrollTop0?o(s,a,i.element):o());var c=t.pointInItemRects({x:e.clientX,y:t.rect.bottom+window.scrollY+r+i.element.scrollTop},this,t.drop);if(this.style[g]=l(t.rect.left-window.scrollX+n,t.rect.top-window.scrollY+r),c){var h=c.element;h.style.transition=w,h.style.borderTopWidth=t.drop.style.borderTopWidth,t.drop.style.borderTopWidth=null,t.drop=h}}},mouseup:function(t,e){o(),t.removeEvt("mousemove"),t.removeEvt("mouseup"),e.stopPropagation();var i=this.getBoundingClientRect();if(window.scrollX+i.left===t.rect.left&&window.scrollY+i.top===t.rect.top)t.reinsert(this);else{var n=t.drop.getBoundingClientRect();t.addEvt(this,"transitionend",this),this.style.transitionDuration=w,this.style.transitionProperty=g,this.style[g]=l(n.left-window.scrollX,n.top-window.scrollY)}},transitionend:function(t,e){if(e.propertyName===g){t.removeEvt("transitionend"),t.reinsert(this),this.style.transitionProperty=w;var i=t.modelLists[t.origin.list].splice(t.origin.item,1)[0],n=t.itemCoordinates(this);t.modelLists[n.list].splice(n.item,0,i),t.callback.dropped&&t.callback.dropped.call(this,t)}}};e.exports=n},{"css-injector":3,templex:22}],16:[function(t,e,i){!function(){function t(t,e){return Object.prototype.hasOwnProperty.call(t,e)}function i(){return 1}function n(t){d||"string"==typeof t||"number"==typeof t||(d=!0,console.error(new TypeError("LRU: key must be a string or number. Almost certainly a bug! "+typeof t).stack))}function o(t){return this instanceof o?("number"==typeof t&&(t={max:t}),t||(t={}),this._max=t.max,(!this._max||"number"!=typeof this._max||this._max<=0)&&(this._max=1/0),this._lengthCalculator=t.length||i,"function"!=typeof this._lengthCalculator&&(this._lengthCalculator=i),this._allowStale=t.stale||!1,this._maxAge=t.maxAge||null,this._dispose=t.dispose,void this.reset()):new o(t)}function r(t,e,i){n(e);var o=t._cache[e];return o&&(s(t,o)?(c(t,o),t._allowStale||(o=void 0)):i&&l(t,o),o&&(o=o.value)),o}function s(t,e){if(!e||!e.maxAge&&!t._maxAge)return!1;var i=!1,n=Date.now()-e.now;return i=e.maxAge?n>e.maxAge:t._maxAge&&n>t._maxAge}function l(t,e){u(t,e),e.lu=t._mru++,t._lruList[e.lu]=e}function a(t){for(;t._lrut._max;)c(t,t._lruList[t._lru])}function u(t,e){for(delete t._lruList[e.lu];t._lru=t)&&(t=1/0),this._max=t,this._length>this._max&&a(this)},get:function(){return this._max},enumerable:!0}),Object.defineProperty(o.prototype,"lengthCalculator",{set:function(t){if("function"!=typeof t){this._lengthCalculator=i,this._length=this._itemCount;for(var e in this._cache)this._cache[e].length=1}else{this._lengthCalculator=t,this._length=0;for(var e in this._cache)this._cache[e].length=this._lengthCalculator(this._cache[e].value),this._length+=this._cache[e].length}this._length>this._max&&a(this)},get:function(){return this._lengthCalculator},enumerable:!0}),Object.defineProperty(o.prototype,"length",{get:function(){return this._length},enumerable:!0}),Object.defineProperty(o.prototype,"itemCount",{get:function(){return this._itemCount},enumerable:!0}),o.prototype.forEach=function(t,e){e=e||this;for(var i=0,n=this._itemCount,o=this._mru-1;o>=0&&n>i;o--)if(this._lruList[o]){i++;var r=this._lruList[o];s(this,r)&&(c(this,r),this._allowStale||(r=void 0)),r&&t.call(e,r.value,r.key,this)}},o.prototype.keys=function(){for(var t=new Array(this._itemCount),e=0,i=this._mru-1;i>=0&&e=0&&e=0&&ethis._max?(c(this,this._cache[e]),!1):(this._dispose&&this._dispose(e,this._cache[e].value),this._cache[e].now=r,this._cache[e].maxAge=o,this._cache[e].value=i,this._length+=s-this._cache[e].length,this._cache[e].length=s,this.get(e),this._length>this._max&&a(this),!0);var l=new h(e,i,this._mru++,s,r,o);return l.length>this._max?(this._dispose&&this._dispose(e,i),!1):(this._length+=l.length,this._lruList[l.lu]=this._cache[e]=l,this._itemCount++,this._length>this._max&&a(this),!0)},o.prototype.has=function(e){if(n(e),!t(this._cache,e))return!1;var i=this._cache[e];return s(this,i)?!1:!0},o.prototype.get=function(t){return n(t),r(this,t,!0)},o.prototype.peek=function(t){return n(t),r(this,t,!1)},o.prototype.pop=function(){var t=this._lruList[this._lru];return c(this,t),t||null},o.prototype.del=function(t){n(t),c(this,this._cache[t])},o.prototype.load=function(t){this.reset();for(var e=Date.now(),i=t.length-1;i>=0;i--){var o=t[i];n(o.k);var r=o.e||0;if(0===r)this.set(o.k,o.v);else{var s=r-e;s>0&&this.set(o.k,o.v,s)}}}}()},{}],17:[function(t,e,i){!function(t,e){"object"==typeof i&&i&&"string"!=typeof i.nodeName?e(i):"function"==typeof define&&define.amd?define(["exports"],e):(t.Mustache={},e(t.Mustache))}(this,function(t){function e(t){return"function"==typeof t}function i(t){return p(t)?"array":typeof t}function n(t){return t.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g,"\\$&")}function o(t,e){return null!=t&&"object"==typeof t&&e in t}function r(t,e){return v.call(t,e)}function s(t){return!r(m,t)}function l(t){return String(t).replace(/[&<>"'`=\/]/g,function(t){return C[t]})}function a(e,i){function o(){if(m&&!C)for(;v.length;)delete f[v.pop()];else v=[];m=!1,C=!1}function r(t){if("string"==typeof t&&(t=t.split(y,2)),!p(t)||2!==t.length)throw new Error("Invalid tags: "+t);l=new RegExp(n(t[0])+"\\s*"),a=new RegExp("\\s*"+n(t[1])),d=new RegExp("\\s*"+n("}"+t[1]))}if(!e)return[];var l,a,d,g=[],f=[],v=[],m=!1,C=!1;r(i||t.tags);for(var E,R,A,M,P,D,k=new h(e);!k.eos();){if(E=k.pos,A=k.scanUntil(l))for(var T=0,F=A.length;F>T;++T)M=A.charAt(T),s(M)?v.push(f.length):C=!0,f.push(["text",M,E,E+1]),E+=1,"\n"===M&&o();if(!k.scan(l))break;if(m=!0,R=k.scan(b)||"name",k.scan(w),"="===R?(A=k.scanUntil(x),k.scan(x),k.scanUntil(a)):"{"===R?(A=k.scanUntil(d),k.scan(S),k.scanUntil(a),R="&"):A=k.scanUntil(a),!k.scan(a))throw new Error("Unclosed tag at "+k.pos);if(P=[R,A,E,k.pos],f.push(P),"#"===R||"^"===R)g.push(P);else if("/"===R){if(D=g.pop(),!D)throw new Error('Unopened section "'+A+'" at '+E);if(D[1]!==A)throw new Error('Unclosed section "'+D[1]+'" at '+E)}else"name"===R||"{"===R||"&"===R?C=!0:"="===R&&r(A)}if(D=g.pop())throw new Error('Unclosed section "'+D[1]+'" at '+k.pos);return c(u(f))}function u(t){for(var e,i,n=[],o=0,r=t.length;r>o;++o)e=t[o],e&&("text"===e[0]&&i&&"text"===i[0]?(i[1]+=e[1],i[3]=e[3]):(n.push(e),i=e));return n}function c(t){for(var e,i,n=[],o=n,r=[],s=0,l=t.length;l>s;++s)switch(e=t[s],e[0]){case"#":case"^":o.push(e),r.push(e),o=e[4]=[];break;case"/":i=r.pop(),i[5]=e[2],o=r.length>0?r[r.length-1][4]:n;break;default:o.push(e)}return n}function h(t){this.string=t,this.tail=t,this.pos=0}function d(t,e){this.view=t,this.cache={".":this.view},this.parent=e}function g(){this.cache={}}var f=Object.prototype.toString,p=Array.isArray||function(t){return"[object Array]"===f.call(t)},v=RegExp.prototype.test,m=/\S/,C={"&":"&","<":"<",">":">",'"':""","'":"'","/":"/","`":"`","=":"="},w=/\s*/,y=/\s+/,x=/\s*=/,S=/\s*\}/,b=/#|\^|\/|>|\{|&|=|!/;h.prototype.eos=function(){return""===this.tail},h.prototype.scan=function(t){var e=this.tail.match(t);if(!e||0!==e.index)return"";var i=e[0];return this.tail=this.tail.substring(i.length),this.pos+=i.length,i},h.prototype.scanUntil=function(t){var e,i=this.tail.search(t);switch(i){case-1:e=this.tail,this.tail="";break;case 0:e="";break;default:e=this.tail.substring(0,i),this.tail=this.tail.substring(i)}return this.pos+=e.length,e},d.prototype.push=function(t){return new d(t,this)},d.prototype.lookup=function(t){var i,n=this.cache;if(n.hasOwnProperty(t))i=n[t];else{for(var r,s,l=this,a=!1;l;){if(t.indexOf(".")>0)for(i=l.view,r=t.split("."),s=0;null!=i&&sa;++a)s=void 0,o=t[a],r=o[0],"#"===r?s=this.renderSection(o,e,i,n):"^"===r?s=this.renderInverted(o,e,i,n):">"===r?s=this.renderPartial(o,e,i,n):"&"===r?s=this.unescapedValue(o,e):"name"===r?s=this.escapedValue(o,e):"text"===r&&(s=this.rawValue(o)),void 0!==s&&(l+=s);return l},g.prototype.renderSection=function(t,i,n,o){function r(t){return s.render(t,i,n)}var s=this,l="",a=i.lookup(t[1]);if(a){if(p(a))for(var u=0,c=a.length;c>u;++u)l+=this.renderTokens(t[4],i.push(a[u]),n,o);else if("object"==typeof a||"string"==typeof a||"number"==typeof a)l+=this.renderTokens(t[4],i.push(a),n,o);else if(e(a)){if("string"!=typeof o)throw new Error("Cannot use higher-order sections without the original template");a=a.call(i.view,o.slice(t[3],t[5]),r),null!=a&&(l+=a)}else l+=this.renderTokens(t[4],i,n,o);return l}},g.prototype.renderInverted=function(t,e,i,n){var o=e.lookup(t[1]);return!o||p(o)&&0===o.length?this.renderTokens(t[4],e,i,n):void 0},g.prototype.renderPartial=function(t,i,n){if(n){var o=e(n)?n(t[1]):n[t[1]];return null!=o?this.renderTokens(this.parse(o),i,n,o):void 0}},g.prototype.unescapedValue=function(t,e){var i=e.lookup(t[1]);return null!=i?i:void 0},g.prototype.escapedValue=function(e,i){var n=i.lookup(e[1]);return null!=n?t.escape(n):void 0},g.prototype.rawValue=function(t){return t[1]},t.name="mustache.js",t.version="2.2.1",t.tags=["{{","}}"];var E=new g;t.clearCache=function(){return E.clearCache()},t.parse=function(t,e){return E.parse(t,e)},t.render=function(t,e,n){if("string"!=typeof t)throw new TypeError('Invalid template! Template should be a "string" but "'+i(t)+'" was given as the first argument for mustache#render(template, view, partials)');return E.render(t,e,n)},t.to_html=function(i,n,o,r){var s=t.render(i,n,o);return e(r)?void r(s):s},t.escape=l,t.Scanner=h,t.Context=d,t.Writer=g})},{}],18:[function(t,e,i){"use strict";function n(t){return t instanceof n?t:this instanceof n?(this.originalValue=t,void(this.o=t||{})):new n(t)}n.chain=function(t){var e=n(t);return e.chaining=!0,e},n.prototype={value:function(){return this.originalValue},each:function(t,e){var i=this.o;return Object.keys(i).forEach(function(e){t.call(this,i[e],e,i)},e||i),this},find:function(t,e){var i,n=this.o;return n&&(i=Object.keys(n).find(function(e){return t.call(this,n[e],e,n)},e||n),void 0!==i&&(i=n[i])),i},filter:function(t,e){var i=this.o,n=[];return i&&Object.keys(i).forEach(function(e){t.call(this,i[e],e,i)&&n.push(i[e])},e||i),n},map:function(t,e){var i=this.o,n=[];return i&&Object.keys(i).forEach(function(e){n.push(t.call(this,i[e],e,i))},e||i),n},reduce:function(t,e,i){var n=this.o;return n&&Object.keys(n).forEach(function(i,o){e=o||void 0!==e?t(e,n[i],i,n):n[i]},i||n),e},extend:function(t){var e=this.o;return Array.prototype.slice.call(arguments).forEach(function(t){if(t)for(var i in t)e[i]=t[i]}),this.chaining?this:e},extendOwn:function(t){var e=this.o;return Array.prototype.slice.call(arguments).forEach(function(t){n(t).each(function(t,i){e[i]=t})}),this.chaining?this:e}},Array.prototype.find||(Array.prototype.find=function(t){if(null===this)throw new TypeError("Array.prototype.find called on null or undefined");if("function"!=typeof t)throw new TypeError("predicate must be a function");for(var e,i=Object(this),n=i.length>>>0,o=arguments[1],r=0;n>r;r++)if(e=i[r],t.call(o,e,r,i))return e}),e.exports=n},{}],19:[function(t,e,i){"use strict";function n(t,e){Object.defineProperty(this,t,{value:e,writable:!1,enumerable:!0,configurable:!1})}function o(t,e){n.call(this,"x",Number(t)||0),n.call(this,"y",Number(e)||0)}function r(t,e,i,r){t=Number(t)||0,e=Number(e)||0,i=Number(i)||0,r=Number(r)||0,0>i&&(t+=i,i=-i),0>r&&(e+=r,r=-r),n.call(this,"origin",new o(t,e)),n.call(this,"extent",new o(i,r)),n.call(this,"corner",new o(t+i,e+r)),n.call(this,"center",new o(t+i/2,e+r/2))}o.prototype={plus:function(t){return new o(this.x+t.x,this.y+t.y)},plusXY:function(t,e){return new o(this.x+(t||0),this.y+(e||0))},minus:function(t){return new o(this.x-t.x,this.y-t.y)},min:function(t){return new o(Math.min(this.x,t.x),Math.min(this.y,t.y))},max:function(t){return new o(Math.max(this.x,t.x),Math.max(this.y,t.y))},distance:function(t){var e=t.x-this.x,i=t.y-this.y;return Math.sqrt(e*e+i*i)},equals:function(t){var e=!1;return t&&(e=this.x===t.x&&this.y===t.y),e},greaterThan:function(t){return this.x>t.x&&this.y>t.y},lessThan:function(t){return this.x=t.x&&this.y>=t.y},lessThanOrEqualTo:function(t){return this.x<=t.x&&this.y<=t.y},within:function(t){var e=t.origin.x,i=e+t.extent.x,n=t.origin.y,o=n+t.extent.y;return t.extent.x<0&&(e=i,i=t.origin.x),t.extent.y<0&&(n=o,o=t.origin.y),e<=this.x&&this.xi;i++)for(var o=this.origin.y,r=this.corner.y;r>o;o++)t.call(e,i,o)},intersect:function(t,e,i){var n=null,o=this.origin.max(t.origin),s=this.corner.min(t.corner),l=s.minus(o);return l.x>0&&l.y>0?n=new r(o.x,o.y,l.x,l.y):"function"==typeof e&&(n=e.call(i||this,t)),n},intersects:function(t){return t.corner.x>this.origin.x&&t.corner.y>this.origin.y&&t.origin.x\|\:\[\]])/g,a=".*",u=".",c="("+a+")",h="_",d="%",g=new RegExp("("+[h,d,"\\[\\^?[^-\\]]+]","\\[\\^?[^-\\]]\\-[^\\]]]"].join("|")+")","g");(o.clearCache=function(t){return t?r[t]&&(delete r[t],s--):(r={},s=0),s})(),o.getCacheSize=function(){return s},o.cached=function(t,e){var i=r[t];if(i)i.when=(new Date).getTime(),void 0!==e&&(i.keep=e);else{if(s===o.cacheMax){var n,l,a=[],u=0;for(n in r)if(i=r[n],!i.keep){for(l=0;u>l&&!(i.when=t?[t,e]:[e,t]}function n(t,e){return t[0]<=e[0]&&e[0]<=t[1]||t[0]<=e[1]&&e[1]<=t[1]||e[0]=o&&n>r?s.push([r+1,n]):o>i&&r>=n?s.push([i,o-1]):o>i&&n>r?(s.push([i,o-1]),s.push([r+1,n])):(i>r||o>n)&&s.push(t),s}function s(t,e){var i=Math.min(Math.min.apply(Math,t),Math.min.apply(Math,e)),n=Math.max(Math.max.apply(Math,t),Math.max.apply(Math,e));return[i,n]}e.prototype={select:function(t,e){this.storeState();var r=i(t,e),l=[0,1];return this.selection.forEach(function(t){n(t,r)||o(t,r)?r=s(t,r):l.push(t)}),l.push(r),l[1]=this.selection.length,this.selection.splice.apply(this.selection,l),this},deselect:function(t,e){var o=i(t,e),s=[0,0];return this.selection.forEach(function(t){if(n(t,o)){var e=r(t,o);s=s.concat(e)}else s.push(t)}),s[1]=this.selection.length,this.selection.splice.apply(this.selection,s),this},clear:function(){return this.states.length=0,this.selection.length=0,this},clearMostRecentSelection:function(){0!==this.states.length&&(this.selection=this.states.pop())},isSelected:function(t){return this.selection.some(function(e){return e[0]<=t&&t<=e[1]})},isEmpty:function(){return 0===this.selection.length},getSelections:function(){var t=[];return this.selection.forEach(function(e){for(var i=e[0];i<=e[1];i++)t.push(i)}),t.sort(function(t,e){return t-e}),t}},t.exports=e})("object"==typeof e&&e||(window.RangeSelectionModel={}),"object"==typeof e&&e.exports||(window.RangeSelectionModel.exports={}))||"object"==typeof e||(window.RangeSelectionModel=window.RangeSelectionModel.exports)},{}],22:[function(require,module,exports){function templex(t){var e=this instanceof Array?this:[this];return arguments.length>1&&e.unshift(arguments),t.replace(templex.regexp,templex.merger.bind(e))}templex.regexp=/\{(.*?)\}/g,templex["with"]=function(t,e){return"with(this["+t+"]){"+e+"}"},templex.cache=[],templex.deref=function(key){if(!(this.length in templex.cache)){for(var code="return eval(expr)",i=0;i1&&i.unshift(r.pop()),r=[r.join(" ")],i.length&&(r=r.concat(s(t,e,i,n))),r}function l(t,e,i,n){return s(t,e,a(i).split(" "),n)}function a(t){return t.toString().trim().replace(/\s\s+/g," ")}function u(t,e,i,n,o,r,s,l){l||(l=!0),r||(r=5),t.beginPath(),t.moveTo(e+r,i),t.lineTo(e+n-r,i),t.quadraticCurveTo(e+n,i,e+n,i+r),t.lineTo(e+n,i+o-r),t.quadraticCurveTo(e+n,i+o,e+n-r,i+o),t.lineTo(e+r,i+o),t.quadraticCurveTo(e,i+o,e,i+o-r),t.lineTo(e,i+r),t.quadraticCurveTo(e,i,e+r,i),t.closePath(),l&&t.stroke(),s&&t.fill(),t.closePath()}var c=t("extend-me").Base,h=c.extend("CellProvider",{initialize:function(){this.cellCache={},this.initializeCells()},getCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},getColumnHeaderCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},getRowHeaderCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},paintButton:function(t,e){var i=e.value,n=e.x,o=e.y,r=e.bounds,s=r.x+2,l=r.y+2,a=r.width-3,c=r.height-3,h=c/2,d=t.createLinearGradient(s,l,s,l+c); -e.mouseDown?(d.addColorStop(0,"#B5CBED"),d.addColorStop(1,"#4d74ea")):(d.addColorStop(0,"#ffffff"),d.addColorStop(1,"#aaaaaa")),t.fillStyle=d,t.strokeStyle="#000000",u(t,s,l,a,c,h,d,!0);var g=(a-e.getTextWidth(t,i))/2,f=(c-e.getTextHeight(t.font).descent)/2;"middle"!==t.textBaseline&&(t.textBaseline="middle"),t.fillStyle="#000000",e.backgroundColor="rgba(0,0,0,0)",t.fillText(i,s+g,l+f),e.buttonCells[n+","+o]=!0},defaultCellPaint:function(t,e){var i,o,r,s,l,a=e.value,u=e.bounds.x,c=e.bounds.y,h=e.bounds.width,d=e.bounds.height,g=e.headerTextWrapping,f=2,p=0===e.y;a&&a.constructor===Array&&(i=a[0],o=a[2],a=a[1],a&&"object"==typeof a&&"HTMLImageElement"===a.constructor.name&&(r=a,a=null),i&&"IMG"!==i.nodeName&&(i=null),o&&"IMG"!==o.nodeName&&(o=null),r&&"IMG"!==r.nodeName&&(r=null)),a=n(e,a),a=e.formatter(a),t.font!==e.font&&(t.font=e.font),"left"!==t.textAlign&&(t.textAlign="left"),"middle"!==t.textBaseline&&(t.textBaseline="middle");var v;e.backgroundColor&&(t.fillStyle=v=n(e,e.isSelected?e.backgroundSelectionColor:e.backgroundColor),e.isColumnHovered&&(t.fillStyle=e.hoverColumnColor),e.isRowHovered&&(t.fillStyle=e.hoverRowColor),e.isCellHovered&&(t.fillStyle=e.hoverCellColor),t.fillRect(u,c,h,d));var m=n(e,e.isSelected?e.foregroundSelectionColor:e.color);t.fillStyle!==m&&(t.fillStyle=m,t.strokeStyle=m),p&&g?this.renderMultiLineText(t,u,c,d,h,e,a):this.renderSingleLineText(t,u,c,d,h,e,a);var C=0;if(i&&(l=Math.round((d-i.height)/2),t.drawImage(i,u+f,c+l),C=Math.max(i.width+2)),o&&h>1.75*d){l=Math.round((d-o.height)/2);var w=u+h-o.width;v?(t.fillStyle=v,t.fillRect(w,c,o.width,d)):t.clearRect(w,c,o.width,d),t.drawImage(o,w,c+l),C=Math.max(o.width+2)}r&&(l=Math.round((d-r.height)/2),s=Math.round((h-r.width)/2),t.drawImage(r,u+h-s-r.width,c+l),C=Math.max(r.width+2)),e.cellBorderThickness&&(t.beginPath(),t.rect(u,c,h,d),t.lineWidth=e.cellBorderThickness,t.strokeStyle=e.cellBorderStyle,t.stroke(),t.closePath()),e.minWidth=e.minWidth+2*C},renderMultiLineText:function(t,e,i,n,o,r,s){var u=l(t,r,s,o);if(1===u.length)return this.renderSingleLineText(t,e,i,n,o,r,a(s));var c=r.cellPadding,h=0,d=r.voffset,g=r.halign,f=r.getTextHeight(r.font).height;switch(g){case"right":h=o-c;break;case"center":h=o/2;break;case"left":h=c}var p=0,v=Math.ceil(f/2);d+=Math.ceil((n-(u.length-1)*f)/2),h=Math.max(p,h),d=Math.max(v,d),t.save(),t.rect(e,i,o,n),t.clip(),t.textAlign=g;for(var m=0;mt?"0"+t:t+""};e.exports={date:function(t){var e=t.getFullYear()+"-"+n(t.getMonth()+1)+"-"+n(t.getDay());return e},"default":function(t){return t+""}}},{}],25:[function(t,e,i){"use strict";function n(t,e,i){var n=this;this.div="string"==typeof t?document.querySelector(t):t,m("grid"),this.lastEdgeSelection=[0,0],this.lnfProperties=Object.create(S),this.isWebkit=navigator.userAgent.toLowerCase().indexOf("webkit")>-1,this.selectionModel=new v,this.localCellEditors={},this.selectionModel.getGrid=function(){return n},this.cellEditors=Object.create(this.localCellEditors),this.renderOverridesCache={},this.behavior=e(this),this.div.oncontextmenu=function(t){return t.preventDefault(),!1},this.clearMouseDown(),this.dragExtent=new h(0,0),this.numRows=0,this.numColumns=0,this.pluginsDo(function(t){t.installOn&&t.installOn(n)}),i=i||{},i.top=i.top||0,i.right=i.right||"-200px",i.bottom=i.bottom||0,i.left=i.left||0,y||(y=!0,r()),this.initRenderer(),this.initCanvas(i),this.initScrollbars(),this.initLocalCellEditors(),document.body.addEventListener("copy",function(t){n.checkClipboardCopy(t)}),this.getCanvas().resize(),this.dialog=new C(this)}function o(t){var e=t.origin,i=t.corner,n=Math.min(e.x,i.x),o=Math.min(e.y,i.y),r=Math.max(e.x,i.x),s=Math.max(e.y,i.y),l=new d(n,o,r-n,s-o);return l}function r(){s(x);var t=document.createElement("paper-button");t.style.display="none",t.setAttribute("disabled",!0),document.body.appendChild(t);var e=window.getComputedStyle(t),i=document.createElement("section");i.style.display="none",i.setAttribute("hero",!0),document.body.appendChild(i);var n=window.getComputedStyle(document.querySelector("html")),o=window.getComputedStyle(document.querySelector("html, body")),r=window.getComputedStyle(i);x.columnHeaderBackgroundColor=e.color,x.rowHeaderBackgroundColor=e.color,x.topLeftBackgroundColor=e.color,x.lineColor=e.backgroundColor,x.backgroundColor2=o.backgroundColor,x.color=n.color,x.fontFamily=n.fontFamily,x.backgroundColor=r.backgroundColor,t.setAttribute("disabled",!1),t.setAttribute("secondary",!0),t.setAttribute("raised",!0),e=window.getComputedStyle(t),x.columnHeaderColor=e.color,x.rowHeaderColor=e.color,x.topLeftColor=e.color,x.backgroundSelectionColor=e.backgroundColor,x.foregroundSelectionColor=e.color,t.setAttribute("secondary",!1),t.setAttribute("warning",!0),x.columnHeaderForegroundSelectionColor=e.color,x.columnHeaderBackgroundSelectionColor=e.backgroundColor,x.rowHeaderForegroundSelectionColor=e.color,x.fixedColumnBackgroundSelectionColor=e.backgroundColor,("rgba(0, 0, 0, 0)"===x.columnHeaderBackgroundSelectionColor||"transparent"===x.lineColor)&&s(x),document.body.removeChild(t),document.body.removeChild(i)}function s(t){for(var e in t)t.hasOwnProperty(e)&&delete t[e]}function l(t){var e="function"==typeof t?t():t;return e||0===e?e:""}var a=t("extend-me");a.debug=!0;var u=t("finbars"),c=t("fincanvas"),h=t("rectangular").Point,d=t("rectangular").Rectangle,g=t("object-iterators"),f=t("./defaults"),p=t("./Renderer"),v=t("./SelectionModel"),m=t("./stylesheets"),C=t("./TableDialog"),w=t("./Formatters"),y=!1,x=Object.create(f),S=Object.create(x),b={};n.prototype={constructor:n.prototype.constructor,behavior:null,isWebkit:!0,mouseDown:[],dragExtent:null,vScrollValue:0,hScrollValue:0,rectangular:null,selectionModel:null,cellEditor:null,sbHScroller:null,sbVScroller:null,sbPrevVScrollValue:null,sbPrevHScrollValue:null,cellEditors:null,renderOverridesCache:{},hoverCell:null,scrollingNow:!1,lastEdgeSelection:null,setAttribute:function(t,e){this.div.setAttribute(t,e)},reset:function(){var t=this;this.lastEdgeSelection=[0,0],this.lnfProperties=Object.create(S),this.selectionModel=new v,this.selectionModel.getGrid=function(){return t},this.cellEditors=Object.create(this.localCellEditors),this.renderOverridesCache={},this.clearMouseDown(),this.dragExtent=new h(0,0),this.numRows=0,this.numColumns=0,this.vScrollValue=0,this.hScrollValue=0,this.cellEditor=null,this.sbPrevVScrollValue=null,this.sbPrevHScrollValue=null,this.hoverCell=null,this.scrollingNow=!1,this.lastEdgeSelection=[0,0],this.getBehavior().reset(),this.getRenderer().reset(),this.getCanvas().resize(),this.behaviorChanged()},getProperties:function(){return this.getPrivateState()},_getProperties:function(){return this.lnfProperties},computeCellsBounds:function(){var t=this.getRenderer();t&&t.computeCellsBounds()},initCellEditor:function(t){this.localCellEditors[t.alias]=t,t.grid=this},initLocalCellEditors:function(){var t=["Textfield","Choice","Color","Date","Slider","Spinner","Filter"],e=this;t.forEach(function(t){e.initCellEditor(new n.cellEditors[t])}),this.localCellEditors["int"]=this.localCellEditors.spinner,this.localCellEditors["float"]=this.localCellEditors.spinner,this.localCellEditors.date=this.localCellEditors.date,this.localCellEditors.string=this.localCellEditors.extfield},toggleColumnPicker:function(){this.getBehavior().toggleColumnPicker()},isHovered:function(t,e){var i=this.getHoverCell();return i?i.x===t&&i.y===e:!1},registerFormatter:function(t,e){w[t]=e},getFormatter:function(t){var e=w[t];return e?e:w["default"]},formatValue:function(t,e){var i=this.getFormatter(t);return i(e)},isColumnHovered:function(t){var e=this.getHoverCell();return e?e.x===t:!1},isRowResizeable:function(){return this.resolveProperty("rowResize")},isCheckboxOnlyRowSelections:function(){return this.resolveProperty("checkboxOnlyRowSelections")},isRowHovered:function(t){var e=this.getHoverCell();return e?e.y===t:!1},getHoverCell:function(){return this.hoverCell},setHoverCell:function(t){var e=this.hoverCell,i=new h(t.x,t.y);e&&e.equals(i)||(this.hoverCell=i,this.fireSyntheticOnCellEnterEvent(i),this.repaint())},addGlobalProperties:function(t){if(S)this._addGlobalProperties(t);else{var e=this;setTimeout(function(){e.addGlobalProperties(t)},10)}},_addGlobalProperties:function(t){g(t).each(function(t,e){S[e]=t})},addProperties:function(t){var e=this.getProperties();g(t).each(function(i,n){e[n]=t[n]}),this.refreshProperties()},refreshProperties:function(){this.checkScrollbarVisibility(),this.getBehavior().defaultRowHeight=null,this.isColumnAutosizing()&&this.getBehavior().autosizeAllColumns()},getPrivateState:function(){return this.getBehavior().getPrivateState()},setState:function(t){var e=this;this.getBehavior().setState(t),setTimeout(function(){e.behaviorChanged(),e.synchronizeScrollingBoundries()},100)},getState:function(){return this.getBehavior().getState()},getMouseDown:function(){var t=this.mouseDown.length-1;return 0>t?null:this.mouseDown[t]},popMouseDown:function(){0!==this.mouseDown.length&&(this.mouseDown.length=this.mouseDown.length-1)},clearMouseDown:function(){this.mouseDown=[new h(-1,-1)],this.dragExtent=null},setMouseDown:function(t){this.mouseDown.push(t)},getDragExtent:function(){return this.dragExtent},setDragExtent:function(t){this.dragExtent=t},pluginsDo:function(t){},getCellProvider:function(){var t=this.getBehavior().getCellProvider();return t},gridRenderedNotification:function(){this.updateRenderedSizes(),this.cellEditor&&this.cellEditor.gridRenderedNotification(),this.checkColumnAutosizing(),this.fireSyntheticGridRenderedEvent()},checkColumnAutosizing:function(){var t=this.getBehavior();t.autoSizeRowNumberColumn(),this.isColumnAutosizing()&&t.checkColumnAutosizing(!1)},updateRenderedSizes:function(){var t=this.getBehavior();t.setRenderedColumnCount(this.getVisibleColumns()+1),t.setRenderedRowCount(this.getVisibleRows()+1)},checkClipboardCopy:function(t){if(this.hasFocus()){t.preventDefault();var e=this.getSelectionAsTSV();t.clipboardData.setData("text/plain",e)}},hasSelections:function(){return this.getSelectionModel?this.getSelectionModel().hasSelections():void 0},getSelectionAsTSV:function(){var t=this.getSelectionModel();if(t.hasSelections()){var e=this.getSelectionMatrix();return e=e[e.length-1],this.getMatrixSelectionAsTSV(e)}return t.hasRowSelections()?this.getMatrixSelectionAsTSV(this.getRowSelectionMatrix()):t.hasColumnSelections()?this.getMatrixSelectionAsTSV(this.getColumnSelectionMatrix()):void 0},getMatrixSelectionAsTSV:function(t){if(t.length){var e=t.length,i=t[0].length,n=e*i,o=[];if(n>2e4)return alert("selection size is too big to copy to the paste buffer"),"";for(var r=0;i>r;r++){for(var s=0;e>s;s++)o.push(t[s][r]),e>s&&o.push(" ");i>r&&o.push("\n")}var l=o.join("");return l}return""},hasFocus:function(){return this.getCanvas().hasFocus()},clearSelections:function(){var t=this.isCheckboxOnlyRowSelections();this.getSelectionModel().clear(t),this.clearMouseDown()},clearMostRecentSelection:function(){var t=this.isCheckboxOnlyRowSelections();this.getSelectionModel().clearMostRecentSelection(t)},clearMostRecentColumnSelection:function(){this.getSelectionModel().clearMostRecentColumnSelection()},clearMostRecentRowSelection:function(){this.getSelectionModel().clearMostRecentRowSelection()},select:function(t,e,i,n){0>t||0>e||this.getSelectionModel().select(t,e,i,n)},isSelected:function(t,e){return this.getSelectionModel().isSelected(t,e)},isCellSelectedInRow:function(t){var e=this.getSelectionModel(),i=e.isCellSelectedInRow(t);return i},isCellSelectedInColumn:function(t){var e=this.getSelectionModel(),i=e.isCellSelectedInColumn(t);return i},getSelectionModel:function(){return this.selectionModel},getBehavior:function(){return this.behavior},setBehavior:function(t){this.behavior=t,this.behavior.setGrid(this),this.behavior.changed=this.behaviorChanged.bind(this),this.behavior.shapeChanged=this.behaviorShapeChanged.bind(this),this.behavior.stateChanged=this.behaviorStateChanged.bind(this)},behaviorChanged:function(){(this.numColumns!==this.getColumnCount()||this.numRows!==this.getRowCount())&&(this.numColumns=this.getColumnCount(),this.numRows=this.getRowCount(),this.behaviorShapeChanged()),this.computeCellsBounds(),this.repaint()},getBounds:function(){var t=this.getRenderer();if(t)return t.getBounds()},resolveProperty:function(t){return this.getProperties()[t]},behaviorShapeChanged:function(){this.synchronizeScrollingBoundries()},behaviorStateChanged:function(){this.getRenderer().computeCellsBounds(),this.repaint()},repaint:function(){var t=this.resolveProperty("repaintImmediately"),e=this.getCanvas();e&&(t===!0?e.paintNow():e.repaint())},paintNow:function(){var t=this.getCanvas();t.paintNow()},useHiDPI:function(){return this.resolveProperty("useHiDPI")!==!1},initCanvas:function(t){var e=this,i=this.divCanvas=document.createElement("div");this.div.appendChild(i),this.canvas=new c(i,this.renderer);var n=i.style;n.position="absolute",n.top=t.top,n.right=t.right,n.bottom=t.bottom,n.left=t.left,this.canvas.resizeNotification=function(){e.resized()},this.addFinEventListener("fin-canvas-mousemove",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseMove(n)}}),this.addFinEventListener("fin-canvas-mousedown",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.keys=t.detail.keys,n.primitiveEvent=t,e.mouseDownState=n,e.delegateMouseDown(n),e.fireSyntheticMouseDownEvent(n),e.repaint()}}),this.addFinEventListener("fin-canvas-mouseup",function(t){if(!e.resolveProperty("readOnly")){e.dragging=!1,e.isScrollingNow()&&e.setScrollingNow(!1),e.columnDragAutoScrolling&&(e.columnDragAutoScrolling=!1);var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseUp(n),e.mouseDownState&&e.fireSyntheticButtonPressedEvent(e.mouseDownState),e.mouseDownState=null,e.fireSyntheticMouseUpEvent(n)}}),this.addFinEventListener("fin-canvas-dblclick",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.fireSyntheticDoubleClickEvent(n,t),e.delegateDoubleClick(n)}}),this.addFinEventListener("fin-canvas-tap",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,n.keys=t.detail.keys,e.fireSyntheticClickEvent(n),e.delegateTap(n)}}),this.addFinEventListener("fin-canvas-drag",function(t){if(!e.resolveProperty("readOnly")){e.dragging=!0;var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseDrag(n)}}),this.addFinEventListener("fin-canvas-keydown",function(t){var i=t.detail["char"];-1!==["DELETE"].indexOf(i)&&(e.isEditing()||setTimeout(function(){e.takeFocus()},50)),e.resolveProperty("readOnly")||(e.fireSyntheticKeydownEvent(t),e.delegateKeyDown(t))}),this.addFinEventListener("fin-canvas-keyup",function(t){e.resolveProperty("readOnly")||(e.fireSyntheticKeyupEvent(t),e.delegateKeyUp(t))}),this.addFinEventListener("fin-canvas-track",function(t){if(!e.resolveProperty("readOnly")&&!e.dragging){var i=t.detail.primitiveEvent;Math.abs(i.dy)>Math.abs(i.dx)?i.yDirection>0?e.scrollVBy(-2):i.yDirection<-0&&e.scrollVBy(2):i.xDirection>0?e.scrollHBy(-1):i.xDirection<-0&&e.scrollHBy(1)}}),this.addFinEventListener("fin-canvas-wheelmoved",function(t){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateWheelMoved(n)}),this.addFinEventListener("fin-canvas-mouseout",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateMouseExit(n)}}),this.addFinEventListener("fin-canvas-context-menu",function(t){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateContextMenu(n)}),this.div.removeAttribute("tabindex")},convertViewPointToDataPoint:function(t){return this.getBehavior().convertViewPointToDataPoint(t)},convertDataPointToViewPoint:function(t){return this.getBehavior().convertDataPointToViewPoint(t)},addFinEventListener:function(t,e){this.canvas.addEventListener(t,e)},setScrollingNow:function(t){this.scrollingNow=t},isScrollingNow:function(){return this.scrollingNow},overColumnDivider:function(t){var e=t.primitiveEvent.detail.mouse.x,i=this.getRenderer().overColumnDivider(e);return i},overRowDivider:function(t){var e=t.primitiveEvent.detail.mouse.y,i=this.getRenderer().overRowDivider(e);return i},beCursor:function(t){t||(t="default"),this.div.style.cursor=t},delegateWheelMoved:function(t){var e=this.getBehavior();e.onWheelMoved(this,t)},delegateMouseExit:function(t){var e=this.getBehavior();e.handleMouseExit(this,t)},delegateContextMenu:function(t){var e=this.getBehavior();e.onContextMenu(this,t)},delegateMouseMove:function(t){var e=this.getBehavior();e.onMouseMove(this,t)},delegateMouseDown:function(t){var e=this.getBehavior();e.handleMouseDown(this,t)},delegateMouseUp:function(t){var e=this.getBehavior();e.onMouseUp(this,t)},delegateTap:function(t){var e=this.getBehavior();e.onTap(this,t)},delegateMouseDrag:function(t){var e=this.getBehavior();e.onMouseDrag(this,t)},delegateDoubleClick:function(t){var e=this.getBehavior();e.onDoubleClick(this,t)},delegateHoldPulse:function(t){var e=this.getBehavior();e.onHoldPulse(this,t)},delegateKeyDown:function(t){var e=this.getBehavior();e.onKeyDown(this,t)},delegateKeyUp:function(t){var e=this.getBehavior();e.onKeyUp(this,t)},stopEditing:function(){this.cellEditor&&this.isEditing()&&(this.cellEditor.stopEditing&&this.cellEditor.stopEditing(),this.cellEditor=null)},registerCellEditor:function(t,e){this.cellEditors[t]=e},getDataBounds:function(){var t=200,e=this.canvas.bounds,i=new d(0,0,e.origin.x+e.extent.x-t,e.origin.y+e.extent.y);return i},getRowNumbersWidth:function(){return this.isShowRowNumbers()?this.getRenderer().getRowNumbersWidth():0},getCanvas:function(){return this.canvas},editAt:function(t,e){this.cellEditor=t;var i=e.gridCell,n=i.x,o=i.y;if(!(0>n||0>o)){var r=new h(n,o);this.setMouseDown(r),this.setDragExtent(new h(0,0)),t.beginEditAt(r)}},isColumnVisible:function(t){var e=this.getRenderer().isColumnVisible(t);return e},isDataRowVisible:function(t){var e=this.getRenderer().isRowVisible(t);return e},isDataVisible:function(t,e){var i=this.isDataRowVisible(e)&&this.isColumnVisible(t);return i},insureModelColIsVisible:function(t,e){var i=this.getColumnCount()-1,n=t;return e>0&&n++,this.isColumnVisible(n)&&t!==i?!1:(this.scrollBy(e,0),!0)},insureModelRowIsVisible:function(t,e){var i=this.getRowCount()-1,n=t;return e>0&&n++,this.isDataRowVisible(n)&&t!==i?!1:(this.scrollBy(0,e),!0)},scrollBy:function(t,e){this.scrollHBy(t),this.scrollVBy(e)},scrollVBy:function(t){var e=this.sbVScroller.range.max,i=this.getVScrollValue(),n=Math.min(e,Math.max(0,i+t));n!==i&&this.setVScrollValue(n)},scrollHBy:function(t){var e=this.sbHScroller.range.max,i=this.getHScrollValue(),n=Math.min(e,Math.max(0,i+t));n!==i&&this.setHScrollValue(n)},getGridCellFromMousePoint:function(t){var e=this.getRenderer().getGridCellFromMousePoint(t);return e},getBoundsOfCell:function(t){var e=this.getRenderer().getBoundsOfCell(t),i=new d(e.x,e.y,e.width,e.height);return i},resized:function(){this.synchronizeScrollingBoundries()},cellClicked:function(t){var e=t.gridCell,i=this.getColumnCount(),n=this.getRowCount();if(!(e.x>i||e.y>n)){var o=this.getHoverCell(),r=this.getVScrollValue(),s=o.x;o.y<0&&(r=0),o=new h(s,o.y+r),this.getBehavior().cellClicked(o,t)}},setTotalsValueNotification:function(t,e,i,n){this.fireSyntheticSetTotalsValue(t,e,i,n)},fireSyntheticSetTotalsValue:function(t,e,i,n){var o=new CustomEvent("fin-set-totals-value",{detail:{x:t,y:e,value:i,area:n?"bottom":"top"}});this.canvas.dispatchEvent(o)},fireSyntheticEditorKeyUpEvent:function(t,e){var i=new CustomEvent("fin-editor-key-up",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorKeyDownEvent:function(t,e){var i=new CustomEvent("fin-editor-key-down",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorKeyPressEvent:function(t,e){var i=new CustomEvent("fin-editor-key-press",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorDataChangeEvent:function(t,e,i){var n=new CustomEvent("fin-editor-data-change",{detail:{input:t,oldValue:e,newValue:i},cancelable:!0});return this.canvas.dispatchEvent(n)},fireSyntheticRowSelectionChangedEvent:function(){var t=new CustomEvent("fin-row-selection-changed",{detail:{rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(t)},fireSyntheticColumnSelectionChangedEvent:function(){var t=new CustomEvent("fin-column-selection-changed",{detail:{rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(t)},selectionChanged:function(){var t=this.getSelectedRows(),e=new CustomEvent("fin-selection-changed",{detail:{rows:t,columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(e)},getRowSelection:function(){function t(t,o){i[o]=l(n.getValue(e,t))}var e,i,n=this,o=this.getSelectionModel().getSelectedRows(),r=this.getColumnCount(),s={};for(e=0;r>e;e++)i=new Array(o.length),s[this.getField(e)]=i,o.forEach(t);return s},getRowSelectionMatrix:function(){function t(t,n){r[e][n]=l(i.getValue(e,t))}var e,i=this,n=this.getSelectionModel().getSelectedRows(),o=this.getColumnCount(),r=new Array(o);for(e=0;o>e;e++)r[e]=new Array(n.length),n.forEach(t);return r},getColumnSelectionMatrix:function(){var t=this.getSelectedColumns(),e=this.getRowCount(),i=new Array(t.length),n=this;return t.forEach(function(t,o){i[o]=new Array(e);for(var r=0;e>r;r++)i[o][r]=l(n.getValue(t,r))}),i},getColumnSelection:function(){var t=this.getSelectedColumns(),e={},i=this.getRowCount(),n=this;return t.forEach(function(t){var o=new Array(i);e[n.getField(t)]=o;for(var r=0;i>r;r++)o[r]=l(n.getValue(t,r))}),e},getSelection:function(){var t=this,e=this.getSelections(),i=new Array(e.length);return e.forEach(function(e,n){i[n]=t._getSelection(e)}),i},_getSelection:function(t){t=o(t);for(var e,i=t.extent.x+1,n=t.extent.y+1,r=t.origin.x,s=t.origin.y,a={},u=0;i>u;u++){var c=new Array(n);for(a[this.getField(u+r)]=c,e=0;n>e;e++)c[e]=l(this.getValue(r+u,s+e))}return a},getSelectionMatrix:function(){var t=this,e=this.getSelections(),i=new Array(e.length);return e.forEach(function(e,n){i[n]=t._getSelectionMatrix(e)}),i},_getSelectionMatrix:function(t){t=o(t);for(var e=t.extent.x+1,i=t.extent.y+1,n=t.origin.x,r=t.origin.y,s=[],a=0;e>a;a++){var u=new Array(i);s[a]=u;for(var c=0;i>c;c++)u[c]=l(this.getValue(n+a,r+c))}return s},fireSyntheticContextMenuEvent:function(t){t.gridCell=this.convertViewPointToDataPoint(t.gridCell);var e=new CustomEvent("fin-context-menu",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(e)},fireSyntheticMouseUpEvent:function(t){var e=new CustomEvent("fin-mouseup",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(e)},fireSyntheticMouseDownEvent:function(t){var e=new CustomEvent("fin-mousedown",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.getSelectionModel().getSelections()}});this.canvas.dispatchEvent(e)},isViewableButton:function(t,e){return this.getRenderer().isViewableButton(t,e)},fireSyntheticButtonPressedEvent:function(t){var e=t.dataCell,i=t.gridCell;if(this.isViewableButton(e.x,e.y)){var n=new CustomEvent("fin-button-pressed",{detail:{gridCell:i}});this.canvas.dispatchEvent(n)}},fireSyntheticKeydownEvent:function(t){var e=new CustomEvent("fin-keydown",{detail:t.detail});this.canvas.dispatchEvent(e)},fireSyntheticKeyupEvent:function(t){var e=new CustomEvent("fin-keyup",{detail:t.detail});this.canvas.dispatchEvent(e)},fireSyntheticFilterAppliedEvent:function(t){var e=new CustomEvent("fin-filter-applied",{detail:t});this.canvas&&this.canvas.dispatchEvent(e)},fireSyntheticOnCellEnterEvent:function(t){var e={gridCell:t,time:Date.now(),grid:this},i=new CustomEvent("fin-cell-enter",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticGroupsChangedEvent:function(t){var e={groups:t,time:Date.now(),grid:this},i=new CustomEvent("fin-groups-changed",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticOnCellExitEvent:function(t){var e={gridCell:t,time:Date.now(),grid:this},i=new CustomEvent("fin-cell-exit",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticClickEvent:function(t){this.stopEditing();var e=t.gridCell,i={gridCell:e,mousePoint:t.mousePoint,keys:t.keys,primitiveEvent:t,time:Date.now(),grid:this};this.getBehavior().enhanceDoubleClickEvent(i);var n=new CustomEvent("fin-click",{detail:i});this.canvas.dispatchEvent(n)},fireSyntheticDoubleClickEvent:function(t){this.stopEditing();var e=t.gridCell,i=this.getBehavior(),n={gridCell:e,mousePoint:t.mousePoint,time:Date.now(),grid:this};i.enhanceDoubleClickEvent(t);var o=new CustomEvent("fin-double-click",{detail:n});i.cellDoubleClicked(e,t),this.canvas.dispatchEvent(o)},fireSyntheticGridRenderedEvent:function(){var t=new CustomEvent("fin-grid-rendered",{detail:{source:this,time:Date.now()}});this.canvas&&this.canvas.dispatchEvent(t)},fireScrollEvent:function(t,e,i){var n=new CustomEvent(t,{detail:{oldValue:e,value:i,time:Date.now()}});this.canvas.dispatchEvent(n)},setVScrollValue:function(t){t=Math.round(t);var e=this.sbVScroller.range.max;t=Math.min(e,Math.max(0,t));var i=this;if(t!==this.vScrollValue){this.getBehavior()._setScrollPositionY(t);var n=this.vScrollValue;this.vScrollValue=t,this.scrollValueChangedNotification(),setTimeout(function(){i.fireScrollEvent("fin-scroll-y",n,t)})}},getVScrollValue:function(){return this.vScrollValue},setHScrollValue:function(t){t=Math.round(t);var e=this.sbHScroller.range.max;t=Math.min(e,Math.max(0,t));var i=this;if(t!==this.hScrollValue){this.getBehavior()._setScrollPositionX(t);var n=this.hScrollValue;this.hScrollValue=t,this.scrollValueChangedNotification(),setTimeout(function(){i.fireScrollEvent("fin-scroll-x",n,t),i.synchronizeScrollingBoundries()})}},getHScrollValue:function(){return this.hScrollValue},takeFocus:function(){this.isEditing()?this.stopEditing():this.getCanvas().takeFocus()},editorTakeFocus:function(){return this.cellEditor?this.cellEditor.takeFocus():void 0},isEditing:function(){return this.cellEditor?this.cellEditor.isEditing:!1},initScrollbars:function(){var t=this,e=new u({orientation:"horizontal",onchange:t.setHScrollValue.bind(t),cssStylesheetReferenceElement:this.div}),i=new u({orientation:"vertical",onchange:t.setVScrollValue.bind(t),paging:{up:t.pageUp.bind(t),down:t.pageDown.bind(t)}});this.sbHScroller=e,this.sbVScroller=i;var n=this.resolveProperty("hScrollbarClassPrefix"),o=this.resolveProperty("vScrollbarClassPrefix");n&&""!==n&&(this.sbHScroller.classPrefix=n),o&&""!==o&&(this.sbVScroller.classPrefix=o),this.div.appendChild(e.bar),this.div.appendChild(i.bar),this.resizeScrollbars()},resizeScrollbars:function(){this.sbHScroller.shortenBy(this.sbVScroller).resize(),this.sbVScroller.resize()},setVScrollbarValues:function(t){this.sbVScroller.range={min:0,max:t}},setHScrollbarValues:function(t){this.sbHScroller.range={min:0,max:t}},scrollValueChangedNotification:function(){(this.hScrollValue!==this.sbPrevHScrollValue||this.vScrollValue!==this.sbPrevVScrollValue)&&(this.sbPrevHScrollValue=this.hScrollValue,this.sbPrevVScrollValue=this.vScrollValue,this.cellEditor&&this.cellEditor.scrollValueChangedNotification(),this.computeCellsBounds())},getValue:function(t,e){return this.getBehavior().getValue(t,e)},setValue:function(t,e,i){this.getBehavior().setValue(t,e,i)},getColumnAlignment:function(t){return this.getBehavior().getColumnAlignment(t)},synchronizeScrollingBoundries:function(){var t=this.getBehavior(),e=this.getFixedColumnCount(),i=this.getFixedRowCount(),n=this.getColumnCount(),o=this.getRowCount(),r=this.getBounds();if(r){for(var s=r.height-t.getFixedRowsMaxHeight()-15,l=r.width-200-t.getFixedColumnsMaxWidth()-15,a=0,u=0;n>a;a++){var c=this.getColumnWidth(n-a-1);if(u+=c,u>l)break}for(var h=0,d=0;o>h;h++){var g=this.getRowHeight(o-h-1);if(d+=g,d>s)break}var f=Math.max(0,n-e-a);this.setHScrollbarValues(f);var p=1+Math.max(0,o-i-h);this.setVScrollbarValues(p),this.setHScrollValue(Math.min(this.getHScrollValue(),f)),this.setVScrollValue(Math.min(this.getVScrollValue(),p)),this.computeCellsBounds(),this.repaint(),this.resizeScrollbars()}},getVisibleRows:function(){return this.getRenderer().getVisibleRows()},getVisibleColumns:function(){ -return this.getRenderer().getVisibleColumns()},initRenderer:function(){this.renderer=new p(this)},getRenderer:function(){return this.renderer},getColumnWidth:function(t){return this.getBehavior().getColumnWidth(t)},setColumnWidth:function(t,e){this.stopEditing(),this.getBehavior().setColumnWidth(t,e)},getColumnEdge:function(t){return this.getBehavior().getColumnEdge(t,this.getRenderer())},getFixedColumnsWidth:function(){return this.getBehavior().getFixedColumnsWidth()},getRowHeight:function(t){return this.getBehavior().getRowHeight(t)},setRowHeight:function(t,e){this.stopEditing(),this.getBehavior().setRowHeight(t,e)},getFixedRowsHeight:function(){return this.getBehavior().getFixedRowsHeight()},getColumnCount:function(){return this.getBehavior().getColumnCount()},getRowCount:function(){return this.getBehavior().getRowCount()},getUnfilteredRowCount:function(){return this.getBehavior().getUnfilteredRowCount()},getFixedColumnCount:function(){return this.getBehavior().getFixedColumnCount()},getFixedRowCount:function(){return this.getBehavior().getFixedRowCount()},topLeftClicked:function(t){this.getBehavior().topLeftClicked(this,t)},rowHeaderClicked:function(t){this.getBehavior().rowHeaderClicked(this,t)},columnHeaderClicked:function(t){this.getBehavior().columnHeaderClicked(this,t)},_activateEditor:function(t){var e=t.gridCell;this.activateEditor(e.x,e.y)},activateEditor:function(t,e){if(this.isEditable()||this.isFilterRow(e)){var i=this.getCellEditorAt(t,e);if(i){var n=i.getEditorPoint();if(i){if(n.x===t&&n.y===e&&i.isEditing)return;this.isEditing()&&this.stopEditing(),event.gridCell={x:t,y:e},this.editAt(i,event)}}}},getCellEditorAt:function(t,e){return this.getBehavior()._getCellEditorAt(t,e)},toggleHiDPI:function(){this.useHiDPI()?this.removeAttribute("hidpi"):this.setAttribute("hidpi",null),this.canvas.resize()},getHiDPI:function(t){if(window.devicePixelRatio&&this.useHiDPI()){var e=window.devicePixelRatio||1,i=t.webkitBackingStorePixelRatio||t.mozBackingStorePixelRatio||t.msBackingStorePixelRatio||t.oBackingStorePixelRatio||t.backingStorePixelRatio||1,n=e/i;return n}return 1},getRenderedWidth:function(t){return this.renderer.getRenderedWidth(t)},getRenderedHeight:function(t){return this.renderer.getRenderedHeight(t)},resolveCellEditor:function(t){return this.cellEditors[t]},updateCursor:function(){var t=this.getBehavior(),e=t.getCursorAt(-1,-1),i=this.getHoverCell();if(i&&i.x>-1&&i.y>-1){var n=i.x+this.getHScrollValue();e=t.getCursorAt(n,i.y+this.getVScrollValue())}this.beCursor(e)},repaintCell:function(t,e){this.getRenderer().repaintCell(t,e)},isDraggingColumn:function(){return!!this.renderOverridesCache.dragger},pageUp:function(){var t=this.getRenderer().getPageUpRow();return this.setVScrollValue(t),t},pageDown:function(){var t=this.getRenderer().getPageDownRow();return this.setVScrollValue(t),t},pageLeft:function(){console.log("page left")},pageRight:function(){console.log("page right")},getRenderedData:function(){var t,e=this.getBehavior(),i=this.getRenderer(),n=this.getColumnCount(),o=i.getVisibleRows(),r=new Array(n),s=new Array(o);return r.forEach(function(t,i){r[i]=e.getColumnId(i,0)}),s.forEach(function(i,n){t=s[n]={hierarchy:e.getFixedColumnValue(0,n)},r.forEach(function(i,o){t[i]=e.getValue(o,n)})}),s},getSelectedRow:function(){var t=this.getSelectionModel().getSelections();if(t.length){for(var e=this.getBehavior(),i=this.getColumnCount(),n=t[0].origin.y,o={},r=0;i>r;r++)o[e.getColumnId(r,0)]=e.getValue(r,n);return o}},fireRequestCellEdit:function(t,e){var i=new CustomEvent("fin-request-cell-edit",{cancelable:!0,detail:{value:e,gridCell:t,time:Date.now()}});return this.canvas.dispatchEvent(i)},fireBeforeCellEdit:function(t,e,i,n){var o=new CustomEvent("fin-before-cell-edit",{cancelable:!0,detail:{oldValue:e,newValue:i,gridCell:t,time:Date.now(),input:n,row:this.getRow(t.y)}}),r=this.canvas.dispatchEvent(o);return r},fireAfterCellEdit:function(t,e,i,n){var o=new CustomEvent("fin-after-cell-edit",{detail:{newValue:i,oldValue:e,gridCell:t,time:Date.now(),input:n,row:this.getRow(t.y)}});this.canvas.dispatchEvent(o)},autosizeColumn:function(t){var e=this.getBehavior().getColumn(t);e.checkColumnAutosizing(!0),this.computeCellsBounds()},setFocusable:function(t){this.getCanvas().setFocusable(t)},getVisibleColumnsCount:function(){return this.getRenderer().getVisibleColumnsCount()},getVisibleRowsCount:function(){return this.getRenderer().getVisibleRowsCount()},updateSize:function(){this.canvas.checksize()},stopPaintThread:function(){this.canvas.stopPaintThread()},stopResizeThread:function(){this.canvas.stopResizeThread()},restartResizeThread:function(){this.canvas.restartResizeThread()},restartPaintThread:function(){this.canvas.restartPaintThread()},swapColumns:function(t,e){this.getBehavior().swapColumns(t,e)},endDragColumnNotification:function(){this.getBehavior().endDragColumnNotification()},getFixedColumnsMaxWidth:function(){return this.getBehavior().getFixedColumnsMaxWidth()},isMouseDownInHeaderArea:function(){var t=this.getHeaderColumnCount(),e=this.getHeaderRowCount(),i=this.getMouseDown();return i.x-1,r=e.indexOf("SHIFT")>-1;o||r?(o&&(n?i.deselectColumn(t):i.selectColumn(t)),r&&(i.clear(),i.selectColumn(this.lastEdgeSelection[0],t))):(i.clear(),n||i.selectColumn(t)),n||r||(this.lastEdgeSelection[0]=t),this.repaint(),this.fireSyntheticColumnSelectionChangedEvent()},toggleSelectRow:function(t,e){var i=this.getFilterRowIndex()+1;if(!(i>t)){e=e||[];var n=this.isSingleRowSelectionMode(),o=this.getSelectionModel(),r=o.isRowSelected(t),s=e.indexOf("CTRL")>-1,l=e.indexOf("SHIFT")>-1;s||l?(s&&(r?o.deselectRow(t):(n&&o.clearRowSelection(),o.selectRow(t))),l&&(o.clear(),o.selectRow(this.lastEdgeSelection[1],t))):r?o.deselectRow(t):o.selectRow(t),r||l||(this.lastEdgeSelection[1]=t),this.repaint()}},selectViewportCell:function(t,e){var i=this.getHeaderRowCount(),n=this.getRenderer(),o=n.getVisibleColumns()[t],r=n.getVisibleRows()[e];this.clearSelections(),this.select(o,r+i,0,0),this.setMouseDown(this.newPoint(o,r+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToViewportCell:function(t,e){var i=this.getSelections();if(i&&0!==i.length){var n=this.getHeaderRowCount(),o=this.getRenderer(),r=o.getVisibleColumns()[t],s=o.getVisibleRows()[e]+n,l=i[0],a=l.origin;this.setDragExtent(this.newPoint(r-a.x,s-a.y)),this.select(a.x,a.y,r-a.x,s-a.y),this.repaint()}},selectFinalCellOfCurrentRow:function(){var t=this.getColumnCount()-1,e=this.getSelectedRows()[0],i=this.getHeaderRowCount();this.clearSelections(),this.scrollBy(this.getColumnCount(),0),this.select(t,e+i,0,0),this.setMouseDown(this.newPoint(t,e+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToFinalCellOfCurrentRow:function(){var t=this.getSelections();if(t&&0!==t.length){var e=t[0],i=e.origin,n=e.extent,o=this.getColumnCount();this.scrollBy(o,0),this.clearSelections(),this.select(i.x,i.y,o-i.x-1,n.y),this.repaint()}},selectFirstCellOfCurrentRow:function(){var t=0,e=this.getSelectedRows()[0],i=this.getHeaderRowCount();this.clearSelections(),this.setHScrollValue(0),this.select(t,e+i,0,0),this.setMouseDown(this.newPoint(t,e+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToFirstCellOfCurrentRow:function(){var t=this.getSelections();if(t&&0!==t.length){var e=t[0],i=e.origin,n=e.extent;this.clearSelections(),this.select(i.x,i.y,-i.x,n.y),this.setHScrollValue(0),this.repaint()}},selectFinalCell:function(){this.selectCell(this.getColumnCount()-1,this.getRowCount()-1),this.scrollBy(this.getColumnCount(),this.getRowCount()),this.repaint()},selectToFinalCell:function(){var t=this.getSelections();if(t&&0!==t.length){var e=t[0],i=e.origin,n=this.getColumnCount(),o=this.getRowCount();this.clearSelections(),this.select(i.x,i.y,n-i.x-1,o-i.y-1),this.scrollBy(n,o),this.repaint()}},isShowRowNumbers:function(){return this.resolveProperty("showRowNumbers")},isEditable:function(){return this.resolveProperty("editable")===!0},isShowFilterRow:function(){return this.resolveProperty("showFilterRow")},isShowHeaderRow:function(){return this.resolveProperty("showHeaderRow")},getHeaderRowCount:function(){return this.getBehavior().getHeaderRowCount()},isFilterRow:function(t){return t===this.getFilterRowIndex()},getFilterRowIndex:function(){return this.isShowFilterRow()?this.isShowHeaderRow()?1:0:-1},setGroups:function(t){this.getBehavior().setGroups(t)},filterClicked:function(t){this.activateEditor(t.gridCell.x,t.gridCell.y)},hasHierarchyColumn:function(){return this.getBehavior().hasHierarchyColumn()},isHierarchyColumn:function(t){return this.hasHierarchyColumn()?0===t:!1},checkScrollbarVisibility:function(){},isColumnOrRowSelected:function(){return this.getSelectionModel().isColumnOrRowSelected()},selectColumn:function(t,e){this.getSelectionModel().selectColumn(t,e)},selectRow:function(t,e){this.isSingleRowSelectionMode()?(this.getSelectionModel().clearRowSelection(),t=e):e=e||t;var i=Math.min(t,e),n=Math.max(t,e),o=this.getFilterRowIndex()+1;o>i||this.getSelectionModel().selectRow(i,n)},isRowNumberAutosizing:function(){return this.resolveProperty("rowNumberAutosizing")},isRowSelected:function(t){return this.getSelectionModel().isRowSelected(t)},isColumnSelected:function(t){return this.getSelectionModel().isColumnSelected(t)},lookupFeature:function(t){return this.getBehavior().lookupFeature(t)},getRow:function(t){return this.getBehavior().getRow(t)},getFieldName:function(t){return this.getBehavior().getFieldName(t)},getColumnIndex:function(t){return this.getBehavior().getColumnIndex(t)},isCellSelection:function(){return this.resolveProperty("cellSelection")===!0},isRowSelection:function(){return this.resolveProperty("rowSelection")===!0},isColumnSelection:function(){return this.resolveProperty("columnSelection")===!0},getComputedRow:function(t){return this.getBehavior().getComputedRow(t)},isColumnAutosizing:function(){return this.resolveProperty("columnAutosizing")===!0},setGlobalFilter:function(t){this.getBehavior().setGlobalFilter(t)},selectRowsFromCells:function(){if(!this.isCheckboxOnlyRowSelections()){var t=this.getSelectionModel();if(this.isSingleRowSelectionMode()){var e=t.getLastSelection();e?this.selectRow(null,e.corner.y):t.clearRowSelection()}else t.selectRowsFromCells()}},selectColumnsFromCells:function(){this.getSelectionModel().selectColumnsFromCells()},getSelectedRows:function(){return this.getBehavior().getSelectedRows()},getSelectedColumns:function(){return this.getBehavior().getSelectedColumns()},getSelections:function(){return this.getBehavior().getSelections()},getLastSelectionType:function(){return this.getSelectionModel().getLastSelectionType()},isCellSelected:function(t,e){return this.getSelectionModel().isCellSelected(t,e)},isInCurrentSelectionRectangle:function(t,e){return this.getSelectionModel().isInCurrentSelectionRectangle(t,e)},selectAllRows:function(){this.getSelectionModel().selectAllRows()},areAllRowsSelected:function(){return this.getSelectionModel().areAllRowsSelected()},toggleSelectAllRows:function(){this.areAllRowsSelected()?this.getSelectionModel().clear():this.selectAllRows(),this.repaint()},getField:function(t){return this.getBehavior().getField(t)},isSingleRowSelectionMode:function(){return this.resolveProperty("singleRowSelectionMode")},newPoint:function(t,e){return new h(t,e)},newRectangle:function(t,e,i,n){return new d(t,e,i,n)},registerFilter:function(t){b[t.alias]=t},getFilterFor:function(t){return b.MyCustomFilter},resolveFilter:function(t){return b[t]},getFormattedValue:function(t,e){e+=this.getHeaderRowCount();var i=this.getColumnProperties(t).format,n=this.getValue(t,e),o=this.getFormatter(i),r=o(n);return r}},e.exports=n},{"./Formatters":24,"./Renderer":27,"./SelectionModel":28,"./TableDialog":29,"./defaults":48,"./stylesheets":84,"extend-me":4,finbars:10,fincanvas:11,"object-iterators":18,rectangular:19}],26:[function(t,e,i){"use strict";e.exports=function(){function t(){this.keys=[],this.data={},this.values=[]}var e=".~.#%_",i=0,n=function(t){var n=typeof t;switch(n){case"number":return e+n+"_"+t;case"string":return e+n+"_"+t;case"boolean":return e+n+"_"+t;case"symbol":return e+n+"_"+t;case"undefined":return e+"undefined";case"object":return t.___finhash?t.___finhash:(t.___finhash=e+i++,t.___finhash);case"function":return t.___finhash?t.___finhash:(t.___finhash=e+i++,t.___finhash)}},o=Object.is||function(t,e){return t===e?0!==t||1/t==1/e:t!=t&&e!=e},r=function(t,e){if(e!=e||0===e)for(var i=t.length;i--&&!o(t[i],e););else i=[].indexOf.call(t,e);return i};return t.prototype.set=function(t,e){var i=n(t);void 0===this.data[i]&&(this.keys.push(t),this.values.push(e)),this.data[i]=e},t.prototype.get=function(t){var e=n(t);return this.data[e]},t.prototype.getIfAbsent=function(t,e){var i=this.get(t);return void 0===i&&(i=e(t,this)),i},t.prototype.size=function(){return this.keys.length},t.prototype.clear=function(){this.keys.length=0,this.data={}},t.prototype["delete"]=function(t){var e=n(t);if(void 0!==this.data[e]){var i=r(this.keys,t);this.keys.splice(i,1),this.values.splice(i,1),delete this.data[e]}},t.prototype.forEach=function(t){for(var e=this.keys,i=0;it&&(i=t,t>=f&&(i+=d,void 0===s&&(s=i),l=i),!(R>y||i>=g));t++)o=this.getColumnWidth(i),R+=o,this.columnEdges[t+1]=Math.round(R),this.visibleColumns[t]=i,this.columnEdgesIndexMap[i]=t,S=S+Math.round(o/2)+b,this.insertionBounds.push(S),b=Math.round(o/2);for(e=0;p>e&&(n=e,e>=v&&(n+=h,void 0===a&&(a=n),u=n),!(A>x||n>=p));e++)r=this.getRowHeight(n),A+=r,this.rowEdges[e+1]=Math.round(A),this.visibleRows[e]=n,this.rowEdgesIndexMap[n]=e;this.viewHeight=x,this.dataWindow=c.newRectangle(s,a,l-s,u-a)},resolveProperty:function(t){return this.getGrid().resolveProperty(t)},getGrid:function(){return this.grid},_paint:function(t){this.grid&&(this.renderGrid(t),this.getGrid().gridRenderedNotification())},getVisibleRowsCount:function(){return this.visibleRows.length-1},getVisibleScrollHeight:function(){var t=this.getGrid(),e=t.getFixedRowsHeight();return this.viewHeight-e},getVisibleRows:function(){return this.visibleRows},getVisibleColumnsCount:function(){return this.visibleColumns.length-1},getVisibleColumns:function(){return this.visibleColumns},overColumnDivider:function(t){t=Math.round(t);var e=this.getColumnEdges(),i=e.indexOf(t-1);return 0>i&&(i=e.indexOf(t)),0>i&&(i=e.indexOf(t-2)),0>i&&(i=e.indexOf(t+1)),0>i&&(i=e.indexOf(t-3)),i},overRowDivider:function(t){t=Math.round(t);var e=this.rowEdges.indexOf(t+1);return 0>e&&(e=this.rowEdges.indexOf(t)),0>e&&(e=this.rowEdges.indexOf(t-1)),e},getBoundsOfCell:function(t){return this._getBoundsOfCell(t.x,t.y)},_getBoundsOfCell:function(t,e){var i,n=!1,o=!1,r=this.cell,s=this.columnEdgesIndexMap[t];void 0===s&&(s=this.columnEdgesIndexMap[t-1],n=!0);var l,a,u,c=this.columnEdges[s],h=this.columnEdges[s+1],d=h-c;if(r.x=n?h:c,r.width=n?0:d,0>e){var g=this.getGrid(),f=g.getBehavior(),p=this.getBounds();u=f.getDefaultRowHeight(),l=p.height+e*u,a=l+u}else i=this.rowEdgesIndexMap[e],void 0===i&&(i=this.rowEdgesIndexMap[e-1],o=!0),l=this.rowEdges[i],a=this.rowEdges[i+1],u=a-l;return r.y=o?a:l,r.height=o?0:u,r},getColumnFromPixelX:function(t){for(var e=0,i=this.getGrid(),n=this.getFixedColumnCount(),o=i.getHScrollValue(),r=this.getColumnEdges(),s=1;st)return s>n&&(s+=o),s-1;return s>n&&(s+=o),s-1},getGridCellFromMousePoint:function(t){var e,i,n,o,r=this.getGrid(),s=r.getBehavior(),l=0,a=0,u=0,c=this.getColumnEdges(),h=this.getFixedColumnCount(),d=this.getFixedRowCount(),g=this.getScrollLeft(),f=this.getScrollTop();for(n=0;n=h&&(n+=g),o>=d&&(o+=f);var v=-1,m=s.getColumn(n);return m&&(v=m.index),{gridCell:r.newPoint(n,o),mousePoint:r.newPoint(e,i),viewPoint:p,dataCell:r.newPoint(v,o)}},isColumnVisible:function(t){var e=-1!==this.visibleColumns.indexOf(t);return e},getFinalVisableColumnBoundary:function(){var t=this.isLastColumnVisible(),e=t?2:1,i=this.getColumnEdges()[this.getColumnEdges().length-e],n=Math.min(i,this.getBounds().width-200);return n},isRowVisible:function(t){var e=-1!==this.visibleRows.indexOf(t);return e},isSelected:function(t,e){return this.getGrid().isSelected(t,e)},renderGrid:function(t){t.beginPath(),this.paintCells(t),this.paintGridlines(t),this.renderOverrides(t),this.renderFocusCell(t),t.closePath()},focusLineStep:[[5,5],[0,1,5,4],[0,2,5,3],[0,3,5,2],[0,4,5,1],[0,5,5,0],[1,5,4,0],[2,5,3,0],[3,5,2,0],[4,5,1,0]],renderFocusCell:function(t){t.beginPath(),this._renderFocusCell(t),t.closePath()},_renderFocusCell:function(t){var e=this.getGrid(),i=e.getSelectionModel().getSelections();if(i&&0!==i.length){var n=i[i.length-1],o=n.origin;if(-1!==o.x){var r=this.getVisibleColumns(),s=this.getVisibleRows(),l=r[r.length-1],a=s[s.length-1],u=n.extent,c=Math.min(o.x,o.x+u.x),h=Math.min(o.y,o.y+u.y);if(!(c>l||h>a)){var d=Math.max(o.x,o.x+u.x)+1;d=Math.min(d,1+l);var g=Math.max(o.y,o.y+u.y)+1;g=Math.min(g,1+a);var f=this._getBoundsOfCell(c,h),p=Math.round(void 0===f.x?e.getFixedColumnsWidth():f.x),v=Math.round(void 0===f.y?e.getFixedRowsHeight():f.y),m=this._getBoundsOfCell(d,g),C=Math.round(void 0===m.x?e.getFixedColumnsWidth():m.x),w=Math.round(void 0===m.y?e.getFixedRowsHeight():m.y),y=Math.min(p,C),x=Math.min(v,w),S=1+C-p,b=1+w-v;y===C&&(S=p-C),x===w&&(b=v-w),1>S*b||(t.rect(y,x,S,b),t.fillStyle=this.resolveProperty("selectionRegionOverlayColor"),t.fill(),t.lineWidth=1,t.strokeStyle=this.resolveProperty("selectionRegionOutlineColor"),t.stroke())}}}},renderOverrides:function(t){var e=this.getGrid(),i=e.renderOverridesCache;for(var n in i)if(i.hasOwnProperty(n)){var o=i[n];o&&this.renderOverride(t,o)}},renderOverride:function(t,e){var i=e.hdpiratio,n=e.startX,o=e.width+1,r=e.height,s=e.ctx,l=t.getImageData(n,0,Math.round(o*i),Math.round(r*i));s.putImageData(l,0,0),t.fillStyle=this.resolveProperty("backgroundColor2"),t.fillRect(Math.round(n/i),0,o,r)},isHovered:function(t,e){var i=this.getGrid();return i.isHovered(t,e)&&i.resolveProperty("hoverCellHighlight")===!0},isRowHovered:function(t){var e=this.getGrid();return e.isRowHovered(t)&&e.resolveProperty("hoverRowHighlight")===!0},isColumnHovered:function(t){var e=this.getGrid();return e.isColumnHovered(t)&&e.resolveProperty("hoverColumnHighlight")===!0},isCellSelectedInRow:function(t){return this.getGrid().isCellSelectedInRow(t)},isCellSelectedInColumn:function(t){return this.getGrid().isCellSelectedInColumn(t)},getScrollTop:function(){var t=this.getGrid().getVScrollValue();return t},getScrollLeft:function(){var t=this.getGrid().getHScrollValue();return t},getBehavior:function(){return this.getGrid().getBehavior()},getColumnEdges:function(){return this.columnEdges},getRowEdges:function(){return this.rowEdges},getRowHeight:function(t){var e=this.getBehavior().getRowHeight(t);return e},getColumnWidth:function(t){var e=this.getGrid().getColumnWidth(t);return e},isLastColumnVisible:function(){var t=this.getColumnCount()-1,e=-1!==this.visibleColumns.indexOf(t);return e},getRenderedWidth:function(t){return this.getColumnEdges()[t]},getRenderedHeight:function(t){return this.rowEdges[t]},getCanvas:function(){return this.getGrid().getCanvas()},isDraggingColumn:function(){return this.getGrid().isDraggingColumn()},getPageUpRow:function(){for(var t=this.getBehavior(),e=this.getVisibleScrollHeight(),i=this.getGrid().getFixedRowCount(),n=this.dataWindow.origin.y-i,o=0;e>o&&n>-1;)o+=t.getRowHeight(n),n--;return n+1},getPageDownRow:function(){var t=this.getGrid().getFixedRowCount(),e=this.dataWindow.corner.y-t-1;return e},getColumnCount:function(){return this.getGrid().getColumnCount()},getRowCount:function(){return this.getGrid().getRowCount()},getFixedColumnCount:function(){return this.getGrid().getFixedColumnCount()},getFixedRowCount:function(){return this.getGrid().getFixedRowCount()},getHeaderRowCount:function(){return this.getGrid().getHeaderRowCount()},getHeaderColumnCount:function(){return this.getGrid().getHeaderColumnCount()},paintCells:function(t){var e,i,o,r,s,l,a,u=this.getColumnEdges(),c=this.rowEdges,h=this.getVisibleColumns(),d=this.getVisibleRows(),g=this.getBehavior(),f=0,p=0,v=this.getBounds().height,m=this.getGrid().isShowRowNumbers()?-1:0,C=h.length;if(this.buttonCells={},C)for(o=m;C>o;o++,f+=a){for(s=h[o],this.renderedColumnMinWidths[s]=0,e=g.getColumnProperties(s).renderCellError,t.save(),a=u[o-m]-f,t.beginPath(),t.rect(f,p,a,v),t.clip(),r=0;r=0&&c>i,g=0>i,f=n.isFilterRow(i),p=n.isHierarchyColumn(e),v=n.isRowSelected(i),m=n.isColumnSelected(e),C=n.isCellSelected(e,i),w=n.isCellSelectedInColumn(e),y=n.isCellSelectedInRow(i),x=n.areAllRowsSelected();h&&-1===e||p?(v?(l=l.rowHeaderRowSelection,a=Object.create(l),a.isSelected=!0):(l=l.rowHeader,a=Object.create(l),a.isSelected=y),a.isUserDataArea=!1):d||g?(f?(l=l.filterProperties,a=Object.create(l),a.isSelected=!1):m?(l=l.columnHeaderColumnSelection,a=Object.create(l),a.isSelected=!0):(l=l.columnHeader,a=Object.create(l),a.isSelected=w),a.isUserDataArea=!1):p?(l=l.rowHeader,a=Object.create(l),a.isSelected=y):(a=Object.create(l),a.isSelected=C||v||m,a.isUserDataArea=!0);var S=i-c+1;-1===e?(0===i?a.value=[s.checkbox(x),"",null]:f?a.value=[s.filter(!1),"",null]:d||g?a.value="":a.value=[s.checkbox(v),S,null],a.halign="right"):(a.value=n.getValue(e,i),a.halign=n.getColumnAlignment(e)),a.isColumnHovered=this.isColumnHovered(e),a.isRowHovered=this.isRowHovered(i),a.isCellHovered=this.isHovered(e,i),a.bounds=this._getBoundsOfCell(e,i),a.isCellSelected=C,a.isRowSelected=v,a.isColumnSelected=m,a.isInCurrentSelectionRectangle=n.isInCurrentSelectionRectangle(e,i);var b=n.mouseDownState;if(b){var E=b.gridCell;a.mouseDown=E.x===e&&E.y===i}a.x=e,a.y=i,r.cellPropertiesPrePaintNotification(a);var R=r.getCellRenderer(a,e,i),A=r.getCellProperties(e,i);o(a).extendOwn(A),a.buttonCells=this.buttonCells;var M=a.isUserDataArea?a.format:"default";a.formatter=this.getGrid().getFormatter(M),R.paint(t,a),this.renderedColumnMinWidths[e]=Math.max(a.minWidth||0,this.renderedColumnMinWidths[e]),u.preferredWidth=this.renderedColumnMinWidths[e]}},isViewableButton:function(t,e){var i=t+","+e;return this.buttonCells[i]===!0},getRowNumbersWidth:function(){var t=this.getColumnEdges();return 0===t.length?0:t[0]},startAnimator:function(){var t,e=this;t=function(){e.animate(),requestAnimationFrame(t)},requestAnimationFrame(t)},animate:function(){var t=this.getCanvas().canvasCTX;t.beginPath(),t.save(),this.renderFocusCell(t),t.restore(),t.closePath()},getBounds:function(){return this.bounds},setBounds:function(t){return this.bounds=t}});e.exports=l},{"../images":2,"extend-me":4,"object-iterators":18}],28:[function(t,e,i){"use strict";function n(){this.selections=[],this.flattenedX=[],this.flattenedY=[],this.rowSelectionModel=new o,this.columnSelectionModel=new o,this.setLastSelectionType("")}var o=t("sparse-boolean-array");n.prototype={allRowsSelected:!1,getGrid:function(){return null},getLastSelection:function(){var t=this.selections,e=t[t.length-1];return e},getLastSelectionType:function(){return this.lastSelectionType},setLastSelectionType:function(t){this.lastSelectionType=t},select:function(t,e,i,n){var o=this.getGrid().newRectangle(t,e,i,n);this.selections.push(o),this.flattenedX.push(o.flattenXAt(0)),this.flattenedY.push(o.flattenYAt(0)),this.setLastSelectionType("cell"),this.getGrid().selectionChanged()},toggleSelect:function(t,e,i,n){var o,r;o=this.selections.find(function(o,s){return r=s,o.origin.x===t&&o.origin.y===e&&o.extent.x===i&&o.extent.y===n}),o?(this.selections.splice(r,1),this.flattenedX.splice(r,1),this.flattenedY.splice(r,1),this.getGrid().selectionChanged()):this.select(t,e,i,n)},clearMostRecentSelection:function(t){t=t===!0,t||this.setAllRowsSelected(!1),this.selections.length=Math.max(0,this.selections.length-1),this.flattenedX.length=Math.max(0,this.flattenedX.length-1),this.flattenedY.length=Math.max(0,this.flattenedY.length-1)},clearMostRecentColumnSelection:function(){this.columnSelectionModel.clearMostRecentSelection(),this.setLastSelectionType("column")},clearMostRecentRowSelection:function(){this.rowSelectionModel.clearMostRecentSelection(),this.setLastSelectionType("row")},clearRowSelection:function(){this.rowSelectionModel.clear(),this.setLastSelectionType("row")},getSelections:function(){return this.selections},hasSelections:function(){return 0!==this.selections.length},hasRowSelections:function(){return!this.rowSelectionModel.isEmpty()},hasColumnSelections:function(){return!this.columnSelectionModel.isEmpty()},isCellSelectedInRow:function(t){return this._isCellSelected(this.flattenedX,0,t)},isCellSelectedInColumn:function(t){return this._isCellSelected(this.flattenedY,t,0)},isSelected:function(t,e){return this.isColumnSelected(t)||this.isRowSelected(e)||this._isCellSelected(this.selections,t,e)},isCellSelected:function(t,e){return this._isCellSelected(this.selections,t,e)},_isCellSelected:function(t,e,i){var n=this;return!!t.find(function(t){return n.rectangleContains(t,e,i)})},clear:function(t){t=t===!0,this.selections.length=0,this.flattenedX.length=0,this.flattenedY.length=0,this.columnSelectionModel.clear(),t||(this.setAllRowsSelected(!1),this.rowSelectionModel.clear())},isRectangleSelected:function(t,e,i,n){return!!this.selections.find(function(o){return o.origin.x===t&&o.origin.y===e&&o.extent.x===i&&o.extent.y===n})},isColumnSelected:function(t){return this.columnSelectionModel.isSelected(t)},isRowSelected:function(t){return this.allRowsSelected||this.rowSelectionModel.isSelected(t)},selectColumn:function(t,e){this.columnSelectionModel.select(t,e),this.setLastSelectionType("column")},selectAllRows:function(){this.clear(),this.setAllRowsSelected(!0)},setAllRowsSelected:function(t){this.allRowsSelected=t},areAllRowsSelected:function(){return this.allRowsSelected},selectRow:function(t,e){this.rowSelectionModel.select(t,e),this.setLastSelectionType("row")},deselectColumn:function(t,e){this.columnSelectionModel.deselect(t,e),this.setLastSelectionType("column")},deselectRow:function(t,e){this.rowSelectionModel.deselect(t,e),this.setLastSelectionType("row")},getSelectedRows:function(){if(this.areAllRowsSelected()){for(var t=this.getGrid(),e=t.getHeaderRowCount(),i=t.getRowCount()-e,n=new Array(i),o=0;i>o;o++)n[o]=o+e;return n}return this.rowSelectionModel.getSelections()},getSelectedColumns:function(){return this.columnSelectionModel.getSelections()},isColumnOrRowSelected:function(){return!this.columnSelectionModel.isEmpty()||!this.rowSelectionModel.isEmpty()},getFlattenedYs:function(){var t=[],e={};return this.selections.forEach(function(i){for(var n=i.origin.y,o=i.extent.y+1,r=0;o>r;r++){var s=r+n;e[s]||(t.push(s),e[s]=!0)}}),t.sort(function(t,e){return t-e}),t},selectRowsFromCells:function(t){t=t||0;var e=this.rowSelectionModel;this.setAllRowsSelected(!1),e.clear(),this.selections.forEach(function(i){var n=i.origin.y,o=i.extent.y;e.select(n+t,n+o+t)})},selectColumnsFromCells:function(t){t=t||0;var e=this.columnSelectionModel;e.clear(),this.selections.forEach(function(i){var n=i.origin.x,o=i.extent.x;e.select(n+t,n+o+t)})},isInCurrentSelectionRectangle:function(t,e){var i=this.selections[this.selections.length-1];return i&&this.rectangleContains(i,t,e)},rectangleContains:function(t,e,i){var n=t.origin.x,o=t.origin.y,r=n+t.extent.x,s=o+t.extent.y;t.extent.x<0&&(n=r,r=t.origin.x),t.extent.y<0&&(o=s,s=t.origin.y);var l=e>=n&&i>=o&&r>=e&&s>=i;return l}},e.exports=n},{"sparse-boolean-array":21}],29:[function(t,e,i){"use strict";var n=t("extend-me").Base,o=500,r=o+"ms ease-in",s=n.extend("TableDialog",{initialize:function(t){this.grid=t,this.initializeOverlaySurface(),this.openNow=!1},isOpen:function(){return this.openNow},open:function(){if(!this.isOpen()){this.openNow=!0;var t=this;this.overlay.style.backgroundColor=this.grid.resolveProperty("backgroundColor"),this.overlay.style.top=this.overlay.style.bottom=this.overlay.style.right=this.overlay.style.left=0,t.overlay.style.webkitTransition="",this.overlay.style.margin="15px 35px 35px 15px",this.overlay.style.opacity=0, -this.overlay.style.zIndex=100,this.closeTransition=function(){this.overlay.style.opacity=0},this._closer||(this._closer=function(e){var i=t.getCharFor(e.keyCode).toLowerCase(),n=t.grid.resolveProperty("editorActivationKeys");(n.indexOf(i)>-1||27===e.keyCode)&&(e.preventDefault(),t.close())}),requestAnimationFrame(function(){t.overlay.style.webkitTransition="opacity "+o+"ms ease-in",requestAnimationFrame(function(){document.addEventListener("keydown",t._closer,!1),t.overlay.style.opacity=.95})}),setTimeout(function(){t.overlay.focus()},100)}},openFrom:function(t){if(!this.isOpen()){this.openNow=!0;var e=this,i=this.overlay.style;i.backgroundColor=this.grid.resolveProperty("backgroundColor");var n=this.grid.div.getBoundingClientRect(),o=t.y+"px "+(n.width-(t.x+t.width))+"px "+(n.height-(t.y+t.height))+"px "+t.x+"px";i.webkitTransition="",i.top=i.right=i.bottom=i.left=0,i.margin=o,i.zIndex=100,i.opacity=1,this.closeTransition=function(){i.margin=o},this._closer||(this._closer=function(t){var i=e.getCharFor(t.keyCode).toLowerCase(),n=e.grid.resolveProperty("editorActivationKeys");(n.indexOf(i)>-1||27===t.keyCode)&&(t.preventDefault(),e.close())}),requestAnimationFrame(function(){document.addEventListener("keydown",e._closer,!1),requestAnimationFrame(function(){requestAnimationFrame(function(){i.webkitTransition="margin-top "+r+", margin-right "+r+", margin-bottom "+r+", margin-left "+r,i.margin="15px 35px 35px 15px"})})}),setTimeout(function(){e.overlay.focus()},100)}},close:function(){this.openNow=!1,document.removeEventListener("keydown",this._closer,!1);var t=this;requestAnimationFrame(function(){t.closeTransition()}),setTimeout(function(){t.clear(),t.overlay.style.zIndex=-1e3,t.onClose&&(t.onClose(),t.onClose=void 0),t.grid.takeFocus()},o)},initializeOverlaySurface:function(){this.overlay=document.createElement("div"),this.overlay.setAttribute("tabindex",0),this.overlay.addEventListener("wheel",function(t){t.stopPropagation()});var t=this.overlay.style;t.outline="none",t.boxShadow="0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)",t.position="absolute",t.margin=0,t.overflow="hidden",t.opacity=0,t.zIndex=10,this.grid.div.appendChild(this.overlay)},getCharFor:function(t){var e=this.grid.getCanvas().getCharMap();return e[t][0]},clear:function(){this.overlay.innerHTML=""},querySelector:function(t){var e=this.overlay.querySelector(t);return e},getAnimationTime:function(){return o}});e.exports=s},{"extend-me":4}],30:[function(t,e,i){"use strict";var n=t("object-iterators"),o=t("extend-me").Base,r=t("./Column"),s=t("../CellProvider"),l=["columnHeader","columnHeaderColumnSelection","filterProperties","rowHeader","rowHeaderRowSelection","rowNumbersProperties","treeColumnProperties","treeColumnPropertiesColumnSelection"],a=o.extend("Behavior",{initialize:function(t){t.setBehavior(this),this.initializeFeatureChain(t),this.getDataModel(),this.cellProvider=this.createCellProvider(),this.renderedColumnCount=30,this.renderedRowCount=60,this.dataUpdates={}},initializeFeatureChain:function(t){var e=this;this.features.forEach(function(t){e.setNextFeature(new t)}),this.featureChain.initializeOn(t)},features:[],tableState:null,grid:null,editorTypes:["choice","textfield","color","slider","spinner","date"],featureChain:null,dataModel:null,baseModel:null,scrollPositionX:0,scrollPositionY:0,featureMap:{},allColumns:[],columns:[],reset:function(){this.cellProvider=this.createCellProvider(),this.renderedColumnCount=30,this.renderedRowCount=60,this.dataUpdates={},this.clearColumns(),this.clearState(),this.getDataModel().reset(),this.createColumns()},clearColumns:function(){this.columns=[],this.allColumns=[],this.columns[-1]=this.newColumn(-1,""),this.columns[-2]=this.newColumn(-2,"Tree"),this.allColumns[-1]=this.columns[-1],this.allColumns[-2]=this.columns[-2]},getColumn:function(t){return this.columns[t]},getColumnId:function(t){return this.getColumn(t).label},newColumn:function(t,e){var i=this.createColumnProperties();return this.getPrivateState().columnProperties[t]=i,new r(this,t,e)},addColumn:function(t,e){var i=this.newColumn(t,e);return this.columns.push(i),this.allColumns.push(i),i},createColumns:function(){},createColumnProperties:function(){var t=this.getPrivateState(),e=Object.create(t);return e.rowNumbersProperties=Object.create(e,{foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundSelectionColor},set:function(t){this.columnHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundSelectionColor},set:function(t){this.columnHeaderBackgroundSelectionColor=t}}}),e.rowHeader=Object.create(e,{font:{configurable:!0,get:function(){return this.rowHeaderFont},set:function(t){this.rowHeaderFont=t}},color:{configurable:!0,get:function(){return this.rowHeaderColor},set:function(t){this.rowHeaderColor=t}},backgroundColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundColor},set:function(t){this.rowHeaderBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderForegroundSelectionColor},set:function(t){this.rowHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundSelectionColor},set:function(t){this.rowHeaderBackgroundSelectionColor=t}}}),e.columnHeader=Object.create(e,{format:{value:"default"},font:{configurable:!0,get:function(){return this.columnHeaderFont},set:function(t){this.columnHeaderFont=t}},color:{configurable:!0,get:function(){return this.columnHeaderColor},set:function(t){this.columnHeaderColor=t}},backgroundColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundColor},set:function(t){this.columnHeaderBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundSelectionColor},set:function(t){this.columnHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundSelectionColor},set:function(t){this.columnHeaderBackgroundSelectionColor=t}}}),e.columnHeaderColumnSelection=Object.create(e.columnHeader,{foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundColumnSelectionColor},set:function(t){this.columnHeaderForegroundColumnSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundColumnSelectionColor},set:function(t){this.columnHeaderBackgroundColumnSelectionColor=t}}}),e.rowHeaderRowSelection=Object.create(e.rowHeader,{foregroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderForegroundRowSelectionColor},set:function(t){this.rowHeaderForegroundRowSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundRowSelectionColor},set:function(t){this.rowHeaderBackgroundRowSelectionColor=t}}}),e.filterProperties=Object.create(e,{font:{configurable:!0,get:function(){return this.filterFont},set:function(t){this.filterFont=t}},color:{configurable:!0,get:function(){return this.filterColor},set:function(t){this.filterColor=t}},backgroundColor:{configurable:!0,get:function(){return this.filterBackgroundColor},set:function(t){this.filterBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.filterForegroundSelectionColor},set:function(t){this.filterForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.filterBackgroundSelectionColor},set:function(t){this.filterBackgroundSelectionColor=t}},cellBorderStyle:{configurable:!0,get:function(){return this.filterCellBorderStyle},set:function(t){this.filterCellBorderStyle=t}},cellBorderThickness:{configurable:!0,get:function(){return this.filterCellBorderThickness},set:function(t){this.filterCellBorderThickness=t}}}),e.treeColumnProperties=Object.create(e,{font:{configurable:!0,get:function(){return this.treeColumnFont},set:function(t){this.treeColumnFont=t}},color:{configurable:!0,get:function(){return this.treeColumnColor},set:function(t){this.treeColumnColor=t}},backgroundColor:{configurable:!0,get:function(){return this.treeColumnBackgroundColor},set:function(t){this.treeColumnBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnForegroundSelectionColor},set:function(t){this.treeColumnForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnBackgroundSelectionColor},set:function(t){this.treeColumnBackgroundSelectionColor=t}}}),e.treeColumnPropertiesColumnSelection=Object.create(e.treeColumnProperties,{foregroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnForegroundColumnSelectionColor},set:function(t){this.treeColumnForegroundColumnSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnBackgroundColumnSelectionColor},set:function(t){this.treeColumnBackgroundColumnSelectionColor=t}}}),e},getColumnWidth:function(t){var e=this.getColumn(t);if(!e)return this.resolveProperty("defaultColumnWidth");var i=e.getWidth();return i},setColumnWidth:function(t,e){this.getColumn(t).setWidth(e),this.stateChanged()},getDataModel:function(){if(null===this.dataModel){var t=this.getDefaultDataModel();this.setDataModel(t)}return this.dataModel},getCellRenderer:function(t,e,i){return this.getColumn(e).getCellRenderer(t,i)},setDataModel:function(t){this.dataModel=t},setComplexFilter:function(t,e){var i=this.getColumn(t);i&&i.setComplexFilter(e)},getComplexFilter:function(t){var e=this.getColumn(t);return e?e.getComplexFilter():void 0},applyFilters:function(){},clearObjectProperties:function(t,e){for(var i in t)t.hasOwnProperty(i)&&(void 0===e||!e&&l.indexOf(i)>=0||e&&l.indexOf(i)<0)&&delete t[i]},getPrivateState:function(){return this.tableState||(this.tableState=this.getDefaultState()),this.tableState},getState:function(){var t=JSON.parse(JSON.stringify(this.getPrivateState()));return this.clearObjectProperties(t.columnProperties,!1),t},clearState:function(){this.tableState=null},getDefaultState:function(){var t=this.getGrid()._getProperties(),e=Object.create(t);return n(e).extendOwn({rowHeights:{},cellProperties:{},columnProperties:[]}),e},setState:function(t){if(!t.columnIndexes){var e=this.getFields();t.columnIndexes=[];for(var i=0;ii;i++)e+=this.getRowHeight(i);return e},getRowHeight:function(t){var e=this.getPrivateState();if(e.rowHeights){var i=e.rowHeights[t];if(i)return i}return this.getDefaultRowHeight()},getDefaultRowHeight:function(){return this.defaultRowHeight||(this.defaultRowHeight=this.resolveProperty("defaultRowHeight")),this.defaultRowHeight},setRowHeight:function(t,e){var i=this.getPrivateState();i.rowHeights[t]=Math.max(5,e),this.stateChanged()},getFixedRowsMaxHeight:function(){return this.getFixedRowsHeight()},getFixedColumnsWidth:function(){var t=this.getFixedColumnCount(),e=0;this.getGrid().isShowRowNumbers()&&(e=this.getColumnWidth(-1));for(var i=0;t>i;i++)e+=this.getColumnWidth(i);return e},getFixedColumnsMaxWidth:function(){var t=this.getFixedColumnsWidth();return t},_setScrollPositionY:function(t){this.setScrollPositionY(t),this.changed()},_setScrollPositionX:function(t){this.setScrollPositionX(t),this.changed()},setRenderedColumnCount:function(t){this.renderedColumnCount=t},setRenderedRowCount:function(t){this.renderedRowCount=t},_fixedRowClicked:function(t,e){var i=this.translateColumnIndex(this.getScrollPositionX()+e.gridCell.x-this.getFixedColumnCount()),n=this.grid.newPoint(i,e.gridCell.y);e.gridCell=n,this.fixedRowClicked(t,e)},_fixedColumnClicked:function(t,e){var i=this.grid.newPoint(e.gridCell.x,this.getScrollPositionY()+e.gridCell.y-this.getFixedRowCount());e.gridCell=i,this.fixedColumnClicked(t,e)},moveSingleSelect:function(t,e,i){this.featureChain&&(this.featureChain.moveSingleSelect(t,e,i),this.setCursor(t))},setCursor:function(t){t.updateCursor(),this.featureChain.setCursor(t)},onMouseMove:function(t,e){this.featureChain&&(this.featureChain.handleMouseMove(t,e),this.setCursor(t))},onTap:function(t,e){this.featureChain&&(this.featureChain.handleTap(t,e),this.setCursor(t))},onContextMenu:function(t,e){var i=t.fireSyntheticContextMenuEvent(e);i&&this.featureChain&&(this.featureChain.handleContextMenu(t,e),this.setCursor(t))},onWheelMoved:function(t,e){this.featureChain&&(this.featureChain.handleWheelMoved(t,e),this.setCursor(t))},onMouseUp:function(t,e){this.featureChain&&(this.featureChain.handleMouseUp(t,e),this.setCursor(t))},onMouseDrag:function(t,e){this.featureChain&&(this.featureChain.handleMouseDrag(t,e),this.setCursor(t))},onKeyDown:function(t,e){this.featureChain&&(this.featureChain.handleKeyDown(t,e),this.setCursor(t))},onKeyUp:function(t,e){this.featureChain&&(this.featureChain.handleKeyUp(t,e),this.setCursor(t))},onDoubleClick:function(t,e){this.featureChain&&(this.featureChain.handleDoubleClick(t,e),this.setCursor(t))},onHoldPulse:function(t,e){this.featureChain&&(this.featureChain.handleHoldPulse(t,e),this.setCursor(t))},toggleColumnPicker:function(){var t=this.grid.dialog,e=this;t.isOpen()?t.close():(this.buildColumnPicker(t.overlay),t.onClose=function(){e.updateFromColumnPicker(t.overlay)},t.open())},handleMouseDown:function(t,e){this.featureChain&&(this.featureChain.handleMouseDown(t,e),this.setCursor(t))},handleMouseExit:function(t,e){this.featureChain&&(this.featureChain.handleMouseExit(t,e),this.setCursor(t))},changed:function(){},shapeChanged:function(){},isColumnReorderable:function(){return!0},getColumnProperties:function(t){var e=this.columns[t];if(!e)return{isNull:!0};var i=e.getProperties();return i?i:{isNull:!0}},setColumnProperties:function(t,e){var i=this.allColumns[t].getProperties();n(i).extendOwn(e),this.changed()},getField:function(t){if(-1===t)return"tree";var e=this.getColumn(t);return e.getField()},getHeader:function(t){if(-1===t)return"Tree";var e=this.getColumn(t);return e.getHeader()},setColumnDescriptors:function(t){var e,i=t.visible,n=this.getPrivateState(),o=i.length,r=[];for(e=0;o>e;e++)r.push(i[e].id);n.columnIndexes=r,this.changed()},getHiddenColumnDescriptors:function(){for(var t=this.getPrivateState(),e=t.columnIndexes,i=[],n=this.getColumnCount(),o=0;n>o;o++)-1===e.indexOf(o)&&i.push({id:o,label:this.getHeader(o),field:this.getField(o)});return i},hideColumns:function(t){for(var e=this.getPrivateState(),i=e.columnIndexes,n=0;ns;s++)if(i=this.getValue(s),n=this.typeOf(i),o!==n){if(!r||"number"!=typeof i)return"mixed";o="float"}return o},typeOf:function(t){var e=typeof t;switch(e){case"object":return t.constructor.name.toLowerCase();case"number":return parseInt(t)===t?"int":"float";default:return e}},getProperties:function(){return this.behavior.getPrivateState().columnProperties[this.index]},setProperties:function(t){var e=this.behavior.getPrivateState().columnProperties[this.index];this.clearObjectProperties(e,!1),o(e).extendOwn(t)},toggleSort:function(t){this.dataModel.toggleSort(this.index,t)},getCellEditorAt:function(t,e){return this.dataModel.getCellEditorAt(this.index,e)},getHeader:function(){return this.label},getField:function(){return this.dataModel.getFields()[this.index]}},e.exports=n},{"object-iterators":18}],32:[function(t,e,i){"use strict";function n(t,e){this.setComponent(e),this.setGrid(t)}n.prototype={constructor:n.prototype.constructor,component:null,grid:null,getGrid:function(){return this.grid},setGrid:function(t){this.grid=t,this.getComponent().setGrid(t)},getBehavior:function(){return this.getGrid().getBehavior()},changed:function(){this.getBehavior().changed()},getPrivateState:function(){return this.getGrid().getPrivateState()},applyState:function(){},setComponent:function(t){this.component=t},getComponent:function(){return this.component},setGlobalFilter:function(t){return this.getComponent().setGlobalFilter(t)},getData:function(){return this.getComponent().getData()},getFilteredData:function(){return this.getComponent().getFilteredData()},getValue:function(t,e){return this.getComponent().getValue(t,e)},getUnfilteredValue:function(t,e){return this.getComponent().getUnfilteredValue(t,e)},setValue:function(t,e,i){this.getComponent().setValue(t,e,i)},getColumnCount:function(){return this.getComponent().getColumnCount()},applyFilters:function(){return this.getComponent().applyFilters()},getRowCount:function(){return this.getComponent().getRowCount()},getCellRenderer:function(t,e,i,n,o){return this.getComponent().getCellRenderer(t,e,i,n,o)},getRowHeight:function(t){return this.getComponent().getRowHeight(t)},getColumnEdge:function(t,e){return this.getComponent().getColumnEdge(t,e)},getColumnWidth:function(t){return this.getComponent().getColumnWidth(t)},setColumnWidth:function(t,e){this.getComponent().setColumnWidth(t,e)},toggleSort:function(t,e){this.getComponent().toggleSort(t,e)},getColumnProperties:function(t){return this.getComponent().getColumnProperties(t)},setColumnProperties:function(t,e){this.getComponent().setColumnProperties(t,e)},getHeaders:function(){return this.getComponent().getHeaders()},getFields:function(){return this.getComponent().getFields()},setFields:function(t){this.getComponent().setFields(t)},getCellProperties:function(t,e){return this.getComponent().getCellProperties(t,e)},setCellProperties:function(t,e,i){this.getComponent().setCellProperties(t,e,i)},getRow:function(t){return this.getComponent().getRow(t)},getTopTotals:function(){return this.getComponent().getTopTotals()},setTopTotals:function(t){this.getComponent().setTopTotals(t)},getBottomTotals:function(){return this.getComponent().getBottomTotals()},setBottomTotals:function(t){this.getComponent().setBottomTotals(t)},setData:function(t){return this.getComponent().setData(t)},hasHierarchyColumn:function(){return this.getComponent().hasHierarchyColumn()},setHeaders:function(t){return this.getComponent().setHeaders(t)},cellClicked:function(t,e){return this.getComponent().cellClicked(t,e)},getAvailableGroups:function(){return this.getComponent().getAvailableGroups()},getGroups:function(){return this.getComponent().getGroups()},setGroups:function(t){this.getComponent().setGroups(t)},getHiddenColumns:function(){return this.getComponent().getHiddenColumns()},getVisibleColumns:function(){return this.getComponent().getVisibleColumns()},setAggregates:function(t){return this.getComponent().setAggregates(t)},reset:function(){this.getComponent().reset()},getCellEditorAt:function(t,e){return this.getComponent().getCellEditorAt(t,e)},getUnfilteredRowCount:function(){return this.getComponent().getUnfilteredRowCount()}},e.exports=n},{}],33:[function(t,e,i){"use strict";var n=t("list-dragon"),o=t("./Behavior"),r=t("./DataModelDecorator"),s=t("../dataModels/JSON"),l=t("../features/index"),a=t("../stylesheets"),u=t("../local_node_modules/finanalytics").aggregations,c=o.extend("behaviors.JSON",{initialize:function(t,e){this.setData(e)},features:[l.CellSelection,l.KeyPaging,l.ColumnPicker,l.ColumnResizing,l.RowResizing,l.Filters,l.RowSelection,l.ColumnSelection,l.ColumnMoving,l.ColumnSorting,l.CellEditing,l.CellClick,l.OnHover],aggregations:u,createColumns:function(){var t=this.getDataModel(),e=t.getColumnCount(),i=t.getHeaders(),n=t.getFields();this.clearColumns();for(var o=0;e>o;o++){var r=i[o],s=this.addColumn(o,r),l=s.getProperties();l.field=n[o],l.header=r,l.complexFilter=null}},getDefaultDataModel:function(){var t=new s,e=new r(this.getGrid(),t);return e.setComponent(t),e},applyFilters:function(){this.dataModel.applyFilters()},setHeaders:function(t){this.getDataModel().setHeaders(t)},getHeaders:function(){return this.getDataModel().getHeaders()},setFields:function(t){this.getDataModel().setFields(t),this.createColumns()},getFields:function(){return this.getDataModel().getFields()},setData:function(t){this.getDataModel().setData(t),this.createColumns();var e=this;this.getGrid().isColumnAutosizing()?(setTimeout(function(){e.autosizeAllColumns()},100),e.changed()):setTimeout(function(){e.allColumns[-1].checkColumnAutosizing(!0),e.changed()})},setTopTotals:function(t){this.getDataModel().setTopTotals(t)},getTopTotals:function(){return this.getDataModel().getTopTotals()},setBottomTotals:function(t){this.getDataModel().setBottomTotals(t)},getBottomTotals:function(){return this.getDataModel().getBottomTotals()},setColumns:function(t){this.getDataModel().setColumns(t)},enhanceDoubleClickEvent:function(t){t.row=this.getRow(t.gridCell.y)},setDataProvider:function(t){this.getDataModel().setDataProvider(t)},hasHierarchyColumn:function(){return this.getDataModel().hasHierarchyColumn()},getColumnAlignment:function(t){return 0===t&&this.hasHierarchyColumn()?"left":"center"},getRowSelectionMatrix:function(t){return this.getDataModel().getRowSelectionMatrix(t)},getColumnSelectionMatrix:function(t){return this.getDataModel().getColumnSelectionMatrix(t)},getSelectionMatrix:function(t){return this.getDataModel().getSelectionMatrix(t)},getRowSelection:function(){var t=this.getSelectedRows();return this.getDataModel().getRowSelection(t)},getColumnSelection:function(){var t=this.getSelectedColumns();return this.getDataModel().getColumnSelection(t)},getSelection:function(){var t=this.getSelections();return this.getDataModel().getSelection(t)},buildColumnPicker:function(t){if(!this.isColumnReorderable())return!1;var e={cssStylesheetReferenceElement:t},i={models:this.getGroups(),title:"Groups"},o={models:this.getAvailableGroups(),title:"Available Groups"},r={models:this.getHiddenColumns(),title:"Hidden Columns"},s={models:this.getVisibleColumns(),title:"Visible Columns"},l=new n([i,o],e),u=new n([r,s],e),c=[l,u];return a("list-dragon",t),c.forEach(function(e){e.modelLists.forEach(function(e){t.appendChild(e.container)})}),t.lists={group:i.models,availableGroups:o.models,hidden:r.models,visible:s.models},!0},getGroups:function(){return this.getDataModel().getGroups()},getAvailableGroups:function(){return this.getDataModel().getAvailableGroups()},getHiddenColumns:function(){return this.getDataModel().getHiddenColumns()},getVisibleColumns:function(){return this.getDataModel().getVisibleColumns()},setColumnDescriptors:function(t){var e=this.columns[0];this.columns.length=0,e&&"Tree"===e.label&&this.columns.push(e);for(var i=0;iu;u++){var c=e.getUnfilteredValue(l,u);a.set(c,c)}var h=a.values;h.sort(),h.length>0&&h[0].length>0&&h.unshift(""),this.setItems(h)}},showEditor:function(){var t=this;this.input.style.display="inline",setTimeout(function(){t.showDropdown(t.input)},50)},preShowEditorNotification:function(){this.autopopulate(),this.setEditorValue(this.initialValue)},setItems:function(t){this.items=t,this.updateView()},initializeInput:function(t){var e=this;n.prototype.initializeInput.apply(this,[t]),t.onchange=function(){e.stopEditing()}}});e.exports=r},{"../Mappy":26,"./Simple":41}],38:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Color",{alias:"color",template:function(){}});e.exports=o},{"./Simple":41}],39:[function(t,e,i){"use strict";var n=t("./Simple"),o=function(t){var e=t.match(/(\d+)/g);return new window.Date(e[0],e[1]-1,e[2])},r=function(t){return 10>t?"0"+t:t+""},s=n.extend("Date",{alias:"date",template:function(){},setEditorValue:function(t){t&&"Date"===t.constructor.name&&(t=t.getFullYear()+"-"+r(t.getMonth()+1)+"-"+r(t.getDay())),this.getInput().value=t+""},getEditorValue:function(){var t=this.getInput().value;return t=o(t)}});e.exports=s},{"./Simple":41}],40:[function(t,e,i){"use strict";var n=t("./CellEditor"),o=n.extend("Filter",{alias:"filter",initialize:function(){var t=document.createElement("div"),e=t.style;e.position="absolute",e.top=e.bottom="44px",e.right=e.left="1em",e.overflowY="scroll";var i=document.createElement("table");t.appendChild(i),e=i.style,e.width=e.height="100%";var n=document.createElement("tr"),o=document.createElement("td");i.appendChild(n),n.appendChild(o),this.title=document.createElement("div"),this.title.innerHTML="Filter Editor",this.dialog=document.createElement("div"),this.content=o,this.buttons=document.createElement("div"),e=this.dialog.style,e.position="absolute",e.top=e.left=e.right=e.bottom=0,e.whiteSpace="nowrap",e=this.title.style,e.position="absolute",e.top=e.left=e.right=0,e.height="44px",e.whiteSpace="nowrap",e.textAlign="center",e.padding="11px",e=this.buttons.style,e.position="absolute",e.left=e.right=e.bottom=0,e.height="44px",e.whiteSpace="nowrap",e.textAlign="center",e.padding="8px",this.dialog.appendChild(this.title),this.dialog.appendChild(t),this.dialog.appendChild(this.buttons),this.ok=document.createElement("button"),this.ok.style.borderRadius="8px",this.ok.style.width="5.5em",this.cancel=document.createElement("button"),this.cancel.style.marginLeft="2em",this.cancel.style.borderRadius="8px",this.cancel.style.width="5.5em",this["delete"]=document.createElement("button"),this["delete"].style.marginLeft="2em",this["delete"].style.borderRadius="8px",this["delete"].style.width="5.5em",this.reset=document.createElement("button"),this.reset.style.marginLeft="2em",this.reset.style.borderRadius="8px",this.reset.style.width="5.5em",this.ok.innerHTML="ok",this.cancel.innerHTML="cancel",this["delete"].innerHTML="delete",this.reset.innerHTML="reset",this.buttons.appendChild(this.ok),this.buttons.appendChild(this.reset),this.buttons.appendChild(this["delete"]),this.buttons.appendChild(this.cancel);var r=this;this.ok.onclick=function(){r.okPressed()},this.cancel.onclick=function(){r.cancelPressed()},this["delete"].onclick=function(){r.deletePressed()},this.reset.onclick=function(){r.resetPressed()}},tearDown:function(){this.content.innerHTML=""},okPressed:function(){var t=this.getGrid().dialog;t.onOkPressed()},cancelPressed:function(){var t=this.getGrid().dialog;t.onCancelPressed()},deletePressed:function(){var t=this.getGrid().dialog;t.onDeletePressed()},resetPressed:function(){var t=this.getGrid().dialog;t.onResetPressed()},beginEditAt:function(t){var e=this.getGrid(),i=e.getBehavior(),n=e.dialog,o=t.x,r=i.getColumnId(o),s=i.getField(o),l=i.getColumn(o).getType(),a=function(){return[{name:s,alias:r,type:l}]};this.title.innerHTML="filter for "+r+" column";var u=e.getFilterFor(o);if(n.isOpen())n.close();else{var c=this;n.clear(),n.overlay.appendChild(this.dialog),u.initialize(a),n.onOkPressed=function(){u.onOk&&u.onOk()||(c.tearDown(),i.setComplexFilter(o,{type:u.alias,state:u.getState()}),n.close(),i.applyFilters(),i.changed())},n.onCancelPressed=function(){u.onCancel&&u.onCancel()||(c.tearDown(),n.close(),u=void 0)},n.onDeletePressed=function(){u.onDelete&&u.onDelete()||(c.tearDown(),i.setComplexFilter(o,void 0),n.close(),i.applyFilters(),i.changed())},n.onResetPressed=function(){u.onReset&&u.onReset()||(c.tearDown(),u.initialize(n),u.onShow&&u.onShow(c.content))};var h=e._getBoundsOfCell(o,t.y),d=e.div.getBoundingClientRect().left-e.divCanvas.getBoundingClientRect().left;h.x=h.x-d,n.openFrom(h);var g=i.getComplexFilter(o);g&&u.setState(g.state),setTimeout(function(){u.onShow&&u.onShow(c.content)},n.getAnimationTime()+10)}}});e.exports=o},{"./CellEditor":36}],41:[function(t,e,i){"use strict";var n=t("./CellEditor.js"),o=n.extend("Simple",{input:null,alias:"simple",initialize:function(){this.editorPoint={x:0,y:0}},initializeInput:function(t){var e=this;t.addEventListener("keyup",function(t){!t||13!==t.keyCode&&27!==t.keyCode&&8!==t.keyCode||(t.preventDefault(),8===t.keyCode?e.clearStopEditing():27===t.keyCode?e.cancelEditing():e.stopEditing(),e.getGrid().repaint(),e.getGrid().takeFocus()),e.getGrid().fireSyntheticEditorKeyUpEvent(e,t)}),t.addEventListener("keydown",function(t){e.getGrid().fireSyntheticEditorKeyDownEvent(e,t)}),t.addEventListener("keypress",function(t){e.getGrid().fireSyntheticEditorKeyPressEvent(e,t)}),t.onblur=function(t){e.cancelEditing()},t.style.position="absolute",t.style.display="none",t.style.border="solid 2px black",t.style.outline=0,t.style.padding=0,t.style.zIndex=1e3,t.style.boxShadow="white 0px 0px 1px 1px"},getEditorValue:function(){var t=this.getInput().value;return t},setEditorValue:function(t){this.getInput().value=t+""},clearStopEditing:function(){this.setEditorValue(""),this.stopEditing()},cancelEditing:function(){this.isEditing&&(this.getInput().value=null,this.isEditing=!1,this.hideEditor())},showEditor:function(){this.getInput().style.display="inline"},hideEditor:function(){this.getInput().style.display="none"},takeFocus:function(){var t=this;setTimeout(function(){t.input.focus(),t.selectAll()},300)},selectAll:function(){},originOffset:function(){return[0,0]},setBounds:function(t){var e=this.originOffset(),i="translate("+(t.x-1+e[0])+"px,"+(t.y-1+e[1])+"px)",n=this.getInput();n.style.boxSizing="border-box",n.style.webkitTransform=i,n.style.MozTransform=i,n.style.msTransform=i,n.style.OTransform=i,n.style.width=t.width+2+"px",n.style.height=t.height+2+"px"},saveEditorValue:function(){var t=this.getEditorPoint(),e=this.getEditorValue();if(e!==this.initialValue){parseFloat(this.initialValue)===this.initialValue&&(e=parseFloat(e));var i=this.getGrid().fireBeforeCellEdit(t,this.initialValue,e,this);i&&(this.getBehavior().setValue(t.x,t.y,e),this.getGrid().fireAfterCellEdit(t,this.initialValue,e,this))}},_moveEditor:function(){var t=this.getGrid(),e=this.getEditorPoint(),i=t._getBoundsOfCell(e.x,e.y),n=t.div.getBoundingClientRect().left-t.divCanvas.getBoundingClientRect().left;i.x=i.x-n,this.setBounds(i)},moveEditor:function(){this._moveEditor(),this.takeFocus()},beginEditAt:function(t){this.isAdded||(this.isAdded=!0,this.grid.div.appendChild(this.getInput())),this.setEditorPoint(t);var e=this.getBehavior(),i=e.getValue(t.x,t.y);"Array"===i.constructor.name&&(i=i[1]);var n=this.grid.fireRequestCellEdit(t,i);n&&(this.initialValue=i,this.isEditing=!0,this.setCheckEditorPositionFlag(),this.checkEditor())},checkEditor:function(){if(this.checkEditorPositionFlag&&(this.checkEditorPositionFlag=!1,this.isEditing)){var t=this.getEditorPoint();this.grid.isDataVisible(t.x,t.y)?(this.preShowEditorNotification(),this.attachEditor(),this.moveEditor(),this.showEditor()):this.hideEditor()}},attachEditor:function(){var t=this.getInput();this.grid.div.appendChild(t)},preShowEditorNotification:function(){this.setEditorValue(this.initialValue)},getInput:function(){return this.input||(this.input=this.getDefaultInput()),this.input},getDefaultInput:function(){var t=document.createElement("DIV");t.innerHTML=this.getHTML();var e=t.firstChild;return this.initializeInput(e),e},updateView:function(){var t=this.getInput(),e=t.parentNode,i=this.getDefaultInput();this.input=i,e.replaceChild(i,t)},showDropdown:function(t){var e;e=document.createEvent("MouseEvents"),e.initMouseEvent("mousedown",!0,!0,window),t.dispatchEvent(e)}});e.exports=o},{"./CellEditor.js":36}],42:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Slider",{alias:"slider",template:function(){}});e.exports=o},{"./Simple":41}],43:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Spinner",{alias:"spinner",template:function(){}});e.exports=o},{"./Simple":41}],44:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Textfield",{alias:"textfield",template:function(){},selectAll:function(){this.input.setSelectionRange(0,this.input.value.length)},initializeInput:function(t){var e=this;t.addEventListener("keyup",function(t){!t||13!==t.keyCode&&27!==t.keyCode||(t.preventDefault(),27===t.keyCode?e.cancelEditing():e.stopEditing(),e.getGrid().repaint(),e.getGrid().takeFocus()),e.getGrid().isFilterRow(e.getEditorPoint().y)&&setTimeout(function(){e.saveEditorValue(),e._moveEditor()}),e.getGrid().fireSyntheticEditorKeyUpEvent(e,t)}),t.addEventListener("keydown",function(t){e.getGrid().fireSyntheticEditorKeyDownEvent(e,t)}),t.addEventListener("keypress",function(t){e.getGrid().fireSyntheticEditorKeyPressEvent(e,t)}),t.onblur=function(t){e.cancelEditing()},t.style.position="absolute",t.style.display="none",t.style.border="solid 2px black",t.style.outline=0,t.style.padding=0,t.style.zIndex=1e3,t.style.boxShadow="white 0px 0px 1px 1px"}});e.exports=o},{"./Simple":41}],45:[function(t,e,i){"use strict";e.exports={CellEditor:t("./CellEditor"),Textfield:t("./Textfield"),Choice:t("./Choice"),Color:t("./Color"),Date:t("./Date"),Slider:t("./Slider"),Spinner:t("./Spinner"),Filter:t("./Filter")}},{"./CellEditor":36,"./Choice":37,"./Color":38,"./Date":39,"./Filter":40,"./Slider":42,"./Spinner":43,"./Textfield":44}],46:[function(t,e,i){"use strict";var n=t("extend-me").Base,o="A".charCodeAt(0),r=n.extend("DataModel",{next:null,grid:null,setGrid:function(t){this.grid=t},getGrid:function(){return this.grid},getBehavior:function(){return this.getGrid().getBehavior()},changed:function(){this.getBehavior().changed()},getPrivateState:function(){return this.getGrid().getPrivateState()},applyState:function(){},alphaFor:function(t){var e=Math.floor(t/26),i=t%26,n="";return e>0&&(n+=this.alpha(e-1)),n+=this.alpha(i)},alpha:function(t){return String.fromCharCode(o+t)},getCellEditorAt:function(t,e){}});e.exports=r},{"extend-me":4}],47:[function(t,e,i){"use strict";function n(t){return"function"==typeof t?t():t}function o(t){return t=t.toLowerCase(),function(e){return e=n(e),(e+"").toLowerCase().indexOf(t)>-1}}var r=t("../local_node_modules/finanalytics"),s=t("./DataModel"),l=t("../../images"),a="▲",u="▼",c={isNullObject:function(){return!0},getFields:function(){return[]},getHeaders:function(){return[]},getColumnCount:function(){return 0},getRowCount:function(){return 0},getAggregateTotals:function(){return[]},hasAggregates:function(){return!1},hasGroups:function(){return!1},getRow:function(){return null}},h=s.extend("dataModels.JSON",{source:c,preglobalfilter:c,prefilter:c,presorter:c,analytics:c,postglobalfilter:c,postfilter:c,postsorter:c,topTotals:[],bottomTotals:[],hasAggregates:function(){return this.analytics.hasAggregates()},hasGroups:function(){return this.analytics.hasGroups()},getDataSource:function(){return this.postsorter},getFilterSource:function(){return this.postfilter},getGlobalFilterSource:function(){return this.postglobalfilter},getSortingSource:function(){return this.postsorter},getData:function(){return this.source.data},getFilteredData:function(){for(var t=this.getDataSource(),e=t.getRowCount(),i=new Array(e),n=0;e>n;n++)i[n]=t.getRow(n);return i},getValue:function(t,e){var i,n=this.hasHierarchyColumn(),o=this.getGrid(),r=o.getHeaderRowCount();return n?-2===t&&(t=0):this.hasAggregates()&&(t+=1),i=r>e?this.getHeaderRowValue(t,e):this.getDataSource().getValue(t,e-r)},getHeaderRowValue:function(t,e){var i;if(void 0===e)i=this.getHeaders()[Math.max(t,0)];else if(0>e){var n=this.getBottomTotals();i=n[n.length+e][t]}else{var o=this.getGrid(),r=o.isShowFilterRow(),s=o.isShowHeaderRow(),a=(r?1:0)+(s?1:0);if(e>=a)i=this.getTopTotals()[e-a][t];else{if(!s||0!==e){i=this.getFilter(t);var u=l.filter(i.length);return[null,i,u]}i=this.getHeaders()[t];var c=this.getSortImageForColumn(t);c&&(i=c+i)}}return i},setValue:function(t,e,i){var n=this.hasHierarchyColumn(),o=this.getGrid(),r=o.getHeaderRowCount();n?-2===t&&(t=0):this.hasAggregates()&&(t+=1),r>e?this.setHeaderRowValue(t,e,i):this.getDataSource().setValue(t,e-r,i),this.changed()},setHeaderRowValue:function(t,e,i){if(void 0===i)return this._setHeader(t,e);var n=this.getGrid(),o=n.isShowFilterRow(),r=n.isShowHeaderRow(),s=o&&r,l=(o?1:0)+(r?1:0);if(e>=l)this.getTopTotals()[e-l][t]=i;else{if(-1===t)return;if(s){if(0===e)return this._setHeader(t,i);this.setFilter(t,i)}else{if(!o)return this._setHeader(t,i);this.setFilter(t,i)}}return""},getColumnProperties:function(t){var e=this.getBehavior().allColumns[t];return e?e.getProperties():void 0},getFilter:function(t){var e=this.getColumnProperties(t);return e?e.filter||"":""},getComplexFilter:function(t){var e=this.getColumnProperties(t);if(!e)return"";var i=e.complexFilter;if(i){var o=this.getGrid().resolveFilter(i.type),r=o.create(i.state);return function(t){var e=n(t);return r(e)}}},setFilter:function(t,e){var i=this.getColumnProperties(t);i.filter=e,this.applyAnalytics()},getColumnCount:function(){var t=this.getGrid().resolveProperty("showTreeColumn")===!0,e=this.hasAggregates(),i=e&&!t?-1:0;return this.analytics.getColumnCount()+i},getRowCount:function(){var t=this.getGrid(),e=this.getDataSource().getRowCount();return e+=t.getHeaderRowCount()},getHeaders:function(){return this.analytics.getHeaders()},setHeaders:function(t){this.getDataSource().setHeaders(t)},setFields:function(t){this.getDataSource().setFields(t)},getFields:function(){return this.getDataSource().getFields()},setData:function(t){this.source=new r.JSDataSource(t),this.analytics=new r.DataSourceAggregator(this.source),this.postglobalfilter=new r.DataSourceGlobalFilter(this.analytics),this.postfilter=new r.DataSourceFilter(this.postglobalfilter),this.postsorter=new r.DataSourceSorterComposite(this.postfilter),this.applyAnalytics()},setTopTotals:function(t){this.topTotals=t},getTopTotals:function(){return this.hasAggregates()?this.getDataSource().getGrandTotals():this.topTotals},setBottomTotals:function(t){this.bottomTotals=t},getBottomTotals:function(){return this.hasAggregates()?this.getDataSource().getGrandTotals():this.bottomTotals},setGroups:function(t){this.analytics.setGroupBys(t),this.applyAnalytics(),this.getGrid().fireSyntheticGroupsChangedEvent(this.getGroups())},getGroups:function(){for(var t=this.getHeaders().slice(0),e=this.getFields().slice(0),i=this.analytics.groupBys,n=[],o=0;o0&&o(l);h&&(n.add(s-r,this.createFormattedFilter(u,h)),i.push({column:e.label,format:c?"complex":a}))}.bind(this)),n.applyAll(),t.fireSyntheticFilterAppliedEvent({details:i})},createFormattedFilter:function(t,e){return function(i){var n=t(i);return e(n)}},toggleSort:function(t,e){this.incrementSortState(t,e),this.applyAnalytics()},incrementSortState:function(t,e){t++;var i=this.getPrivateState(),n=e.indexOf("CTRL")>-1;i.sorts=i.sorts||[];var o=i.sorts.indexOf(t);-1===o&&(o=i.sorts.indexOf(-1*t)),o>-1?i.sorts[o]>0?i.sorts[o]=-1*i.sorts[o]:i.sorts.splice(o,1):n||0===i.sorts.length?i.sorts.unshift(t):(i.sorts.length=0,i.sorts.unshift(t)),i.sorts.length>3&&(i.sorts.length=3)},applySorts:function(){var t=this.getSortingSource(),e=this.getPrivateState().sorts,i=this.hasAggregates()?1:0;if(e&&0!==e.length)for(var n=0;nn&&(n=i.indexOf(-1*t),e=!1),0>n)return null;var o=i.length-n,r=e?a:u;return o+r+" "},cellClicked:function(t,e){if(this.hasAggregates()&&0===e.gridCell.x){var i=this.getGrid(),n=i.getHeaderRowCount(),o=e.gridCell.y-n;this.getDataSource().click(o),this.applyFilters(),this.applySorts(),this.changed()}},getRow:function(t){var e=this.getGrid(),i=e.getHeaderRowCount();if(i>t&&!this.hasAggregates()){var n=this.getTopTotals();return n[t-(i-n.length)]}return this.getDataSource().getRow(t-i)},buildRow:function(t){var e=this.getColumnCount(),i=[].concat(this.getFields()),n={};this.hasAggregates()&&(n.tree=this.getValue(-2,t),i.shift());for(var o=0;e>o;o++)n[i[o]]=this.getValue(o,t);return n},getComputedRow:function(t){for(var e=this.getRowContextFunction([t]),i=this.getFields(),n={},o=0;o=o&&i.x>=r?t.cellClicked(e):this.next&&this.next.handleTap(t,e)}});e.exports=o},{"./Feature.js":59}],51:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("CellEditing",{alias:"CellEditing",handleDoubleClick:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,i)?t._activateEditor(e):this.next&&this.next.handleDoubleClick(t,e)},handleTap:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,!i)?t._activateEditor(e):this.next&&this.next.handleTap(t,e)},handleHoldPulse:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,!i)?t._activateEditor(e):this.next&&this.next.handleHoldPulse(t,e)},checkActivateEditor:function(t,e,i){var n=t.getBehavior(),o=n.getHeaderRowCount(),r=n.getHeaderColumnCount(),s=e.gridCell,l=t.isFilterRow(s.y),a=i&&s.x>=r&&(l||s.y>=o);return a}});e.exports=o},{"./Feature.js":59}],52:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("CellSelection",{alias:"CellSelection",currentDrag:null,lastDragCell:null,sbLastAuto:0,sbAutoStart:0,handleMouseUp:function(t,e){this.dragging&&(this.dragging=!1),this.next&&this.next.handleMouseUp(t,e)},handleMouseDown:function(t,e){var i=e.primitiveEvent.detail.isRightClick,n=t.getBehavior(),o=e.gridCell,r=e.viewPoint,s=o.x,l=o.y,a=n.getHeaderRowCount(),u=n.getHeaderColumnCount(),c=n.getColumnCount(),h=r.x>=c,d=a>l||u>s;if(!t.isCellSelection()||i||d||h)this.next&&this.next.handleMouseDown(t,e);else{var g=t.getFixedColumnCount(),f=t.getFixedRowCount();r.xn.origin.x+n.extent.x&&(o=1),this.currentDrag.y>n.origin.y+n.extent.y&&(r=1);var c=o,h=r;a&&(c=0),u&&(h=0),this.lastDragCell=i.plusXY(c,h),t.scrollBy(o,r),this.handleMouseDragCellSelection(t,i,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){var n=-1!==i.indexOf("CTRL"),o=-1!==i.indexOf("SHIFT"),r=t.getMouseDown(),s=e.x,l=e.y;if(!(0>s||0>l)){if(s===r.x&&l===r.y)return t.clearMostRecentSelection(),t.popMouseDown(),void t.repaint();n||o||t.clearSelections(),o?(t.clearMostRecentSelection(),t.select(r.x,r.y,s-r.x+1,l-r.y+1),t.setDragExtent(t.newPoint(s-r.x+1,l-r.y))):(t.select(s,l,0,0),t.setMouseDown(t.newPoint(s,l)),t.setDragExtent(t.newPoint(0,0))),t.repaint()}},handleDOWNSHIFT:function(t){this.moveShiftSelect(t,0,1)},handleUPSHIFT:function(t){this.moveShiftSelect(t,0,-1)},handleLEFTSHIFT:function(t){this.moveShiftSelect(t,-1,0)},handleRIGHTSHIFT:function(t){this.moveShiftSelect(t,1,0)},handleDOWN:function(t,e){e.primitiveEvent.preventDefault();var i=this.getAutoScrollAcceleration();this.moveSingleSelect(t,0,i)},handleUP:function(t,e){e.primitiveEvent.preventDefault();var i=this.getAutoScrollAcceleration();this.moveSingleSelect(t,0,-i)},handleLEFT:function(t){this.moveSingleSelect(t,-1,0)},handleRIGHT:function(t){this.moveSingleSelect(t,1,0)},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now()},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e,i){var n=t.getColumnCount()-1,o=t.getRowCount()-1,r=t.getVisibleColumns()-1,s=t.getVisibleRows()-1;t.resolveProperty("scrollingEnabled")||(n=Math.min(n,r),o=Math.min(o,s));var l=t.getMouseDown(),a=t.getDragExtent(),u=a.x+e,c=a.y+i;u=Math.min(n-l.x,Math.max(-l.x,u)),c=Math.min(o-l.y,Math.max(-l.y,c)),t.clearMostRecentSelection(),t.select(l.x,l.y,u,c),t.setDragExtent(t.newPoint(u,c)),t.insureModelColIsVisible(u+l.x,e)&&this.pingAutoScroll(),t.insureModelRowIsVisible(c+l.y,i)&&this.pingAutoScroll(),t.repaint()},moveSingleSelect:function(t,e,i){var n=t.getColumnCount()-1,o=t.getRowCount()-1,r=t.getVisibleColumnsCount()-1,s=t.getVisibleRowsCount()-1,l=t.getHeaderRowCount(),a=t.getHeaderColumnCount();t.resolveProperty("scrollingEnabled")||(n=Math.min(n,r),o=Math.min(o,s));var u=t.getMouseDown().plus(t.getDragExtent()),c=u.x+e,h=u.y+i;c=Math.min(n,Math.max(a,c)),h=Math.min(o,Math.max(l,h)),t.clearSelections(),t.select(c,h,0,0),t.setMouseDown(t.newPoint(c,h)),t.setDragExtent(t.newPoint(0,0)),t.insureModelColIsVisible(c,e)&&this.pingAutoScroll(),t.insureModelRowIsVisible(h,i)&&this.pingAutoScroll(),t.repaint()}});e.exports=o},{"./Feature.js":59}],53:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnAutosizing",{alias:"ColumnAutosizing",handleDoubleClick:function(t,e){var i=t.getHeaderRowCount(),n=e.gridCell;n.y<=i?t.autosizeColumn(n.x):this.next&&this.next.handleDoubleClick(t,e)}});e.exports=o},{"./Feature.js":59}],54:[function(t,e,i){"use strict";var n,o,r,s,l=t("./Feature.js"),a=150,u=l.extend("ColumnMoving",{alias:"ColumnMoving",floaterAnimationQueue:[],columnDragAutoScrollingRight:!1,columnDragAutoScrollingLeft:!1,dragArmed:!1,dragging:!1,dragCol:-1,dragOffset:0,initializeOn:function(t){this.isFloatingNow=!1,this.initializeAnimationSupport(t),this.next&&this.next.initializeOn(t)},initializeAnimationSupport:function(t){n||(n=document.createElement("canvas"),n.setAttribute("width","0px"),n.setAttribute("height","0px"), -document.body.appendChild(n),o=n.getContext("2d")),r||(r=document.createElement("canvas"),r.setAttribute("width","0px"),r.setAttribute("height","0px"),document.body.appendChild(r),s=r.getContext("2d"))},getCanDragCursorName:function(){return"-webkit-grab"},getDraggingCursorName:function(){return"-webkit-grabbing"},handleMouseDrag:function(t,e){var i,n=e.gridCell,o=Math.abs(e.primitiveEvent.detail.dragstart.x-e.primitiveEvent.detail.mouse.x);return 10>o?void(this.next&&this.next.handleMouseDrag(t,e)):(this.isHeaderRow(t,e)&&this.dragArmed&&!this.dragging?(this.dragging=!0,this.dragCol=n.x,this.dragOffset=e.mousePoint.x,this.detachChain(),i=e.primitiveEvent.detail.mouse.x-this.dragOffset,this.createDragColumn(t,i,this.dragCol)):this.next&&this.next.handleMouseDrag(t,e),void(this.dragging&&(i=e.primitiveEvent.detail.mouse.x-this.dragOffset,this.dragColumn(t,i))))},handleMouseDown:function(t,e){t.getBehavior().isColumnReorderable()&&this.isHeaderRow(t,e)&&-1!==e.gridCell.x&&(this.dragArmed=!0,this.cursor=this.getDraggingCursorName(),t.clearSelections()),this.next&&this.next.handleMouseDown(t,e)},handleMouseUp:function(t,e){if(this.dragging){this.cursor=null;var i=this;this.endDragColumn(t),setTimeout(function(){i.attachChain()},200)}this.dragCol=-1,this.dragging=!1,this.dragArmed=!1,this.cursor=null,t.repaint(),this.next&&this.next.handleMouseUp(t,e)},handleMouseMove:function(t,e){!this.dragging&&e.mousePoint.y<5&&0===e.viewPoint.y?this.cursor=this.getCanDragCursorName():this.cursor=null,this.next&&this.next.handleMouseMove(t,e),this.isHeaderRow(t,e)&&this.dragging&&(this.cursor=this.getDraggingCursorName())},floatColumnTo:function(t,e){this.floatingNow=!0;var i,n,o=t.getRenderer(),r=o.getColumnEdges(),s=t.getHScrollValue(),l=t.renderOverridesCache.floater.columnIndex,a=t.renderOverridesCache.dragger.columnIndex,u=t.renderOverridesCache.dragger.hdpiratio,c=t.getFixedColumnCount(),h=t.getColumnWidth(a),d=t.getColumnWidth(l),g=t.getVisibleColumnsCount(),f=0,p=0;a>=c&&(f=s),l>=c&&(p=s),e?(i=r[Math.min(g,a-f)],n=r[Math.min(g,l-p)],t.renderOverridesCache.dragger.startX=(i+d)*u,t.renderOverridesCache.floater.startX=i*u):(n=r[Math.min(g,l-p)],i=n+h,t.renderOverridesCache.dragger.startX=n*u,t.renderOverridesCache.floater.startX=i*u),t.swapColumns(a,l),t.renderOverridesCache.dragger.columnIndex=l,t.renderOverridesCache.floater.columnIndex=a,this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(t,n,i)),this.doFloaterAnimation(t)},doColumnMoveAnimation:function(t,e,i){var n=this;return function(){var o=r;o.style.display="inline",n.setCrossBrowserProperty(o,"transform","translate("+e+"px, 0px)"),requestAnimationFrame(function(){n.setCrossBrowserProperty(o,"transition",(n.isWebkit?"-webkit-":"")+"transform "+a+"ms ease"),n.setCrossBrowserProperty(o,"transform","translate("+i+"px, -2px)")}),t.repaint(),setTimeout(function(){n.setCrossBrowserProperty(o,"transition",""),t.renderOverridesCache.floater=null,t.repaint(),n.doFloaterAnimation(t),requestAnimationFrame(function(){o.style.display="none",n.isFloatingNow=!1})},a+50)}},doFloaterAnimation:function(t){if(0===this.floaterAnimationQueue.length)return this.floatingNow=!1,void t.repaint();var e=this.floaterAnimationQueue.pop();e()},createFloatColumn:function(t,e){var i=t.getFixedColumnCount(),n=t.getHScrollValue();i>e&&(n=0);var o=t.getRenderer(),l=o.getColumnEdges(),a=t.getColumnWidth(e),u=t.div.clientHeight,c=r,h=c.style,d=t.div.getBoundingClientRect();h.top=d.top-2+"px",h.left=d.left+"px",h.position="fixed";var g=t.getHiDPI(s);c.setAttribute("width",Math.round(a*g)+"px"),c.setAttribute("height",Math.round(u*g)+"px"),h.boxShadow="0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)",h.width=a+"px",h.height=u+"px",h.borderTop="1px solid "+o.resolveProperty("lineColor"),h.backgroundColor=o.resolveProperty("backgroundColor");var f=l[e-n];f*=g,s.scale(g,g),t.renderOverridesCache.floater={columnIndex:e,ctx:s,startX:f,width:a,height:u,hdpiratio:g},h.zIndex="4",this.setCrossBrowserProperty(c,"transform","translate("+f+"px, -2px)"),h.cursor=this.getDraggingCursorName(),t.repaint()},setCrossBrowserProperty:function(t,e,i){var n=e[0].toUpperCase()+e.substr(1);this.setProp(t,"webkit"+n,i),this.setProp(t,"Moz"+n,i),this.setProp(t,"ms"+n,i),this.setProp(t,"O"+n,i),this.setProp(t,e,i)},setProp:function(t,e,i){e in t.style&&(t.style[e]=i)},createDragColumn:function(t,e,i){var r=t.getFixedColumnCount(),s=t.getHScrollValue();r>i&&(s=0);var l=t.getRenderer(),a=l.getColumnEdges(),u=t.getHiDPI(o),c=t.getColumnWidth(i),h=t.div.clientHeight,d=n,g=t.div.getBoundingClientRect(),f=d.style;f.top=g.top+"px",f.left=g.left+"px",f.position="fixed",f.opacity=.85,f.boxShadow="0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)",f.borderTop="1px solid "+l.resolveProperty("lineColor"),f.backgroundColor=t.renderer.resolveProperty("backgroundColor"),d.setAttribute("width",Math.round(c*u)+"px"),d.setAttribute("height",Math.round(h*u)+"px"),f.width=c+"px",f.height=h+"px";var p=a[i-s];p*=u,o.scale(u,u),t.renderOverridesCache.dragger={columnIndex:i,ctx:o,startX:p,width:c,height:h,hdpiratio:u},this.setCrossBrowserProperty(d,"transform","translate("+e+"px, -5px)"),f.zIndex="5",f.cursor=this.getDraggingCursorName(),t.repaint()},dragColumn:function(t,e){var i=this,r=this.columnDragAutoScrollingRight||this.columnDragAutoScrollingLeft,s=t.getHiDPI(o),l=t.renderOverridesCache.dragger.columnIndex,u=t.renderOverridesCache.dragger.width,c=0,h=t.renderer.getFinalVisableColumnBoundary()-u;e=Math.min(e,h+15),e=Math.max(c-15,e);var d=c>e&&0!==l,g=e>h,f=n;this.setCrossBrowserProperty(f,"transition",(i.isWebkit?"-webkit-":"")+"transform 0ms ease, box-shadow "+a+"ms ease"),this.setCrossBrowserProperty(f,"transform","translate("+e+"px, -10px)"),requestAnimationFrame(function(){f.style.display="inline"});var p=t.renderer.getColumnFromPixelX(e+f.width/2/s);d&&(p=0),g&&(p=t.getColumnCount()-1);var v=l>p;if(v=v||p-l>=1,!v||g||r){if(c-10>e&&this.checkAutoScrollToLeft(t,e),e>c-10&&(this.columnDragAutoScrollingLeft=!1),g||e>h+10)return void this.checkAutoScrollToRight(t,e);h+10>e&&(this.columnDragAutoScrollingRight=!1)}else{var m=p>l;if(this.isFloatingNow)return;this.isFloatingNow=!0,this.createFloatColumn(t,p),this.floatColumnTo(t,m)}},checkAutoScrollToRight:function(t,e){this.columnDragAutoScrollingRight||(this.columnDragAutoScrollingRight=!0,this._checkAutoScrollToRight(t,e))},_checkAutoScrollToRight:function(t,e){if(this.columnDragAutoScrollingRight){var i=t.getHScrollValue();if(t.dragging&&!(i>t.sbHScrollConfig.rangeStop-2)){var n=t.renderOverridesCache.dragger.columnIndex;t.scrollBy(1,0);var o=n+1;console.log(o,n),t.swapColumns(o,n),t.renderOverridesCache.dragger.columnIndex=o,setTimeout(this._checkAutoScrollToRight.bind(this,t,e),250)}}},checkAutoScrollToLeft:function(t,e){this.columnDragAutoScrollingLeft||(this.columnDragAutoScrollingLeft=!0,this._checkAutoScrollToLeft(t,e))},_checkAutoScrollToLeft:function(t,e){if(this.columnDragAutoScrollingLeft){var i=t.getHScrollValue();if(t.dragging&&!(1>i)){var n=t.renderOverridesCache.dragger.columnIndex;t.swapColumns(n+i,n+i-1),t.scrollBy(-1,0),setTimeout(this._checkAutoScrollToLeft.bind(this,t,e),250)}}},endDragColumn:function(t){var e=t.getFixedColumnCount(),i=t.getHScrollValue(),o=t.renderOverridesCache.dragger.columnIndex;e>o&&(i=0);var r=t.getRenderer(),s=r.getColumnEdges(),l=this,u=s[o-i],c=n;l.setCrossBrowserProperty(c,"transition",(l.isWebkit?"-webkit-":"")+"transform "+a+"ms ease, box-shadow "+a+"ms ease"),l.setCrossBrowserProperty(c,"transform","translate("+u+"px, -1px)"),c.style.boxShadow="0px 0px 0px #888888",setTimeout(function(){t.renderOverridesCache.dragger=null,t.repaint(),requestAnimationFrame(function(){c.style.display="none",t.endDragColumnNotification()})},a+50)},isHeaderRow:function(t,e){var i=e.viewPoint,n=0===i.y;return n}});e.exports=u},{"./Feature.js":59}],55:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnPicker",{alias:"ColumnPicker",handleKeyUp:function(t,e){var i=e.detail["char"].toLowerCase(),n=t.resolveProperty("editorActivationKeys");n.indexOf(i)>-1&&t.toggleColumnPicker()}});e.exports=o},{"./Feature.js":59}],56:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnResizing",{alias:"ColumnResizing",dragIndex:-2,dragStart:-1,dragIndexStartingSize:-1,getMouseValue:function(t){return t.primitiveEvent.detail.mouse.x},getGridCellValue:function(t){return t.y},getScrollValue:function(t){return t.getHScrollValue()},getAreaSize:function(t,e){return t.getColumnWidth(e)},setAreaSize:function(t,e,i){t.setColumnWidth(e,i)},getPreviousAbsoluteSize:function(t,e){return t.getRenderedWidth(e)},overAreaDivider:function(t,e){return t.overColumnDivider(e)},isFirstFixedOtherArea:function(t,e){return this.isFirstFixedRow(t,e)},getCursorName:function(){return"col-resize"},handleMouseDrag:function(t,e){if(this.dragIndex>-2){var i=this.getMouseValue(e),n=this.getScrollValue(t);this.dragIndex-1&&this.isFirstFixedOtherArea(t,e)){var o=this.getScrollValue(t);n-2){this.cursor=null,this.dragIndex=-2,e.primitiveEvent.stopPropagation();var n=this;t.synchronizeScrollingBoundries(),setTimeout(function(){n.attachChain()},200)}else this.next&&this.next.handleMouseUp(t,e)},handleMouseMove:function(t,e){this.dragIndex>-2||(this.cursor=null,this.next&&this.next.handleMouseMove(t,e),this.checkForAreaResizeCursorChange(t,e))},checkForAreaResizeCursorChange:function(t,e){var i=this.isEnabled(t);i&&this.overAreaDivider(t,e)>-1&&this.isFirstFixedOtherArea(t,e)?this.cursor=this.getCursorName():this.cursor=null},getFixedAreaCount:function(t){var e=t.getFixedColumnCount()+(t.isShowRowNumbers()?1:0)+(t.hasHierarchyColumn()?1:0);return e},handleDoubleClick:function(t,e){var i=this.isEnabled(t),n=this.overAreaDivider(t,e)>-1,o=t.getHeaderRowCount(),r=e.gridCell;i&&n&&r.y<=o?t.autosizeColumn(r.x-1):this.next&&this.next.handleDoubleClick(t,e)},isEnabled:function(t){return!0}});e.exports=o},{"./Feature.js":59}],57:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnSelection",{alias:"ColumnSelection",currentDrag:null,lastDragCell:null,sbLastAuto:0,sbAutoStart:0,handleMouseUp:function(t,e){return this.dragging&&(this.dragging=!1),this.next?void this.next.handleMouseUp(t,e):void 0},handleMouseDown:function(t,e){if((!t.isColumnSelection()||e.mousePoint.y<5)&&this.next)return void this.next.handleMouseDown(t,e);var i=e.primitiveEvent.detail.isRightClick,n=e.gridCell,o=e.viewPoint,r=n.x,s=n.y,l=t.isShowHeaderRow()&&0===s&&-1!==r;if(i||!l)this.next&&this.next.handleMouseDown(t,e);else{var a=t.getFixedColumnCount();o.xi.origin.x+i.extent.x&&(n=1);var u=n,c=o;l&&(u=0),a&&(c=0),this.lastDragCell=e.plusXY(u,c),t.scrollBy(n,o),this.handleMouseDragCellSelection(t,e,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){t.stopEditing();var n=-1!==i.indexOf("SHIFT"),o=t.getMouseDown(),r=e.x,s=e.y;0>r||0>s||(n?(t.clearMostRecentColumnSelection(),t.selectColumn(r,o.x),t.setDragExtent(t.newPoint(r-o.x,0))):(t.toggleSelectColumn(r,i),t.setMouseDown(t.newPoint(r,s)),t.setDragExtent(t.newPoint(0,0))),t.repaint())},handleDOWNSHIFT:function(t){},handleUPSHIFT:function(t){},handleLEFTSHIFT:function(t){this.moveShiftSelect(t,-1)},handleRIGHTSHIFT:function(t){this.moveShiftSelect(t,1)},handleDOWN:function(t){},handleUP:function(t){},handleLEFT:function(t){this.moveSingleSelect(t,-1)},handleRIGHT:function(t){this.moveSingleSelect(t,1)},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now()},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e){var i=t.getColumnCount()-1,n=t.getVisibleColumns()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown(),r=t.getDragExtent(),s=r.x+e;s=Math.min(i-o.x,Math.max(-o.x,s)),t.clearMostRecentColumnSelection(),t.selectColumn(o.x,o.x+s),t.setDragExtent(t.newPoint(s,0)),t.insureModelColIsVisible(s+o.x,e)&&this.pingAutoScroll(),t.repaint()},moveSingleSelect:function(t,e){var i=t.getColumnCount()-1,n=t.getVisibleColumnsCount()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown().plus(t.getDragExtent()),r=o.x+e;r=Math.min(i,Math.max(0,r)),t.clearSelections(),t.selectColumn(r),t.setMouseDown(t.newPoint(r,0)),t.setDragExtent(t.newPoint(0,0)),t.insureModelColIsVisible(r,e)&&this.pingAutoScroll(),t.repaint()},isColumnDragging:function(t){var e=t.lookupFeature("ColumnMoving");if(!e)return!1;var i=e.dragging&&!this.dragging;return i}});e.exports=o},{"./Feature.js":59}],58:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnSorting",{alias:"ColumnSorting",handleDoubleClick:function(t,e){var i=e.gridCell;if(t.isShowHeaderRow()&&0===i.y&&-1!==i.x){var n=e.primitiveEvent.detail.keys;t.toggleSort(i.x,n)}else this.next&&this.next.handleDoubleClick(t,e)},handleMouseMove:function(t,e){var i=e.gridCell.y;this.isFixedRow(t,e)&&1>i?this.cursor="pointer":this.cursor=null,this.next&&this.next.handleMouseMove(t,e)}});e.exports=o},{"./Feature.js":59}],59:[function(t,e,i){"use strict";var n=t("extend-me").Base,o=n.extend("Feature",{next:null,detached:null,cursor:null,currentHoverCell:null,setNext:function(t){this.next?this.next.setNext(t):(this.next=t,this.detached=t)},detachChain:function(){this.next=null},attachChain:function(){this.next=this.detached},handleMouseMove:function(t,e){this.next&&this.next.handleMouseMove(t,e)},handleMouseExit:function(t,e){this.next&&this.next.handleMouseExit(t,e)},handleMouseEnter:function(t,e){this.next&&this.next.handleMouseEnter(t,e)},handleMouseDown:function(t,e){this.next&&this.next.handleMouseDown(t,e)},handleMouseUp:function(t,e){this.next&&this.next.handleMouseUp(t,e)},handleKeyDown:function(t,e){this.next&&this.next.handleKeyDown(t,e)},handleKeyUp:function(t,e){this.next&&this.next.handleKeyUp(t,e)},handleWheelMoved:function(t,e){this.next&&this.next.handleWheelMoved(t,e)},handleDoubleClick:function(t,e){this.next&&this.next.handleDoubleClick(t,e)},handleHoldPulse:function(t,e){this.next&&this.next.handleHoldPulse(t,e)},handleTap:function(t,e){this.next&&this.next.handleTap(t,e)},handleMouseDrag:function(t,e){this.next&&this.next.handleMouseDrag(t,e)},handleContextMenu:function(t,e){this.next&&this.next.handleContextMenu(t,e)},moveSingleSelect:function(t,e,i){this.next&&this.next.moveSingleSelect(t,e,i)},isFixedRow:function(t,e){var i=e.viewPoint,n=i.yr;if(t.isRowSelection()&&!i&&l){var a=t.getFixedRowCount();o.yi.origin.y+i.extent.y&&(o=1);var u=n,c=o;l&&(u=0),a&&(c=0),this.lastDragCell=e.plusXY(u,c),t.scrollBy(n,o),this.handleMouseDragCellSelection(t,e,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){t.stopEditing();var n=-1!==i.indexOf("SHIFT"),o=t.getMouseDown(),r=e.x,s=e.y;0>r||0>s||(n?(t.clearMostRecentRowSelection(),t.selectRow(s,o.y),t.setDragExtent(t.newPoint(0,s-o.y))):(t.toggleSelectRow(s,i),t.setMouseDown(t.newPoint(r,s)),t.setDragExtent(t.newPoint(0,0))),t.repaint())},handleDOWNSHIFT:function(t){this.moveShiftSelect(t,1)},handleUPSHIFT:function(t){this.moveShiftSelect(t,-1)},handleLEFTSHIFT:function(t){},handleRIGHTSHIFT:function(t){},handleDOWN:function(t){this.moveSingleSelect(t,1)},handleUP:function(t){this.moveSingleSelect(t,-1)},handleLEFT:function(t){},handleRIGHT:function(t){var e=t.getMouseDown().plus(t.getDragExtent()),i=t.getColumnCount()-1,n=t.getHeaderColumnCount()+t.getHScrollValue(),o=e.y;n=Math.min(i,n),t.clearSelections(),t.select(n,o,0,0),t.setMouseDown(t.newPoint(n,o)),t.setDragExtent(t.newPoint(0,0)),t.repaint()},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now()},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e){var i=t.getRowCount()-1,n=t.getVisibleRows()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown(),r=t.getDragExtent(),s=r.y+e;s=Math.min(i-o.y,Math.max(-o.y,s)),t.clearMostRecentRowSelection(),t.selectRow(o.y,o.y+s),t.setDragExtent(t.newPoint(0,s)),t.insureModelRowIsVisible(s+o.y,e)&&this.pingAutoScroll(),t.fireSyntheticRowSelectionChangedEvent(),t.repaint()},moveSingleSelect:function(t,e){var i=t.getRowCount()-1,n=t.getVisibleRowsCount()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown().plus(t.getDragExtent()),r=o.y+e;r=Math.min(i,Math.max(0,r)),t.clearSelections(),t.selectRow(r),t.setMouseDown(t.newPoint(0,r)),t.setDragExtent(t.newPoint(0,0)),t.insureModelRowIsVisible(r,e)&&this.pingAutoScroll(),t.fireSyntheticRowSelectionChangedEvent(),t.repaint()},isSingleRowSelection:function(){return!0}});e.exports=o},{"./Feature.js":59}],65:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ThumbwheelScrolling",{alias:"ThumbwheelScrolling",handleWheelMoved:function(t,e){if(t.resolveProperty("scrollingEnabled")){var i=e.primitiveEvent,n=i.wheelDeltaY||-i.deltaY,o=i.wheelDeltaX||-i.deltaX;n>0?t.scrollBy(0,-1):-0>n?t.scrollBy(0,1):o>0?t.scrollBy(-1,0):-0>o&&t.scrollBy(1,0)}}});e.exports=o},{"./Feature.js":59}],66:[function(t,e,i){"use strict";e.exports={Feature:t("./Feature"),CellClick:t("./CellClick"),CellEditing:t("./CellEditing"),CellSelection:t("./CellSelection"),ColumnAutosizing:t("./ColumnAutosizing"),ColumnMoving:t("./ColumnMoving"),ColumnResizing:t("./ColumnResizing"),ColumnSelection:t("./ColumnSelection"),ColumnSorting:t("./ColumnSorting"),Filters:t("./Filters"),KeyPaging:t("./KeyPaging"),OnHover:t("./OnHover"),ColumnPicker:t("./ColumnPicker"),RowResizing:t("./RowResizing"),RowSelection:t("./RowSelection"),ThumbwheelScrolling:t("./ThumbwheelScrolling")}},{"./CellClick":50,"./CellEditing":51,"./CellSelection":52,"./ColumnAutosizing":53,"./ColumnMoving":54,"./ColumnPicker":55,"./ColumnResizing":56,"./ColumnSelection":57,"./ColumnSorting":58,"./Feature":59,"./Filters":60,"./KeyPaging":61,"./OnHover":62,"./RowResizing":63,"./RowSelection":64,"./ThumbwheelScrolling":65}],67:[function(t,e,i){"use strict";e.exports=function(){function t(t){this.label=t,this.data=[""],this.rowIndexes=[],this.hasChildren=!1,this.depth=0,this.height=1,this.expanded=!1}var e=" ";return t.prototype.isNullObject=!1,t.prototype.getValue=function(t){return this.data[t]},t.prototype.prune=function(t){this.depth=t,this.data[0]=this.computeDepthString()},t.prototype.computeDepthString=function(){var t=e.substring(0,2+3*this.depth)+this.label;return t},t.prototype.computeHeight=function(){return 1},t.prototype.getAllRowIndexes=function(){return this.rowIndexes},t.prototype.computeAggregates=function(t){this.applyAggregates(t)},t.prototype.applyAggregates=function(t){var e=t.hasGroups()?1:0,i=this.getAllRowIndexes();if(0!==i.length){var n=t.aggregates,o=this.data;o.length=n.length+e;var r=t.sorterInstance;r.indexes=i;for(var s=0;s0},t.prototype.hasAggregates=function(){return this.aggregates.length>0},t.prototype.apply=function(){this.buildGroupTree()},t.prototype.clearGroups=function(){this.groupBys.length=0},t.prototype.clearAggregations=function(){this.aggregates.length=0,this.headers.length=0},t.prototype.buildGroupTree=function(){var t,e,i,l,a,u=function(t,e){return l=new r(t),e.set(t,l),l},c=function(t,e){return l=new s(t),e.set(t,l),l},h=this.groupBys,d=this.dataSource,g=d.getRowCount();if(this.presortGroups)for(t=0;te;e++){for(t=0;ti;i++)e[i]=i;this.indexes=e},t.prototype.setData=function(t){this.dataSource.setData(t)},t}()},{}],73:[function(t,e,i){"use strict";var n=t("./DataSourceDecorator");e.exports=function(){function t(t){n.call(this,t,!1),this.filters=[]}return t.prototype=Object.create(n.prototype),t.prototype.add=function(t,e){e.columnIndex=t,this.filters.push(e)},t.prototype.set=function(t,e){e.columnIndex=t,this.filters.push(e)},t.prototype.clearAll=function(){this.filters.length=0,this.indexes.length=0},t.prototype.applyAll=function(){if(0===this.filters.length)return void(this.indexes.length=0);var t=this.indexes;t.length=0;for(var e=this.dataSource.getRowCount(),i=0;e>i;i++)this.applyFiltersTo(i)&&t.push(i)},t.prototype.applyFiltersTo=function(t){for(var e=this.filters,i=!0,n=0;no;o++)this.applyFilterTo(o)&&i.push(o)},t.prototype.applyFilterTo=function(t){for(var e=!1,i=this.filter,n=this.visibleColumnMap.length,o=this.dataSource.getRow(t),r=0;n>r;r++){var s=this.visibleColumnMap[r];if(e=e||i(this.dataSource.getValue(s,t),o,t))return!0}return!1},t}()},{"./DataSourceDecorator":72}],75:[function(t,e,i){"use strict";var n=t("./Utils.js"),o=t("./DataSourceDecorator"),r=function(t){var e="f"===(typeof t)[0],i=e?t():t;return i};e.exports=function(){function t(t){o.call(this,t),this.descendingSort=!1}return t.prototype=Object.create(o.prototype),t.prototype.sortOn=function(t,e){if(0===e)return void(this.indexes.length=0);this.initializeIndexVector();var i=this;n.stableSort(this.indexes,function(e){var n=i.dataSource.getValue(t,e);return n=r(n)},e)},t}()},{"./DataSourceDecorator":72,"./Utils.js":79}],76:[function(t,e,i){"use strict";var n=t("./DataSourceDecorator"),o=t("./DataSourceSorter");e.exports=function(){function t(t){n.call(this,t),this.sorts=[],this.last=this.dataSource}return t.prototype=Object.create(n.prototype),t.prototype.getRow=function(t){return this.last.getRow(t)},t.prototype.sortOn=function(t,e){this.sorts.push([t,e])},t.prototype.applySorts=function(){for(var t=this.sorts,e=this.dataSource,i=0;io;o++)i+=e.getValue(t,o);return i}},min:function(t){return function(e){for(var i=1/0,n=e.getRowCount(),o=0;n>o;o++)i=Math.min(i,e.getValue(t,o));return i}},max:function(t){return function(e){for(var i=-(1/0),n=e.getRowCount(),o=0;n>o;o++)i=Math.max(i,e.getValue(t,o));return i}},avg:function(t){return function(e){for(var i=0,n=e.getRowCount(),o=0;n>o;o++)i+=e.getValue(t,o);return i/n}},first:function(t){return function(e){return e.getValue(t,0)}},last:function(t){return function(e){var i=e.getRowCount();return e.getValue(t,i-1)}},stddev:function(t){return function(e){var i,n=0,o=e.getRowCount();for(i=0;o>i;i++)n+=e.getValue(t,i);var r=n/o,s=0;for(i=0;o>i;i++){var l=e.getValue(t,i)-r;s+=l*l}var a=Math.sqrt(s/o);return a}}}}()},{}],81:[function(t,e,i){"use strict";var n=t("./JSDataSource"),o=t("./DataSourceSorter"),r=t("./DataSourceSorterComposite"),s=t("./DataSourceFilter"),l=t("./DataSourceGlobalFilter"),a=t("./DataSourceAggregator"),u=t("./aggregations");e.exports=function(){return{JSDataSource:n,DataSourceSorter:o,DataSourceSorterComposite:r,DataSourceFilter:s,DataSourceGlobalFilter:l,DataSourceAggregator:a,aggregations:u}}()},{"./DataSourceAggregator":71,"./DataSourceFilter":73,"./DataSourceGlobalFilter":74,"./DataSourceSorter":75,"./DataSourceSorterComposite":76,"./JSDataSource":77,"./aggregations":80}],82:[function(t,e,i){"use strict";var n=function(t,e){return function(i,n){var o=i[0],r=n[0];if(o===r)o=e?n[1]:i[1],r=e?i[1]:n[1];else{if(null===r)return-1;if(null===o)return 1}return t(o,r)}},o=function(t,e){return t-e},r=function(t,e){return e-t},s=function(t,e){return e>t?-1:1},l=function(t,e){return t>e?-1:1},a=function(t){return"number"===t?n(o,!1):n(s,!1)},u=function(t){return"number"===t?n(r,!0):n(l,!0)};e.exports=function(){function t(t,e,i){var n,o;if(0!==t.length&&(void 0===i&&(i=1),0!==i)){var r=typeof e(0);n=1===i?a(r):u(r);var s=new Array(t.length);for(o=0;o div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }","div.dragon-list > ul { top: 46px; }","div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {"," content: '\\2b24';"," color: #b6b6b6;"," font-size: 30px;"," margin: 8px 14px 8px 8px; }","li.dragon-pop { opacity:.8; }"]};e.exports=n},{"css-injector":3}]},{},[49]); \ No newline at end of file +!function t(e,i,n){function o(s,l){if(!i[s]){if(!e[s]){var a="function"==typeof require&&require;if(!l&&a)return a(s,!0);if(r)return r(s,!0);throw new Error("Cannot find module '"+s+"'")}var u=i[s]={exports:{}};e[s][0].call(u.exports,function(t){var i=e[s][1][t];return o(i?i:t)},u,u.exports,t,e,i,n)}return i[s].exports}for(var r="function"==typeof require&&require,s=0;s div, div.dragon-list > ul > li, li.dragon-pop { line-height: 46px; }","div.dragon-list > ul { top: 46px; }","div.dragon-list > ul > li:not(:last-child)::before, li.dragon-pop::before {"," content: '\\2b24';"," color: #b6b6b6;"," font-size: 30px;"," margin: 8px 14px 8px 8px; }","li.dragon-pop { opacity:.8; }"]};e.exports=n},{"css-injector":4}],2:[function(t,e,i){e.exports={calendar:{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAAc0lEQVR4nIXQwQkCMRSE4U9ZLMCT9Xjaq2AfNhfYU5oQLMAOtoN48EWei5iBIRPe/yYQ3qrhf1lFG7iKcEaJxSfukUvMWgdHavt0uWHtg2QwxXnAnJZ2uOLyVZtybzzhgWNmfoFl0/YB87NbzR1cjP9xeQHSDC6mcL1xFQAAAABJRU5ErkJggg=="},checked:{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAYJJREFUOE+NkstLglEQxf0fahG0iFrUxm2ElFDYLohCqCDaCAkWPaxIRbFFEJEaGEKLDCoMETRFUAMLyaIHBUG6sSKIMtKFqEhLT818ZUgmDhzu3DPn9z0uV1RrmUwmyGQyqNVqfFvViwBxu5RFPZuLSyGMKhz/qlEsRV19K8xm6y+w7bpBPFnAferjj3bdQX6DpHcAUwavAHUN2RGIZxBJZHH2mC/TUeydwwTZvBegLENNgw7sX6Wh1FswNmPEmjPCDyGRRwCtW9E3tMgdAtQw7GZjYcNX+gza2wJ3ZXsSZUuQ0vWCOV8SHfJJ/uluhbHUj1v8PKNMszIoQNRMHCShD6Wh8zyhrbOPwz8w+STKlCCJ7oRNUzQH63kBs5thBghePXxlj2aUoSxDPcuXPNiLAc5EEZ6HIkbmV2DYiXBPHs0o079+K0DTVj/s11mE00A0L+g4VcDp10qKZMAzytBhMaTRaPmYg885DlcSzSij0eoEiIouoUqlqqqaL2rlEok+Ad4vlfzPoVDsAAAAAElFTkSuQmCC"},"down-rectangle":{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAAkAAAAECAYAAABcDxXOAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAadEVYdFNvZnR3YXJlAFBhaW50Lk5FVCB2My41LjExR/NCNwAAABpJREFUGFdjgIL/eDAKIKgABggqgAE0BQwMAPTlD/Fpi0JfAAAAAElFTkSuQmCC"},"filter-off":{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAAChSURBVChTzZHBCoUgFET9TqEiskgyWoutQvRLRIr+cR7XQAjiJW/1BgZmMUevXsY5xy9OoDEGMcYiUzeB67qibVuwQjVNA6311V+WBeM4vsLDMEApde/1fY9pmtI453neHEKAlBJd1z0fXtc16PbjODK07zvmeUZVVd8nooc75zJIOX3Gm6i0bVsGKf8xKIRIuyJTLgJJ3nvQzsjW2geIsQ/pr9hMVrSncAAAAABJRU5ErkJggg=="},"filter-on":{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAMCAYAAABSgIzaAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjAuNWWFMmUAAACoSURBVChTY3BqfP2fHAzWmDbj7f8p294RhVOBasEa02e+/e/VBmQQCTxaX/9PnvYGoj5ywpv/Qd2ENft3vv4f1gfVBAP+nW/+h/a+ATtn1q73KHjytvdgg3070DTBgHvL6/8g22fsQGiaDmSHA21xaybgIpDHixa8hWssnA8NDEIApCh3LkIjiD2INYJCL2X6W3B8gdhEaQQBUOCA4gyE8+e9xaKJgQEA/74BNE3cElkAAAAASUVORK5CYII="},unchecked:{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA0AAAAPCAYAAAA/I0V3AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwgAADsIBFShKgAAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAARBJREFUOE+9krtug1AQRPldSio7FQ1tZImOkoKOBomGT0EURC5ino54yTw90WywQhTkIkVWGoF2zuxdrlD+t0zThKZpT0Vmxb8CQRCg6zr0fb8rer7vfwcPxxdcrx+YpgnzPGNZlh9ibxxHlGUJshLSdV0at9tNpg7DIBrX5+OkPM9BVkKGYSBJEtR1jbZrBdiqbVtUVYU0TUFWQq+nE+I4xvvlImGaW7FHjwxZCVmWhbfzGVmWoSgKWXUr9uiRISshx3FkEldomubXauzRI0NWQp7nyUR+NG/rfr/jUXxnjx5vmKyEbNuWox9Xvid6ZMhK6HA4wnVdhGGIKIp2RY8MWQmx+JuoqvpUZFb8L6UonyYL3uOtrFH+AAAAAElFTkSuQmCC"},"up-down-spin":{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGJJREFUOE+lkwEKACEIBH2Zb/PnHsoGeaVJDUjGOgRRpKpkiIj+y4MME3eDR7kaKOVNsJyMNjIHzGy9YnW6J7qIcrriQimeCqORNABd0fpRTkt8uVUj7EsxC6vs/q3e/Q6iD2bwnByjPXHNAAAAAElFTkSuQmCC"},"up-down":{type:"image/png",data:"iVBORw0KGgoAAAANSUhEUgAAAA4AAAAPCAYAAADUFP50AAAABGdBTUEAALGPC/xhBQAAAAlwSFlzAAAOwQAADsEBuJFr7QAAABh0RVh0U29mdHdhcmUAcGFpbnQubmV0IDQuMC41ZYUyZQAAAGFJREFUOE+lkkEKQCEIRD2ZJ3Ph3iN4WD9GflpYhj0YYowpGgJmbikd3gjMDFokwbuT1iAiurG5nomgqo5QaPo9ERQRI6Jf7sfGjudy2je23+i0Wl2oQ85TOdlfrJQOazF8br+rqTXQKn0AAAAASUVORK5CYII="}}},{}],3:[function(t,e,i){"use strict";var n=t("object-iterators"),o=t("./images");n(o).each(function(t,e){var i=new Image;i. src="data:"+t.type+";base64,"+t.data,o[e]=i}),o.checkbox=function(t){return o[t?"checked":"unchecked"]},o.filter=function(t){return o[t?"filter-on":"filter-off"]},e.exports=o},{"./images":2,"object-iterators":21}],4:[function(t,e,i){"use strict";function n(t,e,i){if("string"==typeof i){if(i=document.querySelector(i),!i)throw"Cannot find reference element for CSS injection."}else if(i&&!(i instanceof Element))throw"Given value not a reference element.";var o=i&&i.parentNode||document.head||document.getElementsByTagName("head")[0];if(!e||(e=n.idPrefix+e,!o.querySelector("#"+e))){var r=document.createElement("style");return r.type="text/css",e&&(r.id=e),t instanceof Array&&(t=t.join("\n")),t="\n"+t+"\n",r.styleSheet?r.styleSheet.cssText=t:r.appendChild(document.createTextNode(t)),void 0===i&&(i=o.firstChild),o.insertBefore(r,i),r}}n.idPrefix="injected-stylesheet-",e.exports=n},{}],5:[function(t,e,i){"use strict";function n(t,e){function i(){e.preInitialize&&e.preInitialize.apply(this,arguments),o.apply(this,arguments),e.postInitialize&&e.postInitialize.apply(this,arguments)}function r(t,i){s[i]=e[t]}switch(arguments.length){case 0:e={};break;case 1:if(e=t,"object"!=typeof e)throw"Single parameter overload must be object.";t=void 0;break;case 2:if("string"!=typeof t||"object"!=typeof e)throw"Two parameter overload must be string, object.";break;default:throw"Too many parameters"}i.extend=n;var s=i.prototype=Object.create(this.prototype);s.constructor=i,t&&n.debug&&(s.$$CLASS_NAME=t);for(var l in e)if(e.hasOwnProperty(l)){var a=e[l];switch(l){case"initializeOwn":break;case"aliases":for(var u in a)a.hasOwnProperty(u)&&r(a[u],u);break;default:"string"==typeof a&&"#"===a[0]?r(a,l.substr(1)):s[l]=a}}return i}function o(){function t(n){var o=Object.getPrototypeOf(n);o.constructor!==Object&&(t(o),o.hasOwnProperty("initialize")&&o.initialize.apply(e,i))}var e=this,i=arguments;t(e)}n.Base=function(){},n.Base.extend=n,e.exports=n},{}],6:[function(t,e,i){"use strict";function n(t){if("object"!=typeof t){var e="Expected `state` parameter to be an object.";throw"string"==typeof t&&(e+=" See `JSON.parse()`."),c.Error(e)}}function o(t){var e=t.target,i=this[e.className]||this[e.parentNode.className];i&&(this.detachChooser&&this.detachChooser(),i.call(this,t),t.stopPropagation()),this.eventHandler&&this.eventHandler(t)}function r(t){if(this instanceof v&&!this.children.length)throw new c.Error("Empty subexpression (no filters).");this.children.forEach(function(e){e instanceof h?e.validate(t):e.children.length&&r.call(e,t)})}function s(t){var e=this,i=t.target.getBoundingClientRect();if(!i.width)return void setTimeout(function(){s.call(e,t)},50);var n=Object.keys(v.prototype.editors),o=this.chooser=document.createElement("select");o.className="filter-tree-chooser",o.size=n.length,n.forEach(function(t){var i=e.editors[t].prototype.name||t;i=i.replace("?","≟"),o.add(new Option(i,t))}),o.onmouseover=function(t){t.target.selected=!0},o.style.left=i.left+19+"px",o.style.top=i.bottom+"px",this.detachChooser=l.bind(this),window.addEventListener("click",this.detachChooser),o.onclick=function(){e.children.push(new e.editors[o.value]({parent:e}))},o.onmouseout=function(){o.selectedIndex=-1},this.el.appendChild(o),this.chooserTarget=t.target,this.chooserTarget.classList.add("as-menu-header")}function l(){var t=this.chooser;t&&(this.el.removeChild(t),this.chooserTarget.classList.remove("as-menu-header"),t.onclick=t.onmouseout=null,window.removeEventListener("click",this.detachChooser),delete this.detachChooser,delete this.chooser)}var a=t("unstrungify"),u=t("./js/css"),c=t("./js/FilterNode"),h=t("./js/FilterLeaf"),d=t("./js/template"),g=t("./js/tree-operators"),f=0,p=/^filter-tree: /,v=c.extend("FilterTree",{preInitialize:function(t){u("filter-tree-base",t&&t.cssStylesheetReferenceElement),t.editors&&(this.editors=t.editors)},destroy:function(){l.call(this)},editors:{Default:h},addEditor:function(t,e){e?this.editors[t]=h.extend(e):delete this.editors[t]},newView:function(){this.el=d("tree",++f),this.el.addEventListener("click",o.bind(this))},getState:a,getJSON:function(){var t=JSON.stringify(this,null,this.JSONspace);return t?t:""},setJSON:function(t){this.setState(JSON.parse(t))},load:function(t){if(t){if(n(t),!(g[t.operator]||void 0===t.operator&&1===t.children.length))throw c.Error("Expected `operator` property to be one of: "+Object.keys(g));if(this.operator=t.operator,!(t.children instanceof Array&&t.children.length))throw c.Error("Expected `children` property to be a non-empty array.");this.children=[];var e=this;t.children.forEach(function(t){var i;if("object"!=typeof t)throw e.Error("Expected child to be an object containing either `children`, `editor`, or neither.");i=t.children?v:e.editors[t.editor||"Default"],e.children.push(new i({state:t,parent:e}))})}else{var i=Object.keys(this.editors),o=1===i.length;this.children=o?[new this.editors[i[0]]({parent:this})]:[],this.operator="op-and"}},render:function(){var t=this.el.querySelector("input[value="+this.operator+"]");if(t.checked=!0,this["filter-tree-op-choice"]({target:t}),!this.children.length&&Object.keys(this.editors).length>1){var e=this.el.querySelector(".filter-tree-add-filter");this["filter-tree-add-filter"]({target:e})}c.prototype.render.call(this)},"filter-tree-op-choice":function(t){var e=t.target;this.operator=e.value;var i=this.el.querySelectorAll("label>input.filter-tree-op-choice[name="+e.name+"]");Array.prototype.slice.call(i).forEach(function(t){t.parentElement.style.textDecoration=t.checked?"none":"line-through"});for(var n in g)this.el.classList.remove(n);this.el.classList.add(this.operator)},"filter-tree-add-filter":function(t){var e=Object.keys(this.editors);1===e.length?this.children.push(new this.editors[e[0]]({parent:this})):s.call(this,t)},"filter-tree-add":function(){this.children.push(new v({parent:this}))},"filter-tree-remove":function(t){var e=t.target,i=e.parentElement,n=this.children,o=e.nextElementSibling;n.forEach(function(t,e){t.el===o&&(delete n[e],i.remove())})},validate:function(t){t=t||{};var e,i=void 0===t.focus||t.focus,n=void 0===t.alert||t.alert,o=t.rethrow===!0;try{r.call(this,i)}catch(s){if(e=s.message,o||!p.test(e))throw s;n&&(e=e.replace(p,""),window.alert(e))}return e},test:function m(t){var e=g[this.operator],i=e.seed,n=!0;return this.children.find(function(o){return o?(n=!1,o instanceof h?i=e.reduce(i,o.test(t)):o.children.length&&(i=e.reduce(i,m.call(o,t))),i===e.abort):!1}),n||(e.negate?!i:i)},toJSON:function C(){var t={operator:this.operator,children:[]};this.children.forEach(function(e){if(e)if(e instanceof h)t.children.push(e);else if(e.children.length){var i=C.call(e);i&&t.children.push(i)}});var e=c.prototype.toJSON.call(this);return Object.keys(e).forEach(function(i){t[i]=e[i]}),t.children.length?t:void 0},getSqlWhereClause:function w(){var t=g[this.operator].SQL,e="";return this.children.forEach(function(i,n){var o=n?" "+t.op+" ":"";i&&(i instanceof h?e+=o+i.getSqlWhereClause():i.children.length&&(e+=o+w.call(i)))}),e||(e="NULL IS NULL"),t.beg+e+t.end}});e.exports=v},{"./js/FilterLeaf":7,"./js/FilterNode":8,"./js/css":9,"./js/template":11,"./js/tree-operators":12,unstrungify:26}],7:[function(t,e,i){"use strict";function n(t,e){var i,o;return o=t.find(function(t){return(t.options||t)instanceof Array?i=n(t.options||t,e):t.name===e}),i||o}function o(t){var e=t.target;if(e.classList.remove("filter-tree-error"),u.setWarningClass(e),!e.multiple&&e.value)for(;(e=e.nextElementSibling)&&(!("name"in e)||""!==e.value.trim()););e&&""===e.value.trim()&&(e.value="",u.clickIn(e)),this.eventHandler&&this.eventHandler(t)}function r(t){setTimeout(function(){t.classList.add("filter-tree-error"),u.clickIn(t)},0)}function s(t){var e,i;switch(t.type){case"checkbox":case"radio":for(t=document.querySelectorAll("input[name='"+t.name+"']:enabled:checked"),e=[],i=0;it?-1:t>e?1:0}var u=t("./FilterNode"),c=t("./template"),h=t("./leaf-operators"),d={to:Number,not:isNaN},g={to:function(t){return new Date(t)},not:isNaN},f=u.extend("FilterLeaf",{name:"column : value",preInitialize:function(){this.onChange=o.bind(this)},operators:h,operatorsOptions:h.options,destroy:function(){if(this.controls)for(var t in this.controls)this.controls[t].removeEventListener("change",this.onChange)},newView:function(){var t=this.parent.nodeFields||this.fields;if(!t)throw u.Error("Terminal node requires a fields list.");var e=this.el=document.createElement("span");e.className="filter-tree-default",this.controls={column:this.makeElement(e,t,"column",!0),operator:this.makeElement(e,this.operatorsOptions,"operator"),value:this.makeElement(e)},e.appendChild(document.createElement("br"))},makeElement:function(t,e,i,n){var o,r,s,a=e?"select":"input";return e&&1===e.length?(r=e[0],o=document.createElement("input"),o.type="hidden",o.value=r.name||r.alias||r,s=document.createElement("span"),s.innerHTML=r.alias||r.name||r,s.appendChild(o),t.appendChild(s)):(o=l(a,e,i,n),"text"===o.type&&this.eventHandler&&this.el.addEventListener("keyup",this.eventHandler),this.el.addEventListener("change",this.onChange),u.setWarningClass(o),t.appendChild(o)),o},load:function(t){if(t){var e,i,n,o,r,s=[];for(var l in t)if("fields"!==l&&"editor"!==l)switch(e=t[l],i=this.controls[l],i.type){case"checkbox":case"radio":for(i=document.querySelectorAll("input[name='"+i.name+"']"),n=0;n=0;break;case"select-multiple":for(i=i.options,n=0,o=!1;n=0,i[n].selected=r;u.setWarningClass(i,o);break;default:i.value=e,u.setWarningClass(i)||i.value===e||s.push({key:l,value:e})}if(s.length){var a=s.length>1,h=c(a?"notes":"note"),d=h.lastElementChild;s.forEach(function(t){var e=a?document.createElement("li"):d;for(t=c("optionMissing",t.key,t.value);t.length;)e.appendChild(t[0]);a&&d.appendChild(e)}),i.parentNode.replaceChild(h,i.parentNode.lastElementChild)}}},converters:{number:d,"int":d,"float":d,date:g},validate:function(t){var e,i,o;for(e in this.controls){var l=this.controls[e],a=s(l).trim();if(""===a)throw t&&r(l),new u.Error("Blank "+e+" control.\nComplete the filter or delete it.");this[e]=a}if(this.op=this.operators[this.operator],this.converter=void 0,this.op.type)this.converter=this.converters[this.op.type];else for(e in this.controls)/^column/.test(e)&&(i=this.parent.nodeFields||this.fields,o=n(i,this[e]),o&&o.type&&(this.converter=this.converters[o.type]))},p:function(t){return t[this.column]},q:function(){return this.value},test:function(t){var e,i,n,o,r;return void 0===(e=this.p(t))||void 0===(i=this.q(t))?!1:!(r=this.converter)||r.not(n=r.to(e))||r.not(o=r.to(i))?this.op.test(e,i):this.op.test(n,o)},toJSON:function(t){var e={};this.editor&&(e.editor=this.editor);for(var i in this.controls)e[i]=this[i];return this.parent.nodeFields||this.fields===this.parent.fields||(e.fields=this.fields),e},getSqlWhereClause:function(){return this.SQL_QUOTED_IDENTIFIER+this.column+this.SQL_QUOTED_IDENTIFIER+" "+("function"==typeof this.op.sql?this.op.sql(this.value):(this.op.sql||this.operator)+h.sq(this.value))}});e.exports=f},{"./FilterNode":8,"./leaf-operators":10,"./template":11}],8:[function(t,e,i){"use strict";function n(t,e,i,n){return e&&e[t]||i&&i[t]||n&&n[t]}var o=t("extend-me"),r=o.Base,s=t("./template");o.debug=!0;var l="OL",a="LI",u=r.extend({initialize:function(t){var e=t&&t.parent,i=t&&(t.state||t.json&&JSON.parse(t.json));this.parent=e,this.nodeFields=n("nodeFields",t,i),this.fields=n("fields",t,i,e),this.editor=n("editor",t,i,e),this.eventHandler=n("eventHandler",t,i,e),this.setState(i)},render:function(){if(this.parent){var t=document.createElement(a);t.appendChild(s("removeButton")),t.appendChild(this.el),this.parent.el.querySelector(l).appendChild(t)}},setState:function(t){var e=this.el;this.newView(),this.load(t),this.render(),e&&!this.parent&&e.parentNode.replaceChild(this.el,e)},toJSON:function(){var t={};if(this.toJsonOptions){var e=this,i=[];this.toJsonOptions.fields&&(i.push("fields"),i.push("nodeFields")),this.toJsonOptions.editor&&i.push("editor"),i.forEach(function(i){(!e.parent||e[i]&&e[i]!==e.parent[i])&&(t[i]=e[i])})}return t},SQL_QUOTED_IDENTIFIER:'"'});u.setWarningClass=function(t,e){return arguments.length<2&&(e=t.value),t.classList[e?"remove":"add"]("filter-tree-warning"),e},u.Error=function(t){return new Error("filter-tree: "+t)},u.clickIn=function(t){t&&("SELECT"===t.tagName?setTimeout(function(){t.dispatchEvent(new MouseEvent("mousedown"))},0):t.focus())},e.exports=u},{"./template":11,"extend-me":5}],9:[function(t,e,i){"use strict";var n,o=t("css-injector");n=".filter-tree{font-family:sans-serif;font-size:10pt;line-height:1.5em}.filter-tree label{font-weight:400}.filter-tree input[type=checkbox],.filter-tree input[type=radio]{left:3px;margin-right:3px}.filter-tree ol{margin-top:0}.filter-tree-add,.filter-tree-add-filter,.filter-tree-remove{cursor:pointer}.filter-tree-add,.filter-tree-add-filter{font-style:italic;color:#444;font-size:90%}.filter-tree-add-filter{margin:3px 0 3px 3em;width:120px;display:inline-block}.filter-tree-add-filter:hover,.filter-tree-add:hover{text-decoration:underline}.filter-tree-add-filter.as-menu-header,.filter-tree-add.as-menu-header{background-color:#fff;font-weight:700;font-style:normal}.filter-tree-add-filter.as-menu-header:hover{text-decoration:inherit}.filter-tree-add-filter>div,.filter-tree-add>div,.filter-tree-remove{display:inline-block;width:15px;height:15px;border-radius:8px;background-color:#8c8;font-size:11.5px;font-weight:700;color:#fff;text-align:center;line-height:normal;font-style:normal;font-family:sans-serif;text-shadow:0 0 1.5px grey;margin-right:4px}.filter-tree-add-filter>div:before,.filter-tree-add>div:before{content:'\\ff0b'}.filter-tree-remove{background-color:#e88;border:0}.filter-tree-remove:before{content:'\\2212'}.filter-tree li::after{font-size:70%;font-style:italic;font-weight:700;color:#080}.filter-tree>ol>li:last-child::after{display:none}.op-or>ol>li::after{margin-left:2.5em;content:'— OR —'}.op-and>ol>li::after{margin-left:2.5em;content:'— AND —'}.op-nor>ol>li::after{margin-left:2.5em;content:'— NOR —'}.filter-tree-default>:enabled{margin:0 .4em;background-color:#ddd;border:0}.filter-tree-default>input[type=text]{width:8em;padding:0 5px}.filter-tree-default>select{border:0}.filter-tree-default>.filter-tree-warning{background-color:#ffc}.filter-tree-default>.filter-tree-error{background-color:#Fcc}.filter-tree .footnotes{font-size:6pt;margin:2px 0 0;line-height:normal;white-space:normal;color:#999}.filter-tree .footnotes>ol{margin:0;padding-left:2em}.filter-tree .footnotes>ol>li{margin:2px 0}.filter-tree .footnotes .field-name,.filter-tree .footnotes .field-value{font-weight:700;color:#777}.filter-tree .footnotes .field-value:after,.filter-tree .footnotes .field-value:before{content:'\"'}.filter-tree .footnotes .field-value{font-family:monospace}.filter-tree-chooser{position:absolute;font-size:9pt;outline:0;box-shadow:5px 5px 10px grey}",e.exports=o.bind(this,n)},{"css-injector":4}],10:[function(t,e,i){"use strict";function n(t,e){return e.trim().replace(/\s*,\s*/g,",").split(",").indexOf(t.toString())}function o(t,e){return t.toString().toLowerCase().indexOf(e.toString().toLowerCase())}function r(t,e){return t.toString().toLowerCase().substr(0,e)}function s(t,e){return t.toString().toLowerCase().substr(-e,e)}function l(t,e,i,n){var o=n.replace(/([\[_%\]])/g,"[$1]");return i+h(t+o+e)}function a(t){return"IN ('"+c(t).replace(/\s*,\s*/g,"', '")+"')"}function u(t){return"NOT "+a(t)}function c(t){return t.replace(/'/g,"''")}function h(t){return" '"+c(t)+"'"}var d,g,f,p,v,m=t("regexp-like"),C="LIKE ",w="NOT "+C,y="%",b={"<":{test:function(t,e){return e>t}},"≤":{test:function(t,e){return e>=t},sql:"<="},"=":{test:function(t,e){return t===e}},"≥":{test:function(t,e){return t>=e},sql:">="},">":{test:function(t,e){return t>e}},"≠":{test:function(t,e){return t!==e},sql:"<>"},LIKE:{test:function(t,e){return m.cached(e,!0).test(t)},type:"string"},"NOT LIKE":{test:function(t,e){return!m.cached(e,!0).test(t)},type:"string"},IN:{test:function(t,e){return n(t,e)>=0},sql:a,type:"string"},"NOT IN":{test:function(t,e){return n(t,e)<0},sql:u,type:"string"},CONTAINS:{test:function(t,e){return o(t,e)>=0},sql:l.bind(this,y,y,C),type:"string"},"NOT CONTAINS":{test:function(t,e){return o(t,e)<0},sql:l.bind(this,y,y,w),type:"string"},BEGINS:{test:function(t,e){return e=e.toString().toLowerCase(),r(t,e.length)===e},sql:l.bind(this,"",y,C),type:"string"},"NOT BEGINS":{test:function(t,e){return e=e.toString().toLowerCase(),r(t,e.length)!==e},sql:l.bind(this,"",y,w),type:"string"},ENDS:{test:function(t,e){return e=e.toString().toLowerCase(),s(t,e.length)===e},sql:l.bind(this,y,"",C),type:"string"},"NOT ENDS":{test:function(t,e){return e=e.toString().toLowerCase(),s(t,e.length)!==e},sql:l.bind(this,y,"",w),type:"string"}};d=["="],d.label="Equality",g=["<","≤","≠","≥",">"],g.label="Inquality",f=["IN","NOT IN"],f.label="Set scan",p=["CONTAINS","NOT CONTAINS","BEGINS","NOT BEGINS","ENDS","NOT ENDS"],p.label="String scan",v={options:["LIKE","NOT LIKE"],label:"Pattern matching"},b.options=[d,g,f,p,v],b.sq=h,e.exports=b},{"regexp-like":23}],11:[function(t,e,i){"use strict";function n(t){var e,i=document.createElement("div"),n=r[t].toString().match(s)[1],a=[n].concat(Array.prototype.slice.call(arguments,1)),u={};for(l.lastIndex=0;e=l.exec(n);)u[e[1]]=!0;return e=Object.keys(u),e.length&&(e.forEach(function(t){i.textContent=a[t],a[t]=i.innerHTML}),a[0]=n.replace(l,"{$1}")),i.innerHTML=o.apply(this,a),1===i.children.length&&1===i.childNodes.length?i.firstChild:i.childNodes}var o=t("templex"),r={tree:function(){},removeButton:function(){},note:function(){},notes:function(){},optionMissing:function(){}},s=/\/\*\s*([^]+?)\s+\*\//,l=/\{(\d+)\:encode\}/g;e.exports=n},{templex:25}],12:[function(t,e,i){"use strict";function n(t,e){return t&&e}function o(t,e){return t||e}var r={"op-and":{reduce:n,seed:!0,abort:!1,negate:!1,SQL:{op:"AND",beg:"(",end:")"}},"op-or":{reduce:o,seed:!1,abort:!0,negate:!1,SQL:{op:"OR",beg:"(",end:")"}},"op-nor":{reduce:o,seed:!1,abort:!0,negate:!0,SQL:{op:"OR",beg:"NOT (",end:")"}}};e.exports=r},{}],13:[function(t,e,i){"use strict";function n(t){var e=this._bound={};for(s in u)e[s]=u[s].bind(this);var i=document.createElement("div");i.classList.add("thumb"),i.onclick=e.shortStop,i.onmouseover=e.onmouseover,this.thumb=i;var o=document.createElement("div");o.classList.add("finbar-vertical"),o.appendChild(i),this.paging&&(o.onclick=e.onclick),this.bar=o,t=t||{},this.orientation="vertical",this._min=this._index=0,this._max=100;for(var s in t)if(t.hasOwnProperty(s)){var c=t[s];switch(s){case"index":this._index=c;break;case"range":r(c),this._min=c.min,this._max=c.max,this.contentSize=c.max-c.min+1;break;default:"_"!==s.charAt(0)&&"function"!=typeof n.prototype[s]&&(this[s]=c)}}l(a,"finbar-base",t.cssStylesheetReferenceElement)}function o(t){for(var e=1;e'+t.replace("mouse","")+""}),e.appendChild(n),t={},i.forEach(function(e){t[e]=n.getElementsByClassName(e)[0]})}return t},_addEvt:function(t){var e=this.testPanelItem&&this.testPanelItem[t];e&&e.classList.add("listening"),window.addEventListener(t,this._bound["on"+t])},_removeEvt:function(t){var e=this.testPanelItem&&this.testPanelItem[t];e&&e.classList.remove("listening"),window.removeEventListener(t,this._bound["on"+t])}};var a,u={shortStop:function(t){t.stopPropagation()},onwheel:function(t){this.index+=t[this.deltaProp],t.stopPropagation(),t.preventDefault()},onclick:function(t){var e=this.thumb.getBoundingClientRect(),i=t[this.oh.coordinate].thumb,div.finbar-vertical>.thumb{position:absolute;background-color:#d3d3d3;-webkit-box-shadow:0 0 1px #000;-moz-box-shadow:0 0 1px #000;box-shadow:0 0 1px #000;border-radius:4px;margin:2px;opacity:.4;transition:opacity .5s}div.finbar-horizontal>.thumb.hover,div.finbar-vertical>.thumb.hover{opacity:1;transition:opacity .5s}div.finbar-vertical{top:0;bottom:0;right:0;width:11px}div.finbar-vertical>.thumb{top:0;right:0;width:7px}div.finbar-horizontal{left:0;right:0;bottom:0;height:11px}div.finbar-horizontal>.thumb{left:0;bottom:0;height:7px}",e.exports=n},{"css-injector":4}],14:[function(t,e,i){"use strict";function n(t,e,i){var n=this;this.div=t,this.component=e,i=i||{},this.doubleClickDelay=i.doubleClickDelay||325,this.dragEndtime=Date.now(),this.canvas=document.createElement("canvas"),this.div.appendChild(this.canvas),this.canvas.style.outline="none",this.canvasCTX=this.canvas.getContext("2d"),this.gc=new u(this.canvasCTX),this.buffer=document.createElement("canvas"),this.bufferCTX=this.buffer.getContext("2d"),this.bufferGC=new u(this.bufferCTX),this.mouseLocation=new l.Point(-1,-1),this.dragstart=new l.Point(-1,-1),this.bounds=new l.Rectangle(0,0,0,0),this.hasMouse=!1,document.addEventListener("mousemove",function(t){(n.hasMouse||n.isDragging())&&n.finmousemove(t)}),document.addEventListener("mouseup",function(t){n.finmouseup(t)}),document.addEventListener("wheel",function(t){n.finwheelmoved(t)}),document.addEventListener("keydown",function(t){n.finkeydown(t)}),document.addEventListener("keyup",function(t){n.finkeyup(t)}),this.canvas.onmouseover=function(){n.hasMouse=!0},this.canvas.addEventListener("focus",function(t){n.finfocusgained(t)}),this.canvas.addEventListener("blur",function(t){n.finfocuslost(t)}),this.canvas.addEventListener("mousedown",function(t){n.finmousedown(t)}),this.canvas.addEventListener("mouseout",function(t){n.hasMouse=!1,n.finmouseout(t)}),this.canvas.addEventListener("click",function(t){n.finclick(t)}),this.canvas.addEventListener("contextmenu",function(t){return n.fincontextmenu(t),t.preventDefault(),!1}),a.addEventListener(this.canvas,"tap",function(t){n.fintap(t)}),a.addEventListener(this.canvas,"holdpulse",function(t){n.finholdpulse(t)}),a.addEventListener(this.canvas,"flick",function(t){n.finflick(t)}),a.addEventListener(this.canvas,"release",function(t){n.finrelease(t)}),a.addEventListener(this.canvas,"trackstart",function(t){n.fintrackstart(t)}),a.addEventListener(this.canvas,"track",function(t){n.fintrack(t)}),a.addEventListener(this.canvas,"trackend",function(t){n.fintrackend(t)}),this.canvas.setAttribute("tabindex",0),this.canvas.contentEditable=!0,this.resize(),this.beginResizing(),this.beginPainting()}function o(t){if(g){for(var e=0;ei;i++)t[i]=e;return t[27]=["ESC","ESCSHIFT"],t[192]=["`","~"],t[49]=["1","!"],t[50]=["2","@"],t[51]=["3","#"],t[52]=["4","$"],t[53]=["5","%"],t[54]=["6","^"],t[55]=["7","&"],t[56]=["8","*"],t[57]=["9","("],t[48]=["0",")"],t[189]=["-","_"],t[187]=["=","+"],t[8]=["BACKSPACE","BACKSPACESHIFT"],t[46]=["DELETE","DELETESHIFT"],t[9]=["TAB","TABSHIFT"],t[81]=["q","Q"],t[87]=["w","W"],t[69]=["e","E"],t[82]=["r","R"],t[84]=["t","T"],t[89]=["y","Y"],t[85]=["u","U"],t[73]=["i","I"],t[79]=["o","O"],t[80]=["p","P"],t[219]=["[","{"],t[221]=["]","}"],t[220]=["\\","|"],t[220]=["CAPSLOCK","CAPSLOCKSHIFT"],t[65]=["a","A"],t[83]=["s","S"],t[68]=["d","D"],t[70]=["f","F"],t[71]=["g","G"],t[72]=["h","H"],t[74]=["j","J"],t[75]=["k","K"],t[76]=["l","L"],t[186]=[";",":"],t[222]=["'","|"],t[13]=["RETURN","RETURNSHIFT"],t[16]=["SHIFT","SHIFT"],t[90]=["z","Z"],t[88]=["x","X"],t[67]=["c","C"],t[86]=["v","V"],t[66]=["b","B"],t[78]=["n","N"],t[77]=["m","M"],t[188]=[",","<"],t[190]=[".",">"],t[191]=["/","?"],t[16]=["SHIFT","SHIFT"],t[17]=["CTRL","CTRLSHIFT"],t[18]=["ALT","ALTSHIFT"],t[91]=["COMMANDLEFT","COMMANDLEFTSHIFT"],t[32]=["SPACE","SPACESHIFT"],t[93]=["COMMANDRIGHT","COMMANDRIGHTSHIFT"],t[18]=["ALT","ALTSHIFT"],t[38]=["UP","UPSHIFT"],t[37]=["LEFT","LEFTSHIFT"],t[40]=["DOWN","DOWNSHIFT"],t[39]=["RIGHT","RIGHTSHIFT"],t[33]=["PAGEUP","PAGEUPSHIFT"],t[34]=["PAGEDOWN","PAGEDOWNSHIFT"],t[35]=["PAGERIGHT","PAGERIGHTSHIFT"],t[36]=["PAGELEFT","PAGELEFTSHIFT"],t[112]=["F1","F1SHIFT"],t[113]=["F2","F2SHIFT"],t[114]=["F3","F3SHIFT"],t[115]=["F4","F4SHIFT"],t[116]=["F5","F5SHIFT"],t[117]=["F6","F6SHIFT"],t[118]=["F7","F7SHIFT"],t[119]=["F8","F8SHIFT"],t[120]=["F9","F9SHIFT"],t[121]=["F10","F10SHIFT"],t[122]=["F11","F1S1HIFT"],t[123]=["F12","F121HIFT"],t}var l=t("rectangular"),a=t("./js/polymergestures.dev.js"),u=t("./js/GraphicsContext.js"),c=200,h=[],d=[],g=!0,f=!0,p=s();n.prototype={constructor:n.prototype.constructor,div:null,component:null,gestures:a,canvas:null,canvasCTX:null,focuser:null,buffer:null,ctx:null,mouseLocation:null,holdPulseCount:-1,dragstart:null,origin:null,bounds:null,dirty:!1,size:null,mousedown:!1,dragging:!1,repeatKeyCount:0,repeatKey:null,repeatKeyStartTime:0,currentKeys:[],hasMouse:!1,lastDoubleClickTime:0,dragEndTime:0,lastRepaintTime:0,addEventListener:function(t,e){this.canvas.addEventListener(t,e)},stopPaintLoop:function(){g=!1},restartPaintLoop:function(){g||(g=!0,requestAnimationFrame(o))},stopResizeLoop:function(){f=!1},restartResizeLoop:function(){f||(f=!0,setInterval(r,200))},detached:function(){this.stopPainting(),this.stopResizing()},useHiDPI:function(){return this.component.resolveProperty("useHiDPI")},useBitBlit:function(){return this.component.resolveProperty("useBitBlit")},getFPS:function(){var t=this.component.resolveProperty("repaintIntervalRate");return t?parseInt(t):0},tickPaint:function(t){var e=this.getFPS();if(0!==e){var i=1e3/e,n=t-this.lastRepaintTime;n>i&&this.dirty&&(this.lastRepaintTime=t-n%i,this.paintNow())}},beginPainting:function(){var t=this;this.dirty=!0,this.tickPainter=function(e){t.tickPaint(e)},h.push(this)},stopPainting:function(){h.splice(h.indexOf(this),1)},beginResizing:function(){var t=this;this.tickResizer=function(){t.checksize()},d.push(this)},stopResizing:function(){d.splice(d.indexOf(this),1)},start:function(){this.beginPainting(),this.beginResizing()},stop:function(){this.stopPainting(),this.stopResizing()},checksize:function(){var t=this.div.getBoundingClientRect();(t.width!==this.size.width||t.height!==this.size.height)&&this.sizeChangedNotification()},sizeChangedNotification:function(){this.resize()},resize:function(){var t=this.size=this.div.getBoundingClientRect();this.canvas.width=this.buffer.width=t.width,this.canvas.height=this.buffer.height=t.height;var e=1,i=this.useBitBlit(),n=window.devicePixelRatio&&this.useHiDPI();if(n){var o=window.devicePixelRatio||1,r=this.canvasCTX.webkitBackingStorePixelRatio||this.canvasCTX.mozBackingStorePixelRatio||this.canvasCTX.msBackingStorePixelRatio||this.canvasCTX.oBackingStorePixelRatio||this.canvasCTX.backingStorePixelRatio||1;e=o/r}var s=this.canvas.getAttribute("width"),a=this.canvas.getAttribute("height");this.canvas.width=s*e,this.canvas.height=a*e,this.buffer.width=s*e,this.buffer.height=a*e,this.canvas.style.width=s+"px",this.canvas.style.height=a+"px",this.buffer.style.width=s+"px",this.buffer.style.height=a+"px",this.bufferCTX.scale(e,e),n&&!i&&this.canvasCTX.scale(e,e),this.bounds=new l.Rectangle(0,0,t.width,t.height);var u=this.component;u&&u.setBounds(this.bounds),this.resizeNotification(),this.paintNow()},resizeNotification:function(){},getBounds:function(){return this.bounds},paintNow:function(){var t=this;this.safePaintImmediately(function(e){e.clearRect(0,0,t.canvas.width,t.canvas.height);var i=t.component;i&&i._paint(e),t.dirty=!1})},safePaintImmediately:function(t){var e=this.useBitBlit(),i=e?this.bufferGC:this.gc;try{i.save(),t(i)}catch(n){console.error(n)}finally{i.restore()}e&&this.flushBuffer()},flushBuffer:function(){this.buffer.width>0&&this.buffer.height>0&&this.canvasCTX.drawImage(this.buffer,0,0)},dispatchNewEvent:function(t,e,i){return i={detail:i||{}},i.detail.primitiveEvent=t,this.canvas.dispatchEvent(new CustomEvent(e,i))},dispatchNewMouseKeysEvent:function(t,e,i){return i=i||{},i.mouse=this.mouseLocation,i.keys=this.currentKeys,this.dispatchNewEvent(t,e,i)},finmousemove:function(t){!this.isDragging()&&this.mousedown&&(this.beDragging(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-dragstart",{isRightClick:this.isRightClick(t)}),this.dragstart=new l.Point(this.mouseLocation.x,this.mouseLocation.y)),this.mouseLocation=this.getLocal(t),this.isDragging()&&this.dispatchNewMouseKeysEvent(t,"fin-canvas-drag",{dragstart:this.dragstart,isRightClick:this.isRightClick(t)}),this.bounds.contains(this.mouseLocation)&&this.dispatchNewMouseKeysEvent(t,"fin-canvas-mousemove")},finmousedown:function(t){this.mouseLocation=this.mouseDownLocation=this.getLocal(t),this.mousedown=!0,this.dispatchNewMouseKeysEvent(t,"fin-canvas-mousedown",{isRightClick:this.isRightClick(t)}),this.takeFocus()},finmouseup:function(t){this.isDragging()&&(this.dispatchNewMouseKeysEvent(t,"fin-canvas-dragend",{dragstart:this.dragstart,isRightClick:this.isRightClick(t)}),this.beNotDragging(),this.dragEndtime=Date.now()),this.mousedown=!1,this.dispatchNewMouseKeysEvent(t,"fin-canvas-mouseup",{isRightClick:this.isRightClick(t)})},finmouseout:function(t){this.mousedown||(this.mouseLocation=new l.Point(-1,-1)),this.dispatchNewMouseKeysEvent(t,"fin-canvas-mouseout")},finwheelmoved:function(t){!this.isDragging()&&this.hasFocus()&&(t.preventDefault(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-wheelmoved",{isRightClick:this.isRightClick(t)}))},finclick:function(t){this.doubleClickTimer&&Date.now()-this.lastClickTimen||i-this.dragEndtime<100||setTimeout(function(){e._fintap(t)},180)},_fintap:function(t){var e=Date.now(),i=e-this.lastDoubleClickTime;300>i||(this.mouseDownLocation&&(this.mouseLocation=this.mouseDownLocation,this.mouseDownLocation=void 0),this.dispatchNewMouseKeysEvent(t,"fin-canvas-tap",{isRightClick:this.isRightClick(t)}))},findblclick:function(t){this.mouseLocation=this.getLocal(t),this.lastDoubleClickTime=Date.now(),this.dispatchNewMouseKeysEvent(t,"fin-canvas-dblclick",{isRightClick:this.isRightClick(t)})},getCharMap:function(){return p},finkeydown:function(t){if(this.hasFocus()){var e=t.shiftKey?p[t.keyCode][1]:p[t.keyCode][0];t.repeat?this.repeatKey===e?this.repeatKeyCount++:(this.repeatKey=e,this.repeatKeyStartTime=Date.now()):(this.repeatKey=null,this.repeatKeyCount=0,this.repeatKeyStartTime=0),-1===this.currentKeys.indexOf(e)&&this.currentKeys.push(e),this.dispatchNewEvent(t,"fin-canvas-keydown",{alt:t.altKey,ctrl:t.ctrlKey,"char":e,code:t.charCode,key:t.keyCode,meta:t.metaKey,repeatCount:this.repeatKeyCount,repeatStartTime:this.repeatKeyStartTime,shift:t.shiftKey,identifier:t.keyIdentifier,currentKeys:this.currentKeys.slice(0)})}},finkeyup:function(t){var e=t.shiftKey?p[t.keyCode][1]:p[t.keyCode][0];this.currentKeys.splice(this.currentKeys.indexOf(e),1),this.hasFocus()&&(this.repeatKeyCount=0,this.repeatKey=null,this.repeatKeyStartTime=0,this.dispatchNewEvent(t,"fin-canvas-keyup",{alt:t.altKey,ctrl:t.ctrlKey,"char":e,code:t.charCode,key:t.keyCode,meta:t.metaKey,repeat:t.repeat,shift:t.shiftKey,identifier:t.keyIdentifier,currentKeys:this.currentKeys.slice(0)}))},finfocusgained:function(t){this.dispatchNewEvent(t,"fin-canvas-focus-gained")},finfocuslost:function(t){this.dispatchNewEvent(t,"fin-canvas-focus-lost")},fincontextmenu:function(t){t.ctrlKey&&-1===this.currentKeys.indexOf("CTRL")&&this.currentKeys.push("CTRL"),this.doubleRightClickTimer&&Date.now()-this.lastClickTime=0?t=this.walk(t,o):e=this.walk(e,-o);t&&e&&t!==e;)t=t.parentNode||t.host,e=e.parentNode||e.host;return t},walk:function(t,e){for(var i=0;t&&e>i;i++)t=t.parentNode||t.host;return t},depth:function(t){for(var e=0;t;)e++,t=t.parentNode||t.host;return e},deepContains:function(t,e){var i=this.LCA(t,e);return i===t},insideNode:function(t,e,i){var n=t.getBoundingClientRect();return n.left<=e&&e<=n.right&&n.top<=i&&i<=n.bottom},path:function(t){var i;if(e&&t.path&&t.path.length)i=t.path;else{i=[];for(var n=this.findTarget(t);n;)i.push(n),n=n.parentNode||n.host}return i}};t.targetFinding=s,t.findTarget=s.findTarget.bind(s),t.deepContains=s.deepContains.bind(s),t.insideNode=s.insideNode}(i),function(){function t(t){return"html /deep/ "+e(t)}function e(t){return'[touch-action="'+t+'"]'}function i(t){return"{ -ms-touch-action: "+t+"; touch-action: "+t+";}"}var n=["none","auto","pan-x","pan-y",{rule:"pan-x pan-y",selectors:["pan-x pan-y","pan-y pan-x"]},"manipulation"],o="",r="string"==typeof document.head.style.touchAction,s=!window.ShadowDOMPolyfill&&document.head.createShadowRoot;if(r){n.forEach(function(n){String(n)===n?(o+=e(n)+i(n)+"\n",s&&(o+=t(n)+i(n)+"\n")):(o+=n.selectors.map(e)+i(n.rule)+"\n",s&&(o+=n.selectors.map(t)+i(n.rule)+"\n"))});var l=document.createElement("style");l.textContent=o,document.head.appendChild(l)}}(),function(t){var e=["bubbles","cancelable","view","detail","screenX","screenY","clientX","clientY","ctrlKey","altKey","shiftKey","metaKey","button","relatedTarget","pageX","pageY"],i=[!1,!1,null,null,0,0,0,0,!1,!1,!1,!1,0,null,0,0],n=function(){return function(){}},o={preventTap:n,makeBaseEvent:function(t,e){var i=document.createEvent("Event");return i.initEvent(t,e.bubbles||!1,e.cancelable||!1),i.preventTap=o.preventTap(i),i},makeGestureEvent:function(t,e){e=e||Object.create(null);for(var i,n=this.makeBaseEvent(t,e),o=0,r=Object.keys(e);o-1?this.values[i]=e:(this.keys.push(t),this.values.push(e))},has:function(t){return this.keys.indexOf(t)>-1},"delete":function(t){var e=this.keys.indexOf(t);e>-1&&(this.keys.splice(e,1),this.values.splice(e,1))},get:function(t){var e=this.keys.indexOf(t);return this.values[e]},clear:function(){this.keys.length=0,this.values.length=0},forEach:function(t,e){this.values.forEach(function(i,n){t.call(e,i,this.keys[n],this)},this)},pointers:function(){return this.keys.length}},t.PointerMap=e}(i),function(t){var e,i=["bubbles","cancelable","view","detail","screenX","screenY","clientX","clientY","ctrlKey","altKey","shiftKey","metaKey","button","relatedTarget","buttons","pointerId","width","height","pressure","tiltX","tiltY","pointerType","hwTimestamp","isPrimary","type","target","currentTarget","which","pageX","pageY","timeStamp","preventTap","tapPrevented","_source"],n=[!1,!1,null,null,0,0,0,0,!1,!1,!1,!1,0,null,0,0,0,0,0,0,0,"",0,!1,"",null,null,0,0,0,0,function(){},!1],o="undefined"!=typeof SVGElementInstance,r=t.eventFactory,s={IS_IOS:!1,pointermap:new t.PointerMap,requiredGestures:new t.PointerMap,eventMap:Object.create(null),eventSources:Object.create(null),eventSourceList:[],gestures:[],dependencyMap:{down:{listeners:0,index:-1},up:{listeners:0,index:-1}},gestureQueue:[],registerSource:function(t,e){var i=e,n=i.events;n&&(n.forEach(function(t){i[t]&&(this.eventMap[t]=i[t].bind(i))},this),this.eventSources[t]=i,this.eventSourceList.push(i))},registerGesture:function(t,e){var i=Object.create(null);i.listeners=0,i.index=this.gestures.length;for(var n,o=0;oo&&(i=this.eventSourceList[o]);o++)i.register.call(i,t,e)},unregister:function(t){for(var e,i=this.eventSourceList.length,n=0;i>n&&(e=this.eventSourceList[n]);n++)e.unregister.call(e,t)},down:function(t){this.requiredGestures.set(t.pointerId,e),this.fireEvent("down",t)},move:function(t){t.type="move",this.fillGestureQueue(t)},up:function(t){this.fireEvent("up",t),this.requiredGestures["delete"](t.pointerId)},cancel:function(t){t.tapPrevented=!0,this.fireEvent("up",t),this.requiredGestures["delete"](t.pointerId)},addGestureDependency:function(t,e){var i=t._pgEvents;if(i&&e)for(var n,o,r,s=Object.keys(i),l=0;l0&&(n=this.dependencyMap[r],o=n?n.index:-1,e[o]=!0)},eventHandler:function(i){var n=i.type;if("touchstart"===n||"mousedown"===n||"pointerdown"===n||"MSPointerDown"===n)if(i._handledByPG||(e={}),this.IS_IOS){var o=i;if("touchstart"===n){var r=i.changedTouches[0];o={target:i.target,clientX:r.clientX,clientY:r.clientY,path:i.path}}for(var s,l=i.path||t.targetFinding.path(o),a=0;an&&(i=e[n]);n++)this.addEvent(t,i)},unlisten:function(t,e){for(var i,n=0,o=e.length;o>n&&(i=e[n]);n++)this.removeEvent(t,i)},addEvent:function(t,e){t.addEventListener(e,this.boundHandler)},removeEvent:function(t,e){t.removeEventListener(e,this.boundHandler)},makeEvent:function(t,e){var i=r.makePointerEvent(t,e);return i.preventDefault=e.preventDefault,i.tapPrevented=e.tapPrevented,i._target=i._target||e.target,i},fireEvent:function(t,e){var i=this.makeEvent(t,e);return this.dispatchEvent(i)},cloneEvent:function(t){for(var e,r=Object.create(null),s=0;s0&&t._pgListeners--,0===t._pgListeners&&s.unregister(t),t._pgEvents&&(t._pgEvents[i]>0?t._pgEvents[i]--:t._pgEvents[i]=0)),Boolean(n)},t.removeEventListener=function(e,i,n,o){n&&(t.deactivateGesture(e,i),e.removeEventListener(i,n,o))}}(i),function(t){var e=t.dispatcher,i=e.pointermap,n=25,o=[0,1,4,2],r=0,s=/Linux.*Firefox\//i,l=function(){if(s.test(navigator.userAgent))return!1;try{return 1===new MouseEvent("test",{buttons:1}).buttons}catch(t){return!1}}(),a={POINTER_ID:1,POINTER_TYPE:"mouse",events:["mousedown","mousemove","mouseup"],exposes:["down","up","move"],register:function(t){e.listen(t,this.events)},unregister:function(t){t.nodeType!==Node.DOCUMENT_NODE&&e.unlisten(t,this.events)},lastTouches:[],isEventSimulatedFromTouch:function(t){for(var e,i=this.lastTouches,o=t.clientX,r=t.clientY,s=0,l=i.length;l>s&&(e=i[s]);s++){var a=Math.abs(o-e.x),u=Math.abs(r-e.y);if(n>=a&&n>=u)return!0}},prepareEvent:function(t){var i=e.cloneEvent(t);if(i.pointerId=this.POINTER_ID,i.isPrimary=!0,i.pointerType=this.POINTER_TYPE,i._source="mouse",!l){var n=t.type,s=o[t.which]||0;"mousedown"===n?r|=s:"mouseup"===n&&(r&=~s),i.buttons=r}return i},mousedown:function(n){if(!this.isEventSimulatedFromTouch(n)){var o=(i.has(this.POINTER_ID),this.prepareEvent(n));o.target=t.findTarget(n),i.set(this.POINTER_ID,o.target),e.down(o)}},mousemove:function(t){if(!this.isEventSimulatedFromTouch(t)){var n=i.get(this.POINTER_ID);if(n){var o=this.prepareEvent(t);o.target=n,0===(l?o.buttons:o.which)?(l||(r=o.buttons=0),e.cancel(o),this.cleanupMouse(o.buttons)):e.move(o)}}},mouseup:function(n){if(!this.isEventSimulatedFromTouch(n)){var o=this.prepareEvent(n);o.relatedTarget=t.findTarget(n),o.target=i.get(this.POINTER_ID),e.up(o),this.cleanupMouse(o.buttons)}},cleanupMouse:function(t){0===t&&i["delete"](this.POINTER_ID)}};t.mouseEvents=a}(i),function(t){var e=t.dispatcher,i=(t.targetFinding.allShadows.bind(t.targetFinding),e.pointermap),n=(Array.prototype.map.call.bind(Array.prototype.map),2500),o=25,r=200,s=20,l=!1,a={IS_IOS:!1,events:["touchstart","touchmove","touchend","touchcancel"],exposes:["down","up","move"],register:function(t,i){(this.IS_IOS?i:!i)&&e.listen(t,this.events)},unregister:function(t){this.IS_IOS||e.unlisten(t,this.events)},scrollTypes:{EMITTER:"none",XSCROLLER:"pan-x",YSCROLLER:"pan-y"},touchActionToScrollType:function(t){var e=t,i=this.scrollTypes;return e===i.EMITTER?"none":e===i.XSCROLLER?"X":e===i.YSCROLLER?"Y":"XY"},POINTER_TYPE:"touch",firstTouch:null,isPrimaryTouch:function(t){return this.firstTouch===t.identifier},setPrimaryTouch:function(t){(0===i.pointers()||1===i.pointers()&&i.has(1))&&(this.firstTouch=t.identifier,this.firstXY={X:t.clientX,Y:t.clientY},this.firstTarget=t.target,this.scrolling=null,this.cancelResetClickCount())},removePrimaryPointer:function(t){t.isPrimary&&(this.firstTouch=null,this.firstXY=null,this.resetClickCount())},clickCount:0,resetId:null,resetClickCount:function(){var t=function(){this.clickCount=0,this.resetId=null}.bind(this);this.resetId=setTimeout(t,r)},cancelResetClickCount:function(){this.resetId&&clearTimeout(this.resetId)},typeToButtons:function(t){var e=0;return("touchstart"===t||"touchmove"===t)&&(e=1),e},findTarget:function(e,n){if("touchstart"===this.currentTouchEvent.type){if(this.isPrimaryTouch(e)){var o={clientX:e.clientX,clientY:e.clientY,path:this.currentTouchEvent.path,target:this.currentTouchEvent.target};return t.findTarget(o)}return t.findTarget(e)}return i.get(n)},touchToPointer:function(t){var i=this.currentTouchEvent,n=e.cloneEvent(t),o=n.pointerId=t.identifier+2;n.target=this.findTarget(t,o),n.bubbles=!0,n.cancelable=!0,n.detail=this.clickCount,n.buttons=this.typeToButtons(i.type),n.width=t.webkitRadiusX||t.radiusX||0,n.height=t.webkitRadiusY||t.radiusY||0,n.pressure=t.webkitForce||t.force||.5,n.isPrimary=this.isPrimaryTouch(t),n.pointerType=this.POINTER_TYPE,n._source="touch";var r=this;return n.preventDefault=function(){r.scrolling=!1,r.firstXY=null,i.preventDefault()},n},processTouches:function(t,e){var n=t.changedTouches;this.currentTouchEvent=t;for(var o,r,s=0;s=u}return i}},findTouch:function(t,e){for(var i,n=0,o=t.length;o>n&&(i=t[n]);n++)if(i.identifier===e)return!0},vacuumTouches:function(t){var e=t.touches;if(i.pointers()>=e.length){var n=[];i.forEach(function(t,i){if(1!==i&&!this.findTouch(e,i-2)){var o=t;n.push(o)}},this),n.forEach(function(t){this.cancel(t),i["delete"](t.pointerId)},this)}},touchstart:function(t){this.vacuumTouches(t),this.setPrimaryTouch(t.changedTouches[0]),this.dedupSynthMouse(t),this.scrolling||(this.clickCount++,this.processTouches(t,this.down))},down:function(t){e.down(t)},touchmove:function(t){if(l)t.cancelable&&this.processTouches(t,this.move);else if(this.scrolling){if(this.firstXY){var e=t.changedTouches[0],i=e.clientX-this.firstXY.X,n=e.clientY-this.firstXY.Y,o=Math.sqrt(i*i+n*n);o>=s&&(this.touchcancel(t),this.scrolling=!0,this.firstXY=null)}}else null===this.scrolling&&this.shouldScroll(t)?this.scrolling=!0:(this.scrolling=!1,t.preventDefault(),this.processTouches(t,this.move))},move:function(t){e.move(t)},touchend:function(t){this.dedupSynthMouse(t),this.processTouches(t,this.up)},up:function(i){i.relatedTarget=t.findTarget(i),e.up(i)},cancel:function(t){e.cancel(t)},touchcancel:function(t){t._cancel=!0,this.processTouches(t,this.cancel)},cleanUpPointer:function(t){i["delete"](t.pointerId),this.removePrimaryPointer(t)},dedupSynthMouse:function(e){var i=t.mouseEvents.lastTouches,o=e.changedTouches[0];if(this.isPrimaryTouch(o)){var r={x:o.clientX,y:o.clientY};i.push(r);var s=function(t,e){var i=t.indexOf(e); +i>-1&&t.splice(i,1)}.bind(null,i,r);setTimeout(s,n)}}},u=Event.prototype.stopImmediatePropagation||Event.prototype.stopPropagation;document.addEventListener("click",function(e){var i=e.clientX,n=e.clientY,r=function(t){var e=Math.abs(i-t.x),r=Math.abs(n-t.y);return o>=e&&o>=r},s=t.mouseEvents.lastTouches.some(r),l=t.targetFinding.path(e);if(s){for(var c=0;c0?1:-1},calcPositionDelta:function(t,e){var i=0,n=0;return t&&e&&(i=e.pageX-t.pageX,n=e.pageY-t.pageY),{x:i,y:n}},fireTrack:function(t,e,n){var o=n,r=this.calcPositionDelta(o.downEvent,e),s=this.calcPositionDelta(o.lastMoveEvent,e);if(s.x)o.xDirection=this.clampDir(s.x);else if("trackx"===t)return;if(s.y)o.yDirection=this.clampDir(s.y);else if("tracky"===t)return;var l={bubbles:!0,cancelable:!0,trackInfo:o.trackInfo,relatedTarget:e.relatedTarget,pointerType:e.pointerType,pointerId:e.pointerId,_source:"track"};"tracky"!==t&&(l.x=e.x,l.dx=r.x,l.ddx=s.x,l.clientX=e.clientX,l.pageX=e.pageX,l.screenX=e.screenX,l.xDirection=o.xDirection),"trackx"!==t&&(l.dy=r.y,l.ddy=s.y,l.y=e.y,l.clientY=e.clientY,l.pageY=e.pageY,l.screenY=e.screenY,l.yDirection=o.yDirection);var a=i.makeGestureEvent(t,l);o.downTarget.dispatchEvent(a)},down:function(t){if(t.isPrimary&&("mouse"===t.pointerType?1===t.buttons:!0)){var e={downEvent:t,downTarget:t.target,trackInfo:{},lastMoveEvent:null,xDirection:0,yDirection:0,tracking:!1};n.set(t.pointerId,e)}},move:function(t){var e=n.get(t.pointerId);if(e){if(!e.tracking){var i=this.calcPositionDelta(e.downEvent,t),o=i.x*i.x+i.y*i.y;o>this.WIGGLE_THRESHOLD&&(e.tracking=!0,e.lastMoveEvent=e.downEvent,this.fireTrack("trackstart",t,e))}e.tracking&&(this.fireTrack("track",t,e),this.fireTrack("trackx",t,e),this.fireTrack("tracky",t,e)),e.lastMoveEvent=t}},up:function(t){var e=n.get(t.pointerId);e&&(e.tracking&&this.fireTrack("trackend",t,e),n["delete"](t.pointerId))}};e.registerGesture("track",o)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n={HOLD_DELAY:200,WIGGLE_THRESHOLD:16,events:["down","move","up"],exposes:["hold","holdpulse","release"],heldPointer:null,holdJob:null,pulse:function(){var t=Date.now()-this.heldPointer.timeStamp,e=this.held?"holdpulse":"hold";this.fireHold(e,t),this.held=!0},cancel:function(){clearInterval(this.holdJob),this.held&&this.fireHold("release"),this.held=!1,this.heldPointer=null,this.target=null,this.holdJob=null},down:function(t){t.isPrimary&&!this.heldPointer&&(this.heldPointer=t,this.target=t.target,this.holdJob=setInterval(this.pulse.bind(this),this.HOLD_DELAY))},up:function(t){this.heldPointer&&this.heldPointer.pointerId===t.pointerId&&this.cancel()},move:function(t){if(this.heldPointer&&this.heldPointer.pointerId===t.pointerId){var e=t.clientX-this.heldPointer.clientX,i=t.clientY-this.heldPointer.clientY;e*e+i*i>this.WIGGLE_THRESHOLD&&this.cancel()}},fireHold:function(t,e){var n={bubbles:!0,cancelable:!0,pointerType:this.heldPointer.pointerType,pointerId:this.heldPointer.pointerId,x:this.heldPointer.clientX,y:this.heldPointer.clientY,_source:"hold"};e&&(n.holdTime=e);var o=i.makeGestureEvent(t,n);this.target.dispatchEvent(o)}};e.registerGesture("hold",n)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n=new t.PointerMap,o={events:["down","up"],exposes:["tap"],down:function(t){t.isPrimary&&!t.tapPrevented&&n.set(t.pointerId,{target:t.target,buttons:t.buttons,x:t.clientX,y:t.clientY})},shouldTap:function(t,e){var i=!0;return"mouse"===t.pointerType&&(i=1^t.buttons&&1&e.buttons),i&&!t.tapPrevented},up:function(e){var o=n.get(e.pointerId);if(o&&this.shouldTap(e,o)){var r=t.targetFinding.LCA(o.target,e.relatedTarget);if(r){var s=i.makeGestureEvent("tap",{bubbles:!0,cancelable:!0,x:e.clientX,y:e.clientY,detail:e.detail,pointerType:e.pointerType,pointerId:e.pointerId,altKey:e.altKey,ctrlKey:e.ctrlKey,metaKey:e.metaKey,shiftKey:e.shiftKey,_source:"tap"});r.dispatchEvent(s)}}n["delete"](e.pointerId)}};i.preventTap=function(t){return function(){t.tapPrevented=!0,n["delete"](t.pointerId)}},e.registerGesture("tap",o)}(i),function(t){var e=t.dispatcher,i=t.eventFactory,n=new t.PointerMap,o=180/Math.PI,r={events:["down","up","move","cancel"],exposes:["pinchstart","pinch","pinchend","rotate"],defaultActions:{pinch:"none",rotate:"none"},reference:{},down:function(e){if(n.set(e.pointerId,e),2==n.pointers()){var i=this.calcChord(),o=this.calcAngle(i);this.reference={angle:o,diameter:i.diameter,target:t.targetFinding.LCA(i.a.target,i.b.target)},this.firePinch("pinchstart",i.diameter,i)}},up:function(t){var e=n.get(t.pointerId),i=n.pointers();if(e){if(2===i){var o=this.calcChord();this.firePinch("pinchend",o.diameter,o)}n["delete"](t.pointerId)}},move:function(t){n.has(t.pointerId)&&(n.set(t.pointerId,t),n.pointers()>1&&this.calcPinchRotate())},cancel:function(t){this.up(t)},firePinch:function(t,e,n){var o=e/this.reference.diameter,r=i.makeGestureEvent(t,{bubbles:!0,cancelable:!0,scale:o,centerX:n.center.x,centerY:n.center.y,_source:"pinch"});this.reference.target.dispatchEvent(r)},fireRotate:function(t,e){var n=Math.round((t-this.reference.angle)%360),o=i.makeGestureEvent("rotate",{bubbles:!0,cancelable:!0,angle:n,centerX:e.center.x,centerY:e.center.y,_source:"pinch"});this.reference.target.dispatchEvent(o)},calcPinchRotate:function(){var t=this.calcChord(),e=t.diameter,i=this.calcAngle(t);e!=this.reference.diameter&&this.firePinch("pinch",e,t),i!=this.reference.angle&&this.fireRotate(i,t)},calcChord:function(){var t=[];n.forEach(function(e){t.push(e)});for(var e,i,o,r=0,s={a:t[0],b:t[1]},l=0;lr&&(r=o,s={a:a,b:c})}return e=Math.abs(s.a.clientX+s.b.clientX)/2,i=Math.abs(s.a.clientY+s.b.clientY)/2,s.center={x:e,y:i},s.diameter=r,s},calcAngle:function(t){var e=t.a.clientX-t.b.clientX,i=t.a.clientY-t.b.clientY;return(360+Math.atan2(i,e)*o)%360}};e.registerGesture("pinch",r)}(i)},{}],18:[function(t,e,i){"use strict";function n(t,e){if(!(this instanceof n))throw d('Not called with "new" keyword.');var i,o,s=this;e=e||{},"string"==typeof t?(o=r(document.querySelectorAll(t)),i=c(o)):t[0]instanceof Element?(o=r(t),i=c(o)):(o=[],i=u(t,e),i.forEach(function(t){o=o.concat(r(t.element.querySelectorAll("li")))})),i.forEach(function(t){t.element.addEventListener("wheel",h)}),o.forEach(function(t,e){var i=t!==t.parentElement.lastElementChild?s.addEvt(t,"mousedown",t,!0):{element:t};o[e]=i}),g="transform"in o[0].element.style?"transform":"-webkit-transform",this.modelLists=i,this.items=o,this.bindings={},this.callback={},m(v,"list-dragon-base",e.cssStylesheetReferenceElement)}function o(t,e,i){if(t){var n=0>p&&t>=0||0===p&&0!==t||p>0&&0>=t;p=t>0?Math.min(50,t):Math.max(-50,t),n&&(clearInterval(f),f=setInterval(function(t){var e=i.scrollTop+p;0>p&&t>e||p>0&&e>t?(i.scrollTop=t,clearInterval(f)):i.scrollTop=e},125))}else clearInterval(f),p=0}function r(t){return Array.prototype.slice.call(t)}function s(t,e){return e.top<=t.y&&t.y<=e.bottom&&e.left<=t.x&&t.x<=e.right}function l(t,e){return"translate("+Math.floor(t+window.scrollX)+"px,"+Math.floor(e+window.scrollY)+"px)"}function a(t){var e=document.createTextNode(t);return document.createElement("a").appendChild(e).parentNode.innerHTML}function u(t,e){var i=e.label||"{label}";return t.forEach(function(n,o){var r=n.label||i,s=void 0!==n.htmlEncode&&n.htmlEncode||e.htmlEncode,l=document.createElement("div"),u=document.createElement("ul");if(n.models)Object.keys(n).forEach(function(t){"models"!==t&&(n.models[t]=n[t])}),t[o]=n=n.models;else{if(!(n instanceof Array))throw d("List [{1}] not an array of models (with or without additional properties) OR an object (with a `models` property containing an array of models).",o);n.models=n}n.forEach(function(t){var i=t.label||r,o=void 0!==t.htmlEncode&&t.htmlEncode||s,l="object"==typeof t?t:{label:t},c=C.call([l,n,e],i),h=document.createElement("li");h.innerHTML=o?a(c):c,u.appendChild(h)});var c=document.createElement("li");if(c.innerHTML=" ",u.appendChild(c),n.title){var h=document.createElement("div");h.innerHTML=s?a(n.title):n.title,l.appendChild(h)}l.appendChild(u),l.className=n.cssClassNames||e.cssClassNames||"dragon-list",n.element=u,n.container=l}),t}function c(t){var e=[];return t.forEach(function(t){var i=t.parentElement,n=i.parentElement,o=[];e.find(function(t){return t.element===i})||(r(i.querySelectorAll("li")).forEach(function(t){t!==i.lastElementChild&&o.push(t.innerHTML)}),o.element=i,o.container=n,e.push(o))}),e}function h(t){t.stopPropagation()}function d(){return"list-dragon: "+C.apply(this,Array.prototype.slice.call(arguments))}var g,f,p,v,m=t("css-injector"),C=t("templex"),w=null;v="div.dragon-list{position:relative;background-color:#fff}div.dragon-list>div,div.dragon-list>ul{position:absolute;left:0;right:0}div.dragon-list>div{text-align:center;background-color:#00796b;color:#fff;box-shadow:0 3px 6px rgba(0,0,0,.16),0 3px 6px rgba(0,0,0,.23);overflow:hidden;white-space:nowrap}div.dragon-list>ul{overflow-y:auto;bottom:0;margin:0;padding:0;box-shadow:0 1px 3px rgba(0,0,0,.12),0 1px 2px rgba(0,0,0,.24)}div.dragon-list>ul>li,li.dragon-pop{white-space:nowrap;list-style-type:none;border:0 solid #f4f4f4;border-bottom:1px solid #e0e0e0;cursor:move;transition:border-top-width .2s}div.dragon-list>ul>li:last-child{height:0;border-bottom:none}li.dragon-pop{position:fixed;background-color:#fff;border:1px solid #e0e0e0;left:0;top:0;overflow-x:hidden;box-shadow:rgba(0,0,0,.188235) 0 10px 20px,rgba(0,0,0,.227451) 0 6px 6px}",n.prototype={addEvt:function(t,e,i,n){var o={handler:y[e].bind(t,this),element:i||window};return n||(this.bindings[e]=o),o.element.addEventListener(e,o.handler),o},removeEvt:function(t){var e=this.bindings[t];delete this.bindings[t],e.element.removeEventListener(t,e.handler)},removeAllEventListeners:function(){for(var t in this.bindings){var e=this.bindings[t];e.element.removeEventListener(t,e.handler)}this.items.forEach(function(t){t.handler&&t.element.removeEventListener("mousedown",t.handler)}),this.modelLists.forEach(function(t){t.element.removeEventListener("wheel",h)})},pointInListRects:function(t){return this.modelLists.find(function(e){var i=e.element.getBoundingClientRect();return i={left:window.scrollX+i.left,top:window.scrollY+i.top,right:window.scrollX+i.right,bottom:window.scrollY+i.bottom,width:i.width,height:i.height},e.rect=i,s(t,i)?(e.rect=i,!0):!1})},pointInItemRects:function(t,e,i){return this.items.find(function(n){var o=n.element;return o!==e&&o!==i&&s(t,n.rect)})},getAllItemBoundingRects:function(){var t,e=this.modelLists;this.items.forEach(function(i){var n=i.element,o=n.parentElement,r=e.find(function(t){return t.element===o});if(void 0===r.isDropTarget||"function"==typeof r.isDropTarget&&r.isDropTarget()||r.isDropTarget){var s=n.getBoundingClientRect(),l=s.bottom;n===o.lastElementChild?(l=o.getBoundingClientRect().bottom,l0&&(i.element.scrollTop>0&&(s=u-(i.rect.top+5))<0?o(s,0,i.element):i.element.scrollTop0?o(s,a,i.element):o());var c=t.pointInItemRects({x:e.clientX,y:t.rect.bottom+window.scrollY+r+i.element.scrollTop},this,t.drop);if(this.style[g]=l(t.rect.left-window.scrollX+n,t.rect.top-window.scrollY+r),c){var h=c.element;h.style.transition=w,h.style.borderTopWidth=t.drop.style.borderTopWidth,t.drop.style.borderTopWidth=null,t.drop=h}}},mouseup:function(t,e){o(),t.removeEvt("mousemove"),t.removeEvt("mouseup"),e.stopPropagation();var i=this.getBoundingClientRect();if(window.scrollX+i.left===t.rect.left&&window.scrollY+i.top===t.rect.top)t.reinsert(this);else{var n=t.drop.getBoundingClientRect();t.addEvt(this,"transitionend",this),this.style.transitionDuration=w,this.style.transitionProperty=g,this.style[g]=l(n.left-window.scrollX,n.top-window.scrollY)}},transitionend:function(t,e){if(e.propertyName===g){t.removeEvt("transitionend"),t.reinsert(this),this.style.transitionProperty=w;var i=t.modelLists[t.origin.list].splice(t.origin.item,1)[0],n=t.itemCoordinates(this);t.modelLists[n.list].splice(n.item,0,i),t.callback.dropped&&t.callback.dropped.call(this,t)}}};e.exports=n},{"css-injector":4,templex:25}],19:[function(t,e,i){!function(){function t(t,e){return Object.prototype.hasOwnProperty.call(t,e)}function i(){return 1}function n(t){d||"string"==typeof t||"number"==typeof t||(d=!0,console.error(new TypeError("LRU: key must be a string or number. Almost certainly a bug! "+typeof t).stack))}function o(t){return this instanceof o?("number"==typeof t&&(t={max:t}),t||(t={}),this._max=t.max,(!this._max||"number"!=typeof this._max||this._max<=0)&&(this._max=1/0),this._lengthCalculator=t.length||i,"function"!=typeof this._lengthCalculator&&(this._lengthCalculator=i),this._allowStale=t.stale||!1,this._maxAge=t.maxAge||null,this._dispose=t.dispose,void this.reset()):new o(t)}function r(t,e,i){n(e);var o=t._cache[e];return o&&(s(t,o)?(c(t,o),t._allowStale||(o=void 0)):i&&l(t,o),o&&(o=o.value)),o}function s(t,e){if(!e||!e.maxAge&&!t._maxAge)return!1;var i=!1,n=Date.now()-e.now;return i=e.maxAge?n>e.maxAge:t._maxAge&&n>t._maxAge}function l(t,e){u(t,e),e.lu=t._mru++,t._lruList[e.lu]=e}function a(t){for(;t._lrut._max;)c(t,t._lruList[t._lru])}function u(t,e){for(delete t._lruList[e.lu];t._lru=t)&&(t=1/0),this._max=t,this._length>this._max&&a(this)},get:function(){return this._max},enumerable:!0}),Object.defineProperty(o.prototype,"lengthCalculator",{set:function(t){if("function"!=typeof t){this._lengthCalculator=i,this._length=this._itemCount;for(var e in this._cache)this._cache[e].length=1}else{this._lengthCalculator=t,this._length=0;for(var e in this._cache)this._cache[e].length=this._lengthCalculator(this._cache[e].value),this._length+=this._cache[e].length}this._length>this._max&&a(this)},get:function(){return this._lengthCalculator},enumerable:!0}),Object.defineProperty(o.prototype,"length",{get:function(){return this._length},enumerable:!0}),Object.defineProperty(o.prototype,"itemCount",{get:function(){return this._itemCount},enumerable:!0}),o.prototype.forEach=function(t,e){e=e||this;for(var i=0,n=this._itemCount,o=this._mru-1;o>=0&&n>i;o--)if(this._lruList[o]){i++;var r=this._lruList[o];s(this,r)&&(c(this,r),this._allowStale||(r=void 0)),r&&t.call(e,r.value,r.key,this)}},o.prototype.keys=function(){for(var t=new Array(this._itemCount),e=0,i=this._mru-1;i>=0&&e=0&&e=0&&ethis._max?(c(this,this._cache[e]),!1):(this._dispose&&this._dispose(e,this._cache[e].value),this._cache[e].now=r,this._cache[e].maxAge=o,this._cache[e].value=i,this._length+=s-this._cache[e].length,this._cache[e].length=s,this.get(e),this._length>this._max&&a(this),!0);var l=new h(e,i,this._mru++,s,r,o);return l.length>this._max?(this._dispose&&this._dispose(e,i),!1):(this._length+=l.length,this._lruList[l.lu]=this._cache[e]=l,this._itemCount++,this._length>this._max&&a(this),!0)},o.prototype.has=function(e){if(n(e),!t(this._cache,e))return!1;var i=this._cache[e];return s(this,i)?!1:!0},o.prototype.get=function(t){return n(t),r(this,t,!0)},o.prototype.peek=function(t){return n(t),r(this,t,!1)},o.prototype.pop=function(){var t=this._lruList[this._lru];return c(this,t),t||null},o.prototype.del=function(t){n(t),c(this,this._cache[t])},o.prototype.load=function(t){this.reset();for(var e=Date.now(),i=t.length-1;i>=0;i--){var o=t[i];n(o.k);var r=o.e||0;if(0===r)this.set(o.k,o.v);else{var s=r-e;s>0&&this.set(o.k,o.v,s)}}}}()},{}],20:[function(t,e,i){!function(t,e){"object"==typeof i&&i&&"string"!=typeof i.nodeName?e(i):"function"==typeof define&&define.amd?define(["exports"],e):(t.Mustache={},e(t.Mustache))}(this,function(t){function e(t){return"function"==typeof t}function i(t){return p(t)?"array":typeof t}function n(t){return t.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g,"\\$&")}function o(t,e){return null!=t&&"object"==typeof t&&e in t}function r(t,e){return v.call(t,e)}function s(t){return!r(m,t)}function l(t){return String(t).replace(/[&<>"'`=\/]/g,function(t){return C[t]})}function a(e,i){function o(){if(m&&!C)for(;v.length;)delete f[v.pop()];else v=[];m=!1,C=!1}function r(t){if("string"==typeof t&&(t=t.split(y,2)),!p(t)||2!==t.length)throw new Error("Invalid tags: "+t);l=new RegExp(n(t[0])+"\\s*"),a=new RegExp("\\s*"+n(t[1])),d=new RegExp("\\s*"+n("}"+t[1]))}if(!e)return[];var l,a,d,g=[],f=[],v=[],m=!1,C=!1;r(i||t.tags);for(var E,A,R,D,T,P,M=new h(e);!M.eos();){if(E=M.pos,R=M.scanUntil(l))for(var k=0,F=R.length;F>k;++k)D=R.charAt(k),s(D)?v.push(f.length):C=!0,f.push(["text",D,E,E+1]),E+=1,"\n"===D&&o();if(!M.scan(l))break;if(m=!0,A=M.scan(S)||"name",M.scan(w),"="===A?(R=M.scanUntil(b),M.scan(b),M.scanUntil(a)):"{"===A?(R=M.scanUntil(d),M.scan(x),M.scanUntil(a),A="&"):R=M.scanUntil(a),!M.scan(a))throw new Error("Unclosed tag at "+M.pos);if(T=[A,R,E,M.pos],f.push(T),"#"===A||"^"===A)g.push(T);else if("/"===A){if(P=g.pop(),!P)throw new Error('Unopened section "'+R+'" at '+E);if(P[1]!==R)throw new Error('Unclosed section "'+P[1]+'" at '+E)}else"name"===A||"{"===A||"&"===A?C=!0:"="===A&&r(R)}if(P=g.pop())throw new Error('Unclosed section "'+P[1]+'" at '+M.pos);return c(u(f))}function u(t){for(var e,i,n=[],o=0,r=t.length;r>o;++o)e=t[o],e&&("text"===e[0]&&i&&"text"===i[0]?(i[1]+=e[1],i[3]=e[3]):(n.push(e),i=e));return n}function c(t){for(var e,i,n=[],o=n,r=[],s=0,l=t.length;l>s;++s)switch(e=t[s],e[0]){case"#":case"^":o.push(e),r.push(e),o=e[4]=[];break;case"/":i=r.pop(),i[5]=e[2],o=r.length>0?r[r.length-1][4]:n;break;default:o.push(e)}return n}function h(t){this.string=t,this.tail=t,this.pos=0}function d(t,e){this.view=t,this.cache={".":this.view},this.parent=e}function g(){this.cache={}}var f=Object.prototype.toString,p=Array.isArray||function(t){return"[object Array]"===f.call(t)},v=RegExp.prototype.test,m=/\S/,C={"&":"&","<":"<",">":">",'"':""","'":"'","/":"/","`":"`","=":"="},w=/\s*/,y=/\s+/,b=/\s*=/,x=/\s*\}/,S=/#|\^|\/|>|\{|&|=|!/;h.prototype.eos=function(){return""===this.tail},h.prototype.scan=function(t){var e=this.tail.match(t);if(!e||0!==e.index)return"";var i=e[0];return this.tail=this.tail.substring(i.length),this.pos+=i.length,i},h.prototype.scanUntil=function(t){var e,i=this.tail.search(t);switch(i){case-1:e=this.tail,this.tail="";break;case 0:e="";break;default:e=this.tail.substring(0,i),this.tail=this.tail.substring(i)}return this.pos+=e.length,e},d.prototype.push=function(t){return new d(t,this)},d.prototype.lookup=function(t){var i,n=this.cache;if(n.hasOwnProperty(t))i=n[t];else{for(var r,s,l=this,a=!1;l;){if(t.indexOf(".")>0)for(i=l.view,r=t.split("."),s=0;null!=i&&sa;++a)s=void 0,o=t[a],r=o[0],"#"===r?s=this.renderSection(o,e,i,n):"^"===r?s=this.renderInverted(o,e,i,n):">"===r?s=this.renderPartial(o,e,i,n):"&"===r?s=this.unescapedValue(o,e):"name"===r?s=this.escapedValue(o,e):"text"===r&&(s=this.rawValue(o)),void 0!==s&&(l+=s);return l},g.prototype.renderSection=function(t,i,n,o){function r(t){return s.render(t,i,n)}var s=this,l="",a=i.lookup(t[1]);if(a){if(p(a))for(var u=0,c=a.length;c>u;++u)l+=this.renderTokens(t[4],i.push(a[u]),n,o);else if("object"==typeof a||"string"==typeof a||"number"==typeof a)l+=this.renderTokens(t[4],i.push(a),n,o);else if(e(a)){if("string"!=typeof o)throw new Error("Cannot use higher-order sections without the original template");a=a.call(i.view,o.slice(t[3],t[5]),r),null!=a&&(l+=a)}else l+=this.renderTokens(t[4],i,n,o);return l}},g.prototype.renderInverted=function(t,e,i,n){var o=e.lookup(t[1]);return!o||p(o)&&0===o.length?this.renderTokens(t[4],e,i,n):void 0},g.prototype.renderPartial=function(t,i,n){if(n){var o=e(n)?n(t[1]):n[t[1]];return null!=o?this.renderTokens(this.parse(o),i,n,o):void 0}},g.prototype.unescapedValue=function(t,e){var i=e.lookup(t[1]);return null!=i?i:void 0},g.prototype.escapedValue=function(e,i){var n=i.lookup(e[1]);return null!=n?t.escape(n):void 0},g.prototype.rawValue=function(t){return t[1]},t.name="mustache.js",t.version="2.2.1",t.tags=["{{","}}"];var E=new g;t.clearCache=function(){return E.clearCache()},t.parse=function(t,e){return E.parse(t,e)},t.render=function(t,e,n){if("string"!=typeof t)throw new TypeError('Invalid template! Template should be a "string" but "'+i(t)+'" was given as the first argument for mustache#render(template, view, partials)');return E.render(t,e,n)},t.to_html=function(i,n,o,r){var s=t.render(i,n,o);return e(r)?void r(s):s},t.escape=l,t.Scanner=h,t.Context=d,t.Writer=g})},{}],21:[function(t,e,i){"use strict";function n(t){return t instanceof n?t:this instanceof n?(this.originalValue=t,void(this.o=t||{})):new n(t)}n.chain=function(t){var e=n(t);return e.chaining=!0,e},n.prototype={value:function(){return this.originalValue},each:function(t,e){var i=this.o;return Object.keys(i).forEach(function(e){t.call(this,i[e],e,i)},e||i),this},find:function(t,e){var i,n=this.o;return n&&(i=Object.keys(n).find(function(e){return t.call(this,n[e],e,n)},e||n),void 0!==i&&(i=n[i])),i},filter:function(t,e){var i=this.o,n=[];return i&&Object.keys(i).forEach(function(e){t.call(this,i[e],e,i)&&n.push(i[e])},e||i),n},map:function(t,e){var i=this.o,n=[];return i&&Object.keys(i).forEach(function(e){n.push(t.call(this,i[e],e,i))},e||i),n},reduce:function(t,e,i){var n=this.o;return n&&Object.keys(n).forEach(function(i,o){e=o||void 0!==e?t(e,n[i],i,n):n[i]},i||n),e},extend:function(t){var e=this.o;return Array.prototype.slice.call(arguments).forEach(function(t){if(t)for(var i in t)e[i]=t[i]}),this.chaining?this:e},extendOwn:function(t){var e=this.o;return Array.prototype.slice.call(arguments).forEach(function(t){n(t).each(function(t,i){e[i]=t})}),this.chaining?this:e}},Array.prototype.find||(Array.prototype.find=function(t){if(null===this)throw new TypeError("Array.prototype.find called on null or undefined");if("function"!=typeof t)throw new TypeError("predicate must be a function");for(var e,i=Object(this),n=i.length>>>0,o=arguments[1],r=0;n>r;r++)if(e=i[r],t.call(o,e,r,i))return e}),e.exports=n},{}],22:[function(t,e,i){"use strict";function n(t,e){Object.defineProperty(this,t,{value:e,writable:!1,enumerable:!0,configurable:!1})}function o(t,e){n.call(this,"x",Number(t)||0),n.call(this,"y",Number(e)||0)}function r(t,e,i,r){t=Number(t)||0,e=Number(e)||0,i=Number(i)||0,r=Number(r)||0,0>i&&(t+=i,i=-i),0>r&&(e+=r,r=-r),n.call(this,"origin",new o(t,e)),n.call(this,"extent",new o(i,r)),n.call(this,"corner",new o(t+i,e+r)),n.call(this,"center",new o(t+i/2,e+r/2))}o.prototype={plus:function(t){return new o(this.x+t.x,this.y+t.y)},plusXY:function(t,e){return new o(this.x+(t||0),this.y+(e||0))},minus:function(t){return new o(this.x-t.x,this.y-t.y)},min:function(t){return new o(Math.min(this.x,t.x),Math.min(this.y,t.y))},max:function(t){return new o(Math.max(this.x,t.x),Math.max(this.y,t.y))},distance:function(t){var e=t.x-this.x,i=t.y-this.y;return Math.sqrt(e*e+i*i)},equals:function(t){var e=!1;return t&&(e=this.x===t.x&&this.y===t.y),e},greaterThan:function(t){return this.x>t.x&&this.y>t.y},lessThan:function(t){return this.x=t.x&&this.y>=t.y},lessThanOrEqualTo:function(t){return this.x<=t.x&&this.y<=t.y},within:function(t){var e=t.origin.x,i=e+t.extent.x,n=t.origin.y,o=n+t.extent.y;return t.extent.x<0&&(e=i,i=t.origin.x),t.extent.y<0&&(n=o,o=t.origin.y),e<=this.x&&this.xi;i++)for(var o=this.origin.y,r=this.corner.y;r>o;o++)t.call(e,i,o)},intersect:function(t,e,i){var n=null,o=this.origin.max(t.origin),s=this.corner.min(t.corner),l=s.minus(o);return l.x>0&&l.y>0?n=new r(o.x,o.y,l.x,l.y):"function"==typeof e&&(n=e.call(i||this,t)),n},intersects:function(t){return t.corner.x>this.origin.x&&t.corner.y>this.origin.y&&t.origin.x\|\:\[\]])/g,r=".*",s=".",l="("+r+")",a="_",u="%",c=new RegExp("("+[a,u,"\\[\\^?[^-\\]]+]","\\[\\^?[^-\\]]\\-[^\\]]]"].join("|")+")","g"); +n.reserve=function(t){return t.replace(o,"\\$1")};var h,d;(n.clearCache=function(t){return t?h[t]&&(delete h[t],d--):(h={},d=0),d})(),n.getCacheSize=function(){return d},n.cached=function(t,e,i){"string"==typeof t&&(i=e,e=t,t=!1);var o=e+(i?"i":"c"),r=h[o];if(r)r.when=(new Date).getTime(),void 0!==t&&(r.keep=t);else{if(d===n.cacheMax){var s,l,a=[],u=0;for(s in h)if(r=h[s],!r.keep){for(l=0;u>l&&!(r.when=t?[t,e]:[e,t]}function n(t,e){return t[0]<=e[0]&&e[0]<=t[1]||t[0]<=e[1]&&e[1]<=t[1]||e[0]=o&&n>r?s.push([r+1,n]):o>i&&r>=n?s.push([i,o-1]):o>i&&n>r?(s.push([i,o-1]),s.push([r+1,n])):(i>r||o>n)&&s.push(t),s}function s(t,e){var i=Math.min(Math.min.apply(Math,t),Math.min.apply(Math,e)),n=Math.max(Math.max.apply(Math,t),Math.max.apply(Math,e));return[i,n]}e.prototype={select:function(t,e){this.storeState();var r=i(t,e),l=[0,1];return this.selection.forEach(function(t){n(t,r)||o(t,r)?r=s(t,r):l.push(t)}),l.push(r),l[1]=this.selection.length,this.selection.splice.apply(this.selection,l),this},deselect:function(t,e){var o=i(t,e),s=[0,0];return this.selection.forEach(function(t){if(n(t,o)){var e=r(t,o);s=s.concat(e)}else s.push(t)}),s[1]=this.selection.length,this.selection.splice.apply(this.selection,s),this},clear:function(){return this.states.length=0,this.selection.length=0,this},clearMostRecentSelection:function(){0!==this.states.length&&(this.selection=this.states.pop())},isSelected:function(t){return this.selection.some(function(e){return e[0]<=t&&t<=e[1]})},isEmpty:function(){return 0===this.selection.length},getSelections:function(){var t=[];return this.selection.forEach(function(e){for(var i=e[0];i<=e[1];i++)t.push(i)}),t.sort(function(t,e){return t-e}),t}},t.exports=e})("object"==typeof e&&e||(window.RangeSelectionModel={}),"object"==typeof e&&e.exports||(window.RangeSelectionModel.exports={}))||"object"==typeof e||(window.RangeSelectionModel=window.RangeSelectionModel.exports)},{}],25:[function(require,module,exports){function templex(t){var e=this instanceof Array?this:[this];return arguments.length>1&&e.unshift(arguments),t.replace(templex.regexp,templex.merger.bind(e))}templex.regexp=/\{(.*?)\}/g,templex["with"]=function(t,e){return"with(this["+t+"]){"+e+"}"},templex.cache=[],templex.deref=function(key){if(!(this.length in templex.cache)){for(var code="return eval(expr)",i=0;i-1,this.selectionModel=new C(this),this.localCellEditors={},this.cellEditors=Object.create(this.localCellEditors),this.renderOverridesCache={},this.behavior=e(this),this.div.oncontextmenu=function(t){return t.preventDefault(),!1},this.clearMouseDown(),this.dragExtent=new g(0,0),this.numRows=0,this.numColumns=0,this.pluginsDo(function(t){t.installOn&&t.installOn(n)}),i=i||{},i.top=i.top||0,i.right=i.right||"-200px",i.bottom=i.bottom||0,i.left=i.left||0,S||(S=!0,s()),this.initRenderer(),this.initCanvas(i),this.initScrollbars(),this.initLocalCellEditors(),document.body.addEventListener("copy",function(t){n.checkClipboardCopy(t)}),this.getCanvas().resize(),this.dialog=new y(this),this.filter=new x}function o(t,e){return p(e).each(function(e,i){"object"==typeof t[i]&&"object"==typeof e?o(t[i],e):void 0===e?delete t[i]:t[i]=e}),t}function r(t){var e=t.origin,i=t.corner,n=Math.min(e.x,i.x),o=Math.min(e.y,i.y),r=Math.max(e.x,i.x),s=Math.max(e.y,i.y),l=new f(n,o,r-n,s-o);return l}function s(){l(E);var t=document.createElement("paper-button");t.style.display="none",t.setAttribute("disabled",!0),document.body.appendChild(t);var e=window.getComputedStyle(t),i=document.createElement("section");i.style.display="none",i.setAttribute("hero",!0),document.body.appendChild(i);var n=window.getComputedStyle(document.querySelector("html")),o=window.getComputedStyle(document.querySelector("html, body")),r=window.getComputedStyle(i);E.columnHeaderBackgroundColor=e.color,E.rowHeaderBackgroundColor=e.color,E.topLeftBackgroundColor=e.color,E.lineColor=e.backgroundColor,E.backgroundColor2=o.backgroundColor,E.color=n.color,E.fontFamily=n.fontFamily,E.backgroundColor=r.backgroundColor,t.setAttribute("disabled",!1),t.setAttribute("secondary",!0),t.setAttribute("raised",!0),e=window.getComputedStyle(t),E.columnHeaderColor=e.color,E.rowHeaderColor=e.color,E.topLeftColor=e.color,E.backgroundSelectionColor=e.backgroundColor,E.foregroundSelectionColor=e.color,t.setAttribute("secondary",!1),t.setAttribute("warning",!0),E.columnHeaderForegroundSelectionColor=e.color,E.columnHeaderBackgroundSelectionColor=e.backgroundColor,E.rowHeaderForegroundSelectionColor=e.color,E.fixedColumnBackgroundSelectionColor=e.backgroundColor,("rgba(0, 0, 0, 0)"===E.columnHeaderBackgroundSelectionColor||"transparent"===E.lineColor)&&l(E),document.body.removeChild(t),document.body.removeChild(i)}function l(t){for(var e in t)t.hasOwnProperty(e)&&delete t[e]}function a(t){var e="function"==typeof t?t():t;return e||0===e?e:""}var u=t("extend-me"),c=t("./lib/deprecated");u.debug=!0;var h=t("finbars"),d=t("fincanvas"),g=t("rectangular").Point,f=t("rectangular").Rectangle,p=t("object-iterators"),v=t("./defaults"),m=t("./lib/Renderer"),C=t("./lib/SelectionModel"),w=t("../css/stylesheets"),y=t("./lib/TableDialog"),b=t("./lib/Formatters"),x=t("./lib/CustomFilter"),S=!1,E=Object.create(v),A=Object.create(E);n.prototype={constructor:n.prototype.constructor,deprecated:c,behavior:null,isWebkit:!0,mouseDown:[],dragExtent:null,vScrollValue:0,hScrollValue:0,rectangular:null,selectionModel:null,cellEditor:null,sbHScroller:null,sbVScroller:null,sbPrevVScrollValue:null,sbPrevHScrollValue:null,cellEditors:null,renderOverridesCache:{},hoverCell:null,scrollingNow:!1,lastEdgeSelection:null,setAttribute:function(t,e){this.div.setAttribute(t,e)},reset:function(){this.lastEdgeSelection=[0,0],this.lnfProperties=Object.create(A),this.selectionModel=new C(this),this.cellEditors=Object.create(this.localCellEditors),this.renderOverridesCache={},this.clearMouseDown(),this.dragExtent=new g(0,0),this.numRows=0,this.numColumns=0,this.vScrollValue=0,this.hScrollValue=0,this.cellEditor=null,this.sbPrevVScrollValue=null,this.sbPrevHScrollValue=null,this.hoverCell=null,this.scrollingNow=!1,this.lastEdgeSelection=[0,0],this.behavior.reset(),this.getRenderer().reset(),this.getCanvas().resize(),this.behaviorChanged()},getProperties:function(){return this.getPrivateState()},_getProperties:function(){return this.lnfProperties},computeCellsBounds:function(){var t=this.getRenderer();t&&t.computeCellsBounds()},initCellEditor:function(t){this.localCellEditors[t.alias]=t,t.grid=this},initLocalCellEditors:function(){var t=["Textfield","Choice","Color","Date","Slider","Spinner","Filter"],e=this;t.forEach(function(t){e.initCellEditor(new n.cellEditors[t])}),this.localCellEditors["int"]=this.localCellEditors.spinner,this.localCellEditors["float"]=this.localCellEditors.spinner,this.localCellEditors.date=this.localCellEditors.date,this.localCellEditors.string=this.localCellEditors.extfield},toggleColumnPicker:function(){this.behavior.toggleColumnPicker()},isHovered:function(t,e){var i=this.getHoverCell();return i&&i.x===t&&i.y===e},registerFormatter:function(t,e){b[t]=e},getFormatter:function(t){var e=b[t];return e?e:b["default"]},formatValue:function(t,e){var i=this.getFormatter(t);return i(e)},isColumnHovered:function(t){var e=this.getHoverCell();return e&&e.x===t},isRowResizeable:function(){return this.resolveProperty("rowResize")},isCheckboxOnlyRowSelections:function(){return this.resolveProperty("checkboxOnlyRowSelections")},isRowHovered:function(t){var e=this.getHoverCell();return e&&e.y===t},getHoverCell:function(){return this.hoverCell},setHoverCell:function(t){var e=this.hoverCell,i=new g(t.x,t.y);e&&e.equals(i)||(this.hoverCell=i,this.fireSyntheticOnCellEnterEvent(i),this.repaint())},addGlobalProperties:function(t){if(A)this._addGlobalProperties(t);else{var e=this;setTimeout(function(){e.addGlobalProperties(t)},10)}},_addGlobalProperties:function(t){p(t).each(function(t,e){A[e]=t})},refreshProperties:function(){this.checkScrollbarVisibility(),this.behavior.defaultRowHeight=null,this.isColumnAutosizing()&&this.behavior.autosizeAllColumns()},addProperties:function(t){var e=this.getProperties();o(e,t),this.refreshProperties()},getPrivateState:function(){return this.behavior.getPrivateState()},setState:function(t){var e=this;this.behavior.setState(t),setTimeout(function(){e.behaviorChanged(),e.synchronizeScrollingBoundries()},100)},getState:function(){return this.behavior.getState()},getMouseDown:function(){var t=this.mouseDown.length-1;return 0>t?null:this.mouseDown[t]},popMouseDown:function(){0!==this.mouseDown.length&&(this.mouseDown.length=this.mouseDown.length-1)},clearMouseDown:function(){this.mouseDown=[new g(-1,-1)],this.dragExtent=null},setMouseDown:function(t){this.mouseDown.push(t)},getDragExtent:function(){return this.dragExtent},setDragExtent:function(t){this.dragExtent=t},pluginsDo:function(t){},getCellProvider:function(){var t=this.behavior.getCellProvider();return t},gridRenderedNotification:function(){this.updateRenderedSizes(),this.cellEditor&&this.cellEditor.gridRenderedNotification(),this.checkColumnAutosizing(),this.fireSyntheticGridRenderedEvent()},checkColumnAutosizing:function(){var t=this.behavior;t.autoSizeRowNumberColumn(),this.isColumnAutosizing()&&t.checkColumnAutosizing(!1)},updateRenderedSizes:function(){this.behavior.setRenderedColumnCount(this.getVisibleColumns()+1),this.behavior.setRenderedRowCount(this.getVisibleRows()+1)},checkClipboardCopy:function(t){if(this.hasFocus()){t.preventDefault();var e=this.getSelectionAsTSV();t.clipboardData.setData("text/plain",e)}},hasSelections:function(){return this.getSelectionModel?this.selectionModel.hasSelections():void 0},getSelectionAsTSV:function(){var t=this.selectionModel;if(t.hasSelections()){var e=this.getSelectionMatrix();return e=e[e.length-1],this.getMatrixSelectionAsTSV(e)}return t.hasRowSelections()?this.getMatrixSelectionAsTSV(this.getRowSelectionMatrix()):t.hasColumnSelections()?this.getMatrixSelectionAsTSV(this.getColumnSelectionMatrix()):void 0},getMatrixSelectionAsTSV:function(t){if(t.length){var e=t.length,i=t[0].length,n=e*i,o=[];if(n>2e4)return alert("selection size is too big to copy to the paste buffer"),"";for(var r=0;i>r;r++){for(var s=0;e>s;s++)o.push(t[s][r]),e>s&&o.push(" ");i>r&&o.push("\n")}var l=o.join("");return l}return""},hasFocus:function(){return this.getCanvas().hasFocus()},clearSelections:function(){var t=this.isCheckboxOnlyRowSelections();this.selectionModel.clear(t),this.clearMouseDown()},clearMostRecentSelection:function(){var t=this.isCheckboxOnlyRowSelections();this.selectionModel.clearMostRecentSelection(t)},clearMostRecentColumnSelection:function(){this.selectionModel.clearMostRecentColumnSelection()},clearMostRecentRowSelection:function(){},clearRowSelection:function(){this.selectionModel.clearRowSelection(),this.behavior.getDataModel().getComponent().clearSelectedData()},select:function(t,e,i,n){0>t||0>e||this.selectionModel.select(t,e,i,n)},isSelected:function(t,e){return this.selectionModel.isSelected(t,e)},isCellSelectedInRow:function(t){var e=this.selectionModel,i=e.isCellSelectedInRow(t);return i},isCellSelectedInColumn:function(t){var e=this.selectionModel,i=e.isCellSelectedInColumn(t);return i},getSelectionModel:function(){return this.deprecated("selectionModel",{since:"0.2"})},getBehavior:function(){return this.deprecated("behavior",{since:"0.2"})},setBehavior:function(t){this.behavior=t,this.behavior.setGrid(this),this.behavior.changed=this.behaviorChanged.bind(this),this.behavior.shapeChanged=this.behaviorShapeChanged.bind(this),this.behavior.stateChanged=this.behaviorStateChanged.bind(this)},behaviorChanged:function(){(this.numColumns!==this.getColumnCount()||this.numRows!==this.getRowCount())&&(this.numColumns=this.getColumnCount(),this.numRows=this.getRowCount(),this.behaviorShapeChanged()),this.computeCellsBounds(),this.repaint()},getBounds:function(){var t=this.getRenderer();return t&&t.getBounds()},resolveProperty:function(t){for(var e=t.split("."),i=this.getProperties();e.length;)i=i[e.shift()];return i},behaviorShapeChanged:function(){this.synchronizeScrollingBoundries()},behaviorStateChanged:function(){this.getRenderer().computeCellsBounds(),this.repaint()},repaint:function(){var t=this.resolveProperty("repaintImmediately"),e=this.getCanvas();e&&(t===!0?e.paintNow():e.repaint())},paintNow:function(){var t=this.getCanvas();t.paintNow()},useHiDPI:function(){return this.resolveProperty("useHiDPI")!==!1},initCanvas:function(t){var e=this,i=this.divCanvas=document.createElement("div");this.div.appendChild(i),this.canvas=new d(i,this.renderer);var o=i.style;o.position="absolute",o.top=t.top,o.right=t.right,o.bottom=t.bottom,o.left=t.left,this.canvas.resizeNotification=function(){e.resized()},this.addFinEventListener("fin-canvas-mousemove",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseMove(n)}}),this.addFinEventListener("fin-canvas-mousedown",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.keys=t.detail.keys,n.primitiveEvent=t,e.mouseDownState=n,e.delegateMouseDown(n),e.fireSyntheticMouseDownEvent(n),e.repaint()}}),this.addFinEventListener("fin-canvas-mouseup",function(t){if(!e.resolveProperty("readOnly")){e.dragging=!1,e.isScrollingNow()&&e.setScrollingNow(!1),e.columnDragAutoScrolling&&(e.columnDragAutoScrolling=!1);var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseUp(n),e.mouseDownState&&e.fireSyntheticButtonPressedEvent(e.mouseDownState),e.mouseDownState=null,e.fireSyntheticMouseUpEvent(n)}}),this.addFinEventListener("fin-canvas-dblclick",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.fireSyntheticDoubleClickEvent(n,t),e.delegateDoubleClick(n)}}),this.addFinEventListener("fin-canvas-tap",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,n.keys=t.detail.keys,e.fireSyntheticClickEvent(n),e.delegateTap(n)}}),this.addFinEventListener("fin-canvas-drag",function(t){if(!e.resolveProperty("readOnly")){e.dragging=!0;var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t,e.delegateMouseDrag(n)}}),this.addFinEventListener("fin-canvas-keydown",function(t){if(!e.resolveProperty("readOnly")){if(e.resolveProperty("editOnKeydown")){var i,o,r,s,l=t.detail["char"];e.isEditing()||"F2"!==l&&!(i=1===l.length)&&!(o="DELETE"===l||"BACKSPACE"===l)||(r=e.selectionModel.getLastSelection(),r&&(s=e.activateEditor(r.origin.x,r.origin.y),s instanceof n.cellEditors.Simple&&(i?s.input.value=l:o&&(s.input.value=""),t.detail.primitiveEvent.preventDefault())))}e.fireSyntheticKeydownEvent(t),e.delegateKeyDown(t)}}),this.addFinEventListener("fin-canvas-keyup",function(t){e.resolveProperty("readOnly")||(e.fireSyntheticKeyupEvent(t),e.delegateKeyUp(t))}),this.addFinEventListener("fin-canvas-track",function(t){if(!e.resolveProperty("readOnly")&&!e.dragging){var i=t.detail.primitiveEvent;Math.abs(i.dy)>Math.abs(i.dx)?i.yDirection>0?e.scrollVBy(-2):i.yDirection<-0&&e.scrollVBy(2):i.xDirection>0?e.scrollHBy(-1):i.xDirection<-0&&e.scrollHBy(1)}}),this.addFinEventListener("fin-canvas-wheelmoved",function(t){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateWheelMoved(n)}),this.addFinEventListener("fin-canvas-mouseout",function(t){if(!e.resolveProperty("readOnly")){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateMouseExit(n)}}),this.addFinEventListener("fin-canvas-context-menu",function(t){var i=t.detail.mouse,n=e.getGridCellFromMousePoint(i);n.primitiveEvent=t.detail.primitiveEvent,e.delegateContextMenu(n)}),this.div.removeAttribute("tabindex")},convertViewPointToDataPoint:function(t){return this.behavior.convertViewPointToDataPoint(t)},convertDataPointToViewPoint:function(t){return this.behavior.convertDataPointToViewPoint(t)},addFinEventListener:function(t,e){this.canvas.addEventListener(t,e)},setScrollingNow:function(t){this.scrollingNow=t},isScrollingNow:function(){return this.scrollingNow},overColumnDivider:function(t){var e=t.primitiveEvent.detail.mouse.x,i=this.getRenderer().overColumnDivider(e);return i},overRowDivider:function(t){var e=t.primitiveEvent.detail.mouse.y,i=this.getRenderer().overRowDivider(e);return i},beCursor:function(t){t||(t="default"),this.div.style.cursor=t},delegateWheelMoved:function(t){this.behavior.onWheelMoved(this,t)},delegateMouseExit:function(t){this.behavior.handleMouseExit(this,t)},delegateContextMenu:function(t){this.behavior.onContextMenu(this,t)},delegateMouseMove:function(t){this.behavior.onMouseMove(this,t)},delegateMouseDown:function(t){this.behavior.handleMouseDown(this,t)},delegateMouseUp:function(t){this.behavior.onMouseUp(this,t)},delegateTap:function(t){this.behavior.onTap(this,t)},delegateMouseDrag:function(t){this.behavior.onMouseDrag(this,t)},delegateDoubleClick:function(t){this.behavior.onDoubleClick(this,t)},delegateHoldPulse:function(t){this.behavior.onHoldPulse(this,t)},delegateKeyDown:function(t){this.behavior.onKeyDown(this,t)},delegateKeyUp:function(t){this.behavior.onKeyUp(this,t)},stopEditing:function(){this.cellEditor&&this.isEditing()&&(this.cellEditor.stopEditing&&this.cellEditor.stopEditing(),this.cellEditor=null)},registerCellEditor:function(t,e){this.cellEditors[t]=e},getDataBounds:function(){var t=200,e=this.canvas.bounds,i=new f(0,0,e.origin.x+e.extent.x-t,e.origin.y+e.extent.y);return i},getRowNumbersWidth:function(){return this.isShowRowNumbers()?this.getRenderer().getRowNumbersWidth():0},getCanvas:function(){return this.canvas},editAt:function(t,e){this.cellEditor=t;var i=e.gridCell,n=i.x,o=i.y;if(n>=0&&o>=0){var r=new g(n,o);this.setMouseDown(r),this.setDragExtent(new g(0,0)),t.beginEditAt(r)}},isColumnVisible:function(t){return this.getRenderer().isColumnVisible(t)},isDataRowVisible:function(t){return this.getRenderer().isRowVisible(t)},isDataVisible:function(t,e){return this.isDataRowVisible(e)&&this.isColumnVisible(t)},insureModelColIsVisible:function(t,e){var i=this.getColumnCount()-1,n=t+(e>0),o=!this.isColumnVisible(n)||t===i;return o&&this.scrollBy(e,0),o},insureModelRowIsVisible:function(t,e){var i=this.getRowCount()-1,n=t+(e>0),o=!this.isDataRowVisible(n)||t===i;return o&&this.scrollBy(0,e),o},scrollBy:function(t,e){this.scrollHBy(t),this.scrollVBy(e)},scrollVBy:function(t){var e=this.sbVScroller.range.max,i=this.getVScrollValue(),n=Math.min(e,Math.max(0,i+t));n!==i&&this.setVScrollValue(n)},scrollHBy:function(t){var e=this.sbHScroller.range.max,i=this.getHScrollValue(),n=Math.min(e,Math.max(0,i+t));n!==i&&this.setHScrollValue(n)},getGridCellFromMousePoint:function(t){var e=this.getRenderer().getGridCellFromMousePoint(t);return e},getBoundsOfCell:function(t){var e=this.getRenderer().getBoundsOfCell(t),i=new f(e.x,e.y,e.width,e.height);return i},resized:function(){this.synchronizeScrollingBoundries()},cellClicked:function(t){var e=t.gridCell;if(e.x<=this.getColumnCount()&&e.y<=this.getRowCount()){var i=this.getHoverCell(),n=i.x,o=i.y;o>=0&&(o+=this.getVScrollValue()),this.behavior.cellClicked(new g(n,o),t)}},setTotalsValueNotification:function(t,e,i,n){this.fireSyntheticSetTotalsValue(t,e,i,n)},fireSyntheticSetTotalsValue:function(t,e,i,n){var o=new CustomEvent("fin-set-totals-value",{detail:{x:t,y:e,value:i,area:n?"bottom":"top"}});this.canvas.dispatchEvent(o)},fireSyntheticEditorKeyUpEvent:function(t,e){var i=new CustomEvent("fin-editor-keyup",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorKeyDownEvent:function(t,e){var i=new CustomEvent("fin-editor-keydown",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorKeyPressEvent:function(t,e){var i=new CustomEvent("fin-editor-keypress",{detail:{input:t,keyEvent:e}});this.canvas.dispatchEvent(i)},fireSyntheticEditorDataChangeEvent:function(t,e,i){var n=new CustomEvent("fin-editor-data-change",{detail:{input:t,oldValue:e,newValue:i},cancelable:!0});return this.canvas.dispatchEvent(n)},fireSyntheticRowSelectionChangedEvent:function(){var t=new CustomEvent("fin-row-selection-changed",{detail:{rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(t)},fireSyntheticColumnSelectionChangedEvent:function(){var t=new CustomEvent("fin-column-selection-changed",{detail:{rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(t)},selectionChanged:function(){var t=this.getSelectedRows(),e=new CustomEvent("fin-selection-changed",{detail:{rows:t,columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(e)},getRowSelection:function(){function t(t,o){i[o]=a(n.getValue(e,t))}var e,i,n=this,o=this.selectionModel.getSelectedRows(),r=this.getColumnCount(),s={};for(e=0;r>e;e++)i=new Array(o.length),s[this.getField(e)]=i,o.forEach(t);return s},getRowSelectionMatrix:function(){function t(t,n){r[e][n]=a(i.getValue(e,t))}var e,i=this,n=this.selectionModel.getSelectedRows(),o=this.getColumnCount(),r=new Array(o);for(e=0;o>e;e++)r[e]=new Array(n.length),n.forEach(t);return r},getColumnSelectionMatrix:function(){var t=this.getSelectedColumns(),e=this.getRowCount(),i=new Array(t.length),n=this;return t.forEach(function(t,o){i[o]=new Array(e);for(var r=0;e>r;r++)i[o][r]=a(n.getValue(t,r))}),i},getColumnSelection:function(){var t=this.getSelectedColumns(),e={},i=this.getRowCount(),n=this;return t.forEach(function(t){var o=new Array(i);e[n.getField(t)]=o;for(var r=0;i>r;r++)o[r]=a(n.getValue(t,r))}),e},getSelection:function(){var t=this,e=this.getSelections(),i=new Array(e.length);return e.forEach(function(e,n){i[n]=t._getSelection(e)}),i},_getSelection:function(t){t=r(t);for(var e,i=t.extent.x+1,n=t.extent.y+1,o=t.origin.x,s=t.origin.y,l={},u=0;i>u;u++){var c=new Array(n);for(l[this.getField(u+o)]=c,e=0;n>e;e++)c[e]=a(this.getValue(o+u,s+e))}return l},getSelectionMatrix:function(){var t=this,e=this.getSelections(),i=new Array(e.length);return e.forEach(function(e,n){i[n]=t._getSelectionMatrix(e)}),i},_getSelectionMatrix:function(t){t=r(t);for(var e=t.extent.x+1,i=t.extent.y+1,n=t.origin.x,o=t.origin.y,s=[],l=0;e>l;l++){var u=new Array(i);s[l]=u;for(var c=0;i>c;c++)u[c]=a(this.getValue(n+l,o+c))}return s},fireSyntheticContextMenuEvent:function(t){t.gridCell=this.convertViewPointToDataPoint(t.gridCell);var e=new CustomEvent("fin-context-menu",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(e)},fireSyntheticMouseUpEvent:function(t){var e=new CustomEvent("fin-mouseup",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(e)},fireSyntheticMouseDownEvent:function(t){this.stopEditing();var e=new CustomEvent("fin-mousedown",{detail:{gridCell:t.gridCell,mousePoint:t.mousePoint,viewPoint:t.viewPoint,primitiveEvent:t.primitiveEvent,rows:this.getSelectedRows(),columns:this.getSelectedColumns(),selections:this.selectionModel.getSelections()}});this.canvas.dispatchEvent(e)},isViewableButton:function(t,e){return this.getRenderer().isViewableButton(t,e)},fireSyntheticButtonPressedEvent:function(t){var e=t.dataCell,i=t.gridCell;if(this.isViewableButton(e.x,e.y)){var n=new CustomEvent("fin-button-pressed",{detail:{gridCell:i}});this.canvas.dispatchEvent(n)}},fireSyntheticKeydownEvent:function(t){var e=new CustomEvent("fin-keydown",{detail:t.detail});this.canvas.dispatchEvent(e)},fireSyntheticKeyupEvent:function(t){var e=new CustomEvent("fin-keyup",{detail:t.detail});this.canvas.dispatchEvent(e)},fireSyntheticFilterAppliedEvent:function(t){var e=new CustomEvent("fin-filter-applied",{detail:t});this.canvas&&this.canvas.dispatchEvent(e)},fireSyntheticOnCellEnterEvent:function(t){var e={gridCell:t,time:Date.now(),grid:this},i=new CustomEvent("fin-cell-enter",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticGroupsChangedEvent:function(t){var e={groups:t,time:Date.now(),grid:this},i=new CustomEvent("fin-groups-changed",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticOnCellExitEvent:function(t){var e={gridCell:t,time:Date.now(),grid:this},i=new CustomEvent("fin-cell-exit",{detail:e});this.canvas.dispatchEvent(i)},fireSyntheticClickEvent:function(t){var e=t.gridCell,i={gridCell:e,mousePoint:t.mousePoint,keys:t.keys,primitiveEvent:t,time:Date.now(),grid:this};this.behavior.enhanceDoubleClickEvent(i);var n=new CustomEvent("fin-click",{detail:i});this.canvas.dispatchEvent(n)},fireSyntheticDoubleClickEvent:function(t){this.stopEditing();var e=t.gridCell,i={gridCell:e,mousePoint:t.mousePoint,time:Date.now(),grid:this};this.behavior.enhanceDoubleClickEvent(t);var n=new CustomEvent("fin-double-click",{detail:i});this.behavior.cellDoubleClicked(e,t),this.canvas.dispatchEvent(n)},fireSyntheticGridRenderedEvent:function(){var t=new CustomEvent("fin-grid-rendered",{detail:{source:this,time:Date.now()}});this.canvas&&this.canvas.dispatchEvent(t)},fireScrollEvent:function(t,e,i){var n=new CustomEvent(t,{detail:{oldValue:e,value:i,time:Date.now()}});this.canvas.dispatchEvent(n)},setVScrollValue:function(t){t=Math.round(t);var e=this.sbVScroller.range.max;t=Math.min(e,Math.max(0,t));var i=this;if(t!==this.vScrollValue){this.behavior._setScrollPositionY(t);var n=this.vScrollValue;this.vScrollValue=t,this.scrollValueChangedNotification(),setTimeout(function(){i.fireScrollEvent("fin-scroll-y",n,t)})}},getVScrollValue:function(){return this.vScrollValue},setHScrollValue:function(t){t=Math.round(t);var e=this.sbHScroller.range.max;t=Math.min(e,Math.max(0,t));var i=this;if(t!==this.hScrollValue){this.behavior._setScrollPositionX(t);var n=this.hScrollValue;this.hScrollValue=t,this.scrollValueChangedNotification(),setTimeout(function(){i.fireScrollEvent("fin-scroll-x",n,t),i.synchronizeScrollingBoundries()})}},getHScrollValue:function(){return this.hScrollValue},takeFocus:function(){this.isEditing()?this.stopEditing():this.getCanvas().takeFocus()},editorTakeFocus:function(){return this.cellEditor?this.cellEditor.takeFocus():void 0},isEditing:function(){return this.cellEditor&&this.cellEditor.isEditing},initScrollbars:function(){var t=this,e=new h({orientation:"horizontal",onchange:t.setHScrollValue.bind(t),cssStylesheetReferenceElement:this.div}),i=new h({orientation:"vertical",onchange:t.setVScrollValue.bind(t),paging:{up:t.pageUp.bind(t),down:t.pageDown.bind(t)}});this.sbHScroller=e,this.sbVScroller=i;var n=this.resolveProperty("hScrollbarClassPrefix"),o=this.resolveProperty("vScrollbarClassPrefix");n&&""!==n&&(this.sbHScroller.classPrefix=n),o&&""!==o&&(this.sbVScroller.classPrefix=o),this.div.appendChild(e.bar),this.div.appendChild(i.bar),this.resizeScrollbars()},resizeScrollbars:function(){this.sbHScroller.shortenBy(this.sbVScroller).resize(),this.sbVScroller.resize()},setVScrollbarValues:function(t){this.sbVScroller.range={min:0,max:t}},setHScrollbarValues:function(t){this.sbHScroller.range={min:0,max:t}},scrollValueChangedNotification:function(){(this.hScrollValue!==this.sbPrevHScrollValue||this.vScrollValue!==this.sbPrevVScrollValue)&&(this.sbPrevHScrollValue=this.hScrollValue,this.sbPrevVScrollValue=this.vScrollValue,this.cellEditor&&this.cellEditor.scrollValueChangedNotification(),this.computeCellsBounds())},getValue:function(t,e){return this.behavior.getValue(t,e)},setValue:function(t,e,i){this.behavior.setValue(t,e,i)},getColumnAlignment:function(t){return this.behavior.getColumnAlignment(t)},synchronizeScrollingBoundries:function(){var t=this.getFixedColumnCount(),e=this.getFixedRowCount(),i=this.getColumnCount(),n=this.getRowCount(),o=this.getBounds();if(o){for(var r=o.height-this.behavior.getFixedRowsMaxHeight()-15,s=o.width-200-this.behavior.getFixedColumnsMaxWidth()-15,l=0,a=0;i>l;l++){var u=this.getColumnWidth(i-l-1);if(a+=u,a>s)break}for(var c=0,h=0;n>c;c++){var d=this.getRowHeight(n-c-1);if(h+=d,h>r)break}var g=Math.max(0,i-t-l);this.setHScrollbarValues(g);var f=1+Math.max(0,n-e-c);this.setVScrollbarValues(f),this.setHScrollValue(Math.min(this.getHScrollValue(),g)),this.setVScrollValue(Math.min(this.getVScrollValue(),f)),this.computeCellsBounds(),this.repaint(),this.resizeScrollbars()}},getVisibleRows:function(){return this.getRenderer().getVisibleRows()},getVisibleColumns:function(){return this.getRenderer().getVisibleColumns()},initRenderer:function(){this.renderer=new m(this)},getRenderer:function(){return this.renderer},getColumnWidth:function(t){return this.behavior.getColumnWidth(t)},setColumnWidth:function(t,e){this.stopEditing(),this.behavior.setColumnWidth(t,e)},getColumnEdge:function(t){return this.behavior.getColumnEdge(t,this.getRenderer())},getFixedColumnsWidth:function(){return this.behavior.getFixedColumnsWidth()},getRowHeight:function(t){return this.behavior.getRowHeight(t)},setRowHeight:function(t,e){this.stopEditing(),this.behavior.setRowHeight(t,e)},getFixedRowsHeight:function(){return this.behavior.getFixedRowsHeight()},getColumnCount:function(){return this.behavior.getColumnCount()},getRowCount:function(){return this.behavior.getRowCount()},getUnfilteredRowCount:function(){return this.behavior.getUnfilteredRowCount()},getFixedColumnCount:function(){return this.behavior.getFixedColumnCount()},getFixedRowCount:function(){return this.behavior.getFixedRowCount()},topLeftClicked:function(t){this.behavior.topLeftClicked(this,t)},rowHeaderClicked:function(t){this.behavior.rowHeaderClicked(this,t)},columnHeaderClicked:function(t){this.behavior.columnHeaderClicked(this,t)},_activateEditor:function(t){var e=t.gridCell;this.activateEditor(e.x,e.y)},activateEditor:function(t,e){var i;if((this.isEditable()||this.isFilterRow(e))&&(i=this.getCellEditorAt(t,e))){var n=i.getEditorPoint();if(i){if(n.x===t&&n.y===e&&i.isEditing)return;this.isEditing()&&this.stopEditing(),event.gridCell={x:t,y:e},this.editAt(i,event)}}return i; +},getCellEditorAt:function(t,e){return this.behavior._getCellEditorAt(t,e)},toggleHiDPI:function(){this.useHiDPI()?this.removeAttribute("hidpi"):this.setAttribute("hidpi",null),this.canvas.resize()},getHiDPI:function(t){if(window.devicePixelRatio&&this.useHiDPI()){var e=window.devicePixelRatio||1,i=t.webkitBackingStorePixelRatio||t.mozBackingStorePixelRatio||t.msBackingStorePixelRatio||t.oBackingStorePixelRatio||t.backingStorePixelRatio||1,n=e/i;return n}return 1},getRenderedWidth:function(t){return this.renderer.getRenderedWidth(t)},getRenderedHeight:function(t){return this.renderer.getRenderedHeight(t)},resolveCellEditor:function(t){return this.cellEditors[t]},updateCursor:function(){var t=this.behavior.getCursorAt(-1,-1),e=this.getHoverCell();if(e&&e.x>-1&&e.y>-1){var i=e.x+this.getHScrollValue();t=this.behavior.getCursorAt(i,e.y+this.getVScrollValue())}this.beCursor(t)},repaintCell:function(t,e){this.getRenderer().repaintCell(t,e)},isDraggingColumn:function(){return!!this.renderOverridesCache.dragger},pageUp:function(){var t=this.getRenderer().getPageUpRow();return this.setVScrollValue(t),t},pageDown:function(){var t=this.getRenderer().getPageDownRow();return this.setVScrollValue(t),t},pageLeft:function(){console.log("page left")},pageRight:function(){console.log("page right")},getRenderedData:function(){var t,e=this.behavior,i=this.getRenderer(),n=this.getColumnCount(),o=i.getVisibleRows(),r=new Array(n),s=new Array(o);return r.forEach(function(t,i){r[i]=e.getColumnId(i,0)}),s.forEach(function(i,n){t=s[n]={hierarchy:e.getFixedColumnValue(0,n)},r.forEach(function(i,o){t[i]=e.getValue(o,n)})}),s},getSelectedRow:function(){var t=this.selectionModel.getSelections();if(t.length){for(var e=this.behavior,i=this.getColumnCount(),n=t[0].origin.y,o={},r=0;i>r;r++)o[e.getColumnId(r,0)]=e.getValue(r,n);return o}},fireRequestCellEdit:function(t,e){var i=new CustomEvent("fin-request-cell-edit",{cancelable:!0,detail:{value:e,gridCell:t,time:Date.now()}});return this.canvas.dispatchEvent(i)},fireBeforeCellEdit:function(t,e,i,n){var o=new CustomEvent("fin-before-cell-edit",{cancelable:!0,detail:{oldValue:e,newValue:i,gridCell:t,time:Date.now(),input:n,row:this.getRow(t.y)}}),r=this.canvas.dispatchEvent(o);return r},fireAfterCellEdit:function(t,e,i,n){var o=new CustomEvent("fin-after-cell-edit",{detail:{newValue:i,oldValue:e,gridCell:t,time:Date.now(),input:n,row:this.getRow(t.y)}});this.canvas.dispatchEvent(o)},autosizeColumn:function(t){var e=this.behavior.getColumn(t);e.checkColumnAutosizing(!0),this.computeCellsBounds()},setFocusable:function(t){this.getCanvas().setFocusable(t)},getVisibleColumnsCount:function(){return this.getRenderer().getVisibleColumnsCount()},getVisibleRowsCount:function(){return this.getRenderer().getVisibleRowsCount()},updateSize:function(){this.canvas.checksize()},stopPaintThread:function(){this.canvas.stopPaintThread()},stopResizeThread:function(){this.canvas.stopResizeThread()},restartResizeThread:function(){this.canvas.restartResizeThread()},restartPaintThread:function(){this.canvas.restartPaintThread()},swapColumns:function(t,e){this.behavior.swapColumns(t,e)},endDragColumnNotification:function(){this.behavior.endDragColumnNotification()},getFixedColumnsMaxWidth:function(){return this.behavior.getFixedColumnsMaxWidth()},isMouseDownInHeaderArea:function(){var t=this.getHeaderColumnCount(),e=this.getHeaderRowCount(),i=this.getMouseDown();return i.x-1,r=e.indexOf("SHIFT")>-1;o||r?(o&&(n?i.deselectColumn(t):i.selectColumn(t)),r&&(i.clear(),i.selectColumn(this.lastEdgeSelection[0],t))):(i.clear(),n||i.selectColumn(t)),n||r||(this.lastEdgeSelection[0]=t),this.repaint(),this.fireSyntheticColumnSelectionChangedEvent()},toggleSelectRow:function(t,e){if(t>this.getFilterRowIndex()){e=e||[];var i=this.selectionModel,n=i.isRowSelected(t),o=e.indexOf("SHIFT")>=0;n?i.deselectRow(t):(this.singleSelect(),i.selectRow(t)),o&&(i.clear(),i.selectRow(this.lastEdgeSelection[1],t)),n||o||(this.lastEdgeSelection[1]=t),this.repaint()}},singleSelect:function(){var t=this.isCheckboxOnlyRowSelections(),e=this.isSingleRowSelectionMode(),i=this.mouseDownState.primitiveEvent.detail.primitiveEvent.ctrlKey,n=t&&e||!t&&(!i||e);return n&&this.selectionModel.clearRowSelection(),n},selectViewportCell:function(t,e){var i=this.getHeaderRowCount(),n=this.getRenderer(),o=n.getVisibleColumns()[t],r=n.getVisibleRows()[e];this.clearSelections(),this.select(o,r+i,0,0),this.setMouseDown(this.newPoint(o,r+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToViewportCell:function(t,e){var i=this.getSelections();if(i&&i.length){var n=this.getHeaderRowCount(),o=this.getRenderer(),r=o.getVisibleColumns()[t],s=o.getVisibleRows()[e]+n,l=i[0],a=l.origin;this.setDragExtent(this.newPoint(r-a.x,s-a.y)),this.select(a.x,a.y,r-a.x,s-a.y),this.repaint()}},selectFinalCellOfCurrentRow:function(){var t=this.getColumnCount()-1,e=this.getSelectedRows()[0],i=this.getHeaderRowCount();this.clearSelections(),this.scrollBy(this.getColumnCount(),0),this.select(t,e+i,0,0),this.setMouseDown(this.newPoint(t,e+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToFinalCellOfCurrentRow:function(){var t=this.getSelections();if(t&&t.length){var e=t[0],i=e.origin,n=e.extent,o=this.getColumnCount();this.scrollBy(o,0),this.clearSelections(),this.select(i.x,i.y,o-i.x-1,n.y),this.repaint()}},selectFirstCellOfCurrentRow:function(){var t=0,e=this.getSelectedRows()[0],i=this.getHeaderRowCount();this.clearSelections(),this.setHScrollValue(0),this.select(t,e+i,0,0),this.setMouseDown(this.newPoint(t,e+i)),this.setDragExtent(this.newPoint(0,0)),this.repaint()},selectToFirstCellOfCurrentRow:function(){var t=this.getSelections();if(t&&t.length){var e=t[0],i=e.origin,n=e.extent;this.clearSelections(),this.select(i.x,i.y,-i.x,n.y),this.setHScrollValue(0),this.repaint()}},selectFinalCell:function(){this.selectCell(this.getColumnCount()-1,this.getRowCount()-1),this.scrollBy(this.getColumnCount(),this.getRowCount()),this.repaint()},selectToFinalCell:function(){var t=this.getSelections();if(t&&t.length){var e=t[0],i=e.origin,n=this.getColumnCount(),o=this.getRowCount();this.clearSelections(),this.select(i.x,i.y,n-i.x-1,o-i.y-1),this.scrollBy(n,o),this.repaint()}},isShowRowNumbers:function(){return this.resolveProperty("showRowNumbers")},isEditable:function(){return this.resolveProperty("editable")===!0},isShowFilterRow:function(){return this.resolveProperty("showFilterRow")},isShowHeaderRow:function(){return this.resolveProperty("showHeaderRow")},getHeaderRowCount:function(){return this.behavior.getHeaderRowCount()},isFilterRow:function(t){return t===this.getFilterRowIndex()},getFilterRowIndex:function(){return this.isShowFilterRow()?this.isShowHeaderRow()?1:0:-1},setGroups:function(t){this.behavior.setGroups(t)},filterClicked:function(t){this.activateEditor(t.gridCell.x,t.gridCell.y)},hasHierarchyColumn:function(){return this.behavior.hasHierarchyColumn()},isHierarchyColumn:function(t){return this.hasHierarchyColumn()&&0===t},checkScrollbarVisibility:function(){},isColumnOrRowSelected:function(){return this.selectionModel.isColumnOrRowSelected()},selectColumn:function(t,e){this.selectionModel.selectColumn(t,e)},selectRow:function(t,e){var i=this.selectionModel,n=this.getFilterRowIndex()+1;this.singleSelect()?t=e:e=e||t;var o=Math.min(t,e);if(o>=n){var r=Math.max(t,e);i.selectRow(o,r)}},isRowNumberAutosizing:function(){return this.resolveProperty("rowNumberAutosizing")},isRowSelected:function(t){return this.selectionModel.isRowSelected(t)},isColumnSelected:function(t){return this.selectionModel.isColumnSelected(t)},lookupFeature:function(t){return this.behavior.lookupFeature(t)},getRow:function(t){return this.behavior.getRow(t)},getFieldName:function(t){return this.behavior.getFieldName(t)},getColumnIndex:function(t){return this.behavior.getColumnIndex(t)},isCellSelection:function(){return this.resolveProperty("cellSelection")===!0},isRowSelection:function(){return this.resolveProperty("rowSelection")===!0},isColumnSelection:function(){return this.resolveProperty("columnSelection")===!0},getComputedRow:function(t){return this.behavior.getComputedRow(t)},isColumnAutosizing:function(){return this.resolveProperty("columnAutosizing")===!0},setGlobalFilter:function(t){this.behavior.setGlobalFilter(t)},selectRowsFromCells:function(){if(!this.isCheckboxOnlyRowSelections()){var t,e=this.mouseDownState.primitiveEvent.detail.primitiveEvent.ctrlKey;e&&!this.isSingleRowSelectionMode()?this.selectionModel.selectRowsFromCells(0,e):(t=this.selectionModel.getLastSelection())?this.selectRow(null,t.corner.y):this.clearRowSelection()}},selectColumnsFromCells:function(){this.selectionModel.selectColumnsFromCells()},getSelectedRows:function(){return this.behavior.getSelectedRows()},getSelectedColumns:function(){return this.behavior.getSelectedColumns()},getSelections:function(){return this.behavior.getSelections()},getLastSelectionType:function(){return this.selectionModel.getLastSelectionType()},isCellSelected:function(t,e){return this.selectionModel.isCellSelected(t,e)},isInCurrentSelectionRectangle:function(t,e){return this.selectionModel.isInCurrentSelectionRectangle(t,e)},selectAllRows:function(){this.selectionModel.selectAllRows()},areAllRowsSelected:function(){return this.selectionModel.areAllRowsSelected()},toggleSelectAllRows:function(){this.areAllRowsSelected()?this.selectionModel.clear():this.selectAllRows(),this.repaint()},getField:function(t){return this.behavior.getField(t)},isSingleRowSelectionMode:function(){return this.resolveProperty("singleRowSelectionMode")},newPoint:function(t,e){return new g(t,e)},newRectangle:function(t,e,i,n){return new f(t,e,i,n)},getFormattedValue:function(t,e){e+=this.getHeaderRowCount();var i=this.getColumnProperties(t).format,n=this.getValue(t,e),o=this.getFormatter(i),r=o(n);return r}},e.exports=n},{"../css/stylesheets":1,"./defaults":47,"./lib/CustomFilter":68,"./lib/Formatters":69,"./lib/Renderer":71,"./lib/SelectionModel":72,"./lib/TableDialog":73,"./lib/deprecated":74,"extend-me":5,finbars:13,fincanvas:14,"object-iterators":21,rectangular:22}],28:[function(t,e,i){"use strict";var n=t("object-iterators"),o=t("../lib/Base"),r=t("./Column"),s=t("../lib/CellProvider"),l=["columnHeader","columnHeaderColumnSelection","filterProperties","rowHeader","rowHeaderRowSelection","rowNumbersProperties","treeColumnProperties","treeColumnPropertiesColumnSelection"],a={isNull:!0},u=o.extend("Behavior",{initialize:function(t){t.setBehavior(this),this.initializeFeatureChain(t),this.getDataModel(),this.cellProvider=this.createCellProvider(),this.renderedColumnCount=30,this.renderedRowCount=60,this.dataUpdates={}},initializeFeatureChain:function(t){var e=this;this.features.forEach(function(t){e.setNextFeature(new t)}),this.featureChain.initializeOn(t)},features:[],tableState:null,grid:null,editorTypes:["choice","textfield","color","slider","spinner","date"],featureChain:null,dataModel:null,baseModel:null,scrollPositionX:0,scrollPositionY:0,featureMap:{},allColumns:[],columns:[],reset:function(){this.cellProvider=this.createCellProvider(),this.renderedColumnCount=30,this.renderedRowCount=60,this.dataUpdates={},this.clearColumns(),this.clearState(),this.getDataModel().reset(),this.createColumns()},clearColumns:function(){this.columns=[],this.allColumns=[],this.columns[-1]=this.newColumn(-1,""),this.columns[-2]=this.newColumn(-2,"Tree"),this.allColumns[-1]=this.columns[-1],this.allColumns[-2]=this.columns[-2]},getColumn:function(t){return this.columns[t]},getColumnId:function(t){return this.getColumn(t).label},newColumn:function(t,e){var i=this.createColumnProperties();return this.getPrivateState().columnProperties[t]=i,new r(this,t,e)},addColumn:function(t,e){var i=this.newColumn(t,e);return this.columns.push(i),this.allColumns.push(i),i},createColumns:function(){},createColumnProperties:function(){var t=this.getPrivateState(),e=Object.create(t);return e.rowNumbersProperties=Object.create(e,{foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundSelectionColor},set:function(t){this.columnHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundSelectionColor},set:function(t){this.columnHeaderBackgroundSelectionColor=t}}}),e.rowHeader=Object.create(e,{font:{configurable:!0,get:function(){return this.rowHeaderFont},set:function(t){this.rowHeaderFont=t}},color:{configurable:!0,get:function(){return this.rowHeaderColor},set:function(t){this.rowHeaderColor=t}},backgroundColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundColor},set:function(t){this.rowHeaderBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderForegroundSelectionColor},set:function(t){this.rowHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundSelectionColor},set:function(t){this.rowHeaderBackgroundSelectionColor=t}}}),e.columnHeader=Object.create(e,{format:{value:"default"},font:{configurable:!0,get:function(){return this.columnHeaderFont},set:function(t){this.columnHeaderFont=t}},color:{configurable:!0,get:function(){return this.columnHeaderColor},set:function(t){this.columnHeaderColor=t}},backgroundColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundColor},set:function(t){this.columnHeaderBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundSelectionColor},set:function(t){this.columnHeaderForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundSelectionColor},set:function(t){this.columnHeaderBackgroundSelectionColor=t}}}),e.columnHeaderColumnSelection=Object.create(e.columnHeader,{foregroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderForegroundColumnSelectionColor},set:function(t){this.columnHeaderForegroundColumnSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.columnHeaderBackgroundColumnSelectionColor},set:function(t){this.columnHeaderBackgroundColumnSelectionColor=t}}}),e.rowHeaderRowSelection=Object.create(e.rowHeader,{foregroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderForegroundRowSelectionColor},set:function(t){this.rowHeaderForegroundRowSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.rowHeaderBackgroundRowSelectionColor},set:function(t){this.rowHeaderBackgroundRowSelectionColor=t}}}),e.filterProperties=Object.create(e,{font:{configurable:!0,get:function(){return this.filterFont},set:function(t){this.filterFont=t}},color:{configurable:!0,get:function(){return this.filterColor},set:function(t){this.filterColor=t}},backgroundColor:{configurable:!0,get:function(){return this.filterBackgroundColor},set:function(t){this.filterBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.filterForegroundSelectionColor},set:function(t){this.filterForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.filterBackgroundSelectionColor},set:function(t){this.filterBackgroundSelectionColor=t}},cellBorderStyle:{configurable:!0,get:function(){return this.filterCellBorderStyle},set:function(t){this.filterCellBorderStyle=t}},cellBorderThickness:{configurable:!0,get:function(){return this.filterCellBorderThickness},set:function(t){this.filterCellBorderThickness=t}}}),e.treeColumnProperties=Object.create(e,{font:{configurable:!0,get:function(){return this.treeColumnFont},set:function(t){this.treeColumnFont=t}},color:{configurable:!0,get:function(){return this.treeColumnColor},set:function(t){this.treeColumnColor=t}},backgroundColor:{configurable:!0,get:function(){return this.treeColumnBackgroundColor},set:function(t){this.treeColumnBackgroundColor=t}},foregroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnForegroundSelectionColor},set:function(t){this.treeColumnForegroundSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnBackgroundSelectionColor},set:function(t){this.treeColumnBackgroundSelectionColor=t}}}),e.treeColumnPropertiesColumnSelection=Object.create(e.treeColumnProperties,{foregroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnForegroundColumnSelectionColor},set:function(t){this.treeColumnForegroundColumnSelectionColor=t}},backgroundSelectionColor:{configurable:!0,get:function(){return this.treeColumnBackgroundColumnSelectionColor},set:function(t){this.treeColumnBackgroundColumnSelectionColor=t}}}),e},getColumnWidth:function(t){var e=this.getColumn(t);if(!e)return this.resolveProperty("defaultColumnWidth");var i=e.getWidth();return i},setColumnWidth:function(t,e){this.getColumn(t).setWidth(e),this.stateChanged()},getDataModel:function(){if(null===this.dataModel){var t=this.getDefaultDataModel();this.setDataModel(t)}return this.dataModel},getCellRenderer:function(t,e,i){return this.getColumn(e).getCellRenderer(t,i)},setDataModel:function(t){this.dataModel=t},setComplexFilter:function(t,e){var i=this.getColumn(t);i&&i.setComplexFilter(e)},getComplexFilter:function(t){var e=this.getColumn(t);return e&&e.getComplexFilter()},applyAnalytics:function(){},clearObjectProperties:function(t,e){for(var i in t)t.hasOwnProperty(i)&&(void 0===e||!e&&l.indexOf(i)>=0||e&&l.indexOf(i)<0)&&delete t[i]},getPrivateState:function(){return this.tableState||(this.tableState=this.getDefaultState()),this.tableState},getState:function(){var t=JSON.parse(JSON.stringify(this.getPrivateState()));return this.clearObjectProperties(t.columnProperties,!1),t},clearState:function(){this.tableState=null},getDefaultState:function(){var t=this.grid._getProperties(),e=Object.create(t);return n(e).extendOwn({rowHeights:{},cellProperties:{},columnProperties:[]}),e},setState:function(t){if(!t.columnIndexes){var e=this.getFields();t.columnIndexes=[];for(var i=0;ii;i++)e+=this.getRowHeight(i);return e},getRowHeight:function(t){var e=this.getPrivateState().rowHeights;return e&&e[t]||this.getDefaultRowHeight()},getDefaultRowHeight:function(){return this.defaultRowHeight||(this.defaultRowHeight=this.resolveProperty("defaultRowHeight")),this.defaultRowHeight},setRowHeight:function(t,e){var i=this.getPrivateState();i.rowHeights[t]=Math.max(5,e),this.stateChanged()},getFixedRowsMaxHeight:function(){return this.getFixedRowsHeight()},getFixedColumnsWidth:function(){var t=this.getFixedColumnCount(),e=0;this.grid.isShowRowNumbers()&&(e=this.getColumnWidth(-1));for(var i=0;t>i;i++)e+=this.getColumnWidth(i);return e},getFixedColumnsMaxWidth:function(){return this.getFixedColumnsWidth()},_setScrollPositionY:function(t){this.setScrollPositionY(t),this.changed()},_setScrollPositionX:function(t){this.setScrollPositionX(t),this.changed()},setRenderedColumnCount:function(t){this.renderedColumnCount=t},setRenderedRowCount:function(t){this.renderedRowCount=t},_fixedRowClicked:function(t,e){var i=this.translateColumnIndex(this.getScrollPositionX()+e.gridCell.x-this.getFixedColumnCount()),n=this.grid.newPoint(i,e.gridCell.y);e.gridCell=n,this.fixedRowClicked(t,e)},_fixedColumnClicked:function(t,e){var i=this.grid.newPoint(e.gridCell.x,this.getScrollPositionY()+e.gridCell.y-this.getFixedRowCount());e.gridCell=i,this.fixedColumnClicked(t,e)},moveSingleSelect:function(t,e,i){this.featureChain&&(this.featureChain.moveSingleSelect(t,e,i),this.setCursor(t))},setCursor:function(t){t.updateCursor(),this.featureChain.setCursor(t)},onMouseMove:function(t,e){this.featureChain&&(this.featureChain.handleMouseMove(t,e),this.setCursor(t))},onTap:function(t,e){this.featureChain&&(this.featureChain.handleTap(t,e),this.setCursor(t))},onContextMenu:function(t,e){var i=t.fireSyntheticContextMenuEvent(e);i&&this.featureChain&&(this.featureChain.handleContextMenu(t,e),this.setCursor(t))},onWheelMoved:function(t,e){this.featureChain&&(this.featureChain.handleWheelMoved(t,e),this.setCursor(t))},onMouseUp:function(t,e){this.featureChain&&(this.featureChain.handleMouseUp(t,e),this.setCursor(t))},onMouseDrag:function(t,e){this.featureChain&&(this.featureChain.handleMouseDrag(t,e),this.setCursor(t))},onKeyDown:function(t,e){this.featureChain&&(this.featureChain.handleKeyDown(t,e),this.setCursor(t))},onKeyUp:function(t,e){this.featureChain&&(this.featureChain.handleKeyUp(t,e),this.setCursor(t))},onDoubleClick:function(t,e){this.featureChain&&(this.featureChain.handleDoubleClick(t,e),this.setCursor(t))},onHoldPulse:function(t,e){this.featureChain&&(this.featureChain.handleHoldPulse(t,e),this.setCursor(t))},toggleColumnPicker:function(){var t=this.grid.dialog,e=this;t.isOpen()?t.close():(this.buildColumnPicker(t.overlay),t.onClose=function(){e.updateFromColumnPicker(t.overlay)},t.open())},handleMouseDown:function(t,e){this.featureChain&&(this.featureChain.handleMouseDown(t,e),this.setCursor(t))},handleMouseExit:function(t,e){this.featureChain&&(this.featureChain.handleMouseExit(t,e),this.setCursor(t))},changed:function(){},shapeChanged:function(){},isColumnReorderable:function(){return!0},getColumnProperties:function(t){var e=this.columns[t];if(!e)return a;var i=e.getProperties();return i?i:a},setColumnProperties:function(t,e){var i=this.allColumns[t].getProperties();n(i).extendOwn(e),this.changed()},getField:function(t){return-1===t?"tree":this.getColumn(t).getField()},getHeader:function(t){return-1===t?"Tree":this.getColumn(t).getHeader()},setColumnDescriptors:function(t){var e,i=t.visible,n=this.getPrivateState(),o=i.length,r=[];for(e=0;o>e;e++)r.push(i[e].id);n.columnIndexes=r,this.changed()},getHiddenColumnDescriptors:function(){for(var t=this.getPrivateState(),e=t.columnIndexes,i=[],n=this.getColumnCount(),o=0;n>o;o++)-1===e.indexOf(o)&&i.push({id:o,label:this.getHeader(o),field:this.getField(o)});return i},hideColumns:function(t){for(var e=this.getPrivateState(),i=e.columnIndexes,n=0;ns;s++)if(i=this.getValue(s),n=this.typeOf(i),o!==n){if(!r||"number"!=typeof i)return"mixed";o="float"}return o},typeOf:function(t){var e=typeof t;switch(e){case"object":return t.constructor.name.toLowerCase();case"number":return parseInt(t)===t?"int":"float";default:return e}},getProperties:function(){return this.behavior.getPrivateState().columnProperties[this.index]},setProperties:function(t){var e=this.behavior.getPrivateState().columnProperties[this.index];this.clearObjectProperties(e,!1),o(e).extendOwn(t)},toggleSort:function(t){this.dataModel.toggleSort(this.index,t)},getCellEditorAt:function(t,e){return this.dataModel.getCellEditorAt(this.index,e)},getHeader:function(){return this.label},getField:function(){ +return this.dataModel.getFields()[this.index]}},e.exports=n},{"object-iterators":21}],30:[function(t,e,i){"use strict";function n(t,e){this.setComponent(e),this.setGrid(t)}var o=t("../lib/deprecated");n.prototype={constructor:n.prototype.constructor,deprecated:o,component:null,grid:null,getGrid:function(){return this.deprecated("grid",{since:"0.2"})},setGrid:function(t){this.grid=t,this.getComponent().setGrid(t)},getBehavior:function(){return this.deprecated("grid.behavior",{since:"0.2"})},changed:function(){this.grid.behavior.changed()},getPrivateState:function(){return this.grid.getPrivateState()},applyState:function(){},setComponent:function(t){this.component=t},getComponent:function(){return this.component},setGlobalFilter:function(t){return this.getComponent().setGlobalFilter(t)},getData:function(){return this.getComponent().getData()},getFilteredData:function(){return this.getComponent().getFilteredData()},getValue:function(t,e){return this.getComponent().getValue(t,e)},getUnfilteredValue:function(t,e){return this.getComponent().getUnfilteredValue(t,e)},setValue:function(t,e,i){this.getComponent().setValue(t,e,i)},getColumnCount:function(){return this.getComponent().getColumnCount()},applyAnalytics:function(){return this.getComponent().applyAnalytics()},getRowCount:function(){return this.getComponent().getRowCount()},getCellRenderer:function(t,e,i,n,o){return this.getComponent().getCellRenderer(t,e,i,n,o)},getRowHeight:function(t){return this.getComponent().getRowHeight(t)},getColumnEdge:function(t,e){return this.getComponent().getColumnEdge(t,e)},getColumnWidth:function(t){return this.getComponent().getColumnWidth(t)},setColumnWidth:function(t,e){this.getComponent().setColumnWidth(t,e)},toggleSort:function(t,e){this.getComponent().toggleSort(t,e)},getColumnProperties:function(t){return this.getComponent().getColumnProperties(t)},setColumnProperties:function(t,e){this.getComponent().setColumnProperties(t,e)},getHeaders:function(){return this.getComponent().getHeaders()},getFields:function(){return this.getComponent().getFields()},setFields:function(t){this.getComponent().setFields(t)},getCellProperties:function(t,e){return this.getComponent().getCellProperties(t,e)},setCellProperties:function(t,e,i){this.getComponent().setCellProperties(t,e,i)},getRow:function(t){return this.getComponent().getRow(t)},getTopTotals:function(){return this.getComponent().getTopTotals()},setTopTotals:function(t){this.getComponent().setTopTotals(t)},getBottomTotals:function(){return this.getComponent().getBottomTotals()},setBottomTotals:function(t){this.getComponent().setBottomTotals(t)},setData:function(t){return this.getComponent().setData(t)},hasHierarchyColumn:function(){return this.getComponent().hasHierarchyColumn()},setHeaders:function(t){return this.getComponent().setHeaders(t)},cellClicked:function(t,e){return this.getComponent().cellClicked(t,e)},getAvailableGroups:function(){return this.getComponent().getAvailableGroups()},getGroups:function(){return this.getComponent().getGroups()},setGroups:function(t){this.getComponent().setGroups(t)},getHiddenColumns:function(){return this.getComponent().getHiddenColumns()},getVisibleColumns:function(){return this.getComponent().getVisibleColumns()},setAggregates:function(t){return this.getComponent().setAggregates(t)},reset:function(){this.getComponent().reset()},getCellEditorAt:function(t,e){return this.getComponent().getCellEditorAt(t,e)},getUnfilteredRowCount:function(){return this.getComponent().getUnfilteredRowCount()}},e.exports=n},{"../lib/deprecated":74}],31:[function(t,e,i){"use strict";var n=t("list-dragon"),o=t("./Local"),r=t("./DataModelDecorator"),s=t("../dataModels/JSON"),l=t("../features/index"),a=t("../../css/stylesheets"),u=t("../local_node_modules/finanalytics").aggregations,c=o.extend("behaviors.JSON",{initialize:function(t,e){this.setData(e)},features:[l.CellSelection,l.KeyPaging,l.ColumnPicker,l.ColumnResizing,l.RowResizing,l.Filters,l.RowSelection,l.ColumnSelection,l.ColumnMoving,l.ColumnSorting,l.CellEditing,l.CellClick,l.OnHover],aggregations:u,createColumns:function(){var t=this.getDataModel(),e=t.getColumnCount(),i=t.getHeaders(),n=t.getFields();this.clearColumns();for(var o=0;e>o;o++){var r=i[o],s=this.addColumn(o,r),l=s.getProperties();l.field=n[o],l.header=r,l.complexFilter=null}},getDefaultDataModel:function(){var t=new s,e=new r(this.grid,t);return e.setComponent(t),e},applyAnalytics:function(){this.dataModel.applyAnalytics()},setHeaders:function(t){this.getDataModel().setHeaders(t)},getHeaders:function(){return this.getDataModel().getHeaders()},setFields:function(t){this.getDataModel().setFields(t),this.createColumns()},getFields:function(){return this.getDataModel().getFields()},setData:function(t){this.getDataModel().setData(t),this.createColumns();var e=this;this.grid.isColumnAutosizing()?(setTimeout(function(){e.autosizeAllColumns()},100),e.changed()):setTimeout(function(){e.allColumns[-1].checkColumnAutosizing(!0),e.changed()})},setTopTotals:function(t){this.getDataModel().setTopTotals(t)},getTopTotals:function(){return this.getDataModel().getTopTotals()},setBottomTotals:function(t){this.getDataModel().setBottomTotals(t)},getBottomTotals:function(){return this.getDataModel().getBottomTotals()},setColumns:function(t){this.getDataModel().setColumns(t)},enhanceDoubleClickEvent:function(t){t.row=this.getRow(t.gridCell.y)},setDataProvider:function(t){this.getDataModel().setDataProvider(t)},hasHierarchyColumn:function(){return this.getDataModel().hasHierarchyColumn()},getColumnAlignment:function(t){return 0===t&&this.hasHierarchyColumn()?"left":"center"},getRowSelectionMatrix:function(t){return this.getDataModel().getRowSelectionMatrix(t)},getColumnSelectionMatrix:function(t){return this.getDataModel().getColumnSelectionMatrix(t)},getSelectionMatrix:function(t){return this.getDataModel().getSelectionMatrix(t)},getRowSelection:function(){var t=this.getSelectedRows();return this.getDataModel().getRowSelection(t)},getColumnSelection:function(){var t=this.getSelectedColumns();return this.getDataModel().getColumnSelection(t)},getSelection:function(){var t=this.getSelections();return this.getDataModel().getSelection(t)},buildColumnPicker:function(t){if(!this.isColumnReorderable())return!1;var e={cssStylesheetReferenceElement:t},i={models:this.getGroups(),title:"Groups"},o={models:this.getAvailableGroups(),title:"Available Groups"},r={models:this.getHiddenColumns(),title:"Hidden Columns"},s={models:this.getVisibleColumns(),title:"Visible Columns"},l=new n([i,o],e),u=new n([r,s],e),c=[l,u];return a("list-dragon",t),c.forEach(function(e){e.modelLists.forEach(function(e){t.appendChild(e.container)})}),t.lists={group:i.models,availableGroups:o.models,hidden:r.models,visible:s.models},!0},getGroups:function(){return this.getDataModel().getGroups()},getAvailableGroups:function(){return this.getDataModel().getAvailableGroups()},getHiddenColumns:function(){return this.getDataModel().getHiddenColumns()},getVisibleColumns:function(){return this.getDataModel().getVisibleColumns()},setColumnDescriptors:function(t){var e=this.columns[0];this.columns.length=0,e&&"Tree"===e.label&&this.columns.push(e);for(var i=0;ia;a++){var u=t.getUnfilteredValue(s,a);l.set(u,u)}var c=l.values;c.sort(),c.length>0&&c[0].length>0&&c.unshift(""),this.setItems(c)}},showEditor:function(){var t=this;this.input.style.display="inline",setTimeout(function(){t.showDropdown(t.input)},50)},preShowEditorNotification:function(){this.autopopulate(),this.setEditorValue(this.initialValue)},setItems:function(t){this.items=t,this.updateView()},initializeInput:function(t){var e=this;n.prototype.initializeInput.apply(this,[t]),t.onchange=function(){e.stopEditing()}}});e.exports=r},{"../lib/Mappy":70,"./Simple":40}],37:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Color",{alias:"color",template:function(){}});e.exports=o},{"./Simple":40}],38:[function(t,e,i){"use strict";function n(t){var e=t.match(/(\d+)/g);return new window.Date(e[0],e[1]-1,e[2])}var o=t("./Simple"),r=t("../lib/Formatters"),s=o.extend("Date",{alias:"date",template:function(){},setEditorValue:function(t){null!=t&&"Date"===t.constructor.name&&(t=r.date(t)),this.getInput().value=t+""},getEditorValue:function(){var t=this.getInput().value;return t=n(t)}});e.exports=s},{"../lib/Formatters":69,"./Simple":40}],39:[function(t,e,i){"use strict";var n=t("./CellEditor"),o=n.extend("Filter",{alias:"filter",initialize:function(){var t=document.createElement("div"),e=t.style;e.position="absolute",e.top=e.bottom="44px",e.right=e.left="1em",e.overflowY="scroll";var i=document.createElement("table");t.appendChild(i),e=i.style,e.width=e.height="100%";var n=document.createElement("tr"),o=document.createElement("td");i.appendChild(n),n.appendChild(o),this.title=document.createElement("div"),this.title.innerHTML="Filter Editor",this.dialog=document.createElement("div"),this.content=o,this.buttons=document.createElement("div"),e=this.dialog.style,e.position="absolute",e.top=e.left=e.right=e.bottom=0,e.whiteSpace="nowrap",e=this.title.style,e.position="absolute",e.top=e.left=e.right=0,e.height="44px",e.whiteSpace="nowrap",e.textAlign="center",e.padding="11px",e=this.buttons.style,e.position="absolute",e.left=e.right=e.bottom=0,e.height="44px",e.whiteSpace="nowrap",e.textAlign="center",e.padding="8px",this.dialog.appendChild(this.title),this.dialog.appendChild(t),this.dialog.appendChild(this.buttons),this.ok=document.createElement("button"),this.ok.style.borderRadius="8px",this.ok.style.width="5.5em",this.cancel=document.createElement("button"),this.cancel.style.marginLeft="2em",this.cancel.style.borderRadius="8px",this.cancel.style.width="5.5em",this["delete"]=document.createElement("button"),this["delete"].style.marginLeft="2em",this["delete"].style.borderRadius="8px",this["delete"].style.width="5.5em",this.reset=document.createElement("button"),this.reset.style.marginLeft="2em",this.reset.style.borderRadius="8px",this.reset.style.width="5.5em",this.ok.innerHTML="ok",this.cancel.innerHTML="cancel",this["delete"].innerHTML="delete",this.reset.innerHTML="reset",this.buttons.appendChild(this.ok),this.buttons.appendChild(this.reset),this.buttons.appendChild(this["delete"]),this.buttons.appendChild(this.cancel);var r=this;this.ok.onclick=function(){r.okPressed()},this.cancel.onclick=function(){r.cancelPressed()},this["delete"].onclick=function(){r.deletePressed()},this.reset.onclick=function(){r.resetPressed()}},tearDown:function(){this.content.innerHTML=""},okPressed:function(){var t=this.grid.dialog;t.onOkPressed()},cancelPressed:function(){var t=this.grid.dialog;t.onCancelPressed()},deletePressed:function(){var t=this.grid.dialog;t.onDeletePressed()},resetPressed:function(){var t=this.grid.dialog;t.onResetPressed()},beginEditAt:function(t){var e=this.grid.behavior,i=this.grid.dialog,n=t.x,o=e.getColumnId(n),r=e.getField(n),s=e.getColumn(n).getType(),l=function(){return[{name:r,alias:o,type:s}]};this.title.innerHTML="Manage Filters";var a=this.grid.filter;if(i.isOpen())i.close();else{var u=this;i.clear(),i.overlay.appendChild(this.dialog),a.initialize(l),i.onOkPressed=function(){a.onOk&&a.onOk()||(u.tearDown(),e.setComplexFilter(n,{state:a.getState()}),i.close(),e.applyAnalytics(),e.changed())},i.onCancelPressed=function(){a.onCancel&&a.onCancel()||(u.tearDown(),i.close(),a=void 0)},i.onDeletePressed=function(){a.onDelete&&a.onDelete()||(u.tearDown(),e.setComplexFilter(n,void 0),i.close(),e.applyAnalytics(),e.changed())},i.onResetPressed=function(){a.onReset&&a.onReset()||(u.tearDown(),a.initialize(i),a.onShow&&a.onShow(u.content))};var c=this.grid._getBoundsOfCell(n,t.y),h=this.grid.div.getBoundingClientRect().left-this.grid.divCanvas.getBoundingClientRect().left;c.x=c.x-h,i.openFrom(c);var d=e.getComplexFilter(n);d&&a.setState(d.state),setTimeout(function(){a.onShow&&a.onShow(u.content)},i.getAnimationTime()+10)}}});e.exports=o},{"./CellEditor":35}],40:[function(t,e,i){"use strict";var n=t("./CellEditor.js"),o=n.extend("Simple",{input:null,alias:"simple",initialize:function(){this.editorPoint={x:0,y:0}},specialKeyups:{9:"stopEditing",13:"stopEditing",27:"cancelEditing"},keyup:function(t){if(t){var e=this.specialKeyups[t.keyCode];e&&(t.preventDefault(),this[e](),this.grid.repaint(),this.grid.takeFocus()),this.grid.fireSyntheticEditorKeyUpEvent(this,t)}},initializeInput:function(t){var e=this;t.addEventListener("keyup",this.keyup.bind(this)),t.addEventListener("keydown",function(t){e.grid.fireSyntheticEditorKeyDownEvent(e,t)}),t.addEventListener("keypress",function(t){e.grid.fireSyntheticEditorKeyPressEvent(e,t)}),t.onblur=function(t){e.cancelEditing()},t.style.position="absolute",t.style.display="none",t.style.border="solid 2px black",t.style.outline=0,t.style.padding=0,t.style.boxShadow="white 0px 0px 1px 1px"},getEditorValue:function(){var t=this.getInput().value;return t},setEditorValue:function(t){this.getInput().value=t},clearStopEditing:function(){this.setEditorValue(""),this.stopEditing()},cancelEditing:function(){this.isEditing&&(this.getInput().value=null,this.isEditing=!1,this.hideEditor())},showEditor:function(){this.getInput().style.display="inline"},hideEditor:function(){this.getInput().style.display="none"},takeFocus:function(){var t=this;setTimeout(function(){var e=t.input.style.transform;t.input.style.transform="translate(0,0)",t.input.focus(),t.selectAll(),t.input.style.transform=e})},selectAll:function(){},originOffset:function(){return[0,0]},setBounds:function(t){var e=this.originOffset(),i="translate("+(t.x-1+e[0])+"px,"+(t.y-1+e[1])+"px)",n=this.getInput();n.style.boxSizing="border-box",n.style.webkitTransform=i,n.style.MozTransform=i,n.style.msTransform=i,n.style.OTransform=i,n.style.width=t.width+2+"px",n.style.height=t.height+2+"px"},saveEditorValue:function(){var t=this.getEditorPoint(),e=this.getEditorValue();if(e!==this.initialValue){parseFloat(this.initialValue)===this.initialValue&&(e=parseFloat(e));var i=this.grid.fireBeforeCellEdit(t,this.initialValue,e,this);i&&(this.grid.behavior.setValue(t.x,t.y,e),this.grid.fireAfterCellEdit(t,this.initialValue,e,this))}},_moveEditor:function(){var t=this.getEditorPoint(),e=this.grid._getBoundsOfCell(t.x,t.y),i=this.grid.div.getBoundingClientRect().left-this.grid.divCanvas.getBoundingClientRect().left;e.x=e.x-i,this.setBounds(e)},moveEditor:function(){this._moveEditor(),this.takeFocus()},beginEditAt:function(t){this.isAdded||(this.isAdded=!0,this.attachEditor()),this.setEditorPoint(t);var e=this.grid.behavior.getValue(t.x,t.y);"Array"===e.constructor.name&&(e=e[1]);var i=this.grid.fireRequestCellEdit(t,e);i&&(this.initialValue=e,this.isEditing=!0,this.setCheckEditorPositionFlag(),this.checkEditor())},checkEditor:function(){if(this.checkEditorPositionFlag&&(this.checkEditorPositionFlag=!1,this.isEditing)){var t=this.getEditorPoint();this.grid.isDataVisible(t.x,t.y)?(this.preShowEditorNotification(),this.attachEditor(),this.moveEditor(),this.showEditor()):this.hideEditor()}},attachEditor:function(){var t=this.getInput(),e=this.grid.div,i=e.querySelectorAll(".finbar-horizontal, .finbar-vertical");e.insertBefore(t,i.length?i[0]:null)},preShowEditorNotification:function(){this.setEditorValue(this.initialValue)},getInput:function(){return this.input||(this.input=this.getDefaultInput()),this.input},getDefaultInput:function(){var t=document.createElement("DIV");t.innerHTML=this.getHTML();var e=t.firstChild;return this.initializeInput(e),e},updateView:function(){var t=this.getInput(),e=t.parentNode,i=this.getDefaultInput();this.input=i,e.replaceChild(i,t)},showDropdown:function(t){var e;e=document.createEvent("MouseEvents"),e.initMouseEvent("mousedown",!0,!0,window),t.dispatchEvent(e)}});e.exports=o},{"./CellEditor.js":35}],41:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Slider",{alias:"slider",template:function(){}});e.exports=o},{"./Simple":40}],42:[function(t,e,i){"use strict";var n=t("./Simple"),o=n.extend("Spinner",{alias:"spinner",template:function(){}});e.exports=o},{"./Simple":40}],43:[function(t,e,i){"use strict";function n(){this.saveEditorValue(),this._moveEditor()}var o=t("./Simple"),r=o.extend("Textfield",{alias:"textfield",template:function(){},selectAll:function(){this.input.setSelectionRange(0,this.input.value.length)},specialKeyups:{9:"stopEditing",13:"stopEditing",27:"cancelEditing"},keyup:function(t){t&&(o.prototype.keyup.call(this,t),this.grid.isFilterRow(this.getEditorPoint().y)&&setTimeout(n.bind(this)))}});e.exports=r},{"./Simple":40}],44:[function(t,e,i){"use strict";e.exports={CellEditor:t("./CellEditor"),Textfield:t("./Textfield"),Choice:t("./Choice"),Color:t("./Color"),Date:t("./Date"),Simple:t("./Simple"),Slider:t("./Slider"),Spinner:t("./Spinner"),Filter:t("./Filter")}},{"./CellEditor":35,"./Choice":36,"./Color":37,"./Date":38,"./Filter":39,"./Simple":40,"./Slider":41,"./Spinner":42,"./Textfield":43}],45:[function(t,e,i){"use strict";var n=t("../lib/Base"),o="A".charCodeAt(0),r=n.extend("DataModel",{next:null,grid:null,setGrid:function(t){this.grid=t},getGrid:function(){return this.deprecated("grid",{since:"0.2"})},getBehavior:function(){return this.deprecated("grid.behavior",{since:"0.2"})},changed:function(){this.grid.behavior.changed()},getPrivateState:function(){return this.grid.getPrivateState()},applyState:function(){},alphaFor:function(t){var e=Math.floor(t/26),i=t%26,n="";return e>0&&(n+=this.alpha(e-1)),n+=this.alpha(i)},alpha:function(t){return String.fromCharCode(o+t)},getCellEditorAt:function(t,e){}});e.exports=r},{"../lib/Base":66}],46:[function(t,e,i){"use strict";function n(t){return"function"==typeof t?t():t}function o(t){return t=t.toLowerCase(),function(e){return e=n(e),(e+"").toLowerCase().indexOf(t)>-1}}function r(){if(this.grid.selectionModel.hasRowSelections()){var t=this.getFilteredData(),e=this.grid.getSelectedRows(),i=this.selectedData;i.forEach(function(e,n){t.indexOf(e)>0&&delete i[n]}),e.forEach(function(e){var n=t[e];i.indexOf(n)<0&&i.push(n)})}}function s(){if(this.selectedData.length){var t=this.grid.selectionModel,e=this.grid.getHeaderRowCount(),i=this.getFilteredData();t.clearRowSelection(),this.selectedData.forEach(function(n){var o=i.indexOf(n);o>=0&&t.selectRow(e+o)})}}function l(){0===this.analytics.aggregates.length&&this.quietlySetAggregates({}),this.analytics.apply()}function a(){var t=this.getVisibleColumns();this.getGlobalFilterSource().apply(t);var e=[],i=this.getFilterSource(),n=0;i.clearAll(),t.forEach(function(t){var r=t.index,s=this.getFilter(r),l=t.getProperties().format,a=this.grid.getFormatter(l),u=this.getComplexFilter(r),c=u||s.length>0&&o(s);c&&(i.add(r-n,this.createFormattedFilter(a,c)),e.push({column:t.label,format:u?"complex":l}))}.bind(this)),i.applyAll(),this.grid.fireSyntheticFilterAppliedEvent({details:e})}function u(){var t=this.getSortingSource(),e=this.getPrivateState().sorts,i=this.hasAggregates()?1:0;if(e&&0!==e.length)for(var n=0;nn;n++)i[n]=t.getRow(n);return i},getValue:function(t,e){var i,n=this.hasHierarchyColumn(),o=this.grid.getHeaderRowCount();return n?-2===t&&(t=0):this.hasAggregates()&&(t+=1),i=o>e?this.getHeaderRowValue(t,e):this.getDataSource().getValue(t,e-o)},getHeaderRowValue:function(t,e){var i;if(void 0===e)i=this.getHeaders()[Math.max(t,0)];else if(0>e){var n=this.getBottomTotals();i=n[n.length+e][t]}else{var o=this.grid.isShowFilterRow(),r=this.grid.isShowHeaderRow(),s=(o?1:0)+(r?1:0);if(e>=s)i=this.getTopTotals()[e-s][t];else{if(!r||0!==e){i=this.getFilter(t);var l=d.filter(i.length);return[null,i,l]}i=this.getHeaders()[t];var a=this.getSortImageForColumn(t);a&&(i=a+i)}}return i},setValue:function(t,e,i){var n=this.hasHierarchyColumn(),o=this.grid.getHeaderRowCount();n?-2===t&&(t=0):this.hasAggregates()&&(t+=1),o>e?this.setHeaderRowValue(t,e,i):this.getDataSource().setValue(t,e-o,i),this.changed()},setHeaderRowValue:function(t,e,i){if(void 0===i)return this._setHeader(t,e);var n=this.grid.isShowFilterRow(),o=this.grid.isShowHeaderRow(),r=n&&o,s=(n?1:0)+(o?1:0);if(e>=s)this.getTopTotals()[e-s][t]=i;else{if(-1===t)return;if(r){if(0===e)return this._setHeader(t,i);this.setFilter(t,i)}else{if(!n)return this._setHeader(t,i);this.setFilter(t,i)}}return""},getColumnProperties:function(t){var e=this.grid.behavior.allColumns[t];return e?e.getProperties():void 0},getFilter:function(t){var e,i;return(i=this.getColumnProperties(t))&&(e=i.filter),e||""},getComplexFilter:function(t){var e,i,o,r,s;return(i=this.getColumnProperties(t))&&(o=i.complexFilter)&&(r=this.grid.filter)&&(s=r.create(o.state))&&(e=function(t){var e=n(t);return s(e)}),e},setFilter:function(t,e){var i=this.getColumnProperties(t);i.filter=e,this.applyAnalytics()},getColumnCount:function(){var t=this.grid.resolveProperty("showTreeColumn")===!0,e=this.hasAggregates(),i=e&&!t?-1:0;return this.analytics.getColumnCount()+i},getRowCount:function(){var t=this.getDataSource().getRowCount();return t+=this.grid.getHeaderRowCount()},getHeaders:function(){return this.analytics.getHeaders()},setHeaders:function(t){this.getDataSource().setHeaders(t)},setFields:function(t){this.getDataSource().setFields(t)},getFields:function(){return this.getDataSource().getFields()},setData:function(t){this.source=new c.JSDataSource(t),this.analytics=new c.DataSourceAggregator(this.source),this.postglobalfilter=new c.DataSourceGlobalFilter(this.analytics),this.postfilter=new c.DataSourceFilter(this.postglobalfilter),this.postsorter=new c.DataSourceSorterComposite(this.postfilter),this.applyAnalytics()},setTopTotals:function(t){this.topTotals=t},getTopTotals:function(){return this.hasAggregates()?this.getDataSource().getGrandTotals():this.topTotals},setBottomTotals:function(t){this.bottomTotals=t},getBottomTotals:function(){return this.hasAggregates()?this.getDataSource().getGrandTotals():this.bottomTotals},setGroups:function(t){this.analytics.setGroupBys(t),this.applyAnalytics(),this.grid.fireSyntheticGroupsChangedEvent(this.getGroups())},getGroups:function(){for(var t=this.getHeaders().slice(0),e=this.getFields().slice(0),i=this.analytics.groupBys,n=[],o=0;o-1;i.sorts=i.sorts||[];var o=i.sorts.indexOf(t);-1===o&&(o=i.sorts.indexOf(-1*t)),o>-1?i.sorts[o]>0?i.sorts[o]=-1*i.sorts[o]:i.sorts.splice(o,1):n||0===i.sorts.length?i.sorts.unshift(t):(i.sorts.length=0,i.sorts.unshift(t)),i.sorts.length>3&&(i.sorts.length=3)},getSortImageForColumn:function(t){t++;var e=!0,i=this.getPrivateState().sorts;if(!i)return null;var n=i.indexOf(t);if(0>n&&(n=i.indexOf(-1*t),e=!1),0>n)return null;var o=i.length-n,r=e?g:f;return o+r+" "},cellClicked:function(t,e){if(this.hasAggregates()&&0===e.gridCell.x){var i=this.grid.getHeaderRowCount(),n=e.gridCell.y-i;this.getDataSource().click(n),this.applyAnalytics(!0),this.changed()}},getRow:function(t){var e=this.grid.getHeaderRowCount();if(e>t&&!this.hasAggregates()){var i=this.getTopTotals();return i[t-(e-i.length)]}return this.getDataSource().getRow(t-e)},buildRow:function(t){var e=this.getColumnCount(),i=[].concat(this.getFields()),n={};this.hasAggregates()&&(n.tree=this.getValue(-2,t),i.shift());for(var o=0;e>o;o++)n[i[o]]=this.getValue(o,t);return n},getComputedRow:function(t){for(var e=this.getRowContextFunction([t]),i=this.getFields(),n={},o=0;o=t.behavior.getHeaderRowCount()&&e.gridCell.x>=t.behavior.getHeaderColumnCount()?t.cellClicked(e):this.next&&this.next.handleTap(t,e)}});e.exports=o},{"./Feature.js":58}],50:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("CellEditing",{alias:"CellEditing",handleDoubleClick:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,i)?t._activateEditor(e):this.next&&this.next.handleDoubleClick(t,e)},handleTap:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,!i)?t._activateEditor(e):this.next&&this.next.handleTap(t,e)},handleHoldPulse:function(t,e){var i=t.resolveProperty("editOnDoubleClick");this.checkActivateEditor(t,e,!i)?t._activateEditor(e):this.next&&this.next.handleHoldPulse(t,e)},checkActivateEditor:function(t,e,i){var n=t.behavior.getHeaderRowCount(),o=t.behavior.getHeaderColumnCount(),r=e.gridCell,s=t.isFilterRow(r.y),l=i&&r.x>=o&&(s||r.y>=n);return l}});e.exports=o},{"./Feature.js":58}],51:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("CellSelection",{alias:"CellSelection",currentDrag:null,lastDragCell:null,sbLastAuto:0,sbAutoStart:0,handleMouseUp:function(t,e){this.dragging&&(this.dragging=!1),this.next&&this.next.handleMouseUp(t,e)},handleMouseDown:function(t,e){var i=e.primitiveEvent.detail.isRightClick,n=e.gridCell,o=e.viewPoint,r=n.x,s=n.y,l=t.behavior.getHeaderRowCount(),a=t.behavior.getHeaderColumnCount(),u=t.behavior.getColumnCount(),c=o.x>=u,h=l>s||a>r;if(!t.isCellSelection()||i||h||c)this.next&&this.next.handleMouseDown(t,e);else{var d=t.getFixedColumnCount(),g=t.getFixedRowCount();o.xn.origin.x+n.extent.x&&(o=1),this.currentDrag.y>n.origin.y+n.extent.y&&(r=1);var c=o,h=r;a&&(c=0),u&&(h=0),this.lastDragCell=i.plusXY(c,h),t.scrollBy(o,r),this.handleMouseDragCellSelection(t,i,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){var n=i.indexOf("CTRL")>=0,o=i.indexOf("SHIFT")>=0,r=t.getMouseDown(),s=e.x,l=e.y;if(!(0>s||0>l)){if(n&&s===r.x&&l===r.y)return t.clearMostRecentSelection(),t.popMouseDown(),void t.repaint();n||o||t.clearSelections(),o?(t.clearMostRecentSelection(),t.select(r.x,r.y,s-r.x+1,l-r.y+1),t.setDragExtent(t.newPoint(s-r.x+1,l-r.y))):(t.select(s,l,0,0),t.setMouseDown(t.newPoint(s,l)),t.setDragExtent(t.newPoint(0,0))),t.repaint()}},handleDOWNSHIFT:function(t){this.moveShiftSelect(t,0,1)},handleUPSHIFT:function(t){this.moveShiftSelect(t,0,-1)},handleLEFTSHIFT:function(t){this.moveShiftSelect(t,-1,0)},handleRIGHTSHIFT:function(t){this.moveShiftSelect(t,1,0)},handleDOWN:function(t,e){e.primitiveEvent.preventDefault();var i=this.getAutoScrollAcceleration();this.moveSingleSelect(t,0,i)},handleUP:function(t,e){e.primitiveEvent.preventDefault();var i=this.getAutoScrollAcceleration();this.moveSingleSelect(t,0,-i)},handleLEFT:function(t){this.moveSingleSelect(t,-1,0)},handleRIGHT:function(t){this.moveSingleSelect(t,1,0)},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now()},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e,i){var n=t.getColumnCount()-1,o=t.getRowCount()-1,r=t.getVisibleColumns()-1,s=t.getVisibleRows()-1;t.resolveProperty("scrollingEnabled")||(n=Math.min(n,r),o=Math.min(o,s));var l=t.getMouseDown(),a=t.getDragExtent(),u=a.x+e,c=a.y+i;u=Math.min(n-l.x,Math.max(-l.x,u)),c=Math.min(o-l.y,Math.max(-l.y,c)),t.clearMostRecentSelection(),t.select(l.x,l.y,u,c),t.setDragExtent(t.newPoint(u,c)),t.insureModelColIsVisible(u+l.x,e)&&this.pingAutoScroll(),t.insureModelRowIsVisible(c+l.y,i)&&this.pingAutoScroll(),t.repaint()},moveSingleSelect:function(t,e,i){var n=t.getColumnCount()-1,o=t.getRowCount()-1,r=t.getVisibleColumnsCount()-1,s=t.getVisibleRowsCount()-1,l=t.getHeaderRowCount(),a=t.getHeaderColumnCount();t.resolveProperty("scrollingEnabled")||(n=Math.min(n,r),o=Math.min(o,s));var u=t.getMouseDown().plus(t.getDragExtent()),c=u.x+e,h=u.y+i;c=Math.min(n,Math.max(a,c)),h=Math.min(o,Math.max(l,h)),t.clearSelections(),t.select(c,h,0,0),t.setMouseDown(t.newPoint(c,h)),t.setDragExtent(t.newPoint(0,0)),t.insureModelColIsVisible(c,e)&&this.pingAutoScroll(),t.insureModelRowIsVisible(h,i)&&this.pingAutoScroll(),t.repaint()}});e.exports=o},{"./Feature.js":58}],52:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnAutosizing",{alias:"ColumnAutosizing",handleDoubleClick:function(t,e){var i=t.getHeaderRowCount(),n=e.gridCell;n.y<=i?t.autosizeColumn(n.x):this.next&&this.next.handleDoubleClick(t,e)}});e.exports=o},{"./Feature.js":58}],53:[function(t,e,i){"use strict";var n,o,r,s,l=t("./Feature.js"),a=150,u=l.extend("ColumnMoving",{alias:"ColumnMoving",floaterAnimationQueue:[],columnDragAutoScrollingRight:!1,columnDragAutoScrollingLeft:!1,dragArmed:!1,dragging:!1,dragCol:-1,dragOffset:0,initializeOn:function(t){this.isFloatingNow=!1,this.initializeAnimationSupport(t),this.next&&this.next.initializeOn(t)},initializeAnimationSupport:function(t){n||(n=document.createElement("canvas"),n.setAttribute("width","0px"),n.setAttribute("height","0px"),document.body.appendChild(n),o=n.getContext("2d")),r||(r=document.createElement("canvas"),r.setAttribute("width","0px"),r.setAttribute("height","0px"),document.body.appendChild(r),s=r.getContext("2d"))},getCanDragCursorName:function(){return"-webkit-grab"},getDraggingCursorName:function(){return"-webkit-grabbing"},handleMouseDrag:function(t,e){var i,n=e.gridCell,o=Math.abs(e.primitiveEvent.detail.dragstart.x-e.primitiveEvent.detail.mouse.x);return 10>o?void(this.next&&this.next.handleMouseDrag(t,e)):(this.isHeaderRow(t,e)&&this.dragArmed&&!this.dragging?(this.dragging=!0,this.dragCol=n.x,this.dragOffset=e.mousePoint.x,this.detachChain(),i=e.primitiveEvent.detail.mouse.x-this.dragOffset,this.createDragColumn(t,i,this.dragCol)):this.next&&this.next.handleMouseDrag(t,e),void(this.dragging&&(i=e.primitiveEvent.detail.mouse.x-this.dragOffset,this.dragColumn(t,i))))},handleMouseDown:function(t,e){t.behavior.isColumnReorderable()&&this.isHeaderRow(t,e)&&-1!==e.gridCell.x&&(this.dragArmed=!0,this.cursor=this.getDraggingCursorName(),t.clearSelections()),this.next&&this.next.handleMouseDown(t,e)},handleMouseUp:function(t,e){if(this.dragging){this.cursor=null;var i=this;this.endDragColumn(t),setTimeout(function(){i.attachChain()},200)}this.dragCol=-1,this.dragging=!1,this.dragArmed=!1,this.cursor=null,t.repaint(),this.next&&this.next.handleMouseUp(t,e)},handleMouseMove:function(t,e){!this.dragging&&e.mousePoint.y<5&&0===e.viewPoint.y?this.cursor=this.getCanDragCursorName():this.cursor=null,this.next&&this.next.handleMouseMove(t,e),this.isHeaderRow(t,e)&&this.dragging&&(this.cursor=this.getDraggingCursorName())},floatColumnTo:function(t,e){this.floatingNow=!0;var i,n,o=t.getRenderer(),r=o.getColumnEdges(),s=t.getHScrollValue(),l=t.renderOverridesCache.floater.columnIndex,a=t.renderOverridesCache.dragger.columnIndex,u=t.renderOverridesCache.dragger.hdpiratio,c=t.getFixedColumnCount(),h=t.getColumnWidth(a),d=t.getColumnWidth(l),g=t.getVisibleColumnsCount(),f=0,p=0;a>=c&&(f=s),l>=c&&(p=s),e?(i=r[Math.min(g,a-f)],n=r[Math.min(g,l-p)],t.renderOverridesCache.dragger.startX=(i+d)*u,t.renderOverridesCache.floater.startX=i*u):(n=r[Math.min(g,l-p)],i=n+h,t.renderOverridesCache.dragger.startX=n*u,t.renderOverridesCache.floater.startX=i*u),t.swapColumns(a,l),t.renderOverridesCache.dragger.columnIndex=l,t.renderOverridesCache.floater.columnIndex=a,this.floaterAnimationQueue.unshift(this.doColumnMoveAnimation(t,n,i)),this.doFloaterAnimation(t)},doColumnMoveAnimation:function(t,e,i){var n=this;return function(){var o=r;o.style.display="inline",n.setCrossBrowserProperty(o,"transform","translate("+e+"px, 0px)"),requestAnimationFrame(function(){n.setCrossBrowserProperty(o,"transition",(n.isWebkit?"-webkit-":"")+"transform "+a+"ms ease"),n.setCrossBrowserProperty(o,"transform","translate("+i+"px, -2px)")}),t.repaint(),setTimeout(function(){n.setCrossBrowserProperty(o,"transition",""),t.renderOverridesCache.floater=null,t.repaint(),n.doFloaterAnimation(t),requestAnimationFrame(function(){o.style.display="none",n.isFloatingNow=!1})},a+50)}},doFloaterAnimation:function(t){if(0===this.floaterAnimationQueue.length)return this.floatingNow=!1,void t.repaint();var e=this.floaterAnimationQueue.pop();e()},createFloatColumn:function(t,e){var i=t.getFixedColumnCount(),n=t.getHScrollValue();i>e&&(n=0);var o=t.getRenderer(),l=o.getColumnEdges(),a=t.getColumnWidth(e),u=t.div.clientHeight,c=r,h=c.style,d=t.div.getBoundingClientRect();h.top=d.top-2+"px",h.left=d.left+"px",h.position="fixed";var g=t.getHiDPI(s);c.setAttribute("width",Math.round(a*g)+"px"),c.setAttribute("height",Math.round(u*g)+"px"),h.boxShadow="0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23)",h.width=a+"px",h.height=u+"px",h.borderTop="1px solid "+o.resolveProperty("lineColor"),h.backgroundColor=o.resolveProperty("backgroundColor");var f=l[e-n];f*=g,s.scale(g,g),t.renderOverridesCache.floater={columnIndex:e,ctx:s,startX:f,width:a,height:u,hdpiratio:g},h.zIndex="4",this.setCrossBrowserProperty(c,"transform","translate("+f+"px, -2px)"),h.cursor=this.getDraggingCursorName(),t.repaint()},setCrossBrowserProperty:function(t,e,i){var n=e[0].toUpperCase()+e.substr(1);this.setProp(t,"webkit"+n,i),this.setProp(t,"Moz"+n,i),this.setProp(t,"ms"+n,i),this.setProp(t,"O"+n,i),this.setProp(t,e,i)},setProp:function(t,e,i){e in t.style&&(t.style[e]=i)},createDragColumn:function(t,e,i){var r=t.getFixedColumnCount(),s=t.getHScrollValue();r>i&&(s=0);var l=t.getRenderer(),a=l.getColumnEdges(),u=t.getHiDPI(o),c=t.getColumnWidth(i),h=t.div.clientHeight,d=n,g=t.div.getBoundingClientRect(),f=d.style;f.top=g.top+"px",f.left=g.left+"px",f.position="fixed",f.opacity=.85,f.boxShadow="0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)",f.borderTop="1px solid "+l.resolveProperty("lineColor"),f.backgroundColor=t.renderer.resolveProperty("backgroundColor"),d.setAttribute("width",Math.round(c*u)+"px"),d.setAttribute("height",Math.round(h*u)+"px"),f.width=c+"px",f.height=h+"px";var p=a[i-s];p*=u,o.scale(u,u),t.renderOverridesCache.dragger={columnIndex:i,ctx:o,startX:p,width:c,height:h,hdpiratio:u},this.setCrossBrowserProperty(d,"transform","translate("+e+"px, -5px)"),f.zIndex="5",f.cursor=this.getDraggingCursorName(),t.repaint()},dragColumn:function(t,e){var i=this,r=this.columnDragAutoScrollingRight||this.columnDragAutoScrollingLeft,s=t.getHiDPI(o),l=t.renderOverridesCache.dragger.columnIndex,u=t.renderOverridesCache.dragger.width,c=0,h=t.renderer.getFinalVisableColumnBoundary()-u;e=Math.min(e,h+15),e=Math.max(c-15,e);var d=c>e&&0!==l,g=e>h,f=n;this.setCrossBrowserProperty(f,"transition",(i.isWebkit?"-webkit-":"")+"transform 0ms ease, box-shadow "+a+"ms ease"),this.setCrossBrowserProperty(f,"transform","translate("+e+"px, -10px)"),requestAnimationFrame(function(){f.style.display="inline"});var p=t.renderer.getColumnFromPixelX(e+f.width/2/s);d&&(p=0),g&&(p=t.getColumnCount()-1);var v=l>p;if(v=v||p-l>=1,!v||g||r){if(c-10>e&&this.checkAutoScrollToLeft(t,e),e>c-10&&(this.columnDragAutoScrollingLeft=!1),g||e>h+10)return void this.checkAutoScrollToRight(t,e);h+10>e&&(this.columnDragAutoScrollingRight=!1)}else{var m=p>l;if(this.isFloatingNow)return;this.isFloatingNow=!0,this.createFloatColumn(t,p),this.floatColumnTo(t,m)}},checkAutoScrollToRight:function(t,e){this.columnDragAutoScrollingRight||(this.columnDragAutoScrollingRight=!0,this._checkAutoScrollToRight(t,e))},_checkAutoScrollToRight:function(t,e){if(this.columnDragAutoScrollingRight){var i=t.getHScrollValue();if(t.dragging&&!(i>t.sbHScrollConfig.rangeStop-2)){var n=t.renderOverridesCache.dragger.columnIndex;t.scrollBy(1,0);var o=n+1;console.log(o,n),t.swapColumns(o,n),t.renderOverridesCache.dragger.columnIndex=o,setTimeout(this._checkAutoScrollToRight.bind(this,t,e),250)}}},checkAutoScrollToLeft:function(t,e){this.columnDragAutoScrollingLeft||(this.columnDragAutoScrollingLeft=!0,this._checkAutoScrollToLeft(t,e))},_checkAutoScrollToLeft:function(t,e){if(this.columnDragAutoScrollingLeft){var i=t.getHScrollValue();if(t.dragging&&!(1>i)){var n=t.renderOverridesCache.dragger.columnIndex;t.swapColumns(n+i,n+i-1),t.scrollBy(-1,0),setTimeout(this._checkAutoScrollToLeft.bind(this,t,e),250)}}},endDragColumn:function(t){var e=t.getFixedColumnCount(),i=t.getHScrollValue(),o=t.renderOverridesCache.dragger.columnIndex;e>o&&(i=0);var r=t.getRenderer(),s=r.getColumnEdges(),l=this,u=s[o-i],c=n;l.setCrossBrowserProperty(c,"transition",(l.isWebkit?"-webkit-":"")+"transform "+a+"ms ease, box-shadow "+a+"ms ease"),l.setCrossBrowserProperty(c,"transform","translate("+u+"px, -1px)"),c.style.boxShadow="0px 0px 0px #888888",setTimeout(function(){t.renderOverridesCache.dragger=null,t.repaint(),requestAnimationFrame(function(){c.style.display="none",t.endDragColumnNotification()})},a+50)},isHeaderRow:function(t,e){var i=e.viewPoint,n=0===i.y;return n}});e.exports=u},{"./Feature.js":58}],54:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnPicker",{alias:"ColumnPicker",handleKeyUp:function(t,e){var i=e.detail["char"].toLowerCase(),n=t.resolveProperty("editorActivationKeys");n.indexOf(i)>-1&&t.toggleColumnPicker()}});e.exports=o},{"./Feature.js":58}],55:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnResizing",{alias:"ColumnResizing",dragIndex:-2,dragStart:-1,dragIndexStartingSize:-1,getMouseValue:function(t){return t.primitiveEvent.detail.mouse.x},getGridCellValue:function(t){return t.y},getScrollValue:function(t){return t.getHScrollValue()},getAreaSize:function(t,e){return t.getColumnWidth(e)},setAreaSize:function(t,e,i){t.setColumnWidth(e,i)},getPreviousAbsoluteSize:function(t,e){return t.getRenderedWidth(e)},overAreaDivider:function(t,e){return t.overColumnDivider(e)},isFirstFixedOtherArea:function(t,e){return this.isFirstFixedRow(t,e)},getCursorName:function(){return"col-resize"},handleMouseDrag:function(t,e){if(this.dragIndex>-2){var i=this.getMouseValue(e),n=this.getScrollValue(t);this.dragIndex-1&&this.isFirstFixedOtherArea(t,e)){var o=this.getScrollValue(t);n-2){this.cursor=null,this.dragIndex=-2,e.primitiveEvent.stopPropagation();var n=this;t.synchronizeScrollingBoundries(),setTimeout(function(){n.attachChain()},200)}else this.next&&this.next.handleMouseUp(t,e)},handleMouseMove:function(t,e){this.dragIndex>-2||(this.cursor=null,this.next&&this.next.handleMouseMove(t,e),this.checkForAreaResizeCursorChange(t,e))},checkForAreaResizeCursorChange:function(t,e){var i=this.isEnabled(t);i&&this.overAreaDivider(t,e)>-1&&this.isFirstFixedOtherArea(t,e)?this.cursor=this.getCursorName():this.cursor=null},getFixedAreaCount:function(t){var e=t.getFixedColumnCount()+(t.isShowRowNumbers()?1:0)+(t.hasHierarchyColumn()?1:0);return e},handleDoubleClick:function(t,e){var i=this.isEnabled(t),n=this.overAreaDivider(t,e)>-1,o=t.getHeaderRowCount(),r=e.gridCell;i&&n&&r.y<=o?t.autosizeColumn(r.x-1):this.next&&this.next.handleDoubleClick(t,e)},isEnabled:function(t){return!0}});e.exports=o},{"./Feature.js":58}],56:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=50,r=n.extend("ColumnSelection",{alias:"ColumnSelection",currentDrag:null,lastDragCell:null,sbLastAuto:0,sbAutoStart:0,handleMouseUp:function(t,e){this.dragging&&(this.dragging=!1),this.next&&this.next.handleMouseUp(t,e)},handleDoubleClick:function(t,e){this.doubleClickTimer&&(clearTimeout(this.doubleClickTimer),this.doubleClickTimer=void 0),this.next&&this.next.handleDoubleClick(t,e)},handleMouseDown:function(t,e){if(!this.doubleClickTimer){if((!t.isColumnSelection()||e.mousePoint.y<5)&&this.next)return void this.next.handleMouseDown(t,e);var i=e.primitiveEvent.detail.isRightClick,n=e.gridCell,r=e.viewPoint,s=n.x,l=n.y,a=t.isShowHeaderRow()&&0===l&&-1!==s;i||!a?this.next&&this.next.handleMouseDown(t,e):this.doubleClickTimer=setTimeout(function(){this.doubleClickTimer=void 0;var i=t.getFixedColumnCount();r.xi.origin.x+i.extent.x&&(n=1);var u=n,c=o;l&&(u=0),a&&(c=0),this.lastDragCell=e.plusXY(u,c),t.scrollBy(n,o),this.handleMouseDragCellSelection(t,e,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){t.stopEditing();var n=-1!==i.indexOf("SHIFT"),o=t.getMouseDown(),r=e.x,s=e.y;0>r||0>s||(n?(t.clearMostRecentColumnSelection(),t.selectColumn(r,o.x),t.setDragExtent(t.newPoint(r-o.x,0))):(t.toggleSelectColumn(r,i),t.setMouseDown(t.newPoint(r,s)),t.setDragExtent(t.newPoint(0,0))),t.repaint())},handleDOWNSHIFT:function(t){},handleUPSHIFT:function(t){},handleLEFTSHIFT:function(t){this.moveShiftSelect(t,-1)},handleRIGHTSHIFT:function(t){this.moveShiftSelect(t,1)},handleDOWN:function(t){},handleUP:function(t){},handleLEFT:function(t){this.moveSingleSelect(t,-1)},handleRIGHT:function(t){this.moveSingleSelect(t,1)},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now()},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e){var i=t.getColumnCount()-1,n=t.getVisibleColumns()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown(),r=t.getDragExtent(),s=r.x+e;s=Math.min(i-o.x,Math.max(-o.x,s)),t.clearMostRecentColumnSelection(),t.selectColumn(o.x,o.x+s),t.setDragExtent(t.newPoint(s,0)),t.insureModelColIsVisible(s+o.x,e)&&this.pingAutoScroll(),t.repaint()},moveSingleSelect:function(t,e){var i=t.getColumnCount()-1,n=t.getVisibleColumnsCount()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown().plus(t.getDragExtent()),r=o.x+e;r=Math.min(i,Math.max(0,r)),t.clearSelections(),t.selectColumn(r),t.setMouseDown(t.newPoint(r,0)),t.setDragExtent(t.newPoint(0,0)),t.insureModelColIsVisible(r,e)&&this.pingAutoScroll(),t.repaint()},isColumnDragging:function(t){var e=t.lookupFeature("ColumnMoving");if(!e)return!1;var i=e.dragging&&!this.dragging;return i}});e.exports=r},{"./Feature.js":58}],57:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ColumnSorting",{alias:"ColumnSorting",handleDoubleClick:function(t,e){var i=e.gridCell;if(t.isShowHeaderRow()&&0===i.y&&-1!==i.x){var n=e.primitiveEvent.detail.keys;t.toggleSort(i.x,n)}else this.next&&this.next.handleDoubleClick(t,e)},handleMouseMove:function(t,e){var i=e.gridCell.y;this.isFixedRow(t,e)&&1>i?this.cursor="pointer":this.cursor=null,this.next&&this.next.handleMouseMove(t,e)}});e.exports=o},{"./Feature.js":58}],58:[function(t,e,i){"use strict";var n=t("../lib/Base"),o=n.extend("Feature",{next:null,detached:null,cursor:null,currentHoverCell:null,setNext:function(t){this.next?this.next.setNext(t):(this.next=t,this.detached=t)},detachChain:function(){this.next=null},attachChain:function(){this.next=this.detached},handleMouseMove:function(t,e){this.next&&this.next.handleMouseMove(t,e)},handleMouseExit:function(t,e){this.next&&this.next.handleMouseExit(t,e)},handleMouseEnter:function(t,e){this.next&&this.next.handleMouseEnter(t,e)},handleMouseDown:function(t,e){this.next&&this.next.handleMouseDown(t,e)},handleMouseUp:function(t,e){this.next&&this.next.handleMouseUp(t,e)},handleKeyDown:function(t,e){this.next&&this.next.handleKeyDown(t,e)},handleKeyUp:function(t,e){this.next&&this.next.handleKeyUp(t,e)},handleWheelMoved:function(t,e){this.next&&this.next.handleWheelMoved(t,e)},handleDoubleClick:function(t,e){this.next&&this.next.handleDoubleClick(t,e)},handleHoldPulse:function(t,e){this.next&&this.next.handleHoldPulse(t,e)},handleTap:function(t,e){this.next&&this.next.handleTap(t,e)},handleMouseDrag:function(t,e){this.next&&this.next.handleMouseDrag(t,e)},handleContextMenu:function(t,e){this.next&&this.next.handleContextMenu(t,e)},moveSingleSelect:function(t,e,i){this.next&&this.next.moveSingleSelect(t,e,i)},isFixedRow:function(t,e){var i=e.viewPoint,n=i.yr;if(t.isRowSelection()&&!i&&l){var a=t.getFixedRowCount();o.yi.origin.y+i.extent.y&&(o=1);var u=n,c=o;l&&(u=0),a&&(c=0),this.lastDragCell=e.plusXY(u,c),t.scrollBy(n,o),this.handleMouseDragCellSelection(t,e,[]),t.repaint(),setTimeout(this.scrollDrag.bind(this,t),25)}},extendSelection:function(t,e,i){t.stopEditing();var n=-1!==i.indexOf("SHIFT"),o=t.getMouseDown(),r=e.x,s=e.y;0>r||0>s||(n?(t.clearMostRecentRowSelection(),t.selectRow(s,o.y),t.setDragExtent(t.newPoint(0,s-o.y))):(t.toggleSelectRow(s,i),t.setMouseDown(t.newPoint(r,s)),t.setDragExtent(t.newPoint(0,0))),t.repaint())},handleDOWNSHIFT:function(t){this.moveShiftSelect(t,1)},handleUPSHIFT:function(t){this.moveShiftSelect(t,-1)},handleLEFTSHIFT:function(t){},handleRIGHTSHIFT:function(t){},handleDOWN:function(t){this.moveSingleSelect(t,1)},handleUP:function(t){this.moveSingleSelect(t,-1)},handleLEFT:function(t){},handleRIGHT:function(t){var e=t.getMouseDown().plus(t.getDragExtent()),i=t.getColumnCount()-1,n=t.getHeaderColumnCount()+t.getHScrollValue(),o=e.y;n=Math.min(i,n),t.clearSelections(),t.select(n,o,0,0),t.setMouseDown(t.newPoint(n,o)),t.setDragExtent(t.newPoint(0,0)),t.repaint()},getAutoScrollAcceleration:function(){var t=1,e=this.getAutoScrollDuration()/2e3;return t=Math.max(1,Math.floor(e*e*e*e))},setAutoScrollStartTime:function(){this.sbAutoStart=Date.now(); +},pingAutoScroll:function(){var t=Date.now();t-this.sbLastAuto>500&&this.setAutoScrollStartTime(),this.sbLastAuto=Date.now()},getAutoScrollDuration:function(){return Date.now()-this.sbLastAuto>500?0:Date.now()-this.sbAutoStart},moveShiftSelect:function(t,e){var i=t.getRowCount()-1,n=t.getVisibleRows()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown(),r=t.getDragExtent(),s=r.y+e;s=Math.min(i-o.y,Math.max(-o.y,s)),t.clearMostRecentRowSelection(),t.selectRow(o.y,o.y+s),t.setDragExtent(t.newPoint(0,s)),t.insureModelRowIsVisible(s+o.y,e)&&this.pingAutoScroll(),t.fireSyntheticRowSelectionChangedEvent(),t.repaint()},moveSingleSelect:function(t,e){var i=t.getRowCount()-1,n=t.getVisibleRowsCount()-1;t.resolveProperty("scrollingEnabled")||(i=Math.min(i,n));var o=t.getMouseDown().plus(t.getDragExtent()),r=o.y+e;r=Math.min(i,Math.max(0,r)),t.clearSelections(),t.selectRow(r),t.setMouseDown(t.newPoint(0,r)),t.setDragExtent(t.newPoint(0,0)),t.insureModelRowIsVisible(r,e)&&this.pingAutoScroll(),t.fireSyntheticRowSelectionChangedEvent(),t.repaint()},isSingleRowSelection:function(){return!0}});e.exports=o},{"./Feature.js":58}],64:[function(t,e,i){"use strict";var n=t("./Feature.js"),o=n.extend("ThumbwheelScrolling",{alias:"ThumbwheelScrolling",handleWheelMoved:function(t,e){if(t.resolveProperty("scrollingEnabled")){var i=e.primitiveEvent,n=i.wheelDeltaY||-i.deltaY,o=i.wheelDeltaX||-i.deltaX;n>0?t.scrollBy(0,-1):-0>n?t.scrollBy(0,1):o>0?t.scrollBy(-1,0):-0>o&&t.scrollBy(1,0)}}});e.exports=o},{"./Feature.js":58}],65:[function(t,e,i){"use strict";e.exports={Feature:t("./Feature"),CellClick:t("./CellClick"),CellEditing:t("./CellEditing"),CellSelection:t("./CellSelection"),ColumnAutosizing:t("./ColumnAutosizing"),ColumnMoving:t("./ColumnMoving"),ColumnResizing:t("./ColumnResizing"),ColumnSelection:t("./ColumnSelection"),ColumnSorting:t("./ColumnSorting"),Filters:t("./Filters"),KeyPaging:t("./KeyPaging"),OnHover:t("./OnHover"),ColumnPicker:t("./ColumnPicker"),RowResizing:t("./RowResizing"),RowSelection:t("./RowSelection"),ThumbwheelScrolling:t("./ThumbwheelScrolling")}},{"./CellClick":49,"./CellEditing":50,"./CellSelection":51,"./ColumnAutosizing":52,"./ColumnMoving":53,"./ColumnPicker":54,"./ColumnResizing":55,"./ColumnSelection":56,"./ColumnSorting":57,"./Feature":58,"./Filters":59,"./KeyPaging":60,"./OnHover":61,"./RowResizing":62,"./RowSelection":63,"./ThumbwheelScrolling":64}],66:[function(t,e,i){"use strict";var n=t("./deprecated"),o=t("extend-me").Base;o.prototype.deprecated=n,e.exports=o},{"./deprecated":74,"extend-me":5}],67:[function(t,e,i){"use strict";function n(t,e){var i="f"===(typeof e)[0],n=i?e(t):e;return n||0===n?n:""}function o(t,e,i,n,o,r){var s=t.getTextWidth(e,i);switch(e.textAlign){case"center":n-=s/2;break;case"right":n-=s}e.lineWidth=r,e.moveTo(n+.5,o+.5),e.lineTo(n+s+.5,o+.5)}function r(t,e,i,n,o,r){var s=t.getTextHeight(t.font),l=t.getTextWidth(e,i);switch(o-=.4*s.height,e.textAlign){case"center":n-=l/2;break;case"right":n-=l}e.lineWidth=r,e.moveTo(n+.5,o+.5),e.lineTo(n+l+.5,o+.5)}function s(t,e,i,n){if(1===i.length)return i;for(var o,r=[i.shift()];(o=e.getTextWidth(t,r.join(" "))1&&i.unshift(r.pop()),r=[r.join(" ")],i.length&&(r=r.concat(s(t,e,i,n))),r}function l(t,e,i,n){return s(t,e,a(i).split(" "),n)}function a(t){return t.toString().trim().replace(/\s\s+/g," ")}function u(t,e,i,n,o,r,s,l){l||(l=!0),r||(r=5),t.beginPath(),t.moveTo(e+r,i),t.lineTo(e+n-r,i),t.quadraticCurveTo(e+n,i,e+n,i+r),t.lineTo(e+n,i+o-r),t.quadraticCurveTo(e+n,i+o,e+n-r,i+o),t.lineTo(e+r,i+o),t.quadraticCurveTo(e,i+o,e,i+o-r),t.lineTo(e,i+r),t.quadraticCurveTo(e,i,e+r,i),t.closePath(),l&&t.stroke(),s&&t.fill(),t.closePath()}function c(t){if(void 0===t)return 0;var e=t.match(c.regex);if(null===e)return 1;var i=e[4];return void 0===i?0:Number(i)}var h=t("./Base"),d=h.extend("CellProvider",{initialize:function(){this.cellCache={},this.initializeCells()},getCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},getColumnHeaderCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},getRowHeaderCell:function(t){var e=this.cellCache.simpleCellRenderer;return e.config=t,e},paintButton:function(t,e){var i=e.value,n=e.x,o=e.y,r=e.bounds,s=r.x+2,l=r.y+2,a=r.width-3,c=r.height-3,h=c/2,d=t.createLinearGradient(s,l,s,l+c);e.mouseDown?(d.addColorStop(0,"#B5CBED"),d.addColorStop(1,"#4d74ea")):(d.addColorStop(0,"#ffffff"),d.addColorStop(1,"#aaaaaa")),t.fillStyle=d,t.strokeStyle="#000000",u(t,s,l,a,c,h,d,!0);var g=(a-e.getTextWidth(t,i))/2,f=(c-e.getTextHeight(t.font).descent)/2;"middle"!==t.textBaseline&&(t.textBaseline="middle"),t.fillStyle="#000000",e.backgroundColor="rgba(0,0,0,0)",t.fillText(i,s+g,l+f),e.buttonCells[n+","+o]=!0},defaultCellPaint:function(t,e){var i,o,r,s,l,a=e.value,u=e.bounds.x,h=e.bounds.y,d=e.bounds.width,g=e.bounds.height,f=e.headerTextWrapping,p=2,v=0===e.y;a&&a.constructor===Array&&(i=a[0],o=a[2],a=a[1],a&&"object"==typeof a&&"HTMLImageElement"===a.constructor.name&&(r=a,a=null),i&&"IMG"!==i.nodeName&&(i=null),o&&"IMG"!==o.nodeName&&(o=null),r&&"IMG"!==r.nodeName&&(r=null)),a=n(e,a),a=e.formatter(a),t.font!==e.font&&(t.font=e.font),"left"!==t.textAlign&&(t.textAlign="left"),"middle"!==t.textBaseline&&(t.textBaseline="middle");var m,C,w,y;e.isCellHovered&&e.hoverCellHighlight.enabled?w=e.hoverCellHighlight.backgroundColor:e.isRowHovered&&(C=e.hoverRowHighlight).enabled?w=e.isGridColumn||!C.header||void 0===C.header.backgroundColor?C.backgroundColor:C.header.backgroundColor:e.isColumnHovered&&(C=e.hoverColumnHighlight).enabled&&(w=e.isGridRow||!C.header||void 0===C.header.backgroundColor?C.backgroundColor:C.header.backgroundColor),c(w)<1&&(e.isSelected&&(y=n(e,e.backgroundSelectionColor)),c(y)<1&&(m=n(e,e.backgroundColor),c(m)>0&&(t.fillStyle=m,t.fillRect(u,h,d,g))),void 0!==y&&(t.fillStyle=y,t.fillRect(u,h,d,g))),void 0!==w&&(t.fillStyle=w,t.fillRect(u,h,d,g));var b=n(e,e.isSelected?e.foregroundSelectionColor:e.color);t.fillStyle!==b&&(t.fillStyle=b,t.strokeStyle=b),v&&f?this.renderMultiLineText(t,u,h,g,d,e,a):this.renderSingleLineText(t,u,h,g,d,e,a);var x=0;if(i&&(l=Math.round((g-i.height)/2),t.drawImage(i,u+p,h+l),x=Math.max(i.width+2)),o&&d>1.75*g){l=Math.round((g-o.height)/2);var S=u+d-o.width;void 0!==m?(t.fillStyle=m,t.fillRect(S,h,o.width,g)):t.clearRect(S,h,o.width,g),t.drawImage(o,S,h+l),x=Math.max(o.width+2)}r&&(l=Math.round((g-r.height)/2),s=Math.round((d-r.width)/2),t.drawImage(r,u+d-s-r.width,h+l),x=Math.max(r.width+2)),e.cellBorderThickness&&(t.beginPath(),t.rect(u,h,d,g),t.lineWidth=e.cellBorderThickness,t.strokeStyle=e.cellBorderStyle,t.stroke(),t.closePath()),e.minWidth=e.minWidth+2*x},renderMultiLineText:function(t,e,i,n,o,r,s){var u=l(t,r,s,o);if(1===u.length)return this.renderSingleLineText(t,e,i,n,o,r,a(s));var c=r.cellPadding,h=0,d=r.voffset,g=r.halign,f=r.getTextHeight(r.font).height;switch(g){case"right":h=o-c;break;case"center":h=o/2;break;case"left":h=c}var p=0,v=Math.ceil(f/2);d+=Math.ceil((n-(u.length-1)*f)/2),h=Math.max(p,h),d=Math.max(v,d),t.save(),t.rect(e,i,o,n),t.clip(),t.textAlign=g;for(var m=0;mt?"-0":"-")+t}e.exports={date:function(t){return t instanceof Date?t.getFullYear()+n(t.getMonth()+1)+n(t.getDate()):t+""},"default":function(t){return t+""}}},{}],70:[function(t,e,i){"use strict";e.exports=function(){function t(){this.keys=[],this.data={},this.values=[]}var e=".~.#%_",i=0,n=function(t){var n=typeof t;switch(n){case"number":return e+n+"_"+t;case"string":return e+n+"_"+t;case"boolean":return e+n+"_"+t;case"symbol":return e+n+"_"+t;case"undefined":return e+"undefined";case"object":case"function":return t.___finhash||(t.___finhash=e+i++),t.___finhash}},o=Object.is||function(t,e){return t===e?0!==t||1/t==1/e:t!=t&&e!=e},r=function(t,e){if(e!=e||0===e)for(var i=t.length;i--&&!o(t[i],e););else i=[].indexOf.call(t,e);return i};return t.prototype.set=function(t,e){var i=n(t);void 0===this.data[i]&&(this.keys.push(t),this.values.push(e)),this.data[i]=e},t.prototype.get=function(t){var e=n(t);return this.data[e]},t.prototype.getIfAbsent=function(t,e){var i=this.get(t);return void 0===i&&(i=e(t,this)),i},t.prototype.size=function(){return this.keys.length},t.prototype.clear=function(){this.keys.length=0,this.data={}},t.prototype["delete"]=function(t){var e=n(t);if(void 0!==this.data[e]){var i=r(this.keys,t);this.keys.splice(i,1),this.values.splice(i,1),delete this.data[e]}},t.prototype.forEach=function(t){for(var e=this.keys,i=0;it&&(i=t,t>=g&&(i+=h,void 0===s&&(s=i),l=i),!(E>w||i>=d));t++)o=m.getColumnWidth(i),E+=o,this.columnEdges[t+1]=Math.round(E),this.visibleColumns[t]=i,this.columnEdgesIndexMap[i]=t,b=b+Math.round(o/2)+x,this.insertionBounds.push(b),x=Math.round(o/2);for(e=0;f>e&&(n=e,e>=p&&(n+=c,void 0===a&&(a=n),u=n),!(A>y||n>=f));e++)r=m.getRowHeight(n),A+=r,this.rowEdges[e+1]=Math.round(A),this.visibleRows[e]=n,this.rowEdgesIndexMap[n]=e;this.viewHeight=y,this.dataWindow=this.grid.newRectangle(s,a,l-s,u-a)},resolveProperty:function(t){return this.grid.resolveProperty(t)},getGrid:function(){return this.deprecated("grid",{since:"0.2"})},_paint:function(t){this.grid&&(this.renderGrid(t),this.grid.gridRenderedNotification())},getVisibleRowsCount:function(){return this.visibleRows.length-1},getVisibleScrollHeight:function(){return this.viewHeight-this.grid.getFixedRowsHeight()},getVisibleRows:function(){return this.visibleRows},getVisibleColumnsCount:function(){return this.visibleColumns.length-1},getVisibleColumns:function(){return this.visibleColumns},overColumnDivider:function(t){t=Math.round(t);var e=this.getColumnEdges(),i=e.indexOf(t-1);return 0>i&&(i=e.indexOf(t)),0>i&&(i=e.indexOf(t-2)),0>i&&(i=e.indexOf(t+1)),0>i&&(i=e.indexOf(t-3)),i},overRowDivider:function(t){t=Math.round(t);var e=this.rowEdges.indexOf(t+1);return 0>e&&(e=this.rowEdges.indexOf(t)),0>e&&(e=this.rowEdges.indexOf(t-1)),e},getBoundsOfCell:function(t){return this._getBoundsOfCell(t.x,t.y)},_getBoundsOfCell:function(t,e){var i,n=!1,o=!1,r=this.cell,s=this.columnEdgesIndexMap[t];void 0===s&&(s=this.columnEdgesIndexMap[t-1],n=!0);var l,a,u,c=this.columnEdges[s],h=this.columnEdges[s+1],d=h-c;if(r.x=n?h:c,r.width=n?0:d,0>e){var g=this.grid.behavior,f=this.getBounds();u=g.getDefaultRowHeight(),l=f.height+e*u,a=l+u}else i=this.rowEdgesIndexMap[e],void 0===i&&(i=this.rowEdgesIndexMap[e-1],o=!0),l=this.rowEdges[i],a=this.rowEdges[i+1],u=a-l;return r.y=o?a:l,r.height=o?0:u,r},getColumnFromPixelX:function(t){for(var e=0,i=this.getFixedColumnCount(),n=this.grid.getHScrollValue(),o=this.getColumnEdges(),r=1;rt)return r>i&&(r+=n),r-1;return r>i&&(r+=n),r-1},getGridCellFromMousePoint:function(t){var e,i,n,o,r=this.grid.behavior,s=0,l=0,a=0,u=this.getColumnEdges(),c=this.getFixedColumnCount(),h=this.getFixedRowCount(),d=this.getScrollLeft(),g=this.getScrollTop();for(n=0;n=c&&(n+=d),o>=h&&(o+=g);var p=-1,v=r.getColumn(n);return v&&(p=v.index),{gridCell:this.grid.newPoint(n,o),mousePoint:this.grid.newPoint(e,i),viewPoint:f,dataCell:this.grid.newPoint(p,o)}},isColumnVisible:function(t){var e=-1!==this.visibleColumns.indexOf(t);return e},getFinalVisableColumnBoundary:function(){var t=this.isLastColumnVisible(),e=t?2:1,i=this.getColumnEdges()[this.getColumnEdges().length-e],n=Math.min(i,this.getBounds().width-200);return n},isRowVisible:function(t){var e=-1!==this.visibleRows.indexOf(t);return e},isSelected:function(t,e){return this.grid.isSelected(t,e)},renderGrid:function(t){t.beginPath(),this.paintCells(t),this.paintGridlines(t),this.renderOverrides(t),this.renderFocusCell(t),t.closePath()},focusLineStep:[[5,5],[0,1,5,4],[0,2,5,3],[0,3,5,2],[0,4,5,1],[0,5,5,0],[1,5,4,0],[2,5,3,0],[3,5,2,0],[4,5,1,0]],renderFocusCell:function(t){t.beginPath(),this._renderFocusCell(t),t.closePath()},_renderFocusCell:function(t){var e=this.grid.selectionModel.getSelections();if(e&&0!==e.length){var i=e[e.length-1],n=i.origin;if(-1!==n.x){var o=this.getVisibleColumns(),r=this.getVisibleRows(),s=o[o.length-1],l=r[r.length-1],a=i.extent,u=Math.min(n.x,n.x+a.x),c=Math.min(n.y,n.y+a.y);if(!(u>s||c>l)){var h=Math.max(n.x,n.x+a.x)+1;h=Math.min(h,1+s);var d=Math.max(n.y,n.y+a.y)+1;d=Math.min(d,1+l);var g=this._getBoundsOfCell(u,c),f=Math.round(void 0===g.x?this.grid.getFixedColumnsWidth():g.x),p=Math.round(void 0===g.y?this.grid.getFixedRowsHeight():g.y),v=this._getBoundsOfCell(h,d),m=Math.round(void 0===v.x?this.grid.getFixedColumnsWidth():v.x),C=Math.round(void 0===v.y?this.grid.getFixedRowsHeight():v.y),w=Math.min(f,m),y=Math.min(p,C),b=1+m-f,x=1+C-p;w===m&&(b=f-m),y===C&&(x=p-C),1>b*x||(t.rect(w,y,b,x),t.fillStyle=this.resolveProperty("selectionRegionOverlayColor"),t.fill(),t.lineWidth=1,t.strokeStyle=this.resolveProperty("selectionRegionOutlineColor"),t.stroke())}}}},renderOverrides:function(t){var e=this.grid.renderOverridesCache;for(var i in e)if(e.hasOwnProperty(i)){var n=e[i];n&&this.renderOverride(t,n)}},renderOverride:function(t,e){var i=e.hdpiratio,n=e.startX,o=e.width+1,r=e.height,s=e.ctx,l=t.getImageData(n,0,Math.round(o*i),Math.round(r*i));s.putImageData(l,0,0),t.fillStyle=this.resolveProperty("backgroundColor2"),t.fillRect(Math.round(n/i),0,o,r)},getScrollTop:function(){return this.grid.getVScrollValue()},getScrollLeft:function(){return this.grid.getHScrollValue()},getBehavior:function(){return this.deprecated("grid.behavior",{since:"0.2"})},getColumnEdges:function(){return this.columnEdges},getRowEdges:function(){return this.rowEdges},isLastColumnVisible:function(){var t=this.getColumnCount()-1;return-1!==this.visibleColumns.indexOf(t)},getRenderedWidth:function(t){return this.getColumnEdges()[t]},getRenderedHeight:function(t){return this.rowEdges[t]},getCanvas:function(){return this.grid.getCanvas()},isDraggingColumn:function(){return this.grid.isDraggingColumn()},getPageUpRow:function(){for(var t=this.grid,e=this.getVisibleScrollHeight(),i=this.grid.getFixedRowCount(),n=this.dataWindow.origin.y-i,o=0;e>o&&n>-1;)o+=t.getRowHeight(n),n--;return n+1},getPageDownRow:function(){var t=this.grid.getFixedRowCount(),e=this.dataWindow.corner.y-t-1;return e},getColumnCount:function(){return this.grid.getColumnCount()},getRowCount:function(){return this.grid.getRowCount()},getFixedColumnCount:function(){return this.grid.getFixedColumnCount()},getFixedRowCount:function(){return this.grid.getFixedRowCount()},getHeaderRowCount:function(){return this.grid.getHeaderRowCount()},getHeaderColumnCount:function(){return this.grid.getHeaderColumnCount()},paintCells:function(t){var e,i,o,r,s,l,a,u=this.getColumnEdges(),c=this.rowEdges,h=this.getVisibleColumns(),d=this.getVisibleRows(),g=this.grid.behavior,f=0,p=0,v=this.getBounds().height,m=this.grid.isShowRowNumbers()?-1:0,C=h.length;if(this.buttonCells={},C)for(o=m;C>o;o++,f+=a){for(s=h[o],this.renderedColumnMinWidths[s]=0,e=g.getColumnProperties(s).renderCellError,t.save(),a=u[o-m]-f,t.beginPath(),t.rect(f,p,a,v),t.clip(),r=0;r=c,d=0>i,g=!h&&!d,f=n.isFilterRow(i),p=r.getHeaderColumnCount(),v=e>=p,m=n.isShowRowNumbers(),C=n.isHierarchyColumn(e),w=n.isRowSelected(i),y=n.isColumnSelected(e),b=n.isCellSelected(e,i),x=n.isCellSelectedInColumn(e),S=n.isCellSelectedInRow(i),E=n.areAllRowsSelected();m&&-1===e||C?(w?(a=Object.create(l.rowHeaderRowSelection),a.isSelected=!0):(a=Object.create(l.rowHeader),a.isSelected=S),a.isUserDataArea=!1):g||d?(f?(a=Object.create(l.filterProperties),a.isSelected=!1):y?(a=Object.create(l.columnHeaderColumnSelection),a.isSelected=!0):(a=Object.create(l.columnHeader),a.isSelected=x),a.isUserDataArea=!1):C?(a=Object.create(l.rowHeader),a.isSelected=S):(a=Object.create(l),a.isSelected=b||w||y,a.isUserDataArea=!0);var A=i-c+1;if(-1===e?(0===i?a.value=[s.checkbox(E),"",null]:f?a.value=[s.filter(!1),"",null]:g||d?a.value="":a.value=[s.checkbox(w),A,null],a.halign="right"):(a.value=n.getValue(e,i),a.halign=n.getColumnAlignment(e)),a.isGridColumn=v,a.isGridRow=h,a.isColumnHovered=n.isColumnHovered(e)&&v,a.isRowHovered=n.isRowHovered(i)&&h,a.isCellHovered=n.isHovered(e,i)&&v&&h,a.bounds=this._getBoundsOfCell(e,i),a.isCellSelected=b,a.isRowSelected=w,a.isColumnSelected=y,a.isInCurrentSelectionRectangle=n.isInCurrentSelectionRectangle(e,i),n.mouseDownState){var R=n.mouseDownState.gridCell;a.mouseDown=R.x===e&&R.y===i}a.x=e,a.y=i,r.cellPropertiesPrePaintNotification(a);var D=r.getCellRenderer(a,e,i),T=r.getCellProperties(e,i);o(a).extendOwn(T),a.buttonCells=this.buttonCells;var P=a.isUserDataArea?a.format:"default";a.formatter=n.getFormatter(P),D.paint(t,a),this.renderedColumnMinWidths[e]=Math.max(a.minWidth||0,this.renderedColumnMinWidths[e]),u.preferredWidth=this.renderedColumnMinWidths[e]}},isViewableButton:function(t,e){var i=t+","+e;return this.buttonCells[i]===!0},getRowNumbersWidth:function(){var t=this.getColumnEdges();return 0===t.length?0:t[0]},startAnimator:function(){var t,e=this;t=function(){e.animate(),requestAnimationFrame(t)},requestAnimationFrame(t)},animate:function(){var t=this.getCanvas().canvasCTX;t.beginPath(),t.save(),this.renderFocusCell(t),t.restore(),t.closePath()},getBounds:function(){return this.bounds},setBounds:function(t){return this.bounds=t}});e.exports=l},{"../../images/index":3,"./Base":66,"object-iterators":21}],72:[function(t,e,i){"use strict";function n(t){this.grid=t,this.selections=[],this.flattenedX=[],this.flattenedY=[],this.rowSelectionModel=new o,this.columnSelectionModel=new o,this.setLastSelectionType("")}var o=t("sparse-boolean-array");n.prototype={constructor:n.prototype.constructor,allRowsSelected:!1,getLastSelection:function(){var t=this.selections,e=t[t.length-1];return e},getLastSelectionType:function(){return this.lastSelectionType},setLastSelectionType:function(t){this.lastSelectionType=t},select:function(t,e,i,n){var o=this.grid.newRectangle(t,e,i,n);this.selections.push(o),this.flattenedX.push(o.flattenXAt(0)),this.flattenedY.push(o.flattenYAt(0)),this.setLastSelectionType("cell"),this.grid.selectionChanged()},toggleSelect:function(t,e,i,n){var o,r;o=this.selections.find(function(o,s){return r=s,o.origin.x===t&&o.origin.y===e&&o.extent.x===i&&o.extent.y===n}),o?(this.selections.splice(r,1),this.flattenedX.splice(r,1),this.flattenedY.splice(r,1),this.grid.selectionChanged()):this.select(t,e,i,n)},clearMostRecentSelection:function(t){t=t===!0,t||this.setAllRowsSelected(!1),this.selections.length&&--this.selections.length,this.flattenedX.length&&--this.flattenedX.length,this.flattenedY.length&&--this.flattenedY.length},clearMostRecentColumnSelection:function(){this.columnSelectionModel.clearMostRecentSelection(),this.setLastSelectionType("column")},clearMostRecentRowSelection:function(){this.rowSelectionModel.clearMostRecentSelection(),this.setLastSelectionType("row")},clearRowSelection:function(){this.rowSelectionModel.clear(),this.setLastSelectionType("row")},getSelections:function(){return this.selections},hasSelections:function(){return 0!==this.selections.length},hasRowSelections:function(){return!this.rowSelectionModel.isEmpty()},hasColumnSelections:function(){return!this.columnSelectionModel.isEmpty()},isCellSelectedInRow:function(t){return this._isCellSelected(this.flattenedX,0,t)},isCellSelectedInColumn:function(t){return this._isCellSelected(this.flattenedY,t,0)},isSelected:function(t,e){return this.isColumnSelected(t)||this.isRowSelected(e)||this._isCellSelected(this.selections,t,e)},isCellSelected:function(t,e){return this._isCellSelected(this.selections,t,e)},_isCellSelected:function(t,e,i){var n=this;return!!t.find(function(t){return n.rectangleContains(t,e,i)})},clear:function(t){t=t===!0,this.selections.length=0,this.flattenedX.length=0,this.flattenedY.length=0,this.columnSelectionModel.clear(),t||(this.setAllRowsSelected(!1),this.rowSelectionModel.clear())},isRectangleSelected:function(t,e,i,n){return!!this.selections.find(function(o){return o.origin.x===t&&o.origin.y===e&&o.extent.x===i&&o.extent.y===n})},isColumnSelected:function(t){return this.columnSelectionModel.isSelected(t)},isRowSelected:function(t){return this.allRowsSelected||this.rowSelectionModel.isSelected(t)},selectColumn:function(t,e){this.columnSelectionModel.select(t,e),this.setLastSelectionType("column")},selectAllRows:function(){this.clear(),this.setAllRowsSelected(!0)},setAllRowsSelected:function(t){this.allRowsSelected=t},areAllRowsSelected:function(){return this.allRowsSelected},selectRow:function(t,e){this.rowSelectionModel.select(t,e),this.setLastSelectionType("row")},deselectColumn:function(t,e){this.columnSelectionModel.deselect(t,e),this.setLastSelectionType("column")},deselectRow:function(t,e){this.rowSelectionModel.deselect(t,e),this.setLastSelectionType("row")},getSelectedRows:function(){if(this.areAllRowsSelected()){for(var t=this.grid.getHeaderRowCount(),e=this.grid.getRowCount()-t,i=new Array(e),n=0;e>n;n++)i[n]=n+t;return i}return this.rowSelectionModel.getSelections()},getSelectedColumns:function(){return this.columnSelectionModel.getSelections()},isColumnOrRowSelected:function(){return!this.columnSelectionModel.isEmpty()||!this.rowSelectionModel.isEmpty()},getFlattenedYs:function(){var t=[],e={};return this.selections.forEach(function(i){for(var n=i.origin.y,o=i.extent.y+1,r=0;o>r;r++){var s=r+n;e[s]||(t.push(s),e[s]=!0)}}),t.sort(function(t,e){return t-e}),t},selectRowsFromCells:function(t,e){t=t||0,e=e===!0;var i=this.rowSelectionModel;e||(this.setAllRowsSelected(!1),i.clear()),this.selections.forEach(function(e){var n=e.origin.y,o=e.extent.y;n+=t,i.select(n,n+o)})},selectColumnsFromCells:function(t){t=t||0;var e=this.columnSelectionModel;e.clear(),this.selections.forEach(function(i){var n=i.origin.x,o=i.extent.x;n+=t,e.select(n,n+o)})},isInCurrentSelectionRectangle:function(t,e){var i=this.selections[this.selections.length-1];return i&&this.rectangleContains(i,t,e)},rectangleContains:function(t,e,i){var n=t.origin.x,o=t.origin.y,r=n+t.extent.x,s=o+t.extent.y;t.extent.x<0&&(n=r,r=t.origin.x),t.extent.y<0&&(o=s,s=t.origin.y);var l=e>=n&&i>=o&&r>=e&&s>=i;return l}},e.exports=n},{"sparse-boolean-array":24}],73:[function(t,e,i){"use strict";var n=t("./Base"),o=500,r=o+"ms ease-in",s=n.extend("TableDialog",{initialize:function(t){this.grid=t,this.initializeOverlaySurface(),this.openNow=!1},isOpen:function(){return this.openNow},open:function(){if(!this.isOpen()){this.openNow=!0;var t=this;this.overlay.style.backgroundColor=this.grid.resolveProperty("backgroundColor"),this.overlay.style.top=this.overlay.style.bottom=this.overlay.style.right=this.overlay.style.left=0,t.overlay.style.webkitTransition="",this.overlay.style.margin="15px 35px 35px 15px",this.overlay.style.opacity=0,this.overlay.style.zIndex=100,this.closeTransition=function(){this.overlay.style.opacity=0},this._closer||(this._closer=function(e){var i=t.getCharFor(e.keyCode).toLowerCase(),n=t.grid.resolveProperty("editorActivationKeys");(n.indexOf(i)>-1||27===e.keyCode)&&(e.preventDefault(),t.close())}),requestAnimationFrame(function(){t.overlay.style.webkitTransition="opacity "+o+"ms ease-in",requestAnimationFrame(function(){document.addEventListener("keydown",t._closer,!1),t.overlay.style.opacity=.95})}),setTimeout(function(){t.overlay.focus()},100)}},openFrom:function(t){if(!this.isOpen()){this.openNow=!0;var e=this,i=this.overlay.style;i.backgroundColor=this.grid.resolveProperty("backgroundColor");var n=this.grid.div.getBoundingClientRect(),o=t.y+"px "+(n.width-(t.x+t.width))+"px "+(n.height-(t.y+t.height))+"px "+t.x+"px";i.webkitTransition="",i.top=i.right=i.bottom=i.left=0,i.margin=o,i.zIndex=100,i.opacity=1,this.closeTransition=function(){i.margin=o},this._closer||(this._closer=function(t){var i=e.getCharFor(t.keyCode).toLowerCase(),n=e.grid.resolveProperty("editorActivationKeys");(n.indexOf(i)>-1||27===t.keyCode)&&(t.preventDefault(),e.close())}),requestAnimationFrame(function(){document.addEventListener("keydown",e._closer,!1),requestAnimationFrame(function(){requestAnimationFrame(function(){i.webkitTransition="margin-top "+r+", margin-right "+r+", margin-bottom "+r+", margin-left "+r,i.margin="15px 35px 35px 15px"})})}),setTimeout(function(){e.overlay.focus()},100)}},close:function(){this.openNow=!1,document.removeEventListener("keydown",this._closer,!1);var t=this;requestAnimationFrame(function(){t.closeTransition()}),setTimeout(function(){t.clear(),t.overlay.style.zIndex=-1e3,t.onClose&&(t.onClose(),t.onClose=void 0),t.grid.takeFocus()},o)},initializeOverlaySurface:function(){this.overlay=document.createElement("div"),this.overlay.setAttribute("tabindex",0),this.overlay.addEventListener("wheel",function(t){t.stopPropagation()});var t=this.overlay.style;t.outline="none",t.boxShadow="0 19px 38px rgba(0,0,0,0.30), 0 15px 12px rgba(0,0,0,0.22)",t.position="absolute",t.margin=0,t.overflow="hidden",t.opacity=0,t.zIndex=10,this.grid.div.appendChild(this.overlay)},getCharFor:function(t){var e=this.grid.getCanvas().getCharMap();return e[t][0]},clear:function(){this.overlay.innerHTML=""},querySelector:function(t){var e=this.overlay.querySelector(t); +return e},getAnimationTime:function(){return o}});e.exports=s},{"./Base":66}],74:[function(t,e,i){"use strict";console.warn||(console.warn=function(){console.log.apply(console,["WARNING:"].concat(Array.prototype.slice.call(arguments)))});var n=function(t,e){var i,n=t.split("."),o=n[n.length-1],r=e&&e.asOfVersion,s=this;return o="get"+o[0].toUpperCase()+o.substr(1),i="."+o+"() method is deprecated",r&&(i+=" as of v"+e.asOfVersion),i+=". Use ."+t,")"!==t[t.length-1]&&(i+=" property"),i+=" instead. (Will be removed in a future release.)",console.warn(i),n.forEach(function(t){s=s[t]}),s};e.exports=n},{}],75:[function(t,e,i){"use strict";function n(t,e,i,n,r,s){t.fillStyle="#FFD500",t.fill(),t.strokeStyle=t.createPattern(o.caution,"repeat"),t.lineWidth=5,t.beginPath(),t.moveTo(i,n),t.lineTo(i+r,n),t.lineTo(i+r,n+s),t.lineTo(i,n+s),t.lineTo(i,n),t.stroke(),t.beginPath(),t.rect(i,n,r-2,s),t.clip(),t.fillStyle="#A00",t.textAlign="start",t.textBaseline="middle",t.font='bold 6pt "arial narrow", verdana, geneva',t.fillText(e,i+4,n+s/2+.5)}var o=t("../../images/index");e.exports=n},{"../../images/index":3}],76:[function(t,e,i){"use strict";e.exports=function(){function t(t){this.label=t,this.data=[""],this.rowIndexes=[],this.hasChildren=!1,this.depth=0,this.height=1,this.expanded=!1}var e=" ";return t.prototype.isNullObject=!1,t.prototype.getValue=function(t){return this.data[t]},t.prototype.prune=function(t){this.depth=t,this.data[0]=this.computeDepthString()},t.prototype.computeDepthString=function(){var t=e.substring(0,2+3*this.depth)+this.label;return t},t.prototype.computeHeight=function(){return 1},t.prototype.getAllRowIndexes=function(){return this.rowIndexes},t.prototype.computeAggregates=function(t){this.applyAggregates(t)},t.prototype.applyAggregates=function(t){var e=t.hasGroups()?1:0,i=this.getAllRowIndexes();if(0!==i.length){var n=t.aggregates,o=this.data;o.length=n.length+e;var r=t.sorterInstance;r.indexes=i;for(var s=0;s0},t.prototype.hasAggregates=function(){return this.aggregates.length>0},t.prototype.apply=function(){this.buildGroupTree()},t.prototype.clearGroups=function(){this.groupBys.length=0},t.prototype.clearAggregations=function(){this.aggregates.length=0,this.headers.length=0},t.prototype.buildGroupTree=function(){var t,e,i,l,a,u=function(t,e){return l=new r(t),e.set(t,l),l},c=function(t,e){return l=new s(t),e.set(t,l),l},h=this.groupBys,d=this.dataSource,g=d.getRowCount();if(this.presortGroups)for(t=0;te;e++){for(t=0;ti;i++)e[i]=i;this.indexes=e},t.prototype.setData=function(t){this.dataSource.setData(t)},t}()},{}],82:[function(t,e,i){"use strict";var n=t("./DataSourceDecorator");e.exports=function(){function t(t){n.call(this,t,!1),this.filters=[]}return t.prototype=Object.create(n.prototype),t.prototype.add=function(t,e){e.columnIndex=t,this.filters.push(e)},t.prototype.set=function(t,e){e.columnIndex=t,this.filters.push(e)},t.prototype.clearAll=function(){this.filters.length=0,this.indexes.length=0},t.prototype.applyAll=function(){if(0===this.filters.length)return void(this.indexes.length=0);var t=this.indexes;t.length=0;for(var e=this.dataSource.getRowCount(),i=0;e>i;i++)this.applyFiltersTo(i)&&t.push(i)},t.prototype.applyFiltersTo=function(t){for(var e=this.filters,i=!0,n=0;no;o++)this.applyFilterTo(o)&&i.push(o)},t.prototype.applyFilterTo=function(t){for(var e=!1,i=this.filter,n=this.visibleColumnMap.length,o=this.dataSource.getRow(t),r=0;n>r;r++){var s=this.visibleColumnMap[r];if(e=e||i(this.dataSource.getValue(s,t),o,t))return!0}return!1},t}()},{"./DataSourceDecorator":81}],84:[function(t,e,i){"use strict";var n=t("./Utils.js"),o=t("./DataSourceDecorator"),r=function(t){var e="f"===(typeof t)[0],i=e?t():t;return i};e.exports=function(){function t(t){o.call(this,t),this.descendingSort=!1}return t.prototype=Object.create(o.prototype),t.prototype.sortOn=function(t,e){if(0===e)return void(this.indexes.length=0);this.initializeIndexVector();var i=this;n.stableSort(this.indexes,function(e){var n=i.dataSource.getValue(t,e);return n=r(n)},e)},t}()},{"./DataSourceDecorator":81,"./Utils.js":88}],85:[function(t,e,i){"use strict";var n=t("./DataSourceDecorator"),o=t("./DataSourceSorter");e.exports=function(){function t(t){n.call(this,t),this.sorts=[],this.last=this.dataSource}return t.prototype=Object.create(n.prototype),t.prototype.getRow=function(t){return this.last.getRow(t)},t.prototype.sortOn=function(t,e){this.sorts.push([t,e])},t.prototype.applySorts=function(){for(var t=this.sorts,e=this.dataSource,i=0;io;o++)i+=e.getValue(t,o);return i}},min:function(t){return function(e){for(var i=1/0,n=e.getRowCount(),o=0;n>o;o++)i=Math.min(i,e.getValue(t,o));return i}},max:function(t){return function(e){for(var i=-(1/0),n=e.getRowCount(),o=0;n>o;o++)i=Math.max(i,e.getValue(t,o));return i}},avg:function(t){return function(e){for(var i=0,n=e.getRowCount(),o=0;n>o;o++)i+=e.getValue(t,o);return i/n}},first:function(t){return function(e){return e.getValue(t,0)}},last:function(t){return function(e){var i=e.getRowCount();return e.getValue(t,i-1)}},stddev:function(t){return function(e){var i,n=0,o=e.getRowCount();for(i=0;o>i;i++)n+=e.getValue(t,i);var r=n/o,s=0;for(i=0;o>i;i++){var l=e.getValue(t,i)-r;s+=l*l}var a=Math.sqrt(s/o);return a}}}}()},{}],90:[function(t,e,i){"use strict";var n=t("./JSDataSource"),o=t("./DataSourceSorter"),r=t("./DataSourceSorterComposite"),s=t("./DataSourceFilter"),l=t("./DataSourceGlobalFilter"),a=t("./DataSourceAggregator"),u=t("./aggregations");e.exports=function(){return{JSDataSource:n,DataSourceSorter:o,DataSourceSorterComposite:r,DataSourceFilter:s,DataSourceGlobalFilter:l,DataSourceAggregator:a,aggregations:u}}()},{"./DataSourceAggregator":80,"./DataSourceFilter":82,"./DataSourceGlobalFilter":83,"./DataSourceSorter":84,"./DataSourceSorterComposite":85,"./JSDataSource":86,"./aggregations":89}],91:[function(t,e,i){"use strict";var n=function(t,e){return function(i,n){var o=i[0],r=n[0];if(o===r)o=e?n[1]:i[1],r=e?i[1]:n[1];else{if(null===r)return-1;if(null===o)return 1}return t(o,r)}},o=function(t,e){return t-e},r=function(t,e){return e-t},s=function(t,e){return e>t?-1:1},l=function(t,e){return t>e?-1:1},a=function(t){return"number"===t?n(o,!1):n(s,!1)},u=function(t){return"number"===t?n(r,!0):n(l,!0)};e.exports=function(){function t(t,e,i){var n,o;if(0!==t.length&&(void 0===i&&(i=1),0!==i)){var r=typeof e(0);n=1===i?a(r):u(r);var s=new Array(t.length);for(o=0;o