#!python import re from dateutil.parser import parse from dateutil.relativedelta import relativedelta from lxml import etree from svg.charts.graph import Graph from util import grouper, date_range, divide_timedelta_float, TimeScale __all__ = ('Schedule') class Schedule(Graph): """ # === For creating SVG plots of scalar temporal data = Synopsis require 'SVG/Graph/Schedule' # Data sets are label, start, end tripples. data1 = [ "Housesitting", "6/17/04", "6/19/04", "Summer Session", "6/15/04", "8/15/04", ] graph = SVG::Graph::Schedule.new( { :width => 640, :height => 480, :graph_title => title, :show_graph_title => true, :no_css => 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", :stagger_x_labels => true, :stagger_y_labels => true, :x_label_format => "%m/%d/%y", }) graph.add_data({ :data => data1, :title => 'Data', }) print graph.burn() = Description Produces a graph of temporal scalar data. = Examples http://www.germane-software/repositories/public/SVG/test/schedule.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. 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 * SVG::Graph::TimeSeries == Author Sean E. Russell Copyright 2004 Sean E. Russell This software is available under the Ruby license[LICENSE.txt] """ "The format string to be used to format the X axis labels" x_label_format = '%Y-%m-%d %H:%M:%S' """ Use this to set the spacing between dates on the axis. The value must be of the form "\d+ ?((year|month|week|day|hour|minute|second)s?)?" e.g. graph.timescale_divisions = '2 weeks' graph.timescale_divisions = '1 month' graph.timescale_divisions = '3600 seconds' # easier would be '1 hour' """ timescale_divisions = None "The formatting used for the popups. See x_label_format" popup_format = '%Y-%m-%d %H:%M:%S' _min_x_value = None scale_x_divisions = False scale_x_integers = False bar_gap = True stylesheet_names = Graph.stylesheet_names + ['bar.css'] def add_data(self, data): """ Add data to the plot. # A data set with 1 point: Lunch from 12:30 to 14:00 d1 = [ "Lunch", "12:30", "14:00" ] # A data set with 2 points: "Cats" runs from 5/11/03 to 7/15/04, and # "Henry V" runs from 6/12/03 to 8/20/03 d2 = [ "Cats", "5/11/03", "7/15/04", "Henry V", "6/12/03", "8/20/03" ] graph.add_data( :data => d1, :title => 'Meetings' ) graph.add_data( :data => d2, :title => 'Plays' ) Note that the data must be in time,value pairs, and that the date format may be any date that is parseable by ParseDate. Also note that, in this example, we're mixing scales; the data from d1 will probably not be discernable if both data sets are plotted on the same graph, since d1 is too granular. """ # The ruby version does something different here, throwing out # any previously added data. super(Schedule, self).add_data(data) # copied from Bar # TODO, refactor this into a common base class (or mix-in) def get_bar_gap(self, field_size): bar_gap = 10 # default gap if field_size < 10: # adjust for narrow fields bar_gap = field_size / 2 # the following zero's out the gap if bar_gap is False bar_gap = int(self.bar_gap) * bar_gap return bar_gap def validate_data(self, conf): super(Schedule, self).validate_data(conf) msg = "Data supplied must be (title, from, to) tripples!" assert len(conf['data']) % 3 == 0, msg def process_data(self, conf): super(Schedule, self).process_data(conf) data = conf['data'] triples = grouper(3, data) labels, begin_dates, end_dates = zip(*triples) begin_dates = map(self.parse_date, begin_dates) end_dates = map(self.parse_date, end_dates) # reconstruct the triples in a new order reordered_triples = zip(begin_dates, end_dates, labels) # because of the reordering, this will sort by begin_date # then end_date, then label. reordered_triples.sort() conf['data'] = reordered_triples def parse_date(self, date_string): print 'attempting to parse %s as date' % date_string return parse(date_string) def set_min_x_value(self, value): if isinstance(value, basestring): value = self.parse_date(value) self._min_x_value = value def get_min_x_value(self): return self._min_x_value 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): format = lambda x: x.strftime(self.x_label_format) return map(format, self.get_x_values()) def y_label_offset(self, height): return height / -2.0 def get_y_labels(self): # ruby version uses the last data supplied last = -1 data = self.data[last]['data'] begin_dates, start_dates, labels = zip(*data) return labels def draw_data(self): bar_gap = self.get_bar_gap(self.get_field_height()) subbar_height = self.get_field_height() - bar_gap y_mod = (subbar_height / 2) + (self.font_size / 2) x_min,x_max,div = self._x_range() x_range = x_max - x_min width = (float(self.graph_width) - self.font_size*2) # time_scale #scale /= x_range scale = TimeScale(width, x_range) # ruby version uses the last data supplied last = -1 data = self.data[last]['data'] for index, (x_start, x_end, label) in enumerate(data): count = index + 1 # index is 0-based, count is 1-based y = self.graph_height - (self.get_field_height()*count) bar_width = scale*(x_end-x_start) bar_start = scale*(x_start-x_min) etree.SubElement(self.graph, 'rect', { 'x': str(bar_start), 'y': str(y), 'width': str(bar_width), 'height': str(subbar_height), 'class': 'fill%s' % (count+1), }) def _x_range(self): # ruby version uses teh last data supplied last = -1 data = self.data[last]['data'] start_dates, end_dates, labels = zip(*data) all_dates = start_dates + end_dates max_value = max(all_dates) if not self.min_x_value is None: all_dates.append(self.min_x_value) min_value = min(all_dates) range = max_value - min_value right_pad = divide_timedelta_float(range, 20.0) or relativedelta(days=10) scale_range = (max_value + right_pad) - min_value #scale_division = self.scale_x_divisions or (scale_range / 10.0) # todo, remove timescale_x_divisions and use scale_x_divisions only # but as a time delta scale_division = divide_timedelta_float(scale_range, 10.0) # this doesn't make sense, because x is a timescale #if self.scale_x_integers: # scale_division = min(round(scale_division), 1) return min_value, max_value, scale_division def get_x_values(self): x_min, x_max, scale_division = self._x_range() if self.timescale_divisions: pattern = re.compile('(\d+) ?(\w+)') m = pattern.match(self.timescale_divisions) if not m: raise ValueError, "Invalid timescale_divisions: %s" % self.timescale_divisions magnitude = int(m.group(1)) units = m.group(2) parameter = self.lookup_relativedelta_parameter(units) delta = relativedelta(**{parameter:magnitude}) scale_division = delta return date_range(x_min, x_max, scale_division) def lookup_relativedelta_parameter(self, unit_string): from util import reverse_mapping, flatten_mapping unit_string = unit_string.lower() mapping = dict( years = ('years', 'year', 'yrs', 'yr'), months = ('months', 'month', 'mo'), weeks = ('weeks', 'week', 'wks' ,'wk'), days = ('days', 'day'), hours = ('hours', 'hour', 'hr', 'hrs', 'h'), minutes = ('minutes', 'minute', 'min', 'mins', 'm'), seconds = ('seconds', 'second', 'sec', 'secs', 's'), ) mapping = reverse_mapping(mapping) mapping = flatten_mapping(mapping) if not unit_string in mapping: raise ValueError, "%s doesn't match any supported time/date unit" return mapping[unit_string]