-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlargestTriangleThreeBuckets.js
75 lines (58 loc) · 2.67 KB
/
largestTriangleThreeBuckets.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
/*
* The MIT License
Copyright (c) 2018 Alexufo . Original code 2013 by Sveinn Steinarsson
*/
function largestTriangleThreeBuckets(data, threshold) {
// data is linear array [90,87,97...]
var data_length = data.length;
if (threshold >= data_length || threshold === 0) {
return data; // Nothing to do
}
var sampled = [],
sampled_index = 0;
// Bucket size. Leave room for start and end data points
var every = (data_length - 2) / (threshold - 2);
var a = 0, // Initially a is the first point in the triangle
max_area_point,
max_area,
area,
next_a;
sampled[ sampled_index++ ] = data[ a ]; // Always add the first point
for (var i = 0; i < threshold - 2; i++) {
// Calculate point average for next bucket (containing c)
var avg_x = 0,
avg_y = 0,
avg_range_start = Math.floor( ( i + 1 ) * every ) + 1,
avg_range_end = Math.floor( ( i + 2 ) * every ) + 1;
avg_range_end = avg_range_end < data_length ? avg_range_end : data_length;
var avg_range_length = avg_range_end - avg_range_start;
for ( ; avg_range_start<avg_range_end; avg_range_start++ ) {
avg_x += avg_range_start;
avg_y += data[ avg_range_start ] * 1;// * 1 enforces Number (value may be Date)
}
avg_x /= avg_range_length;
avg_y /= avg_range_length;
// Get the range for this bucket
var range_offs = Math.floor( (i + 0) * every ) + 1,
range_to = Math.floor( (i + 1) * every ) + 1;
// Point a
var point_a_x = a * 1, // enforce Number (value may be Date)
point_a_y = data[ a ] * 1;
max_area = area = -1;
for ( ; range_offs < range_to; range_offs++ ) {
// Calculate triangle area over three buckets
area = Math.abs( ( point_a_x - avg_x ) * ( data[ range_offs ] - point_a_y ) -
( point_a_x - range_offs ) * ( avg_y - point_a_y )
) * 0.5;
if ( area > max_area ) {
max_area = area;
max_area_point = data[ range_offs ];
next_a = range_offs; // Next a is this b
}
}
sampled[ sampled_index++ ] = max_area_point; // Pick this point from the bucket
a = next_a; // This a is the next a (chosen b)
}
sampled[ sampled_index++ ] = data[ data_length - 1 ]; // Always add last
return sampled;
}