-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathtime_selection_bar.js
139 lines (113 loc) · 4.52 KB
/
time_selection_bar.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
// TODO: Clean up namespace
var bg_colour = "#AAF"; // Light blue
var line_colour = "#B47570"; // Less light blue
var max_val = 1200; // 20 min
function draw_time_bg () {
var canvas = document.getElementById("time_data");
canvas.width = 120;
var ctx = canvas.getContext("2d");
var curve_end = 70;
// Fill under the curve
ctx.beginPath();
ctx.moveTo(0,2);
ctx.quadraticCurveTo(curve_end*0.8, canvas.height*0.03, curve_end, canvas.height);
ctx.lineTo(curve_end, 2);
ctx.lineTo(0,2);
ctx.closePath();
var gradient = ctx.createLinearGradient(0,0,curve_end,0);
gradient.addColorStop(0, line_colour);
gradient.addColorStop(1, "white");
ctx.fillStyle = gradient;
ctx.fill();
// Light line to close it off
ctx.lineWidth = 1;
ctx.strokeStyle = bg_colour;
ctx.stroke();
ctx.beginPath();
ctx.moveTo(0,2);
ctx.quadraticCurveTo(curve_end*0.8, canvas.height*0.03, curve_end, canvas.height);
// Draw the curve
ctx.lineWidth = 2;
ctx.strokeStyle = line_colour;
ctx.stroke();
}
/* I played with a bunch of different functions before landing on this one.
*
* Linear feels better to click - it is more intuitive - but the range looks weird
* Essentially I want to go up to 20min but the most common points are around 1 min.
*
* Square-root like shapes are ok, best was: return Math.pow(v-elbow,1/1.5);
* Log gave a slightly better distribution
*
* I eventually landed on using both log and linear with a transition point at 1 minute.
* The most common values are around the 1 minute mark, forcing it to be the middle of the range
* is a recognition of that and makes it easy to select those region of values.
*
* Making the lower half linear means that clicking is intuitive and the full range is easily accessible.
* The nature of the log function is that it kicks in almost linearly, so the 1-5min area feels natural too.
*
* Using log allows 10-20 minute values to be selected without distorting the entire scale.
*
* The background image is designed to get across the concept that the scale isn't linear,
* in addition to doing the impossible in representing the passage of time.
*/
function v2m(v) {
// Fix elbow in middle of range
var elbow = 60;
return v > elbow ? v2m(elbow) + Math.log((v-elbow)/50+1) : Math.log((max_val-elbow)/50+1)/elbow*v;
}
function m2v(m) {
// Reverse v2m - to map clicks to values
var elbow_v = 60;
var elbow = v2m(elbow_v);
return m > elbow ? elbow_v + (Math.exp(m-elbow)-1)*50 : m/elbow*elbow_v;
}
function val2y (v) {
// Scale, 1sec - 1hr
var svg = document.getElementById("time_data");
return (1-v2m(v)/v2m(max_val))*svg.getBoundingClientRect().height;
}
function y2val (y) {
var svg = document.getElementById("time_data");
var m = (1-y/svg.getBoundingClientRect().height)*v2m(max_val);
var raw_val = m2v(m);
// To make people happy, we want to bias towards pretty numbers
// There is no practical difference between 60 and 63 seconds...
// But if somebody pushes the 1 min button they expect 60 seconds
if (raw_val > 200 && (raw_val + 100) % 300 < 200) {
return Math.round(raw_val/300)*300; // Nearest 5 minutes
} else if (raw_val > 50 && (raw_val + 10) % 60 < 20) {
return Math.round(raw_val/60)*60; // Nearest minute
} else if (raw_val > 5 && (raw_val + 5) % 10 < 10) {
return Math.round(raw_val/10)*10; // Nearest 10 seconds
} else {
return Math.round(raw_val);
}
}
function time_label(v, label) {
var y = (1-v2m(v)/v2m(max_val))*100;
// Need to undo the viewport scaling, to prevent text distortion
// Viewport is 200x100
var hs = $('#time_data_labels').parents('svg').height()/100;
var ws = $('#time_data_labels').parents('svg').width()/200;
var newText = document.createElementNS("http://www.w3.org/2000/svg","text");
newText.setAttributeNS(null,"text-anchor","end");
newText.setAttributeNS(null,"x",100/hs); // Because of right/end align
newText.setAttributeNS(null,"y",y/ws); // Unsure why this is scaled by width...
newText.setAttributeNS(null,"transform","scale("+hs+","+ws+")");
var textNode = document.createTextNode(label);
newText.appendChild(textNode);
document.getElementById("time_data_labels").appendChild(newText);
}
time_label(10, "10 sec");
time_label(30, "30 sec");
time_label(60, "1 min");
time_label(5*60, "5 min");
time_label(10*60, "10 min");
document.getElementById("time_data").addEventListener('click', function(event) {
if (event.button) return; // Not left-button, ignore
var y = event.pageY - this.getBoundingClientRect().y; // Get local coordinates
var v = y2val(y);
console.log("Click", y, v, v/60);
window.location.href = '/time/'+v;
});