Skip to content

Instantly share code, notes, and snippets.

@jrabary
Created June 12, 2015 15:56
Show Gist options
  • Save jrabary/a724851815e7d5f55b3e to your computer and use it in GitHub Desktop.
Save jrabary/a724851815e7d5f55b3e to your computer and use it in GitHub Desktop.
Scatter with image tooltip
var d3 = require('d3');
require('d3-multiaxis-zoom')(d3);
var inherits = require('inherits');
var utils = require('lightning-client-utils');
var _ = require('lodash');
var colorbrewer = require('colorbrewer');
var TooltipPlugin = require('d3-tip');
TooltipPlugin(d3);
var CustomScatter = function(selector, data, images, opts) {
var margin = {
top: 0,
right: 0,
bottom: 20,
left: 45
};
var defaults = {
tooltips: true
};
opts = _.defaults(opts || {}, defaults);
this.opts = opts
this.data = this._formatData(data)
if(_.has(this.data, 'xaxis')) {
margin.bottom = 57;
}
if(_.has(this.data, 'yaxis')) {
margin.left = 70;
}
this.width = (opts.width || $(selector).width()) - margin.left - margin.right;
this.height = Math.min(($(selector).height() || Infinity), (opts.height || (this.width * 0.6))) - margin.top - margin.bottom;
this.selector = selector;
this.defaultFill = '#deebfa'
this.defaultStroke = '#68a1e5'
this.defaultSize = 8
this.defaultAlpha = 0.9
this.margin = margin
this._init();
};
inherits(CustomScatter, require('events').EventEmitter);
CustomScatter.prototype._init = function() {
var tip;
if(this.opts.tooltips) {
var format = d3.format('.02f');
tip = d3.tip()
.attr('class', 'd3-tip')
.html(function(d) {
return '<img src="' + d.t + '"/><br>Label : ' + d.l ;
});
}
var data = this.data
var height = this.height
var width = this.width
var opts = this.opts
var selector = this.selector
var margin = this.margin
var self = this
var points = data.points
var xDomain = d3.extent(points, function(d) {
return d.x;
});
var yDomain = d3.extent(points, function(d) {
return d.y;
});
var sizeMax = d3.max(points, function(d) {
return d.s;
});
if (sizeMax) {
var padding = sizeMax / 2
} else {
var padding = self.defaultSize / 2
}
var xRange = xDomain[1] - xDomain[0]
var yRange = yDomain[1] - yDomain[0]
this.x = d3.scale.linear()
.domain([xDomain[0] - xRange * 0.1, xDomain[1] + xRange * 0.1])
.range([0 + padding, width - padding]);
this.y = d3.scale.linear()
.domain([yDomain[0] - yRange * 0.1, yDomain[1] + yRange * 0.1])
.range([height - padding , 0 + padding]);
var zoom = d3.behavior.zoom()
.x(this.x)
.y(this.y)
.on('zoom', zoomed);
var svg = d3.select(selector)
.append('svg')
.attr('class', 'scatter-plot')
.attr('width', width + margin.left + margin.right)
.attr('height', height + margin.top + margin.bottom)
.append('svg:g')
.attr('transform', 'translate(' + margin.left + ',' + margin.top + ')')
.call(zoom);
if(this.opts.tooltips) {
svg.call(tip);
}
svg.append('rect')
.attr('width', width)
.attr('height', height)
.attr('class', 'plot');
var makeXAxis = function () {
return d3.svg.axis()
.scale(self.x)
.orient('bottom')
.ticks(5);
};
var makeYAxis = function () {
return d3.svg.axis()
.scale(self.y)
.orient('left')
.ticks(5);
};
this.xAxis = d3.svg.axis()
.scale(self.x)
.orient('bottom')
.ticks(5);
svg.append('g')
.attr('class', 'x axis')
.attr('transform', 'translate(0, ' + height + ')')
.call(self.xAxis);
this.yAxis = d3.svg.axis()
.scale(self.y)
.orient('left')
.ticks(5);
svg.append('g')
.attr('class', 'y axis')
.call(self.yAxis);
svg.append('g')
.attr('class', 'x grid')
.attr('transform', 'translate(0,' + height + ')')
.call(makeXAxis()
.tickSize(-height, 0, 0)
.tickFormat(''));
svg.append('g')
.attr('class', 'y grid')
.call(makeYAxis()
.tickSize(-width, 0, 0)
.tickFormat(''));
var clipId = utils.getUniqueId();
svg.append('svg:clipPath')
.attr('id', clipId)
.append('svg:rect')
.attr('x', 0)
.attr('y', 0)
.attr('width', width)
.attr('height', height);
var chartBody = svg.append('g')
.attr('clip-path', 'url(#' + clipId + ')');
function darken(d, i) {
if(self.opts.tooltips) {
tip.show(d, i);
}
var point = d3.select(this)
var newcolor = d3.hsl(point.style('fill')).darker(0.5)
point.style('fill', d3.rgb(newcolor))
self.emit('hover', d);
console.log('in: ' + i);
}
function brighten(d, i) {
if(self.opts.tooltips) {
tip.hide(d, i);
}
var point = d3.select(this)
var newcolor = d3.hsl(point.style('fill')).brighter(0.5)
point.style('fill', d3.rgb(newcolor))
console.log('out: ' + i);
}
chartBody.selectAll('.dot')
.data(points)
.enter().append('circle')
.attr('class', 'dot')
.attr('r', function(d) { return (d.s ? d.s: self.defaultSize)})
.attr('transform', function(d) {
return 'translate(' + self.x(d.x) + ',' + self.y(d.y) + ')';
})
.style('fill',function(d) { return (d.c ? d.c : self.defaultFill);})
.style('stroke',function(d) { return (d.c ? d.c.darker(0.75) : self.defaultStroke);})
.style('fill-opacity',function(d) { return (d.a ? d.a : self.defaultAlpha);})
.style('stroke-opacity',function(d) { return (d.a ? d.a : self.defaultAlpha);})
.on('mouseover', darken)
.on('mouseout', brighten);
function zoomed() {
svg.select('.x.axis').call(self.xAxis);
svg.select('.y.axis').call(self.yAxis);
svg.select('.x.grid')
.call(makeXAxis()
.tickSize(-height, 0, 0)
.tickFormat(''));
svg.select('.y.grid')
.call(makeYAxis()
.tickSize(-width, 0, 0)
.tickFormat(''));
svg.selectAll('circle')
.attr('transform', function(d) {
return 'translate(' + self.x(d.x) + ',' + self.y(d.y) + ')';
});
}
if(_.has(this.data, 'xaxis')) {
var txt = this.data.xaxis;
if(_.isArray(txt)) {
txt = txt[0];
}
svg.append("text")
.attr("class", "x label")
.attr("text-anchor", "middle")
.attr("x", width / 2)
.attr("y", height + margin.bottom - 5)
.text(txt);
}
if(_.has(this.data, 'yaxis')) {
var txt = this.data.yaxis;
if(_.isArray(txt)) {
txt = txt[0];
}
svg.append("text")
.attr("class", "y label")
.attr("text-anchor", "middle")
.attr("transform", "rotate(-90)")
.attr("x", - height / 2)
.attr("y", -50)
.text(txt);
}
this.brighten = brighten;
this.darken = darken;
this.svg = svg;
this.points = points;
}
CustomScatter.prototype._formatData = function(data) {
retColor = utils.getColorFromData(data);
retSize = data.size || [];
retAlpha = data.alpha || [];
var labels = data.label || [];
var thumbnails = data.thumbnails || []
console.log(thumbnails);
var colorScale = d3.scale.category20c();
data.points = data.points.map(function(d, i) {
d.x = d[0];
d.y = d[1];
d.i = i;
d.l = labels[i];
d.c = d3.rgb(colorScale(d.l));
d.t = thumbnails.length > 1 ? thumbnails[i] : null
d.s = retSize.length > 1 ? retSize[i] : retSize[0];
d.a = retAlpha.length > 1 ? retAlpha[i] : retAlpha[0];
return d;
})
console.log(data.points);
return data;
};
module.exports = CustomScatter;
[
{
"name": "sample",
"data": {
"points": [[0,0],[10,10], [-10, 10], [-10, -10]],
"label": [1,2,3,4],
"thumbnails": [
"",
"",
"",
""
]
}
}
]
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment