(function() { var bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; }; this.Calendar = (function() { function Calendar(timestamps, calendar_activities_path) { this.calendar_activities_path = calendar_activities_path; this.clickDay = bind(this.clickDay, this); this.currentSelectedDate = ''; this.daySpace = 1; this.daySize = 15; this.daySizeWithSpace = this.daySize + (this.daySpace * 2); this.monthNames = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; this.months = []; // Loop through the timestamps to create a group of objects // The group of objects will be grouped based on the day of the week they are this.timestampsTmp = []; var group = 0; var today = new Date() today.setHours(0, 0, 0, 0, 0); var oneYearAgo = new Date(today); oneYearAgo.setFullYear(today.getFullYear() - 1); var days = gl.utils.getDayDifference(oneYearAgo, today); for(var i = 0; i <= days; i++) { var date = new Date(oneYearAgo); date.setDate(date.getDate() + i); var day = date.getDay(); var count = timestamps[dateFormat(date, 'yyyy-mm-dd')]; // Create a new group array if this is the first day of the week // or if is first object if ((day === 0 && i !== 0) || i === 0) { this.timestampsTmp.push([]); group++; } var innerArray = this.timestampsTmp[group - 1]; // Push to the inner array the values that will be used to render map innerArray.push({ count: count || 0, date: date, day: day }); } // Init color functions this.colorKey = this.initColorKey(); this.color = this.initColor(); // Init the svg element this.renderSvg(group); this.renderDays(); this.renderMonths(); this.renderDayTitles(); this.renderKey(); this.initTooltips(); } // Add extra padding for the last month label if it is also the last column Calendar.prototype.getExtraWidthPadding = function(group) { var extraWidthPadding = 0; var lastColMonth = this.timestampsTmp[group - 1][0].date.getMonth(); var secondLastColMonth = this.timestampsTmp[group - 2][0].date.getMonth(); if (lastColMonth != secondLastColMonth) { extraWidthPadding = 3; } return extraWidthPadding; } Calendar.prototype.renderSvg = function(group) { var width = (group + 1) * this.daySizeWithSpace + this.getExtraWidthPadding(group); return this.svg = d3.select('.js-contrib-calendar').append('svg').attr('width', width).attr('height', 167).attr('class', 'contrib-calendar'); }; Calendar.prototype.renderDays = function() { return this.svg.selectAll('g').data(this.timestampsTmp).enter().append('g').attr('transform', (function(_this) { return function(group, i) { _.each(group, function(stamp, a) { var lastMonth, lastMonthX, month, x; if (a === 0 && stamp.day === 0) { month = stamp.date.getMonth(); x = (_this.daySizeWithSpace * i + 1) + _this.daySizeWithSpace; lastMonth = _.last(_this.months); if (lastMonth != null) { lastMonthX = lastMonth.x; } if (lastMonth == null) { return _this.months.push({ month: month, x: x }); } else if (month !== lastMonth.month && x - _this.daySizeWithSpace !== lastMonthX) { return _this.months.push({ month: month, x: x }); } } }); return "translate(" + ((_this.daySizeWithSpace * i + 1) + _this.daySizeWithSpace) + ", 18)"; }; })(this)).selectAll('rect').data(function(stamp) { return stamp; }).enter().append('rect').attr('x', '0').attr('y', (function(_this) { return function(stamp, i) { return _this.daySizeWithSpace * stamp.day; }; })(this)).attr('width', this.daySize).attr('height', this.daySize).attr('title', (function(_this) { return function(stamp) { var contribText, date, dateText; date = new Date(stamp.date); contribText = 'No contributions'; if (stamp.count > 0) { contribText = stamp.count + " contribution" + (stamp.count > 1 ? 's' : ''); } dateText = dateFormat(date, 'mmm d, yyyy'); return contribText + "<br />" + (gl.utils.getDayName(date)) + " " + dateText; }; })(this)).attr('class', 'user-contrib-cell js-tooltip').attr('fill', (function(_this) { return function(stamp) { if (stamp.count !== 0) { return _this.color(Math.min(stamp.count, 40)); } else { return '#ededed'; } }; })(this)).attr('data-container', 'body').on('click', this.clickDay); }; Calendar.prototype.renderDayTitles = function() { var days; days = [ { text: 'M', y: 29 + (this.daySizeWithSpace * 1) }, { text: 'W', y: 29 + (this.daySizeWithSpace * 3) }, { text: 'F', y: 29 + (this.daySizeWithSpace * 5) } ]; return this.svg.append('g').selectAll('text').data(days).enter().append('text').attr('text-anchor', 'middle').attr('x', 8).attr('y', function(day) { return day.y; }).text(function(day) { return day.text; }).attr('class', 'user-contrib-text'); }; Calendar.prototype.renderMonths = function() { return this.svg.append('g').selectAll('text').data(this.months).enter().append('text').attr('x', function(date) { return date.x; }).attr('y', 10).attr('class', 'user-contrib-text').text((function(_this) { return function(date) { return _this.monthNames[date.month]; }; })(this)); }; Calendar.prototype.renderKey = function() { var keyColors; keyColors = ['#ededed', this.colorKey(0), this.colorKey(1), this.colorKey(2), this.colorKey(3)]; return this.svg.append('g').attr('transform', "translate(18, " + (this.daySizeWithSpace * 8 + 16) + ")").selectAll('rect').data(keyColors).enter().append('rect').attr('width', this.daySize).attr('height', this.daySize).attr('x', (function(_this) { return function(color, i) { return _this.daySizeWithSpace * i; }; })(this)).attr('y', 0).attr('fill', function(color) { return color; }); }; Calendar.prototype.initColor = function() { var colorRange; colorRange = ['#ededed', this.colorKey(0), this.colorKey(1), this.colorKey(2), this.colorKey(3)]; return d3.scale.threshold().domain([0, 10, 20, 30]).range(colorRange); }; Calendar.prototype.initColorKey = function() { return d3.scale.linear().range(['#acd5f2', '#254e77']).domain([0, 3]); }; Calendar.prototype.clickDay = function(stamp) { var formatted_date; if (this.currentSelectedDate !== stamp.date) { this.currentSelectedDate = stamp.date; formatted_date = this.currentSelectedDate.getFullYear() + "-" + (this.currentSelectedDate.getMonth() + 1) + "-" + this.currentSelectedDate.getDate(); return $.ajax({ url: this.calendar_activities_path, data: { date: formatted_date }, cache: false, dataType: 'html', beforeSend: function() { return $('.user-calendar-activities').html('<div class="text-center"><i class="fa fa-spinner fa-spin user-calendar-activities-loading"></i></div>'); }, success: function(data) { return $('.user-calendar-activities').html(data); } }); } else { return $('.user-calendar-activities').html(''); } }; Calendar.prototype.initTooltips = function() { return $('.js-contrib-calendar .js-tooltip').tooltip({ html: true }); }; return Calendar; })(); }).call(this);