flot-rails 0.0.2 → 0.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/README.md +2 -2
- data/lib/flot/rails/version.rb +2 -2
- data/vendor/assets/javascripts/excanvas.js +1428 -1427
- data/vendor/assets/javascripts/excanvas.min.js +1 -1
- data/vendor/assets/javascripts/jquery.colorhelpers.js +179 -179
- data/vendor/assets/javascripts/jquery.colorhelpers.min.js +21 -1
- data/vendor/assets/javascripts/jquery.flot.canvas.js +317 -0
- data/vendor/assets/javascripts/jquery.flot.canvas.min.js +28 -0
- data/vendor/assets/javascripts/jquery.flot.categories.js +190 -0
- data/vendor/assets/javascripts/jquery.flot.categories.min.js +44 -0
- data/vendor/assets/javascripts/jquery.flot.crosshair.js +176 -167
- data/vendor/assets/javascripts/jquery.flot.crosshair.min.js +59 -1
- data/vendor/assets/javascripts/jquery.flot.errorbars.js +353 -0
- data/vendor/assets/javascripts/jquery.flot.errorbars.min.js +63 -0
- data/vendor/assets/javascripts/jquery.flot.fillbetween.js +226 -183
- data/vendor/assets/javascripts/jquery.flot.fillbetween.min.js +30 -1
- data/vendor/assets/javascripts/jquery.flot.image.js +241 -238
- data/vendor/assets/javascripts/jquery.flot.image.min.js +53 -1
- data/vendor/assets/javascripts/jquery.flot.js +2980 -2599
- data/vendor/assets/javascripts/jquery.flot.min.js +26 -4
- data/vendor/assets/javascripts/jquery.flot.navigate.js +345 -336
- data/vendor/assets/javascripts/jquery.flot.navigate.min.js +96 -1
- data/vendor/assets/javascripts/jquery.flot.pie.js +561 -499
- data/vendor/assets/javascripts/jquery.flot.pie.min.js +56 -1
- data/vendor/assets/javascripts/jquery.flot.resize.js +60 -60
- data/vendor/assets/javascripts/jquery.flot.resize.min.js +21 -1
- data/vendor/assets/javascripts/jquery.flot.selection.js +360 -344
- data/vendor/assets/javascripts/jquery.flot.selection.min.js +79 -1
- data/vendor/assets/javascripts/jquery.flot.stack.js +188 -184
- data/vendor/assets/javascripts/jquery.flot.stack.min.js +36 -1
- data/vendor/assets/javascripts/jquery.flot.symbol.js +71 -70
- data/vendor/assets/javascripts/jquery.flot.symbol.min.js +14 -1
- data/vendor/assets/javascripts/jquery.flot.threshold.js +142 -103
- data/vendor/assets/javascripts/jquery.flot.threshold.min.js +43 -1
- data/vendor/assets/javascripts/jquery.flot.time.js +424 -0
- data/vendor/assets/javascripts/jquery.flot.time.min.js +9 -0
- metadata +25 -7
@@ -1,183 +1,226 @@
|
|
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
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
1
|
+
/* Flot plugin for computing bottoms for filled line and bar charts.
|
2
|
+
|
3
|
+
Copyright (c) 2007-2013 IOLA and Ole Laursen.
|
4
|
+
Licensed under the MIT license.
|
5
|
+
|
6
|
+
The case: you've got two series that you want to fill the area between. In Flot
|
7
|
+
terms, you need to use one as the fill bottom of the other. You can specify the
|
8
|
+
bottom of each data point as the third coordinate manually, or you can use this
|
9
|
+
plugin to compute it for you.
|
10
|
+
|
11
|
+
In order to name the other series, you need to give it an id, like this:
|
12
|
+
|
13
|
+
var dataset = [
|
14
|
+
{ data: [ ... ], id: "foo" } , // use default bottom
|
15
|
+
{ data: [ ... ], fillBetween: "foo" }, // use first dataset as bottom
|
16
|
+
];
|
17
|
+
|
18
|
+
$.plot($("#placeholder"), dataset, { lines: { show: true, fill: true }});
|
19
|
+
|
20
|
+
As a convenience, if the id given is a number that doesn't appear as an id in
|
21
|
+
the series, it is interpreted as the index in the array instead (so fillBetween:
|
22
|
+
0 can also mean the first series).
|
23
|
+
|
24
|
+
Internally, the plugin modifies the datapoints in each series. For line series,
|
25
|
+
extra data points might be inserted through interpolation. Note that at points
|
26
|
+
where the bottom line is not defined (due to a null point or start/end of line),
|
27
|
+
the current line will show a gap too. The algorithm comes from the
|
28
|
+
jquery.flot.stack.js plugin, possibly some code could be shared.
|
29
|
+
|
30
|
+
*/
|
31
|
+
|
32
|
+
(function ( $ ) {
|
33
|
+
|
34
|
+
var options = {
|
35
|
+
series: {
|
36
|
+
fillBetween: null // or number
|
37
|
+
}
|
38
|
+
};
|
39
|
+
|
40
|
+
function init( plot ) {
|
41
|
+
|
42
|
+
function findBottomSeries( s, allseries ) {
|
43
|
+
|
44
|
+
var i;
|
45
|
+
|
46
|
+
for ( i = 0; i < allseries.length; ++i ) {
|
47
|
+
if ( allseries[ i ].id === s.fillBetween ) {
|
48
|
+
return allseries[ i ];
|
49
|
+
}
|
50
|
+
}
|
51
|
+
|
52
|
+
if ( typeof s.fillBetween === "number" ) {
|
53
|
+
if ( s.fillBetween < 0 || s.fillBetween >= allseries.length ) {
|
54
|
+
return null;
|
55
|
+
}
|
56
|
+
return allseries[ s.fillBetween ];
|
57
|
+
}
|
58
|
+
|
59
|
+
return null;
|
60
|
+
}
|
61
|
+
|
62
|
+
function computeFillBottoms( plot, s, datapoints ) {
|
63
|
+
|
64
|
+
if ( s.fillBetween == null ) {
|
65
|
+
return;
|
66
|
+
}
|
67
|
+
|
68
|
+
var other = findBottomSeries( s, plot.getData() );
|
69
|
+
|
70
|
+
if ( !other ) {
|
71
|
+
return;
|
72
|
+
}
|
73
|
+
|
74
|
+
var ps = datapoints.pointsize,
|
75
|
+
points = datapoints.points,
|
76
|
+
otherps = other.datapoints.pointsize,
|
77
|
+
otherpoints = other.datapoints.points,
|
78
|
+
newpoints = [],
|
79
|
+
px, py, intery, qx, qy, bottom,
|
80
|
+
withlines = s.lines.show,
|
81
|
+
withbottom = ps > 2 && datapoints.format[2].y,
|
82
|
+
withsteps = withlines && s.lines.steps,
|
83
|
+
fromgap = true,
|
84
|
+
i = 0,
|
85
|
+
j = 0,
|
86
|
+
l, m;
|
87
|
+
|
88
|
+
while ( true ) {
|
89
|
+
|
90
|
+
if ( i >= points.length ) {
|
91
|
+
break;
|
92
|
+
}
|
93
|
+
|
94
|
+
l = newpoints.length;
|
95
|
+
|
96
|
+
if ( points[ i ] == null ) {
|
97
|
+
|
98
|
+
// copy gaps
|
99
|
+
|
100
|
+
for ( m = 0; m < ps; ++m ) {
|
101
|
+
newpoints.push( points[ i + m ] );
|
102
|
+
}
|
103
|
+
|
104
|
+
i += ps;
|
105
|
+
|
106
|
+
} else if ( j >= otherpoints.length ) {
|
107
|
+
|
108
|
+
// for lines, we can't use the rest of the points
|
109
|
+
|
110
|
+
if ( !withlines ) {
|
111
|
+
for ( m = 0; m < ps; ++m ) {
|
112
|
+
newpoints.push( points[ i + m ] );
|
113
|
+
}
|
114
|
+
}
|
115
|
+
|
116
|
+
i += ps;
|
117
|
+
|
118
|
+
} else if ( otherpoints[ j ] == null ) {
|
119
|
+
|
120
|
+
// oops, got a gap
|
121
|
+
|
122
|
+
for ( m = 0; m < ps; ++m ) {
|
123
|
+
newpoints.push( null );
|
124
|
+
}
|
125
|
+
|
126
|
+
fromgap = true;
|
127
|
+
j += otherps;
|
128
|
+
|
129
|
+
} else {
|
130
|
+
|
131
|
+
// cases where we actually got two points
|
132
|
+
|
133
|
+
px = points[ i ];
|
134
|
+
py = points[ i + 1 ];
|
135
|
+
qx = otherpoints[ j ];
|
136
|
+
qy = otherpoints[ j + 1 ];
|
137
|
+
bottom = 0;
|
138
|
+
|
139
|
+
if ( px === qx ) {
|
140
|
+
|
141
|
+
for ( m = 0; m < ps; ++m ) {
|
142
|
+
newpoints.push( points[ i + m ] );
|
143
|
+
}
|
144
|
+
|
145
|
+
//newpoints[ l + 1 ] += qy;
|
146
|
+
bottom = qy;
|
147
|
+
|
148
|
+
i += ps;
|
149
|
+
j += otherps;
|
150
|
+
|
151
|
+
} else if ( px > qx ) {
|
152
|
+
|
153
|
+
// we got past point below, might need to
|
154
|
+
// insert interpolated extra point
|
155
|
+
|
156
|
+
if ( withlines && i > 0 && points[ i - ps ] != null ) {
|
157
|
+
intery = py + ( points[ i - ps + 1 ] - py ) * ( qx - px ) / ( points[ i - ps ] - px );
|
158
|
+
newpoints.push( qx );
|
159
|
+
newpoints.push( intery );
|
160
|
+
for ( m = 2; m < ps; ++m ) {
|
161
|
+
newpoints.push( points[ i + m ] );
|
162
|
+
}
|
163
|
+
bottom = qy;
|
164
|
+
}
|
165
|
+
|
166
|
+
j += otherps;
|
167
|
+
|
168
|
+
} else { // px < qx
|
169
|
+
|
170
|
+
// if we come from a gap, we just skip this point
|
171
|
+
|
172
|
+
if ( fromgap && withlines ) {
|
173
|
+
i += ps;
|
174
|
+
continue;
|
175
|
+
}
|
176
|
+
|
177
|
+
for ( m = 0; m < ps; ++m ) {
|
178
|
+
newpoints.push( points[ i + m ] );
|
179
|
+
}
|
180
|
+
|
181
|
+
// we might be able to interpolate a point below,
|
182
|
+
// this can give us a better y
|
183
|
+
|
184
|
+
if ( withlines && j > 0 && otherpoints[ j - otherps ] != null ) {
|
185
|
+
bottom = qy + ( otherpoints[ j - otherps + 1 ] - qy ) * ( px - qx ) / ( otherpoints[ j - otherps ] - qx );
|
186
|
+
}
|
187
|
+
|
188
|
+
//newpoints[l + 1] += bottom;
|
189
|
+
|
190
|
+
i += ps;
|
191
|
+
}
|
192
|
+
|
193
|
+
fromgap = false;
|
194
|
+
|
195
|
+
if ( l !== newpoints.length && withbottom ) {
|
196
|
+
newpoints[ l + 2 ] = bottom;
|
197
|
+
}
|
198
|
+
}
|
199
|
+
|
200
|
+
// maintain the line steps invariant
|
201
|
+
|
202
|
+
if ( withsteps && l !== newpoints.length && l > 0 &&
|
203
|
+
newpoints[ l ] !== null &&
|
204
|
+
newpoints[ l ] !== newpoints[ l - ps ] &&
|
205
|
+
newpoints[ l + 1 ] !== newpoints[ l - ps + 1 ] ) {
|
206
|
+
for (m = 0; m < ps; ++m) {
|
207
|
+
newpoints[ l + ps + m ] = newpoints[ l + m ];
|
208
|
+
}
|
209
|
+
newpoints[ l + 1 ] = newpoints[ l - ps + 1 ];
|
210
|
+
}
|
211
|
+
}
|
212
|
+
|
213
|
+
datapoints.points = newpoints;
|
214
|
+
}
|
215
|
+
|
216
|
+
plot.hooks.processDatapoints.push( computeFillBottoms );
|
217
|
+
}
|
218
|
+
|
219
|
+
$.plot.plugins.push({
|
220
|
+
init: init,
|
221
|
+
options: options,
|
222
|
+
name: "fillbetween",
|
223
|
+
version: "1.0"
|
224
|
+
});
|
225
|
+
|
226
|
+
})(jQuery);
|
@@ -1 +1,30 @@
|
|
1
|
-
|
1
|
+
/* Flot plugin for computing bottoms for filled line and bar charts.
|
2
|
+
|
3
|
+
Copyright (c) 2007-2013 IOLA and Ole Laursen.
|
4
|
+
Licensed under the MIT license.
|
5
|
+
|
6
|
+
The case: you've got two series that you want to fill the area between. In Flot
|
7
|
+
terms, you need to use one as the fill bottom of the other. You can specify the
|
8
|
+
bottom of each data point as the third coordinate manually, or you can use this
|
9
|
+
plugin to compute it for you.
|
10
|
+
|
11
|
+
In order to name the other series, you need to give it an id, like this:
|
12
|
+
|
13
|
+
var dataset = [
|
14
|
+
{ data: [ ... ], id: "foo" } , // use default bottom
|
15
|
+
{ data: [ ... ], fillBetween: "foo" }, // use first dataset as bottom
|
16
|
+
];
|
17
|
+
|
18
|
+
$.plot($("#placeholder"), dataset, { lines: { show: true, fill: true }});
|
19
|
+
|
20
|
+
As a convenience, if the id given is a number that doesn't appear as an id in
|
21
|
+
the series, it is interpreted as the index in the array instead (so fillBetween:
|
22
|
+
0 can also mean the first series).
|
23
|
+
|
24
|
+
Internally, the plugin modifies the datapoints in each series. For line series,
|
25
|
+
extra data points might be inserted through interpolation. Note that at points
|
26
|
+
where the bottom line is not defined (due to a null point or start/end of line),
|
27
|
+
the current line will show a gap too. The algorithm comes from the
|
28
|
+
jquery.flot.stack.js plugin, possibly some code could be shared.
|
29
|
+
|
30
|
+
*/(function(e){function n(e){function t(e,t){var n;for(n=0;n<t.length;++n)if(t[n].id===e.fillBetween)return t[n];return typeof e.fillBetween=="number"?e.fillBetween<0||e.fillBetween>=t.length?null:t[e.fillBetween]:null}function n(e,n,r){if(n.fillBetween==null)return;var i=t(n,e.getData());if(!i)return;var s=r.pointsize,o=r.points,u=i.datapoints.pointsize,a=i.datapoints.points,f=[],l,c,h,p,d,v,m=n.lines.show,g=s>2&&r.format[2].y,y=m&&n.lines.steps,b=!0,w=0,E=0,S,x;for(;;){if(w>=o.length)break;S=f.length;if(o[w]==null){for(x=0;x<s;++x)f.push(o[w+x]);w+=s}else if(E>=a.length){if(!m)for(x=0;x<s;++x)f.push(o[w+x]);w+=s}else if(a[E]==null){for(x=0;x<s;++x)f.push(null);b=!0,E+=u}else{l=o[w],c=o[w+1],p=a[E],d=a[E+1],v=0;if(l===p){for(x=0;x<s;++x)f.push(o[w+x]);v=d,w+=s,E+=u}else if(l>p){if(m&&w>0&&o[w-s]!=null){h=c+(o[w-s+1]-c)*(p-l)/(o[w-s]-l),f.push(p),f.push(h);for(x=2;x<s;++x)f.push(o[w+x]);v=d}E+=u}else{if(b&&m){w+=s;continue}for(x=0;x<s;++x)f.push(o[w+x]);m&&E>0&&a[E-u]!=null&&(v=d+(a[E-u+1]-d)*(l-p)/(a[E-u]-p)),w+=s}b=!1,S!==f.length&&g&&(f[S+2]=v)}if(y&&S!==f.length&&S>0&&f[S]!==null&&f[S]!==f[S-s]&&f[S+1]!==f[S-s+1]){for(x=0;x<s;++x)f[S+s+x]=f[S+x];f[S+1]=f[S-s+1]}}r.points=f}e.hooks.processDatapoints.push(n)}var t={series:{fillBetween:null}};e.plot.plugins.push({init:n,options:t,name:"fillbetween",version:"1.0"})})(jQuery);
|
@@ -1,238 +1,241 @@
|
|
1
|
-
/*
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
skipped when plotting.
|
10
|
-
|
11
|
-
There are two helpers included for retrieving images. The easiest work
|
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
|
-
if (s.
|
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
|
-
if (
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
tmp = 0.5 * (
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
{
|
223
|
-
{
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
1
|
+
/* Flot plugin for plotting images.
|
2
|
+
|
3
|
+
Copyright (c) 2007-2013 IOLA and Ole Laursen.
|
4
|
+
Licensed under the MIT license.
|
5
|
+
|
6
|
+
The data syntax is [ [ image, x1, y1, x2, y2 ], ... ] where (x1, y1) and
|
7
|
+
(x2, y2) are where you intend the two opposite corners of the image to end up
|
8
|
+
in the plot. Image must be a fully loaded Javascript image (you can make one
|
9
|
+
with new Image()). If the image is not complete, it's skipped when plotting.
|
10
|
+
|
11
|
+
There are two helpers included for retrieving images. The easiest work the way
|
12
|
+
that you put in URLs instead of images in the data, like this:
|
13
|
+
|
14
|
+
[ "myimage.png", 0, 0, 10, 10 ]
|
15
|
+
|
16
|
+
Then call $.plot.image.loadData( data, options, callback ) where data and
|
17
|
+
options are the same as you pass in to $.plot. This loads the images, replaces
|
18
|
+
the URLs in the data with the corresponding images and calls "callback" when
|
19
|
+
all images are loaded (or failed loading). In the callback, you can then call
|
20
|
+
$.plot with the data set. See the included example.
|
21
|
+
|
22
|
+
A more low-level helper, $.plot.image.load(urls, callback) is also included.
|
23
|
+
Given a list of URLs, it calls callback with an object mapping from URL to
|
24
|
+
Image object when all images are loaded or have failed loading.
|
25
|
+
|
26
|
+
The plugin supports these options:
|
27
|
+
|
28
|
+
series: {
|
29
|
+
images: {
|
30
|
+
show: boolean
|
31
|
+
anchor: "corner" or "center"
|
32
|
+
alpha: [ 0, 1 ]
|
33
|
+
}
|
34
|
+
}
|
35
|
+
|
36
|
+
They can be specified for a specific series:
|
37
|
+
|
38
|
+
$.plot( $("#placeholder"), [{
|
39
|
+
data: [ ... ],
|
40
|
+
images: { ... }
|
41
|
+
])
|
42
|
+
|
43
|
+
Note that because the data format is different from usual data points, you
|
44
|
+
can't use images with anything else in a specific data series.
|
45
|
+
|
46
|
+
Setting "anchor" to "center" causes the pixels in the image to be anchored at
|
47
|
+
the corner pixel centers inside of at the pixel corners, effectively letting
|
48
|
+
half a pixel stick out to each side in the plot.
|
49
|
+
|
50
|
+
A possible future direction could be support for tiling for large images (like
|
51
|
+
Google Maps).
|
52
|
+
|
53
|
+
*/
|
54
|
+
|
55
|
+
(function ($) {
|
56
|
+
var options = {
|
57
|
+
series: {
|
58
|
+
images: {
|
59
|
+
show: false,
|
60
|
+
alpha: 1,
|
61
|
+
anchor: "corner" // or "center"
|
62
|
+
}
|
63
|
+
}
|
64
|
+
};
|
65
|
+
|
66
|
+
$.plot.image = {};
|
67
|
+
|
68
|
+
$.plot.image.loadDataImages = function (series, options, callback) {
|
69
|
+
var urls = [], points = [];
|
70
|
+
|
71
|
+
var defaultShow = options.series.images.show;
|
72
|
+
|
73
|
+
$.each(series, function (i, s) {
|
74
|
+
if (!(defaultShow || s.images.show))
|
75
|
+
return;
|
76
|
+
|
77
|
+
if (s.data)
|
78
|
+
s = s.data;
|
79
|
+
|
80
|
+
$.each(s, function (i, p) {
|
81
|
+
if (typeof p[0] == "string") {
|
82
|
+
urls.push(p[0]);
|
83
|
+
points.push(p);
|
84
|
+
}
|
85
|
+
});
|
86
|
+
});
|
87
|
+
|
88
|
+
$.plot.image.load(urls, function (loadedImages) {
|
89
|
+
$.each(points, function (i, p) {
|
90
|
+
var url = p[0];
|
91
|
+
if (loadedImages[url])
|
92
|
+
p[0] = loadedImages[url];
|
93
|
+
});
|
94
|
+
|
95
|
+
callback();
|
96
|
+
});
|
97
|
+
}
|
98
|
+
|
99
|
+
$.plot.image.load = function (urls, callback) {
|
100
|
+
var missing = urls.length, loaded = {};
|
101
|
+
if (missing == 0)
|
102
|
+
callback({});
|
103
|
+
|
104
|
+
$.each(urls, function (i, url) {
|
105
|
+
var handler = function () {
|
106
|
+
--missing;
|
107
|
+
|
108
|
+
loaded[url] = this;
|
109
|
+
|
110
|
+
if (missing == 0)
|
111
|
+
callback(loaded);
|
112
|
+
};
|
113
|
+
|
114
|
+
$('<img />').load(handler).error(handler).attr('src', url);
|
115
|
+
});
|
116
|
+
};
|
117
|
+
|
118
|
+
function drawSeries(plot, ctx, series) {
|
119
|
+
var plotOffset = plot.getPlotOffset();
|
120
|
+
|
121
|
+
if (!series.images || !series.images.show)
|
122
|
+
return;
|
123
|
+
|
124
|
+
var points = series.datapoints.points,
|
125
|
+
ps = series.datapoints.pointsize;
|
126
|
+
|
127
|
+
for (var i = 0; i < points.length; i += ps) {
|
128
|
+
var img = points[i],
|
129
|
+
x1 = points[i + 1], y1 = points[i + 2],
|
130
|
+
x2 = points[i + 3], y2 = points[i + 4],
|
131
|
+
xaxis = series.xaxis, yaxis = series.yaxis,
|
132
|
+
tmp;
|
133
|
+
|
134
|
+
// actually we should check img.complete, but it
|
135
|
+
// appears to be a somewhat unreliable indicator in
|
136
|
+
// IE6 (false even after load event)
|
137
|
+
if (!img || img.width <= 0 || img.height <= 0)
|
138
|
+
continue;
|
139
|
+
|
140
|
+
if (x1 > x2) {
|
141
|
+
tmp = x2;
|
142
|
+
x2 = x1;
|
143
|
+
x1 = tmp;
|
144
|
+
}
|
145
|
+
if (y1 > y2) {
|
146
|
+
tmp = y2;
|
147
|
+
y2 = y1;
|
148
|
+
y1 = tmp;
|
149
|
+
}
|
150
|
+
|
151
|
+
// if the anchor is at the center of the pixel, expand the
|
152
|
+
// image by 1/2 pixel in each direction
|
153
|
+
if (series.images.anchor == "center") {
|
154
|
+
tmp = 0.5 * (x2-x1) / (img.width - 1);
|
155
|
+
x1 -= tmp;
|
156
|
+
x2 += tmp;
|
157
|
+
tmp = 0.5 * (y2-y1) / (img.height - 1);
|
158
|
+
y1 -= tmp;
|
159
|
+
y2 += tmp;
|
160
|
+
}
|
161
|
+
|
162
|
+
// clip
|
163
|
+
if (x1 == x2 || y1 == y2 ||
|
164
|
+
x1 >= xaxis.max || x2 <= xaxis.min ||
|
165
|
+
y1 >= yaxis.max || y2 <= yaxis.min)
|
166
|
+
continue;
|
167
|
+
|
168
|
+
var sx1 = 0, sy1 = 0, sx2 = img.width, sy2 = img.height;
|
169
|
+
if (x1 < xaxis.min) {
|
170
|
+
sx1 += (sx2 - sx1) * (xaxis.min - x1) / (x2 - x1);
|
171
|
+
x1 = xaxis.min;
|
172
|
+
}
|
173
|
+
|
174
|
+
if (x2 > xaxis.max) {
|
175
|
+
sx2 += (sx2 - sx1) * (xaxis.max - x2) / (x2 - x1);
|
176
|
+
x2 = xaxis.max;
|
177
|
+
}
|
178
|
+
|
179
|
+
if (y1 < yaxis.min) {
|
180
|
+
sy2 += (sy1 - sy2) * (yaxis.min - y1) / (y2 - y1);
|
181
|
+
y1 = yaxis.min;
|
182
|
+
}
|
183
|
+
|
184
|
+
if (y2 > yaxis.max) {
|
185
|
+
sy1 += (sy1 - sy2) * (yaxis.max - y2) / (y2 - y1);
|
186
|
+
y2 = yaxis.max;
|
187
|
+
}
|
188
|
+
|
189
|
+
x1 = xaxis.p2c(x1);
|
190
|
+
x2 = xaxis.p2c(x2);
|
191
|
+
y1 = yaxis.p2c(y1);
|
192
|
+
y2 = yaxis.p2c(y2);
|
193
|
+
|
194
|
+
// the transformation may have swapped us
|
195
|
+
if (x1 > x2) {
|
196
|
+
tmp = x2;
|
197
|
+
x2 = x1;
|
198
|
+
x1 = tmp;
|
199
|
+
}
|
200
|
+
if (y1 > y2) {
|
201
|
+
tmp = y2;
|
202
|
+
y2 = y1;
|
203
|
+
y1 = tmp;
|
204
|
+
}
|
205
|
+
|
206
|
+
tmp = ctx.globalAlpha;
|
207
|
+
ctx.globalAlpha *= series.images.alpha;
|
208
|
+
ctx.drawImage(img,
|
209
|
+
sx1, sy1, sx2 - sx1, sy2 - sy1,
|
210
|
+
x1 + plotOffset.left, y1 + plotOffset.top,
|
211
|
+
x2 - x1, y2 - y1);
|
212
|
+
ctx.globalAlpha = tmp;
|
213
|
+
}
|
214
|
+
}
|
215
|
+
|
216
|
+
function processRawData(plot, series, data, datapoints) {
|
217
|
+
if (!series.images.show)
|
218
|
+
return;
|
219
|
+
|
220
|
+
// format is Image, x1, y1, x2, y2 (opposite corners)
|
221
|
+
datapoints.format = [
|
222
|
+
{ required: true },
|
223
|
+
{ x: true, number: true, required: true },
|
224
|
+
{ y: true, number: true, required: true },
|
225
|
+
{ x: true, number: true, required: true },
|
226
|
+
{ y: true, number: true, required: true }
|
227
|
+
];
|
228
|
+
}
|
229
|
+
|
230
|
+
function init(plot) {
|
231
|
+
plot.hooks.processRawData.push(processRawData);
|
232
|
+
plot.hooks.drawSeries.push(drawSeries);
|
233
|
+
}
|
234
|
+
|
235
|
+
$.plot.plugins.push({
|
236
|
+
init: init,
|
237
|
+
options: options,
|
238
|
+
name: 'image',
|
239
|
+
version: '1.1'
|
240
|
+
});
|
241
|
+
})(jQuery);
|