/*! Bundled by webpack from entry point: ./_Scripts/LegacyJS/site-analytics.js */
/******/ (function() { // webpackBootstrap
/******/ // The require scope
/******/ var __webpack_require__ = {};
/******/
/************************************************************************/
/******/ /* webpack/runtime/publicPath */
/******/ !function() {
/******/ __webpack_require__.p = "";
/******/ }();
/******/
/************************************************************************/
var __webpack_exports__ = {};
// This entry need to be wrapped in an IIFE because it need to be isolated against other entry modules.
!function() {
/*!*************************************!*\
!*** ./_Scripts/set-public-path.ts ***!
\*************************************/
/*** IMPORTS FROM imports-loader ***/
StackExchange = window.StackExchange = (window.StackExchange || {});
StackOverflow = window.StackOverflow = (window.StackOverflow || {});
"use strict";
// This file gets auto-included into each webpack entrypoint.
// We need to set the public path at runtime based on the GlobalRoot site setting.
// (We don't know at build time where the static files will be,
// since we could be deployed on-premise.)
// The value of that site setting gets written out to the dom in Master.cshtml
// (see JavaScriptHelper.PublicPath()) so that we can read it here
__webpack_require__.p = document.getElementById("webpack-public-path").innerText;
}();
// This entry need to be wrapped in an IIFE because it need to be isolated against other entry modules.
!function() {
/*!*********************************************!*\
!*** ./_Scripts/LegacyJS/site-analytics.js ***!
\*********************************************/
/*** IMPORTS FROM imports-loader ***/
StackExchange = window.StackExchange = (window.StackExchange || {});
StackOverflow = window.StackOverflow = (window.StackOverflow || {});
StackExchange.siteAnalytics = function () {
'use strict';
var graphDisplayData = {};
var margin = { top: 10, right: 10, bottom: 30, left: 40 };
var width = 800;
var height = 300;
var stackColors = ['#f48024', '#0077cc', '#65bb5c', '#6b291b', '#ffcf10', '#f23b14'];
var hiddenLines = {};
var weekly = false;
d3.selection.prototype.tooltip = function (target, options) {
var self = this; // the enetered selection
options = $.extend({
style: { 'z-index': 10 },
reposition: function (d, i) {
return {
left: d3.event.pageX + 15,
top: d3.event.pageY - 15,
};
}
}, options);
var tooltipDiv = d3.select('.graph')
.append('div')
.attr('class', 'events-tooltip')
.style('opacity', 0)
.style('display', 'none')
.style('position', 'absolute');
var reposition = function () {
var newPosition = options.reposition.apply(tooltipDiv, arguments);
tooltipDiv.style('left', newPosition.left + 'px').style('top', newPosition.top + 'px');
return tooltipDiv;
};
for (var i in options.style) {
tooltipDiv.style(i, options.style[i]);
}
self
.style('pointer-events', 'all') // makes svg:g groups work on tooltips too...
.on('mouseover', function () {
tooltipDiv.style('opacity', .95).style('display', 'block');
reposition.apply(this, arguments);
})
.on('mousemove', reposition)
.on('mouseout', function (d) {
tooltipDiv.style('opacity', 0).style('display', 'none');
});
};
var basicFormat = function(i) {
return i.toString();
};
var simplify = function (i) {
if (i >= 1000000 && (i % 100000) == 0) return basicFormat(i / 1000000) + 'M';
if (i >= 1000 && (i % 100) == 0) return basicFormat(i / 1000) + 'k';
return basicFormat(i);
};
var timeFormat = d3.time.format('%d %b');
var simpleDate = function (d) {
var date = new Date(d);
return timeFormat(date);
};
var commify = function (number) {
var numberString = String(number.toFixed(0));
var parts = numberString.split('.');
var whole = parts[0];
var pattern = /(\d+)(\d{3})(,|$)/;
while (pattern.test(whole)) {
whole = whole.replace(pattern, '$1,$2');
}
numberString = (parts.length > 1) ? whole + '.' + parts[1] : whole;
return numberString;
}
var renderToGraph = function ($elem, svg, data, extents, color) {
var x = d3.scale.linear().domain([extents.xMin, extents.xMax]).range([0, width - (margin.right + margin.left)]);
var y = d3.scale.linear().domain([extents.yMin, extents.yMax]).range([height - (margin.top + margin.bottom), 1]);
var valueline = d3.svg.line()
.x(function(d, i) { return x(d[0]); })
.y(function(d, i) { return y(d[1]); });
svg = svg.append('g').attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');
var range = svg.append('g').attr('class', 'range');
range.append('g').append('path').attr('stroke', color).attr('class', 'line').attr('d', valueline(data));
svg.selectAll('.range')
.data(data)
.append('g')
.attr('class', 'circles')
.selectAll('.circles')
.data(data)
.enter()
.append('circle').attr('r', 2.7)
.attr('cx', function (d, i) { return x(d[0]); })
.attr('cy', function(d, i) { return y(d[1]); })
.style('fill', function (d, i) { return color; })
.tooltip($elem, {
reposition: function (d) {
this.html(
('
' +
'
' +
'
{value}
' +
'
{date}
' +
'
').formatUnicorn({
date: $('#weekly').is(':checked') ? (function(n){return"Week of "+n.date})({date:new Date(d[0]).toDateString()}) : new Date(d[0]).toDateString(),
value: commify(d[1]),
color: color
}));
return {
left: d3.event.pageX + 10,
top: d3.event.pageY - 10
};
}
});
};
var calculateGroupMaxMins = function (data, groupName) {
var group = weekly ? data.weekly : data.daily;
var filteredOutHidden = group.filter(function(series) {
return hiddenLines[groupName].indexOf(series.label) < 0;
});
if (filteredOutHidden.length === 0) {
return { xMin: 0, xMax: 0, yMin: 0, yMax: 0 };
}
var xMin = d3.min(filteredOutHidden[0].data, function (d, idx) { return d[0]; });
var xMax = d3.max(filteredOutHidden[0].data, function (d, idx) { return d[0]; });;
var yMin = d3.min(filteredOutHidden[0].data, function (d, idx) { return d[1]; });
var yMax = d3.max(filteredOutHidden[0].data, function (d, idx) { return d[1]; });;
for (var i = 1; i < filteredOutHidden.length; i++) {
if (xMin > d3.min(filteredOutHidden[i].data, function (d, idx) { return d[0]; })) {
xMin = d3.min(filteredOutHidden[i].data, function (d, idx) { return d[0]; });
}
if (xMax < d3.max(filteredOutHidden[i].data, function (d, idx) { return d[0]; })) {
xMax = d3.max(filteredOutHidden[i].data, function (d, idx) { return d[0]; });
}
if (yMin > d3.min(filteredOutHidden[i].data, function (d, idx) { return d[1]; })) {
yMin = d3.min(filteredOutHidden[i].data, function (d, idx) { return d[1]; });
}
if (yMax < d3.max(filteredOutHidden[i].data, function (d, idx) { return d[1]; })) {
yMax = d3.max(filteredOutHidden[i].data, function (d, idx) { return d[1]; });
}
}
return { xMin: xMin, xMax: xMax, yMin: yMin, yMax: yMax };
};
var renderGroup = function ($elem, $svg, data) {
var groupName = $elem.closest('.graph').data('name');
var groupExtents = calculateGroupMaxMins(data, groupName);
var checkboxes = '';
var svg = d3.selectAll($svg.toArray());
var group = weekly ? data.weekly : data.daily;
for (var i = 0; i < group.length; i++) {
var series = group[i];
var isSeriesVisible = hiddenLines[groupName].indexOf(series.label) < 0;
var checkedAttribute = '';
if (isSeriesVisible) {
checkedAttribute = ' checked="checked"';
renderToGraph($elem, svg, series.data, groupExtents, stackColors[i]);
}
checkboxes += ' ' + series.localizedLabel + ' ';
}
var x = d3.scale.linear().domain([groupExtents.xMin, groupExtents.xMax]).range([0, width - (margin.right + margin.left)]);
var y = d3.scale.linear().domain([groupExtents.yMin, groupExtents.yMax]).range([height - (margin.top + margin.bottom), 1]);
var yAxis = d3.svg.axis().scale(y).orient('left').tickFormat(simplify);
svg.append('g').attr('class', 'y axis').attr('transform', 'translate(' + margin.left + ',' + margin.top + ')').call(yAxis)
.append('text').attr('transform', 'rotate(-90)').attr('y', 6).attr('dy', '.71em').style('text-anchor', 'end');
var xAxis = d3.svg.axis().scale(x).orient('bottom').ticks(8).tickFormat(simpleDate);
svg.append('g').attr('class', 'x axis').attr('transform', 'translate(' + margin.left + ',' + (height - margin.bottom) + ')').call(xAxis);
$elem.append(checkboxes + ' ');
var graphCol = $('');
$elem.append(graphCol.append($svg));
};
var mergeGroups = function () {
var daily = [], weekly = [];
for (var i = 0; i < arguments.length; i++) {
daily.push.apply(daily, arguments[i].daily);
weekly.push.apply(weekly, arguments[i].weekly);
daily[i].color = weekly[i].color = stackColors[i];
}
return { daily: daily, weekly: weekly };
};
var displayGraph = function ($elem, data) {
$elem.html(''); // wipe
var $svg = $('
');
renderGroup($elem, $svg, data);
};
var reRenderAllGraphs = function(graphs) {
graphs.each(function(index, element) {
var name = $(element).data('name');
var $elem = $(element);
$elem.addSpinner();
StackExchange.helpers.removeSpinner();
displayGraph($elem, graphDisplayData[name]);
});
};
var buildGraphsAndGraphData = function(graphs) {
graphs.each(function (index, element) {
var name = $(element).data('name');
var $elem = $(element);
$elem.addSpinner();
$.ajax('/site-analytics/' + name, { data: { from: $('#from').val(), to: $('#to').val() } })
.done(function (data) {
if (name === 'traffic') {
graphDisplayData[name] = mergeGroups(data.pageViews, data.visits, data.newVisits);
} else {
graphDisplayData[name] = data;
}
if (!hiddenLines.hasOwnProperty(name)) {
hiddenLines[name] = [];
}
displayGraph($elem, graphDisplayData[name]);
})
.fail(function () {
$elem.append('' + "An error has occurred while retrieving data. Please try again in 24 hours." + '
');
})
.always(function () {
StackExchange.helpers.removeSpinner();
});
});
};
function zip(arrays) {
return arrays[0].map(function (_, i) {
return arrays.map(function (array) { return array[i] })
});
}
var getCsvForGraph = function (graphName) {
var graphData = graphDisplayData[graphName];
var header = ['date'];
var seriesData = $('#weekly').is(':checked') ? graphData.weekly : graphData.daily;
var csv = [header.concat(seriesData.map(function (d) { return d.localizedLabel; }))];
var dataPoints = seriesData.map(function(d) { return d.data; });
var joined = zip(dataPoints).map(function (d) {
var date = new Date(d[0][0]);
var merged = [$.datepicker.formatDate('yy-mm-dd', date)];
for (var i = 0; i < d.length; i++) {
merged.push(d[i][1]);
}
return merged;
});
csv = csv.concat(joined);
return d3.csv.formatRows(csv);
};
var downloadCSV = function (csv, fileName) {
// https://stackoverflow.com/q/29304414
var a = document.createElement('a');
var mimeType = 'text/csv';
if (navigator.msSaveBlob) { // IE10
return navigator.msSaveBlob(new Blob([content], { type: mimeType }), fileName);
} else if ('download' in a) { //html5 A[download]
var blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });
var url = URL.createObjectURL(blob);
a.href = url;
a.setAttribute('download', fileName);
document.body.appendChild(a);
setTimeout(function () {
a.click();
document.body.removeChild(a);
}, 66);
return true;
} else { //do iframe dataURL download (old ch+FF):
var f = document.createElement('iframe');
document.body.appendChild(f);
f.src = 'data:' + mimeType + ',' + encodeURIComponent(content);
setTimeout(function () {
document.body.removeChild(f);
}, 333);
return true;
}
};
var init = function (options) {
var graphs = $('.graph');
buildGraphsAndGraphData(graphs);
graphs.on('change', 'input[type=checkbox]', function (e) {
var $graph = $(e.target).closest('.graph');
var graphName = $graph.data('name');
var series = e.target.name;
// todo: persist hidden series
e.target.checked ? hiddenLines[graphName].splice(hiddenLines[graphName].indexOf(series), 1) : hiddenLines[graphName].push(series);
displayGraph($graph, graphDisplayData[graphName]);
});
$('#weekly').change(function (e) {
// todo: persist weekly/daily view
weekly = e.target.checked;
reRenderAllGraphs(graphs);
});
$('#from').datepicker({ dateFormat: 'yy-M-dd', defaultDate: options.fromDate, maxDate: options.maxDate, minDate: options.minDate });
$('#to').datepicker({ dateFormat: 'yy-M-dd', defaultDate: options.toDate, maxDate: options.maxDate, minDate: options.minDate });
$('#from, #to').change(function() {
var graphs = $('.graph');
buildGraphsAndGraphData(graphs);
});
$(".phaseDate").click(function(e) {
var phaseDate = $(e.target).text();
var fromDate = $('#from').val();
if (fromDate !== phaseDate) {
$('#from').datepicker('setDate', phaseDate);
var graphs = $('.graph');
buildGraphsAndGraphData(graphs);
}
e.preventDefault();
});
$('.csv').click(function(e) {
var graphName = $(e.target).closest('.graph-container').find('.graph').data('name');
var from =$('#from').val();
var to = $('#to').val();
var fileName = "{graphName}_{from}_{to}".formatUnicorn({ graphName: graphName, from: from, to: to }) +
($('#weekly').is(':checked')?'_weekly':'') + '.csv';
var csv = getCsvForGraph(graphName);
downloadCSV(csv, fileName);
});
};
var renderPieChart = function (selector, data, options) {
options = options || {};
options.includeLegend = options.includeLegend || false;
options.resizeRatio = options.resizeRatio || 1;
options.includeTooltip = options.includeTooltip || false;
options.useCustomColors = options.useCustomColors || false;
var customColors = ['#ff7f0e', '#0095ff', '#68ba60'];
var pieWidth = 320 * options.resizeRatio, pieHeight = 320 * options.resizeRatio, radius = 150 * options.resizeRatio;
var color = d3.scale.category10();
var totalVisits = data.reduce(function(prev, curr, i, arr) {
return prev + curr.visits;
}, 0);
var svg = d3.selectAll(selector)
.append('svg:svg')
.data([data])
.attr('width', pieWidth + 150)
.attr('height', pieHeight)
.append('svg:g')
.attr('transform', 'translate(' + pieWidth / 2 + ',' + pieHeight / 2 + ')');
var arc = d3.svg.arc()
.outerRadius(radius);
var pie = d3.layout.pie()
.value(function (d) { return d.visits; });
var arcs = svg.selectAll('g.slice')
.data(pie)
.enter()
.append('svg:g')
.attr('class', 'slice');
arcs.append('svg:path')
.attr('fill', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); })
.attr('d', arc);
arcs.filter(function(d) {
return Math.abs(d.startAngle - d.endAngle) >= Math.PI / 6;
})
.append('svg:text')
.attr('transform', function(d) {
//we have to make sure to set these before calling arc.centroid
d.innerRadius = 0;
d.outerRadius = radius;
return 'translate(' + arc.centroid(d) + ')';
})
.attr('text-anchor', 'middle')
.attr('fill', 'white')
.text(function(d) { return d.data.source; });
if (options.includeTooltip) {
arcs.tooltip($(selector), {
reposition: function(d, i) {
this.html(('' +
'
' +
'
{source}
' +
'
{value}
' +
'
').formatUnicorn({
source: d.data.source,
value: commify(d.data.visits) + ' (' + (d.data.visits / totalVisits * 100).toFixed(1) + '%)',
color: options.useCustomColors ? customColors[i] : color(i)
}));
return {
left: d3.event.pageX + 10,
top: d3.event.pageY - 10,
};
}
});
}
if (options.includeLegend) {
var legendRectSize = 18;
var legendSpacing = 4;
var legend = svg.selectAll('.legend')
.data(data)
.enter()
.append('svg:g')
.attr('class', 'legend')
.attr('transform', function(d, i) {
var rectHeight = legendRectSize + legendSpacing;
var offset = rectHeight * data.length / 2;
var horz = pieWidth / 2 + 5;
var vert = i * offset - pieHeight / 2 + 10;
return 'translate(' + horz + ',' + vert + ')';
});
legend.append('svg:rect')
.attr('width', legendRectSize)
.attr('height', legendRectSize)
.style('fill', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); })
.style('stroke', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); });
legend.append('svg:text')
.attr('x', legendRectSize + legendSpacing)
.attr('y', legendRectSize - legendSpacing)
.attr('font-size', 10)
.text(function (d) { return d.source + ': ' + commify(d.visits) + ' (' + (d.visits / totalVisits * 100).toFixed(1) + '%)'; });
}
};
var renderTable = function (selector, data) {
var rows = '';
$.each(data, function(i, d) {
rows += '' + d.source + ' ' + commify(d.visits) + ' ';
});
$(selector).append(rows);
};
return { init: init, renderPieChart: renderPieChart, renderTable: renderTable };
}();
}();
/******/ })()
;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"site-analytics.js","mappings":";;UAAA;UACA;;;;;WCDA;;;;;;;;;;;;;;;;;;ACAA,6DAA6D;AAC7D,kFAAkF;AAClF,+DAA+D;AAC/D,0CAA0C;AAC1C,8EAA8E;AAC9E,kEAAkE;AAClE,qBAAuB,GAAG,QAAQ,CAAC,cAAc,CAAC,qBAAqB,CAAC,CAAC,SAAS,CAAC;;;;;;;;ACNnF;;;AAGA,kEAAkE;AAClE,kEAAkE;;;AAGlE;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yBAAyB;AACzB;AACA,qBAAqB,eAAe;AACpC;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,EAAE;AAClC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgC,iBAAiB;AACjD,gCAAgC,iBAAiB;AACjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,iBAAiB;AAC3D,yCAAyC,iBAAiB;AAC1D,6CAA6C,eAAe;AAC5D;AACA;AACA;AACA;AACA,iFAAiF,OAAO,QAAQ;AAChG,iCAAiC,MAAM;AACvC,iCAAiC,KAAK;AACtC;AACA,uFAAuF,qCAAqC;AAC5H;AACA;AACA,yBAAyB;AACzB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,qBAAqB;AACrB;AACA;AACA,yEAAyE,cAAc;AACvF,yEAAyE,cAAc;AACvF,yEAAyE,cAAc;AACvF,yEAAyE,cAAc;AACvF;AACA,wBAAwB,8BAA8B;AACtD,6EAA6E,cAAc;AAC3F,6EAA6E,cAAc;AAC3F;AACA,6EAA6E,cAAc;AAC3F,6EAA6E,cAAc;AAC3F;AACA,6EAA6E,cAAc;AAC3F,6EAA6E,cAAc;AAC3F;AACA,6EAA6E,cAAc;AAC3F,6EAA6E,cAAc;AAC3F;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,kBAAkB;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,sBAAsB;AAC9C;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA,wBAAwB;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD,QAAQ,8CAA8C;AACtG;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA,iBAAiB;AACjB;AACA;AACA,iBAAiB;AACjB,SAAS;AACT;AACA;AACA;AACA;AACA,iDAAiD,iBAAiB;AAClE,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+DAA+D,0BAA0B;AACzF;AACA,sDAAsD,gBAAgB;AACtE;AACA;AACA;AACA;AACA,4BAA4B,cAAc;AAC1C;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC;AACpC,8DAA8D,gBAAgB;AAC9E,UAAU,4BAA4B;AACtC,yCAAyC,gBAAgB,cAAc,GAAG;AAC1E;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA,UAAU,OAAO;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA,gCAAgC,0GAA0G;AAC1I,8BAA8B,wGAAwG;AACtI;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B,UAAU,EAAE,KAAK,EAAE,GAAG,kBAAkB,0CAA0C;AAC/G;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,kBAAkB;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4CAA4C,8DAA8D;AAC1G;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,gCAAgC,uBAAuB;AACvD;AACA;AACA;AACA;AACA;AACA,6EAA6E,OAAO,QAAQ;AAC5F,6BAA6B,OAAO;AACpC,6BAA6B,MAAM;AACnC;AACA;AACA;AACA;AACA,qBAAqB;AACrB;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB;AACjB;AACA;AACA;AACA;AACA,iDAAiD,8DAA8D;AAC/G,mDAAmD,8DAA8D;AACjH;AACA;AACA;AACA;AACA;AACA,qCAAqC,uGAAuG;AAC5I;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,aAAa;AACb,CAAC","sources":["webpack://stackoverflow/webpack/bootstrap","webpack://stackoverflow/webpack/runtime/publicPath","webpack://stackoverflow/./_Scripts/set-public-path.ts","webpack://stackoverflow/./_Scripts/LegacyJS/site-analytics.js"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","__webpack_require__.p = \"\";","// This file gets auto-included into each webpack entrypoint.\r\n// We need to set the public path at runtime based on the GlobalRoot site setting.\r\n// (We don't know at build time where the static files will be,\r\n// since we could be deployed on-premise.)\r\n// The value of that site setting gets written out to the dom in Master.cshtml\r\n// (see JavaScriptHelper.PublicPath()) so that we can read it here\r\n__webpack_public_path__ = document.getElementById(\"webpack-public-path\").innerText;\r\n","/*** IMPORTS FROM imports-loader ***/\n\n\nStackExchange = window.StackExchange = (window.StackExchange || {});\nStackOverflow = window.StackOverflow = (window.StackOverflow || {});\n\n\nStackExchange.siteAnalytics = function () {\r\n    'use strict';\r\n\r\n    var graphDisplayData = {};\r\n    var margin = { top: 10, right: 10, bottom: 30, left: 40 };\r\n    var width = 800;\r\n    var height = 300;\r\n    var stackColors = ['#f48024', '#0077cc', '#65bb5c', '#6b291b', '#ffcf10', '#f23b14'];\r\n\r\n    var hiddenLines = {};\r\n    var weekly = false;\r\n\r\n    d3.selection.prototype.tooltip = function (target, options) {\r\n        var self = this; // the enetered selection\r\n        options = $.extend({\r\n            style: { 'z-index': 10 },\r\n            reposition: function (d, i) {\r\n                return {\r\n                    left: d3.event.pageX + 15,\r\n                    top: d3.event.pageY - 15,\r\n                };\r\n            }\r\n        }, options);\r\n        var tooltipDiv = d3.select('.graph')\r\n            .append('div')\r\n            .attr('class', 'events-tooltip')\r\n            .style('opacity', 0)\r\n            .style('display', 'none')\r\n            .style('position', 'absolute');\r\n        var reposition = function () {\r\n            var newPosition = options.reposition.apply(tooltipDiv, arguments);\r\n            tooltipDiv.style('left', newPosition.left + 'px').style('top', newPosition.top + 'px');\r\n\r\n            return tooltipDiv;\r\n        };\r\n\r\n        for (var i in options.style) {\r\n            tooltipDiv.style(i, options.style[i]);\r\n        }\r\n\r\n        self\r\n            .style('pointer-events', 'all') // makes svg:g groups work on tooltips too...\r\n            .on('mouseover', function () {\r\n                tooltipDiv.style('opacity', .95).style('display', 'block');\r\n                reposition.apply(this, arguments);\r\n            })\r\n            .on('mousemove', reposition)\r\n            .on('mouseout', function (d) {\r\n                tooltipDiv.style('opacity', 0).style('display', 'none');\r\n            });\r\n    };\r\n\r\n    var basicFormat = function(i) {\r\n        return i.toString();\r\n    };\r\n\r\n    var simplify = function (i) {\r\n        if (i >= 1000000 && (i % 100000) == 0) return basicFormat(i / 1000000) + 'M';\r\n        if (i >= 1000 && (i % 100) == 0) return basicFormat(i / 1000) + 'k';\r\n        return basicFormat(i);\r\n    };\r\n\r\n    var timeFormat = d3.time.format('%d %b');\r\n\r\n    var simpleDate = function (d) {\r\n        var date = new Date(d);\r\n        return timeFormat(date);\r\n    };\r\n\r\n    var commify = function (number) {\r\n        var numberString = String(number.toFixed(0));\r\n        var parts = numberString.split('.');\r\n        var whole = parts[0];\r\n\r\n        var pattern = /(\\d+)(\\d{3})(,|$)/;\r\n        while (pattern.test(whole)) {\r\n            whole = whole.replace(pattern, '$1,$2');\r\n        }\r\n\r\n        numberString = (parts.length > 1) ? whole + '.' + parts[1] : whole;\r\n        return numberString;\r\n    }\r\n\r\n    var renderToGraph = function ($elem, svg, data, extents, color) {\r\n\r\n        var x = d3.scale.linear().domain([extents.xMin, extents.xMax]).range([0, width - (margin.right + margin.left)]);\r\n        var y = d3.scale.linear().domain([extents.yMin, extents.yMax]).range([height - (margin.top + margin.bottom), 1]);\r\n\r\n        var valueline = d3.svg.line()\r\n            .x(function(d, i) { return x(d[0]); })\r\n            .y(function(d, i) { return y(d[1]); });\r\n\r\n        svg = svg.append('g').attr('transform', 'translate(' + margin.left + ',' + margin.top + ')');\r\n        var range = svg.append('g').attr('class', 'range');\r\n        range.append('g').append('path').attr('stroke', color).attr('class', 'line').attr('d', valueline(data));\r\n        svg.selectAll('.range')\r\n            .data(data)\r\n            .append('g')\r\n            .attr('class', 'circles')\r\n            .selectAll('.circles')\r\n            .data(data)\r\n            .enter()\r\n            .append('circle').attr('r', 2.7)\r\n            .attr('cx', function (d, i) { return x(d[0]); })\r\n            .attr('cy', function(d, i) { return y(d[1]); })\r\n            .style('fill', function (d, i) { return color; })\r\n            .tooltip($elem, {\r\n                reposition: function (d) {\r\n                    this.html(\r\n                        ('<div>' +\r\n                            '<div class=\"series-color\" style=\"background-color: {color};\">&nbsp;</div>' +\r\n                            '<p>{value}</p>' +\r\n                            '<p>{date}</p>' +\r\n                        '</div>').formatUnicorn({\r\n                            date: $('#weekly').is(':checked') ? _s('Week of $date$', { date : new Date(d[0]).toDateString()}) : new Date(d[0]).toDateString(),\r\n                            value: commify(d[1]),\r\n                            color: color\r\n                        }));\r\n\r\n                    return {\r\n                        left: d3.event.pageX + 10,\r\n                        top: d3.event.pageY - 10\r\n                    };\r\n                }\r\n            });\r\n    };\r\n\r\n    var calculateGroupMaxMins = function (data, groupName) {\r\n        var group = weekly ? data.weekly : data.daily;\r\n        var filteredOutHidden = group.filter(function(series) {\r\n            return hiddenLines[groupName].indexOf(series.label) < 0;\r\n        });\r\n\r\n        if (filteredOutHidden.length === 0) {\r\n            return { xMin: 0, xMax: 0, yMin: 0, yMax: 0 };\r\n        }\r\n\r\n        var xMin = d3.min(filteredOutHidden[0].data, function (d, idx) { return d[0]; });\r\n        var xMax = d3.max(filteredOutHidden[0].data, function (d, idx) { return d[0]; });;\r\n        var yMin = d3.min(filteredOutHidden[0].data, function (d, idx) { return d[1]; });\r\n        var yMax = d3.max(filteredOutHidden[0].data, function (d, idx) { return d[1]; });;\r\n\r\n        for (var i = 1; i < filteredOutHidden.length; i++) {\r\n            if (xMin > d3.min(filteredOutHidden[i].data, function (d, idx) { return d[0]; })) {\r\n                xMin = d3.min(filteredOutHidden[i].data, function (d, idx) { return d[0]; });\r\n            }\r\n            if (xMax < d3.max(filteredOutHidden[i].data, function (d, idx) { return d[0]; })) {\r\n                xMax = d3.max(filteredOutHidden[i].data, function (d, idx) { return d[0]; });\r\n            }\r\n            if (yMin > d3.min(filteredOutHidden[i].data, function (d, idx) { return d[1]; })) {\r\n                yMin = d3.min(filteredOutHidden[i].data, function (d, idx) { return d[1]; });\r\n            }\r\n            if (yMax < d3.max(filteredOutHidden[i].data, function (d, idx) { return d[1]; })) {\r\n                yMax = d3.max(filteredOutHidden[i].data, function (d, idx) { return d[1]; });\r\n            }\r\n        }\r\n\r\n        return { xMin: xMin, xMax: xMax, yMin: yMin, yMax: yMax };\r\n    };\r\n\r\n    var renderGroup = function ($elem, $svg, data) {\r\n        var groupName = $elem.closest('.graph').data('name');\r\n        var groupExtents = calculateGroupMaxMins(data, groupName);\r\n        var checkboxes = '<div class=\"col-2\"><ul>';\r\n        var svg = d3.selectAll($svg.toArray());\r\n\r\n        var group = weekly ? data.weekly : data.daily;\r\n        for (var i = 0; i < group.length; i++) {\r\n            var series = group[i];\r\n            var isSeriesVisible = hiddenLines[groupName].indexOf(series.label) < 0;\r\n            var checkedAttribute = '';\r\n            if (isSeriesVisible) {\r\n                checkedAttribute = ' checked=\"checked\"';\r\n                renderToGraph($elem, svg, series.data, groupExtents, stackColors[i]);\r\n            }\r\n            checkboxes += '<li><label style=\"color:' + stackColors[i] + '\"><input type=\"checkbox\" name=\"' + series.label + '\"' + checkedAttribute + '>' + series.localizedLabel + '</label></li>';\r\n        }\r\n\r\n        var x = d3.scale.linear().domain([groupExtents.xMin, groupExtents.xMax]).range([0, width - (margin.right + margin.left)]);\r\n        var y = d3.scale.linear().domain([groupExtents.yMin, groupExtents.yMax]).range([height - (margin.top + margin.bottom), 1]);\r\n\r\n        var yAxis = d3.svg.axis().scale(y).orient('left').tickFormat(simplify);\r\n        svg.append('g').attr('class', 'y axis').attr('transform', 'translate(' + margin.left + ',' + margin.top + ')').call(yAxis)\r\n            .append('text').attr('transform', 'rotate(-90)').attr('y', 6).attr('dy', '.71em').style('text-anchor', 'end');\r\n\r\n        var xAxis = d3.svg.axis().scale(x).orient('bottom').ticks(8).tickFormat(simpleDate);\r\n        svg.append('g').attr('class', 'x axis').attr('transform', 'translate(' + margin.left + ',' + (height - margin.bottom) + ')').call(xAxis);\r\n\r\n        $elem.append(checkboxes + '</ul></div>');\r\n        var graphCol = $('<div class=\"col-10\">');\r\n        $elem.append(graphCol.append($svg));\r\n    };\r\n\r\n    var mergeGroups = function () {\r\n        var daily = [], weekly = [];\r\n\r\n        for (var i = 0; i < arguments.length; i++) {\r\n            daily.push.apply(daily, arguments[i].daily);\r\n            weekly.push.apply(weekly, arguments[i].weekly);\r\n\r\n            daily[i].color = weekly[i].color = stackColors[i];\r\n        }\r\n\r\n        return { daily: daily, weekly: weekly };\r\n    };\r\n\r\n    var displayGraph = function ($elem, data) {\r\n        $elem.html(''); // wipe\r\n        var $svg = $('<svg width=\"' + width + '\" height=\"' + height + '\">');\r\n        renderGroup($elem, $svg, data);\r\n    };\r\n\r\n    var reRenderAllGraphs = function(graphs) {\r\n        graphs.each(function(index, element) {\r\n            var name = $(element).data('name');\r\n            var $elem = $(element);\r\n            $elem.addSpinner();\r\n            StackExchange.helpers.removeSpinner();\r\n            displayGraph($elem, graphDisplayData[name]);\r\n        });\r\n    };\r\n\r\n    var buildGraphsAndGraphData = function(graphs) {\r\n        graphs.each(function (index, element) {\r\n            var name = $(element).data('name');\r\n            var $elem = $(element);\r\n            $elem.addSpinner();\r\n            $.ajax('/site-analytics/' + name, { data: { from: $('#from').val(), to: $('#to').val() } })\r\n                .done(function (data) {\r\n                    if (name === 'traffic') {\r\n                        graphDisplayData[name] = mergeGroups(data.pageViews, data.visits, data.newVisits);\r\n                    } else {\r\n                        graphDisplayData[name] = data;\r\n                    }\r\n                    if (!hiddenLines.hasOwnProperty(name)) {\r\n                        hiddenLines[name] = [];\r\n                    }\r\n                    displayGraph($elem, graphDisplayData[name]);\r\n                })\r\n                .fail(function () {\r\n                    $elem.append('<div class=\"val-message val-error\"><p>' + _s('An error has occurred while retrieving data. Please try again in 24 hours.') + '</p></div>');\r\n                })\r\n                .always(function () {\r\n                    StackExchange.helpers.removeSpinner();\r\n                });\r\n        });\r\n    };\r\n\r\n    function zip(arrays) {\r\n        return arrays[0].map(function (_, i) {\r\n            return arrays.map(function (array) { return array[i] })\r\n        });\r\n    }\r\n\r\n    var getCsvForGraph = function (graphName) {\r\n        var graphData = graphDisplayData[graphName];\r\n\r\n        var header = ['date'];\r\n        var seriesData = $('#weekly').is(':checked') ? graphData.weekly : graphData.daily;\r\n        var csv = [header.concat(seriesData.map(function (d) { return d.localizedLabel; }))];\r\n\r\n        var dataPoints = seriesData.map(function(d) { return d.data; });\r\n\r\n        var joined = zip(dataPoints).map(function (d) {\r\n            var date = new Date(d[0][0]);\r\n            var merged = [$.datepicker.formatDate('yy-mm-dd', date)];\r\n            for (var i = 0; i < d.length; i++) {\r\n                merged.push(d[i][1]);\r\n            }\r\n            return merged;\r\n        });\r\n        csv = csv.concat(joined);\r\n        \r\n        return d3.csv.formatRows(csv);\r\n    };\r\n\r\n    var downloadCSV = function (csv, fileName) {\r\n        // https://stackoverflow.com/q/29304414\r\n\r\n        var a = document.createElement('a');\r\n        var mimeType = 'text/csv';\r\n\r\n        if (navigator.msSaveBlob) { // IE10\r\n            return navigator.msSaveBlob(new Blob([content], { type: mimeType }), fileName);\r\n        } else if ('download' in a) { //html5 A[download]\r\n            var blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });\r\n            var url = URL.createObjectURL(blob);\r\n            a.href = url;\r\n            a.setAttribute('download', fileName);\r\n            document.body.appendChild(a);\r\n            setTimeout(function () {\r\n                a.click();\r\n                document.body.removeChild(a);\r\n            }, 66);\r\n            return true;\r\n        } else { //do iframe dataURL download (old ch+FF):\r\n            var f = document.createElement('iframe');\r\n            document.body.appendChild(f);\r\n            f.src = 'data:' + mimeType + ',' + encodeURIComponent(content);\r\n\r\n            setTimeout(function () {\r\n                document.body.removeChild(f);\r\n            }, 333);\r\n            return true;\r\n        }\r\n    };\r\n\r\n    var init = function (options) {\r\n        var graphs = $('.graph');\r\n        buildGraphsAndGraphData(graphs);\r\n\r\n        graphs.on('change', 'input[type=checkbox]', function (e) {\r\n            var $graph = $(e.target).closest('.graph');\r\n            var graphName = $graph.data('name');\r\n            var series = e.target.name;\r\n\r\n            // todo: persist hidden series\r\n            e.target.checked ? hiddenLines[graphName].splice(hiddenLines[graphName].indexOf(series), 1) : hiddenLines[graphName].push(series);\r\n            displayGraph($graph, graphDisplayData[graphName]);\r\n        });\r\n\r\n        $('#weekly').change(function (e) {\r\n            // todo: persist weekly/daily view\r\n            weekly = e.target.checked;\r\n            reRenderAllGraphs(graphs);\r\n        });\r\n\r\n        $('#from').datepicker({ dateFormat: 'yy-M-dd', defaultDate: options.fromDate, maxDate: options.maxDate, minDate: options.minDate });\r\n        $('#to').datepicker({ dateFormat: 'yy-M-dd', defaultDate: options.toDate, maxDate: options.maxDate, minDate: options.minDate });\r\n\r\n        $('#from, #to').change(function() {\r\n            var graphs = $('.graph');\r\n            buildGraphsAndGraphData(graphs);\r\n        });\r\n\r\n        $(\".phaseDate\").click(function(e) {\r\n            var phaseDate = $(e.target).text();\r\n            var fromDate = $('#from').val();\r\n            if (fromDate !== phaseDate) {\r\n                $('#from').datepicker('setDate', phaseDate);\r\n                var graphs = $('.graph');\r\n                buildGraphsAndGraphData(graphs);\r\n            }\r\n            e.preventDefault();\r\n        });\r\n\r\n        $('.csv').click(function(e) {\r\n            var graphName = $(e.target).closest('.graph-container').find('.graph').data('name');\r\n\r\n            var from =$('#from').val();\r\n            var to = $('#to').val();\r\n            var fileName = \"{graphName}_{from}_{to}\".formatUnicorn({ graphName: graphName, from: from, to: to }) +\r\n                            ($('#weekly').is(':checked')?'_weekly':'') + '.csv';\r\n            var csv = getCsvForGraph(graphName);\r\n\r\n            downloadCSV(csv, fileName);\r\n        });\r\n    };\r\n\r\n    var renderPieChart = function (selector, data, options) {\r\n        options = options || {};\r\n        options.includeLegend = options.includeLegend || false;\r\n        options.resizeRatio = options.resizeRatio || 1;\r\n        options.includeTooltip = options.includeTooltip || false;\r\n        options.useCustomColors = options.useCustomColors || false;\r\n\r\n        var customColors = ['#ff7f0e', '#0095ff', '#68ba60'];\r\n\r\n        var pieWidth = 320 * options.resizeRatio, pieHeight = 320 * options.resizeRatio, radius = 150 * options.resizeRatio;\r\n        var color = d3.scale.category10();\r\n\r\n        var totalVisits = data.reduce(function(prev, curr, i, arr) {\r\n            return prev + curr.visits;\r\n        }, 0);\r\n\r\n        var svg = d3.selectAll(selector)\r\n        .append('svg:svg')\r\n        .data([data])\r\n        .attr('width', pieWidth + 150)\r\n        .attr('height', pieHeight)\r\n        .append('svg:g')\r\n        .attr('transform', 'translate(' + pieWidth / 2 + ',' + pieHeight / 2 + ')');\r\n\r\n        var arc = d3.svg.arc()\r\n            .outerRadius(radius);\r\n\r\n        var pie = d3.layout.pie()\r\n            .value(function (d) { return d.visits; });\r\n\r\n        var arcs = svg.selectAll('g.slice')\r\n            .data(pie)\r\n            .enter()\r\n            .append('svg:g')\r\n            .attr('class', 'slice');\r\n\r\n        arcs.append('svg:path')\r\n            .attr('fill', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); })\r\n            .attr('d', arc);\r\n\r\n        arcs.filter(function(d) {\r\n            return Math.abs(d.startAngle - d.endAngle) >= Math.PI / 6;\r\n            })\r\n            .append('svg:text')\r\n            .attr('transform', function(d) {\r\n                //we have to make sure to set these before calling arc.centroid\r\n                d.innerRadius = 0;\r\n                d.outerRadius = radius;\r\n                return 'translate(' + arc.centroid(d) + ')';\r\n            })\r\n            .attr('text-anchor', 'middle')\r\n            .attr('fill', 'white')\r\n            .text(function(d) { return d.data.source; });\r\n\r\n        if (options.includeTooltip) {\r\n            arcs.tooltip($(selector), {\r\n                reposition: function(d, i) {\r\n                    this.html(('<div>' +\r\n                        '<div class=\"series-color\" style=\"background-color: {color};\">&nbsp;</div>' +\r\n                        '<p>{source}</p>' +\r\n                        '<p>{value}</p>' +\r\n                        '</div>').formatUnicorn({\r\n                        source: d.data.source,\r\n                        value: commify(d.data.visits) + ' (' + (d.data.visits / totalVisits * 100).toFixed(1) + '%)',\r\n                        color: options.useCustomColors ? customColors[i] : color(i)\r\n                    }));\r\n\r\n                    return {\r\n                        left: d3.event.pageX + 10,\r\n                        top: d3.event.pageY - 10,\r\n                    };\r\n                }\r\n            });\r\n        }\r\n\r\n        if (options.includeLegend) {\r\n            var legendRectSize = 18;\r\n            var legendSpacing = 4;\r\n\r\n            var legend = svg.selectAll('.legend')\r\n                .data(data)\r\n                .enter()\r\n                .append('svg:g')\r\n                .attr('class', 'legend')\r\n                .attr('transform', function(d, i) {\r\n                    var rectHeight = legendRectSize + legendSpacing;\r\n                    var offset = rectHeight * data.length / 2;\r\n                    var horz = pieWidth / 2 + 5;\r\n                    var vert = i * offset - pieHeight / 2 + 10;\r\n                    return 'translate(' + horz + ',' + vert + ')';\r\n                });\r\n\r\n            legend.append('svg:rect')\r\n                .attr('width', legendRectSize)\r\n                .attr('height', legendRectSize)\r\n                .style('fill', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); })\r\n                .style('stroke', function (d, i) { return options.useCustomColors ? customColors[i] : color(i); });\r\n\r\n            legend.append('svg:text')\r\n                .attr('x', legendRectSize + legendSpacing)\r\n                .attr('y', legendRectSize - legendSpacing)\r\n                .attr('font-size', 10)\r\n                .text(function (d) { return d.source + ': ' + commify(d.visits) + ' (' + (d.visits / totalVisits * 100).toFixed(1) + '%)'; });\r\n        }\r\n    };\r\n\r\n    var renderTable = function (selector, data) {\r\n        var rows = '';\r\n        $.each(data, function(i, d) {\r\n            rows += '<tr><td>' + d.source + '</td><td>' + commify(d.visits) + '</td><tr>';\r\n        });\r\n        $(selector).append(rows);\r\n    };\r\n\r\n    return { init: init, renderPieChart: renderPieChart, renderTable: renderTable };\r\n}();"],"names":[],"sourceRoot":""}