fix y labels position
[MojoFacets.git] / public / facet_graph.js
1 var data = {
2         min_x: Number.MAX_VALUE,
3         max_x: Number.MIN_VALUE,
4         min_y: Number.MAX_VALUE,
5         max_y: 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.max_x ) data.max_x = v;
20         if ( v < data.min_x ) data.min_x = 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.max_y ) data.max_y = v;
28         if ( v < data.min_y ) data.min_y = v;
29         data.y_data.push( v );
30 });
31
32
33 data.x_range = data.max_x - data.min_x;
34 data.y_range = data.max_y - data.min_y;
35
36
37 var y_num_labels = Math.round( data.height / 20 ); // padding between vertical labels
38 var y_inc = Math.ceil( data.y_range / y_num_labels );
39
40 var y_pos = data.min_y;
41 while( y_pos < data.max_y - y_inc ) {
42         data.y_labels.push( y_pos );
43         y_pos += y_inc;
44 }
45 data.y_labels.push( data.max_y );
46
47 data.numeric = $('span#numeric').length;
48 data.x_inc = data.numeric
49         ? Math.round( data.width / data.x_range )
50         : data.width / data.x_data.length
51         ;
52
53 console.debug( 'data', data );
54
55 var canvas = $('<canvas/>');
56
57 canvas.attr({
58         width: data.width,
59         height: data.height,
60 });
61
62 var canvasContain = $('<div class="chart"></div>')
63         .css({ width: data.width, height: data.height })
64         .append( canvas )
65         .insertBefore( ul );
66
67 var ctx = canvas[0].getContext('2d');
68 ctx.translate( 0, data.height ); // start at bottom left
69 ctx.lineWidth = 2;
70 ctx.strokeStyle = '#ff8800';
71 ctx.fillStyle = '#ffcc88';
72
73 ctx.moveTo( 0, 0 );
74 ctx.beginPath();
75
76 for( var i in data.x_data ) {
77         var x = data.x_data[i];
78         if ( data.numeric ) x = Math.ceil( ( x - data.min_x ) / data.x_range * data.width  );
79         var y = Math.ceil( ( data.y_data[i] - data.min_y ) / data.y_range * data.height );
80         if ( data.numeric ) {
81                 ctx.lineTo( x, -y );
82                 data.x_px.push( x );
83         } else {
84                 var x_px = i * data.x_inc;
85                 console.debug( x_px, y );
86                 ctx.fillRect( x_px, 0, data.x_inc, -y );
87                 ctx.strokeRect( x_px, 0, data.x_inc, -y );
88         }
89 }
90
91 ctx.stroke();
92 ctx.closePath();
93
94 var labels_x = $('<ul class="labels-x"></ul>')
95         .css({ width: data.width, height: data.height, position: 'absolute' });
96
97 var x_pos = 0;
98
99 for( var i in data.x_data ) {
100         if ( Math.abs( data.x_px[i] - x_pos ) > 20 ) {
101                 x_pos = data.x_px[i];
102                 $('<li><span class="line"></span><span class="label">' + data.x_data[i] + '</span></li>')
103                         .css({ left: x_pos })
104                         .appendTo(labels_x);
105         }
106 }
107 labels_x.appendTo(canvasContain);
108
109 var labels_y = $('<ul class="labels-y"></ul>')
110         .css({ width: data.width, height: data.height, position: 'absolute' });
111
112 for( var i in data.y_labels ) {
113                 $('<li><span class="line"></span><span class="label">' + data.y_labels[i] + '</span></li>')
114                         .css({ bottom: Math.ceil( ( data.y_labels[i] - data.min_y ) / data.y_range * data.height ) })
115                         .appendTo(labels_y);
116 }
117 labels_y.appendTo(canvasContain);