97b107757c
svn: r13206
5846 lines
223 KiB
Python
5846 lines
223 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2000-2007 Donald N. Allingham
|
|
# Copyright (C) 2007 Johan Gonqvist <johan.gronqvist@gmail.com>
|
|
# Copyright (C) 2007 Gary Burton <gary.burton@zen.co.uk>
|
|
# Copyright (C) 2007-2009 Stephane Charette <stephanecharette@gmail.com>
|
|
# Copyright (C) 2008 Brian G. Matherly
|
|
# Copyright (C) 2008 Jason M. Simanek <jason@bohemianalps.com>
|
|
# Copyright (C) 2008-2009 Rob G. Healey <robhealey1@gmail.com>
|
|
#
|
|
# 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: $
|
|
|
|
"""
|
|
Narrative Web Page generator.
|
|
"""
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Suggested pylint usage:
|
|
# --max-line-length=100 Yes, I know PEP8 suggest 80, but this has longer lines
|
|
# --argument-rgx='[a-z_][a-z0-9_]{1,30}$' Several identifiers are two characters
|
|
# --variable-rgx='[a-z_][a-z0-9_]{1,30}$' Several identifiers are two characters
|
|
#
|
|
#------------------------------------------------------------------------
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# python modules
|
|
#
|
|
#------------------------------------------------------------------------
|
|
from __future__ import with_statement
|
|
import os, sys
|
|
import re
|
|
try:
|
|
from hashlib import md5
|
|
except ImportError:
|
|
from md5 import md5
|
|
import time, datetime
|
|
import locale
|
|
import shutil
|
|
import codecs
|
|
import tarfile
|
|
import tempfile
|
|
import operator
|
|
from TransUtils import sgettext as _
|
|
from cStringIO import StringIO
|
|
from textwrap import TextWrapper
|
|
from unicodedata import normalize
|
|
|
|
# attempt to import the python exif library?
|
|
try:
|
|
import pyexiv2
|
|
pyexiftaglib = True
|
|
except ImportError:
|
|
pyexiftaglib = False
|
|
|
|
from gen.lib.repotype import RepositoryType
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# Set up logging
|
|
#
|
|
#------------------------------------------------------------------------
|
|
import logging
|
|
log = logging.getLogger(".WebPage")
|
|
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# GRAMPS module
|
|
#
|
|
#------------------------------------------------------------------------
|
|
from gen.lib import UrlType, EventType, Person, date, Date, ChildRefType, \
|
|
FamilyRelType, NameType, Name
|
|
import const
|
|
import Sort
|
|
from gen.plug import PluginManager
|
|
from gen.plug.menu import PersonOption, NumberOption, StringOption, \
|
|
BooleanOption, EnumeratedListOption, FilterOption, \
|
|
NoteOption, MediaOption, DestinationOption
|
|
from ReportBase import (Report, ReportUtils, MenuReportOptions, CATEGORY_WEB,
|
|
Bibliography, CSS_FILES )
|
|
import Utils
|
|
from gui.utils import ProgressMeter
|
|
import ThumbNails
|
|
import ImgManip
|
|
import Mime
|
|
from Utils import probably_alive, xml_lang
|
|
from QuestionDialog import ErrorDialog, WarningDialog
|
|
from BasicUtils import name_displayer as _nd
|
|
from DateHandler import displayer as _dd
|
|
from DateHandler import parser as _dp
|
|
from gen.proxy import PrivateProxyDb, LivingProxyDb
|
|
from gen.lib.eventroletype import EventRoleType
|
|
from libhtmlconst import _CHARACTER_SETS, _CC, _COPY_OPTIONS
|
|
|
|
# import HTML Class from
|
|
# src/plugins/lib/libhtml.py
|
|
from libhtml import Html
|
|
|
|
# import styled notes from
|
|
# src/plugins/lib/libhtmlbackend.py
|
|
from libhtmlbackend import HtmlBackend
|
|
#------------------------------------------------------------------------
|
|
#
|
|
# constants
|
|
#
|
|
#------------------------------------------------------------------------
|
|
# Translatable strings for variables within this plugin
|
|
# gettext carries a huge footprint with it.
|
|
AHEAD = _('Attributes')
|
|
CITY = _('City')
|
|
COUNTY = _('County')
|
|
COUNTRY = _('Country')
|
|
DHEAD = _('Date')
|
|
DESCRHEAD = _('Description')
|
|
LATITUDE = _('Latitude')
|
|
LOCATIONS = _('Alternate Locations')
|
|
LONGITUDE = _('Longitude')
|
|
NHEAD = _('Notes')
|
|
PARISH = _('Church Parish')
|
|
PHEAD = _('Place')
|
|
PHONE = _('Phone')
|
|
POSTAL = _('Postal Code')
|
|
SHEAD = _('Sources')
|
|
ST = _('Status')
|
|
STATE = _('State/ Province')
|
|
STREET = _('Street')
|
|
THEAD = _('Type')
|
|
TMPL = _('Temple')
|
|
VHEAD = _('Value')
|
|
|
|
# define clear blank line for proper styling
|
|
fullclear = Html("div", class_ = "fullclear", inline = True)
|
|
|
|
# Names for stylesheets
|
|
_NARRATIVESCREEN = "narrative-screen.css"
|
|
_NARRATIVEPRINT = "narrative-print.css"
|
|
|
|
# variables for alphabet_navigation()
|
|
_PERSON, _PLACE = 0, 1
|
|
|
|
# Web page filename extensions
|
|
_WEB_EXT = ['.html', '.htm', '.shtml', '.php', '.php3', '.cgi']
|
|
|
|
_INCLUDE_LIVING_VALUE = 99 # Arbitrary number
|
|
_NAME_COL = 3
|
|
|
|
_DEFAULT_MAX_IMG_WIDTH = 800 # resize images that are wider than this (settable in options)
|
|
_DEFAULT_MAX_IMG_HEIGHT = 600 # resize images that are taller than this (settable in options)
|
|
_WIDTH = 160
|
|
_HEIGHT = 50
|
|
_VGAP = 10
|
|
_HGAP = 30
|
|
_SHADOW = 5
|
|
_XOFFSET = 5
|
|
|
|
wrapper = TextWrapper()
|
|
wrapper.break_log_words = True
|
|
wrapper.width = 20
|
|
|
|
_html_dbl_quotes = re.compile(r'([^"]*) " ([^"]*) " (.*)', re.VERBOSE)
|
|
_html_sng_quotes = re.compile(r"([^']*) ' ([^']*) ' (.*)", re.VERBOSE)
|
|
_html_replacement = {
|
|
"&" : "&",
|
|
">" : ">",
|
|
"<" : "<",
|
|
}
|
|
|
|
# This command then defines the 'html_escape' option for escaping
|
|
# special characters for presentation in HTML based on the above list.
|
|
def html_escape(text):
|
|
"""Convert the text and replace some characters with a &# variant."""
|
|
|
|
# First single characters, no quotes
|
|
text = ''.join([_html_replacement.get(c, c) for c in text])
|
|
|
|
# Deal with double quotes.
|
|
while 1:
|
|
m = _html_dbl_quotes.match(text)
|
|
if not m:
|
|
break
|
|
text = m.group(1) + '“' + m.group(2) + '”' + m.group(3)
|
|
# Replace remaining double quotes.
|
|
text = text.replace('"', '"')
|
|
|
|
# Deal with single quotes.
|
|
text = text.replace("'s ", '’s ')
|
|
while 1:
|
|
m = _html_sng_quotes.match(text)
|
|
if not m:
|
|
break
|
|
text = m.group(1) + '‘' + m.group(2) + '’' + m.group(3)
|
|
# Replace remaining single quotes.
|
|
text = text.replace("'", ''')
|
|
|
|
return text
|
|
|
|
def name_to_md5(text):
|
|
"""This creates an MD5 hex string to be used as filename."""
|
|
return md5(text).hexdigest()
|
|
|
|
def conf_priv(obj):
|
|
if obj.get_privacy() != 0:
|
|
return ' priv="%d"' % obj.get_privacy()
|
|
else:
|
|
return ''
|
|
|
|
class BasePage(object):
|
|
"""
|
|
This is the base class to write certain HTML pages.
|
|
"""
|
|
|
|
def __init__(self, report, title, gid=None):
|
|
"""
|
|
report - instance of NavWebReport
|
|
title - text for the <title> tag
|
|
gid - Gramps ID
|
|
"""
|
|
|
|
# class to do conversion of styled notes to html markup
|
|
self._backend = HtmlBackend()
|
|
|
|
self.report = report
|
|
self.title_str = title
|
|
self.gid = gid
|
|
self.src_list = {}
|
|
|
|
self.page_title = ""
|
|
|
|
self.author = Utils.get_researcher().get_name()
|
|
if self.author:
|
|
self.author = self.author.replace(',,,', '')
|
|
self.up = False
|
|
|
|
# TODO. All of these attributes are not necessary, because we have
|
|
# also the options in self.options. Besides, we need to check which
|
|
# are still required.
|
|
self.html_dir = report.options['target']
|
|
self.ext = report.options['ext']
|
|
self.noid = report.options['nogid']
|
|
self.linkhome = report.options['linkhome']
|
|
self.create_media = report.options['gallery']
|
|
|
|
def dump_attributes(self, attrlist, showsrc):
|
|
"""
|
|
dump attribute for object presented in display_attr_list()
|
|
|
|
@param: attrlist = attribute object list
|
|
@param: showsrc = show source references or not?
|
|
"""
|
|
|
|
# begin attributes table
|
|
with Html("table", class_ = "infolist attrlist") as table:
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
attr_header_row = [
|
|
[THEAD, 'Type'],
|
|
[VHEAD, 'Value'],
|
|
[SHEAD, 'Sources'],
|
|
[NHEAD, 'Notes'] ]
|
|
|
|
for (label, colclass) in attr_header_row:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass, inline = True)
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for attr in attrlist:
|
|
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
|
|
attr_data_row = [
|
|
[str(attr.get_type() ) ],
|
|
[attr.get_value() ] ]
|
|
|
|
if showsrc:
|
|
source_row = self.get_citation_links(attr.get_source_references())
|
|
attr_data_row.append(source_row)
|
|
|
|
attr_data_row.append(self.dump_notes(attr.get_note_list() ) )
|
|
|
|
index = 0
|
|
for value in attr_data_row:
|
|
colclass = attr_header_row[index][1]
|
|
|
|
# determine if same row or not?
|
|
samerow = True if (value[0] == " " or colclass == "Type") else False
|
|
|
|
trow += Html("td", value[0], class_ = "Column%s" % colclass,
|
|
inline = samerow)
|
|
index += 1
|
|
|
|
# return table to its callers
|
|
return table
|
|
|
|
def get_citation_links(self, source_ref_list):
|
|
self.bibli = Bibliography()
|
|
|
|
gid_list = []
|
|
lnk = (self.report.cur_fname, self.page_title, self.gid)
|
|
|
|
for sref in source_ref_list:
|
|
handle = sref.get_reference_handle()
|
|
gid_list.append(sref)
|
|
|
|
if handle in self.src_list:
|
|
if lnk not in self.src_list[handle]:
|
|
self.src_list[handle].append(lnk)
|
|
else:
|
|
self.src_list[handle] = [lnk]
|
|
|
|
text = ""
|
|
if len(gid_list):
|
|
text = text + " <sup>"
|
|
for ref in gid_list:
|
|
index, key = self.bibli.add_reference(ref)
|
|
id_ = "%d%s" % (index+1, key)
|
|
text = text + '<a href="#sref%s">%s</a>' % (id_, id_)
|
|
text = text + "</sup>"
|
|
|
|
# return citation list text to its callers
|
|
return text
|
|
|
|
def get_note_format(self, note):
|
|
"""
|
|
will get the note from the database, and will return either the
|
|
styled text or plain note
|
|
"""
|
|
|
|
# retrieve the body of the note
|
|
note_text = note.get()
|
|
|
|
# styled notes
|
|
htmlnotetext = self.styled_note(note.get_styledtext(),
|
|
note.get_format())
|
|
if htmlnotetext:
|
|
text = htmlnotetext
|
|
else:
|
|
text = Html('p', note_text)
|
|
|
|
# return text of the note to its callers
|
|
return text
|
|
|
|
#################################################
|
|
#
|
|
# Will produce styled notes for NarrativeWeb by using:
|
|
# src/plugins/lib/libhtmlbackend.py
|
|
#
|
|
#################################################
|
|
|
|
def styled_note(self, styledtext, format):
|
|
"""
|
|
styledtext : assumed a StyledText object to write
|
|
format : = 0 : Flowed, = 1 : Preformatted
|
|
style_name : name of the style to use for default presentation
|
|
"""
|
|
text = str(styledtext)
|
|
|
|
if not text:
|
|
return ''
|
|
|
|
s_tags = styledtext.get_tags()
|
|
#FIXME: following split should be regex to match \n\s*\n instead?
|
|
markuptext = self._backend.add_markup_from_styled(text, s_tags,
|
|
split='\n\n')
|
|
htmllist = Html("div", id="grampsstylednote")
|
|
if format == 1:
|
|
#preformatted, retain whitespace.
|
|
#so use \n\n for paragraph detection
|
|
#FIXME: following split should be regex to match \n\s*\n instead?
|
|
htmllist += Html('pre', indent=None, inline = True)
|
|
for line in markuptext.split('\n\n'):
|
|
htmllist += Html('p')
|
|
for realline in line.split('\n'):
|
|
htmllist += realline
|
|
htmllist += Html('br')
|
|
|
|
elif format == 0:
|
|
#flowed
|
|
#FIXME: following split should be regex to match \n\s*\n instead?
|
|
for line in markuptext.split('\n\n'):
|
|
htmllist += Html('p')
|
|
htmllist += line
|
|
|
|
return htmllist
|
|
|
|
def dump_notes(self, notelist):
|
|
"""
|
|
dump out of list of notes with very little elements of its own
|
|
|
|
@param: notelist -- list of notes
|
|
"""
|
|
|
|
if not notelist:
|
|
return " "
|
|
db = self.report.database
|
|
|
|
# begin unordered list
|
|
unordered = Html('ul')
|
|
|
|
for notehandle in notelist:
|
|
note = db.get_note_from_handle(notehandle)
|
|
unordered += self.get_note_format(note)
|
|
|
|
# return unordered note list to its callers
|
|
return unordered
|
|
|
|
def display_event_row(self, evt, evt_ref, showplc, showdescr, showsrc,
|
|
shownote, subdirs, hyp):
|
|
"""
|
|
display the event row for class IndividualPage
|
|
|
|
for more information: see get_event_data()
|
|
"""
|
|
db = self.report.database
|
|
|
|
# check to see if place is already in self.place_list?
|
|
lnk = (self.report.cur_fname, self.page_title, self.gid)
|
|
place_handle = evt.get_place_handle()
|
|
if place_handle:
|
|
if place_handle in self.place_list:
|
|
if lnk not in self.place_list[place_handle]:
|
|
self.place_list[place_handle].append(lnk)
|
|
else:
|
|
self.place_list[place_handle] = [lnk]
|
|
|
|
place = self.place_link(place_handle,
|
|
ReportUtils.place_name(db,
|
|
place_handle), up=True)
|
|
else:
|
|
place = ''
|
|
|
|
# begin event table row
|
|
trow = Html("tr")
|
|
|
|
# get event data
|
|
"""
|
|
for more information: see get_event_data()
|
|
"""
|
|
event_data = self.get_event_data(evt, evt_ref, showplc, showdescr, showsrc,
|
|
shownote, subdirs, hyp)
|
|
|
|
for (label, colclass, data) in event_data:
|
|
data = data or " "
|
|
|
|
# determine if information will fit on same line?
|
|
samerow = True if (data == " " or colclass == "Date") else False
|
|
|
|
trow += Html("td", data, class_ = "Column%s" % colclass, inline = samerow)
|
|
|
|
# return events table row to its callers
|
|
return trow
|
|
|
|
def event_link(self, eventtype, handle, gid=None, up=False):
|
|
""" createsa hyperlink for an event based on its type """
|
|
|
|
url = self.report.build_url_fname_html(handle, 'evt', up)
|
|
|
|
|
|
evt_hyper = Html('a', eventtype, href=url, title=eventtype)
|
|
if not self.noid and gid:
|
|
evt_hyper += Html('span', ' [%s] ' % gid, class_ = "grampsid",
|
|
inline = True)
|
|
|
|
# return event hyper link to its callers
|
|
return evt_hyper
|
|
|
|
def get_event_data(self, evt, evt_ref, showplc, showdescr, showsrc, shownote,
|
|
subdirs, hyper, gid=None):
|
|
"""
|
|
retrieve event data from event and evt_ref
|
|
|
|
@param: evt = event from database
|
|
@param: evt_ref = eent reference
|
|
@param: showplc = show the event place or not?
|
|
@param: showdescr = to show the event description or not?
|
|
@param: showsrc = to show the event source references or not?
|
|
@param: shownote = show notes or not?
|
|
@param: subdirs = either True or False
|
|
@param: hyper = to hyperlink the event type or not?
|
|
"""
|
|
db = self.report.database
|
|
|
|
# get event type
|
|
evt_type = get_event_type(evt, evt_ref)
|
|
|
|
# get hyperlink or not?
|
|
if hyper:
|
|
evt_hyper = self.event_link(evt_type, evt_ref.ref, gid, subdirs)
|
|
else:
|
|
evt_hyper = evt_type
|
|
|
|
# get place name
|
|
place_handle = evt.get_place_handle()
|
|
place = db.get_place_from_handle(place_handle)
|
|
|
|
place_hyper = None
|
|
if place:
|
|
place_name = ReportUtils.place_name(db, place_handle)
|
|
place_hyper = self.place_link(place_handle, place_name,
|
|
place.gramps_id, subdirs)
|
|
|
|
# wrap it all up and return to its callers
|
|
# position 0 = translatable label, position 1 = column class
|
|
# position 2 = data
|
|
info = [
|
|
[THEAD, 'Type', evt_hyper],
|
|
[DHEAD, 'Date', _dd.display(evt.get_date_object() )] ]
|
|
|
|
if showplc:
|
|
place_row = [PHEAD, 'Place', place_hyper]
|
|
info.append(place_row)
|
|
|
|
if showdescr:
|
|
descr = evt.get_description()
|
|
descr_row = [DESCRHEAD, 'Description', descr]
|
|
info.append(descr_row)
|
|
|
|
if showsrc:
|
|
srcrefs = evt.get_source_references()
|
|
source_row = [SHEAD, 'Sources', srcrefs]
|
|
info.append(source_row)
|
|
|
|
if shownote:
|
|
notelist = evt.get_note_list()
|
|
notelist.extend(evt_ref.get_note_list() )
|
|
note_row = [NHEAD, 'Notes', notelist]
|
|
info.append(note_row)
|
|
|
|
# return event data information to its callers
|
|
return info
|
|
|
|
def dump_ordinance(self, db, ldsobj, LDSType='Person'):
|
|
"""
|
|
will dump the LDS Ordinance information for either
|
|
a person or a family ...
|
|
|
|
@param: db -- the database in use
|
|
@param: ldsobj -- either person or family
|
|
"""
|
|
objectldsord = ldsobj.lds_ord_list
|
|
if not objectldsord:
|
|
return None
|
|
numberofords = len(objectldsord)
|
|
|
|
def create_LDS_header_row(LDSType):
|
|
""" create the header row for this section """
|
|
|
|
# begin HTML row
|
|
trow = Html("tr")
|
|
|
|
header_row = [
|
|
[THEAD, 'LDSType' ],
|
|
[DHEAD, 'LDSDate' ],
|
|
[TMPL, 'LDSTemple' ],
|
|
[PHEAD, 'LDSPlace' ],
|
|
[ST, 'LDSStatus' ],
|
|
[_('Sealed to '), 'LDSSealed'],
|
|
[SHEAD, 'LDSSources']
|
|
]
|
|
|
|
# finish the label's missing piece
|
|
header_row[5][0] += _('Parents') if LDSType == 'Person' else _('Spouse')
|
|
|
|
for (label, colclass) in header_row:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass, inline = True)
|
|
|
|
# return row back to module
|
|
return trow
|
|
|
|
# begin LDS ordinance table and table head
|
|
with Html("table", class_ = "infolist ldsordlist") as table:
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
# get LDS ord header row
|
|
thead += create_LDS_header_row(LDSType)
|
|
|
|
# start table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for row in range(1, (numberofords + 1)):
|
|
|
|
# get ordinance for this row
|
|
ord = objectldsord[(row - 1)]
|
|
|
|
# 0 = column class, 1 = ordinance data
|
|
lds_ord_data = [
|
|
['LDSType', ord.type2xml()],
|
|
['LDSDate', _dd.display(ord.get_date_object() )],
|
|
['LDSTemple', ord.get_temple()],
|
|
['LDSPlace', ReportUtils.place_name(db, ord.get_place_handle() )],
|
|
['LDSStatus', ord.get_status()],
|
|
['LDSSealed', ord.get_family_handle()],
|
|
['LDSSources', self.get_citation_links(ord.get_source_references() )],
|
|
]
|
|
|
|
# begin ordinance rows
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
|
|
for col in range(1, (len(lds_ord_data) + 1)):
|
|
|
|
# column class for styling
|
|
colclass = lds_ord_data[(col - 1)][0]
|
|
|
|
# actual column data
|
|
value = lds_ord_data[(col - 1)][1]
|
|
value = value or " "
|
|
|
|
trow += Html("td", value, class_ = "Column%s" % colclass,
|
|
inline = True if value == " " else False)
|
|
|
|
# return table to its callers
|
|
return table
|
|
|
|
def source_link(self, handle, name, gid=None, up=False):
|
|
|
|
url = self.report.build_url_fname_html(handle, 'src', up)
|
|
# begin hyperlink
|
|
hyper = Html('a', html_escape(name), href=url, title=html_escape(name))
|
|
if not self.noid and gid:
|
|
hyper += Html('span', '[%s]' % gid, class_ = "grampsid", inline = True)
|
|
|
|
# return hyperlink to its callers
|
|
return hyper
|
|
|
|
def dump_source_references(self, db, sourcelist):
|
|
""" Dump a list of source references """
|
|
|
|
ordered = Html('ol')
|
|
list = Html('li')
|
|
ordered += list
|
|
|
|
source_dict = {}
|
|
# Sort the sources
|
|
for handle in sourcelist:
|
|
|
|
# if source is not None, then add it?
|
|
source = db.get_source_from_handle(handle)
|
|
if source is not None:
|
|
key = source.get_title() + str(source.get_gramps_id())
|
|
source_dict[key] = (source, handle)
|
|
keys = sorted(source_dict, key=locale.strxfrm)
|
|
|
|
for cindex, key in enumerate(keys):
|
|
(source, handle) = source_dict[key]
|
|
source_title = source.get_title()
|
|
|
|
list += self.source_link(handle, title, cindex+1, source.gramps_id, True)
|
|
|
|
# return ordered list to its callers
|
|
return ordered
|
|
|
|
def dump_addresses(self, addrobj, showsrc=True):
|
|
"""
|
|
will display an object's addresses, url list, note list,
|
|
and source references.
|
|
|
|
@param: addrobj = either person or repository address object
|
|
@param: showsrc = True -- person
|
|
False -- repository
|
|
"""
|
|
|
|
def write_address_header(showsrc):
|
|
""" create header row for address """
|
|
|
|
trow = Html("tr")
|
|
addr_header = [
|
|
[DHEAD, "Date"],
|
|
[STREET, "StreetAddress"],
|
|
[CITY, "City"],
|
|
[COUNTY, "County"],
|
|
[STATE, "State"],
|
|
[COUNTRY, "Cntry"],
|
|
[POSTAL, "Postalcode"],
|
|
[PHONE, "Phone"] ]
|
|
|
|
# if showsrc = True -- an individual's address else repository
|
|
if showsrc:
|
|
addr_header.append([SHEAD, 'Sources'])
|
|
|
|
for (label, colclass) in addr_header:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass,
|
|
inline = True)
|
|
|
|
# return table header row back to module
|
|
return trow, addr_header
|
|
|
|
# begin summaryarea division
|
|
with Html("div", id="summaryarea") as summaryarea:
|
|
|
|
# begin address table
|
|
with Html("table") as table:
|
|
summaryarea += table
|
|
|
|
# get table class based on either person or repository
|
|
table.attr = 'class = "infolist addrlist" ' if showsrc \
|
|
else 'class = "infolist repolist" '
|
|
|
|
# begin table head
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
# add header row
|
|
header_row, addr_header = write_address_header(showsrc)
|
|
thead += header_row
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
# get address list from an object; either repository or person
|
|
for address in addrobj.get_address_list():
|
|
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
|
|
addr_data_row = [
|
|
[_dd.display(address.get_date_object() )],
|
|
[address.get_street()],
|
|
[address.get_city()],
|
|
[address.get_county()],
|
|
[address.get_state()],
|
|
[address.get_country()],
|
|
[address.get_postal_code()],
|
|
[address.get_phone()] ]
|
|
|
|
# get source citation list
|
|
if showsrc:
|
|
addr_data_row.append([self.get_citation_links(
|
|
address.get_source_references() )])
|
|
|
|
index = 0
|
|
for value in addr_data_row:
|
|
colclass = addr_header[index][1]
|
|
value = value or " "
|
|
|
|
trow += Html("td", value, class_ = "Column%s" % colclass,
|
|
inline = True)
|
|
index += 1
|
|
|
|
# address: note list
|
|
notelist = self.display_note_list(address.get_note_list())
|
|
if notelist is not None:
|
|
summaryarea += notelist
|
|
|
|
# return summaryarea division to its callers
|
|
return summaryarea
|
|
|
|
def get_copyright_license(self, copyright, up=False):
|
|
"""
|
|
will return either the text or image of the copyright license
|
|
"""
|
|
|
|
text = ''
|
|
if copyright == 0:
|
|
if self.author:
|
|
year = date.Today().get_year()
|
|
text = '© %(year)d %(person)s' % {
|
|
'person' : self.author,
|
|
'year' : year}
|
|
elif 0 < copyright <= len(_CC):
|
|
# Note. This is a URL
|
|
fname = '/'.join(["images", "somerights20.gif"])
|
|
url = self.report.build_url_fname(fname, None, up=False)
|
|
text = _CC[copyright] % {'gif_fname' : url}
|
|
|
|
# return text or image to its callers
|
|
return text
|
|
|
|
def get_name(self, person, maiden_name = None):
|
|
"""
|
|
Return person's name, unless maiden_name given, unless married_name
|
|
listed.
|
|
"""
|
|
|
|
# name_format is the name format that you set in options
|
|
name_format = self.report.options['name_format']
|
|
|
|
# Get all of a person's names
|
|
primary_name = person.get_primary_name()
|
|
married_name = None
|
|
names = [primary_name] + person.get_alternate_names()
|
|
for name in names:
|
|
if int(name.get_type()) == NameType.MARRIED:
|
|
married_name = name
|
|
break # use first
|
|
# Now, decide which to use:
|
|
if maiden_name is not None:
|
|
if married_name is not None:
|
|
name = Name(married_name)
|
|
else:
|
|
name = Name(primary_name)
|
|
name.set_surname(maiden_name)
|
|
else:
|
|
name = Name(primary_name)
|
|
name.set_display_as(name_format)
|
|
return _nd.display_name(name)
|
|
|
|
def display_attr_list(self, attrlist, showsrc):
|
|
"""
|
|
will display an object's attributes
|
|
|
|
@param: attrlist = a list of attributes
|
|
@param: objtype = which object are we using? Ex. Person, Event, Photo
|
|
@param: showsrc = to shown source references or not?
|
|
"""
|
|
if not attrlist:
|
|
return None
|
|
|
|
# begin attributes division and section title
|
|
with Html("div", class_ = "ubsection", id="attributes") as section:
|
|
section += Html('h4', _('Attributes'), inline = True)
|
|
|
|
# add returned table to section
|
|
section += self.dump_attributes(attrlist, showsrc)
|
|
|
|
# return section to its caller
|
|
return section
|
|
|
|
def write_footer(self):
|
|
"""
|
|
Will create and display the footer section of each page...
|
|
|
|
@param: bottom -- whether to specify location of footer section or not?
|
|
"""
|
|
db = self.report.database
|
|
|
|
# begin footer division
|
|
with Html("div", id="footer") as footer:
|
|
|
|
footer_note = self.report.options['footernote']
|
|
if footer_note:
|
|
note = db.get_note_from_gramps_id(footer_note)
|
|
note_text = self.get_note_format(note)
|
|
|
|
user_footer = Html("div", id='user_footer')
|
|
footer += user_footer
|
|
|
|
# attach note
|
|
user_footer += note_text
|
|
|
|
value = _dd.display(date.Today())
|
|
msg = _('Generated by <a href="%(homepage)s">'
|
|
'GRAMPS</a> on %(date)s') % {
|
|
'date': value, 'homepage' : const.URL_HOMEPAGE
|
|
}
|
|
|
|
# optional "link-home" feature; see bug report #2736
|
|
if self.report.options['linkhome']:
|
|
home_person = db.get_default_person()
|
|
if home_person:
|
|
home_person_url = self.report.build_url_fname_html(
|
|
home_person.handle,
|
|
'ppl',
|
|
self.up)
|
|
|
|
home_person_name = self.get_name(home_person)
|
|
msg += _(' Created for <a href="%s">%s</a>') % (
|
|
home_person_url, home_person_name)
|
|
|
|
# creation date
|
|
footer += Html('p', msg, id='createdate')
|
|
|
|
# get copyright license for all pages
|
|
copy_nr = self.report.copyright
|
|
|
|
text = ''
|
|
if copy_nr == 0:
|
|
if self.author:
|
|
year = date.Today().get_year()
|
|
text = '© %(year)d %(person)s' % {
|
|
'person' : self.author,
|
|
'year' : year}
|
|
elif 0 < copy_nr <= len(_CC):
|
|
# Note. This is a URL
|
|
fname = '/'.join(["images", "somerights20.gif"])
|
|
url = self.report.build_url_fname(fname, None, self.up)
|
|
text = _CC[copy_nr] % {'gif_fname' : url}
|
|
footer += Html('p', text, id='copyright')
|
|
|
|
# return footer to its callers
|
|
return footer
|
|
|
|
def write_header(self, title):
|
|
"""
|
|
Note. 'title' is used as currentsection in the navigation links and
|
|
as part of the header title.
|
|
"""
|
|
db = self.report.database
|
|
|
|
# Header constants
|
|
xmllang = xml_lang()
|
|
_META1 = 'name="generator" content="%s %s %s"' % (
|
|
const.PROGRAM_NAME, const.VERSION, const.URL_HOMEPAGE
|
|
)
|
|
_META2 = 'name="author" content="%s"' % self.author
|
|
|
|
page, head, body = Html.page('%s - %s' %
|
|
(html_escape(self.title_str),
|
|
html_escape(title)),
|
|
self.report.encoding, xmllang
|
|
)
|
|
|
|
# create additional meta tags
|
|
meta = (Html('meta', attr = _META1) +
|
|
Html('meta', attr = _META2, indent=False)
|
|
)
|
|
|
|
# Link to media reference regions behaviour stylesheet
|
|
fname = '/'.join(["styles", "behaviour.css"])
|
|
url1= self.report.build_url_fname(fname, None, self.up)
|
|
|
|
# Link to _NARRATIVESCREEN stylesheet
|
|
fname = '/'.join(["styles", _NARRATIVESCREEN])
|
|
url3 = self.report.build_url_fname(fname, None, self.up)
|
|
|
|
# Link to _NARRATIVEPRINT stylesheet
|
|
fname = '/'.join(["styles", _NARRATIVEPRINT])
|
|
url4 = self.report.build_url_fname(fname, None, self.up)
|
|
|
|
# Link to GRAMPS favicon
|
|
fname = '/'.join(['images', 'favicon.ico'])
|
|
url5 = self.report.build_url_image('favicon.ico', 'images', self.up)
|
|
|
|
# create stylesheet and favicon links
|
|
links = [Html('link', href=url5, type='image/x-icon', rel='shortcut icon'),
|
|
Html('link', href=url1, type='text/css', media='screen', rel='stylesheet'),
|
|
Html('link', href=url3, type='text/css', media='screen', rel='stylesheet'),
|
|
Html('link', href=url4, type='text/css', media='print', rel='stylesheet')
|
|
]
|
|
|
|
# add additional meta and link tags
|
|
head += meta
|
|
head += links
|
|
|
|
# begin header section
|
|
headerdiv = (Html("div", id='header') +
|
|
Html('h1', html_escape(self.title_str), id='SiteTitle', inline = True)
|
|
)
|
|
body += headerdiv
|
|
|
|
header_note = self.report.options['headernote']
|
|
if header_note:
|
|
note = db.get_note_from_gramps_id(header_note)
|
|
note_text = self.get_note_format(note)
|
|
|
|
user_header = Html("div", id='user_header')
|
|
headerdiv += user_header
|
|
|
|
# attach note
|
|
user_header += note_text
|
|
|
|
# Begin Navigation Menu
|
|
body += self.display_nav_links(title)
|
|
|
|
# return to its caller, page and body
|
|
return page, body
|
|
|
|
def display_nav_links(self, currentsection):
|
|
"""
|
|
Creates the navigation menu
|
|
"""
|
|
|
|
db = self.report.database
|
|
|
|
# include repositories or not?
|
|
inc_repos = True
|
|
if not self.report.inc_repository or \
|
|
len(db.get_repository_handles()) == 0:
|
|
inc_repos = False
|
|
|
|
navs = [
|
|
(self.report.index_fname, _('Home'), self.report.use_home),
|
|
(self.report.intro_fname, _('Introduction'), self.report.use_intro),
|
|
(self.report.surname_fname, _('Surnames'), True),
|
|
('individuals', _('Individuals'), True),
|
|
('places', _('Places'), True),
|
|
('events', _('Events'), self.report.inc_events),
|
|
('media', _('Media'), self.create_media),
|
|
('download', _('Download'), self.report.inc_download),
|
|
('contact', _('Contact'), self.report.use_contact),
|
|
('sources', _('Sources'), True),
|
|
('repositories', _('Repositories'), inc_repos),
|
|
]
|
|
|
|
navigation = Html("div", id='navigation')
|
|
ul = Html('ul')
|
|
|
|
navs = ((u, n) for u, n, c in navs if c)
|
|
for url_fname, nav_text in navs:
|
|
|
|
if not _has_webpage_extension(url_fname):
|
|
url_fname += self.ext
|
|
|
|
url = self.report.build_url_fname(url_fname, None, self.up)
|
|
|
|
# Define 'currentsection' to correctly set navlink item CSS id
|
|
# 'CurrentSection' for Navigation styling.
|
|
# Use 'self.report.cur_fname' to determine 'CurrentSection' for individual
|
|
# elements for Navigation styling.
|
|
|
|
# Figure out if we need <li class="CurrentSection"> of just plain <li>
|
|
cs = False
|
|
if nav_text == currentsection:
|
|
cs = True
|
|
elif nav_text == _('Surnames'):
|
|
if "srn" in self.report.cur_fname:
|
|
cs = True
|
|
elif _('Surnames') in currentsection:
|
|
cs = True
|
|
elif nav_text == _('Individuals'):
|
|
if "ppl" in self.report.cur_fname:
|
|
cs = True
|
|
elif nav_text == _('Sources'):
|
|
if "src" in self.report.cur_fname:
|
|
cs = True
|
|
elif nav_text == _('Places'):
|
|
if "plc" in self.report.cur_fname:
|
|
cs = True
|
|
elif nav_text == _('Events'):
|
|
if 'evt' in self.report.cur_fname:
|
|
cs = True
|
|
elif nav_text == _('Media'):
|
|
if "img" in self.report.cur_fname:
|
|
cs = True
|
|
|
|
cs = cs and 'class="CurrentSection"' or ''
|
|
ul += (Html('li', attr=cs, inline = True) +
|
|
Html('a', nav_text, href=url)
|
|
)
|
|
|
|
navigation += ul
|
|
|
|
# return navigation menu bar to its caller
|
|
return navigation
|
|
|
|
def display_first_image_as_thumbnail( self, photolist=None):
|
|
db = self.report.database
|
|
|
|
if not photolist or not self.create_media:
|
|
return None
|
|
|
|
photo_handle = photolist[0].get_reference_handle()
|
|
photo = db.get_object_from_handle(photo_handle)
|
|
mime_type = photo.get_mime_type()
|
|
|
|
# begin snapshot division
|
|
with Html("div", class_ = "snapshot") as snapshot:
|
|
|
|
if mime_type:
|
|
try:
|
|
lnkref = (self.report.cur_fname, self.page_title, self.gid)
|
|
self.report.add_lnkref_to_photo(photo, lnkref)
|
|
real_path, newpath = self.report.prepare_copy_media(photo)
|
|
|
|
# TODO. Check if build_url_fname can be used.
|
|
newpath = '/'.join(['..']*3 + [newpath])
|
|
if ( Utils.win ):
|
|
newpath = newpath.replace('\\','/')
|
|
|
|
# begin hyperlink
|
|
# description is given only for the purpose of the alt tag in img element
|
|
snapshot += self.media_link(photo_handle, newpath, '', up=True)
|
|
|
|
except (IOError, OSError), msg:
|
|
WarningDialog(_("Could not add photo to page"), str(msg))
|
|
else:
|
|
|
|
# get media description
|
|
descr = photo.get_description()
|
|
|
|
# begin hyperlink
|
|
snapshot += self.doc_link(photo_handle, descr, up=True)
|
|
|
|
lnk = (self.report.cur_fname, self.page_title, self.gid)
|
|
# FIXME. Is it OK to add to the photo_list of report?
|
|
photo_list = self.report.photo_list
|
|
if photo_handle in photo_list:
|
|
if lnk not in photo_list[photo_handle]:
|
|
photo_list[photo_handle].append(lnk)
|
|
else:
|
|
photo_list[photo_handle] = [lnk]
|
|
|
|
# return snapshot division to its callers
|
|
return snapshot
|
|
|
|
def display_additional_images_as_gallery( self, photolist=None):
|
|
|
|
if not photolist or not self.create_media:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# begin individualgallery division
|
|
with Html("div", class_ = "subsection", id="indivgallery") as section:
|
|
|
|
# begin section title
|
|
section += Html('h4', _('Gallery'), inline = True)
|
|
|
|
displayed = []
|
|
for mediaref in photolist:
|
|
|
|
photo_handle = mediaref.get_reference_handle()
|
|
photo = db.get_object_from_handle(photo_handle)
|
|
if photo_handle in displayed:
|
|
continue
|
|
mime_type = photo.get_mime_type()
|
|
|
|
# get media description
|
|
descr = photo.get_description()
|
|
|
|
if mime_type:
|
|
try:
|
|
lnkref = (self.report.cur_fname, self.page_title, self.gid)
|
|
self.report.add_lnkref_to_photo(photo, lnkref)
|
|
real_path, newpath = self.report.prepare_copy_media(photo)
|
|
# TODO. Check if build_url_fname can be used.
|
|
newpath = '/'.join(['..']*3 + [newpath])
|
|
if ( Utils.win ):
|
|
newpath = newpath.replace('\\','/')
|
|
|
|
# begin hyperlink
|
|
section += self.media_link(photo_handle, newpath, descr, True, False)
|
|
|
|
except (IOError, OSError), msg:
|
|
WarningDialog(_("Could not add photo to page"), str(msg))
|
|
else:
|
|
try:
|
|
|
|
# begin hyperlink
|
|
section += self.doc_link(photo_handle, descr, up=True)
|
|
|
|
lnk = (self.report.cur_fname, self.page_title, self.gid)
|
|
# FIXME. Is it OK to add to the photo_list of report?
|
|
photo_list = self.report.photo_list
|
|
if photo_handle in photo_list:
|
|
if lnk not in photo_list[photo_handle]:
|
|
photo_list[photo_handle].append(lnk)
|
|
else:
|
|
photo_list[photo_handle] = [lnk]
|
|
except (IOError, OSError), msg:
|
|
WarningDialog(_("Could not add photo to page"), str(msg))
|
|
displayed.append(photo_handle)
|
|
|
|
# add clearline for proper styling
|
|
section += fullclear
|
|
|
|
# return indivgallery division to its caller
|
|
return section
|
|
|
|
def display_note_list(self, notelist=None):
|
|
|
|
if not notelist:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# begin narrative division
|
|
with Html("div", class_ = "subsection", id="narrative") as section:
|
|
|
|
for notehandle in notelist:
|
|
note = db.get_note_from_handle(notehandle)
|
|
|
|
if note:
|
|
note_text = self.get_note_format(note)
|
|
try:
|
|
note_text = unicode(note_text)
|
|
except UnicodeDecodeError:
|
|
note_text = unicode(str(note_text), errors='replace')
|
|
|
|
# add section title
|
|
section += Html('h4', _('Narrative'), inline = True)
|
|
|
|
# attach note
|
|
section += note_text
|
|
|
|
# return notes to its callers
|
|
return section
|
|
|
|
def display_url_list(self, urllist=None):
|
|
|
|
if not urllist:
|
|
return None
|
|
|
|
# begin web links division
|
|
with Html("div", class_ = "subsection", id="weblinks") as section:
|
|
|
|
# begin web title
|
|
title = Html('h4', _('Weblinks'), inline = True)
|
|
section += title
|
|
|
|
# ordered list
|
|
ordered = Html('ol')
|
|
section += ordered
|
|
|
|
for url in urllist:
|
|
uri = url.get_path()
|
|
descr = url.get_description()
|
|
if not descr:
|
|
descr = uri
|
|
if url.get_type() == UrlType.EMAIL and not uri.startswith("mailto:"):
|
|
ordered += Html('li') + Html('a',descr, href='mailto:%s' % url)
|
|
|
|
elif url.get_type() == UrlType.WEB_HOME and not uri.startswith("http://"):
|
|
ordered += Html('li') + Html('a', descr, href='http://%s' % url)
|
|
|
|
elif url.get_type() == UrlType.WEB_FTP and not uri.startswith("ftp://"):
|
|
ordered += Html('li') + Html('a', descr, href='ftp://%s' % url)
|
|
else:
|
|
ordered += Html('li') + Html('a', descr, href=url)
|
|
|
|
# return web links to its caller
|
|
return section
|
|
|
|
def display_source_refs(self, bibli):
|
|
if bibli.get_citation_count() == 0:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# source references division and title
|
|
with Html("div", class_ = "subsection", id="sourcerefs") as section:
|
|
section += Html('h4', _('Source References'), inline = True)
|
|
|
|
ordered=Html('ol')
|
|
section += ordered
|
|
|
|
cindex = 0
|
|
for citation in bibli.get_citation_list():
|
|
cindex += 1
|
|
# Add this source to the global list of sources to be displayed
|
|
# on each source page.
|
|
lnk = (self.report.cur_fname, self.page_title, self.gid)
|
|
shandle = citation.get_source_handle()
|
|
if shandle in self.src_list:
|
|
if lnk not in self.src_list[shandle]:
|
|
self.src_list[shandle].append(lnk)
|
|
else:
|
|
self.src_list[shandle] = [lnk]
|
|
|
|
# Add this source and its references to the page
|
|
source = db.get_source_from_handle(shandle)
|
|
title = source.get_title()
|
|
list = Html('li')
|
|
ordered += list
|
|
|
|
hyper = Html('a', name='sref%d' % cindex) + \
|
|
self.source_link(source.handle, title, source.gramps_id, True)
|
|
list += hyper
|
|
|
|
ordered1 = Html('ol')
|
|
list += ordered1
|
|
|
|
for key, sref in citation.get_ref_list():
|
|
|
|
tmp = []
|
|
confidence = Utils.confidence.get(sref.confidence, _('Unknown'))
|
|
if confidence == _('Normal'):
|
|
confidence = None
|
|
|
|
source_data = [
|
|
[DHEAD, _dd.display(sref.date)],
|
|
[_('Page'), sref.page],
|
|
[_('Confidence'), confidence]
|
|
]
|
|
|
|
for (label, data) in source_data:
|
|
if data:
|
|
tmp.append("%s: %s" % (label, data))
|
|
|
|
# get citation note list
|
|
notelist = sref.get_note_list()
|
|
for notehandle in notelist:
|
|
note = db.get_note_from_handle(notehandle)
|
|
|
|
# check if note is styled text or not?
|
|
note_text = self.get_note_format(note)
|
|
tmp.append("%s: %s" % (_('Text'), note_text))
|
|
|
|
if len(tmp):
|
|
list1 = Html('li') + (
|
|
Html('a', '; '.join(tmp), name='sref%d%s' % (cindex, key))
|
|
)
|
|
ordered1 += list1
|
|
|
|
# return section to its callers
|
|
return section
|
|
|
|
def display_references(self, handlelist, up=False):
|
|
|
|
if not handlelist:
|
|
return None
|
|
|
|
# begin references division and title
|
|
with Html("div", class_ = "subsection", id="references") as section:
|
|
section += Html('h4', _('References'), inline = True)
|
|
|
|
ordered = Html('ol')
|
|
section += ordered
|
|
sortlist = sorted(handlelist, key=lambda x:locale.strxfrm(x[1]))
|
|
|
|
for (path, name, gid) in sortlist:
|
|
list = Html('li')
|
|
ordered += list
|
|
|
|
# Note. 'path' already has a filename extension
|
|
url = self.report.build_url_fname(path, None, self.up)
|
|
list += self.person_link(url, name, None, gid)
|
|
|
|
# return references division to its caller
|
|
return section
|
|
|
|
def person_link(self, url, person, name_style, first=True, gid=None,
|
|
thumbnailUrl=None):
|
|
"""
|
|
creates a hyperlink for a person
|
|
|
|
@param: person = person in database
|
|
@param: namestyle = False -- first and suffix only
|
|
= True -- name displayed in name_format variable
|
|
= None -- person is name
|
|
@param: first = show person's name and gramps id if requested and available
|
|
"""
|
|
|
|
# the only place that this will ever equal False
|
|
# is first there is more than one event for a person for class EventListPage
|
|
if first:
|
|
|
|
# see above for explanation
|
|
if name_style:
|
|
person_name = self.get_name(person)
|
|
elif name_style == False:
|
|
person_name = _get_short_name(person.gender, person.primary_name)
|
|
elif name_style == None: # abnormal specialty situation
|
|
person_name = person
|
|
|
|
# 1. start building link to image or person
|
|
hyper = Html('a', href=url)
|
|
|
|
# 2. insert thumbnail if there is one, otherwise insert class = "noThumb"
|
|
if thumbnailUrl:
|
|
hyper += (Html('span', class_ = "thumbnail") +
|
|
Html('img', src= thumbnailUrl, alt = "Image of " + person_name)
|
|
)
|
|
else:
|
|
hyper.attr += ' class= "noThumb"'
|
|
|
|
# 3. insert the person's name
|
|
hyper += person_name
|
|
|
|
# 3. insert gramps id if requested and available
|
|
if not self.noid and gid:
|
|
hyper += Html('span', " [%s]" % gid, class_ = "grampsid", inline = True)
|
|
|
|
else:
|
|
hyper = " "
|
|
|
|
# return hyperlink to its caller
|
|
return hyper
|
|
|
|
def individual_link(self, url, person):
|
|
"""
|
|
creates a hyperlink for a partner in IndividualListPage, and SurnameListPage
|
|
with no image class attached to it.
|
|
|
|
@param: url = hyperlink to person
|
|
@param: person = person to be hyperlinked
|
|
"""
|
|
|
|
person_name = self.get_name(person)
|
|
|
|
# 1. start building link to image or person
|
|
hyper = Html('a', person_name, href=url, title=html_escape(person_name))
|
|
|
|
# return hyperlink to its caller
|
|
return hyper
|
|
|
|
# TODO. Check img_url of callers
|
|
def media_link(self, handle, img_url, name, up, usedescr=True):
|
|
url = self.report.build_url_fname_html(handle, 'img', up)
|
|
|
|
# begin thumbnail division
|
|
with Html("div", class_ = "thumbnail") as thumbnail:
|
|
|
|
# begin hyperlink
|
|
hyper = (Html('a', href=url, title=name) +
|
|
Html('img', src=img_url, alt=name) +
|
|
(Html('p', inline = True) +
|
|
html_escape(name) if usedescr else '')
|
|
)
|
|
# add hyperlink and description to thumbnail division
|
|
thumbnail += hyper
|
|
|
|
# return thumbnail division to its callers
|
|
return thumbnail
|
|
|
|
def doc_link(self, handle, name, up, usedescr=True):
|
|
# TODO. Check extension of handle
|
|
url = self.report.build_url_fname(handle, 'img', up)
|
|
|
|
# begin thumbnail division
|
|
thumbnail = Html("div", class_ = "thumbnail")
|
|
|
|
# begin hyperlink
|
|
hyper = Html('a', href=url, title=name)
|
|
url = self.report.build_url_image('document.png', 'images', up)
|
|
hyper += Html('img', src=url, alt=html_escape(name))
|
|
if usedescr:
|
|
descr = Html('p', html_escape(name), inline = True)
|
|
else:
|
|
descr = ''
|
|
|
|
# add hyperlink and description to thumbnail division
|
|
thumbnail += (hyper, descr)
|
|
|
|
# return thumbnail division to its callers
|
|
return thumbnail
|
|
|
|
def repository_link(self, handle, name, cindex, gid=None, up=False):
|
|
|
|
url = self.report.build_url_fname_html(handle, 'repo', up)
|
|
# begin hyperlink
|
|
hyper = Html('a', html_escape(name), href=url, title=name)
|
|
if not self.noid and gid:
|
|
hyper += Html('span', '[%s]' % gid, class_ = "grampsid", inline = True)
|
|
|
|
# return hyperlink to its callers
|
|
return hyper
|
|
|
|
def place_link(self, handle, name, gid=None, up=False):
|
|
url = self.report.build_url_fname_html(handle, 'plc', up)
|
|
|
|
hyper = Html('a', html_escape(name), href=url, title=name)
|
|
if not self.noid and gid:
|
|
hyper += Html('span', ' [%s] ' % gid, class_ = "grampsid", inline = True)
|
|
|
|
# return hyperlink to its callers
|
|
return hyper
|
|
|
|
# ---------------------------------------------------------------------------------------
|
|
#
|
|
# # Web Page Fortmatter and writer
|
|
#
|
|
# ---------------------------------------------------------------------------------------
|
|
|
|
def mywriter(self, htmlinstance, of):
|
|
"""
|
|
Will format, write, and close the file
|
|
|
|
of -- open file that is being written to
|
|
htmlinstance -- web page created with libhtml
|
|
src/plugins/lib/libhtml.py
|
|
"""
|
|
|
|
htmlinstance.write(lambda line: of.write(line + '\n'))
|
|
|
|
# closes the file
|
|
self.report.close_file(of)
|
|
|
|
class IndividualListPage(BasePage):
|
|
|
|
def __init__(self, report, title, person_handle_list):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
# handles for this module for use in partner column
|
|
report_handle_list = person_handle_list
|
|
|
|
# plugin variables for this module
|
|
showbirth = report.options['showbirth']
|
|
showdeath = report.options['showdeath']
|
|
showpartner = report.options['showpartner']
|
|
showparents = report.options['showparents']
|
|
|
|
of = self.report.create_file("individuals")
|
|
indlistpage, body = self.write_header(_('Individuals'))
|
|
|
|
# begin Individuals division
|
|
with Html("div", class_ = "content", id="Individuals") as section:
|
|
body += section
|
|
|
|
# Individual List description
|
|
msg = _("This page contains an index of all the individuals in the "
|
|
"database, sorted by their last names. Selecting the person’s "
|
|
"name will take you to that person’s individual page.")
|
|
section += Html('p', msg, id='description')
|
|
|
|
# add alphabet navigation after page msg
|
|
alpha_nav = alphabet_navigation(db, person_handle_list, _PERSON)
|
|
if alpha_nav is not None:
|
|
section += alpha_nav
|
|
|
|
# begin table and table head
|
|
with Html("table", class_ = "infolist IndividualList") as table:
|
|
section += table
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
# Table Header -- Surname and Given name columns
|
|
tcell1 = Html("th", _('Surname'), class_ = "ColumnSurname", inline = True)
|
|
tcell2 = Html("th", _('Name'), class_ = "ColumnName", inline = True)
|
|
trow += (tcell1, tcell2)
|
|
|
|
# table header -- show birth column
|
|
if showbirth:
|
|
trow += Html("th", _('Birth'), class_ = "ColumnBirth", inline = True)
|
|
|
|
# table header -- show death column
|
|
if showdeath:
|
|
trow += Html("th", _('Death'), class_ = "ColumnDeath", inline = True)
|
|
|
|
# table header -- show partmer column
|
|
if showpartner:
|
|
trow += Html("th", _('Partner'), class_ = "ColumnPartner", inline = True)
|
|
|
|
# table header -- show parents column
|
|
if showparents:
|
|
trow += Html("th", _('Parents'), class_ = "ColumnParents", inline = True)
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
# list of person handles for this report
|
|
person_handle_list = sort_people(db, person_handle_list)
|
|
|
|
for (surname, handle_list) in person_handle_list:
|
|
first = True
|
|
if surname:
|
|
letter = normalize('NFKC', surname)[0].upper()
|
|
else:
|
|
letter = u' '
|
|
# See : http://www.gramps-project.org/bugs/view.php?id=2933
|
|
(lang_country, modifier ) = locale.getlocale()
|
|
if lang_country == "sv_SE" and ( letter == u'W' or letter == u'V' ):
|
|
letter = u'V,W'
|
|
for person_handle in handle_list:
|
|
person = db.get_person_from_handle(person_handle)
|
|
|
|
# surname column
|
|
trow = Html("tr")
|
|
tcell = Html("td", class_ = "ColumnSurname", inline = True)
|
|
if first:
|
|
trow.attr = 'class="BeginSurname"'
|
|
if surname:
|
|
tcell += Html('a', surname, name=letter, title='Letter %s' % letter,
|
|
inline = True)
|
|
else:
|
|
tcell += " "
|
|
else:
|
|
tcell += " "
|
|
tbody += trow
|
|
trow += tcell
|
|
|
|
first = False
|
|
# firstname column
|
|
tcell = Html("td", class_ = "ColumnName")
|
|
trow += tcell
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl')
|
|
tcell += self.person_link(url, person, False, person.gramps_id)
|
|
|
|
# birth column
|
|
if showbirth:
|
|
tcell = Html("td", class_ = "ColumnBirth", inline = True)
|
|
birth = ReportUtils.get_birth_or_fallback(db, person)
|
|
if birth:
|
|
birth_date = _dd.display(birth.get_date_object())
|
|
if birth.get_type() == EventType.BIRTH:
|
|
tcell += birth_date
|
|
else:
|
|
tcell += Html('em', birth_date)
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# death column
|
|
if showdeath:
|
|
tcell = Html("td", class_ = "ColumnDeath", inline = True)
|
|
death = ReportUtils.get_death_or_fallback(db, person)
|
|
if death:
|
|
death_date = _dd.display(death.get_date_object())
|
|
if death.get_type() == EventType.DEATH:
|
|
tcell += death_date
|
|
else:
|
|
tcell += Html('em', death_date)
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# partner column
|
|
if showpartner:
|
|
tcell = Html("td", class_ = "ColumnPartner")
|
|
family_list = person.get_family_handle_list()
|
|
first_family = True
|
|
partner_name = None
|
|
if family_list:
|
|
for family_handle in family_list:
|
|
family = db.get_family_from_handle(family_handle)
|
|
partner_handle = ReportUtils.find_spouse(person, family)
|
|
if partner_handle:
|
|
partner = db.get_person_from_handle(partner_handle)
|
|
partner_name = self.get_name(partner)
|
|
if not first_family:
|
|
tcell += ', '
|
|
if partner_handle in report_handle_list:
|
|
url = self.report.build_url_fname_html(partner_handle, 'ppl')
|
|
tcell += self.individual_link(url, partner)
|
|
else:
|
|
tcell += partner_name
|
|
first_family = False
|
|
else:
|
|
tcell += " "
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# parents column
|
|
if showparents:
|
|
tcell = Html("td", class_ = "ColumnParents")
|
|
parent_handle_list = person.get_parent_family_handle_list()
|
|
if parent_handle_list:
|
|
parent_handle = parent_handle_list[0]
|
|
family = db.get_family_from_handle(parent_handle)
|
|
father_handle = family.get_father_handle()
|
|
mother_handle = family.get_mother_handle()
|
|
father = db.get_person_from_handle(father_handle)
|
|
mother = db.get_person_from_handle(mother_handle)
|
|
if father:
|
|
father_name = self.get_name(father)
|
|
if mother:
|
|
mother_name = self.get_name(mother)
|
|
if mother and father:
|
|
fathercell = Html('span', father_name,
|
|
class_ = "father fatherNmother")
|
|
mothercell = Html('span', mother_name, class_ = "mother")
|
|
tcell += (fathercell, mothercell)
|
|
elif mother:
|
|
tcell += Html('span', mother_name, class_ = "mother")
|
|
elif father:
|
|
tcell += Html('span', father_name, class_ = "father")
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# create clear line for proper styling
|
|
# create footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(indlistpage, of)
|
|
|
|
class SurnamePage(BasePage):
|
|
|
|
def __init__(self, report, title, surname, person_handle_list, report_handle_list):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
# module variables
|
|
showbirth = report.options['showbirth']
|
|
showdeath = report.options['showdeath']
|
|
showpartner = report.options['showpartner']
|
|
showparents = report.options['showparents']
|
|
|
|
of = self.report.create_file(name_to_md5(surname), 'srn')
|
|
self.up = True
|
|
surnamepage, body = self.write_header("%s - %s" % (_('Surname'), surname))
|
|
|
|
# begin SurnameDetail division
|
|
with Html("div", class_ = "content", id="SurnameDetail") as surnamedetail:
|
|
body += surnamedetail
|
|
|
|
# section title
|
|
surnamedetail += Html('h3', html_escape(surname), inline = True)
|
|
|
|
msg = _("This page contains an index of all the individuals in the "
|
|
"database with the surname of %s. Selecting the person’s name "
|
|
"will take you to that person’s individual page.") % surname
|
|
surnamedetail += Html('p', msg, id='description')
|
|
|
|
# begin surname table and thead
|
|
with Html("table", class_ = "infolist surname") as surname_table:
|
|
surnamedetail += surname_table
|
|
with Html("thead") as thead:
|
|
surname_table += thead
|
|
tabhead = []
|
|
tabhead.append('Name')
|
|
if report.options['showbirth']:
|
|
tabhead.append('Birth')
|
|
if report.options['showdeath']:
|
|
tabhead.append('Death')
|
|
if report.options['showpartner']:
|
|
tabhead.append('Partner')
|
|
if report.options['showparents']:
|
|
tabhead.append('Parents')
|
|
with Html("tr") as trow:
|
|
thead += trow
|
|
|
|
# now spit out whatever is in table head
|
|
for column in tabhead:
|
|
trow += Html("th", _(column), class_ = "Column%s" % column,
|
|
inline = True)
|
|
|
|
# begin table body
|
|
with Html("tbody") as tbody:
|
|
surname_table += tbody
|
|
|
|
for person_handle in person_handle_list:
|
|
|
|
# firstname column
|
|
person = db.get_person_from_handle(person_handle)
|
|
trow = Html("tr")
|
|
tcell = Html("td", class_ = "ColumnName")
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl', True)
|
|
tcell += self.person_link(url, person, False, person.gramps_id)
|
|
trow += tcell
|
|
|
|
# birth column
|
|
if showbirth:
|
|
tcell = Html("td", class_ = "ColumnBirth", inline = True)
|
|
birth = ReportUtils.get_birth_or_fallback(db, person)
|
|
if birth:
|
|
birth_date = _dd.display(birth.get_date_object())
|
|
if birth.get_type() == EventType.BIRTH:
|
|
tcell += birth_date
|
|
else:
|
|
tcell += Html('em', birth_date)
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# death column
|
|
if showdeath:
|
|
tcell = Html("td", class_ = "ColumnDeath", inline = True)
|
|
death = ReportUtils.get_death_or_fallback(db, person)
|
|
if death:
|
|
death_date = _dd.display(death.get_date_object())
|
|
if death.get_type() == EventType.DEATH:
|
|
tcell += death_date
|
|
else:
|
|
tcell += Html('em', death_date)
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# partner column
|
|
if showpartner:
|
|
tcell = Html("td", class_ = "ColumnPartner")
|
|
family_list = person.get_family_handle_list()
|
|
first_family = True
|
|
if family_list:
|
|
for family_handle in family_list:
|
|
family = db.get_family_from_handle(family_handle)
|
|
partner_handle = ReportUtils.find_spouse(person, family)
|
|
if partner_handle:
|
|
partner = db.get_person_from_handle(partner_handle)
|
|
partner_name = self.get_name(partner)
|
|
if not first_family:
|
|
tcell += ','
|
|
if partner_handle in report_handle_list:
|
|
url = self.report.build_url_fname_html(partner_handle, 'ppl', True)
|
|
tcell += self.individual_link(url, partner)
|
|
else:
|
|
tcell += partner_name
|
|
else:
|
|
tcell += " "
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# parents column
|
|
if report.options['showparents']:
|
|
tcell = Html("td", class_ = "ColumnParents")
|
|
parent_handle_list = person.get_parent_family_handle_list()
|
|
if parent_handle_list:
|
|
parent_handle = parent_handle_list[0]
|
|
family = db.get_family_from_handle(parent_handle)
|
|
father_id = family.get_father_handle()
|
|
mother_id = family.get_mother_handle()
|
|
father = db.get_person_from_handle(father_id)
|
|
mother = db.get_person_from_handle(mother_id)
|
|
if father:
|
|
father_name = self.get_name(father)
|
|
if mother:
|
|
mother_name = self.get_name(mother)
|
|
if mother and father:
|
|
tcell += Html('span', father_name,
|
|
class_ = "father fatherNmother") + (
|
|
Html('span', mother_name, class_ = "mother")
|
|
)
|
|
elif mother:
|
|
tcell += Html('span', mother_name, class_ = "mother")
|
|
elif father:
|
|
tcell += Html('span', father_name, class_ = "father")
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
tbody += trow
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(surnamepage, of)
|
|
|
|
class PlaceListPage(BasePage):
|
|
|
|
def __init__(self, report, title, place_handles, src_list):
|
|
BasePage.__init__(self, report, title)
|
|
self.src_list = src_list # TODO verify that this is correct
|
|
db = report.database
|
|
|
|
of = self.report.create_file("places")
|
|
placelistpage, body = self.write_header(_('Places'))
|
|
|
|
# begin places division
|
|
with Html("div", class_ = "content", id="Places") as section:
|
|
body += section
|
|
|
|
msg = _("This page contains an index of all the places in the "
|
|
"database, sorted by their title. Clicking on a place’s "
|
|
"title will take you to that place’s page.")
|
|
section += Html('p', msg, id='description')
|
|
|
|
# begin alphabet navigation
|
|
alpha_nav = alphabet_navigation(db, place_handles, _PLACE)
|
|
if alpha_nav is not None:
|
|
section += alpha_nav
|
|
|
|
# begin places table and table head
|
|
with Html("table", class_ = "infolist placelist") as table:
|
|
section += table
|
|
|
|
# begin table head
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr") + (
|
|
Html("th", _('Letter'), class_ = "ColumnLetter", inline = True),
|
|
Html("th", _('Name'), class_ = "ColumnName", inline = True)
|
|
)
|
|
thead += trow
|
|
|
|
sort = Sort.Sort(db)
|
|
handle_list = sorted(place_handles, key=sort.by_place_title_key)
|
|
last_letter = ''
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for handle in handle_list:
|
|
place = db.get_place_from_handle(handle)
|
|
place_title = ReportUtils.place_name(db, handle)
|
|
|
|
if not place_title:
|
|
continue
|
|
|
|
letter = normalize('NFKC', place_title)[0].upper()
|
|
# See : http://www.gramps-project.org/bugs/view.php?id=2933
|
|
(lang_country, modifier ) = locale.getlocale()
|
|
if lang_country == "sv_SE" and ( letter == u'W' or letter == u'V' ):
|
|
letter = u'V,W'
|
|
|
|
if letter != last_letter:
|
|
last_letter = letter
|
|
trow = Html("tr", class_ = "BeginLetter")
|
|
tbody += trow
|
|
tcell = Html("td", class_ = "olumnLetter", inline = True) + (
|
|
Html('a', last_letter, name=last_letter,
|
|
title="Letter %s" % last_letter)
|
|
)
|
|
else:
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
tcell = Html("td", " ", class_ = "ColumnLetter", inline = True)
|
|
trow += tcell
|
|
|
|
tcell = Html("td", class_ = "ColumnName") + \
|
|
self.place_link(place.handle, place_title, place.gramps_id)
|
|
trow += tcell
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(placelistpage, of)
|
|
|
|
class PlacePage(BasePage):
|
|
|
|
def __init__(self, report, title, place_handle, src_list, place_list):
|
|
db = report.database
|
|
|
|
place = db.get_place_from_handle(place_handle)
|
|
BasePage.__init__(self, report, title, place.gramps_id)
|
|
self.src_list = src_list # TODO verify that this is correct
|
|
|
|
of = self.report.create_file(place.get_handle(), 'plc')
|
|
self.up = True
|
|
self.page_title = ReportUtils.place_name(db, place_handle)
|
|
placepage, body = self.write_header(_('Places'))
|
|
|
|
# begin PlaceDetail Division
|
|
with Html("div", class_ = "content", id="PlaceDetail") as placedetail:
|
|
body += placedetail
|
|
|
|
media_list = place.get_media_list()
|
|
thumbnail = self.display_first_image_as_thumbnail(media_list)
|
|
if thumbnail is not None:
|
|
placedetail += thumbnail
|
|
|
|
# add section title
|
|
placedetail += Html('h3', html_escape(self.page_title.strip()))
|
|
|
|
# begin summaryarea division and places table
|
|
with Html("div", id='summaryarea') as summaryarea:
|
|
placedetail += summaryarea
|
|
|
|
with Html("table", class_ = "infolist place") as table:
|
|
summaryarea += table
|
|
|
|
if not self.noid:
|
|
trow = Html("tr") + (
|
|
Html("td", _('GRAMPS ID'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", place.gramps_id, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
if place.main_loc:
|
|
ml = place.main_loc
|
|
for val in [
|
|
(LATITUDE, place.lat),
|
|
(LONGITUDE, place.long),
|
|
(STREET, ml.street),
|
|
(CITY, ml.city),
|
|
(PARISH, ml.parish),
|
|
(COUNTY, ml.county),
|
|
(STATE, ml.state),
|
|
(POSTAL, ml.postal),
|
|
(COUNTRY, ml.country),
|
|
(LOCATIONS, place.get_alternate_locations()) ]:
|
|
|
|
if val[1]:
|
|
trow = Html("tr") + (
|
|
Html("td", val[0], class_ = "ColumnAttribute", inline = True),
|
|
Html("td", val[1], class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# place gallery
|
|
if self.create_media:
|
|
placegallery = self.display_additional_images_as_gallery(media_list)
|
|
if placegallery is not None:
|
|
placedetail += placegallery
|
|
|
|
# place notes
|
|
notelist = self.display_note_list(place.get_note_list())
|
|
if notelist is not None:
|
|
placedetail += notelist
|
|
|
|
# place urls
|
|
urllinks = self.display_url_list(place.get_url_list())
|
|
if urllinks is not None:
|
|
placedetail += urllinks
|
|
|
|
# source references
|
|
# sourcerefs = self.get_citation_links(place.get_source_references() )
|
|
# if sourcerefs is not None:
|
|
# placedetail += sourcerefs
|
|
|
|
# place references
|
|
referenceslist = self.display_references(place_list[place.handle])
|
|
if referenceslist is not None:
|
|
placedetail += referenceslist
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(placepage, of)
|
|
|
|
class EventListPage(BasePage):
|
|
|
|
def __init__(self, report, title, event_dict):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file("events")
|
|
eventslistpage, body = self.write_header(_('Events'))
|
|
|
|
# begin events list division
|
|
with Html("div", class_ = "content", id="EventList") as eventlist:
|
|
body += eventlist
|
|
|
|
msg = _("This page contains an index of all the events in the "
|
|
"database, sorted by their type, date (if one is present), "
|
|
"and person’s surname. Clicking on an event’s type "
|
|
"will take you to that event’s page. Clicking on a "
|
|
"person’s name will take you to that person’s page. "
|
|
"The person’s name will only be shown once for their events.")
|
|
eventlist += Html("p", msg, id="description")
|
|
|
|
# begin event list table
|
|
with Html("table", class_ = "infolist eventlist") as table:
|
|
eventlist += table
|
|
|
|
# begin table head
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
# begin table header row
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
for (label, colclass) in [
|
|
[THEAD, 'Type'],
|
|
[DHEAD, 'Date'],
|
|
[_('Person'), 'Person'],
|
|
[_('Partner'), 'Partner'] ]:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass, inline = True)
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for (person, event_list) in event_dict:
|
|
|
|
first = True
|
|
for (evt_type, sort_date, sort_name, evt, evt_ref,
|
|
partner) in event_list:
|
|
|
|
# write eent row data
|
|
trow = self.write_event_row(person, partner, evt_type, evt,
|
|
evt_ref, first)
|
|
tbody += trow
|
|
first = False
|
|
|
|
# and clearline for proper styling
|
|
# and footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page ut for processing
|
|
# and close the file
|
|
self.mywriter(eventslistpage, of)
|
|
|
|
def write_event_row(self, person, partner, evt_type, evt, evt_ref, first):
|
|
"""
|
|
display the event row for class EventListPage()
|
|
|
|
@param: person = person that the event is referenced to
|
|
@param: partner = only used when the event is either a Marriage or Divorce
|
|
@param: evt_type = the type of event
|
|
@param: evt = event
|
|
@param: evt_ref = event reference
|
|
@param: first = used for only showing the person once for list of events
|
|
"""
|
|
subdirs = False
|
|
|
|
# get person's hyperlink
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl', subdirs)
|
|
person_hyper = self.person_link(url, person, True, first)
|
|
|
|
# begin table row
|
|
trow = Html("tr")
|
|
|
|
# get event data
|
|
"""
|
|
for more information: see get_event_data()
|
|
"""
|
|
event_data = self.get_event_data(evt, evt_ref, False, False, False,
|
|
False, subdirs, True)
|
|
|
|
for (label, colclass, data) in event_data:
|
|
data = data or " "
|
|
|
|
# determine if same row or not?
|
|
samerow = True if (data == " " or colclass == "Date") else False
|
|
|
|
trow += Html("td", data, class_ = "Column%s" % colclass, inline = samerow)
|
|
|
|
# display person hyperlink
|
|
trow += Html("td", person_hyper, class_ = "ColumnPerson")
|
|
|
|
# get partner hyperlink
|
|
# display partner if event is either a Marriage or Divorce?
|
|
partner_hyper = " "
|
|
if partner is not None:
|
|
|
|
# get partner hyperlink
|
|
url = self.report.build_url_fname_html(partner.handle, 'ppl', subdirs)
|
|
partner_hyper = self.person_link(url, partner, True)
|
|
trow += Html("td", partner_hyper, class_ = "ColumnPartner")
|
|
|
|
# return EventList row to its caller
|
|
return trow
|
|
|
|
class EventPage(BasePage):
|
|
|
|
def __init__(self, report, title, person, partner, evt_type, event, evt_ref):
|
|
BasePage.__init__(self, report, title, event.gramps_id)
|
|
self.up = True
|
|
db = report.database
|
|
subdirs = True
|
|
|
|
of = self.report.create_file(evt_ref.ref, 'evt')
|
|
eventpage, body = self.write_header(_('Events'))
|
|
|
|
# start event page division
|
|
with Html("div", class_ = "content", id="EventDetail") as eventdetail:
|
|
body += eventdetail
|
|
|
|
# display page itle
|
|
title = _('%(type)s of %(name)s') % {'type' : evt_type.lower(),
|
|
'name' : self.get_name(person) }
|
|
|
|
# line is in place for Peter Lundgren
|
|
title = title[0].upper() + title[1:]
|
|
eventdetail += Html('h3', title, inline = True)
|
|
|
|
# begin eventdetail table
|
|
with Html("table", class_ = "infolist eventlist") as table:
|
|
eventdetail += table
|
|
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
# get event data
|
|
shownote = True
|
|
"""
|
|
for more information: see get_event_data()
|
|
"""
|
|
event_data = self.get_event_data(event, evt_ref, True, True,
|
|
False, shownote, subdirs, False, event.gramps_id)
|
|
|
|
# the first ones are listed here, the rest are shown below
|
|
# minus one because it starts at zero instead of one
|
|
# minus one for note list which is shown below
|
|
for index in xrange(len(event_data) - 2):
|
|
label = event_data[index][0]
|
|
colclass = event_data[index][1]
|
|
data = event_data[index][2] or " "
|
|
|
|
# determine if we are using the same row or not?
|
|
samerow = True if (data == " " or colclass == "Date") else False
|
|
|
|
trow = Html("tr") + (
|
|
Html("td", label, class_ = "ColumnAttribute", inline = True),
|
|
Html('td', data, class_ = "Column%s" % colclass, inline = samerow)
|
|
)
|
|
tbody += trow
|
|
|
|
# get person hyperlink
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl', self.up)
|
|
person_hyper = self.person_link(url, person, True, gid=person.gramps_id)
|
|
trow = Html("tr") + (
|
|
Html("td", _('Person'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", person_hyper, class_ = "ColumnPerson")
|
|
)
|
|
tbody += trow
|
|
|
|
# display partner if type is either Marriage or Divorce
|
|
if partner is not None:
|
|
url = self.report.build_url_fname_html(partner.handle, 'ppl', self.up)
|
|
partner_hyper = self.person_link(url, partner, True, gid=partner.gramps_id)
|
|
trow = Html("tr") + (
|
|
Html("td", _('Partner'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", partner_hyper, class_ = "ColumnPartner")
|
|
)
|
|
tbody += trow
|
|
|
|
# Narrative subsection
|
|
if shownote:
|
|
notelist = event_data[(len(event_data) - 1)][2] or None
|
|
if notelist is not None:
|
|
eventdetail += self.display_note_list(notelist)
|
|
|
|
# get attribute list
|
|
attrlist = event.get_attribute_list()
|
|
attrlist.extend(evt_ref.get_attribute_list() )
|
|
if attrlist:
|
|
eventdetail += self.display_attr_list(attrlist, False)
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the page
|
|
self.mywriter(eventpage, of)
|
|
|
|
class MediaPage(BasePage):
|
|
|
|
def __init__(self, report, title, handle, src_list, my_media_list, info):
|
|
(prev, next, page_number, total_pages) = info
|
|
db = report.database
|
|
|
|
photo = db.get_object_from_handle(handle)
|
|
# TODO. How do we pass my_media_list down for use in BasePage?
|
|
BasePage.__init__(self, report, title, photo.gramps_id)
|
|
|
|
"""
|
|
*************************************
|
|
GRAMPS feature #2634 -- attempt to highlight subregions in media
|
|
objects and link back to the relevant web page.
|
|
|
|
This next section of code builds up the "records" we'll need to
|
|
generate the html/css code to support the subregions
|
|
*************************************
|
|
"""
|
|
|
|
# get all of the backlinks to this media object; meaning all of
|
|
# the people, events, places, etc..., that use this image
|
|
_region_items = set()
|
|
for (classname, newhandle) in db.find_backlink_handles(handle):
|
|
|
|
# for each of the backlinks, get the relevant object from the db
|
|
# and determine a few important things, such as a text name we
|
|
# can use, and the URL to a relevant web page
|
|
_obj = None
|
|
_name = ""
|
|
_linkurl = "#"
|
|
if classname == "Person":
|
|
_obj = db.get_person_from_handle( newhandle )
|
|
# what is the shortest possible name we could use for this person?
|
|
_name = _obj.get_primary_name().get_call_name()
|
|
if not _name or _name == "":
|
|
_name = _obj.get_primary_name().get_first_name()
|
|
_linkurl = report.build_url_fname_html(_obj.handle, 'ppl', True)
|
|
elif classname == "Event":
|
|
_obj = db.get_event_from_handle( newhandle )
|
|
_name = _obj.get_description()
|
|
|
|
# continue looking through the loop for an object...
|
|
if _obj is None:
|
|
continue
|
|
|
|
# get a list of all media refs for this object
|
|
medialist = _obj.get_media_list()
|
|
|
|
# go media refs looking for one that points to this image
|
|
for mediaref in medialist:
|
|
|
|
# is this mediaref for this image? do we have a rect?
|
|
if mediaref.ref == handle and mediaref.rect is not None:
|
|
|
|
(x1, y1, x2, y2) = mediaref.rect
|
|
# GRAMPS gives us absolute coordinates,
|
|
# but we need relative width + height
|
|
w = x2 - x1
|
|
h = y2 - y1
|
|
|
|
# remember all this information, cause we'll need
|
|
# need it later when we output the <li>...</li> tags
|
|
item = (_name, x1, y1, w, h, _linkurl)
|
|
_region_items.add(item)
|
|
"""
|
|
*************************************
|
|
end of code that looks for and prepares the media object regions
|
|
*************************************
|
|
"""
|
|
|
|
of = self.report.create_file(handle, 'img')
|
|
self.up = True
|
|
|
|
self.src_list = src_list
|
|
self.bibli = Bibliography()
|
|
|
|
# get media type to be used primarily with 'img' tags
|
|
mime_type = photo.get_mime_type()
|
|
mtype = Mime.get_description(mime_type)
|
|
|
|
if mime_type:
|
|
note_only = False
|
|
newpath = self.copy_source_file(handle, photo)
|
|
target_exists = newpath is not None
|
|
else:
|
|
note_only = True
|
|
target_exists = False
|
|
|
|
self.copy_thumbnail(handle, photo)
|
|
self.page_title = photo.get_description()
|
|
mediapage, body = self.write_header("%s - %s" % (_('Media'), self.page_title))
|
|
|
|
# begin GalleryDetail division
|
|
mediadetail = Html("div", class_ = "content", id="GalleryDetail")
|
|
body += mediadetail
|
|
|
|
# gallery navigation
|
|
gallerynav = Html("div", id='GalleryNav')
|
|
mediadetail += gallerynav
|
|
if prev:
|
|
gallerynav += self.gallery_nav_link(prev, _('Previous'), True)
|
|
data = _('<strong id="GalleryCurrent">%(page_number)d</strong> of '
|
|
'<strong id="GalleryTotal">%(total_pages)d</strong>' ) % {
|
|
'page_number' : page_number, 'total_pages' : total_pages }
|
|
gallerynav += Html('span', data, id='GalleryPages')
|
|
if next:
|
|
gallerynav += self.gallery_nav_link(next, _('Next'), True)
|
|
|
|
# missing media error msg
|
|
errormsg = _('The file has been moved or deleted.')
|
|
missingimage = Html('span', errormsg, class_ = "MissingImage")
|
|
|
|
# begin summaryarea division
|
|
summaryarea = Html("div", id='summaryarea')
|
|
mediadetail += summaryarea
|
|
if mime_type:
|
|
if mime_type.startswith("image/"):
|
|
if not target_exists:
|
|
mediadisplay = Html("div", id='GalleryDisplay') + \
|
|
missingimage
|
|
summaryarea += mediadisplay
|
|
else:
|
|
# Check how big the image is relative to the requested 'initial'
|
|
# image size. If it's significantly bigger, scale it down to
|
|
# improve the site's responsiveness. We don't want the user to
|
|
# have to await a large download unnecessarily. Either way, set
|
|
# the display image size as requested.
|
|
orig_image_path = Utils.media_path_full(db, photo.get_path())
|
|
(width, height) = ImgManip.image_size(orig_image_path)
|
|
max_width = self.report.options['maxinitialimagewidth']
|
|
max_height = self.report.options['maxinitialimageheight']
|
|
scale_w = (float(max_width)/width) or 1 # the 'or 1' is so that
|
|
# a max of zero is ignored
|
|
|
|
scale_h = (float(max_height)/height) or 1
|
|
scale = min(scale_w, scale_h)
|
|
new_width = int(width*scale)
|
|
new_height = int(height*scale)
|
|
if scale < 0.8:
|
|
# scale factor is significant enough to warrant making a smaller image
|
|
initial_image_path = '%s_init.jpg' % os.path.splitext(newpath)[0]
|
|
initial_image_data = ImgManip.resize_to_jpeg_buffer(orig_image_path,
|
|
new_width, new_height)
|
|
if self.report.archive:
|
|
filed, dest = tempfile.mkstemp()
|
|
os.write(filed, initial_image_data)
|
|
os.close(filed)
|
|
self.report.archive.add(dest, initial_image_path)
|
|
else:
|
|
filed = open(os.path.join(self.html_dir, initial_image_path), 'w')
|
|
filed.write(initial_image_data)
|
|
filed.close()
|
|
else:
|
|
# not worth actually making a smaller image
|
|
initial_image_path = newpath
|
|
|
|
# TODO. Convert disk path to URL.
|
|
url = self.report.build_url_fname(initial_image_path, None, self.up)
|
|
if initial_image_path != newpath:
|
|
scalemsg = Html('p', '(%d x %d).' % (width, height), inline = True)
|
|
summaryarea += scalemsg
|
|
mediadisplay = Html("div", style='width:%dpx; height:%dpx;' % (new_width, new_height))
|
|
summaryarea += mediadisplay
|
|
|
|
# Feature #2634; display the mouse-selectable regions.
|
|
# See the large block at the top of this function where
|
|
# the various regions are stored in _region_items
|
|
if len(_region_items):
|
|
ordered = Html('ol', class_ = "RegionBox")
|
|
mediadisplay += ordered
|
|
while len(_region_items) > 0:
|
|
(name, x, y, w, h, linkurl) = _region_items.pop()
|
|
ordered += Html('li', style='left:%d%%; top:%d%%; width:%d%%; height:%d%%;'
|
|
% (x, y, w, h)) +(
|
|
Html('a', name, href=linkurl)
|
|
)
|
|
|
|
# display the image
|
|
if initial_image_path != newpath:
|
|
url = self.report.build_url_fname(newpath, None, self.up)
|
|
mediadisplay += Html('a', href=url) + (
|
|
Html('img', width=new_width, height=new_height, src=url,
|
|
alt=html_escape(self.page_title))
|
|
)
|
|
else:
|
|
dirname = tempfile.mkdtemp()
|
|
thmb_path = os.path.join(dirname, "temp.png")
|
|
if ThumbNails.run_thumbnailer(mime_type,
|
|
Utils.media_path_full(db,
|
|
photo.get_path()),
|
|
thmb_path, 320):
|
|
try:
|
|
path = self.report.build_path('preview', photo.handle)
|
|
npath = os.path.join(path, photo.handle) + '.png'
|
|
self.report.copy_file(thmb_path, npath)
|
|
path = npath
|
|
os.unlink(thmb_path)
|
|
except IOError:
|
|
path = os.path.join('images', 'document.png')
|
|
else:
|
|
path = os.path.join('images', 'document.png')
|
|
os.rmdir(dirname)
|
|
|
|
mediadisplay = Html("div", id='GalleryDisplay')
|
|
summaryarea += mediadisplay
|
|
if target_exists:
|
|
# TODO. Convert disk path to URL
|
|
url = self.report.build_url_fname(newpath, None, self.up)
|
|
hyper = Html('a', href=url)
|
|
# TODO. Mixup url and path
|
|
# path = convert_disk_path_to_url(path)
|
|
url = self.report.build_url_fname(path, None, self.up)
|
|
if hyper:
|
|
hyper += Html('img', src=url, alt=html_escape(self.page_title))
|
|
else:
|
|
hyper = Html('img', src=url, alt=html_escape(self.page_title))
|
|
if target_exists:
|
|
mediadisplay += hyper
|
|
else:
|
|
mediadisplay += missingimage
|
|
else:
|
|
mediadisplay = Html("div", id='GalleryDisplay')
|
|
summaryarea += mediadisplay
|
|
url = self.report.build_url_image('document.png', 'images', self.up)
|
|
mediadisplay += Html('img', src=url, alt=html_escape(self.page_title))
|
|
|
|
# media title
|
|
title = Html('h3', html_escape(self.page_title.strip()), inline = True)
|
|
summaryarea += title
|
|
|
|
# begin media table
|
|
with Html("table", class_ = "infolist gallery") as table:
|
|
summaryarea += table
|
|
|
|
# GRAMPS id
|
|
if not self.noid:
|
|
trow = Html("tr") + (
|
|
Html("td", _('GRAMPS ID'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", photo.gramps_id, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# mime type
|
|
if mime_type:
|
|
trow = Html("tr") + (
|
|
Html("td", _('File Type'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", mime_type, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# media date
|
|
date = _dd.display(photo.get_date_object())
|
|
if date:
|
|
trow = Html("tr") + (
|
|
Html("td", DHEAD, class_ = "ColumnAttribute", inline = True),
|
|
Html("td", date, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# display image Exif tags/ keys if any?
|
|
if (pyexiftaglib and mime_type.startswith('image/')):
|
|
"""
|
|
# Exif Tags/ Keys
|
|
#
|
|
# Determine if the python exif lib is installed on the system?
|
|
# yes, then use it and determine if the photo has anything written
|
|
# inside of it? No, if not?, then do not show on the media page
|
|
"""
|
|
|
|
image = pyexiv2.Image('%s' % Utils.media_path_full(db,
|
|
photo.get_path()))
|
|
image.readMetadata()
|
|
|
|
# exif data does exists
|
|
if len(image.exifKeys()):
|
|
|
|
# add clearline for better page layout
|
|
mediadetail += fullclear
|
|
|
|
# add exif title header
|
|
mediadetail += Html('h4', _('Image Exif Tags'), inline = True)
|
|
|
|
# begin exif table
|
|
with Html("table", class_ = "exifdata") as table:
|
|
mediadetail += table
|
|
|
|
for keytag in image.exifKeys():
|
|
trow = Html("tr") + (
|
|
Html("td", keytag, class_ = "ColumnAttribute", inline = True),
|
|
Html("td", image[keytag], class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
#################################################
|
|
|
|
# get media notes
|
|
notes = self.display_note_list(photo.get_note_list())
|
|
if notes is not None:
|
|
mediadetail += notes
|
|
|
|
# get attribute list
|
|
attributes = photo.get_attribute_list()
|
|
if attributes:
|
|
attrlist = []
|
|
for attr in attributes:
|
|
attrlist.append(attr)
|
|
mediadetail += self.display_attr_list(attrlist, False)
|
|
|
|
# get media sources
|
|
sources = self.display_media_sources(photo)
|
|
if sources is not None:
|
|
mediadetail += sources
|
|
|
|
# get media references
|
|
references = self.display_references(my_media_list)
|
|
if references is not None:
|
|
mediadetail += references
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(mediapage, of)
|
|
|
|
def gallery_nav_link(self, handle, name, up=False):
|
|
|
|
url = self.report.build_url_fname_html(handle, 'img', up)
|
|
name = html_escape(name)
|
|
hyper = Html('a', name, id=name, href=url, title=name, inline = True)
|
|
|
|
# return hyperlink to its callers
|
|
return hyper
|
|
|
|
def display_media_sources(self, photo):
|
|
|
|
for sref in photo.get_source_references():
|
|
self.bibli.add_reference(sref)
|
|
sourcerefs = self.display_source_refs(self.bibli)
|
|
|
|
# return source references to its callers
|
|
return sourcerefs
|
|
|
|
def copy_source_file(self, handle, photo):
|
|
db = self.report.database
|
|
|
|
ext = os.path.splitext(photo.get_path())[1]
|
|
to_dir = self.report.build_path('images', handle)
|
|
newpath = os.path.join(to_dir, handle) + ext
|
|
|
|
fullpath = Utils.media_path_full(db, photo.get_path())
|
|
try:
|
|
if self.report.archive:
|
|
self.report.archive.add(fullpath, str(newpath))
|
|
else:
|
|
to_dir = os.path.join(self.html_dir, to_dir)
|
|
if not os.path.isdir(to_dir):
|
|
os.makedirs(to_dir)
|
|
shutil.copyfile(fullpath,
|
|
os.path.join(self.html_dir, newpath))
|
|
return newpath
|
|
except (IOError, OSError), msg:
|
|
error = _("Missing media object:") + \
|
|
"%s (%s)" % (photo.get_description(), photo.get_gramps_id())
|
|
WarningDialog(error, str(msg))
|
|
return None
|
|
|
|
def copy_thumbnail(self, handle, photo):
|
|
db = self.report.database
|
|
|
|
to_dir = self.report.build_path('thumb', handle)
|
|
to_path = os.path.join(to_dir, handle) + '.png'
|
|
if photo.get_mime_type():
|
|
from_path = ThumbNails.get_thumbnail_path(Utils.media_path_full(
|
|
db,
|
|
photo.get_path()),
|
|
photo.get_mime_type())
|
|
if not os.path.isfile(from_path):
|
|
from_path = os.path.join(const.IMAGE_DIR, "document.png")
|
|
else:
|
|
from_path = os.path.join(const.IMAGE_DIR, "document.png")
|
|
|
|
self.report.copy_file(from_path, to_path)
|
|
|
|
class SurnameListPage(BasePage):
|
|
|
|
ORDER_BY_NAME = 0
|
|
ORDER_BY_COUNT = 1
|
|
|
|
def __init__(self, report, title, person_handle_list, order_by=ORDER_BY_NAME, filename="surnames"):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
if order_by == self.ORDER_BY_NAME:
|
|
of = self.report.create_file(filename)
|
|
surnamelistpage, body = self.write_header(_('Surnames'))
|
|
else:
|
|
of = self.report.create_file("surnames_count")
|
|
surnamelistpage, body = self.write_header(_('Surnames by person count'))
|
|
|
|
# begin surnames division
|
|
with Html("div", class_ = "content", id="surnames") as surnamelist:
|
|
body += surnamelist
|
|
|
|
# page description
|
|
msg = _( 'This page contains an index of all the '
|
|
'surnames in the database. Selecting a link '
|
|
'will lead to a list of individuals in the '
|
|
'database with this same surname.')
|
|
surnamelist += Html('p', msg, id='description')
|
|
|
|
# add alphabet navigation after page msg
|
|
# only if surname list not surname count
|
|
if order_by == self.ORDER_BY_NAME:
|
|
alpha_nav = alphabet_navigation(db, person_handle_list, _PERSON)
|
|
if alpha_nav is not None:
|
|
surnamelist += alpha_nav
|
|
|
|
if order_by == self.ORDER_BY_COUNT:
|
|
table_id = 'SortByCount'
|
|
else:
|
|
table_id = 'SortByName'
|
|
|
|
# begin surnamelist table and table head
|
|
with Html("table", class_ = "infolist surnamelist", id=table_id) as table:
|
|
surnamelist += table
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr") + (
|
|
Html("th", _('Letter'), class_ = "ColumnLetter", inline = True)
|
|
)
|
|
thead += trow
|
|
|
|
fname = self.report.surname_fname + self.ext
|
|
tcell = Html("th", class_ = "ColumnSurname", inline = True)
|
|
trow += tcell
|
|
hyper = Html('a', _('Surname'), href=fname)
|
|
tcell += hyper
|
|
|
|
fname = "surnames_count" + self.ext
|
|
tcell = Html("th", class_ = "ColumnQuantity", inline = True)
|
|
trow += tcell
|
|
hyper = Html('a', _('Number of People'), href=fname)
|
|
tcell += hyper
|
|
|
|
# begin table body
|
|
with Html("tbody") as tbody:
|
|
table += tbody
|
|
|
|
person_handle_list = sort_people(db, person_handle_list)
|
|
if order_by == self.ORDER_BY_COUNT:
|
|
temp_list = {}
|
|
for (surname, data_list) in person_handle_list:
|
|
index_val = "%90d_%s" % (999999999-len(data_list), surname)
|
|
temp_list[index_val] = (surname, data_list)
|
|
|
|
person_handle_list = []
|
|
for key in sorted(temp_list, key=locale.strxfrm):
|
|
person_handle_list.append(temp_list[key])
|
|
|
|
last_letter = ''
|
|
last_surname = ''
|
|
|
|
for (surname, data_list) in person_handle_list:
|
|
if len(surname) == 0:
|
|
continue
|
|
|
|
# Get a capital normalized version of the first letter of
|
|
# the surname
|
|
if surname:
|
|
letter = normalize('NFKC', surname)[0].upper()
|
|
else:
|
|
letter = u' '
|
|
# See : http://www.gramps-project.org/bugs/view.php?id=2933
|
|
(lang_country, modifier ) = locale.getlocale()
|
|
if lang_country == "sv_SE" and ( letter == u'W' or letter == u'V' ):
|
|
letter = u'V,W'
|
|
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
if letter != last_letter:
|
|
last_letter = letter
|
|
trow.attr = ' class="BeginLetter" '
|
|
|
|
tcell = Html("td", class_ = "ColumnLetter", inline = True) + (
|
|
Html('a', last_letter, name=last_letter)
|
|
)
|
|
trow += tcell
|
|
|
|
tcell = Html("td", class_ = "ColumnSurname") + \
|
|
self.surname_link(name_to_md5(surname), surname)
|
|
trow += tcell
|
|
elif surname != last_surname:
|
|
tcell = Html("td", " ", class_ = "ColumnLetter", inline = True)
|
|
trow += tcell
|
|
tcell = Html("td", class_ = "ColumnSurname") + \
|
|
self.surname_link(name_to_md5(surname), surname)
|
|
trow += tcell
|
|
last_surname = surname
|
|
tcell = Html("td", len(data_list), class_ = "ColumnQuantity", inline = True)
|
|
trow += tcell
|
|
|
|
# create footer section
|
|
# add clearline for proper styling
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(surnamelistpage, of)
|
|
|
|
def surname_link(self, fname, name, opt_val=None, up=False):
|
|
url = self.report.build_url_fname_html(fname, 'srn', up)
|
|
hyper = Html('a', name, href=url, title=name)
|
|
if opt_val is not None:
|
|
hyper += opt_val
|
|
|
|
# return hyperlink to its caller
|
|
return hyper
|
|
|
|
class IntroductionPage(BasePage):
|
|
"""
|
|
This class will create the Introduction page ...
|
|
"""
|
|
|
|
def __init__(self, report, title):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file(report.intro_fname)
|
|
intropage, body = self.write_header(_('Introduction'))
|
|
|
|
# begin Introduction division
|
|
with Html("div", class_ = "content", id="Introduction") as section:
|
|
body += section
|
|
|
|
introimg = report.add_image('introimg')
|
|
if introimg is not None:
|
|
section += introimg
|
|
|
|
note_id = report.options['intronote']
|
|
if note_id:
|
|
note = db.get_note_from_gramps_id(note_id)
|
|
note_text = self.get_note_format(note)
|
|
|
|
# attach note
|
|
section += note_text
|
|
|
|
# add clearline for proper styling
|
|
# create footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(intropage, of)
|
|
|
|
class HomePage(BasePage):
|
|
"""
|
|
This class will create the Home Page ...
|
|
"""
|
|
|
|
def __init__(self, report, title):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file("index")
|
|
homepage, body = self.write_header(_('Home'))
|
|
|
|
# begin home division
|
|
with Html("div", class_ = "content", id="Home") as section:
|
|
body += section
|
|
|
|
homeimg = report.add_image('homeimg')
|
|
if homeimg is not None:
|
|
section += homeimg
|
|
|
|
note_id = report.options['homenote']
|
|
if note_id:
|
|
note = db.get_note_from_gramps_id(note_id)
|
|
note_text = self.get_note_format(note)
|
|
|
|
# attach note
|
|
section += note_text
|
|
|
|
# create clear line for proper styling
|
|
# create footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(homepage, of)
|
|
|
|
class SourceListPage(BasePage):
|
|
|
|
def __init__(self, report, title, handle_set):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
handle_list = list(handle_set)
|
|
source_dict = {}
|
|
|
|
of = self.report.create_file("sources")
|
|
sourcelistpage, body = self.write_header(_('Sources'))
|
|
|
|
# begin source list division
|
|
with Html("div", class_ = "content", id="sources") as section:
|
|
body += section
|
|
|
|
# Sort the sources
|
|
for handle in handle_list:
|
|
source = db.get_source_from_handle(handle)
|
|
key = source.get_title() + str(source.get_gramps_id())
|
|
source_dict[key] = (source, handle)
|
|
|
|
keys = sorted(source_dict, key=locale.strxfrm)
|
|
|
|
msg = _("This page contains an index of all the sources in the "
|
|
"database, sorted by their title. Clicking on a source’s "
|
|
"title will take you to that source’s page.")
|
|
section += Html('p', msg, id='description')
|
|
|
|
# begin sourcelist table and table head
|
|
with Html("table", class_ = "infolist sourcelist") as table:
|
|
section += table
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
for (label, colclass) in [
|
|
(None, 'RowLabel'),
|
|
(_('Name'), 'Name') ]:
|
|
|
|
label = label or " "
|
|
trow += Html("th", label, class_ = "Column%s" % colclass,
|
|
inline = True)
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for index, key in enumerate(keys):
|
|
(source, handle) = source_dict[key]
|
|
|
|
trow = Html("tr") + (
|
|
Html("td", index+1, class_ = "ColumnRowLabel", inline = True)
|
|
)
|
|
tbody += trow
|
|
trow += Html("td", class_ = "ColumnName") + \
|
|
self.source_link(handle, source.get_title(), source.gramps_id)
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(sourcelistpage, of)
|
|
|
|
class SourcePage(BasePage):
|
|
|
|
def __init__(self, report, title, handle, src_list):
|
|
db = report.database
|
|
|
|
source = db.get_source_from_handle(handle)
|
|
BasePage.__init__(self, report, title, source.gramps_id)
|
|
|
|
of = self.report.create_file(source.get_handle(), 'src')
|
|
self.up = True
|
|
sourcepage, body = self.write_header(_('Sources'))
|
|
|
|
# begin source detail division
|
|
with Html("div", class_ = "content", id="SourceDetail") as section:
|
|
body += section
|
|
|
|
media_list = source.get_media_list()
|
|
thumbnail = self.display_first_image_as_thumbnail(media_list)
|
|
if thumbnail is not None:
|
|
section += thumbnail
|
|
|
|
# add section title
|
|
section += Html('h3', html_escape(self.page_title.strip()), inline = True)
|
|
|
|
# begin sources table
|
|
with Html("table", class_ = "infolist source") as table:
|
|
section += table
|
|
|
|
grampsid = None
|
|
if not self.noid:
|
|
grampsid = source.gramps_id
|
|
|
|
for (label, val) in [(_('GRAMPS ID'), grampsid),
|
|
(_('Author'), source.author),
|
|
(_('Publication information'), source.pubinfo),
|
|
(_('Abbreviation'), source.abbrev)]:
|
|
if val:
|
|
trow = Html("tr") + (
|
|
Html("td", label, class_ = "ColumnAttribute"),
|
|
Html("td", val, class_ = "ColumnValue")
|
|
)
|
|
table += trow
|
|
|
|
# additional media
|
|
sourcegallery = self.display_additional_images_as_gallery(media_list)
|
|
if sourcegallery is not None:
|
|
section += sourcegallery
|
|
|
|
# additional notes
|
|
sourcenotes = self.display_note_list(source.get_note_list())
|
|
if sourcenotes is not None:
|
|
section += sourcenotes
|
|
|
|
# references
|
|
source_references = self.display_references(src_list[source.handle])
|
|
if source_references is not None:
|
|
section += source_references
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(sourcepage, of)
|
|
|
|
class MediaListPage(BasePage):
|
|
|
|
def __init__(self, report, title):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file("media")
|
|
medialistpage, body = self.write_header(_('Media'))
|
|
|
|
# begin gallery division
|
|
with Html("div", class_ = "content", id="Gallery") as section:
|
|
body += section
|
|
|
|
msg = _("This page contains an index of all the media objects "
|
|
"in the database, sorted by their title. Clicking on "
|
|
"the title will take you to that media object’s page. "
|
|
"If you see media size densions above an image, click on the "
|
|
"image to see the full sized version. ")
|
|
section += Html('p', msg, id='description')
|
|
|
|
# begin gallery table and table head
|
|
with Html("table", class_ = "infolist gallerylist") as table:
|
|
section += table
|
|
|
|
# begin table head
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr") + (
|
|
Html("th", " ", class_ = "ColumnRowLabel", inline = True),
|
|
Html("th", _('Name'), class_ = "ColumnName", inline = True),
|
|
Html("th", DHEAD, class_ = "ColumnDate", inline = True)
|
|
)
|
|
thead += trow
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
index = 1
|
|
sort = Sort.Sort(db)
|
|
mlist = sorted(self.report.photo_list, key=sort.by_media_title_key)
|
|
|
|
for handle in mlist:
|
|
media = db.get_object_from_handle(handle)
|
|
date = _dd.display(media.get_date_object())
|
|
title = media.get_description()
|
|
if not title:
|
|
title = "[untitled]"
|
|
|
|
trow = Html("tr") + (
|
|
Html("td", index, class_ = "ColumnRowLabel", inline = True),
|
|
)
|
|
tbody += trow
|
|
tcell = Html("td", class_ = "ColumnName") + \
|
|
self.media_ref_link(handle, title)
|
|
trow += tcell
|
|
trow += Html("td", date, class_ = "ColumnDate", inline = True)
|
|
index += 1
|
|
|
|
# add footer section
|
|
# add clearline for proper styling
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(medialistpage, of)
|
|
|
|
def media_ref_link(self, handle, name, up=False):
|
|
|
|
# get media url
|
|
url = self.report.build_url_fname_html(handle, 'img', up)
|
|
|
|
# get name
|
|
name = html_escape(name)
|
|
|
|
# begin hyper link
|
|
hyper = Html('a', name, href=url, title=name)
|
|
|
|
# return hyperlink to its callers
|
|
return hyper
|
|
|
|
class DownloadPage(BasePage):
|
|
"""
|
|
This class will produce the Download Page ...
|
|
"""
|
|
|
|
def __init__(self, report, title):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
# do NOT include a Download Page
|
|
if not self.report.inc_download:
|
|
return None
|
|
|
|
# menu options for class
|
|
# download and description #1
|
|
downloadnote = self.report.downloadnote
|
|
|
|
dlfname1 = self.report.dl_fname1
|
|
dldescr1 = self.report.dl_descr1
|
|
dldescr = ''.join(wrapper.wrap(dldescr1))
|
|
|
|
# download and description #2
|
|
dlfname2 = self.report.dl_fname2
|
|
dldescr2 = self.report.dl_descr2
|
|
dldescr2 = ''.join(wrapper.wrap(dldescr2))
|
|
|
|
# download copyright
|
|
dlcopy = self.report.dl_copy
|
|
|
|
# if no filenames at all, return???
|
|
if not dlfname1 and not dlfname2:
|
|
return
|
|
|
|
of = self.report.create_file("download")
|
|
downloadpage, body = self.write_header(_('Download'))
|
|
|
|
# begin download page and table
|
|
with Html("div", class_ = "content", id="Download") as download:
|
|
body += download
|
|
|
|
# download page note
|
|
if downloadnote:
|
|
note = db.get_note_from_gramps_id(downloadnote)
|
|
note_text = self.get_note_format(note)
|
|
download += Html('p', note_text, id="description")
|
|
|
|
# begin download table
|
|
with Html("table", class_ = "infolist download") as table:
|
|
download += table
|
|
|
|
# table head
|
|
thead = Html("thead")
|
|
table += thead
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
for (label, colclass) in [
|
|
(_('File Name'), 'Filename'),
|
|
(DESCRHEAD, 'Description'),
|
|
(_('License'), 'License'),
|
|
(_('Last Modified'), 'Modified') ]:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass, inline = True)
|
|
|
|
# if dlfname1 is not None, show it???
|
|
if dlfname1:
|
|
|
|
# table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
trow = Html("tr", id='Row01')
|
|
tbody += trow
|
|
|
|
# table row 1, column 1 -- File
|
|
fname = os.path.basename(dlfname1)
|
|
tcell = Html("td", id='Col03', class_ = "Filename") + (
|
|
Html('a', fname, href=dlfname1, alt=dldescr1)
|
|
)
|
|
trow += tcell
|
|
|
|
# table Row 1, column 2 -- File Description
|
|
tcell = Html("td", id='Col01', class_ = "Description",
|
|
inline = True)
|
|
if dldescr1:
|
|
tcell += dldescr1
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# table row 1, column 3 -- Copyright License
|
|
tcell = Html("td", id='Col02', class_ = "License")
|
|
copyright = self.get_copyright_license(dlcopy)
|
|
if copyright:
|
|
tcell += copyright
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# table row 1, column 4 -- Last Modified
|
|
tcell = Html("td", id='Col04', class_ = "Modified", inline = True)
|
|
if os.path.exists(dlfname1):
|
|
modified = os.stat(dlfname1).st_mtime
|
|
last_mod = datetime.datetime.fromtimestamp(modified)
|
|
tcell += last_mod
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# if download filename #2, show it???
|
|
if dlfname2:
|
|
|
|
# begin row #2
|
|
trow = Html("tr", id='Row02')
|
|
tbody += trow
|
|
|
|
# table row 2, column 1 -- File
|
|
fname = os.path.basename(dlfname2)
|
|
tcell = Html("td", id='Col03', class_ = "Filename") + (
|
|
Html('a', fname, href=dlfname2, alt=dldescr2)
|
|
)
|
|
trow += tcell
|
|
|
|
# table row 2, column 2 -- Description
|
|
tcell = Html("td", id='Col01', class_ = "Description",
|
|
inline = True)
|
|
if dldescr2:
|
|
tcell += dldescr2
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# table row 2, column 3 -- Copyright License
|
|
tcell = Html("td", id='Col02', class_ = "License")
|
|
copyright = self.get_copyright_license(dlcopy)
|
|
if copyright:
|
|
tcell += copyright
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# table row 2, column 4 -- Last Modified
|
|
tcell = Html("td", id='Col04', class_ = "Modified", inline = True)
|
|
if os.path.exists(dlfname2):
|
|
modified = os.stat(dlfname2).st_mtime
|
|
last_mod = datetime.datetime.fromtimestamp(modified)
|
|
tcell += last_mod
|
|
else:
|
|
tcell += " "
|
|
trow += tcell
|
|
|
|
# clear line for proper styling
|
|
# create footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(downloadpage, of)
|
|
|
|
class ContactPage(BasePage):
|
|
|
|
def __init__(self, report, title):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file("contact")
|
|
contactpage, body = self.write_header(_('Contact'))
|
|
|
|
# begin contact division
|
|
with Html("div", class_ = "content", id="Contact") as section:
|
|
body += section
|
|
|
|
# begin summaryarea division
|
|
with Html("div", id='summaryarea') as summaryarea:
|
|
section += summaryarea
|
|
|
|
contactimg = report.add_image('contactimg', 200)
|
|
if contactimg is not None:
|
|
summaryarea += contactimg
|
|
|
|
# get researcher information
|
|
r = Utils.get_researcher()
|
|
|
|
with Html("div", id='researcher') as researcher:
|
|
summaryarea += researcher
|
|
if r.name:
|
|
r.name = r.name.replace(',,,', '')
|
|
researcher += Html('h3', r.name, inline = True)
|
|
if r.addr:
|
|
researcher += Html('span', r.addr, id='streetaddress')
|
|
text = "".join([r.city, r.state, r.postal])
|
|
if text:
|
|
city = Html('span', r.city, id='city', inline = True)
|
|
state = Html('span', r.state, id='state', inline = True)
|
|
postal = Html('span', r.postal, id='postalcode', inline = True)
|
|
researcher += (city, state, postal)
|
|
if r.country:
|
|
researcher += Html('span', r.country, id='country', inline = True)
|
|
if r.email:
|
|
researcher += Html('span', id='email') + (
|
|
Html('a', r.email, href='mailto:%s?subject="from GRAMPS Web Site"'
|
|
% r.email, inline = True)
|
|
)
|
|
|
|
# add clear line for proper styling
|
|
summaryarea += fullclear
|
|
|
|
note_id = report.options['contactnote']
|
|
if note_id:
|
|
note = db.get_note_from_gramps_id(note_id)
|
|
note_text = self.get_note_format(note)
|
|
|
|
# attach note
|
|
summaryarea += note_text
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for porcessing
|
|
# and close the file
|
|
self.mywriter(contactpage, of)
|
|
|
|
class IndividualPage(BasePage):
|
|
"""
|
|
This class is used to write HTML for an individual.
|
|
"""
|
|
|
|
gender_map = {
|
|
Person.MALE : _('male'),
|
|
Person.FEMALE : _('female'),
|
|
Person.UNKNOWN : _('unknown'),
|
|
}
|
|
|
|
def __init__(self, report, title, person, ind_list, place_list, src_list, attribute_list):
|
|
BasePage.__init__(self, report, title, person.gramps_id)
|
|
self.person = person
|
|
self.ind_list = ind_list
|
|
self.src_list = src_list # Used by get_citation_links()
|
|
self.bibli = Bibliography()
|
|
self.place_list = place_list
|
|
self.sort_name = self.get_name(person)
|
|
self.name = self.get_name(person)
|
|
self.attribute_list = attribute_list
|
|
db = report.database
|
|
|
|
of = self.report.create_file(person.handle, 'ppl')
|
|
self.up = True
|
|
indivdetpage, body = self.write_header(self.sort_name)
|
|
|
|
# begin individualdetail division
|
|
with Html("div", class_ = "content", id='IndividualDetail') as individualdetail:
|
|
body += individualdetail
|
|
|
|
# display a person's general data
|
|
thumbnail, name, summary = self.display_ind_general()
|
|
|
|
# if there is a thumbnail, add it also?
|
|
if thumbnail is not None:
|
|
individualdetail += thumbnail
|
|
individualdetail += (name, summary)
|
|
|
|
# display a person's events
|
|
sect2 = self.display_ind_events()
|
|
if sect2 is not None:
|
|
individualdetail += sect2
|
|
|
|
# display attributes
|
|
sect3 = self.display_attr_list(attribute_list, True)
|
|
if sect3 is not None:
|
|
individualdetail += sect3
|
|
|
|
# display parents
|
|
sect4 = self.display_ind_parents()
|
|
if sect4 is not None:
|
|
individualdetail += sect4
|
|
|
|
# display relationships
|
|
sect5 = self.display_ind_families()
|
|
if sect5 is not None:
|
|
individualdetail += sect5
|
|
|
|
# display LDS ordinance
|
|
sect6 = self.display_lds_ordinance(person)
|
|
if sect6 is not None:
|
|
individualdetail += sect6
|
|
|
|
# display address(es)
|
|
sect7 = self.display_addresses()
|
|
if sect7 is not None:
|
|
individualdetail += sect7
|
|
|
|
media_list = []
|
|
photo_list = self.person.get_media_list()
|
|
if len(photo_list) > 1:
|
|
media_list = photo_list[1:]
|
|
for handle in self.person.get_family_handle_list():
|
|
family = db.get_family_from_handle(handle)
|
|
media_list += family.get_media_list()
|
|
for evt_ref in family.get_event_ref_list():
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
media_list += event.get_media_list()
|
|
for evt_ref in self.person.get_primary_event_ref_list():
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
if event:
|
|
media_list += event.get_media_list()
|
|
|
|
# display additional images as gallery
|
|
sect8 = self.display_additional_images_as_gallery(media_list)
|
|
if sect8 is not None:
|
|
individualdetail += sect8
|
|
|
|
# display notes
|
|
sect9 = self.display_note_list(self.person.get_note_list())
|
|
if sect9 is not None:
|
|
individualdetail += sect9
|
|
|
|
# display web links
|
|
sect10 = self.display_url_list(self.person.get_url_list())
|
|
if sect10 is not None:
|
|
individualdetail += sect10
|
|
|
|
# display sources
|
|
sect11 = self.display_ind_sources(self.person.get_source_references() )
|
|
if sect11 is not None:
|
|
individualdetail += sect11
|
|
|
|
# display associations
|
|
assocs = self.person.get_person_ref_list()
|
|
if assocs:
|
|
individualdetail += self.display_ind_associations(assocs)
|
|
|
|
# display pedigree
|
|
sect13 = self.display_ind_pedigree()
|
|
if sect13 is not None:
|
|
individualdetail += sect13
|
|
|
|
# display ancestor tree
|
|
if report.options['graph']:
|
|
sect14 = self.display_tree()
|
|
if sect14 is not None:
|
|
individualdetail += sect14
|
|
|
|
# add clearline for proper styling
|
|
# create footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(indivdetpage, of)
|
|
|
|
def draw_box(self, center, col, person):
|
|
db = self.report.database
|
|
|
|
top = center - _HEIGHT/2
|
|
xoff = _XOFFSET+col*(_WIDTH+_HGAP)
|
|
sex = person.gender
|
|
if sex == Person.MALE:
|
|
divclass = "male"
|
|
elif sex == Person.FEMALE:
|
|
divclass = "female"
|
|
else:
|
|
divclass = "unknown"
|
|
|
|
boxbg = Html("div", class_ = "boxbg %s AncCol%s" % (divclass, col),
|
|
style="top: %dpx; left: %dpx;" % (top, xoff+1)
|
|
)
|
|
|
|
person_name = self.get_name(person)
|
|
if person.handle in self.ind_list:
|
|
thumbnailUrl = None
|
|
if self.create_media and col < 5:
|
|
photolist = person.get_media_list()
|
|
if photolist:
|
|
photo_handle = photolist[0].get_reference_handle()
|
|
photo = db.get_object_from_handle(photo_handle)
|
|
mime_type = photo.get_mime_type()
|
|
if mime_type:
|
|
(photoUrl, thumbnailUrl) = self.report.prepare_copy_media(photo)
|
|
thumbnailUrl = '/'.join(['..']*3 + [thumbnailUrl])
|
|
if ( Utils.win ):
|
|
thumbnailUrl = thumbnailUrl.replace('\\','/')
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl', True)
|
|
boxbg += self.person_link(url, person, name_style=True,
|
|
thumbnailUrl=thumbnailUrl)
|
|
else:
|
|
boxbg += Html('span', person_name, class_ = "unlinked", inline = True)
|
|
shadow = Html("div", class_ = "shadow", inline = True, style="top: %dpx; left: %dpx;"
|
|
% (top+_SHADOW, xoff+_SHADOW) )
|
|
|
|
return [boxbg, shadow]
|
|
|
|
def extend_line(self, y0, x0):
|
|
style = "top: %dpx; left: %dpx; width: %dpx"
|
|
bv = Html("div", class_ = "bvline", inline = True,
|
|
style=style % (y0, x0, _HGAP/2)
|
|
)
|
|
gv = Html("div", class_ = "gvline", inline = True,
|
|
style=style % (y0+_SHADOW, x0, _HGAP/2+_SHADOW)
|
|
)
|
|
return [bv, gv]
|
|
|
|
def connect_line(self, y0, y1, col):
|
|
y = min(y0, y1)
|
|
stylew = "top: %dpx; left: %dpx; width: %dpx;"
|
|
styleh = "top: %dpx; left: %dpx; height: %dpx;"
|
|
x0 = _XOFFSET + col * _WIDTH + (col-1)*_HGAP + _HGAP/2
|
|
bv = Html("div", class_ = "bvline", inline = True, style=stylew %
|
|
(y1, x0, _HGAP/2))
|
|
gv = Html("div", class_ = "gvline", inline = True, style=stylew %
|
|
(y1+_SHADOW, x0+_SHADOW, _HGAP/2+_SHADOW))
|
|
bh = Html("div", class_ = "bhline", inline = True, style=styleh %
|
|
(y, x0, abs(y0-y1)))
|
|
gh = Html("div", class_ = "gvline", inline = True, style=styleh %
|
|
(y+_SHADOW, x0+_SHADOW, abs(y0-y1)))
|
|
return [bv, gv, bh, gh]
|
|
|
|
def draw_connected_box(self, center1, center2, col, handle):
|
|
db = self.report.database
|
|
|
|
box = []
|
|
if not handle:
|
|
return box
|
|
person = db.get_person_from_handle(handle)
|
|
box = self.draw_box(center2, col, person)
|
|
box += self.connect_line(center1, center2, col)
|
|
return box
|
|
|
|
def display_tree(self):
|
|
tree = []
|
|
if not self.person.get_main_parents_family_handle():
|
|
return None
|
|
|
|
generations = self.report.options['graphgens']
|
|
max_in_col = 1 << (generations-1)
|
|
max_size = _HEIGHT*max_in_col + _VGAP*(max_in_col+1)
|
|
center = int(max_size/2)
|
|
|
|
with Html("div", id="tree", class_ = "subsection") as tree:
|
|
tree += Html('h4', _('Ancestors'), inline = True)
|
|
with Html("div", id="treeContainer",
|
|
style="width:%dpx; height:%dpx;" %
|
|
(_XOFFSET+(generations)*_WIDTH+(generations-1)*_HGAP,
|
|
max_size)
|
|
) as container:
|
|
tree += container
|
|
container += self.draw_tree(1, generations, max_size,
|
|
0, center, self.person.handle)
|
|
return tree
|
|
|
|
def draw_tree(self, gen_nr, maxgen, max_size, old_center, new_center, phandle):
|
|
db = self.report.database
|
|
|
|
tree = []
|
|
if gen_nr > maxgen:
|
|
return tree
|
|
gen_offset = int(max_size / pow(2, gen_nr+1))
|
|
person = db.get_person_from_handle(phandle)
|
|
if not person:
|
|
return tree
|
|
|
|
if gen_nr == 1:
|
|
tree = self.draw_box(new_center, 0, person)
|
|
else:
|
|
tree = self.draw_connected_box(old_center, new_center, gen_nr-1, phandle)
|
|
|
|
if gen_nr == maxgen:
|
|
return tree
|
|
|
|
family_handle = person.get_main_parents_family_handle()
|
|
if family_handle:
|
|
line_offset = _XOFFSET + gen_nr*_WIDTH + (gen_nr-1)*_HGAP
|
|
tree += self.extend_line(new_center, line_offset)
|
|
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
f_center = new_center-gen_offset
|
|
f_handle = family.get_father_handle()
|
|
tree += self.draw_tree(gen_nr+1, maxgen, max_size,
|
|
new_center, f_center, f_handle)
|
|
|
|
m_center = new_center+gen_offset
|
|
m_handle = family.get_mother_handle()
|
|
tree += self.draw_tree(gen_nr+1, maxgen, max_size,
|
|
new_center, m_center, m_handle)
|
|
return tree
|
|
|
|
def display_ind_sources(self, sourcelist):
|
|
"""
|
|
will create the "Source References" section for a person
|
|
"""
|
|
|
|
for sref in sourcelist:
|
|
self.bibli.add_reference(sref)
|
|
sourcerefs = self.display_source_refs(self.bibli)
|
|
|
|
# return to its caller
|
|
return sourcerefs
|
|
|
|
def display_ind_associations(self, assoclist):
|
|
"""
|
|
display an individual's associations
|
|
"""
|
|
|
|
# begin Associations division
|
|
with Html("div", class_ = "subsection", id="Associations") as section:
|
|
section += Html('h4', _('Associations'), inline = True)
|
|
|
|
with Html("table", class_ = "infolist assoclist") as table:
|
|
section += table
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr")
|
|
thead += trow
|
|
|
|
assoc_row = [
|
|
(_('Relationship'), 'Relationship'),
|
|
(SHEAD, 'Sources'),
|
|
(NHEAD, 'Notes') ]
|
|
|
|
for (label, colclass) in assoc_row:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass,
|
|
inline = True)
|
|
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for person_ref in assoclist:
|
|
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
|
|
index = 0
|
|
for data in [
|
|
[person_ref.get_relation()],
|
|
[self.get_citation_links(person_ref.get_source_references())],
|
|
[self.dump_notes(person_ref.get_note_list())] ]:
|
|
|
|
# get colclass from assoc_row
|
|
colclass = assoc_row[index][1]
|
|
|
|
trow += Html("td", data, class_ = "Column%s" % colclass,
|
|
inline = True)
|
|
index += 1
|
|
|
|
# return section to its callers
|
|
return section
|
|
|
|
def display_ind_pedigree(self):
|
|
"""
|
|
Display an individual's pedigree
|
|
"""
|
|
db = self.report.database
|
|
|
|
# Define helper functions
|
|
|
|
def children_ped(ol):
|
|
if family:
|
|
for child_ref in family.get_child_ref_list():
|
|
child_handle = child_ref.ref
|
|
if child_handle == self.person.handle:
|
|
child_ped(ol)
|
|
else:
|
|
child = db.get_person_from_handle(child_handle)
|
|
ol += Html('li') + self.pedigree_person(child)
|
|
else:
|
|
child_ped(ol)
|
|
return ol
|
|
|
|
def child_ped(ol):
|
|
ol += Html('li', class_ = "thisperson", inline = True) + self.name
|
|
family = self.pedigree_family()
|
|
if family:
|
|
ol += Html('ol', class_ = "spouselist") + family
|
|
return ol
|
|
|
|
# End of helper functions
|
|
|
|
parent_handle_list = self.person.get_parent_family_handle_list()
|
|
if parent_handle_list:
|
|
parent_handle = parent_handle_list[0]
|
|
family = db.get_family_from_handle(parent_handle)
|
|
father_handle = family.get_father_handle()
|
|
mother_handle = family.get_mother_handle()
|
|
mother = db.get_person_from_handle(mother_handle)
|
|
father = db.get_person_from_handle(father_handle)
|
|
else:
|
|
family = None
|
|
father = None
|
|
mother = None
|
|
|
|
with Html("div", id="pedigree", class_ = "subsection") as ped:
|
|
ped += Html('h4', _('Pedigree'), inline = True)
|
|
with Html('ol', class_ = "pedigreegen") as pedol:
|
|
ped += pedol
|
|
if father and mother:
|
|
pedfa = Html('li') + self.pedigree_person(father)
|
|
pedol += pedfa
|
|
with Html('ol') as pedma:
|
|
pedfa += pedma
|
|
pedma += (Html('li', class_ = "spouse") +
|
|
self.pedigree_person(mother) +
|
|
children_ped(Html('ol'))
|
|
)
|
|
elif father:
|
|
pedol += (Html('li') + self.pedigree_person(father) +
|
|
children_ped(Html('ol'))
|
|
)
|
|
elif mother:
|
|
pedol += (Html('li') + self.pedigree_person(mother) +
|
|
children_ped(Html('ol'))
|
|
)
|
|
else:
|
|
pedol += children_ped(Html('ol'))
|
|
return ped
|
|
|
|
def display_ind_general(self):
|
|
"""
|
|
display an individual's general information...
|
|
"""
|
|
db = self.report.database
|
|
|
|
self.page_title = self.sort_name
|
|
thumbnail = self.display_first_image_as_thumbnail(self.person.get_media_list())
|
|
|
|
section_title = Html('h3', self.get_name(self.person), inline = True)
|
|
|
|
# begin summaryarea division
|
|
with Html("div", id='summaryarea') as summaryarea:
|
|
|
|
# begin general details table
|
|
with Html("table", class_ = "infolist") as table:
|
|
summaryarea += table
|
|
|
|
primary_name = self.person.get_primary_name()
|
|
all_names = [primary_name] + self.person.get_alternate_names()
|
|
|
|
# Names [and their sources]
|
|
for name in all_names:
|
|
pname = _nd.display_name(name)
|
|
pname += self.get_citation_links( name.get_source_references() )
|
|
|
|
# if we have just a firstname, then the name is preceeded by ", "
|
|
# which doesn't exactly look very nice printed on the web page
|
|
if pname[:2] == ', ':
|
|
pname = pname[2:]
|
|
|
|
type_ = str( name.get_type() )
|
|
trow = Html("tr") + (
|
|
Html("td", type_, class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
tcell = Html("td", pname, class_ = "ColumnValue", inline = True)
|
|
trow += tcell
|
|
|
|
# display any notes associated with this name
|
|
notelist = name.get_note_list()
|
|
if len(notelist):
|
|
unordered = Html('ul')
|
|
tcell += unordered
|
|
|
|
for notehandle in notelist:
|
|
note = db.get_note_from_handle(notehandle)
|
|
if note:
|
|
note_text = self.get_note_format(note)
|
|
|
|
# attach note
|
|
unordered += note_text
|
|
|
|
# display call names
|
|
first_name = primary_name.get_first_name()
|
|
for name in all_names:
|
|
call_name = name.get_call_name()
|
|
if call_name and call_name != first_name:
|
|
call_name += self.get_citation_links(
|
|
name.get_source_references() )
|
|
trow = Html("tr") + (
|
|
Html("td", _('Call Name'), class_ = "ColumnAttribute",
|
|
inline = True),
|
|
Html("td", call_name, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# display the nickname attribute
|
|
nick_name = self.person.get_nick_name()
|
|
if nick_name and nick_name != first_name:
|
|
nick_name += self.get_citation_links(
|
|
self.person.get_source_references() )
|
|
trow = Html("tr") + (
|
|
Html("td", _('Nick Name'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", nick_name, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# GRAMPS ID
|
|
if not self.noid:
|
|
trow = Html("tr") + (
|
|
Html("td", _('GRAMPS ID'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", self.person.gramps_id, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# Gender
|
|
gender = self.gender_map[self.person.gender]
|
|
trow = Html("tr") + (
|
|
Html("td", _('Gender'), class_ = "ColumnAttribute", inline = True),
|
|
Html("td", gender, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# Age At Death???
|
|
birth_ref = self.person.get_birth_ref()
|
|
birth_date = None
|
|
if birth_ref:
|
|
birth_event = db.get_event_from_handle(birth_ref.ref)
|
|
birth_date = birth_event.get_date_object()
|
|
|
|
if birth_date is not None:
|
|
alive = probably_alive(self.person, db, date.Today())
|
|
death_ref = self.person.get_death_ref()
|
|
death_date = None
|
|
if death_ref:
|
|
death_event = db.get_event_from_handle(death_ref.ref)
|
|
death_date = death_event.get_date_object()
|
|
|
|
if not alive and death_date is not None:
|
|
nyears = death_date - birth_date
|
|
nyears.format(precision=3)
|
|
trow = Html("tr") + (
|
|
Html("td", _('Age at Death'), class_ = "ColumnAttribute",
|
|
inline = True),
|
|
Html("td", nyears, class_ = "ColumnValue", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# return all three pieces to its caller
|
|
# do NOT combine before returning to class IndividualPage
|
|
return thumbnail, section_title, summaryarea
|
|
|
|
def display_ind_events(self):
|
|
"""
|
|
will create the events table
|
|
"""
|
|
|
|
evt_ref_list = self.person.get_event_ref_list()
|
|
|
|
if not evt_ref_list:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# begin events division and section title
|
|
with Html("div", class_ = "subsection", id="events") as section:
|
|
section += Html('h4', _('Events'), inline = True)
|
|
|
|
# begin events table
|
|
with Html("table", class_ = "infolist eventlist") as table:
|
|
section += table
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
"""
|
|
@param: show place
|
|
@param: show description
|
|
@param: show source references
|
|
@param: show note
|
|
"""
|
|
thead += self.display_event_header(True, False, True, False)
|
|
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for evt_ref in evt_ref_list:
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
|
|
# display event row
|
|
"""
|
|
@param: event object
|
|
@param: event_ref = event reference
|
|
@param: show place or not?
|
|
@param: show description or not?
|
|
@param: show source references or not?
|
|
@param: shownote = show notes or not?
|
|
@param: subdirs = True or False
|
|
@param: hyp = show hyperlinked evt type or not?
|
|
"""
|
|
tbody += self.display_event_row(event, evt_ref, True, False, True,
|
|
False, True, True)
|
|
|
|
# return section to its caller
|
|
return section
|
|
|
|
def display_addresses(self):
|
|
"""
|
|
display a person's addresses ...
|
|
"""
|
|
|
|
alist = self.person.get_address_list()
|
|
if not alist:
|
|
return None
|
|
|
|
# begin addresses division and title
|
|
with Html("div", class_ = "subsection", id="Addresses") as section:
|
|
section += Html('h4', _('Addresses'), inline = True)
|
|
|
|
# write out addresses()
|
|
section += self.dump_addresses(self.person)
|
|
|
|
# return address division to its caller
|
|
return section
|
|
|
|
def display_lds_ordinance(self, person):
|
|
"""
|
|
display LDS information for a person or family
|
|
"""
|
|
|
|
ldsordlist = person.lds_ord_list
|
|
if not ldsordlist:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# begin LDS Ordinance division and section title
|
|
with Html("div", class_ = "subsection", id="LDSOrdinance") as section:
|
|
section += Html('h4', _('Latter-Day Saints (LDS) Ordinance'), inline = True)
|
|
|
|
# ump individual LDS ordinance list
|
|
section += self.dump_ordinance(db, self.person)
|
|
|
|
# return section to its caller
|
|
return section
|
|
|
|
def display_child_link(self, child_handle):
|
|
"""
|
|
display child link ...
|
|
"""
|
|
db = self.report.database
|
|
|
|
child = db.get_person_from_handle(child_handle)
|
|
gid = child.gramps_id
|
|
list = Html('li')
|
|
if child_handle in self.ind_list:
|
|
url = self.report.build_url_fname_html(child_handle, 'ppl', True)
|
|
list += self.person_link(url, child, True, gid)
|
|
|
|
else:
|
|
list += self.get_name(child)
|
|
|
|
# return list to its caller
|
|
return list
|
|
|
|
def display_parent(self, handle, title, rel):
|
|
"""
|
|
This will display a parent ...
|
|
"""
|
|
db = self.report.database
|
|
|
|
person = db.get_person_from_handle(handle)
|
|
tcell1 = Html("td", title, class_ = "ColumnAttribute", inline = True)
|
|
tcell2 = Html("td", class_ = "ColumnValue")
|
|
|
|
gid = person.gramps_id
|
|
if handle in self.ind_list:
|
|
url = self.report.build_url_fname_html(handle, 'ppl', True)
|
|
tcell2 += self.person_link(url, person, True, gid)
|
|
else:
|
|
person_name = self.get_name(person)
|
|
tcell2 += person_name
|
|
if rel and rel != ChildRefType(ChildRefType.BIRTH):
|
|
tcell2 += ' (%s)' % str(rel)
|
|
|
|
# return table columns to its caller
|
|
return tcell1, tcell2
|
|
|
|
def display_ind_parents(self):
|
|
"""
|
|
Display a person's parents
|
|
"""
|
|
|
|
birthorder = self.report.options['birthorder']
|
|
parent_list = self.person.get_parent_family_handle_list()
|
|
|
|
if not parent_list:
|
|
return None
|
|
|
|
db = self.report.database
|
|
|
|
# begin parents division
|
|
with Html("div", class_ = "subsection", id="parents") as section:
|
|
section += Html('h4', _('Parents'), inline = True)
|
|
|
|
# begin parents table
|
|
with Html("table", class_ = "infolist") as table:
|
|
section += table
|
|
|
|
first = True
|
|
if parent_list:
|
|
for family_handle in parent_list:
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
# Get the mother and father relationships
|
|
frel = None
|
|
mrel = None
|
|
sibling = set()
|
|
|
|
child_handle = self.person.get_handle()
|
|
child_ref_list = family.get_child_ref_list()
|
|
for child_ref in child_ref_list:
|
|
if child_ref.ref == child_handle:
|
|
frel = child_ref.get_father_relation()
|
|
mrel = child_ref.get_mother_relation()
|
|
break
|
|
|
|
if not first:
|
|
trow = Html("tr") +(
|
|
Html("td", " ", colspan=2, inline = True)
|
|
)
|
|
table += trow
|
|
else:
|
|
first = False
|
|
|
|
father_handle = family.get_father_handle()
|
|
if father_handle:
|
|
trow = Html("tr")
|
|
table += trow
|
|
|
|
tabcol1, tabcol2 = self.display_parent(father_handle, _('Father'), frel)
|
|
trow += (tabcol1, tabcol2)
|
|
mother_handle = family.get_mother_handle()
|
|
if mother_handle:
|
|
trow = Html("tr")
|
|
table += trow
|
|
tabcol1, tabcol2 = self.display_parent(mother_handle, _('Mother'), mrel)
|
|
trow += (tabcol1, tabcol2)
|
|
|
|
first = False
|
|
if len(child_ref_list) > 1:
|
|
childlist = [child_ref.ref for child_ref in child_ref_list]
|
|
for child_handle in childlist:
|
|
sibling.add(child_handle) # remember that
|
|
# we've already "seen" this child
|
|
|
|
# now that we have all natural siblings, display them...
|
|
if len(sibling):
|
|
trow = Html("tr") + (
|
|
Html("td", _('Siblings'), class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
tcell = Html("td", class_ = "ColumnValue")
|
|
trow += tcell
|
|
ordered = Html('ol')
|
|
tcell += ordered
|
|
|
|
if birthorder:
|
|
kids = []
|
|
kids = sorted(add_birthdate(db, sibling))
|
|
|
|
for birth_date, child_handle in kids:
|
|
if child_handle != self.person.handle:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
else:
|
|
|
|
for child_handle in sibling:
|
|
if child_handle != self.person.handle:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
# Also try to identify half-siblings
|
|
half_siblings = set()
|
|
|
|
# if we have a known father...
|
|
showallsiblings = self.report.options['showhalfsiblings']
|
|
if father_handle and showallsiblings:
|
|
# 1) get all of the families in which this father is involved
|
|
# 2) get all of the children from those families
|
|
# 3) if the children are not already listed as siblings...
|
|
# 4) then remember those children since we're going to list them
|
|
father = db.get_person_from_handle(father_handle)
|
|
for family_handle in father.get_family_handle_list():
|
|
family = db.get_family_from_handle(family_handle)
|
|
for half_child_ref in family.get_child_ref_list():
|
|
half_child_handle = half_child_ref.ref
|
|
if half_child_handle not in sibling:
|
|
if half_child_handle != self.person.handle:
|
|
# we have a new step/half sibling
|
|
half_siblings.add(half_child_handle)
|
|
|
|
# do the same thing with the mother (see "father" just above):
|
|
if mother_handle and showallsiblings:
|
|
mother = db.get_person_from_handle(mother_handle)
|
|
for family_handle in mother.get_family_handle_list():
|
|
family = db.get_family_from_handle(family_handle)
|
|
for half_child_ref in family.get_child_ref_list():
|
|
half_child_handle = half_child_ref.ref
|
|
if half_child_handle not in sibling:
|
|
if half_child_handle != self.person.handle:
|
|
# we have a new half sibling
|
|
half_siblings.add(half_child_handle)
|
|
|
|
# now that we have all half- siblings, display them...
|
|
if len(half_siblings):
|
|
trow = Html("tr") + (
|
|
Html("td", _('Half Siblings'), class_ = "ColumnAttribute",
|
|
inline = True),
|
|
)
|
|
table += trow
|
|
tcell = Html("td", class_ = "ColumnValue")
|
|
trow += tcell
|
|
ordered = Html('ol')
|
|
tcell += ordered
|
|
|
|
if birthorder:
|
|
kids = []
|
|
kids = sorted(add_birthdate(db, half_siblings))
|
|
|
|
for birth_date, child_handle in kids:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
else:
|
|
|
|
for child_handle in half_siblings:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
# get step-siblings
|
|
if showallsiblings:
|
|
step_siblings = set()
|
|
|
|
# to find the step-siblings, we need to identify
|
|
# all of the families that can be linked back to
|
|
# the current person, and then extract the children
|
|
# from those families
|
|
all_family_handles = set()
|
|
all_parent_handles = set()
|
|
tmp_parent_handles = set()
|
|
|
|
# first we queue up the parents we know about
|
|
if mother_handle:
|
|
tmp_parent_handles.add(mother_handle)
|
|
if father_handle:
|
|
tmp_parent_handles.add(father_handle)
|
|
|
|
while len(tmp_parent_handles) > 0:
|
|
# pop the next parent from the set
|
|
parent_handle = tmp_parent_handles.pop()
|
|
|
|
# add this parent to our official list
|
|
all_parent_handles.add(parent_handle)
|
|
|
|
# get all families with this parent
|
|
parent = db.get_person_from_handle(parent_handle)
|
|
for family_handle in parent.get_family_handle_list():
|
|
|
|
all_family_handles.add(family_handle)
|
|
|
|
# we already have 1 parent from this family
|
|
# (see "parent" above) so now see if we need
|
|
# to queue up the other parent
|
|
family = db.get_family_from_handle(family_handle)
|
|
tmp_mother_handle = family.get_mother_handle()
|
|
if tmp_mother_handle and \
|
|
tmp_mother_handle != parent and \
|
|
tmp_mother_handle not in tmp_parent_handles and \
|
|
tmp_mother_handle not in all_parent_handles:
|
|
tmp_parent_handles.add(tmp_mother_handle)
|
|
tmp_father_handle = family.get_father_handle()
|
|
if tmp_father_handle and \
|
|
tmp_father_handle != parent and \
|
|
tmp_father_handle not in tmp_parent_handles and \
|
|
tmp_father_handle not in all_parent_handles:
|
|
tmp_parent_handles.add(tmp_father_handle)
|
|
|
|
# once we get here, we have all of the families
|
|
# that could result in step-siblings; note that
|
|
# we can only have step-siblings if the number
|
|
# of families involved is > 1
|
|
|
|
if len(all_family_handles) > 1:
|
|
while len(all_family_handles) > 0:
|
|
# pop the next family from the set
|
|
family_handle = all_family_handles.pop()
|
|
# look in this family for children we haven't yet seen
|
|
family = db.get_family_from_handle(family_handle)
|
|
for step_child_ref in family.get_child_ref_list():
|
|
step_child_handle = step_child_ref.ref
|
|
if step_child_handle not in sibling and \
|
|
step_child_handle not in half_siblings and \
|
|
step_child_handle != self.person.handle:
|
|
# we have a new step sibling
|
|
step_siblings.add(step_child_handle)
|
|
|
|
# now that we have all step- siblings, display them...
|
|
if len(step_siblings):
|
|
trow = Html("tr") + (
|
|
Html("td", _('Step Siblings'), class_ = "ColumnAttribute",
|
|
inline = True)
|
|
)
|
|
table += trow
|
|
tcell = Html("td", class_ = "ColumnValue")
|
|
trow += tcell
|
|
ordered = Html('ol')
|
|
tcell += ordered
|
|
|
|
if birthorder:
|
|
kids = []
|
|
kids = sorted(add_birthdate(db, step_siblings))
|
|
|
|
for birth_date, child_handle in kids:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
else:
|
|
|
|
for child_handle in step_siblings:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
# return parents division to its caller
|
|
return section
|
|
|
|
def display_ind_families(self):
|
|
"""
|
|
Displays a person's relationships ...
|
|
"""
|
|
|
|
family_list = self.person.get_family_handle_list()
|
|
if not family_list:
|
|
return None
|
|
db = self.report.database
|
|
|
|
# begin families division and section title
|
|
with Html("div", class_ = "subsection", id="families") as section:
|
|
section += Html('h4', _('Families'), inline = True)
|
|
|
|
# begin families table
|
|
with Html("table", class_ = "infolist") as table:
|
|
section += table
|
|
|
|
for family_handle in family_list:
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
self.display_partner(family, table)
|
|
|
|
childlist = family.get_child_ref_list()
|
|
if childlist:
|
|
trow = Html("tr") + (
|
|
Html("td", " ", class_ = "ColumnType", inline = True),
|
|
Html("td", _('Children'), class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
tcell = Html("td", class_ = "ColumnValue")
|
|
trow += tcell
|
|
ordered = Html('ol')
|
|
tcell += ordered
|
|
childlist = [child_ref.ref for child_ref in childlist]
|
|
|
|
if self.report.options['birthorder']:
|
|
kids = []
|
|
kids = sorted(add_birthdate(db, childlist))
|
|
|
|
for birth_date, child_handle in kids:
|
|
ordered += self.display_child_link(child_handle)
|
|
else:
|
|
|
|
for child_handle in childlist:
|
|
ordered += self.display_child_link(child_handle)
|
|
|
|
# family LDS ordinance list
|
|
famldslist = family.get_lds_ord_list()
|
|
if famldslist:
|
|
trow = Html("tr") + (
|
|
Html("td", " ", class_ = "ColumnType", inline = True),
|
|
Html("td", ' ', class_ = "ColumnAttribute", inline = True),
|
|
Html("td", self.dump_ordinance(db, family, 'Family'),
|
|
class_ = "ColumnValue")
|
|
)
|
|
table += trow
|
|
|
|
# return section to its caller
|
|
return section
|
|
|
|
def display_partner(self, family, table):
|
|
"""
|
|
display an individual's partner
|
|
"""
|
|
|
|
gender = self.person.gender
|
|
reltype = family.get_relationship()
|
|
db = self.report.database
|
|
|
|
if reltype == FamilyRelType.MARRIED:
|
|
if gender == Person.FEMALE:
|
|
relstr = _("Husband")
|
|
elif gender == Person.MALE:
|
|
relstr = _("Wife")
|
|
else:
|
|
relstr = _("Partner")
|
|
else:
|
|
relstr = _("Partner")
|
|
|
|
partner_handle = ReportUtils.find_spouse(self.person, family)
|
|
if partner_handle:
|
|
partner = db.get_person_from_handle(partner_handle)
|
|
partner_name = self.get_name(partner)
|
|
else:
|
|
partner_name = _("unknown")
|
|
|
|
# family relationship type
|
|
rtype = str(family.get_relationship())
|
|
trow = Html("tr", class_ = "BeginFamily") + (
|
|
Html("td", rtype, class_ = "ColumnType", inline = True),
|
|
Html("td", relstr, class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
tcell = Html("td", class_ = "ColumnValue")
|
|
trow += tcell
|
|
|
|
# display partner's name
|
|
if partner_handle:
|
|
if partner_handle in self.ind_list:
|
|
url = self.report.build_url_fname_html(partner_handle, 'ppl', True)
|
|
tcell += self.person_link(url, partner, True, partner.gramps_id)
|
|
else:
|
|
tcell += partner_name
|
|
|
|
# display family events; such as marriage and divorce events
|
|
family_events = family.get_event_ref_list()
|
|
if family_events:
|
|
trow = Html("tr") + (
|
|
Html("td", " ", class_ = "ColumnType", inline = True),
|
|
Html("td", " ", class_ = "ColumnAttribute", inline = True),
|
|
Html("td", self.format_event(family_events), class_ = "ColumnValue")
|
|
)
|
|
table += trow
|
|
|
|
# return table to its caller
|
|
return table
|
|
|
|
def pedigree_person(self, person):
|
|
"""
|
|
will produce a hyperlink for a pedigree person ...
|
|
"""
|
|
|
|
person_name = self.get_name(person)
|
|
if person.handle in self.ind_list:
|
|
url = self.report.build_url_fname_html(person.handle, 'ppl', True)
|
|
hyper = self.person_link(url, person, name_style=True)
|
|
else:
|
|
hyper = person_name
|
|
|
|
# return hyperlink to its callers
|
|
# can be an actual hyperlink or just a person's name
|
|
return hyper
|
|
|
|
def pedigree_family(self):
|
|
"""
|
|
Returns a family pedigree
|
|
"""
|
|
db = self.report.database
|
|
|
|
ped = []
|
|
for family_handle in self.person.get_family_handle_list():
|
|
rel_family = db.get_family_from_handle(family_handle)
|
|
spouse_handle = ReportUtils.find_spouse(self.person, rel_family)
|
|
if spouse_handle:
|
|
spouse = db.get_person_from_handle(spouse_handle)
|
|
pedsp = (Html('li', class_ = "spouse") +
|
|
self.pedigree_person(spouse)
|
|
)
|
|
ped += [pedsp]
|
|
else:
|
|
pedsp = ped
|
|
childlist = rel_family.get_child_ref_list()
|
|
if childlist:
|
|
with Html('ol') as childol:
|
|
pedsp += [childol]
|
|
for child_ref in childlist:
|
|
child = db.get_person_from_handle(child_ref.ref)
|
|
childol += (Html('li') +
|
|
self.pedigree_person(child)
|
|
)
|
|
return ped
|
|
|
|
def display_event_header(self, showplc, showdescr, showsrc, shownote):
|
|
"""
|
|
will print the event header row for display_event_row() and
|
|
format_event()
|
|
|
|
@param: showplc = show place
|
|
@param: showdescr = show description
|
|
@param: showsrc = show source references
|
|
@param: shownote = show notes or not?
|
|
"""
|
|
# position 0 = translatable label, position 1 = column class, and
|
|
# position 2 = data
|
|
event_header_row = [
|
|
(THEAD, 'Type'),
|
|
(DHEAD, 'Date') ]
|
|
|
|
if showplc:
|
|
place_row = (PHEAD, 'Place')
|
|
event_header_row.append(place_row)
|
|
|
|
if showdescr:
|
|
descr_row = (DESCRHEAD, 'Description')
|
|
event_header_row.append(descr_row)
|
|
|
|
if showsrc:
|
|
source_row = (SHEAD, 'Sources')
|
|
event_header_row.append(source_row)
|
|
|
|
if shownote:
|
|
note_row = (NHEAD, 'Notes')
|
|
event_header_row.append(note_row)
|
|
|
|
trow = Html("tr")
|
|
for (label, colclass) in event_header_row:
|
|
trow += Html("th", label, class_ = "Column%s" % colclass, inline = True)
|
|
|
|
# return header row to its callers
|
|
return trow
|
|
|
|
def format_event(self, eventlist):
|
|
db = self.report.database
|
|
|
|
# begin eventlist table and table header
|
|
with Html("table", class_ = "infolist eventlist") as table:
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
# attach event header row
|
|
"""
|
|
@param: show place
|
|
@param: show description
|
|
@param: show source references
|
|
@param: show note
|
|
"""
|
|
thead += self.display_event_header(True, False, True, False)
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
for event_ref in eventlist:
|
|
event = db.get_event_from_handle(event_ref.ref)
|
|
|
|
# add event body row
|
|
"""
|
|
@param: event object
|
|
@param: event_ref = event reference
|
|
@param: show place or not?
|
|
@param: show description or not?
|
|
@param: show source references or not?
|
|
@param: shownote = show notes or not?
|
|
@param: subdirs = True or False
|
|
@param: hyp = show hyperlinked evt type or not?
|
|
"""
|
|
tbody += self.display_event_row(event, event_ref, True, False,
|
|
True, False, True, True)
|
|
|
|
# return table to its callers
|
|
return table
|
|
|
|
class RepositoryListPage(BasePage):
|
|
"""
|
|
Will create the repository list page
|
|
"""
|
|
|
|
def __init__(self, report, title, repos_dict, keys):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file('repositories')
|
|
repolistpage, body = self.write_header(_('Repositories'))
|
|
|
|
# begin RepositoryList division
|
|
with Html("div", class_ = "content", id="RepositoryList") as repositorylist:
|
|
body += repositorylist
|
|
|
|
msg = _("This page contains an index of all the repositories in the "
|
|
"database, sorted by their title. Clicking on a repositories’s "
|
|
"title will take you to that repositories’s page.")
|
|
repositorylist += Html('p', msg, id='description')
|
|
|
|
# begin repositories table and table head
|
|
with Html("table", class_ = "infolist repolist") as table:
|
|
repositorylist += table
|
|
|
|
thead = Html("thead")
|
|
table += thead
|
|
|
|
trow = Html("tr") + (
|
|
Html("th", " ", class_ = "ColumnRowLabel", inline = True),
|
|
Html("th", _('Type'), class_ = "ColumnType", inline = True),
|
|
Html("th", _('Name'), class_ = "ColumnName", inline = True)
|
|
)
|
|
thead += trow
|
|
|
|
# begin table body
|
|
tbody = Html("tbody")
|
|
table += tbody
|
|
|
|
index = 0
|
|
for index, key in enumerate(keys):
|
|
(repo, handle) = repos_dict[key]
|
|
|
|
trow = Html("tr")
|
|
tbody += trow
|
|
|
|
# index number
|
|
tcell = Html("td", index+1, class_ = "ColumnRowLabel", inline = True)
|
|
trow += tcell
|
|
|
|
# repository type
|
|
rtype = repo.type.xml_str()
|
|
for xtype in RepositoryType._DATAMAP:
|
|
if rtype == xtype[2]:
|
|
rtype = xtype[1]
|
|
break
|
|
if rtype:
|
|
tcell = Html("td", rtype, class_ = "ColumnType", inline = True)
|
|
trow += tcell
|
|
|
|
# repository name and hyperlink
|
|
repo_title = html_escape(repo.name)
|
|
if repo_title:
|
|
tcell = Html("td", class_ = "ColumnName") + \
|
|
self.repository_link(handle, repo_title, repo.gramps_id)
|
|
trow += tcell
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(repolistpage, of)
|
|
|
|
class RepositoryPage(BasePage):
|
|
"""
|
|
will create the individual Repository Pages
|
|
"""
|
|
|
|
def __init__(self, report, title, repo, handle):
|
|
BasePage.__init__(self, report, title)
|
|
db = report.database
|
|
|
|
of = self.report.create_file(handle, 'repo')
|
|
self.up = True
|
|
repositorypage, body = self.write_header(_('Repositories'))
|
|
|
|
# begin RepositoryDetail division and page title
|
|
with Html("div", class_ = "content", id="RepositoryDetail") as repositorydetail:
|
|
body += repositorydetail
|
|
|
|
# repository name
|
|
repositorydetail += Html('h3', repo.name, inline = True)
|
|
|
|
# begin repository table
|
|
with Html("table", class_ = "infolist repolist") as table:
|
|
repositorydetail += table
|
|
|
|
# repository type
|
|
trow = Html("tr") + (
|
|
Html("td", _('Type'), class_ = "ColumnType", inline = True),
|
|
Html("td", str(repo.type), class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
if not self.noid:
|
|
# repo gramps id
|
|
trow = Html("tr") + (
|
|
Html("td", _('GRAMPS ID'), class_ = "ColumnType", inline = True),
|
|
Html("td", repo.gramps_id, class_ = "ColumnAttribute", inline = True)
|
|
)
|
|
table += trow
|
|
|
|
# repository: address(es)
|
|
addresses = self.dump_addresses(repo, False)
|
|
if addresses is not None:
|
|
repositorydetail += addresses
|
|
|
|
# repository: urllist
|
|
urllist = self.display_url_list(repo.get_url_list())
|
|
if urllist is not None:
|
|
repositorydetail += urllist
|
|
|
|
# reposity: notelist
|
|
notelist = self.display_note_list(repo.get_note_list())
|
|
if notelist is not None:
|
|
repositorydetail += notelist
|
|
|
|
# add clearline for proper styling
|
|
# add footer section
|
|
footer = self.write_footer()
|
|
body += (fullclear, footer)
|
|
|
|
# send page out for processing
|
|
# and close the file
|
|
self.mywriter(repositorypage, of)
|
|
|
|
class NavWebReport(Report):
|
|
|
|
def __init__(self, database, options):
|
|
"""
|
|
Create WebReport object that produces the report.
|
|
|
|
The arguments are:
|
|
|
|
database - the GRAMPS database instance
|
|
options - instance of the Options class for this report
|
|
"""
|
|
Report.__init__(self, database, options)
|
|
menu = options.menu
|
|
self.options = {}
|
|
|
|
for optname in menu.get_all_option_names():
|
|
menuopt = menu.get_option_by_name(optname)
|
|
self.options[optname] = menuopt.get_value()
|
|
|
|
if not self.options['incpriv']:
|
|
self.database = PrivateProxyDb(database)
|
|
else:
|
|
self.database = database
|
|
|
|
livinginfo = self.options['living']
|
|
yearsafterdeath = self.options['yearsafterdeath']
|
|
|
|
if livinginfo != _INCLUDE_LIVING_VALUE:
|
|
self.database = LivingProxyDb(self.database,
|
|
livinginfo,
|
|
None,
|
|
yearsafterdeath)
|
|
|
|
filters_option = menu.get_option_by_name('filter')
|
|
self.filter = filters_option.get_filter()
|
|
|
|
self.copyright = self.options['cright']
|
|
self.target_path = self.options['target']
|
|
self.ext = self.options['ext']
|
|
self.css = self.options['css']
|
|
|
|
self.title = self.options['title']
|
|
self.inc_gallery = self.options['gallery']
|
|
self.inc_contact = self.options['contactnote'] or \
|
|
self.options['contactimg']
|
|
|
|
# name format option
|
|
self.name_format = self.options['name_format']
|
|
|
|
# create an event pages or not?
|
|
self.inc_events = self.options['inc_events']
|
|
|
|
# include repository page or not?
|
|
self.inc_repository = self.options['inc_repository']
|
|
|
|
# Download Options Tab
|
|
self.inc_download = self.options['incdownload']
|
|
self.downloadnote = self.options['downloadnote']
|
|
self.dl_fname1 = self.options['down_fname1']
|
|
self.dl_descr1 = self.options['dl_descr1']
|
|
self.dl_fname2 = self.options['down_fname2']
|
|
self.dl_descr2 = self.options['dl_descr2']
|
|
self.dl_copy = self.options['dl_cright']
|
|
|
|
self.encoding = self.options['encoding']
|
|
|
|
self.use_archive = self.options['archive']
|
|
self.use_intro = self.options['intronote'] or \
|
|
self.options['introimg']
|
|
self.use_home = self.options['homenote'] or \
|
|
self.options['homeimg']
|
|
self.use_contact = self.options['contactnote'] or \
|
|
self.options['contactimg']
|
|
|
|
# either include the gender graphics or not?
|
|
self.graph = self.options['graph']
|
|
|
|
# whether to display children in birthorder or entry order?
|
|
self.birthorder = self.options['birthorder']
|
|
|
|
if self.use_home:
|
|
self.index_fname = "index"
|
|
self.surname_fname = "surnames"
|
|
self.intro_fname = "introduction"
|
|
elif self.use_intro:
|
|
self.index_fname = None
|
|
self.surname_fname = "surnames"
|
|
self.intro_fname = "index"
|
|
else:
|
|
self.index_fname = None
|
|
self.surname_fname = "index"
|
|
self.intro_fname = None
|
|
|
|
self.archive = None
|
|
self.cur_fname = None # Internal use. The name of the output file,
|
|
# to be used for the tar archive.
|
|
self.string_io = None
|
|
if self.use_archive:
|
|
self.html_dir = None
|
|
else:
|
|
self.html_dir = self.target_path
|
|
self.warn_dir = True # Only give warning once.
|
|
self.photo_list = {}
|
|
|
|
def write_report(self):
|
|
if not self.use_archive:
|
|
dir_name = self.target_path
|
|
if dir_name is 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
|
|
|
|
try:
|
|
image_dir_name = os.path.join(dir_name, 'images')
|
|
if not os.path.isdir(image_dir_name):
|
|
os.mkdir(image_dir_name)
|
|
|
|
image_dir_name = os.path.join(dir_name, 'thumb')
|
|
if not os.path.isdir(image_dir_name):
|
|
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
|
|
else:
|
|
if os.path.isdir(self.target_path):
|
|
ErrorDialog(_('Invalid file name'),
|
|
_('The archive file must be a file, not a directory'))
|
|
return
|
|
try:
|
|
self.archive = tarfile.open(self.target_path, "w:gz")
|
|
except (OSError, IOError), value:
|
|
ErrorDialog(_("Could not create %s") % self.target_path,
|
|
value)
|
|
return
|
|
|
|
self.progress = ProgressMeter(_("Narrated Web Site Report"), '')
|
|
|
|
# Build the person list
|
|
ind_list = self.build_person_list()
|
|
|
|
# copy all of the neccessary files
|
|
self.copy_narrated_files()
|
|
|
|
place_list = {}
|
|
source_list = {}
|
|
attribute_list = []
|
|
|
|
self.base_pages()
|
|
|
|
# build classes IndividualListPage and IndividualPage
|
|
self.person_pages(ind_list, place_list, source_list)
|
|
|
|
# build classes SurnameListPage and SurnamePage
|
|
self.surname_pages(ind_list)
|
|
|
|
# build PlaceListPage and PlacePage
|
|
self.place_pages(place_list, source_list)
|
|
|
|
# build classes EventListPage and EventPage
|
|
# build the events list only if event pages are being created?
|
|
if self.inc_events:
|
|
event_dict = []
|
|
self.build_events(ind_list, event_dict)
|
|
self.event_pages(event_dict)
|
|
|
|
# build classes SourceListPage and SourcePage
|
|
self.source_pages(source_list)
|
|
|
|
# build MediaListPage and MediaPage
|
|
if self.inc_gallery:
|
|
self.gallery_pages(source_list)
|
|
|
|
# Build source pages a second time to pick up sources referenced
|
|
# by galleries
|
|
self.source_pages(source_list)
|
|
|
|
# build classes RepositoryListPage and RepositoryPage
|
|
repolist = self.database.get_repository_handles()
|
|
if len(repolist):
|
|
self.repository_pages(repolist)
|
|
|
|
# if an archive is being used, close it?
|
|
if self.archive:
|
|
self.archive.close()
|
|
self.progress.close()
|
|
|
|
def build_person_list(self):
|
|
"""
|
|
Builds the person list. Gets all the handles from the database
|
|
and then applies the chosen filter:
|
|
"""
|
|
|
|
# gets the person list and applies the requested filter
|
|
|
|
ind_list = self.database.iter_person_handles()
|
|
self.progress.set_pass(_('Applying Filter...'), self.database.get_number_of_people())
|
|
ind_list = self.filter.apply(self.database, ind_list, self.progress)
|
|
|
|
return ind_list
|
|
|
|
def copy_narrated_files(self):
|
|
"""
|
|
Copy all of the CSS and image files for Narrated Web
|
|
"""
|
|
|
|
# copy behaviour stylesheet
|
|
fname = os.path.join(const.DATA_DIR, "behaviour.css")
|
|
self.copy_file(fname, "behaviour.css", "styles")
|
|
|
|
# copy screen stylesheet
|
|
if self.css:
|
|
fname = os.path.join(const.DATA_DIR, self.css)
|
|
self.copy_file(fname, _NARRATIVESCREEN, "styles")
|
|
|
|
# copy printer stylesheet
|
|
fname = os.path.join(const.DATA_DIR, "Web_Print-Default.css")
|
|
self.copy_file(fname, _NARRATIVEPRINT, "styles")
|
|
|
|
imgs = []
|
|
|
|
# Mainz stylesheet graphics
|
|
# will only be used if Mainz is slected as the stylesheet
|
|
Mainz_images = ["Web_Mainz_Bkgd.png", "Web_Mainz_Header.png",
|
|
"Web_Mainz_Mid.png", "Web_Mainz_MidLight.png"]
|
|
|
|
# Copy Mainz Style Images
|
|
if self.css == "Web_Mainz.css":
|
|
imgs += Mainz_images
|
|
|
|
# Copy the Creative Commons icon if the Creative Commons
|
|
# license is requested???
|
|
if 0 < self.copyright < len(_CC):
|
|
imgs += ["somerights20.gif"]
|
|
|
|
# include GRAMPS favicon
|
|
imgs += ["favicon.ico"]
|
|
|
|
# we need the blank image gif neede by behaviour.css
|
|
imgs += ["blank.gif"]
|
|
|
|
# copy Ancestor Tree graphics if needed???
|
|
if self.graph:
|
|
imgs += ["Web_Gender_Female.png",
|
|
"Web_Gender_FemaleFFF.png",
|
|
"Web_Gender_Male.png",
|
|
"Web_Gender_MaleFFF.png"]
|
|
|
|
for f in imgs:
|
|
from_path = os.path.join(const.IMAGE_DIR, f)
|
|
self.copy_file(from_path, f, "images")
|
|
|
|
def build_events(self, ind_list, event_dict):
|
|
"""
|
|
build a list of events for classes EventListPage and EventPage
|
|
|
|
@param: ind_list = list of handles for persons in this database
|
|
@param: event_dict = a list of events from ind_list
|
|
"""
|
|
db = self.database
|
|
|
|
for person_handle in ind_list:
|
|
person = db.get_person_from_handle(person_handle)
|
|
|
|
# begin events list for each person
|
|
event_list = []
|
|
|
|
# get sort name for sorting later
|
|
last_name = person.get_primary_name().get_surname()
|
|
first_name = person.get_primary_name().get_first_name()
|
|
sort_name = ', '.join([last_name, first_name])
|
|
|
|
partner = None
|
|
for family_handle in person.get_family_handle_list():
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
# get partner for use in Marriage and Divorce
|
|
partner_handle = ReportUtils.find_spouse(person, family)
|
|
if partner_handle:
|
|
partner = db.get_person_from_handle(partner_handle)
|
|
|
|
for evt_ref in family.get_event_ref_list():
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
|
|
# get event type
|
|
evt_type = get_event_type(event, evt_ref)
|
|
|
|
# get sot date as year/month/day, 2009/09/09,
|
|
# or 0000/00/00 for non-existing date
|
|
event_date = event.get_date_object()
|
|
year = event_date.get_year() or 0
|
|
month = event_date.get_month() or 0
|
|
day = event_date.get_day() or 0
|
|
sort_date = '%04d/%02d/%02d' % (year, month, day)
|
|
|
|
# add event data
|
|
event_list.append([evt_type, sort_date, sort_name, event,
|
|
evt_ref, partner])
|
|
|
|
partner = None
|
|
for evt_ref in person.get_primary_event_ref_list():
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
|
|
# get event type
|
|
evt_type = get_event_type(event, evt_ref)
|
|
|
|
# get sot date as year/month/day, see above for further info
|
|
event_date = event.get_date_object()
|
|
year = event_date.get_year() or 0
|
|
month = event_date.get_month() or 0
|
|
day = event_date.get_day() or 0
|
|
sort_date = '%04d/%02d/%02d' % (year, month, day)
|
|
|
|
# add event data
|
|
event_list.append([evt_type, sort_date, sort_name, event,
|
|
evt_ref, partner])
|
|
|
|
# sort the event_list
|
|
event_list.sort()
|
|
|
|
# combine person and their events together
|
|
event_dict.append([person, event_list])
|
|
|
|
# return the events for class EventListPage and EventPage
|
|
return event_dict
|
|
|
|
def build_attributes(self, person):
|
|
""" build a list of attributes for each person """
|
|
db = self.database
|
|
|
|
# get personal attributes
|
|
attribute_list = person.get_attribute_list()
|
|
|
|
for family_handle in person.get_family_handle_list():
|
|
family = db.get_family_from_handle(family_handle)
|
|
|
|
# get family attributes
|
|
attribute_list.extend(family.get_attribute_list() )
|
|
|
|
for evt_ref in family.get_event_ref_list():
|
|
event = db.get_event_from_handle(evt_ref.ref)
|
|
|
|
# get events attributes
|
|
attribute_list.extend(event.get_attribute_list() )
|
|
|
|
# attributes to its caller
|
|
return attribute_list
|
|
|
|
def person_pages(self, ind_list, place_list, source_list):
|
|
|
|
self.progress.set_pass(_('Creating individual pages'), len(ind_list) + 1)
|
|
self.progress.step() # otherwise the progress indicator sits at 100%
|
|
# for a short while from the last step we did,
|
|
# which was to apply the privacy filter
|
|
|
|
IndividualListPage(self, self.title, ind_list)
|
|
|
|
for person_handle in ind_list:
|
|
self.progress.step()
|
|
person = self.database.get_person_from_handle(person_handle)
|
|
|
|
# get attributes for each person
|
|
attribute_list = self.build_attributes(person)
|
|
|
|
IndividualPage(self, self.title, person, ind_list, place_list, source_list,
|
|
attribute_list)
|
|
|
|
def surname_pages(self, ind_list):
|
|
"""
|
|
Generates the surname related pages from list of individual
|
|
people.
|
|
"""
|
|
|
|
local_list = sort_people(self.database, ind_list)
|
|
|
|
self.progress.set_pass(_("Creating surname pages"), len(local_list))
|
|
|
|
SurnameListPage(self, self.title, ind_list, SurnameListPage.ORDER_BY_NAME,
|
|
self.surname_fname)
|
|
|
|
SurnameListPage(self, self.title, ind_list, SurnameListPage.ORDER_BY_COUNT,
|
|
"surnames_count")
|
|
|
|
for (surname, handle_list) in local_list:
|
|
SurnamePage(self, self.title, surname, handle_list, ind_list)
|
|
self.progress.step()
|
|
|
|
def source_pages(self, source_list):
|
|
|
|
self.progress.set_pass(_("Creating source pages"), len(source_list))
|
|
|
|
SourceListPage(self, self.title, source_list.keys())
|
|
|
|
for key in source_list:
|
|
SourcePage(self, self.title, key, source_list)
|
|
self.progress.step()
|
|
|
|
def place_pages(self, place_list, source_list):
|
|
|
|
self.progress.set_pass(_("Creating place pages"), len(place_list))
|
|
|
|
PlaceListPage(self, self.title, place_list, source_list)
|
|
|
|
for place in place_list:
|
|
PlacePage(self, self.title, place, source_list, place_list)
|
|
self.progress.step()
|
|
|
|
def event_pages(self, event_dict):
|
|
"""
|
|
a dump of all the events sorted by event type, date, and surname
|
|
for classes EventListPage and EventPage
|
|
"""
|
|
self.progress.set_pass(_('Creating event pages'), len(event_dict))
|
|
|
|
# send all data to the events list page
|
|
EventListPage(self, self.title, event_dict)
|
|
|
|
for (person, event_list) in event_dict:
|
|
self.progress.step()
|
|
|
|
for (evt_type, sort_date, sort_name, event, evt_ref, partner) in event_list:
|
|
|
|
# create individual event page
|
|
EventPage(self, self.title, person, partner, evt_type, event, evt_ref)
|
|
|
|
def gallery_pages(self, source_list):
|
|
import gc
|
|
|
|
self.progress.set_pass(_("Creating media pages"), len(self.photo_list))
|
|
|
|
MediaListPage(self, self.title)
|
|
|
|
prev = None
|
|
total = len(self.photo_list)
|
|
sort = Sort.Sort(self.database)
|
|
photo_keys = sorted(self.photo_list, key=sort.by_media_title_key)
|
|
|
|
index = 1
|
|
for photo_handle in photo_keys:
|
|
gc.collect() # Reduce memory usage when there are many images.
|
|
if index == total:
|
|
next = None
|
|
else:
|
|
next = photo_keys[index]
|
|
# Notice. Here self.photo_list[photo_handle] is used not self.photo_list
|
|
MediaPage(self, self.title, photo_handle, source_list, self.photo_list[photo_handle],
|
|
(prev, next, index, total))
|
|
self.progress.step()
|
|
prev = photo_handle
|
|
index += 1
|
|
|
|
def base_pages(self):
|
|
|
|
if self.use_home:
|
|
HomePage(self, self.title)
|
|
|
|
if self.inc_contact:
|
|
ContactPage(self, self.title)
|
|
|
|
if self.inc_download:
|
|
DownloadPage(self, self.title)
|
|
|
|
if self.use_intro:
|
|
IntroductionPage(self, self.title)
|
|
|
|
def repository_pages(self, repolist):
|
|
"""
|
|
will create RepositoryPage() and RepositoryListPage()
|
|
"""
|
|
|
|
db = self.database
|
|
repos_dict = {}
|
|
|
|
# Sort the repositories
|
|
for handle in repolist:
|
|
repo = db.get_repository_from_handle(handle)
|
|
key = repo.name + str(repo.get_gramps_id())
|
|
repos_dict[key] = (repo, handle)
|
|
|
|
keys = sorted(repos_dict, key=locale.strxfrm)
|
|
|
|
# set progress bar pass for Repositories
|
|
self.progress.set_pass(_('Creating repository pages'), len(repos_dict))
|
|
|
|
# RepositoryListPage Class
|
|
RepositoryListPage(self, self.title, repos_dict, keys)
|
|
|
|
index = 0
|
|
for index, key in enumerate(keys):
|
|
(repo, handle) = repos_dict[key]
|
|
|
|
# RepositoryPage Class
|
|
RepositoryPage(self, self.title, repo, handle)
|
|
|
|
self.progress.step()
|
|
|
|
def add_image(self, option_name, height=0):
|
|
pic_id = self.options[option_name]
|
|
if pic_id:
|
|
obj = self.database.get_object_from_gramps_id(pic_id)
|
|
mime_type = obj.get_mime_type()
|
|
if mime_type and mime_type.startswith("image"):
|
|
try:
|
|
newpath, thumb_path = self.prepare_copy_media(obj)
|
|
self.copy_file(Utils.media_path_full(self.database, obj.get_path()),
|
|
newpath)
|
|
|
|
# begin image
|
|
image = Html('img')
|
|
img_attr = ''
|
|
if height:
|
|
img_attr += ' height="%d" ' % height
|
|
img_attr += ' src="%s" alt="%s" ' % (newpath, obj.get_description())
|
|
|
|
# add image attributes to image
|
|
image.attr = img_attr
|
|
|
|
# return an image
|
|
return image
|
|
|
|
except (IOError, OSError), msg:
|
|
WarningDialog(_("Could not add photo to page"), str(msg))
|
|
|
|
# no image to return
|
|
return None
|
|
|
|
def build_subdirs(self, subdir, fname, up=False):
|
|
"""
|
|
If subdir is given, then two extra levels of subdirectory are inserted
|
|
between 'subdir' and the filename. The reason is to prevent directories with
|
|
too many entries.
|
|
|
|
For example, this may return "8/1/aec934857df74d36618"
|
|
"""
|
|
subdirs = []
|
|
if subdir:
|
|
subdirs.append(subdir)
|
|
subdirs.append(fname[-1].lower())
|
|
subdirs.append(fname[-2].lower())
|
|
if up:
|
|
subdirs = ['..']*3 + subdirs
|
|
return subdirs
|
|
|
|
|
|
def build_path(self, subdir, fname, up=False):
|
|
"""
|
|
Return the name of the subdirectory.
|
|
|
|
Notice that we DO use os.path.join() here.
|
|
"""
|
|
return os.path.join(*self.build_subdirs(subdir, fname, up))
|
|
|
|
def build_url_image(self, fname, subdir=None, up=False):
|
|
subdirs = []
|
|
if subdir:
|
|
subdirs.append(subdir)
|
|
if up:
|
|
subdirs = ['..']*3 + subdirs
|
|
nname = '/'.join(subdirs + [fname])
|
|
if ( Utils.win ):
|
|
nname = nname.replace('\\','/')
|
|
return nname
|
|
|
|
def build_url_fname_html(self, fname, subdir=None, up=False):
|
|
return self.build_url_fname(fname, subdir, up) + self.ext
|
|
|
|
def build_url_fname(self, fname, subdir=None, up=False):
|
|
"""
|
|
Create part of the URL given the filename and optionally the subdirectory.
|
|
If the subdirectory is given, then two extra levels of subdirectory are inserted
|
|
between 'subdir' and the filename. The reason is to prevent directories with
|
|
too many entries.
|
|
If 'up' is True, then "../../../" is inserted in front of the result.
|
|
|
|
The extension is added to the filename as well.
|
|
|
|
Notice that we do NOT use os.path.join() because we're creating a URL.
|
|
Imagine we run gramps on Windows (heaven forbits), we don't want to
|
|
see backslashes in the URL.
|
|
"""
|
|
if ( Utils.win ):
|
|
fname = fname.replace('\\','/')
|
|
subdirs = self.build_subdirs(subdir, fname, up)
|
|
return '/'.join(subdirs + [fname])
|
|
|
|
def create_file(self, fname, subdir=None):
|
|
if subdir:
|
|
subdir = self.build_path(subdir, fname)
|
|
self.cur_fname = os.path.join(subdir, fname) + self.ext
|
|
else:
|
|
self.cur_fname = fname + self.ext
|
|
if self.archive:
|
|
self.string_io = StringIO()
|
|
of = codecs.EncodedFile(self.string_io, 'utf-8',
|
|
self.encoding, 'xmlcharrefreplace')
|
|
else:
|
|
if subdir:
|
|
subdir = os.path.join(self.html_dir, subdir)
|
|
if not os.path.isdir(subdir):
|
|
os.makedirs(subdir)
|
|
fname = os.path.join(self.html_dir, self.cur_fname)
|
|
of = codecs.EncodedFile(open(fname, "w"), 'utf-8',
|
|
self.encoding, 'xmlcharrefreplace')
|
|
return of
|
|
|
|
def close_file(self, of):
|
|
if self.archive:
|
|
tarinfo = tarfile.TarInfo(self.cur_fname)
|
|
tarinfo.size = len(self.string_io.getvalue())
|
|
tarinfo.mtime = time.time()
|
|
if os.sys.platform != "win32":
|
|
tarinfo.uid = os.getuid()
|
|
tarinfo.gid = os.getgid()
|
|
self.string_io.seek(0)
|
|
self.archive.addfile(tarinfo, self.string_io)
|
|
self.string_io = None
|
|
of.close()
|
|
else:
|
|
of.close()
|
|
self.cur_fname = None
|
|
|
|
def add_lnkref_to_photo(self, photo, lnkref):
|
|
handle = photo.get_handle()
|
|
# FIXME. Is it OK to add to the photo_list of report?
|
|
photo_list = self.photo_list
|
|
if handle in photo_list:
|
|
if lnkref not in photo_list[handle]:
|
|
photo_list[handle].append(lnkref)
|
|
else:
|
|
photo_list[handle] = [lnkref]
|
|
|
|
def prepare_copy_media(self, photo):
|
|
handle = photo.get_handle()
|
|
ext = os.path.splitext(photo.get_path())[1]
|
|
real_path = os.path.join(self.build_path('images', handle), handle + ext)
|
|
thumb_path = os.path.join(self.build_path('thumb', handle), handle + '.png')
|
|
return real_path, thumb_path
|
|
|
|
def copy_file(self, from_fname, to_fname, to_dir=''):
|
|
"""
|
|
Copy a file from a source to a (report) destination.
|
|
If to_dir is not present and if the target is not an archive,
|
|
then the destination directory will be created.
|
|
|
|
Normally 'to_fname' will be just a filename, without directory path.
|
|
|
|
'to_dir' is the relative path name in the destination root. It will
|
|
be prepended before 'to_fname'.
|
|
"""
|
|
if self.archive:
|
|
dest = os.path.join(to_dir, to_fname)
|
|
self.archive.add(from_fname, dest)
|
|
else:
|
|
dest = os.path.join(self.html_dir, to_dir, to_fname)
|
|
|
|
destdir = os.path.dirname(dest)
|
|
if not os.path.isdir(destdir):
|
|
os.makedirs(destdir)
|
|
|
|
if from_fname != dest:
|
|
shutil.copyfile(from_fname, dest)
|
|
elif self.warn_dir:
|
|
WarningDialog(
|
|
_("Possible destination error") + "\n" +
|
|
_("You appear to have set your target directory "
|
|
"to a directory used for data storage. This "
|
|
"could create problems with file management. "
|
|
"It is recommended that you consider using "
|
|
"a different directory to store your generated "
|
|
"web pages."))
|
|
self.warn_dir = False
|
|
|
|
class NavWebOptions(MenuReportOptions):
|
|
"""
|
|
Defines options and provides handling interface.
|
|
"""
|
|
|
|
def __init__(self, name, dbase):
|
|
self.__db = dbase
|
|
self.__archive = None
|
|
self.__target = None
|
|
self.__pid = None
|
|
self.__filter = None
|
|
self.__graph = None
|
|
self.__graphgens = None
|
|
self.__living = None
|
|
self.__yearsafterdeath = None
|
|
MenuReportOptions.__init__(self, name, dbase)
|
|
|
|
def add_menu_options(self, menu):
|
|
"""
|
|
Add options to the menu for the web site.
|
|
"""
|
|
self.__add_report_options(menu)
|
|
self.__add_page_generation_options(menu)
|
|
self.__add_privacy_options(menu)
|
|
self.__add_download_options(menu)
|
|
self.__add_advanced_options(menu)
|
|
|
|
def __add_report_options(self, menu):
|
|
"""
|
|
Options on the "Report Options" tab.
|
|
"""
|
|
category_name = _("Report Options")
|
|
|
|
self.__archive = BooleanOption(_('Store web pages in .tar.gz archive'),
|
|
False)
|
|
self.__archive.set_help(_('Whether to store the web pages in an '
|
|
'archive file'))
|
|
menu.add_option(category_name, 'archive', self.__archive)
|
|
self.__archive.connect('value-changed', self.__archive_changed)
|
|
|
|
self.__target = DestinationOption(_("Destination"),
|
|
os.path.join(const.USER_HOME, "NAVWEB"))
|
|
self.__target.set_help( _("The destination directory for the web "
|
|
"files"))
|
|
menu.add_option(category_name, "target", self.__target)
|
|
|
|
self.__archive_changed()
|
|
|
|
title = StringOption(_("Web site title"), _('My Family Tree'))
|
|
title.set_help(_("The title of the web site"))
|
|
menu.add_option(category_name, "title", title)
|
|
|
|
self.__filter = FilterOption(_("Filter"), 0)
|
|
self.__filter.set_help(
|
|
_("Select filter to restrict people that appear on web site"))
|
|
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()
|
|
|
|
# We must figure out the value of the first option before we can
|
|
# create the EnumeratedListOption
|
|
fmt_list = _nd.get_name_format()
|
|
name_format = EnumeratedListOption(_("Name format"), fmt_list[0][0])
|
|
for num, name, fmt_str, act in fmt_list:
|
|
name_format.add_item(num, name)
|
|
name_format.set_help(_("Select the format to display names"))
|
|
menu.add_option(category_name, "name_format", name_format)
|
|
|
|
ext = EnumeratedListOption(_("File extension"), ".html" )
|
|
for etype in _WEB_EXT:
|
|
ext.add_item(etype, etype)
|
|
ext.set_help( _("The extension to be used for the web files"))
|
|
menu.add_option(category_name, "ext", ext)
|
|
|
|
cright = EnumeratedListOption(_('Copyright'), 0 )
|
|
for index, copt in enumerate(_COPY_OPTIONS):
|
|
cright.add_item(index, copt)
|
|
cright.set_help( _("The copyright to be used for the web files"))
|
|
menu.add_option(category_name, "cright", cright)
|
|
|
|
css = EnumeratedListOption(_('StyleSheet'), CSS_FILES[0][1])
|
|
for style in CSS_FILES:
|
|
css.add_item(style[1], style[0])
|
|
css.set_help( _('The stylesheet to be used for the web page'))
|
|
menu.add_option(category_name, "css", css)
|
|
|
|
self.__graph = BooleanOption(_("Include ancestor graph"), True)
|
|
self.__graph.set_help(_('Whether to include an ancestor graph '
|
|
'on each individual page'))
|
|
menu.add_option(category_name, 'graph', self.__graph)
|
|
self.__graph.connect('value-changed', self.__graph_changed)
|
|
|
|
self.__graphgens = EnumeratedListOption(_('Graph generations'), 4)
|
|
self.__graphgens.add_item(2, "2")
|
|
self.__graphgens.add_item(3, "3")
|
|
self.__graphgens.add_item(4, "4")
|
|
self.__graphgens.add_item(5, "5")
|
|
self.__graphgens.set_help( _("The number of generations to include in "
|
|
"the ancestor graph"))
|
|
menu.add_option(category_name, "graphgens", self.__graphgens)
|
|
|
|
self.__graph_changed()
|
|
|
|
def __add_page_generation_options(self, menu):
|
|
"""
|
|
Options on the "Page Generation" tab.
|
|
"""
|
|
category_name = _("Page Generation")
|
|
|
|
homenote = NoteOption(_('Home page note'))
|
|
homenote.set_help( _("A note to be used on the home page"))
|
|
menu.add_option(category_name, "homenote", homenote)
|
|
|
|
homeimg = MediaOption(_('Home page image'))
|
|
homeimg.set_help( _("An image to be used on the home page"))
|
|
menu.add_option(category_name, "homeimg", homeimg)
|
|
|
|
intronote = NoteOption(_('Introduction note'))
|
|
intronote.set_help( _("A note to be used as the introduction"))
|
|
menu.add_option(category_name, "intronote", intronote)
|
|
|
|
introimg = MediaOption(_('Introduction image'))
|
|
introimg.set_help( _("An image to be used as the introduction"))
|
|
menu.add_option(category_name, "introimg", introimg)
|
|
|
|
contactnote = NoteOption(_("Publisher contact note"))
|
|
contactnote.set_help( _("A note to be used as the publisher contact"))
|
|
menu.add_option(category_name, "contactnote", contactnote)
|
|
|
|
contactimg = MediaOption(_("Publisher contact image"))
|
|
contactimg.set_help( _("An image to be used as the publisher contact"))
|
|
menu.add_option(category_name, "contactimg", contactimg)
|
|
|
|
headernote = NoteOption(_('HTML user header'))
|
|
headernote.set_help( _("A note to be used as the page header"))
|
|
menu.add_option(category_name, "headernote", headernote)
|
|
|
|
footernote = NoteOption(_('HTML user footer'))
|
|
footernote.set_help( _("A note to be used as the page footer"))
|
|
menu.add_option(category_name, "footernote", footernote)
|
|
|
|
self.__gallery = BooleanOption(_("Include images and media objects"), True)
|
|
self.__gallery.set_help(_('Whether to include a gallery of media objects'))
|
|
menu.add_option(category_name, 'gallery', self.__gallery)
|
|
self.__gallery.connect('value-changed', self.__gallery_changed)
|
|
|
|
self.__maxinitialimagewidth = NumberOption(_("Max width of initial image"),
|
|
_DEFAULT_MAX_IMG_WIDTH, 0, 2000)
|
|
self.__maxinitialimagewidth.set_help(_("This allows you to set the maximum width "
|
|
"of the image shown on the media page. Set to 0 for no limit."))
|
|
menu.add_option(category_name, 'maxinitialimagewidth', self.__maxinitialimagewidth)
|
|
|
|
self.__maxinitialimageheight = NumberOption(_("Max height of initial image"),
|
|
_DEFAULT_MAX_IMG_HEIGHT, 0, 2000)
|
|
self.__maxinitialimageheight.set_help(_("This allows you to set the maximum height "
|
|
"of the image shown on the media page. Set to 0 for no limit."))
|
|
menu.add_option(category_name, 'maxinitialimageheight', self.__maxinitialimageheight)
|
|
|
|
self.__gallery_changed()
|
|
|
|
nogid = BooleanOption(_('Suppress GRAMPS ID'), False)
|
|
nogid.set_help(_('Whether to include the Gramps ID of objects'))
|
|
menu.add_option(category_name, 'nogid', nogid)
|
|
|
|
def __add_privacy_options(self, menu):
|
|
"""
|
|
Options on the "Privacy" tab.
|
|
"""
|
|
category_name = _("Privacy")
|
|
|
|
incpriv = BooleanOption(_("Include records marked private"), False)
|
|
incpriv.set_help(_('Whether to include private objects'))
|
|
menu.add_option(category_name, 'incpriv', incpriv)
|
|
|
|
self.__living = EnumeratedListOption(_("Living People"),
|
|
_INCLUDE_LIVING_VALUE )
|
|
self.__living.add_item(LivingProxyDb.MODE_EXCLUDE_ALL,
|
|
_("Exclude"))
|
|
self.__living.add_item(LivingProxyDb.MODE_INCLUDE_LAST_NAME_ONLY,
|
|
_("Include Last Name Only"))
|
|
self.__living.add_item(LivingProxyDb.MODE_INCLUDE_FULL_NAME_ONLY,
|
|
_("Include Full Name Only"))
|
|
self.__living.add_item(_INCLUDE_LIVING_VALUE,
|
|
_("Include"))
|
|
self.__living.set_help(_("How to handle living people"))
|
|
menu.add_option(category_name, "living", self.__living)
|
|
self.__living.connect('value-changed', self.__living_changed)
|
|
|
|
self.__yearsafterdeath = NumberOption(_("Years from death to consider "
|
|
"living"), 30, 0, 100)
|
|
self.__yearsafterdeath.set_help(_("This allows you to restrict "
|
|
"information on people who have not "
|
|
"been dead for very long"))
|
|
menu.add_option(category_name, 'yearsafterdeath',
|
|
self.__yearsafterdeath)
|
|
|
|
self.__living_changed()
|
|
|
|
def __add_download_options(self, menu):
|
|
"""
|
|
Options for the download tab ...
|
|
"""
|
|
|
|
category_name = _("Download")
|
|
|
|
self.__incdownload = BooleanOption(_("Include download page"), False)
|
|
self.__incdownload.set_help(_('Whether to include a database download option'))
|
|
menu.add_option(category_name, 'incdownload', self.__incdownload)
|
|
self.__incdownload.connect('value-changed', self.__download_changed)
|
|
|
|
self.__downloadnote = NoteOption(_('Download page note'))
|
|
self.__downloadnote.set_help( _("A note to be used on the download page"))
|
|
menu.add_option(category_name, "downloadnote", self.__downloadnote)
|
|
|
|
self.__down_fname1 = DestinationOption(_("Download Filename"),
|
|
os.path.join(const.USER_HOME, ""))
|
|
self.__down_fname1.set_help(_("File to be used for downloading of database"))
|
|
menu.add_option(category_name, "down_fname1", self.__down_fname1)
|
|
|
|
self.__dl_descr1 = StringOption(_("Description for download"), _('Smith Family Tree'))
|
|
self.__dl_descr1.set_help(_('Give a description for this file.'))
|
|
menu.add_option(category_name, 'dl_descr1', self.__dl_descr1)
|
|
|
|
self.__down_fname2 = DestinationOption(_("Download Filename"),
|
|
os.path.join(const.USER_HOME, ""))
|
|
self.__down_fname2.set_help(_("File to be used for downloading of database"))
|
|
menu.add_option(category_name, "down_fname2", self.__down_fname2)
|
|
|
|
self.__dl_descr2 = StringOption(_("Description for download"), _('Johnson Family Tree'))
|
|
self.__dl_descr2.set_help(_('Give a description for this file.'))
|
|
menu.add_option(category_name, 'dl_descr2', self.__dl_descr2)
|
|
|
|
self.__dl_cright = EnumeratedListOption(_('Download Copyright License'), 0 )
|
|
for index, copt in enumerate(_COPY_OPTIONS):
|
|
self.__dl_cright.add_item(index, copt)
|
|
self.__dl_cright.set_help( _("The copyright to be used for ths download file?"))
|
|
menu.add_option(category_name, "dl_cright", self.__dl_cright)
|
|
|
|
self.__download_changed()
|
|
|
|
def __add_advanced_options(self, menu):
|
|
"""
|
|
Options on the "Advanced" tab.
|
|
"""
|
|
category_name = _("Advanced")
|
|
|
|
encoding = EnumeratedListOption(_('Character set encoding'), _CHARACTER_SETS[0][1] )
|
|
for eopt in _CHARACTER_SETS:
|
|
encoding.add_item(eopt[1], eopt[0])
|
|
encoding.set_help( _("The encoding to be used for the web files"))
|
|
menu.add_option(category_name, "encoding", encoding)
|
|
|
|
linkhome = BooleanOption(_('Include link to home person on every '
|
|
'page'), False)
|
|
linkhome.set_help(_('Whether to include a link to the home person'))
|
|
menu.add_option(category_name, 'linkhome', linkhome)
|
|
|
|
showbirth = BooleanOption(_("Include a column for birth dates on the "
|
|
"index pages"), True)
|
|
showbirth.set_help(_('Whether to include a birth column'))
|
|
menu.add_option(category_name, 'showbirth', showbirth)
|
|
|
|
showdeath = BooleanOption(_("Include a column for death dates on the "
|
|
"index pages"), False)
|
|
showdeath.set_help(_('Whether to include a death column'))
|
|
menu.add_option(category_name, 'showdeath', showdeath)
|
|
|
|
showpartner = BooleanOption(_("Include a column for partners on the "
|
|
"index pages"), False)
|
|
showpartner.set_help(_('Whether to include a partners column'))
|
|
menu.add_option(category_name, 'showpartner', showpartner)
|
|
|
|
showparents = BooleanOption(_("Include a column for parents on the "
|
|
"index pages"), False)
|
|
showparents.set_help(_('Whether to include a parents column'))
|
|
menu.add_option(category_name, 'showparents', showparents)
|
|
|
|
showallsiblings = BooleanOption(_("Include half and/ or "
|
|
"step-siblings on the individual pages"), False)
|
|
showallsiblings.set_help(_( "Whether to include half and/ or "
|
|
"step-siblings with the parents and siblings"))
|
|
menu.add_option(category_name, 'showhalfsiblings', showallsiblings)
|
|
|
|
birthorder = BooleanOption(_('Sort all children in birth order'), False)
|
|
birthorder.set_help(_('Whether to display children in birth order'
|
|
' or in entry order?'))
|
|
menu.add_option(category_name, 'birthorder', birthorder)
|
|
|
|
inc_events = BooleanOption(_('Include event pages'), False)
|
|
inc_events.set_help(_('Add a complete events list and relevant pages or not'))
|
|
menu.add_option(category_name, 'inc_events', inc_events)
|
|
|
|
inc_repository = BooleanOption(_('Include repository pages'), False)
|
|
inc_repository.set_help(_('Whether to include the Repository Pages or not?'))
|
|
menu.add_option(category_name, 'inc_repository', inc_repository)
|
|
|
|
def __archive_changed(self):
|
|
"""
|
|
Update the change of storage: archive or directory
|
|
"""
|
|
if self.__archive.get_value() == True:
|
|
self.__target.set_extension(".tar.gz")
|
|
self.__target.set_directory_entry(False)
|
|
else:
|
|
self.__target.set_directory_entry(True)
|
|
|
|
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 __graph_changed(self):
|
|
"""
|
|
Handle enabling or disabling the ancestor graph
|
|
"""
|
|
self.__graphgens.set_available(self.__graph.get_value())
|
|
|
|
def __gallery_changed(self):
|
|
"""
|
|
Handles the changing nature of gallery
|
|
"""
|
|
|
|
if self.__gallery.get_value() == False:
|
|
self.__maxinitialimagewidth.set_available(False)
|
|
self.__maxinitialimageheight.set_available(False)
|
|
else:
|
|
self.__maxinitialimagewidth.set_available(True)
|
|
self.__maxinitialimageheight.set_available(True)
|
|
|
|
def __living_changed(self):
|
|
"""
|
|
Handle a change in the living option
|
|
"""
|
|
if self.__living.get_value() == _INCLUDE_LIVING_VALUE:
|
|
self.__yearsafterdeath.set_available(False)
|
|
else:
|
|
self.__yearsafterdeath.set_available(True)
|
|
|
|
def __download_changed(self):
|
|
"""
|
|
Handles the changing nature of include download page
|
|
"""
|
|
|
|
if self.__incdownload.get_value():
|
|
self.__downloadnote.set_available(True)
|
|
self.__down_fname1.set_available(True)
|
|
self.__dl_descr1.set_available(True)
|
|
self.__down_fname2.set_available(True)
|
|
self.__dl_descr2.set_available(True)
|
|
self.__dl_cright.set_available(True)
|
|
else:
|
|
self.__downloadnote.set_available(False)
|
|
self.__down_fname1.set_available(False)
|
|
self.__dl_descr1.set_available(False)
|
|
self.__down_fname2.set_available(False)
|
|
self.__dl_descr2.set_available(False)
|
|
self.__dl_cright.set_available(False)
|
|
|
|
# FIXME. Why do we need our own sorting? Why not use Sort.Sort?
|
|
def sort_people(db, handle_list):
|
|
sname_sub = {}
|
|
sortnames = {}
|
|
|
|
for person_handle in handle_list:
|
|
person = db.get_person_from_handle(person_handle)
|
|
primary_name = person.get_primary_name()
|
|
|
|
if primary_name.group_as:
|
|
surname = primary_name.group_as
|
|
else:
|
|
surname = db.get_name_group_mapping(primary_name.surname)
|
|
|
|
sortnames[person_handle] = _nd.sort_string(primary_name)
|
|
|
|
if surname in sname_sub:
|
|
sname_sub[surname].append(person_handle)
|
|
else:
|
|
sname_sub[surname] = [person_handle]
|
|
|
|
sorted_lists = []
|
|
temp_list = sorted(sname_sub, key=locale.strxfrm)
|
|
|
|
for name in temp_list:
|
|
slist = sorted(((sortnames[x], x) for x in sname_sub[name]),
|
|
key=lambda x:locale.strxfrm(x[0]))
|
|
entries = [x[1] for x in slist]
|
|
sorted_lists.append((name, entries))
|
|
|
|
return sorted_lists
|
|
|
|
# Modified _get_regular_surname from WebCal.py to get prefix, first name, and suffix
|
|
def _get_short_name(gender, name):
|
|
""" Will get prefix and suffix for all people passed through it """
|
|
|
|
short_name = name.get_first_name()
|
|
prefix = name.get_surname_prefix()
|
|
if prefix:
|
|
short_name = prefix + " " + short_name
|
|
if gender == Person.FEMALE:
|
|
return short_name
|
|
else:
|
|
suffix = name.get_suffix()
|
|
if suffix:
|
|
short_name = short_name + ", " + suffix
|
|
return short_name
|
|
|
|
def get_person_keyname(db, handle):
|
|
""" .... """
|
|
person = db.get_person_from_handle(handle)
|
|
return person.get_primary_name().surname
|
|
|
|
def get_place_keyname(db, handle):
|
|
""" ... """
|
|
|
|
return ReportUtils.place_name(db, handle)
|
|
|
|
def get_first_letters(db, handle_list, key):
|
|
""" key is _PLACE or _PERSON ...."""
|
|
|
|
first_letters = []
|
|
|
|
for handle in handle_list:
|
|
if key == _PERSON:
|
|
keyname = get_person_keyname(db, handle)
|
|
else:
|
|
keyname = get_place_keyname(db, handle)
|
|
|
|
if keyname:
|
|
c = normalize('NFKC', keyname)[0].upper()
|
|
# See : http://www.gramps-project.org/bugs/view.php?id=2933
|
|
(lang_country, modifier ) = locale.getlocale()
|
|
if lang_country == "sv_SE" and ( c == u'W' or c == u'V' ):
|
|
first_letters.append(u'V')
|
|
else:
|
|
first_letters.append(c)
|
|
|
|
return first_letters
|
|
|
|
def _has_webpage_extension(url):
|
|
"""
|
|
determine if a filename has an extension or not...
|
|
|
|
url = filename to be checked
|
|
"""
|
|
|
|
for ext in _WEB_EXT:
|
|
if url.endswith(ext):
|
|
return True
|
|
return False
|
|
|
|
def alphabet_navigation(db, handle_list, key):
|
|
"""
|
|
Will create the alphabet navigation bar for classes IndividualListPage,
|
|
SurnameListPage, and PlaceListPage
|
|
|
|
handle_list -- a list of people's or Places' handles
|
|
key -- _PERSON or _PLACE
|
|
"""
|
|
|
|
sorted_set = {}
|
|
|
|
# The comment below from the glibc locale sv_SE in
|
|
# localedata/locales/sv_SE :
|
|
#
|
|
# % The letter w is normally not present in the Swedish alphabet. It
|
|
# % exists in some names in Swedish and foreign words, but is accounted
|
|
# % for as a variant of 'v'. Words and names with 'w' are in Swedish
|
|
# % ordered alphabetically among the words and names with 'v'. If two
|
|
# % words or names are only to be distinguished by 'v' or % 'w', 'v' is
|
|
# % placed before 'w'.
|
|
#
|
|
# See : http://www.gramps-project.org/bugs/view.php?id=2933
|
|
#
|
|
(lang_country, modifier ) = locale.getlocale()
|
|
for ltr in get_first_letters(db, handle_list, key):
|
|
if ltr in sorted_set:
|
|
sorted_set[ltr] += 1
|
|
else:
|
|
sorted_set[ltr] = 1
|
|
|
|
# remove the number of each occurance of each letter
|
|
sorted_alpha_index = sorted((l for l in sorted_set if l != ','),
|
|
key=locale.strxfrm)
|
|
|
|
# if no letters, return None back to its callers
|
|
if not sorted_alpha_index:
|
|
return None
|
|
|
|
# begin alphabet division
|
|
with Html("div", id='alphabet') as section:
|
|
|
|
num_ltrs = len(sorted_alpha_index)
|
|
nrows = (num_ltrs / 35) + 1
|
|
index = 0
|
|
for rows in xrange(nrows):
|
|
unordered = Html('ul')
|
|
section += unordered
|
|
cols = 0
|
|
while (cols <= 35 and index < num_ltrs):
|
|
ltr = sorted_alpha_index[index]
|
|
title_str = _('Surnames') if key == 0 else _('Places')
|
|
if lang_country == "sv_SE" and ltr == u'V':
|
|
title_str += _(' starting with %s') % "V,W"
|
|
unordered += (Html('li', class_ = "letters", inline = True) +
|
|
Html('a', "V,W", href="#V,W", title=title_str)
|
|
)
|
|
else:
|
|
title_str += _(' starting with %s') % ltr
|
|
unordered += Html('li', class_ = "letters", inline = True) + (
|
|
Html('a', ltr, href='#%s' % ltr, title=title_str)
|
|
)
|
|
cols += 1
|
|
index += 1
|
|
|
|
# return alphabet navigation to its callers
|
|
return section
|
|
|
|
def add_birthdate(db, childlist):
|
|
"""
|
|
This will sort a list of child handles in birth order
|
|
"""
|
|
|
|
sorted_children = []
|
|
for child_handle in childlist:
|
|
child = db.get_person_from_handle(child_handle)
|
|
|
|
birth_date = None
|
|
birth_ref = child.get_birth_ref()
|
|
if birth_ref:
|
|
birth_event = db.get_event_from_handle(birth_ref.ref)
|
|
birth_date = birth_event.get_date_object()
|
|
if birth_date is not None:
|
|
year = birth_date.get_year()
|
|
if not year:
|
|
year = 2199
|
|
month = birth_date.get_month()
|
|
if not month:
|
|
month = 12
|
|
day = birth_date.get_day()
|
|
if not day:
|
|
day = 31
|
|
else:
|
|
year, month, day = 2199, 12, 31
|
|
|
|
# get birth date, if there is None, then give fake one...
|
|
birth_date = Date(year, month, day)
|
|
sorted_children.append((birth_date, child_handle))
|
|
|
|
# return the list of child handles and their birthdates
|
|
return sorted_children
|
|
|
|
def get_event_type(event, event_ref):
|
|
""" return the type of an event """
|
|
|
|
evt_name = str(event.get_type())
|
|
|
|
if event_ref.get_role() == EventRoleType.PRIMARY:
|
|
evt_type = u"%(evt_name)s" % locals()
|
|
else:
|
|
evt_role = event_ref.get_role()
|
|
evt_type = u"%(evt_name)s (%(evt_role)s)" % locals()
|
|
|
|
# return event type to its callers
|
|
return evt_type
|
|
|
|
# -------------------------------------------
|
|
#
|
|
# Register Plugin
|
|
#
|
|
# -------------------------------------------
|
|
pmgr = PluginManager.get_instance()
|
|
pmgr.register_report(
|
|
name = 'navwebpage',
|
|
category = CATEGORY_WEB,
|
|
report_class = NavWebReport,
|
|
options_class = NavWebOptions,
|
|
modes = PluginManager.REPORT_MODE_GUI | PluginManager.REPORT_MODE_CLI,
|
|
translated_name = _("Narrated Web Site"),
|
|
status = _("Stable"),
|
|
author_name = "Donald N. Allingham",
|
|
author_email = "don@gramps-project.org",
|
|
description = _("Produces web (HTML) pages for individuals, or a set of "
|
|
"individuals"),
|
|
)
|