# # Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2003-2007 Donald N. Allingham # Copyright (C) 2007-2008 Brian G. Matherly # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program; if not, write to the Free Software # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA # # $Id$ """ Timeline Chart """ #------------------------------------------------------------------------ # # python modules # #------------------------------------------------------------------------ from TransUtils import sgettext as _ #------------------------------------------------------------------------ # # GRAMPS modules # #------------------------------------------------------------------------ from gen.plug.menu import PersonOption, FilterOption, EnumeratedListOption from ReportBase import Report, ReportUtils, MenuReportOptions pt2cm = ReportUtils.pt2cm from gen.plug.docgen import (FontStyle, ParagraphStyle, GraphicsStyle, FONT_SANS_SERIF, DASHED, PARA_ALIGN_CENTER) import Sort from QuestionDialog import ErrorDialog from BasicUtils import name_displayer from Utils import probably_alive from gui.utils import ProgressMeter import gen.lib #------------------------------------------------------------------------ # # Private Functions # #------------------------------------------------------------------------ def _get_sort_functions(sort): return [ (_("Birth Date"),sort.by_birthdate_key), (_("Name"),sort.by_last_name_key), ] #------------------------------------------------------------------------ # # TimeLine # #------------------------------------------------------------------------ class TimeLine(Report): def __init__(self, database, options_class): """ Create the Timeline object that produces the report. The arguments are: database - the GRAMPS database instance person - currently selected person options_class - instance of the Options class for this report This report needs the following parameters (class variables) that come in the options class. filter - Filter to be applied to the people of the database. The option class carries its number, and the function returning the list of filters. sortby - Sorting method to be used. """ Report.__init__(self, database, options_class) menu = options_class.menu self.filter = menu.get_option_by_name('filter').get_filter() self.title = _("Timeline Graph for %s") % self.filter.get_name() sort_func_num = menu.get_option_by_name('sortby').get_value() sort_functions = _get_sort_functions(Sort.Sort(database)) self.sort_name = sort_functions[sort_func_num][0] self.sort_func = sort_functions[sort_func_num][1] self.calendar = gen.lib.date.Date.ui_calendar_names[menu.get_option_by_name('calendar').get_value()] def write_report(self): self.progress = ProgressMeter(_('Timeline')) (low, high) = self.find_year_range() if low == high: if self.standalone: self.doc.close() ErrorDialog(_("Report could not be created"), _("The range of dates chosen was not valid")) return st_size = self.name_size() style_sheet = self.doc.get_style_sheet() font = style_sheet.get_paragraph_style('TLG-Name').get_font() incr = pt2cm(font.get_size()) pad = incr*.75 x1,x2,y1,y2 = (0,0,0,0) start = st_size+0.5 stop = self.doc.get_usable_width()-0.5 size = (stop-start) self.header = 2.0 self.doc.start_page() index = 1 current = 1; length = len(self.plist) self.progress.set_pass(_('Sorting dates...'), 1) self.plist.sort(key=self.sort_func) self.progress.set_pass(_('Calculating timeline...'), len(self.plist)) for p_id in self.plist: self.progress.step() p = self.database.get_person_from_handle(p_id) birth = ReportUtils.get_birth_or_fallback(self.database, p) if birth: b = birth.get_date_object().to_calendar(self.calendar).get_year() else: b = None death = ReportUtils.get_death_or_fallback(self.database, p) if death: d = death.get_date_object().to_calendar(self.calendar).get_year() else: d = None n = name_displayer.display_formal(p) self.doc.draw_text('TLG-text', n,incr+pad, self.header + (incr+pad)*index) y1 = self.header + (pad+incr)*index y2 = self.header + ((pad+incr)*index)+incr y3 = (y1+y2)/2.0 w = 0.05 if b: start_offset = ((float(b-low)/float(high-low)) * (size)) x1 = start+start_offset path = [(x1,y1),(x1+w,y3),(x1,y2),(x1-w,y3)] self.doc.draw_path('TLG-line',path) if d: start_offset = ((float(d-low)/float(high-low)) * (size)) x1 = start+start_offset path = [(x1,y1),(x1+w,y3),(x1,y2),(x1-w,y3)] self.doc.draw_path('TLG-solid',path) if b and d: start_offset = ((float(b-low)/float(high-low)) * size) + w stop_offset = ((float(d-low)/float(high-low)) * size) - w x1 = start+start_offset x2 = start+stop_offset self.doc.draw_line('open',x1,y3,x2,y3) if (y2 + incr) >= self.doc.get_usable_height(): if current != length: self.build_grid(low, high,start,stop) self.doc.end_page() self.doc.start_page() self.build_grid(low, high,start,stop) index = 1 x1,x2,y1,y2 = (0,0,0,0) else: index += 1; current += 1 self.progress.close() self.build_grid(low, high,start,stop) self.doc.end_page() def build_grid(self,year_low,year_high,start_pos,stop_pos): """ Draws the grid outline for the chart. Sets the document label, draws the vertical lines, and adds the year labels. Arguments are: year_low - lowest year on the chart year_high - highest year on the chart start_pos - x position of the lowest leftmost grid line stop_pos - x position of the rightmost grid line """ width = self.doc.get_usable_width() style_sheet = self.doc.get_style_sheet() title_font = style_sheet.get_paragraph_style('TLG-Title').get_font() normal_font = style_sheet.get_paragraph_style('TLG-Name').get_font() label_font = style_sheet.get_paragraph_style('TLG-Label').get_font() byline = _("%(calendar_type)s Calendar, Sorted by %(sortby)s") % \ {"calendar_type": self.calendar, "sortby": self.sort_name} self.doc.center_text('TLG-title',self.title + "\n" + byline,width/2.0,0) label_y = self.header - (pt2cm(normal_font.get_size())*1.2) top_y = self.header bottom_y = self.doc.get_usable_height() incr = (year_high - year_low)/5 delta = (stop_pos - start_pos)/ 5 for val in range(0,6): year_str = str(year_low + (incr*val)) xpos = start_pos+(val*delta) self.doc.center_text('TLG-label', year_str, xpos, label_y) self.doc.draw_line('TLG-grid', xpos, top_y, xpos, bottom_y) def find_year_range(self): low = 999999 high = -999999 self.plist = self.filter.apply(self.database, self.database.iter_person_handles()) for p_id in self.plist: p = self.database.get_person_from_handle(p_id) birth = ReportUtils.get_birth_or_fallback(self.database, p) if birth: b = birth.get_date_object().to_calendar(self.calendar).get_year() else: b = None death = ReportUtils.get_death_or_fallback(self.database, p) if death: d = death.get_date_object().to_calendar(self.calendar).get_year() else: d = None if b: low = min(low,b) high = max(high,b) if d: low = min(low,d) high = max(high,d) # round the dates to the nearest decade low = int((low/10))*10 high = int(((high+9)/10))*10 # Make sure the difference is a multiple of 50 so all year ranges land # on a decade. low -= 50 - ((high-low) % 50) if low is None: low = high if high is None: high = low return (low, high) def name_size(self): self.plist = self.filter.apply(self.database, self.database.iter_person_handles()) style_sheet = self.doc.get_style_sheet() gstyle = style_sheet.get_draw_style('TLG-text') pname = gstyle.get_paragraph_style() pstyle = style_sheet.get_paragraph_style(pname) font = pstyle.get_font() size = 0 for p_id in self.plist: p = self.database.get_person_from_handle(p_id) n = name_displayer.display_formal(p) size = max(self.doc.string_width(font, n),size) return pt2cm(size) #------------------------------------------------------------------------ # # TimeLineOptions # #------------------------------------------------------------------------ class TimeLineOptions(MenuReportOptions): def __init__(self, name, dbase): self.__pid = None self.__filter = None self.__db = dbase MenuReportOptions.__init__(self, name, dbase) def add_menu_options(self, menu): category_name = _("Report Options") self.__filter = FilterOption(_("Filter"), 0) self.__filter.set_help( _("Determines what people are included in the report")) menu.add_option(category_name, "filter", self.__filter) self.__filter.connect('value-changed', self.__filter_changed) self.__pid = PersonOption(_("Filter Person")) self.__pid.set_help(_("The center person for the filter")) menu.add_option(category_name, "pid", self.__pid) self.__pid.connect('value-changed', self.__update_filters) self.__update_filters() sortby = EnumeratedListOption(_('Sort by'), 0 ) idx = 0 for item in _get_sort_functions(Sort.Sort(self.__db)): sortby.add_item(idx,item[0]) idx += 1 sortby.set_help( _("Sorting method to use")) menu.add_option(category_name,"sortby",sortby) self.__calendar = EnumeratedListOption(_("Calendar"), 0) self.__calendar.set_help(_("The calendar which determines the year span")) idx = 0 for calendar in gen.lib.date.Date.ui_calendar_names: self.__calendar.add_item(idx, calendar) idx += 1 menu.add_option(category_name, "calendar", self.__calendar) def __update_filters(self): """ Update the filter list based on the selected person """ gid = self.__pid.get_value() person = self.__db.get_person_from_gramps_id(gid) filter_list = ReportUtils.get_person_filters(person, False) self.__filter.set_filters(filter_list) def __filter_changed(self): """ Handle filter change. If the filter is not specific to a person, disable the person option """ filter_value = self.__filter.get_value() if filter_value in [1, 2, 3, 4]: # Filters 1, 2, 3 and 4 rely on the center person self.__pid.set_available(True) else: # The rest don't self.__pid.set_available(False) def make_default_style(self,default_style): """Make the default output style for the Timeline report.""" # Paragraph Styles f = FontStyle() f.set_size(10) f.set_type_face(FONT_SANS_SERIF) p = ParagraphStyle() p.set_font(f) p.set_description(_("The style used for the person's name.")) default_style.add_paragraph_style("TLG-Name",p) f = FontStyle() f.set_size(8) f.set_type_face(FONT_SANS_SERIF) p = ParagraphStyle() p.set_font(f) p.set_alignment(PARA_ALIGN_CENTER) p.set_description(_("The style used for the year labels.")) default_style.add_paragraph_style("TLG-Label",p) f = FontStyle() f.set_size(14) f.set_type_face(FONT_SANS_SERIF) p = ParagraphStyle() p.set_font(f) p.set_alignment(PARA_ALIGN_CENTER) p.set_description(_("The style used for the title of the page.")) default_style.add_paragraph_style("TLG-Title",p) """ Graphic Styles TLG-grid - 0.5pt wide line dashed line. Used for the lines that make up the grid. TLG-line - 0.5pt wide line. Used for the line connecting two endpoints and for the birth marker. TLG-solid - 0.5pt line with a black fill color. Used for the date of death marker. TLG-text - Contains the TLG-Name paragraph style used for the individual's name. TLG-title - Contains the TLG-Title paragraph style used for the title of the document. TLG-label - Contains the TLG-Label paragraph style used for the year label's in the document. """ g = GraphicsStyle() g.set_line_width(0.5) g.set_color((0,0,0)) default_style.add_draw_style("TLG-line",g) g = GraphicsStyle() g.set_line_width(0.5) g.set_color((0,0,0)) g.set_fill_color((0,0,0)) default_style.add_draw_style("TLG-solid",g) g = GraphicsStyle() g.set_line_width(0.5) g.set_color((0,0,0)) g.set_fill_color((255,255,255)) default_style.add_draw_style("open",g) g = GraphicsStyle() g.set_line_width(0.5) g.set_line_style(DASHED) g.set_color((0,0,0)) default_style.add_draw_style("TLG-grid",g) g = GraphicsStyle() g.set_paragraph_style("TLG-Name") g.set_color((0,0,0)) g.set_fill_color((255,255,255)) g.set_line_width(0) default_style.add_draw_style("TLG-text",g) g = GraphicsStyle() g.set_paragraph_style("TLG-Title") g.set_color((0,0,0)) g.set_fill_color((255,255,255)) g.set_line_width(0) default_style.add_draw_style("TLG-title",g) g = GraphicsStyle() g.set_paragraph_style("TLG-Label") g.set_color((0,0,0)) g.set_fill_color((255,255,255)) g.set_line_width(0) default_style.add_draw_style("TLG-label",g)