# # Gramps - a GTK+/GNOME based genealogy program # # Copyright (C) 2003 Donald N. Allingham # Copyright (C) 2003 Tim Waugh # # 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$ #------------------------------------------------------------------------ # # gramps modules # #------------------------------------------------------------------------ import Report import TextDoc import RelLib import Errors from QuestionDialog import ErrorDialog from intl import gettext as _ #------------------------------------------------------------------------ # # AncestorsReport # #------------------------------------------------------------------------ class AncestorsReport (Report.Report): def __init__(self,database,person,max,pgbrk,doc,output,newpage=0): self.map = {} self.database = database self.start = person self.max_generations = max self.pgbrk = pgbrk self.doc = doc table = TextDoc.TableStyle () table.set_column_widths ([15, 85]) table.set_width (100) doc.add_table_style ("PersonNoSpouse", table) table = TextDoc.TableStyle () table.set_column_widths ([10, 15, 75]) table.set_width (100) doc.add_table_style ("ChildNoSpouse", table) for nspouse in range (1, 3): table = TextDoc.TableStyle () table.set_width (100) widths = [15, 100 - 15 * (nspouse + 1)] widths.extend ([15] * nspouse) table.set_column_widths (widths) doc.add_table_style ("PersonWithSpouse%d" % nspouse, table) table = TextDoc.TableStyle () table.set_width (100) widths = [10, 15, 90 - 15 * (nspouse + 1)] widths.extend ([15] * nspouse) table.set_column_widths (widths) doc.add_table_style ("ChildWithSpouse%d"% nspouse, table) cell = TextDoc.TableCellStyle () cell.set_padding (1) # each side makes 2cm, the size of the photo doc.add_cell_style ("PaddedCell", cell) cell = TextDoc.TableCellStyle () cell.set_padding (1) # each side makes 2cm, the size of the photo cell.set_left_border (1) cell.set_top_border (1) cell.set_right_border (1) cell.set_bottom_border (1) doc.add_cell_style ("NoPhoto", cell) cell = TextDoc.TableCellStyle () doc.add_cell_style ("Photo", cell) cell = TextDoc.TableCellStyle () doc.add_cell_style ("Entry", cell) if output: self.standalone = 1 self.doc.open(output) else: self.standalone = 0 if newpage: self.doc.page_break() self.sref_map = {} self.sref_index = 1 def write_report(self): name = self.person_name (self.start) self.doc.start_paragraph("Title") title = _("Ancestors of %s") % name self.doc.write_text(title) self.doc.end_paragraph() self.doc.start_paragraph ("Generation") self.doc.write_text ("Generation 1") self.doc.end_paragraph () self.write_paragraphs (self.person (self.start, suppress_children = 1, needs_name = 1)) families = [self.start.getMainParents ()] if len (families) > 0: self.generation (self.max_generations, families, [self.start]) if self.standalone: self.doc.close() return def write_paragraphs (self, paragraphs): for (fn, params) in paragraphs: if len (params) == 0: fn () elif len (params) == 1: fn (params[0]) elif len (params) == 2: fn (params[0], params[1]) elif len (params) == 3: fn (params[0], params[1], params[2]) elif len (params) == 4: fn (params[0], params[1], params[2], params[3]) else: self.doc.write_text ("Call to %s with params %s" % (str (fn), str (params))) def family (self, family, already_described): ret = [] father = family.getFather () mother = family.getMother () if father: ret.extend (self.person (father, short_form = father in already_described, already_described = already_described, needs_name = not mother, from_family = family)) if mother: ret.extend (self.person (mother, short_form = mother in already_described, already_described = already_described, needs_name = not father, from_family = family)) children = family.getChildList () if len (children): ret.append ((self.doc.start_paragraph, ['ChildTitle'])) ret.append ((self.doc.write_text, ['Their children:'])) ret.append ((self.doc.end_paragraph, [])) for child in children: ret.extend (self.person (child, suppress_children = 1, short_form=child in already_described, already_described = already_described, needs_name = 1, from_family = family)) return ret def generation (self, generations, families, already_described, thisgen = 2): if generations > 0 and len (families): people = [] for family in families: people.extend (self.family (family, already_described)) if len (people): if self.pgbrk: self.doc.page_break() self.doc.start_paragraph ("Generation") self.doc.write_text ("Generation %d" % thisgen) self.doc.end_paragraph () self.write_paragraphs (people) next_families = [] for family in families: father = family.getFather () if father: already_described.append (father) father_family = father.getMainParents () if father_family: next_families.append (father_family) mother = family.getMother () if mother: already_described.append (mother) mother_family = mother.getMainParents () if mother_family: next_families.append (mother_family) self.generation (generations - 1, next_families, already_described, thisgen + 1) def person (self, person, suppress_children = 0, short_form = 0, already_described = [], needs_name = 0, from_family = None): ret = [] name = self.person_name (person) if name: photos = person.getPhotoList () bits = '' bits += self.short_occupation (person) bits += self.long_born_died (person) if not suppress_children: bits += self.parents_of (person) else: bits += '.' bits += self.married_whom (person, from_family, suppress_children) bits += self.inline_notes (person) longnotes = self.long_notes (person, suppress_children, already_described) if (bits != '.' or longnotes or photos or suppress_children or needs_name): # We have something to say about this person. spouse = [] if from_family: from_family_father = from_family.getFather () from_family_mother = from_family.getMother () else: from_family_father = from_family_mother = None for family in person.getFamilyList (): for partner in [family.getFather (), family.getMother ()]: if partner == person or not partner: continue if (suppress_children or (partner != from_family_father and partner != from_family_mother)): for photo in partner.getPhotoList ()[:1]: spouse.append ((self.doc.add_photo, [photo.ref.getPath (), 'right', 2, 2])) if suppress_children and len (already_described): style = "Child" else: style = "Person" if len (spouse): style += "WithSpouse%d" % len (spouse) else: style += "NoSpouse" ret.append ((self.doc.start_table, [style, style])) ret.append ((self.doc.start_row, [])) if suppress_children and len (already_described): # Can't do proper formatting with TextDoc, so cheat. ret.append ((self.doc.start_cell, ["PaddedCell"])) ret.append ((self.doc.end_cell, [])) if len (photos) == 0: ret.append ((self.doc.start_cell, ["NoPhoto"])) ret.append ((self.doc.end_cell, [])) else: ret.append ((self.doc.start_cell, ["Photo"])) for photo in photos[:1]: ret.append ((self.doc.add_photo, [photo.ref.getPath (), 'left', 2, 2])) ret.append ((self.doc.end_cell, [])) ret.append ((self.doc.start_cell, ["Entry"])) ret.append ((self.doc.start_paragraph, ["Entry"])) ret.append ((self.doc.write_text, [name])) if short_form: ret.append ((self.doc.write_text, [" (mentioned above)."])) else: ret.append ((self.doc.write_text, [bits])) ret.append ((self.doc.end_paragraph, [])) ret.append ((self.doc.end_cell, [])) for s in spouse: ret.append ((self.doc.start_cell, ["Photo"])) ret.append (s) ret.append ((self.doc.end_cell, [])) ret.append ((self.doc.end_row, [])) ret.append ((self.doc.end_table, [])) if not short_form: ret.extend (longnotes) return ret def short_occupation (self, person): occupation = '' for event in person.getEventList (): if event.getName () == 'Occupation': if occupation: return '' occupation = event.getDescription () if occupation: return ' (%s)' % occupation return '' def event_info (self, event): info = '' name = event.getName () if name != 'Birth' and name != 'Death' and name != 'Marriage': info += name description = event.getDescription () if description: info += ': ' + description dateobj = event.getDateObj () if dateobj: text = dateobj.getText () if text: info += ' ' + text[0].lower() + text[1:] elif dateobj.getValid (): if dateobj.isRange (): info += ' ' elif (dateobj.getDayValid () and dateobj.getMonthValid () and dateobj.getYearValid ()): info += ' on ' else: info += ' in ' info += dateobj.getDate () placename = event.getPlaceName () if placename: info += ' in ' + placename note = event.getNote () if note: info += ' (' + note + ')' return info def long_born_died (self, person): ret = '' born_info = self.event_info (person.getBirth ()) if born_info: ret = ", born" + born_info died_info = self.event_info (person.getDeath ()) if died_info: if born_info: ret += '; ' else: ret += ', ' ret += 'died' + died_info return ret def Pronoun (self, person): if person.getGender () == RelLib.Person.female: return 'She' else: return 'He' def son_or_daughter (self, person): if person.getGender () == RelLib.Person.female: return 'daughter' else: return 'son' def parents_of (self, person): childof = ('. ' + self.Pronoun (person) + ' is the ' + self.son_or_daughter (person) + ' of ') family = person.getMainParents () ret = '' if family: fathername = mothername = None father = family.getFather () if father: fathername = self.person_name (father) mother = family.getMother () if mother: mothername = self.person_name (mother) if not mother and not father: pass elif not father: ret = childof + mothername elif not mother: ret = childof + fathername else: ret = childof + fathername + ' and ' + mothername return ret + '.' def first_name_or_nick (self, person): nickname = person.getNickName () if nickname: return nickname name = person.getPrimaryName ().getFirstName () return name.split (' ')[0] def title (self, person): name = person.getPrimaryName () t = name.getTitle () if t: return t if person.getGender () == RelLib.Person.female: if name.getType () == 'Married Name': return 'Mrs.' return 'Miss' return 'Mr.' def person_name (self, person): primary = person.getPrimaryName () name = primary.getTitle () if name: name += ' ' first = primary.getFirstName () last = primary.getSurname () first_replaced = first.replace ('?', '') if first_replaced == '': name += self.title (person) else: name += first if last.replace ('?', '') == '': if first_replaced == '': name += ' (unknown)' else: name += ' ' + last suffix = primary.getSuffix () if suffix: name += ', ' + suffix type = primary.getType () if type != 'Birth Name': name += ' (%s)' % type return name def married_whom (self, person, from_family, listing_children = 0): pronoun = ' ' + self.Pronoun (person) first_marriage = 1 ret = '' for family in person.getFamilyList (): mother = family.getMother () for spouse in [family.getFather (), mother]: if spouse == person or not spouse: continue children = '' childlist = family.getChildList () child_count = len (childlist) if ((listing_children or family != from_family) and child_count > 0): children = ', and they had ' if child_count == 1: children += 'a child named ' else: children += str (child_count) + ' children: ' count = 1 for child in childlist: children += self.first_name_or_nick (child) if child_count - count > 1: children += ', ' elif child_count - count == 1: children += ' and ' count += 1 marriage = family.getMarriage () if not first_marriage: ret += pronoun + ' later married ' ret += self.person_name (spouse) if marriage: ret += self.event_info (marriage) ret += children + '.' elif (listing_children or spouse == mother or family != from_family): ret += pronoun + ' married ' ret += self.person_name (spouse) if marriage: ret += self.event_info (marriage) ret += children + '.' first_marriage = 0 return ret def inline_notes (self, person): note = person.getNote () if not (note == '' or note.find ('\n') != -1): return ' ' + note return '' def long_details (self, note, paras): para_end = note.find ('\n') if para_end != -1: paras = self.long_details (note[note.find ('\n') + 1:], paras) else: para_end = len (note) paras.insert (0, (self.doc.end_paragraph, [])) paras.insert (0, (self.doc.write_text, [note[:para_end]])) paras.insert (0, (self.doc.start_paragraph, ['Details'])) return paras def long_notes (self, person, suppress_children = 0, already_described = []): note = person.getNote () if note != '' and note.find ('\n') != -1: paras = self.long_details (note, []) else: paras = [] events = person.getEventList () if len (events) > 0: paras.append ((self.doc.start_paragraph, ['SubEntry'])) paras.append ((self.doc.write_text, ["More about " + self.first_name_or_nick (person) + ":"])) paras.append ((self.doc.end_paragraph, [])) for event in events: paras.append ((self.doc.start_paragraph, ['Details'])) paras.append ((self.doc.write_text, [self.event_info (event)])) paras.append ((self.doc.end_paragraph, [])) return paras def _make_default_style(self): """Make the default output style for the Ancestors report.""" font = TextDoc.FontStyle() font.set(face=TextDoc.FONT_SANS_SERIF,size=16,bold=1,italic=1) para = TextDoc.ParagraphStyle() para.set_font(font) para.set_header_level(1) para.set_alignment(TextDoc.PARA_ALIGN_CENTER) para.set(pad=0.5) para.set_description(_('The style used for the title of the page.')) self.default_style.add_style("Title",para) font = TextDoc.FontStyle() font.set(face=TextDoc.FONT_SANS_SERIF,size=14,italic=1) para = TextDoc.ParagraphStyle() para.set_font(font) para.set_header_level(2) para.set(pad=0.5) para.set_alignment(TextDoc.PARA_ALIGN_CENTER) para.set_description(_('The style used for the generation header.')) self.default_style.add_style("Generation",para) para = TextDoc.ParagraphStyle() para.set(lmargin=1.0,pad=0.25) para.set_description(_('The basic style used for the text display.')) self.default_style.add_style("Entry",para) details_font = TextDoc.FontStyle() details_font.set(face=TextDoc.FONT_SANS_SERIF,size=8,italic=1) para = TextDoc.ParagraphStyle() para.set(lmargin=1.5,pad=0,font = details_font) para.set_description(_('Style for details about a person.')) self.default_style.add_style("Details",para) para = TextDoc.ParagraphStyle() para.set(lmargin=1.0,pad=0.25) para.set_description(_('The basic style used for the text display.')) self.default_style.add_style("SubEntry",para) para = TextDoc.ParagraphStyle() para.set(pad=0.05) para.set_description(_('The basic style used for the text display.')) self.default_style.add_style("Endnotes",para) para = TextDoc.ParagraphStyle() para.set(lmargin=2.5,pad=0.05) para.set_description(_('Introduction to the children.')) self.default_style.add_style("ChildTitle",para) #------------------------------------------------------------------------ # # # #------------------------------------------------------------------------ class AncestorReportDialog(Report.TextReportDialog): def __init__(self,database,person): Report.TextReportDialog.__init__(self,database,person) #------------------------------------------------------------------------ # # Customization hooks # #------------------------------------------------------------------------ def doc_uses_tables (self): return 1 def get_title(self): """The window title for this dialog""" return "%s - %s - GRAMPS" % (_("Ancestors Report"),_("Text Reports")) def get_header(self, name): """The header line at the top of the dialog contents""" return _("Ancestors for %s") % name def get_target_browser_title(self): """The title of the window created when the 'browse' button is clicked in the 'Save As' frame.""" return _("Save Ancestor Report") def get_stylesheet_savefile(self): """Where to save styles for this report.""" return "ancestors_report.xml" def make_default_style(self): _make_default_style(self) def make_report(self): """Create the object that will produce the Ancestors Report. All user dialog has already been handled and the output file opened.""" try: MyReport = AncestorsReport(self.db, self.person, self.max_gen, self.pg_brk, self.doc, self.target_path) MyReport.write_report() except Errors.ReportError, msg: (m1,m2) = msg.messages() ErrorDialog(m1,m2) except Errors.FilterError, msg: (m1,m2) = msg.messages() ErrorDialog(m1,m2) except: import DisplayTrace DisplayTrace.DisplayTrace() #------------------------------------------------------------------------ # # Standalone report function # #------------------------------------------------------------------------ def report(database,person): AncestorReportDialog(database,person) #------------------------------------------------------------------------ # # # #------------------------------------------------------------------------ def get_xpm_image(): return [ "48 48 33 1", " c None", ". c #1A1A1A", "+ c #847B6E", "@ c #B7AC9C", "# c #D1D1D0", "$ c #EEE2D0", "% c #6A655C", "& c #868686", "* c #F1EADF", "= c #5C5854", "- c #B89C73", "; c #E2C8A1", "> c #55524C", ", c #F5EEE6", "' c #4F4E4C", ") c #A19C95", "! c #B3966E", "~ c #CDC8BF", "{ c #F6F2ED", "] c #A6A5A4", "^ c #413F3F", "/ c #D8D1C5", "( c #968977", "_ c #BAB9B6", ": c #FAFAF9", "< c #BEA27B", "[ c #E9DAC2", "} c #9D9385", "| c #E4E3E3", "1 c #7A7062", "2 c #E6D3B4", "3 c #BAA488", "4 c #322E2B", " ", " ", " (+(+++++111%1%%%%===%1 ", " +______________@_@)&==1 ", " +_::::::::::::::*|#_&&}> ", " &_:::::::::::::::{|#]1~}^ ", " +_::::::::::::::::{|#=|~&4 ", " +_::::]]]]]]]]:::::|{':|~&4 ", " +_::::::::::::::::::{'::|~&4 ", " +_:::::::::::::::::::'*::|~&^ ", " +_:::::::::::::::::::'|*::|~}> ", " 1_::::]]]]]]]]]]]]:::'~|{::|_}% ", " 1_:::::::::::::::::::'..4^'=1+%1 ", " +_::::]]]]]]]]]]]]:::|__])&+%=^% ", " 1_::::::::::::::::::::|#__)&&+'^ ", " 1_::::]]]]]]]]]::::::::|#~_])&%^ ", " 1_::::::::::::::::::::{||#~_])14 ", " 1_::::]]]]]]]]]]]]]]]]]]&}#~_]+4 ", " 1_::::::::::::::::::{{{{||#~~@&4 ", " %_::::]]]]]]]]]]]]]]]])))}(~~~&4 ", " %_:::::::::::::::::{{{{{*|#/~_(4 ", " %_::::]]]]]]]]]]]]]]])))))}2;/}4 ", " %_:::::::::::::::{{{{{***||[#~}4 ", " %_::::]]]]]]]]]])]))))))))}2/;)4 ", " %_::::::::::::::{{{{{**|$$[/2~!4 ", " %_::::]]]]]]]]){{{{******$$[2/}4 ", " %_::::::::::::{{{{****$$$$$[2/!4 ", " =_::::]]]]]]])]))))))))})}}[2/!4 ", " %_:::::::::{{{{{{**|$$$$$$[[2;)4 ", " =_::::]]]])]]))))))))))}}}}[22!4 ", " %_::::::::{{{{{|**|$$[$[[[[[22}4 ", " =_::::]]])])))))))))}}}}}}}222-4 ", " =_:::::{{{{{|{*|$$$$$[[[[22222!4 ", " =_::::)]])))))))))}}}}}}(}(2;2-4 ", " =_:::{{{{{{***|$$$$$[[[[22222;-4 ", " =_:::{])))))))))}}}}}}}(}((2;;<4 ", " >_:{{{{{{**|$$$$$[[[[22222;2;;-4 ", " >_{{{{)))))))}}}}}}}(!(((((;;;-4 ", " >_{{{{|**|*$$$$$[[[[22222;;;;;!4 ", " '_{{{{****$$$$$2[[222222;2;;;;-4 ", " '@{{****$$$$$[[[2[222;;2;;;;;;!4 ", " >]{******$$$[$[2[[2222;;;;;;;;!4 ", " '_****$$$$[$[[[[2222;2;;;;;;;;!4 ", " '@__@@@@@@@33<3<<<<<<-<-!!!!!!!4 ", " 44444444444444444444444444444444 ", " ", " ", " "] #------------------------------------------------------------------------ # # # #------------------------------------------------------------------------ from Plugins import register_report register_report( report, _("Ancestors Report"), category=_("Text Reports"), status=(_("Beta")), description= _("Produces a detailed ancestral report."), xpm=get_xpm_image(), author_name="Tim Waugh", author_email="twaugh@redhat.com" )