mirror of https://github.com/Kozea/pygal.git
Python to generate nice looking SVG graph
http://pygal.org/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
170 lines
5.2 KiB
170 lines
5.2 KiB
#!/usr/bin/env python |
|
import SVG |
|
|
|
def get_pairs( i ): |
|
i = iter( i ) |
|
while True: yield i.next(), i.next() |
|
|
|
class Plot( SVG.Plot ): |
|
"""=== For creating SVG plots of scalar temporal data |
|
|
|
= Synopsis |
|
|
|
require 'SVG/Graph/TimeSeriess' |
|
|
|
# Data sets are x,y pairs |
|
data1 = ["6/17/72", 11, "1/11/72", 7, "4/13/04 17:31", 11, |
|
"9/11/01", 9, "9/1/85", 2, "9/1/88", 1, "1/15/95", 13] |
|
data2 = ["8/1/73", 18, "3/1/77", 15, "10/1/98", 4, |
|
"5/1/02", 14, "3/1/95", 6, "8/1/91", 12, "12/1/87", 6, |
|
"5/1/84", 17, "10/1/80", 12] |
|
|
|
graph = SVG::Graph::TimeSeries.new( { |
|
:width => 640, |
|
:height => 480, |
|
:graph_title => title, |
|
:show_graph_title => true, |
|
:no_css => true, |
|
:key => true, |
|
:scale_x_integers => true, |
|
:scale_y_integers => true, |
|
:min_x_value => 0, |
|
:min_y_value => 0, |
|
:show_data_labels => true, |
|
:show_x_guidelines => true, |
|
:show_x_title => true, |
|
:x_title => "Time", |
|
:show_y_title => true, |
|
:y_title => "Ice Cream Cones", |
|
:y_title_text_direction => :bt, |
|
:stagger_x_labels => true, |
|
:x_label_format => "%m/%d/%y", |
|
}) |
|
|
|
graph.add_data({ |
|
:data => projection |
|
:title => 'Projected', |
|
}) |
|
|
|
graph.add_data({ |
|
:data => actual, |
|
:title => 'Actual', |
|
}) |
|
|
|
print graph.burn() |
|
|
|
= Description |
|
|
|
Produces a graph of temporal scalar data. |
|
|
|
= Examples |
|
|
|
http://www.germane-software/repositories/public/SVG/test/timeseries.rb |
|
|
|
= Notes |
|
|
|
The default stylesheet handles upto 10 data sets, if you |
|
use more you must create your own stylesheet and add the |
|
additional settings for the extra data sets. You will know |
|
if you go over 10 data sets as they will have no style and |
|
be in black. |
|
|
|
Unlike the other types of charts, data sets must contain x,y pairs: |
|
|
|
[ "12:30", 2 ] # A data set with 1 point: ("12:30",2) |
|
[ "01:00",2, "14:20",6] # A data set with 2 points: ("01:00",2) and |
|
# ("14:20",6) |
|
|
|
Note that multiple data sets within the same chart can differ in length, |
|
and that the data in the datasets needn't be in order; they will be ordered |
|
by the plot along the X-axis. |
|
|
|
The dates must be parseable by ParseDate, but otherwise can be |
|
any order of magnitude (seconds within the hour, or years) |
|
|
|
= See also |
|
|
|
* SVG::Graph::Graph |
|
* SVG::Graph::BarHorizontal |
|
* SVG::Graph::Bar |
|
* SVG::Graph::Line |
|
* SVG::Graph::Pie |
|
* SVG::Graph::Plot |
|
|
|
== Author |
|
|
|
Sean E. Russell <serATgermaneHYPHENsoftwareDOTcom> |
|
|
|
Copyright 2004 Sean E. Russell |
|
This software is available under the Ruby license[LICENSE.txt] |
|
""" |
|
popup_format = x_label_format = '%Y-%m-%d %H:%M:%S' |
|
__doc_popup_format_ = "The formatting usped for the popups. See x_label_format" |
|
__doc_x_label_format_ = "The format string used to format the X axis labels. See strftime." |
|
|
|
def add_data( self, data ): |
|
"""Add data to the plot. |
|
d1 = [ "12:30", 2 ] # A data set with 1 point: ("12:30",2) |
|
d2 = [ "01:00",2, "14:20",6] # A data set with 2 points: ("01:00",2) and |
|
# ("14:20",6) |
|
graph.add_data( |
|
:data => d1, |
|
:title => 'One' |
|
) |
|
graph.add_data( |
|
:data => d2, |
|
:title => 'Two' |
|
) |
|
|
|
Note that the data must be in time,value pairs, and that the date format |
|
may be any date that is parseable by ParseDate.""" |
|
super( self.__class__, self ).add_data( data ) |
|
|
|
def validate_data( self, data ): |
|
if len( data['data'] ) % 2 != 0: raise "Expecting an even set of data points for TimeSeries." |
|
|
|
def process_data( self, data ): |
|
pairs = get_pairs( data['data'] ) |
|
pairs = map( lambda (x,y): (self.parse_date(x),y), pairs ) |
|
pairs.sort() |
|
data['data'] = zip( *pairs ) |
|
|
|
def get_min_x_value( self ): |
|
return self._min_x_value |
|
def set_min_x_value( self, date ): |
|
self._min_x_value = self.parse_date( date ) |
|
min_x_value = property( get_min_x_value, set_min_x_value ) |
|
|
|
def format( self, x, y ): |
|
return x.strftime( self.popup_format ) |
|
|
|
def get_x_labels( self ): |
|
return map( lambda t: t.strftime( self.x_label_format ), self.get_x_values() ) |
|
|
|
def get_x_values( self ): |
|
result = self.get_x_timescale_division_values() |
|
if result: return result |
|
return range( *self.x_range() ) |
|
|
|
def get_x_timescale_division_values( self ): |
|
if not self.timescale_divisions: return |
|
min, max, scale_division = self.x_range() |
|
m = re.match( '(?P<amount>\d+) ?(?P<division_units>days|weeks|months|years|hours|minutes|seconds)?', self.timescale_divisions ) |
|
# copy amount and division_units into the local namespace |
|
vars.update( m.groupdict() ) |
|
division_units = division_units or 'days' |
|
amount = int( amount ) |
|
if not amount: return |
|
if division_units == 'weeks': |
|
amount *= 7 |
|
division_units = 'days' |
|
# strip off the plural (s) |
|
division_units = division_units[:-1] |
|
result = self.get_time_range( min, max, step, units ) |
|
return result |
|
|
|
def get_time_range( self, start, stop, step, units ): |
|
current = start |
|
while current < stop: |
|
yield current |
|
current.replace( **{ units: current.getattr( units ) + step } ) |