Files
asd-control/webroot/js/pivot.js
2018-07-06 07:31:22 +02:00

1825 lines
64 KiB
JavaScript

(function() {
var callWithJQuery,
indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) return i; } return -1; },
slice = [].slice,
bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; },
hasProp = {}.hasOwnProperty;
callWithJQuery = function(pivotModule) {
if (typeof exports === "object" && typeof module === "object") {
return pivotModule(require("jquery"));
} else if (typeof define === "function" && define.amd) {
return define(["jquery"], pivotModule);
} else {
return pivotModule(jQuery);
}
};
callWithJQuery(function($) {
/*
Utilities
*/
var PivotData, addSeparators, aggregatorTemplates, aggregators, dayNamesEn, derivers, getSort, locales, mthNamesEn, naturalSort, numberFormat, pivotTableRenderer, rd, renderers, rx, rz, sortAs, usFmt, usFmtInt, usFmtPct, zeroPad;
addSeparators = function(nStr, thousandsSep, decimalSep) {
var rgx, x, x1, x2;
nStr += '';
x = nStr.split('.');
x1 = x[0];
x2 = x.length > 1 ? decimalSep + x[1] : '';
rgx = /(\d+)(\d{3})/;
while (rgx.test(x1)) {
x1 = x1.replace(rgx, '$1' + thousandsSep + '$2');
}
return x1 + x2;
};
numberFormat = function(opts) {
var defaults;
defaults = {
digitsAfterDecimal: 2,
scaler: 1,
thousandsSep: ",",
decimalSep: ".",
prefix: "",
suffix: ""
};
opts = $.extend({}, defaults, opts);
return function(x) {
var result;
if (isNaN(x) || !isFinite(x)) {
return "";
}
result = addSeparators((opts.scaler * x).toFixed(opts.digitsAfterDecimal), opts.thousandsSep, opts.decimalSep);
return "" + opts.prefix + result + opts.suffix;
};
};
usFmt = numberFormat();
usFmtInt = numberFormat({
digitsAfterDecimal: 0
});
usFmtPct = numberFormat({
digitsAfterDecimal: 1,
scaler: 100,
suffix: "%"
});
aggregatorTemplates = {
count: function(formatter) {
if (formatter == null) {
formatter = usFmtInt;
}
return function() {
return function(data, rowKey, colKey) {
return {
count: 0,
push: function() {
return this.count++;
},
value: function() {
return this.count;
},
format: formatter
};
};
};
},
uniques: function(fn, formatter) {
if (formatter == null) {
formatter = usFmtInt;
}
return function(arg) {
var attr;
attr = arg[0];
return function(data, rowKey, colKey) {
return {
uniq: [],
push: function(record) {
var ref;
if (ref = record[attr], indexOf.call(this.uniq, ref) < 0) {
return this.uniq.push(record[attr]);
}
},
value: function() {
return fn(this.uniq);
},
format: formatter,
numInputs: attr != null ? 0 : 1
};
};
};
},
sum: function(formatter) {
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var attr;
attr = arg[0];
return function(data, rowKey, colKey) {
return {
sum: 0,
push: function(record) {
if (!isNaN(parseFloat(record[attr]))) {
return this.sum += parseFloat(record[attr]);
}
},
value: function() {
return this.sum;
},
format: formatter,
numInputs: attr != null ? 0 : 1
};
};
};
},
extremes: function(mode, formatter) {
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var attr;
attr = arg[0];
return function(data, rowKey, colKey) {
return {
val: null,
sorter: getSort(data != null ? data.sorters : void 0, attr),
push: function(record) {
var ref, ref1, ref2, x;
x = record[attr];
if (mode === "min" || mode === "max") {
x = parseFloat(x);
if (!isNaN(x)) {
this.val = Math[mode](x, (ref = this.val) != null ? ref : x);
}
}
if (mode === "first") {
if (this.sorter(x, (ref1 = this.val) != null ? ref1 : x) <= 0) {
this.val = x;
}
}
if (mode === "last") {
if (this.sorter(x, (ref2 = this.val) != null ? ref2 : x) >= 0) {
return this.val = x;
}
}
},
value: function() {
return this.val;
},
format: function(x) {
if (isNaN(x)) {
return x;
} else {
return formatter(x);
}
},
numInputs: attr != null ? 0 : 1
};
};
};
},
quantile: function(q, formatter) {
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var attr;
attr = arg[0];
return function(data, rowKey, colKey) {
return {
vals: [],
push: function(record) {
var x;
x = parseFloat(record[attr]);
if (!isNaN(x)) {
return this.vals.push(x);
}
},
value: function() {
var i;
if (this.vals.length === 0) {
return null;
}
this.vals.sort(function(a, b) {
return a - b;
});
i = (this.vals.length - 1) * q;
return (this.vals[Math.floor(i)] + this.vals[Math.ceil(i)]) / 2.0;
},
format: formatter,
numInputs: attr != null ? 0 : 1
};
};
};
},
runningStat: function(mode, ddof, formatter) {
if (mode == null) {
mode = "mean";
}
if (ddof == null) {
ddof = 1;
}
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var attr;
attr = arg[0];
return function(data, rowKey, colKey) {
return {
n: 0.0,
m: 0.0,
s: 0.0,
push: function(record) {
var m_new, x;
x = parseFloat(record[attr]);
if (isNaN(x)) {
return;
}
this.n += 1.0;
if (this.n === 1.0) {
return this.m = x;
} else {
m_new = this.m + (x - this.m) / this.n;
this.s = this.s + (x - this.m) * (x - m_new);
return this.m = m_new;
}
},
value: function() {
if (mode === "mean") {
if (this.n === 0) {
return 0 / 0;
} else {
return this.m;
}
}
if (this.n <= ddof) {
return 0;
}
switch (mode) {
case "var":
return this.s / (this.n - ddof);
case "stdev":
return Math.sqrt(this.s / (this.n - ddof));
}
},
format: formatter,
numInputs: attr != null ? 0 : 1
};
};
};
},
sumOverSum: function(formatter) {
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var denom, num;
num = arg[0], denom = arg[1];
return function(data, rowKey, colKey) {
return {
sumNum: 0,
sumDenom: 0,
push: function(record) {
if (!isNaN(parseFloat(record[num]))) {
this.sumNum += parseFloat(record[num]);
}
if (!isNaN(parseFloat(record[denom]))) {
return this.sumDenom += parseFloat(record[denom]);
}
},
value: function() {
return this.sumNum / this.sumDenom;
},
format: formatter,
numInputs: (num != null) && (denom != null) ? 0 : 2
};
};
};
},
sumOverSumBound80: function(upper, formatter) {
if (upper == null) {
upper = true;
}
if (formatter == null) {
formatter = usFmt;
}
return function(arg) {
var denom, num;
num = arg[0], denom = arg[1];
return function(data, rowKey, colKey) {
return {
sumNum: 0,
sumDenom: 0,
push: function(record) {
if (!isNaN(parseFloat(record[num]))) {
this.sumNum += parseFloat(record[num]);
}
if (!isNaN(parseFloat(record[denom]))) {
return this.sumDenom += parseFloat(record[denom]);
}
},
value: function() {
var sign;
sign = upper ? 1 : -1;
return (0.821187207574908 / this.sumDenom + this.sumNum / this.sumDenom + 1.2815515655446004 * sign * Math.sqrt(0.410593603787454 / (this.sumDenom * this.sumDenom) + (this.sumNum * (1 - this.sumNum / this.sumDenom)) / (this.sumDenom * this.sumDenom))) / (1 + 1.642374415149816 / this.sumDenom);
},
format: formatter,
numInputs: (num != null) && (denom != null) ? 0 : 2
};
};
};
},
fractionOf: function(wrapped, type, formatter) {
if (type == null) {
type = "total";
}
if (formatter == null) {
formatter = usFmtPct;
}
return function() {
var x;
x = 1 <= arguments.length ? slice.call(arguments, 0) : [];
return function(data, rowKey, colKey) {
return {
selector: {
total: [[], []],
row: [rowKey, []],
col: [[], colKey]
}[type],
inner: wrapped.apply(null, x)(data, rowKey, colKey),
push: function(record) {
return this.inner.push(record);
},
format: formatter,
value: function() {
return this.inner.value() / data.getAggregator.apply(data, this.selector).inner.value();
},
numInputs: wrapped.apply(null, x)().numInputs
};
};
};
}
};
aggregatorTemplates.countUnique = function(f) {
return aggregatorTemplates.uniques((function(x) {
return x.length;
}), f);
};
aggregatorTemplates.listUnique = function(s) {
return aggregatorTemplates.uniques((function(x) {
return x.join(s);
}), (function(x) {
return x;
}));
};
aggregatorTemplates.max = function(f) {
return aggregatorTemplates.extremes('max', f);
};
aggregatorTemplates.min = function(f) {
return aggregatorTemplates.extremes('min', f);
};
aggregatorTemplates.first = function(f) {
return aggregatorTemplates.extremes('first', f);
};
aggregatorTemplates.last = function(f) {
return aggregatorTemplates.extremes('last', f);
};
aggregatorTemplates.median = function(f) {
return aggregatorTemplates.quantile(0.5, f);
};
aggregatorTemplates.average = function(f) {
return aggregatorTemplates.runningStat("mean", 1, f);
};
aggregatorTemplates["var"] = function(ddof, f) {
return aggregatorTemplates.runningStat("var", ddof, f);
};
aggregatorTemplates.stdev = function(ddof, f) {
return aggregatorTemplates.runningStat("stdev", ddof, f);
};
aggregators = (function(tpl) {
return {
"Count": tpl.count(usFmtInt),
"Count Unique Values": tpl.countUnique(usFmtInt),
"List Unique Values": tpl.listUnique(", "),
"Sum": tpl.sum(usFmt),
"Integer Sum": tpl.sum(usFmtInt),
"Average": tpl.average(usFmt),
"Median": tpl.median(usFmt),
"Sample Variance": tpl["var"](1, usFmt),
"Sample Standard Deviation": tpl.stdev(1, usFmt),
"Minimum": tpl.min(usFmt),
"Maximum": tpl.max(usFmt),
"First": tpl.first(usFmt),
"Last": tpl.last(usFmt),
"Sum over Sum": tpl.sumOverSum(usFmt),
"80% Upper Bound": tpl.sumOverSumBound80(true, usFmt),
"80% Lower Bound": tpl.sumOverSumBound80(false, usFmt),
"Sum as Fraction of Total": tpl.fractionOf(tpl.sum(), "total", usFmtPct),
"Sum as Fraction of Rows": tpl.fractionOf(tpl.sum(), "row", usFmtPct),
"Sum as Fraction of Columns": tpl.fractionOf(tpl.sum(), "col", usFmtPct),
"Count as Fraction of Total": tpl.fractionOf(tpl.count(), "total", usFmtPct),
"Count as Fraction of Rows": tpl.fractionOf(tpl.count(), "row", usFmtPct),
"Count as Fraction of Columns": tpl.fractionOf(tpl.count(), "col", usFmtPct)
};
})(aggregatorTemplates);
renderers = {
"Table": function(data, opts) {
return pivotTableRenderer(data, opts);
},
"Table Barchart": function(data, opts) {
return $(pivotTableRenderer(data, opts)).barchart();
},
"Heatmap": function(data, opts) {
return $(pivotTableRenderer(data, opts)).heatmap("heatmap", opts);
},
"Row Heatmap": function(data, opts) {
return $(pivotTableRenderer(data, opts)).heatmap("rowheatmap", opts);
},
"Col Heatmap": function(data, opts) {
return $(pivotTableRenderer(data, opts)).heatmap("colheatmap", opts);
}
};
locales = {
en: {
aggregators: aggregators,
renderers: renderers,
localeStrings: {
renderError: "An error occurred rendering the PivotTable results.",
computeError: "An error occurred computing the PivotTable results.",
uiRenderError: "An error occurred rendering the PivotTable UI.",
selectAll: "Select All",
selectNone: "Select None",
tooMany: "(too many to list)",
filterResults: "Filter values",
apply: "Apply",
cancel: "Cancel",
totals: "Totals",
vs: "vs",
by: "by"
}
}
};
mthNamesEn = ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"];
dayNamesEn = ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"];
zeroPad = function(number) {
return ("0" + number).substr(-2, 2);
};
derivers = {
bin: function(col, binWidth) {
return function(record) {
return record[col] - record[col] % binWidth;
};
},
dateFormat: function(col, formatString, utcOutput, mthNames, dayNames) {
var utc;
if (utcOutput == null) {
utcOutput = false;
}
if (mthNames == null) {
mthNames = mthNamesEn;
}
if (dayNames == null) {
dayNames = dayNamesEn;
}
utc = utcOutput ? "UTC" : "";
return function(record) {
var date;
date = new Date(Date.parse(record[col]));
if (isNaN(date)) {
return "";
}
return formatString.replace(/%(.)/g, function(m, p) {
switch (p) {
case "y":
return date["get" + utc + "FullYear"]();
case "m":
return zeroPad(date["get" + utc + "Month"]() + 1);
case "n":
return mthNames[date["get" + utc + "Month"]()];
case "d":
return zeroPad(date["get" + utc + "Date"]());
case "w":
return dayNames[date["get" + utc + "Day"]()];
case "x":
return date["get" + utc + "Day"]();
case "H":
return zeroPad(date["get" + utc + "Hours"]());
case "M":
return zeroPad(date["get" + utc + "Minutes"]());
case "S":
return zeroPad(date["get" + utc + "Seconds"]());
default:
return "%" + p;
}
});
};
}
};
rx = /(\d+)|(\D+)/g;
rd = /\d/;
rz = /^0/;
naturalSort = (function(_this) {
return function(as, bs) {
var a, a1, b, b1, nas, nbs;
if ((bs != null) && (as == null)) {
return -1;
}
if ((as != null) && (bs == null)) {
return 1;
}
if (typeof as === "number" && isNaN(as)) {
return -1;
}
if (typeof bs === "number" && isNaN(bs)) {
return 1;
}
nas = +as;
nbs = +bs;
if (nas < nbs) {
return -1;
}
if (nas > nbs) {
return 1;
}
if (typeof as === "number" && typeof bs !== "number") {
return -1;
}
if (typeof bs === "number" && typeof as !== "number") {
return 1;
}
if (typeof as === "number" && typeof bs === "number") {
return 0;
}
if (isNaN(nbs) && !isNaN(nas)) {
return -1;
}
if (isNaN(nas) && !isNaN(nbs)) {
return 1;
}
a = String(as);
b = String(bs);
if (a === b) {
return 0;
}
if (!(rd.test(a) && rd.test(b))) {
return (a > b ? 1 : -1);
}
a = a.match(rx);
b = b.match(rx);
while (a.length && b.length) {
a1 = a.shift();
b1 = b.shift();
if (a1 !== b1) {
if (rd.test(a1) && rd.test(b1)) {
return a1.replace(rz, ".0") - b1.replace(rz, ".0");
} else {
return (a1 > b1 ? 1 : -1);
}
}
}
return a.length - b.length;
};
})(this);
sortAs = function(order) {
var i, l_mapping, mapping, x;
mapping = {};
l_mapping = {};
for (i in order) {
x = order[i];
mapping[x] = i;
if (typeof x === "string") {
l_mapping[x.toLowerCase()] = i;
}
}
return function(a, b) {
if ((mapping[a] != null) && (mapping[b] != null)) {
return mapping[a] - mapping[b];
} else if (mapping[a] != null) {
return -1;
} else if (mapping[b] != null) {
return 1;
} else if ((l_mapping[a] != null) && (l_mapping[b] != null)) {
return l_mapping[a] - l_mapping[b];
} else if (l_mapping[a] != null) {
return -1;
} else if (l_mapping[b] != null) {
return 1;
} else {
return naturalSort(a, b);
}
};
};
getSort = function(sorters, attr) {
var sort;
if (sorters != null) {
if ($.isFunction(sorters)) {
sort = sorters(attr);
if ($.isFunction(sort)) {
return sort;
}
} else if (sorters[attr] != null) {
return sorters[attr];
}
}
return naturalSort;
};
/*
Data Model class
*/
PivotData = (function() {
function PivotData(input, opts) {
var ref, ref1, ref2, ref3, ref4, ref5, ref6, ref7, ref8, ref9;
if (opts == null) {
opts = {};
}
this.getAggregator = bind(this.getAggregator, this);
this.getRowKeys = bind(this.getRowKeys, this);
this.getColKeys = bind(this.getColKeys, this);
this.sortKeys = bind(this.sortKeys, this);
this.arrSort = bind(this.arrSort, this);
this.input = input;
this.aggregator = (ref = opts.aggregator) != null ? ref : aggregatorTemplates.count()();
this.aggregatorName = (ref1 = opts.aggregatorName) != null ? ref1 : "Count";
this.colAttrs = (ref2 = opts.cols) != null ? ref2 : [];
this.rowAttrs = (ref3 = opts.rows) != null ? ref3 : [];
this.valAttrs = (ref4 = opts.vals) != null ? ref4 : [];
this.sorters = (ref5 = opts.sorters) != null ? ref5 : {};
this.rowOrder = (ref6 = opts.rowOrder) != null ? ref6 : "key_a_to_z";
this.colOrder = (ref7 = opts.colOrder) != null ? ref7 : "key_a_to_z";
this.derivedAttributes = (ref8 = opts.derivedAttributes) != null ? ref8 : {};
this.filter = (ref9 = opts.filter) != null ? ref9 : (function() {
return true;
});
this.tree = {};
this.rowKeys = [];
this.colKeys = [];
this.rowTotals = {};
this.colTotals = {};
this.allTotal = this.aggregator(this, [], []);
this.sorted = false;
PivotData.forEachRecord(this.input, this.derivedAttributes, (function(_this) {
return function(record) {
if (_this.filter(record)) {
return _this.processRecord(record);
}
};
})(this));
}
PivotData.forEachRecord = function(input, derivedAttributes, f) {
var addRecord, compactRecord, i, j, k, l, len1, record, ref, results, results1, tblCols;
if ($.isEmptyObject(derivedAttributes)) {
addRecord = f;
} else {
addRecord = function(record) {
var k, ref, v;
for (k in derivedAttributes) {
v = derivedAttributes[k];
record[k] = (ref = v(record)) != null ? ref : record[k];
}
return f(record);
};
}
if ($.isFunction(input)) {
return input(addRecord);
} else if ($.isArray(input)) {
if ($.isArray(input[0])) {
results = [];
for (i in input) {
if (!hasProp.call(input, i)) continue;
compactRecord = input[i];
if (!(i > 0)) {
continue;
}
record = {};
ref = input[0];
for (j in ref) {
if (!hasProp.call(ref, j)) continue;
k = ref[j];
record[k] = compactRecord[j];
}
results.push(addRecord(record));
}
return results;
} else {
results1 = [];
for (l = 0, len1 = input.length; l < len1; l++) {
record = input[l];
results1.push(addRecord(record));
}
return results1;
}
} else if (input instanceof jQuery) {
tblCols = [];
$("thead > tr > th", input).each(function(i) {
return tblCols.push($(this).text());
});
return $("tbody > tr", input).each(function(i) {
record = {};
$("td", this).each(function(j) {
return record[tblCols[j]] = $(this).text();
});
return addRecord(record);
});
} else {
throw new Error("unknown input format");
}
};
PivotData.prototype.forEachMatchingRecord = function(criteria, callback) {
return PivotData.forEachRecord(this.input, this.derivedAttributes, (function(_this) {
return function(record) {
var k, ref, v;
if (!_this.filter(record)) {
return;
}
for (k in criteria) {
v = criteria[k];
if (v !== ((ref = record[k]) != null ? ref : "null")) {
return;
}
}
return callback(record);
};
})(this));
};
PivotData.prototype.arrSort = function(attrs) {
var a, sortersArr;
sortersArr = (function() {
var l, len1, results;
results = [];
for (l = 0, len1 = attrs.length; l < len1; l++) {
a = attrs[l];
results.push(getSort(this.sorters, a));
}
return results;
}).call(this);
return function(a, b) {
var comparison, i, sorter;
for (i in sortersArr) {
if (!hasProp.call(sortersArr, i)) continue;
sorter = sortersArr[i];
comparison = sorter(a[i], b[i]);
if (comparison !== 0) {
return comparison;
}
}
return 0;
};
};
PivotData.prototype.sortKeys = function() {
var v;
if (!this.sorted) {
this.sorted = true;
v = (function(_this) {
return function(r, c) {
return _this.getAggregator(r, c).value();
};
})(this);
switch (this.rowOrder) {
case "value_a_to_z":
this.rowKeys.sort((function(_this) {
return function(a, b) {
return naturalSort(v(a, []), v(b, []));
};
})(this));
break;
case "value_z_to_a":
this.rowKeys.sort((function(_this) {
return function(a, b) {
return -naturalSort(v(a, []), v(b, []));
};
})(this));
break;
default:
this.rowKeys.sort(this.arrSort(this.rowAttrs));
}
switch (this.colOrder) {
case "value_a_to_z":
return this.colKeys.sort((function(_this) {
return function(a, b) {
return naturalSort(v([], a), v([], b));
};
})(this));
case "value_z_to_a":
return this.colKeys.sort((function(_this) {
return function(a, b) {
return -naturalSort(v([], a), v([], b));
};
})(this));
default:
return this.colKeys.sort(this.arrSort(this.colAttrs));
}
}
};
PivotData.prototype.getColKeys = function() {
this.sortKeys();
return this.colKeys;
};
PivotData.prototype.getRowKeys = function() {
this.sortKeys();
return this.rowKeys;
};
PivotData.prototype.processRecord = function(record) {
var colKey, flatColKey, flatRowKey, l, len1, len2, n, ref, ref1, ref2, ref3, rowKey, x;
colKey = [];
rowKey = [];
ref = this.colAttrs;
for (l = 0, len1 = ref.length; l < len1; l++) {
x = ref[l];
colKey.push((ref1 = record[x]) != null ? ref1 : "null");
}
ref2 = this.rowAttrs;
for (n = 0, len2 = ref2.length; n < len2; n++) {
x = ref2[n];
rowKey.push((ref3 = record[x]) != null ? ref3 : "null");
}
flatRowKey = rowKey.join(String.fromCharCode(0));
flatColKey = colKey.join(String.fromCharCode(0));
this.allTotal.push(record);
if (rowKey.length !== 0) {
if (!this.rowTotals[flatRowKey]) {
this.rowKeys.push(rowKey);
this.rowTotals[flatRowKey] = this.aggregator(this, rowKey, []);
}
this.rowTotals[flatRowKey].push(record);
}
if (colKey.length !== 0) {
if (!this.colTotals[flatColKey]) {
this.colKeys.push(colKey);
this.colTotals[flatColKey] = this.aggregator(this, [], colKey);
}
this.colTotals[flatColKey].push(record);
}
if (colKey.length !== 0 && rowKey.length !== 0) {
if (!this.tree[flatRowKey]) {
this.tree[flatRowKey] = {};
}
if (!this.tree[flatRowKey][flatColKey]) {
this.tree[flatRowKey][flatColKey] = this.aggregator(this, rowKey, colKey);
}
return this.tree[flatRowKey][flatColKey].push(record);
}
};
PivotData.prototype.getAggregator = function(rowKey, colKey) {
var agg, flatColKey, flatRowKey;
flatRowKey = rowKey.join(String.fromCharCode(0));
flatColKey = colKey.join(String.fromCharCode(0));
if (rowKey.length === 0 && colKey.length === 0) {
agg = this.allTotal;
} else if (rowKey.length === 0) {
agg = this.colTotals[flatColKey];
} else if (colKey.length === 0) {
agg = this.rowTotals[flatRowKey];
} else {
agg = this.tree[flatRowKey][flatColKey];
}
return agg != null ? agg : {
value: (function() {
return null;
}),
format: function() {
return "";
}
};
};
return PivotData;
})();
$.pivotUtilities = {
aggregatorTemplates: aggregatorTemplates,
aggregators: aggregators,
renderers: renderers,
derivers: derivers,
locales: locales,
naturalSort: naturalSort,
numberFormat: numberFormat,
sortAs: sortAs,
PivotData: PivotData
};
/*
Default Renderer for hierarchical table layout
*/
pivotTableRenderer = function(pivotData, opts) {
var aggregator, c, colAttrs, colKey, colKeys, defaults, getClickHandler, i, j, r, result, rowAttrs, rowKey, rowKeys, spanSize, tbody, td, th, thead, totalAggregator, tr, txt, val, x;
defaults = {
table: {
clickCallback: null
},
localeStrings: {
totals: "Totals"
}
};
opts = $.extend(true, {}, defaults, opts);
colAttrs = pivotData.colAttrs;
rowAttrs = pivotData.rowAttrs;
rowKeys = pivotData.getRowKeys();
colKeys = pivotData.getColKeys();
if (opts.table.clickCallback) {
getClickHandler = function(value, rowValues, colValues) {
var attr, filters, i;
filters = {};
for (i in colAttrs) {
if (!hasProp.call(colAttrs, i)) continue;
attr = colAttrs[i];
if (colValues[i] != null) {
filters[attr] = colValues[i];
}
}
for (i in rowAttrs) {
if (!hasProp.call(rowAttrs, i)) continue;
attr = rowAttrs[i];
if (rowValues[i] != null) {
filters[attr] = rowValues[i];
}
}
return function(e) {
return opts.table.clickCallback(e, value, filters, pivotData);
};
};
}
result = document.createElement("table");
result.className = "pvtTable";
spanSize = function(arr, i, j) {
var l, len, n, noDraw, ref, ref1, stop, x;
if (i !== 0) {
noDraw = true;
for (x = l = 0, ref = j; 0 <= ref ? l <= ref : l >= ref; x = 0 <= ref ? ++l : --l) {
if (arr[i - 1][x] !== arr[i][x]) {
noDraw = false;
}
}
if (noDraw) {
return -1;
}
}
len = 0;
while (i + len < arr.length) {
stop = false;
for (x = n = 0, ref1 = j; 0 <= ref1 ? n <= ref1 : n >= ref1; x = 0 <= ref1 ? ++n : --n) {
if (arr[i][x] !== arr[i + len][x]) {
stop = true;
}
}
if (stop) {
break;
}
len++;
}
return len;
};
thead = document.createElement("thead");
for (j in colAttrs) {
if (!hasProp.call(colAttrs, j)) continue;
c = colAttrs[j];
tr = document.createElement("tr");
if (parseInt(j) === 0 && rowAttrs.length !== 0) {
th = document.createElement("th");
th.setAttribute("colspan", rowAttrs.length);
th.setAttribute("rowspan", colAttrs.length);
tr.appendChild(th);
}
th = document.createElement("th");
th.className = "pvtAxisLabel";
th.textContent = c;
tr.appendChild(th);
for (i in colKeys) {
if (!hasProp.call(colKeys, i)) continue;
colKey = colKeys[i];
x = spanSize(colKeys, parseInt(i), parseInt(j));
if (x !== -1) {
th = document.createElement("th");
th.className = "pvtColLabel";
th.textContent = colKey[j];
th.setAttribute("colspan", x);
if (parseInt(j) === colAttrs.length - 1 && rowAttrs.length !== 0) {
th.setAttribute("rowspan", 2);
}
tr.appendChild(th);
}
}
if (parseInt(j) === 0) {
th = document.createElement("th");
th.className = "pvtTotalLabel pvtRowTotalLabel";
th.innerHTML = opts.localeStrings.totals;
th.setAttribute("rowspan", colAttrs.length + (rowAttrs.length === 0 ? 0 : 1));
tr.appendChild(th);
}
thead.appendChild(tr);
}
if (rowAttrs.length !== 0) {
tr = document.createElement("tr");
for (i in rowAttrs) {
if (!hasProp.call(rowAttrs, i)) continue;
r = rowAttrs[i];
th = document.createElement("th");
th.className = "pvtAxisLabel";
th.textContent = r;
tr.appendChild(th);
}
th = document.createElement("th");
if (colAttrs.length === 0) {
th.className = "pvtTotalLabel pvtRowTotalLabel";
th.innerHTML = opts.localeStrings.totals;
}
tr.appendChild(th);
thead.appendChild(tr);
}
result.appendChild(thead);
tbody = document.createElement("tbody");
for (i in rowKeys) {
if (!hasProp.call(rowKeys, i)) continue;
rowKey = rowKeys[i];
tr = document.createElement("tr");
for (j in rowKey) {
if (!hasProp.call(rowKey, j)) continue;
txt = rowKey[j];
x = spanSize(rowKeys, parseInt(i), parseInt(j));
if (x !== -1) {
th = document.createElement("th");
th.className = "pvtRowLabel";
th.textContent = txt;
th.setAttribute("rowspan", x);
if (parseInt(j) === rowAttrs.length - 1 && colAttrs.length !== 0) {
th.setAttribute("colspan", 2);
}
tr.appendChild(th);
}
}
for (j in colKeys) {
if (!hasProp.call(colKeys, j)) continue;
colKey = colKeys[j];
aggregator = pivotData.getAggregator(rowKey, colKey);
val = aggregator.value();
td = document.createElement("td");
td.className = "pvtVal row" + i + " col" + j;
td.textContent = aggregator.format(val);
td.setAttribute("data-value", val);
if (getClickHandler != null) {
td.onclick = getClickHandler(val, rowKey, colKey);
}
tr.appendChild(td);
}
totalAggregator = pivotData.getAggregator(rowKey, []);
val = totalAggregator.value();
td = document.createElement("td");
td.className = "pvtTotal rowTotal";
td.textContent = totalAggregator.format(val);
td.setAttribute("data-value", val);
if (getClickHandler != null) {
td.onclick = getClickHandler(val, rowKey, []);
}
td.setAttribute("data-for", "row" + i);
tr.appendChild(td);
tbody.appendChild(tr);
}
tr = document.createElement("tr");
th = document.createElement("th");
th.className = "pvtTotalLabel pvtColTotalLabel";
th.innerHTML = opts.localeStrings.totals;
th.setAttribute("colspan", rowAttrs.length + (colAttrs.length === 0 ? 0 : 1));
tr.appendChild(th);
for (j in colKeys) {
if (!hasProp.call(colKeys, j)) continue;
colKey = colKeys[j];
totalAggregator = pivotData.getAggregator([], colKey);
val = totalAggregator.value();
td = document.createElement("td");
td.className = "pvtTotal colTotal";
td.textContent = totalAggregator.format(val);
td.setAttribute("data-value", val);
if (getClickHandler != null) {
td.onclick = getClickHandler(val, [], colKey);
}
td.setAttribute("data-for", "col" + j);
tr.appendChild(td);
}
totalAggregator = pivotData.getAggregator([], []);
val = totalAggregator.value();
td = document.createElement("td");
td.className = "pvtGrandTotal";
td.textContent = totalAggregator.format(val);
td.setAttribute("data-value", val);
if (getClickHandler != null) {
td.onclick = getClickHandler(val, [], []);
}
tr.appendChild(td);
tbody.appendChild(tr);
result.appendChild(tbody);
result.setAttribute("data-numrows", rowKeys.length);
result.setAttribute("data-numcols", colKeys.length);
return result;
};
/*
Pivot Table core: create PivotData object and call Renderer on it
*/
$.fn.pivot = function(input, inputOpts, locale) {
var defaults, e, localeDefaults, localeStrings, opts, pivotData, result, x;
if (locale == null) {
locale = "en";
}
if (locales[locale] == null) {
locale = "en";
}
defaults = {
cols: [],
rows: [],
vals: [],
rowOrder: "key_a_to_z",
colOrder: "key_a_to_z",
dataClass: PivotData,
filter: function() {
return true;
},
aggregator: aggregatorTemplates.count()(),
aggregatorName: "Count",
sorters: {},
derivedAttributes: {},
renderer: pivotTableRenderer
};
localeStrings = $.extend(true, {}, locales.en.localeStrings, locales[locale].localeStrings);
localeDefaults = {
rendererOptions: {
localeStrings: localeStrings
},
localeStrings: localeStrings
};
opts = $.extend(true, {}, localeDefaults, $.extend({}, defaults, inputOpts));
result = null;
try {
pivotData = new opts.dataClass(input, opts);
try {
result = opts.renderer(pivotData, opts.rendererOptions);
} catch (_error) {
e = _error;
if (typeof console !== "undefined" && console !== null) {
console.error(e.stack);
}
result = $("<span>").html(opts.localeStrings.renderError);
}
} catch (_error) {
e = _error;
if (typeof console !== "undefined" && console !== null) {
console.error(e.stack);
}
result = $("<span>").html(opts.localeStrings.computeError);
}
x = this[0];
while (x.hasChildNodes()) {
x.removeChild(x.lastChild);
}
return this.append(result);
};
/*
Pivot Table UI: calls Pivot Table core above with options set by user
*/
$.fn.pivotUI = function(input, inputOpts, overwrite, locale) {
var a, aggregator, attr, attrLength, attrValues, c, colOrderArrow, defaults, e, existingOpts, fn1, i, initialRender, l, len1, len2, len3, localeDefaults, localeStrings, materializedInput, n, o, opts, ordering, pivotTable, recordsProcessed, ref, ref1, ref2, ref3, refresh, refreshDelayed, renderer, rendererControl, rowOrderArrow, shownAttributes, shownInAggregators, shownInDragDrop, tr1, tr2, uiTable, unused, unusedAttrsVerticalAutoCutoff, unusedAttrsVerticalAutoOverride, x;
if (overwrite == null) {
overwrite = false;
}
if (locale == null) {
locale = "en";
}
if (locales[locale] == null) {
locale = "en";
}
defaults = {
derivedAttributes: {},
aggregators: locales[locale].aggregators,
renderers: locales[locale].renderers,
hiddenAttributes: [],
hiddenFromAggregators: [],
hiddenFromDragDrop: [],
menuLimit: 500,
cols: [],
rows: [],
vals: [],
rowOrder: "key_a_to_z",
colOrder: "key_a_to_z",
dataClass: PivotData,
exclusions: {},
inclusions: {},
unusedAttrsVertical: 85,
autoSortUnusedAttrs: false,
onRefresh: null,
filter: function() {
return true;
},
sorters: {}
};
localeStrings = $.extend(true, {}, locales.en.localeStrings, locales[locale].localeStrings);
localeDefaults = {
rendererOptions: {
localeStrings: localeStrings
},
localeStrings: localeStrings
};
existingOpts = this.data("pivotUIOptions");
if ((existingOpts == null) || overwrite) {
opts = $.extend(true, {}, localeDefaults, $.extend({}, defaults, inputOpts));
} else {
opts = existingOpts;
}
try {
attrValues = {};
materializedInput = [];
recordsProcessed = 0;
PivotData.forEachRecord(input, opts.derivedAttributes, function(record) {
var attr, base, ref, value;
if (!opts.filter(record)) {
return;
}
materializedInput.push(record);
for (attr in record) {
if (!hasProp.call(record, attr)) continue;
if (attrValues[attr] == null) {
attrValues[attr] = {};
if (recordsProcessed > 0) {
attrValues[attr]["null"] = recordsProcessed;
}
}
}
for (attr in attrValues) {
value = (ref = record[attr]) != null ? ref : "null";
if ((base = attrValues[attr])[value] == null) {
base[value] = 0;
}
attrValues[attr][value]++;
}
return recordsProcessed++;
});
uiTable = $("<table>", {
"class": "pvtUi"
}).attr("cellpadding", 5);
rendererControl = $("<td>");
renderer = $("<select>").addClass('pvtRenderer').appendTo(rendererControl).bind("change", function() {
return refresh();
});
ref = opts.renderers;
for (x in ref) {
if (!hasProp.call(ref, x)) continue;
$("<option>").val(x).html(x).appendTo(renderer);
}
unused = $("<td>").addClass('pvtAxisContainer pvtUnused');
shownAttributes = (function() {
var results;
results = [];
for (a in attrValues) {
if (indexOf.call(opts.hiddenAttributes, a) < 0) {
results.push(a);
}
}
return results;
})();
shownInAggregators = (function() {
var l, len1, results;
results = [];
for (l = 0, len1 = shownAttributes.length; l < len1; l++) {
c = shownAttributes[l];
if (indexOf.call(opts.hiddenFromAggregators, c) < 0) {
results.push(c);
}
}
return results;
})();
shownInDragDrop = (function() {
var l, len1, results;
results = [];
for (l = 0, len1 = shownAttributes.length; l < len1; l++) {
c = shownAttributes[l];
if (indexOf.call(opts.hiddenFromDragDrop, c) < 0) {
results.push(c);
}
}
return results;
})();
unusedAttrsVerticalAutoOverride = false;
if (opts.unusedAttrsVertical === "auto") {
unusedAttrsVerticalAutoCutoff = 120;
} else {
unusedAttrsVerticalAutoCutoff = parseInt(opts.unusedAttrsVertical);
}
if (!isNaN(unusedAttrsVerticalAutoCutoff)) {
attrLength = 0;
for (l = 0, len1 = shownInDragDrop.length; l < len1; l++) {
a = shownInDragDrop[l];
attrLength += a.length;
}
unusedAttrsVerticalAutoOverride = attrLength > unusedAttrsVerticalAutoCutoff;
}
if (opts.unusedAttrsVertical === true || unusedAttrsVerticalAutoOverride) {
unused.addClass('pvtVertList');
} else {
unused.addClass('pvtHorizList');
}
fn1 = function(attr) {
var attrElem, checkContainer, closeFilterBox, controls, filterItem, filterItemExcluded, finalButtons, hasExcludedItem, len2, n, placeholder, ref1, sorter, triangleLink, v, value, valueCount, valueList, values;
values = (function() {
var results;
results = [];
for (v in attrValues[attr]) {
results.push(v);
}
return results;
})();
hasExcludedItem = false;
valueList = $("<div>").addClass('pvtFilterBox').hide();
valueList.append($("<h4>").append($("<span>").text(attr), $("<span>").addClass("count").text("(" + values.length + ")")));
if (values.length > opts.menuLimit) {
valueList.append($("<p>").html(opts.localeStrings.tooMany));
} else {
if (values.length > 5) {
controls = $("<p>").appendTo(valueList);
sorter = getSort(opts.sorters, attr);
placeholder = opts.localeStrings.filterResults;
$("<input>", {
type: "text"
}).appendTo(controls).attr({
placeholder: placeholder,
"class": "pvtSearch"
}).bind("keyup", function() {
var accept, accept_gen, filter;
filter = $(this).val().toLowerCase().trim();
accept_gen = function(prefix, accepted) {
return function(v) {
var real_filter, ref1;
real_filter = filter.substring(prefix.length).trim();
if (real_filter.length === 0) {
return true;
}
return ref1 = Math.sign(sorter(v.toLowerCase(), real_filter)), indexOf.call(accepted, ref1) >= 0;
};
};
accept = filter.startsWith(">=") ? accept_gen(">=", [1, 0]) : filter.startsWith("<=") ? accept_gen("<=", [-1, 0]) : filter.startsWith(">") ? accept_gen(">", [1]) : filter.startsWith("<") ? accept_gen("<", [-1]) : filter.startsWith("~") ? function(v) {
if (filter.substring(1).trim().length === 0) {
return true;
}
return v.toLowerCase().match(filter.substring(1));
} : function(v) {
return v.toLowerCase().indexOf(filter) !== -1;
};
return valueList.find('.pvtCheckContainer p label span.value').each(function() {
if (accept($(this).text())) {
return $(this).parent().parent().show();
} else {
return $(this).parent().parent().hide();
}
});
});
controls.append($("<br>"));
$("<button>", {
type: "button"
}).appendTo(controls).html(opts.localeStrings.selectAll).bind("click", function() {
valueList.find("input:visible:not(:checked)").prop("checked", true).toggleClass("changed");
return false;
});
$("<button>", {
type: "button"
}).appendTo(controls).html(opts.localeStrings.selectNone).bind("click", function() {
valueList.find("input:visible:checked").prop("checked", false).toggleClass("changed");
return false;
});
}
checkContainer = $("<div>").addClass("pvtCheckContainer").appendTo(valueList);
ref1 = values.sort(getSort(opts.sorters, attr));
for (n = 0, len2 = ref1.length; n < len2; n++) {
value = ref1[n];
valueCount = attrValues[attr][value];
filterItem = $("<label>");
filterItemExcluded = false;
if (opts.inclusions[attr]) {
filterItemExcluded = (indexOf.call(opts.inclusions[attr], value) < 0);
} else if (opts.exclusions[attr]) {
filterItemExcluded = (indexOf.call(opts.exclusions[attr], value) >= 0);
}
hasExcludedItem || (hasExcludedItem = filterItemExcluded);
$("<input>").attr("type", "checkbox").addClass('pvtFilter').attr("checked", !filterItemExcluded).data("filter", [attr, value]).appendTo(filterItem).bind("change", function() {
return $(this).toggleClass("changed");
});
filterItem.append($("<span>").addClass("value").text(value));
filterItem.append($("<span>").addClass("count").text("(" + valueCount + ")"));
checkContainer.append($("<p>").append(filterItem));
}
}
closeFilterBox = function() {
if (valueList.find("[type='checkbox']").length > valueList.find("[type='checkbox']:checked").length) {
attrElem.addClass("pvtFilteredAttribute");
} else {
attrElem.removeClass("pvtFilteredAttribute");
}
valueList.find('.pvtSearch').val('');
valueList.find('.pvtCheckContainer p').show();
return valueList.hide();
};
finalButtons = $("<p>").appendTo(valueList);
if (values.length <= opts.menuLimit) {
$("<button>", {
type: "button"
}).text(opts.localeStrings.apply).appendTo(finalButtons).bind("click", function() {
if (valueList.find(".changed").removeClass("changed").length) {
refresh();
}
return closeFilterBox();
});
}
$("<button>", {
type: "button"
}).text(opts.localeStrings.cancel).appendTo(finalButtons).bind("click", function() {
valueList.find(".changed:checked").removeClass("changed").prop("checked", false);
valueList.find(".changed:not(:checked)").removeClass("changed").prop("checked", true);
return closeFilterBox();
});
triangleLink = $("<span>").addClass('pvtTriangle').html(" &#x25BE;").bind("click", function(e) {
var left, ref2, top;
ref2 = $(e.currentTarget).position(), left = ref2.left, top = ref2.top;
return valueList.css({
left: left + 10,
top: top + 10
}).show();
});
attrElem = $("<li>").addClass("axis_" + i).append($("<span>").addClass('pvtAttr').text(attr).data("attrName", attr).append(triangleLink));
if (hasExcludedItem) {
attrElem.addClass('pvtFilteredAttribute');
}
return unused.append(attrElem).append(valueList);
};
for (i in shownInDragDrop) {
if (!hasProp.call(shownInDragDrop, i)) continue;
attr = shownInDragDrop[i];
fn1(attr);
}
tr1 = $("<tr>").appendTo(uiTable);
aggregator = $("<select>").addClass('pvtAggregator').bind("change", function() {
return refresh();
});
ref1 = opts.aggregators;
for (x in ref1) {
if (!hasProp.call(ref1, x)) continue;
aggregator.append($("<option>").val(x).html(x));
}
ordering = {
key_a_to_z: {
rowSymbol: "&varr;",
colSymbol: "&harr;",
next: "value_a_to_z"
},
value_a_to_z: {
rowSymbol: "&darr;",
colSymbol: "&rarr;",
next: "value_z_to_a"
},
value_z_to_a: {
rowSymbol: "&uarr;",
colSymbol: "&larr;",
next: "key_a_to_z"
}
};
rowOrderArrow = $("<a>", {
role: "button"
}).addClass("pvtRowOrder").data("order", opts.rowOrder).html(ordering[opts.rowOrder].rowSymbol).bind("click", function() {
$(this).data("order", ordering[$(this).data("order")].next);
$(this).html(ordering[$(this).data("order")].rowSymbol);
return refresh();
});
colOrderArrow = $("<a>", {
role: "button"
}).addClass("pvtColOrder").data("order", opts.colOrder).html(ordering[opts.colOrder].colSymbol).bind("click", function() {
$(this).data("order", ordering[$(this).data("order")].next);
$(this).html(ordering[$(this).data("order")].colSymbol);
return refresh();
});
$("<td>").addClass('pvtVals').appendTo(tr1).append(aggregator).append(rowOrderArrow).append(colOrderArrow).append($("<br>"));
$("<td>").addClass('pvtAxisContainer pvtHorizList pvtCols').appendTo(tr1);
tr2 = $("<tr>").appendTo(uiTable);
tr2.append($("<td>").addClass('pvtAxisContainer pvtRows').attr("valign", "top"));
pivotTable = $("<td>").attr("valign", "top").addClass('pvtRendererArea').appendTo(tr2);
if (opts.unusedAttrsVertical === true || unusedAttrsVerticalAutoOverride) {
uiTable.find('tr:nth-child(1)').prepend(rendererControl);
uiTable.find('tr:nth-child(2)').prepend(unused);
} else {
uiTable.prepend($("<tr>").append(rendererControl).append(unused));
}
this.html(uiTable);
ref2 = opts.cols;
for (n = 0, len2 = ref2.length; n < len2; n++) {
x = ref2[n];
this.find(".pvtCols").append(this.find(".axis_" + ($.inArray(x, shownInDragDrop))));
}
ref3 = opts.rows;
for (o = 0, len3 = ref3.length; o < len3; o++) {
x = ref3[o];
this.find(".pvtRows").append(this.find(".axis_" + ($.inArray(x, shownInDragDrop))));
}
if (opts.aggregatorName != null) {
this.find(".pvtAggregator").val(opts.aggregatorName);
}
if (opts.rendererName != null) {
this.find(".pvtRenderer").val(opts.rendererName);
}
initialRender = true;
refreshDelayed = (function(_this) {
return function() {
var exclusions, inclusions, len4, newDropdown, numInputsToProcess, pivotUIOptions, pvtVals, ref4, ref5, subopts, t, u, unusedAttrsContainer, vals;
subopts = {
derivedAttributes: opts.derivedAttributes,
localeStrings: opts.localeStrings,
rendererOptions: opts.rendererOptions,
sorters: opts.sorters,
cols: [],
rows: [],
dataClass: opts.dataClass
};
numInputsToProcess = (ref4 = opts.aggregators[aggregator.val()]([])().numInputs) != null ? ref4 : 0;
vals = [];
_this.find(".pvtRows li span.pvtAttr").each(function() {
return subopts.rows.push($(this).data("attrName"));
});
_this.find(".pvtCols li span.pvtAttr").each(function() {
return subopts.cols.push($(this).data("attrName"));
});
_this.find(".pvtVals select.pvtAttrDropdown").each(function() {
if (numInputsToProcess === 0) {
return $(this).remove();
} else {
numInputsToProcess--;
if ($(this).val() !== "") {
return vals.push($(this).val());
}
}
});
if (numInputsToProcess !== 0) {
pvtVals = _this.find(".pvtVals");
for (x = t = 0, ref5 = numInputsToProcess; 0 <= ref5 ? t < ref5 : t > ref5; x = 0 <= ref5 ? ++t : --t) {
newDropdown = $("<select>").addClass('pvtAttrDropdown').append($("<option>")).bind("change", function() {
return refresh();
});
for (u = 0, len4 = shownInAggregators.length; u < len4; u++) {
attr = shownInAggregators[u];
newDropdown.append($("<option>").val(attr).text(attr));
}
pvtVals.append(newDropdown);
}
}
if (initialRender) {
vals = opts.vals;
i = 0;
_this.find(".pvtVals select.pvtAttrDropdown").each(function() {
$(this).val(vals[i]);
return i++;
});
initialRender = false;
}
subopts.aggregatorName = aggregator.val();
subopts.vals = vals;
subopts.aggregator = opts.aggregators[aggregator.val()](vals);
subopts.renderer = opts.renderers[renderer.val()];
subopts.rowOrder = rowOrderArrow.data("order");
subopts.colOrder = colOrderArrow.data("order");
exclusions = {};
_this.find('input.pvtFilter').not(':checked').each(function() {
var filter;
filter = $(this).data("filter");
if (exclusions[filter[0]] != null) {
return exclusions[filter[0]].push(filter[1]);
} else {
return exclusions[filter[0]] = [filter[1]];
}
});
inclusions = {};
_this.find('input.pvtFilter:checked').each(function() {
var filter;
filter = $(this).data("filter");
if (exclusions[filter[0]] != null) {
if (inclusions[filter[0]] != null) {
return inclusions[filter[0]].push(filter[1]);
} else {
return inclusions[filter[0]] = [filter[1]];
}
}
});
subopts.filter = function(record) {
var excludedItems, k, ref6, ref7;
if (!opts.filter(record)) {
return false;
}
for (k in exclusions) {
excludedItems = exclusions[k];
if (ref6 = "" + ((ref7 = record[k]) != null ? ref7 : 'null'), indexOf.call(excludedItems, ref6) >= 0) {
return false;
}
}
return true;
};
pivotTable.pivot(materializedInput, subopts);
pivotUIOptions = $.extend({}, opts, {
cols: subopts.cols,
rows: subopts.rows,
colOrder: subopts.colOrder,
rowOrder: subopts.rowOrder,
vals: vals,
exclusions: exclusions,
inclusions: inclusions,
inclusionsInfo: inclusions,
aggregatorName: aggregator.val(),
rendererName: renderer.val()
});
_this.data("pivotUIOptions", pivotUIOptions);
if (opts.autoSortUnusedAttrs) {
unusedAttrsContainer = _this.find("td.pvtUnused.pvtAxisContainer");
$(unusedAttrsContainer).children("li").sort(function(a, b) {
return naturalSort($(a).text(), $(b).text());
}).appendTo(unusedAttrsContainer);
}
pivotTable.css("opacity", 1);
if (opts.onRefresh != null) {
return opts.onRefresh(pivotUIOptions);
}
};
})(this);
refresh = (function(_this) {
return function() {
pivotTable.css("opacity", 0.5);
return setTimeout(refreshDelayed, 10);
};
})(this);
refresh();
this.find(".pvtAxisContainer").sortable({
update: function(e, ui) {
if (ui.sender == null) {
return refresh();
}
},
connectWith: this.find(".pvtAxisContainer"),
items: 'li',
placeholder: 'pvtPlaceholder'
});
} catch (_error) {
e = _error;
if (typeof console !== "undefined" && console !== null) {
console.error(e.stack);
}
this.html(opts.localeStrings.uiRenderError);
}
return this;
};
/*
Heatmap post-processing
*/
$.fn.heatmap = function(scope, opts) {
var colorScaleGenerator, heatmapper, i, j, l, n, numCols, numRows, ref, ref1, ref2;
if (scope == null) {
scope = "heatmap";
}
numRows = this.data("numrows");
numCols = this.data("numcols");
colorScaleGenerator = opts != null ? (ref = opts.heatmap) != null ? ref.colorScaleGenerator : void 0 : void 0;
if (colorScaleGenerator == null) {
colorScaleGenerator = function(values) {
var max, min;
min = Math.min.apply(Math, values);
max = Math.max.apply(Math, values);
return function(x) {
var nonRed;
nonRed = 255 - Math.round(255 * (x - min) / (max - min));
return "rgb(255," + nonRed + "," + nonRed + ")";
};
};
}
heatmapper = (function(_this) {
return function(scope) {
var colorScale, forEachCell, values;
forEachCell = function(f) {
return _this.find(scope).each(function() {
var x;
x = $(this).data("value");
if ((x != null) && isFinite(x)) {
return f(x, $(this));
}
});
};
values = [];
forEachCell(function(x) {
return values.push(x);
});
colorScale = colorScaleGenerator(values);
return forEachCell(function(x, elem) {
return elem.css("background-color", colorScale(x));
});
};
})(this);
switch (scope) {
case "heatmap":
heatmapper(".pvtVal");
break;
case "rowheatmap":
for (i = l = 0, ref1 = numRows; 0 <= ref1 ? l < ref1 : l > ref1; i = 0 <= ref1 ? ++l : --l) {
heatmapper(".pvtVal.row" + i);
}
break;
case "colheatmap":
for (j = n = 0, ref2 = numCols; 0 <= ref2 ? n < ref2 : n > ref2; j = 0 <= ref2 ? ++n : --n) {
heatmapper(".pvtVal.col" + j);
}
}
heatmapper(".pvtTotal.rowTotal");
heatmapper(".pvtTotal.colTotal");
return this;
};
/*
Barchart post-processing
*/
return $.fn.barchart = function(opts) {
var barcharter, i, l, numCols, numRows, ref;
numRows = this.data("numrows");
numCols = this.data("numcols");
barcharter = (function(_this) {
return function(scope) {
var forEachCell, max, min, range, scaler, values;
forEachCell = function(f) {
return _this.find(scope).each(function() {
var x;
x = $(this).data("value");
if ((x != null) && isFinite(x)) {
return f(x, $(this));
}
});
};
values = [];
forEachCell(function(x) {
return values.push(x);
});
max = Math.max.apply(Math, values);
if (max < 0) {
max = 0;
}
range = max;
min = Math.min.apply(Math, values);
if (min < 0) {
range = max - min;
}
scaler = function(x) {
return 100 * x / (1.4 * range);
};
return forEachCell(function(x, elem) {
var bBase, bgColor, text, wrapper;
text = elem.text();
wrapper = $("<div>").css({
"position": "relative",
"height": "55px"
});
bgColor = "gray";
bBase = 0;
if (min < 0) {
bBase = scaler(-min);
}
if (x < 0) {
bBase += scaler(x);
bgColor = "darkred";
x = -x;
}
wrapper.append($("<div>").css({
"position": "absolute",
"bottom": bBase + "%",
"left": 0,
"right": 0,
"height": scaler(x) + "%",
"background-color": bgColor
}));
wrapper.append($("<div>").text(text).css({
"position": "relative",
"padding-left": "5px",
"padding-right": "5px"
}));
return elem.css({
"padding": 0,
"padding-top": "5px",
"text-align": "center"
}).html(wrapper);
});
};
})(this);
for (i = l = 0, ref = numRows; 0 <= ref ? l < ref : l > ref; i = 0 <= ref ? ++l : --l) {
barcharter(".pvtVal.row" + i);
}
barcharter(".pvtTotal.colTotal");
return this;
};
});
}).call(this);
//# sourceMappingURL=pivot.js.map