gramps/src/plugins/drawreport/DescendTree.py
Raphael Ackermann 30d6eebd0b 0002542: Crash whilst generating web pages from command line
Throwing Report Error if center person is not in database. Added catching of Report Error to CommandLineReport
if using GUI it is not possible to cause this, as you have to select a person that is in the DB. However on the command line you can specify any PID and even no person with that pid exists an error was thrown.

svn: r13004
2009-08-14 07:14:25 +00:00

548 lines
20 KiB
Python

#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2007 Donald N. Allingham
# Copyright (C) 2007-2008 Brian G. Matherly
# Copyright (C) 2009 Craig J. Anderson
#
# 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$
"""Reports/Graphical Reports/Descendant Tree"""
#------------------------------------------------------------------------
#
# python modules
#
#------------------------------------------------------------------------
from BasicUtils import name_displayer
from Errors import ReportError
from gen.plug import PluginManager
from gen.plug.docgen import (GraphicsStyle, FontStyle, ParagraphStyle,
FONT_SANS_SERIF, PARA_ALIGN_CENTER)
from gen.plug.menu import TextOption, NumberOption, BooleanOption, PersonOption
from ReportBase import Report, MenuReportOptions, ReportUtils, CATEGORY_DRAW
from SubstKeywords import SubstKeywords
from TransUtils import sgettext as _
#------------------------------------------------------------------------
#
# GRAMPS modules
#
#------------------------------------------------------------------------
pt2cm = ReportUtils.pt2cm
cm2pt = ReportUtils.cm2pt
#------------------------------------------------------------------------
#
# Constants
#
#------------------------------------------------------------------------
_BORN = _('short for born|b.')
_MARR = _('short for married|m.')
_DIED = _('short for died|d.')
_LINE_HORIZONTAL = 1
_LINE_VERTICAL = 2
_LINE_ANGLE = 3
_PERSON_DIRECT = 1
_PERSON_SPOUSE = 2
#------------------------------------------------------------------------
#
# Layout class
#
#------------------------------------------------------------------------
class GenChart(object):
def __init__(self, generations):
self.generations = generations
self.map = {}
self.array = {}
self.sparray = {}
self.max_x = 0
self.max_y = 0
def get_xy(self, x, y):
if y not in self.array:
return 0
return self.array[y].get(x,0)
def set_xy(self, x, y, value):
self.max_x = max(self.max_x,x)
self.max_y = max(self.max_y,y)
if y not in self.array:
self.array[y] = {}
self.array[y][x] = value
def get_sp(self, col_x, row_y):
"""gets whether person at x,y
is a direct descendent or a spouse"""
if (col_x, row_y) not in self.sparray:
return None
return self.sparray[col_x, row_y]
def set_sp(self, col_x, row_y, value):
"""sets whether person at x,y
is a direct descendent or a spouse"""
self.sparray[col_x, row_y] = value
def dimensions(self):
return (self.max_y+1, self.max_x+1)
def not_blank(self, line):
for i in line:
if i and isinstance(i, tuple):
return 1
return 0
#------------------------------------------------------------------------
#
# DescendTree
#
#------------------------------------------------------------------------
class DescendTree(Report):
def __init__(self, database, options_class):
"""
Create DescendTree 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.
dispf - Display format for the output box.
singlep - Whether to scale to fit on a single page.
maxgen - Maximum number of generations to include.
"""
Report.__init__(self, database, options_class)
menu = options_class.menu
self.display = menu.get_option_by_name('dispf').get_value()
self.max_generations = menu.get_option_by_name('maxgen').get_value()
self.force_fit = menu.get_option_by_name('singlep').get_value()
self.incblank = menu.get_option_by_name('incblank').get_value()
pid = menu.get_option_by_name('pid').get_value()
center_person = database.get_person_from_gramps_id(pid)
if (center_person == None) :
raise ReportError(_("Person %s is not in the Database") % pid )
self.showspouse = menu.get_option_by_name('shows').get_value()
name = name_displayer.display_formal(center_person)
self.title = _("Descendant Chart for %s") % name
self.map = {}
self.text = {}
self.box_width = 0
self.box_height = 0
self.lines = 0
self.scale = 1
self.box_gap = 0.2
self.genchart = GenChart(32)
self.apply_filter(center_person.get_handle(),0,0)
self.calc()
if self.force_fit:
self.scale_styles()
def add_person(self, person_handle, col_x, row_y, spouse_level):
"""Add a new person into the x,y position
also sets wether the person is:
- a direct descendent or a spouse
- the max length of the text/box, and number of lines"""
self.genchart.set_sp(col_x, row_y, spouse_level)
if person_handle is not None:
self.genchart.set_xy(col_x, row_y, person_handle)
else:
#make sure that a box prints
self.genchart.set_xy(col_x, row_y, ".")
#make a blank box.
self.text[(col_x, row_y)] = [""]
return
style_sheet = self.doc.get_style_sheet()
pstyle = style_sheet.get_paragraph_style("DC2-Normal")
font = pstyle.get_font()
em = self.doc.string_width(font,"m")
subst = SubstKeywords(self.database, person_handle)
self.text[(col_x, row_y)] = subst.replace_and_clean(self.display)
for line in self.text[(col_x, row_y)]:
this_box_width = self.doc.string_width(font, line) + 2*em
self.box_width = max(self.box_width, this_box_width)
self.lines = max(self.lines, len(self.text[(col_x, row_y)]))
def apply_filter(self, person_handle, col_x, row_y):
"""traverse the ancestors recursively until either the end
of a line is found, or until we reach the maximum number of
generations that we want to deal with"""
if col_x/2 >= self.max_generations:
return 0
person = self.database.get_person_from_handle(person_handle)
self.add_person(person_handle, col_x, row_y, _PERSON_DIRECT)
working_col = 1
next_col = 0
for family_handle in person.get_family_handle_list():
family = self.database.get_family_from_handle(family_handle)
if self.showspouse:
spouse_handle = ReportUtils.find_spouse(person, family)
self.add_person(spouse_handle, col_x, row_y+working_col,
_PERSON_SPOUSE)
working_col += 1
for child_ref in family.get_child_ref_list():
next_col += self.apply_filter(child_ref.ref, col_x+2,
row_y+next_col)
working_col = next_col = max(working_col, next_col)
return working_col
def add_lines(self):
(maxy, maxx) = self.genchart.dimensions()
for y in range(0, maxy+1):
for x in range(0, maxx+1):
# skip columns reserved for rows - no data here
if x%2:
continue
# if we have a direct child to the right of a person
# check to see if the child is a descendant of the person
if self.genchart.get_sp(x+2, y) == _PERSON_DIRECT:
if self.genchart.get_sp(x, y) == _PERSON_DIRECT:
self.genchart.set_xy(x+1, y , _LINE_HORIZONTAL)
continue
elif self.genchart.get_sp(x, y) == _PERSON_SPOUSE and \
self.genchart.get_sp(x, y-1) != _PERSON_DIRECT:
self.genchart.set_xy(x+1, y , _LINE_HORIZONTAL)
continue
else:
continue
self.genchart.set_xy(x+1, y, _LINE_ANGLE)
# look through the entries ABOVE this one. All direct people
# in the next column are descendants until we hit the first
# direct person (marked with _LINE_HORIZONTAL)
last = y-1
while last > 0:
if self.genchart.get_xy(x+1, last) == 0:
self.genchart.set_xy(x+1, last, _LINE_VERTICAL)
else:
break
last -= 1
def write_report(self):
(maxy,maxx) = self.genchart.dimensions()
if maxx <> 1:
maxx = (maxx-1)*2
else:
#no descendants
maxx = 1
maxh = int((self.uh-0.75)/(self.box_height*1.25))
if self.force_fit:
self.print_page(0, maxx, 0, maxy, 0, 0)
else:
starty = 0
coly = 0
while starty < maxy:
startx = 0
colx = 0
while startx < maxx:
stopx = min(maxx, startx+self.generations_per_page*2)
stopy = min(maxy, starty+maxh)
self.print_page(startx, stopx, starty, stopy, colx, coly)
colx += 1
startx += self.generations_per_page*2
coly += 1
starty += maxh
def calc(self):
"""
calc - calculate the maximum width that a box needs to be. From
that and the page dimensions, calculate the proper place to put
the elements on a page.
"""
style_sheet = self.doc.get_style_sheet()
self.add_lines()
self.box_pad_pts = 10
if self.title and self.force_fit:
pstyle = style_sheet.get_paragraph_style("DC2-Title")
tfont = pstyle.get_font()
self.offset = pt2cm(1.25 * tfont.get_size())
gstyle = style_sheet.get_draw_style("DC2-box")
shadow_height = gstyle.get_shadow_space()
else:
# Make space for the page number labels at the bottom.
p = style_sheet.get_paragraph_style("DC2-Normal")
font = p.get_font()
lheight = pt2cm(1.2*font.get_size())
lwidth = pt2cm(1.1*self.doc.string_width(font,"(00,00)"))
self.page_label_x_offset = self.doc.get_usable_width() - lwidth
self.page_label_y_offset = self.doc.get_usable_height() - lheight
self.offset = pt2cm(1.25 * font.get_size())
shadow_height = 0
self.uh = self.doc.get_usable_height() - self.offset - shadow_height
uw = self.doc.get_usable_width() - pt2cm(self.box_pad_pts)
calc_width = pt2cm(self.box_width + self.box_pad_pts) + self.box_gap
self.box_width = pt2cm(self.box_width)
pstyle = style_sheet.get_paragraph_style("DC2-Normal")
font = pstyle.get_font()
self.box_height = self.lines*pt2cm(1.25*font.get_size())
self.scale = 1
if self.force_fit:
(maxy, maxx) = self.genchart.dimensions()
bw = (calc_width/(uw/(maxx+1)))
bh = (self.box_height*(1.25)+self.box_gap)/(self.uh/maxy)
self.scale = max(bw/2, bh)
self.box_width = self.box_width/self.scale
self.box_height = self.box_height/self.scale
self.box_pad_pts = self.box_pad_pts/self.scale
self.box_gap = self.box_gap/self.scale
# maxh = int((self.uh)/(self.box_height+self.box_gap))
maxw = int(uw/calc_width)
# build array of x indices
self.generations_per_page = maxw
self.delta = pt2cm(self.box_pad_pts) + self.box_width + self.box_gap
if not self.force_fit:
calc_width = self.box_width + pt2cm(self.box_pad_pts)
remain = self.doc.get_usable_width() - \
((self.generations_per_page)*calc_width)
self.delta += remain/float(self.generations_per_page)
def scale_styles(self):
"""
Scale the styles for this report. This must be done in the constructor.
"""
style_sheet = self.doc.get_style_sheet()
g = style_sheet.get_draw_style("DC2-box")
g.set_shadow(g.get_shadow(),g.get_shadow_space()/self.scale)
g.set_line_width(g.get_line_width()/self.scale)
style_sheet.add_draw_style("DC2-box",g)
p = style_sheet.get_paragraph_style("DC2-Normal")
font = p.get_font()
font.set_size(font.get_size()/self.scale)
p.set_font(font)
style_sheet.add_paragraph_style("DC2-Normal", p)
self.doc.set_style_sheet(style_sheet)
def print_page(self, startx, stopx, starty, stopy, colx, coly):
if not self.incblank:
blank = True
for y in range(starty, stopy):
for x in range(startx, stopx):
if self.genchart.get_xy(x, y) != 0:
blank = False
break
if not blank: break
if blank: return
self.doc.start_page()
if self.title and self.force_fit:
self.doc.center_text('DC2-title', self.title,
self.doc.get_usable_width()/2,0)
phys_y = 1
bh = self.box_height * 1.25
for y in range(starty, stopy):
phys_x = 0
for x in range(startx, stopx):
value = self.genchart.get_xy(x, y)
if isinstance(value, basestring):
text = '\n'.join(self.text[(x, y)])
xbegin = phys_x*self.delta
yend = phys_y*bh+self.offset
self.doc.draw_box("DC2-box",
text,
xbegin,
yend,
self.box_width,
self.box_height)
elif value == _LINE_HORIZONTAL:
xbegin = phys_x*self.delta
ystart = (phys_y*bh + self.box_height/2.0) + self.offset
xstart = xbegin + self.box_width
xstop = (phys_x+1)*self.delta
self.doc.draw_line('DC2-line', xstart, ystart, xstop,
ystart)
elif value == _LINE_VERTICAL:
ystart = ((phys_y-1)*bh + self.box_height/2.0) + self.offset
ystop = (phys_y*bh + self.box_height/2.0) + self.offset
xlast = (phys_x*self.delta) + self.box_width + self.box_gap
self.doc.draw_line('DC2-line', xlast, ystart, xlast, ystop)
elif value == _LINE_ANGLE:
ystart = ((phys_y-1)*bh + self.box_height/2.0) + self.offset
ystop = (phys_y*bh + self.box_height/2.0) + self.offset
xlast = (phys_x*self.delta) + self.box_width + self.box_gap
xnext = (phys_x+1)*self.delta
self.doc.draw_line('DC2-line', xlast, ystart, xlast, ystop)
self.doc.draw_line('DC2-line', xlast, ystop, xnext, ystop)
if x%2:
phys_x +=1
phys_y += 1
if not self.force_fit:
self.doc.draw_text('DC2-box',
'(%d,%d)' % (colx+1, coly+1),
self.page_label_x_offset,
self.page_label_y_offset)
self.doc.end_page()
#------------------------------------------------------------------------
#
# DescendTreeOptions
#
#------------------------------------------------------------------------
class DescendTreeOptions(MenuReportOptions):
"""
Defines options and provides handling interface.
"""
def __init__(self, name, dbase):
MenuReportOptions.__init__(self, name, dbase)
def add_menu_options(self, menu):
"""
Add options to the menu for the descendant report.
"""
category_name = _("Tree Options")
pid = PersonOption(_("Center Person"))
pid.set_help(_("The center person for the tree"))
menu.add_option(category_name, "pid", pid)
max_gen = NumberOption(_("Generations"), 10, 1, 50)
max_gen.set_help(_("The number of generations to include in the tree"))
menu.add_option(category_name, "maxgen", max_gen)
disp = TextOption( _("Display Format"),
["$n","%s $b" % _BORN,"%s $d" %_DIED] )
disp.set_help(_("Display format for the outputbox."))
menu.add_option(category_name, "dispf", disp)
scale = BooleanOption(_('Sc_ale to fit on a single page'), True)
scale.set_help(_("Whether to scale to fit on a single page."))
menu.add_option(category_name, "singlep", scale)
blank = BooleanOption(_('Include Blank Pages'), True)
blank.set_help(_("Whether to include pages that are blank."))
menu.add_option(category_name, "incblank", blank)
shows = BooleanOption(_('Show Sp_ouses'), True)
shows.set_help(_("Whether to show spouses in the tree."))
menu.add_option(category_name, "shows", shows)
def make_default_style(self,default_style):
"""Make the default output style for the Ancestor Tree."""
## Paragraph Styles:
f = FontStyle()
f.set_size(9)
f.set_type_face(FONT_SANS_SERIF)
p = ParagraphStyle()
p.set_font(f)
p.set_description(_('The basic style used for the text display.'))
default_style.add_paragraph_style("DC2-Normal", p)
f = FontStyle()
f.set_size(16)
f.set_type_face(FONT_SANS_SERIF)
p = ParagraphStyle()
p.set_font(f)
p.set_alignment(PARA_ALIGN_CENTER)
p.set_description(_('The basic style used for the title display.'))
default_style.add_paragraph_style("DC2-Title", p)
## Draw styles
g = GraphicsStyle()
g.set_paragraph_style("DC2-Normal")
g.set_shadow(1, 0.2)
g.set_fill_color((255, 255, 255))
default_style.add_draw_style("DC2-box", g)
g = GraphicsStyle()
g.set_paragraph_style("DC2-Title")
g.set_color((0, 0, 0))
g.set_fill_color((255, 255, 255))
g.set_line_width(0)
default_style.add_draw_style("DC2-title", g)
g = GraphicsStyle()
default_style.add_draw_style("DC2-line", g)
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
pmgr = PluginManager.get_instance()
pmgr.register_report(
name = 'descend_chart',
category = CATEGORY_DRAW,
report_class = DescendTree,
options_class = DescendTreeOptions,
modes = PluginManager.REPORT_MODE_GUI | \
PluginManager.REPORT_MODE_BKI | \
PluginManager.REPORT_MODE_CLI,
translated_name = _("Descendant Tree"),
status = _("Stable"),
author_name = "Donald N. Allingham",
author_email = "don@gramps-project.org",
description = _("Produces a graphical descendant tree"),
)