gramps/src/plugins/NavWebPage.py
Don Allingham 030375a94d * src/plugins/NavWebPage.py: More enhancements
* src/Sort.py: sort places by title


svn: r4026
2005-02-13 03:54:47 +00:00

1263 lines
47 KiB
Python

#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2000-2005 Donald N. Allingham
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Pubilc 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$
"Web Site/Generate Web Site"
#------------------------------------------------------------------------
#
# python modules
#
#------------------------------------------------------------------------
import os
import shutil
from gettext import gettext as _
#------------------------------------------------------------------------
#
# GNOME/gtk
#
#------------------------------------------------------------------------
import gtk
import gnome.ui
#------------------------------------------------------------------------
#
# GRAMPS module
#
#------------------------------------------------------------------------
import os
import RelLib
import const
import GrampsKeys
import GenericFilter
import Sort
import Report
import Errors
import Utils
from QuestionDialog import ErrorDialog
import ReportOptions
import BaseDoc
from NameDisplay import displayer as _nd
from DateHandler import displayer as _dd
import ReportUtils
_NARRATIVE = "narrative.css"
_css = [
'BODY {\nfont-family: "Arial", "Helvetica", sans-serif;',
'letter-spacing: 0.05em;\nbackground-color: #fafaff;',
'color: #003;\n}',
'P,BLOCKQUOTE {\nfont-size: 14px;\n}',
'DIV {\nmargin: 2px;\npadding: 2px;\n}',
'TD {\nvertical-align: top;\n}',
'H1 {',
'font-family: "Verdana", "Bistream Vera Sans", "Arial", "Helvetica", sans-serif;',
'font-weight: bolder;\nfont-size: 160%;\nmargin: 2px;\n}\n',
'H2 {',
'font-family: "Verdana", "Bistream Vera Sans", "Arial", "Helvetica", sans-serif;',
'font-weight: bolder;\nfont-style: italic;\nfont-size: 150%;\n}',
'H3 {\nfont-weight: bold;\nmargin: 0;\npadding-top: 10px;',
'padding-bottom: 10px;\ncolor: #336;\n}',
'H4 {\nmargin-top: 1em;\nmargin-bottom: 0.3em;',
'padding-left: 4px;\nbackground-color: #667;\ncolor: #fff;\n}',
'H5 {\nmargin-bottom: 0.5em;\n}',
'H6 {\nfont-weight: normal;\nfont-style: italic;',
'font-size: 100%;\nmargin-left: 1em;\nmargin-top: 1.3em;',
'margin-bottom: 0.8em;\n}',
'HR {\nheight: 0;\nwidth: 0;\nmargin: 0;\nmargin-top: 1px;',
'margin-bottom: 1px;\npadding: 0;\nborder-top: 0;',
'border-color: #e0e0e9;\n}',
'A:link {\ncolor: #006;\ntext-decoration: underline;\n}',
'A:visited {\ncolor: #669;\ntext-decoration: underline;\n}',
'A:hover {\nbackground-color: #eef;\ncolor: #000;',
'text-decoration: underline;\n}',
'A:active {\nbackground-color: #eef;\ncolor: #000;\ntext-decoration: none;\n}',
'.navheader {\npadding: 4px;\nbackground-color: #e0e0e9;',
'margin: 2px;\n}',
'.navtitle {\nfont-size: 160%;\ncolor: #669;\nmargin: 2px;\n}',
'.navbyline {\nfloat: right;\nfont-size: 14px;\nmargin: 2px;',
'padding: 4px;\n}',
'.nav {\nmargin: 0;\nmargin-bottom: 4px;\npadding: 0;',
'font-size: 14px;\nfont-weight: bold;\n}',
'.summaryarea {',
'min-height: 100px;',
'height: expression(document.body.clientHeight < 1 ? "100px" : "100px" );',
'}',
'.portrait {\njustify: center;\nmargin: 5px;\nmargin-right: 20px;',
'padding: 3px;\nborder-color: #336;\nborder-width: 1px;\n}',
'.snapshot {\nfloat: right;\nmargin: 5px;\nmargin-right: 20px;',
'padding: 3px;\n}',
'.thumbnail {\nheight: 100px;\nborder-color: #336;\nborder-width: 1px;\n}',
'.leftwrap {\nfloat: left;\nmargin: 2px;\nmargin-right: 10px;\n}',
'.rightwrap {\nfloat: right;\nmargin: 2px;\nmargin-left: 10px;\n}',
'TABLE.infolist {\nborder: 0;\nfont-size: 14px;\n}',
'TD.category {\npadding: 3px;\npadding-right: 3em;',
'font-weight: bold;\n}',
'TD.field {\npadding: 3px;\npadding-right: 3em;\n}',
'TD.data {\npadding: 3px;\npadding-right: 3em;',
'font-weight: bold;\n}',
'.pedigree {\nmargin: 0;\nmargin-left: 2em;\npadding: 0;',
'background-color: #e0e0e9;\nborder: 1px;\n}',
'.pedigreeind {\nfont-size: 14px;\nmargin: 0;\npadding: 2em;',
'padding-top: 0.25em;\npadding-bottom: 0.5em;\n}',
'.footer {\nmargin: 1em;\nfont-size: 12px;\nfloat: right;\n}',
]
class BasePage:
def __init__(self,title):
self.title_str = title
def lnkfmt(self,text):
return text.replace(' ','%20')
def display_footer(self,ofile):
ofile.write('<br>\n')
ofile.write('<br>\n')
ofile.write('<hr>\n')
ofile.write('<div class="footer">')
ofile.write('Generated by <a href="http://gramps.sourceforge.net">GRAMPS</a> on 13 December 2004.')
ofile.write('</div>\n')
ofile.write('</body>\n')
ofile.write('</html>\n')
def display_header(self,ofile,title,author=""):
ofile.write('<!DOCTYPE HTML PUBLIC ')
ofile.write('"-//W3C//DTD HTML 4.01 Transitional//EN">\n')
ofile.write('<html>\n<head>\n')
ofile.write('<title>%s</title>\n' % self.title_str)
ofile.write('<meta http-equiv="Content-Type" content="text/html; ')
ofile.write('charset=ISO-8859-1">\n')
ofile.write('<link href="%s" ' % _NARRATIVE)
ofile.write('rel="stylesheet" type="text/css">\n')
ofile.write('<link href="favicon.png" rel="Shortcut Icon">\n')
ofile.write('</head>\n')
ofile.write('<body>\n')
ofile.write('<div class="navheader">\n')
ofile.write(' <div class="navbyline">By: %s</div>\n' % author)
ofile.write(' <h1 class="navtitle">%s</h1>\n' % self.title_str)
ofile.write(' <hr>\n')
ofile.write(' <div class="nav">\n')
ofile.write(' <a href="index.html">Home</a> &nbsp;\n')
ofile.write(' <a href="introduction.html">Introduction</a> &nbsp;\n')
ofile.write(' <a href="surnames.html">Surnames</a> &nbsp;\n')
ofile.write(' <a href="individuals.html">Individuals</a> &nbsp;\n')
ofile.write(' <a href="sources.html">Sources</a> &nbsp;\n')
ofile.write(' <a href="places.html">Places</a> &nbsp;\n')
ofile.write(' <a href="download.html">Download</a> &nbsp;\n')
ofile.write(' <a href="contact.html">Contact</a> &nbsp;\n')
ofile.write(' </div>\n')
ofile.write(' </div>\n')
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class IndividualListPage(BasePage):
def __init__(self, db, title, person_handle_list, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"individuals.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Individuals'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Individuals'))
ofile.write('<p>%s</p>\n' % _('Index of individuals, sorted by last name.'))
ofile.write('<blockquote>\n')
ofile.write('<table class="infolist" cellspacing="0" ')
ofile.write('cellpadding="0" border="0">\n')
ofile.write('<tr><td class="field"><u><b>%s</b></u></td>\n' % _('Surname'))
ofile.write('<td class="field"><u><b>%s</b></u></td>\n' % _('Name'))
ofile.write('</tr>\n')
self.sort = Sort.Sort(db)
person_handle_list.sort(self.sort.by_last_name)
last_surname = ""
for person_handle in person_handle_list:
person = db.get_person_from_handle(person_handle)
n = person.get_primary_name().get_surname()
if n != last_surname:
ofile.write('<tr><td colspan="2">&nbsp;</td></tr>\n')
ofile.write('<tr><td class="category">')
if n != last_surname:
ofile.write('<a name="%s">%s</a>' % (self.lnkfmt(n),n))
else:
ofile.write('&nbsp')
ofile.write('</td><td class="data">')
ofile.write(person.get_primary_name().get_first_name())
ofile.write(' <a href="%s.html">' % person.gramps_id)
ofile.write("<sup>[%s]</sup>" % person.gramps_id)
ofile.write('</a></td></tr>\n')
last_surname = n
ofile.write('</table>\n</blockquote>\n')
self.display_footer(ofile)
ofile.close()
return
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class PlaceListPage(BasePage):
def __init__(self, db, title, handle_list, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"places.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Places'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Places'))
ofile.write('<p>%s</p>\n' % _('Index of all the places in the '
'project.'))
ofile.write('<blockquote>\n')
ofile.write('<table class="infolist" cellspacing="0" ')
ofile.write('cellpadding="0" border="0">\n')
ofile.write('<tr><td class="field"><u>')
ofile.write('<b>%s</b></u></td>\n' % _('Letter'))
ofile.write('<td class="field"><u>')
ofile.write('<b>%s</b></u></td>\n' % _('Place'))
ofile.write('</tr>\n')
self.sort = Sort.Sort(db)
handle_list.sort(self.sort.by_place_title)
last_name = ""
last_letter = ''
for handle in handle_list:
place = db.get_place_from_handle(handle)
n = place.title
if len(n) == 0:
continue
if n[0] != last_letter:
last_letter = n[0]
ofile.write('<tr><td colspan="2">&nbsp;</td></tr>\n')
ofile.write('<tr><td class="category">%s</td>' % last_letter)
ofile.write('<td class="data">')
ofile.write(n)
ofile.write(' <sup><a href="%s.html">' % place.gramps_id)
ofile.write('[%s]' % place.gramps_id)
ofile.write('</a></sup></td></tr>')
elif n != last_letter:
last_surname = n
ofile.write('<tr><td class="category">&nbsp;</td>')
ofile.write('<td class="data">')
ofile.write(n)
ofile.write(' <sup><a href="%s">' % place.gramps_id)
ofile.write('[%s]' % place.gramps_id)
ofile.write('</a></sup></td></tr>')
ofile.write('</table>\n</blockquote>\n')
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class SurnameListPage(BasePage):
def __init__(self, db, title, person_handle_list, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"surnames.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Surnames'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Surnames'))
ofile.write('<p>%s</p>\n' % _('Index of all the surnames in the '
'project. The links lead to a list '
'of individuals in the database with '
'this same surname.'))
ofile.write('<blockquote>\n')
ofile.write('<table class="infolist" cellspacing="0" ')
ofile.write('cellpadding="0" border="0">\n')
ofile.write('<tr><td class="field"><u>')
ofile.write('<b>%s</b></u></td>\n' % _('Letter'))
ofile.write('<td class="field"><u>')
ofile.write('<b>%s</b></u></td>\n' % _('Surname'))
ofile.write('</tr>\n')
self.sort = Sort.Sort(db)
person_handle_list.sort(self.sort.by_last_name)
last_surname = ""
last_letter = ''
for person_handle in person_handle_list:
person = db.get_person_from_handle(person_handle)
n = person.get_primary_name().get_surname()
if len(n) == 0:
continue
if n[0] != last_letter:
last_letter = n[0]
ofile.write('<tr><td class="category">%s</td>' % last_letter)
ofile.write('<td class="data">')
ofile.write('<a href="individuals.html#%s">' % self.lnkfmt(n))
ofile.write(n)
ofile.write('</a></td></tr>')
elif n != last_surname:
last_surname = n
ofile.write('<tr><td class="category">&nbsp;</td>')
ofile.write('<td class="data">')
ofile.write('<a href="individuals.html#%s">' % self.lnkfmt(n))
ofile.write(n)
ofile.write('</a></td></tr>')
ofile.write('</table>\n</blockquote>\n')
self.display_footer(ofile)
ofile.close()
return
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class IntroductionPage(BasePage):
def __init__(self, db, title, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"introduction.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Introduction'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Introduction'))
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class HomePage(BasePage):
def __init__(self, db, title, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"index.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Home'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Home'))
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class SourcesPage(BasePage):
def __init__(self, db, title, handle_list, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"sources.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Sources'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Sources'))
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class DownloadPage(BasePage):
def __init__(self, db, title, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"download.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Download'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Download'))
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class ContactPage(BasePage):
def __init__(self, db, title, html_dir):
BasePage.__init__(self,title)
page_name = os.path.join(html_dir,"contact.html")
ofile = open(page_name, "w")
self.display_header(ofile,_('Contact'),
db.get_researcher().get_name())
ofile.write('<h3>%s</h3>\n' % _('Contact'))
self.display_footer(ofile)
ofile.close()
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class IndividualPage(BasePage):
gender_map = {
RelLib.Person.MALE : const.male,
RelLib.Person.FEMALE : const.female,
RelLib.Person.UNKNOWN : const.unknown,
}
def __init__(self, db, person, title, dirpath, ind_list):
BasePage.__init__(self,title)
self.person = person
self.db = db
self.ind_list = ind_list
self.dirpath = dirpath
gramps_id = self.person.get_gramps_id()
self.sort_name = _nd.sorted(self.person)
self.name = _nd.sorted(self.person)
ofile = open(os.path.join(dirpath,"%s.html" % gramps_id), "w")
self.display_header(ofile, 'My Family Tree',
self.db.get_researcher().get_name())
self.display_ind_general(ofile)
self.display_ind_events(ofile)
self.display_ind_relationships(ofile)
self.display_ind_narrative(ofile)
self.display_ind_sources(ofile)
self.display_ind_pedigree(ofile)
self.display_footer(ofile)
ofile.close()
def display_ind_sources(self,ofile):
sreflist = self.person.get_source_references()
if not sreflist:
return
ofile.write('<h4>%s</h4>\n' % _('Sources'))
ofile.write('<hr>\n')
ofile.write('<table class="infolist" cellpadding="0" ')
ofile.write('cellspacing="0" border="0">\n')
index = 1
for sref in sreflist:
source = self.db.get_source_from_handle(sref.get_base_handle())
author = source.get_author()
title = source.get_title()
publisher = source.get_publication_info()
date = _dd.display(sref.get_date())
ofile.write('<tr><td class="field">%d. ' % index)
values = []
if author:
values.append(author)
if title:
values.append(title)
if publisher:
values.append(publisher)
if date:
values.append(date)
ofile.write(', '.join(values))
ofile.write('</td></tr>\n')
ofile.write('</table>\n')
def display_ind_pedigree(self,ofile):
parent_handle_list = self.person.get_parent_family_handle_list()
if parent_handle_list:
(parent_handle, mrel,frel) = parent_handle_list[0]
family = self.db.get_family_from_handle(parent_handle)
father_id = family.get_father_handle()
mother_id = family.get_mother_handle()
mother = self.db.get_person_from_handle(mother_id)
father = self.db.get_person_from_handle(father_id)
else:
family = None
father = None
mother = None
ofile.write('<h4>%s</h4>\n' % _('Pedigree'))
ofile.write('<hr>\n<br>\n')
ofile.write('<table class="pedigree">\n')
ofile.write('<tr><td>\n')
if father or mother:
ofile.write('<blockquote class="pedigreeind">\n')
if father:
self.pedigree_person(ofile,father)
if mother:
self.pedigree_person(ofile,mother)
ofile.write('<blockquote class="pedigreeind">\n')
if family:
for child_handle in family.get_child_handle_list():
if child_handle == self.person.handle:
ofile.write('| <strong>%s</strong><br>\n' % self.name)
self.pedigree_family(ofile)
else:
child = self.db.get_person_from_handle(child_handle)
self.pedigree_person(ofile,child)
else:
ofile.write('| <strong>%s</strong><br>\n' % self.name)
self.pedigree_family(ofile)
ofile.write('</blockquote>\n')
if father or mother:
ofile.write('</blockquote>\n')
ofile.write('</td>\n</tr>\n</table>\n')
def display_ind_general(self,ofile):
photolist = self.person.get_media_list()
if photolist:
photo_handle = photolist[0].get_reference_handle()
photo = self.db.get_object_from_handle(photo_handle)
newpath = self.person.gramps_id + os.path.splitext(photo.get_path())[1]
shutil.copyfile(photo.get_path(),os.path.join(self.dirpath,newpath))
ofile.write('<div class="snapshot">\n')
ofile.write('<a href="%s">' % newpath)
ofile.write('<img class="thumbnail" border="0" src="%s" ' % newpath)
ofile.write('height="100"></a>')
ofile.write('</div>\n')
ofile.write('<div class="summaryarea">\n')
ofile.write('<h3>%s</h3>\n' % self.sort_name)
ofile.write('<table class="infolist" cellpadding="0" cellspacing="0" ')
ofile.write('border="0">\n')
# Gender
ofile.write('<tr><td class="field">%s</td>\n' % _('Gender'))
gender = self.gender_map[self.person.gender]
ofile.write('<td class="data">%s</td>\n' % gender)
ofile.write('</tr>\n')
# Birth
handle = self.person.get_birth_handle()
if handle:
event = self.db.get_event_from_handle(handle)
ofile.write('<tr><td class="field">%s</td>\n' % _('Birth'))
ofile.write('<td class="data">%s</td>\n' % self.format_event(event))
ofile.write('</tr>\n')
# Death
handle = self.person.get_death_handle()
if handle:
event = self.db.get_event_from_handle(handle)
ofile.write('<tr><td class="field">%s</td>\n' % _('Death'))
ofile.write('<td class="data">%s</td>\n' % self.format_event(event))
ofile.write('</tr>\n')
ofile.write('</table>\n')
ofile.write('</div>\n')
def display_ind_events(self,ofile):
ofile.write('<h4>%s</h4>\n' % _('Events'))
ofile.write('<hr>\n')
ofile.write('<table class="infolist" cellpadding="0" cellspacing="0" ')
ofile.write('border="0">\n')
for event_id in self.person.get_event_list():
event = self.db.get_event_from_handle(event_id)
ofile.write('<tr><td class="field">%s</td>\n' % event.get_name())
ofile.write('<td class="data">\n')
ofile.write(self.format_event(event))
ofile.write('</td>\n')
ofile.write('</tr>\n')
ofile.write('</table>\n')
def display_ind_narrative(self,ofile):
ofile.write('<h4>%s</h4>\n' % _('Narrative'))
ofile.write('<hr>\n')
noteobj = self.person.get_note_object()
if noteobj:
format = noteobj.get_format()
text = noteobj.get()
if format:
text = "<pre>" + "<br>".join(text.split("\n"))
else:
text = "</p><p>".join(text.split("\n"))
ofile.write('<p>%s</p>\n' % text)
def display_parent(self,ofile,handle,title):
use_link = handle in self.ind_list
person = self.db.get_person_from_handle(handle)
ofile.write('<td class="field">%s</td>\n' % title)
ofile.write('<td class="data">')
ofile.write(_nd.display(person))
if use_link:
val = person.gramps_id
ofile.write('&nbsp;<sup><a href="%s.html">[%s]</a></sup>' % (val,val))
ofile.write('</td>\n')
def display_ind_relationships(self,ofile):
parent_list = self.person.get_parent_family_handle_list()
family_list = self.person.get_family_handle_list()
if not parent_list and not family_list:
return
ofile.write('<h4>%s</h4>\n' % _("Relationships"))
ofile.write('<hr>\n')
ofile.write('<table class="infolist" cellpadding="0" cellspacing="0" border="0">\n')
if parent_list:
for (family_handle,mrel,frel) in parent_list:
family = self.db.get_family_from_handle(family_handle)
ofile.write('<tr><td colspan="3">&nbsp;</td></tr>\n')
ofile.write('<tr><td class="category">%s</td>\n' % _("Parents"))
father_handle = family.get_father_handle()
if father_handle:
self.display_parent(ofile,father_handle,_('Father'))
ofile.write('</tr><tr><td>&nbsp;</td>\n')
mother_handle = family.get_mother_handle()
if mother_handle:
self.display_parent(ofile,mother_handle,_('Mother'))
ofile.write('</tr>\n')
ofile.write('<tr><td colspan="3">&nbsp;</td></tr>\n')
if family_list:
ofile.write('<tr><td class="category">%s</td>\n' % _("Spouses"))
first = True
for family_handle in family_list:
family = self.db.get_family_from_handle(family_handle)
self.display_spouse(ofile,family,first)
first = False
childlist = family.get_child_handle_list()
if childlist:
ofile.write('<tr><td>&nbsp;</td>\n')
ofile.write('<td class="field">%s</td>\n' % _("Children"))
ofile.write('<td class="data">\n')
for child_handle in childlist:
use_link = child_handle in self.ind_list
child = self.db.get_person_from_handle(child_handle)
if use_link:
ofile.write('<a href="%s.html">' % child.get_gramps_id())
ofile.write(_nd.display(child))
if use_link:
ofile.write('</a>\n')
ofile.write("<br>\n")
ofile.write('</td>\n</tr>\n')
ofile.write('</table>\n')
def display_spouse(self,ofile,family,first=True):
gender = self.person.get_gender()
reltype = family.get_relationship()
if reltype == RelLib.Family.MARRIED:
if gender == RelLib.Person.FEMALE:
relstr = _("Husband")
elif gender == RelLib.Person.MALE:
relstr = _("Wife")
else:
relstr = _("Partner")
else:
relstr = _("Partner")
spouse_id = ReportUtils.find_spouse(self.person,family)
if spouse_id:
spouse = self.db.get_person_from_handle(spouse_id)
name = _nd.display(spouse)
else:
name = _("unknown")
if not first:
ofile.write('<tr><td>&nbsp;</td></tr>\n')
ofile.write('<td>&nbsp;</td>')
ofile.write('<td class="field">%s</td>\n' % relstr)
ofile.write('<td class="data">')
use_link = spouse_id in self.ind_list
if use_link:
ofile.write('<a href="%s.html">' % spouse.get_gramps_id())
ofile.write(name)
if use_link:
ofile.write('</a>')
ofile.write('</td>\n</tr>\n')
for event_id in family.get_event_list():
event = self.db.get_event_from_handle(event_id)
ofile.write('<tr><td>&nbsp;</td>\n')
ofile.write('<td class="field">%s</td>\n' % event.get_name())
ofile.write('<td class="data">\n')
ofile.write(self.format_event(event))
ofile.write('</td>\n</tr>\n')
def pedigree_person(self,ofile,person,bullet='|'):
person_link = person.handle in self.ind_list
ofile.write('%s ' % bullet)
if person_link:
ofile.write('<a href="%s.html">' % person.gramps_id)
ofile.write(_nd.display(person))
if person_link:
ofile.write('</a>')
ofile.write('<br>\n')
def pedigree_family(self,ofile):
for family_handle in self.person.get_family_handle_list():
rel_family = self.db.get_family_from_handle(family_handle)
spouse_handle = ReportUtils.find_spouse(self.person,rel_family)
if spouse_handle:
spouse = self.db.get_person_from_handle(spouse_handle)
self.pedigree_person(ofile,spouse,'&bull;')
childlist = rel_family.get_child_handle_list()
if childlist:
ofile.write('<blockquote class="pedigreeind">\n')
for child_handle in childlist:
child = self.db.get_person_from_handle(child_handle)
self.pedigree_person(ofile,child)
ofile.write('</blockquote>\n')
def format_event(self,event):
descr = event.get_description()
place = ReportUtils.place_name(self.db,event.get_place_handle())
date = _dd.display(event.get_date_object())
tmap = {'description' : descr, 'date' : date, 'place' : place}
if descr and date and place:
text = _('%(description)s, &nbsp;&nbsp; %(date)s &nbsp;&nbsp; at &nbsp&nbsp; %(place)s') % tmap
elif descr and date:
text = _('%(description)s, &nbsp;&nbsp; %(date)s &nbsp;&nbsp;') % tmap
elif descr:
text = descr
elif date and place:
text = _('%(date)s &nbsp;&nbsp; at &nbsp&nbsp; %(place)s') % tmap
elif date:
text = date
elif place:
text = place
else:
text = '\n'
return text
#------------------------------------------------------------------------
#
# WebReport
#
#------------------------------------------------------------------------
class WebReport(Report.Report):
def __init__(self,database,person,options_class):
"""
Creates WebReport 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
od
HTMLimg
HTMLrestrictinfo
HTMLincpriv
HTMLnotxtsi
HTMLlnktoalphabet
HTMLsplita
HTMLplaceidx
HTMLshorttree
HTMLidxcol
HTMLimagedir
HTMLincid
HTMLidurl
HTMLlinktidx
HTMLext
HTMLtreed
HTMLidxt
HTMLidxbirth
yearso
"""
self.database = database
self.start_person = person
self.options_class = options_class
filter_num = options_class.get_filter_number()
filters = options_class.get_report_filters(person)
filters.extend(GenericFilter.CustomFilters.get_filters())
self.filter = filters[filter_num]
self.template_name = options_class.handler.template_name
self.target_path = options_class.handler.options_dict['HTMLod']
self.ext = options_class.handler.options_dict['HTMLext']
self.id_link = options_class.handler.options_dict['HTMLlinktidx']
self.photos = options_class.handler.options_dict['HTMLimg']
self.restrict = options_class.handler.options_dict['HTMLrestrictinfo']
self.private = options_class.handler.options_dict['HTMLincpriv']
self.srccomments = options_class.handler.options_dict['HTMLcmtxtsi']
self.image_dir = options_class.handler.options_dict['HTMLimagedir']
self.title = options_class.handler.options_dict['HTMLtitle']
self.separate_alpha = options_class.handler.options_dict['HTMLsplita']
self.depth = options_class.handler.options_dict['HTMLtreed']
self.sort = Sort.Sort(self.database)
def get_progressbar_data(self):
return (_("Generate HTML reports - GRAMPS"),
'<span size="larger" weight="bold">%s</span>' %
_("Creating Web Pages"))
def write_report(self):
dir_name = self.target_path
if dir_name == None:
dir_name = os.getcwd()
elif not os.path.isdir(dir_name):
parent_dir = os.path.dirname(dir_name)
if not os.path.isdir(parent_dir):
ErrorDialog(_("Neither %s nor %s are directories") % \
(dir_name,parent_dir))
return
else:
try:
os.mkdir(dir_name)
except IOError, value:
ErrorDialog(_("Could not create the directory: %s") % \
dir_name + "\n" + value[1])
return
except:
ErrorDialog(_("Could not create the directory: %s") % \
dir_name)
return
if self.image_dir:
image_dir_name = os.path.join(dir_name, self.image_dir)
else:
image_dir_name = dir_name
if not os.path.isdir(image_dir_name) and self.photos != 0:
try:
os.mkdir(image_dir_name)
except IOError, value:
ErrorDialog(_("Could not create the directory: %s") % \
image_dir_name + "\n" + value[1])
return
except:
ErrorDialog(_("Could not create the directory: %s") % \
image_dir_name)
return
ind_list = self.filter.apply(self.database,
self.database.get_person_handles(sort_handles=False))
progress_steps = len(ind_list)
if len(ind_list) > 1:
progress_steps = progress_steps+1
progress_steps = progress_steps+1
self.progress_bar_setup(float(progress_steps))
self.write_css(dir_name)
HomePage(self.database,self.title,dir_name)
ContactPage(self.database,self.title,dir_name)
DownloadPage(self.database,self.title,dir_name)
IntroductionPage(self.database,self.title,dir_name)
for person_handle in ind_list:
person = self.database.get_person_from_handle(person_handle)
idoc = IndividualPage(self.database,person,self.title,dir_name, ind_list)
self.progress_bar_step()
while gtk.events_pending():
gtk.main_iteration()
if len(ind_list) > 1:
IndividualListPage(self.database, self.title, ind_list, dir_name)
SurnameListPage(self.database, self.title, ind_list, dir_name)
self.progress_bar_step()
while gtk.events_pending():
gtk.main_iteration()
SourcesPage(self.database,self.title,
self.database.get_source_handles(),dir_name)
PlaceListPage(self.database,self.title,
self.database.get_place_handles(),dir_name)
self.progress_bar_done()
def write_css(self,dir_name):
f = open(os.path.join(dir_name,_NARRATIVE), "w")
f.write('\n'.join(_css))
f.close()
def add_styles(self,doc):
pass
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class WebReportOptions(ReportOptions.ReportOptions):
"""
Defines options and provides handling interface.
"""
def __init__(self,name,person_id=None):
ReportOptions.ReportOptions.__init__(self,name,person_id)
def set_new_options(self):
# Options specific for this report
self.options_dict = {
'HTMLod' : '',
'HTMLimg' : 2,
'HTMLrestrictinfo' : 0,
'HTMLincpriv' : 0,
'HTMLcmtxtsi' : 0,
'HTMLlnktoalphabet' : 0,
'HTMLsplita' : 0,
'HTMLshorttree' : 1,
'HTMLimagedir' : 'images',
'HTMLtitle' : 'My Family Tree',
'HTMLincid' : 0,
'HTMLidurl' : '',
'HTMLlinktidx' : 1,
'HTMLext' : 'html',
'HTMLtreed' : 3,
'HTMLidxt' : '',
'HTMLidxbirth' : 0,
'HTMLplaceidx' : 0,
'HTMLyearso' : 1,
}
self.options_help = {
}
def enable_options(self):
# Semi-common options that should be enabled for this report
self.enable_dict = {
'filter' : 0,
}
def get_report_filters(self,person):
"""Set up the list of possible content filters."""
if person:
name = person.get_primary_name().get_name()
handle = person.get_handle()
else:
name = 'PERSON'
handle = ''
all = GenericFilter.GenericFilter()
all.set_name(_("Entire Database"))
all.add_rule(GenericFilter.Everyone([]))
des = GenericFilter.GenericFilter()
des.set_name(_("Descendants of %s") % name)
des.add_rule(GenericFilter.IsDescendantOf([handle,1]))
df = GenericFilter.GenericFilter()
df.set_name(_("Descendant Families of %s") % name)
df.add_rule(GenericFilter.IsDescendantFamilyOf([handle]))
ans = GenericFilter.GenericFilter()
ans.set_name(_("Ancestors of %s") % name)
ans.add_rule(GenericFilter.IsAncestorOf([handle,1]))
com = GenericFilter.GenericFilter()
com.set_name(_("People with common ancestor with %s") % name)
com.add_rule(GenericFilter.HasCommonAncestorWith([handle]))
return [all,des,df,ans,com]
def add_user_options(self,dialog):
priv_msg = _("Do not include records marked private")
restrict_msg = _("Restrict information on living people")
no_img_msg = _("Do not use images")
no_limg_msg = _("Do not use images for living people")
no_com_msg = _("Do not include comments and text in source information")
imgdir_msg = _("Image subdirectory")
title_msg = _("Web site title")
ext_msg = _("File extension")
sep_alpha_msg = _("Split alphabetical sections to separate pages")
tree_msg = _("Include short ancestor tree")
self.no_private = gtk.CheckButton(priv_msg)
self.no_private.set_active(not self.options_dict['HTMLincpriv'])
self.restrict_living = gtk.CheckButton(restrict_msg)
self.restrict_living.set_active(self.options_dict['HTMLrestrictinfo'])
# FIXME: document this:
# 0 -- no images of any kind
# 1 -- no living images, but some images
# 2 -- any images
images = self.options_dict['HTMLimg']
self.no_images = gtk.CheckButton(no_img_msg)
self.no_images.set_active(not images)
self.no_living_images = gtk.CheckButton(no_limg_msg)
self.no_living_images.set_sensitive(not images)
self.no_living_images.set_active(images in (0,1))
self.no_comments = gtk.CheckButton(no_com_msg)
self.no_comments.set_active(not self.options_dict['HTMLcmtxtsi'])
self.imgdir = gtk.Entry()
self.imgdir.set_text(self.options_dict['HTMLimagedir'])
self.title = gtk.Entry()
self.title.set_text(self.options_dict['HTMLtitle'])
self.linkpath = gtk.Entry()
self.linkpath.set_sensitive(self.options_dict['HTMLincid'])
self.linkpath.set_text(self.options_dict['HTMLidurl'])
self.ext = gtk.combo_box_new_text()
ext_options = ['.html','.htm','.shtml','.php','.php3','.cgi']
for text in ext_options:
self.ext.append_text(text)
def_ext = "." + self.options_dict['HTMLext']
self.ext.set_active(ext_options.index(def_ext))
dialog.add_option(title_msg,self.title)
dialog.add_option(imgdir_msg,self.imgdir)
dialog.add_option(ext_msg,self.ext)
title = _("Privacy")
dialog.add_frame_option(title,None,self.no_private)
dialog.add_frame_option(title,None,self.restrict_living)
dialog.add_frame_option(title,None,self.no_images)
dialog.add_frame_option(title,None,self.no_living_images)
dialog.add_frame_option(title,None,self.no_comments)
self.no_images.connect('toggled',self.on_nophotos_toggled)
def parse_user_options(self,dialog):
"""Parse the privacy options frame of the dialog. Save the
user selected choices for later use."""
self.options_dict['HTMLrestrictinfo'] = int(self.restrict_living.get_active())
self.options_dict['HTMLincpriv'] = int(not self.no_private.get_active())
self.options_dict['HTMLimagedir'] = unicode(self.imgdir.get_text())
self.options_dict['HTMLtitle'] = unicode(self.title.get_text())
#html_ext = unicode(self.ext.entry.get_text().strip())
html_ext = ".html"
if html_ext[0] == '.':
html_ext = html_ext[1:]
self.options_dict['HTMLext'] = html_ext
self.options_dict['HTMLidurl'] = unicode(self.linkpath.get_text().strip())
self.options_dict['HTMLcmtxtsi'] = int(not self.no_comments.get_active())
if self.no_images.get_active():
photos = 0
elif self.no_living_images.get_active():
photos = 1
else:
photos = 2
self.options_dict['HTMLimg'] = photos
self.options_dict['HTMLod'] = dialog.target_path
#------------------------------------------------------------------------
#
# Callback functions from the dialog
#
#------------------------------------------------------------------------
def show_link(self,obj):
self.linkpath.set_sensitive(obj.get_active())
def on_nophotos_toggled(self,obj):
"""Keep the 'restrict photos' checkbox in line with the 'no
photos' checkbox. If there are no photos included, it makes
no sense to worry about restricting which photos are included,
now does it?"""
self.no_living_images.set_sensitive(not obj.get_active())
def make_default_style(self,default_style):
"""Make the default output style for the Web Pages Report."""
pass
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
class WebReportDialog(Report.ReportDialog):
def __init__(self,database,person):
self.database = database
self.person = person
name = "navwebpage"
translated_name = _("Generate Web Site")
self.options_class = WebReportOptions(name)
self.category = const.CATEGORY_WEB
Report.ReportDialog.__init__(self,database,person,self.options_class,
name,translated_name)
self.style_name = None
response = self.window.run()
if response == gtk.RESPONSE_OK:
try:
self.make_report()
except (IOError,OSError),msg:
ErrorDialog(str(msg))
self.window.destroy()
def setup_style_frame(self):
"""The style frame is not used in this dialog."""
pass
def parse_style_frame(self):
"""The style frame is not used in this dialog."""
pass
def get_title(self):
"""The window title for this dialog"""
return "%s - %s - GRAMPS" % (_("Generate Web Site"),_("Web Page"))
def get_target_browser_title(self):
"""The title of the window created when the 'browse' button is
clicked in the 'Save As' frame."""
return _("Target Directory")
def get_target_is_directory(self):
"""This report creates a directory full of files, not a single file."""
return 1
def get_default_directory(self):
"""Get the name of the directory to which the target dialog
box should default. This value can be set in the preferences
panel."""
return self.options_class.handler.options_dict['HTMLod']
def make_document(self):
"""Do Nothing. This document will be created in the
make_report routine."""
pass
def setup_format_frame(self):
"""Do nothing, since we don't want a format frame (HTML only)"""
pass
def setup_post_process(self):
"""The format frame is not used in this dialog. Hide it, and
set the output notebook to always display the html template
page."""
self.output_notebook.set_current_page(1)
def parse_format_frame(self):
"""The format frame is not used in this dialog."""
pass
def make_report(self):
"""Create the object that will produce the web pages."""
try:
MyReport = WebReport(self.database,self.person,self.options_class)
"""
self.target_path,
self.max_gen, self.photos, self.filter,
self.restrict, self.private, self.srccomments,
self.include_link, self.include_mini_tree,
self.selected_style,
self.img_dir_text,self.template_name,
self.use_id,self.id_link,self.use_gendex,self.use_places,
self.html_ext,self.include_alpha_links,
self.separate_alpha,self.n_cols,
self.birth_dates,self.year_only)
"""
MyReport.write_report()
except Errors.FilterError, msg:
(m1,m2) = msg.messages()
ErrorDialog(m1,m2)
#------------------------------------------------------------------------
#
#
#
#------------------------------------------------------------------------
def cl_report(database,name,category,options_str_dict):
clr = Report.CommandLineReport(database,name,category,WebReportOptions,options_str_dict)
# Exit here if show option was given
if clr.show:
return
try:
MyReport = WebReport(database,clr.person,clr.option_class)
MyReport.write_report()
except:
import DisplayTrace
DisplayTrace.DisplayTrace()
#------------------------------------------------------------------------
#
# Empty class to keep the BaseDoc-targeted format happy
#
#------------------------------------------------------------------------
class EmptyDoc:
def __init__(self,styles,type,template,orientation,source=None):
pass
def init(self):
pass
#-------------------------------------------------------------------------
#
#
#
#-------------------------------------------------------------------------
from PluginMgr import register_report
register_report(
name = 'navwebpage',
category = const.CATEGORY_WEB,
report_class = WebReportDialog,
options_class = cl_report,
modes = Report.MODE_GUI | Report.MODE_CLI,
translated_name = _("Narrative Web Site"),
status=(_("Beta")),
description=_("Generates web (HTML) pages for individuals, or a set of individuals."),
)