rename min/max
[MojoFacets.git] / public / facet_graph.js
1 var data = {
2         x_min: Number.MAX_VALUE,
3         x_max: Number.MIN_VALUE,
4         y_min: Number.MAX_VALUE,
5         y_max: Number.MIN_VALUE,
6         x_data: [],
7         y_data: [],
8         x_px: [],
9         y_labels: [],
10         width: 600,
11         height: 400,
12 };
13
14 var ul = $('ul#facet');
15
16 ul.find('li label').each( function(){
17         var v = parseFloat( $(this).text() );
18         if ( isNaN(v) ) v = 0;
19         if ( v > data.x_max ) data.x_max = v;
20         if ( v < data.x_min ) data.x_min = v;
21         data.x_data.push( v );
22 });
23
24 ul.find('li span.count').each( function(){
25         var v = parseFloat( $(this).text() ); // FIXME not numeric!
26         if ( isNaN(v) ) v = 0;
27         if ( v > data.y_max ) data.y_max = v;
28         if ( v < data.y_min ) data.y_min = v;
29         data.y_data.push( v );
30 });
31
32 data.y_min = 0; // XXX force to 0, because it's count
33
34 data.x_range = data.x_max - data.x_min;
35 data.y_range = data.y_max - data.y_min;
36
37
38 var y_num_labels = Math.round( data.height / 20 ); // padding between vertical labels
39 var y_inc = Math.ceil( data.y_range / y_num_labels );
40
41 var y_pos = data.y_min;
42 var y_last_pos = Math.ceil( data.y_max - y_inc / 2 );
43 while( y_pos < y_last_pos ) {
44         data.y_labels.push( y_pos );
45         y_pos += y_inc;
46 }
47 data.y_labels.push( data.y_max );
48
49 data.numeric = $('span#numeric').length;
50 data.x_inc = data.numeric
51         ? Math.round( data.width / data.x_range )
52         : data.width / data.x_data.length
53         ;
54
55 console.debug( 'data', data );
56
57 var canvas = $('<canvas/>');
58
59 canvas.attr({
60         width: data.width,
61         height: data.height,
62 });
63
64 var canvasContain = $('<div class="chart"></div>')
65         .css({ width: data.width, height: data.height })
66         .append( canvas )
67         .insertBefore( ul );
68
69 var ctx = canvas[0].getContext('2d');
70 ctx.translate( 0, data.height ); // start at bottom left
71 ctx.lineWidth = 2;
72 ctx.strokeStyle = '#ff8800';
73 ctx.fillStyle = '#ffcc88';
74
75 ctx.moveTo( 0, 0 );
76 ctx.beginPath();
77
78 for( var i in data.x_data ) {
79         var x = data.x_data[i];
80         if ( data.numeric ) x = Math.ceil( ( x - data.x_min ) / data.x_range * data.width  );
81         var y = Math.ceil( ( data.y_data[i] - data.y_min ) / data.y_range * data.height );
82         if ( data.numeric ) {
83                 ctx.lineTo( x, -y );
84                 data.x_px.push( x );
85         } else {
86                 var x_px = i * data.x_inc;
87                 console.debug( x_px, y );
88                 ctx.fillRect( x_px, 0, data.x_inc, -y );
89                 ctx.strokeRect( x_px, 0, data.x_inc, -y );
90         }
91 }
92
93 ctx.stroke();
94 ctx.closePath();
95
96 if ( data.numeric ) {
97
98 var labels_x = $('<ul class="labels-x"></ul>')
99         .css({ width: data.width, height: data.height, position: 'absolute' });
100
101 for( var x_pos = 0; x_pos < data.width; x_pos += data.x_inc ) {
102         var x_val = ( x_pos / data.width * data.x_range ) + data.x_min;
103         $('<li><span class="label">' + x_val + '</span></li>')
104                 .css({ left: x_pos })
105                 .appendTo(labels_x);
106 }
107
108 $('<li><span class="label">' + data.x_max + '</span></li>')
109                 .css({ right: 0 })
110                 .appendTo(labels_x);
111
112 labels_x.appendTo(canvasContain);
113
114 } // data.numeric
115
116 var labels_y = $('<ul class="labels-y"></ul>')
117         .css({ width: data.width, height: data.height, position: 'absolute' });
118
119 for( var i in data.y_labels ) {
120                 $('<li><span class="line"></span><span class="label">' + data.y_labels[i] + '</span></li>')
121                         .css({ bottom: Math.ceil( ( data.y_labels[i] - data.y_min ) / data.y_range * data.height ) })
122                         .appendTo(labels_y);
123 }
124 labels_y.appendTo(canvasContain);