flot-rails 0.0.2 → 0.0.3
Sign up to get free protection for your applications and to get access to all the features.
- 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);
|