wheelChart = function(data, style) {
const pi = Math.PI,
mins = d3.range(0, 1440);
const properties = {
width: width,
height: style.height,
margin: style.margin,
arc_thickness: style.thickness,
radius_padding: style.padding
},
center = {
x: properties.width / 2,
y: properties.height / 2
},
clockRadius = (properties.height - (4 * properties.margin)) / 2,
clockInnerRadius = clockRadius - properties.arc_thickness,
angle_domain = d3.extent(mins);
let angleScale = d3.scaleLinear()
.domain(angle_domain)
.range([0, 2*pi]);
let colorScale = function(n) {
var color;
data.map(function(d) {
if (n >= d.start.minute && n <= d.end.minute)
color = d3.interpolateRgb(d.start.color, d.end.color)
((n - d.start.minute) / (d.end.minute - d.start.minute));
}); return color;
}
let arc = d3.arc()
.innerRadius(clockInnerRadius)
.outerRadius(clockRadius)
.startAngle(d => angleScale(d))
.endAngle(d => angleScale(d) + pi*2 / mins.length);
let remaining_arc = d3.arc()
.innerRadius(clockInnerRadius - properties.arc_thickness)
.outerRadius(clockInnerRadius)
.startAngle(d => angleScale(currentDateTime))
.endAngle(d => angleScale(daylight_remaining.end));
let lineRadial = d3.lineRadial();
var WheelChart = {};
WheelChart.addCalendar = function (cal, el, tooltip) {
var selected_valence =
use_valences == "One Valence" ? 1 : use_valences == "No Valences" ? 0 : cal.valence;
var outer = clockRadius + selected_valence*(properties.radius_padding + properties.arc_thickness),
inner = outer - properties.arc_thickness;
var arc_function = d3.arc()
.innerRadius(inner)
.outerRadius(outer)
.startAngle(d => angleScale(d.start))
.endAngle(d => angleScale(d.end));
el.append("g")
.attr("class", "calendar")
.selectAll("path")
.data(cal.events)
.enter()
.append("path")
.attr("d", arc_function)
.attr("fill", d => cal.color)
.attr("opacity", 0.8)
.attr("stroke-width", 1)
.attr("stroke", cal.color)
.on("mouseover", function(d) {
d3.selectAll(".tooltip").style("visibility", "hidden");
tooltip
.style("visibility", "visible")
.text(cal.name + ' - ' + d.name);
d3.select(this)
.attr("stroke", "black");
}).on("mousemove", function() {
tooltip
.style("top", (d3.event.pageY-10)+"px")
.style("left",(d3.event.pageX+20)+"px");
}).on("mouseout", function() {
tooltip.style("visibility", "hidden");
d3.select(this).attr("stroke-width", 0);
});
}
var svg = d3.create("svg")
.attr("viewBox", [0, 0, properties.width, properties.height]);
const tooltip = d3.select("body").append("div")
.attr("class", "tooltip")
.style("position", "absolute")
.style("visibility", "hidden")
.style("font-size", "1rem")
.style("background", "rgba(255,255,255,0.8)")
.style("padding", "0.1rem 0.5rem")
.style("color", "#000")
.style("border-radius", "3px")
.style("border", "1px solid #000");
var container = svg.append("g")
.attr("class", "chart-container")
.attr("transform", `translate(${center.x}, ${center.y})`);
var wheel = container.append("g")
.attr("class", "wheel")
.selectAll("path")
.data(mins)
.enter()
.append("path")
.attr("d", arc)
.attr("fill", d => colorScale(d))
.attr("stroke", d => colorScale(d));
if (currentDateTime <= daylight_remaining.end && currentDateTime >= daylight_remaining.start) {
container.append("g")
.append("path")
.attr("d", remaining_arc)
.attr("fill", d => "#D00000")
.attr("opacity", 0.15);
container.append("g")
.attr("class", "light-left")
.selectAll("text")
.data([currentDateTime])
.enter()
.append("text")
.attr("transform", d =>`translate(
${(clockInnerRadius - 30) * Math.sin(angleScale(daylight_remaining.half_angle))},
${-(clockInnerRadius - 30) * Math.cos(angleScale(daylight_remaining.half_angle))})`)
.attr("dy", "0.35em")
.attr("font-weight", 700)
.attr("font-size", 24)
.text(daylight_remaining.left + '%');
}
container.append("g")
.attr("class", "needle")
.selectAll("path")
.data([currentDateTime])
.enter()
.append("path")
.attr("d", d => lineRadial([[0, 0], [angleScale(d), clockInnerRadius]]))
.attr("stroke", "black")
.attr("stroke-width", 10)
.attr("stroke-linecap", "round");
container.select("g.needle")
.append("circle")
.attr("cx", 0)
.attr("cy", 0)
.attr("r", 20)
.attr("stroke", "black")
.attr("stroke-width", 10)
.style("fill", "white");
WheelChart.svg = svg;
WheelChart.container = container;
WheelChart.tooltip = tooltip;
return WheelChart;
}