function circleLegend(selection, domain, range) {
let instance = {}
const api = {
domain: domain,
range: range,
values: [8, 34, 89],
x: 500,
y: 500,
suffix:'',
circleColor: '#888',
textPadding: 80,
textColor: '#454545',
fontSize: 14,
fontFamily: 'Arial',
format: d3.format("~g"),
title: 'Default title'
}
const sqrtScale = d3.scaleSqrt()
.domain(api.domain)
.range(api.range)
instance.render = function () {
const s = selection.append('g')
.attr('class', 'legend-wrap')
.attr('transform', 'translate(0,' + sqrtScale(d3.max(Math.abs(api.values))) + ')')
s.append('g')
.attr('class', 'values-wrap')
.selectAll('circle')
.data(api.values)
.enter().append('circle')
.attr('class', d => 'values values-' + d)
.attr('r', d => Math.abs(sqrtScale(d)))
.attr('cx', api.x)
.attr('cy', d => api.y - Math.abs(sqrtScale(d)))
.style('fill', 'none')
.style('stroke', api.circleColor)
.style('opacity', 0.9)
s.append('g')
.attr('class', 'values-line-wrap')
.selectAll('.values-labels')
.data(api.values)
.enter().append('line')
.attr('x1', d => api.x + sqrtScale(d))
.attr('x2', api.x + Math.abs(sqrtScale(Math.max(Math.abs(api.domain[0]),Math.abs(api.domain[1])))) + 14)
.attr('y1', d => api.y - Math.abs(sqrtScale(d)))
.attr('y2', d => api.y - Math.abs(sqrtScale(d)))
.style('stroke', api.textColor)
.style('stroke-dasharray', ('2,2'))
s.append('g')
.attr('class', 'values-labels-wrap')
.selectAll('.values-labels')
.data(api.values)
.enter().append('text')
.attr('x', api.x + Math.abs(sqrtScale(Math.max(Math.abs(api.domain[0]),Math.abs(api.domain[1])))) + 14 + api.textPadding)
.attr('y', d => (api.y - Math.abs(sqrtScale(d))) + 4)
.attr('shape-rendering', 'crispEdges')
.style('text-anchor', 'end')
.style('fill', api.textColor)
.attr('font-family', api.fontFamily)
.attr('font-size', api.fontSize)
.text(d => api.format(d) + api.suffix)
s.append('g')
.selectAll('text')
.data(api.values)
.enter().append('text')
.attr('x', api.x )
.attr('y', api.y + 20)
.attr('font-family', api.fontFamily)
.attr('font-size', api.fontSize + 4)
.style('text-anchor', 'middle')
.text(d => api.title)
return instance
}
for (let key in api) {
instance[key] = getSet(key, instance).bind(api)
}
return instance
function getSet(option, component) {
return function (_) {
if (! arguments.length) {
return this[option];
}
this[option] = _;
return component;
}
}
}