diff --git a/ChangeLog b/ChangeLog index 55016b45f..de2c34e23 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,28 @@ +2008-01-22 Brian Matherly + * src/plugins/KinshipReport.py: + * src/plugins/DetDescendantReport.py: + * src/plugins/DescendReport.py: + * src/plugins/IndivComplete.py: + * src/plugins/CalculateEstimatedDates.py: + * src/plugins/BookReport.py: + * src/plugins/TimeLine.py: + * src/plugins/Calendar.py: + * src/plugins/AncestorReport.py: + * src/plugins/MarkerReport.py: + * src/plugins/DescendChart.py: + * src/plugins/EndOfLineReport.py: + * src/plugins/AncestorChart.py: + * src/plugins/DetAncestralReport.py: + * src/plugins/CustomBookText.py: + * src/plugins/FamilyGroup.py: + * src/plugins/GVRelGraph.py: + * src/plugins/GVHourGlass.py: + * src/plugins/StatisticsChart.py: + * src/plugins/FanChart.py: + * src/PluginUtils/__init__.py: + * src/PluginUtils/_MenuOptions.py: + More refactoring in the report system. Book report should work again. + 2008-01-22 Raphael Ackermann * src/ReportBase/_Bibliography.py * src/ReportBase/_Endnotes.py diff --git a/src/PluginUtils/_MenuOptions.py b/src/PluginUtils/_MenuOptions.py index 1db378f4f..ddbe1c396 100644 --- a/src/PluginUtils/_MenuOptions.py +++ b/src/PluginUtils/_MenuOptions.py @@ -357,10 +357,10 @@ class EnumeratedListOption(Option): #------------------------------------------------------------------------- # -# PersonFilterOption class +# FilterOption class # #------------------------------------------------------------------------- -class PersonFilterOption(EnumeratedListOption): +class FilterOption(EnumeratedListOption): """ This class describes an option that provides a list of person filters. Each possible value represents one of the possible filters. diff --git a/src/PluginUtils/__init__.py b/src/PluginUtils/__init__.py index a3e5b91d1..cb4ac5828 100644 --- a/src/PluginUtils/__init__.py +++ b/src/PluginUtils/__init__.py @@ -30,7 +30,7 @@ # of the list. from _MenuOptions import \ NumberOption, BooleanOption, TextOption, \ - EnumeratedListOption, PersonFilterOption, StringOption, ColourOption, \ + EnumeratedListOption, FilterOption, StringOption, ColourOption, \ PersonOption, PersonListOption, SurnameColourOption, FamilyOption from _GuiOptions import GuiMenuOptions from _PluginMgr import \ diff --git a/src/plugins/AncestorChart.py b/src/plugins/AncestorChart.py index 829fa1d35..34da091b4 100644 --- a/src/plugins/AncestorChart.py +++ b/src/plugins/AncestorChart.py @@ -179,14 +179,15 @@ class AncestorChart(Report): compress - Whether to compress chart. """ Report.__init__(self,database,person,options_class) - - self.display = options_class.handler.options_dict['dispf'] - self.max_generations = options_class.handler.options_dict['maxgen'] - self.force_fit = options_class.handler.options_dict['singlep'] - self.incblank = options_class.handler.options_dict['incblank'] - self.compress = options_class.handler.options_dict['compress'] - pid = options_class.handler.options_dict['pid'] + menu = options_class.menu + self.display = menu.get_option_by_name('dispf').get_value() + self.max_generations = menu.get_option_by_name('maxgen').get_value() + self.force_fit = menu.get_option_by_name('singlep').get_value() + self.incblank = menu.get_option_by_name('incblank').get_value() + self.compress = menu.get_option_by_name('compress').get_value() + + pid = menu.get_option_by_name('pid').get_value() center_person = database.get_person_from_gramps_id(pid) name = name_displayer.display_formal(center_person) diff --git a/src/plugins/AncestorReport.py b/src/plugins/AncestorReport.py index 4afc7479d..d8d77fba7 100644 --- a/src/plugins/AncestorReport.py +++ b/src/plugins/AncestorReport.py @@ -28,7 +28,6 @@ # python modules # #------------------------------------------------------------------------ -import gtk import math from gettext import gettext as _ @@ -52,10 +51,15 @@ from gen.lib import ChildRefType # #------------------------------------------------------------------------ def log2(val): + """ + Calculate the log base 2 of a number + """ return int(math.log10(val)/math.log10(2)) class AncestorReport(Report): - + """ + Ancestor Report class + """ def __init__(self, database, person, options_class): """ Creates the AncestorReport object that produces the Ahnentafel report. @@ -74,16 +78,18 @@ class AncestorReport(Report): """ - Report.__init__(self,database,person,options_class) + Report.__init__(self, database, person, options_class) self.map = {} - self.max_generations = options_class.handler.options_dict['maxgen'] - self.pgbrk = options_class.handler.options_dict['pagebbg'] - self.opt_namebrk = options_class.handler.options_dict['namebrk'] - pid = options_class.handler.options_dict['pid'] + + menu = options_class.menu + self.max_generations = menu.get_option_by_name('maxgen').get_value() + self.pgbrk = menu.get_option_by_name('pagebbg').get_value() + self.opt_namebrk = menu.get_option_by_name('namebrk').get_value() + pid = menu.get_option_by_name('pid').get_value() self.center_person = database.get_person_from_gramps_id(pid) - def apply_filter(self,person_handle,index,generation=1): + def apply_filter(self, person_handle, index, generation=1): """ Recursive function to walk back all parents of the current person. When max_generations are hit, we stop the traversal. @@ -125,34 +131,36 @@ class AncestorReport(Report): # people defined as the birth parents, we will select based on # priority in the list - if not father_handle and ref[0].get_father_relation() == ChildRefType.BIRTH: + if not father_handle and \ + ref[0].get_father_relation() == ChildRefType.BIRTH: father_handle = family.get_father_handle() - if not mother_handle and ref[0].get_mother_relation() == ChildRefType.BIRTH: + if not mother_handle and \ + ref[0].get_mother_relation() == ChildRefType.BIRTH: mother_handle = family.get_mother_handle() - # Recursively call the function. It is okay if the handle is None, since - # routine handles a handle of None + # Recursively call the function. It is okay if the handle is None, + # since routine handles a handle of None self.apply_filter(father_handle, index*2, generation+1) self.apply_filter(mother_handle, (index*2)+1, generation+1) def write_report(self): """ - The routine the actually creates the report. At this point, the document is - opened and ready for writing. + The routine the actually creates the report. At this point, the document + is opened and ready for writing. """ - # Call apply_filter to build the self.map array of people in the database that - # match the ancestry. + # Call apply_filter to build the self.map array of people in the + # database that match the ancestry. - self.apply_filter(self.center_person.get_handle(),1) + self.apply_filter(self.center_person.get_handle(), 1) - # Write the title line. Set in INDEX marker so that this section will be + # Write the title line. Set in INDEX marker so that this section will be # identified as a major category if this is included in a Book report. name = name_displayer.display_formal(self.center_person) title = _("Ahnentafel Report for %s") % name - mark = BaseDoc.IndexMark(title, BaseDoc.INDEX_TYPE_TOC,1 ) + mark = BaseDoc.IndexMark(title, BaseDoc.INDEX_TYPE_TOC, 1) self.doc.start_paragraph("AHN-Title") self.doc.write_text(title, mark) self.doc.end_paragraph() @@ -174,9 +182,9 @@ class AncestorReport(Report): generation += 1 # Create the Generation title, set an index marker - mark = BaseDoc.IndexMark(title,BaseDoc.INDEX_TYPE_TOC,2) + mark = BaseDoc.IndexMark(title, BaseDoc.INDEX_TYPE_TOC, 2) self.doc.start_paragraph("AHN-Generation") - self.doc.write_text(_("Generation %d") % generation,mark) + self.doc.write_text(_("Generation %d") % generation, mark) self.doc.end_paragraph() # Build the entry @@ -191,8 +199,8 @@ class AncestorReport(Report): self.doc.write_text(name.strip(), mark) self.doc.end_bold() - # terminate with a period if it is not already terminated. This can happen - # if the person's name ends with something 'Jr.' + # terminate with a period if it is not already terminated. + # This can happen if the person's name ends with something 'Jr.' if name[-1:] == '.': self.doc.write_text(" ") else: @@ -208,9 +216,12 @@ class AncestorReport(Report): primary_name = person.get_primary_name() first = primary_name.get_first_name() - self.doc.write_text(ReportUtils.born_str(self.database,person,first)) - self.doc.write_text(ReportUtils.died_str(self.database,person,0)) - self.doc.write_text(ReportUtils.buried_str(self.database,person,0)) + self.doc.write_text( + ReportUtils.born_str(self.database, person, first)) + self.doc.write_text( + ReportUtils.died_str(self.database, person, 0)) + self.doc.write_text( + ReportUtils.buried_str(self.database, person, 0)) self.doc.end_paragraph() @@ -225,10 +236,10 @@ class AncestorOptions(MenuReportOptions): Defines options and provides handling interface. """ - def __init__(self,name,dbstate=None): - MenuReportOptions.__init__(self,name,dbstate) + def __init__(self, name, dbstate=None): + MenuReportOptions.__init__(self, name, dbstate) - def add_menu_options(self,menu,dbstate): + def add_menu_options(self, menu, dbstate): """ Add options to the menu for the ancestor report. """ @@ -236,21 +247,22 @@ class AncestorOptions(MenuReportOptions): pid = PersonOption(_("Center Person")) pid.set_help(_("The center person for the report")) - menu.add_option(category_name,"pid",pid) + menu.add_option(category_name, "pid", pid) - maxgen = NumberOption(_("Generations"),10,1,15) + maxgen = NumberOption(_("Generations"), 10, 1, 15) maxgen.set_help(_("The number of generations to include in the report")) - menu.add_option(category_name,"maxgen",maxgen) + menu.add_option(category_name, "maxgen", maxgen) - pagebbg = BooleanOption(_("Page break between generations"),False) - pagebbg.set_help(_("Whether to start a new page after each generation.")) - menu.add_option(category_name,"pagebbg",pagebbg) + pagebbg = BooleanOption(_("Page break between generations"), False) + pagebbg.set_help( + _("Whether to start a new page after each generation.")) + menu.add_option(category_name, "pagebbg", pagebbg) - namebrk = BooleanOption(_("Add linebreak after each name"),False) + namebrk = BooleanOption(_("Add linebreak after each name"), False) namebrk.set_help(_("Indicates if a line break should follow the name.")) - menu.add_option(category_name,"namebrk",namebrk) + menu.add_option(category_name, "namebrk", namebrk) - def make_default_style(self,default_style): + def make_default_style(self, default_style): """ Make the default output style for the Ahnentafel report. @@ -284,7 +296,7 @@ class AncestorOptions(MenuReportOptions): # AHN-Title # font = BaseDoc.FontStyle() - font.set(face=BaseDoc.FONT_SANS_SERIF,size=16,bold=1) + font.set(face=BaseDoc.FONT_SANS_SERIF, size=16, bold=1) para = BaseDoc.ParagraphStyle() para.set_font(font) para.set_header_level(1) @@ -292,30 +304,30 @@ class AncestorOptions(MenuReportOptions): para.set_bottom_margin(0.25) para.set_alignment(BaseDoc.PARA_ALIGN_CENTER) para.set_description(_('The style used for the title of the page.')) - default_style.add_paragraph_style("AHN-Title",para) + default_style.add_paragraph_style("AHN-Title", para) # # AHN-Generation # font = BaseDoc.FontStyle() - font.set(face=BaseDoc.FONT_SANS_SERIF,size=14,italic=1) + font.set(face=BaseDoc.FONT_SANS_SERIF, size=14, italic=1) para = BaseDoc.ParagraphStyle() para.set_font(font) para.set_header_level(2) para.set_top_margin(0.125) para.set_bottom_margin(0.125) para.set_description(_('The style used for the generation header.')) - default_style.add_paragraph_style("AHN-Generation",para) + default_style.add_paragraph_style("AHN-Generation", para) # # AHN-Entry # para = BaseDoc.ParagraphStyle() - para.set(first_indent=-1.0,lmargin=1.0) + para.set(first_indent=-1.0, lmargin=1.0) para.set_top_margin(0.125) para.set_bottom_margin(0.125) para.set_description(_('The basic style used for the text display.')) - default_style.add_paragraph_style("AHN-Entry",para) + default_style.add_paragraph_style("AHN-Entry", para) #------------------------------------------------------------------------ # diff --git a/src/plugins/BookReport.py b/src/plugins/BookReport.py index 4b7a6044f..f1521e4f6 100644 --- a/src/plugins/BookReport.py +++ b/src/plugins/BookReport.py @@ -46,9 +46,9 @@ log = logging.getLogger(".BookReport") # #------------------------------------------------------------------------- try: - from xml.sax import make_parser,handler,SAXParseException + from xml.sax import make_parser, handler, SAXParseException except: - from _xmlplus.sax import make_parser,handler,SAXParseException + from _xmlplus.sax import make_parser, handler, SAXParseException #------------------------------------------------------------------------- # @@ -63,7 +63,6 @@ import gtk.glade # gramps modules # #------------------------------------------------------------------------- -from gen.lib import Person import const import Utils import ListModel @@ -71,14 +70,13 @@ import Errors import BaseDoc from QuestionDialog import WarningDialog, ErrorDialog from PluginUtils import bkitems_list, register_report, Plugins -from PluginUtils import PersonOption, PersonFilterOption +from PluginUtils import PersonOption, FilterOption, FamilyOption import ManagedWindow # Import from specific modules in ReportBase from ReportBase._Constants import CATEGORY_BOOK, MODE_GUI, MODE_CLI from ReportBase._BookFormatComboBox import BookFormatComboBox from ReportBase._BareReportDialog import BareReportDialog -from ReportBase._ReportDialog import ReportDialog from ReportBase._DocReportDialog import DocReportDialog from ReportBase._CommandLineReport import CommandLineReport from ReportBase._ReportOptions import ReportOptions @@ -90,28 +88,83 @@ from BasicUtils import name_displayer as _nd # Private Functions # #------------------------------------------------------------------------ -def _get_subject(options,db): +def _initialize_options(options, dbstate): + """ + Validates all options by making sure that their values are consistent with + the database. + + menu: The Menu class + dbase: the database the options will be applied to + """ + dbase = dbstate.get_database() + if not hasattr(options, "menu"): + return + menu = options.menu + + for name in menu.get_all_option_names(): + option = menu.get_option_by_name(name) + + if isinstance(option, PersonOption): + person = dbstate.get_active_person() + option.set_value(person.get_gramps_id()) + + elif isinstance(option, FamilyOption): + person = dbstate.get_active_person() + family_list = person.get_family_handle_list() + if family_list: + family_handle = family_list[0] + else: + family_handle = dbase.get_family_handles()[0] + family = dbase.get_family_from_handle(family_handle) + option.set_value(family.get_gramps_id()) + +def _get_subject(options, dbase): """ Attempts to determine the subject of a set of options. The subject would likely be a person (using a PersonOption) or a filter (using a - PersonFilterOption) + FilterOption) options: The ReportOptions class - db: the database for which it corresponds + dbase: the database for which it corresponds """ - if not hasattr(options,"menu"): + if not hasattr(options, "menu"): return _("Not Applicable") menu = options.menu + option_names = menu.get_all_option_names() + for name in option_names: option = menu.get_option_by_name(name) - if isinstance(option, PersonFilterOption): + + if isinstance(option, FilterOption): return option.get_filter().get_name() + elif isinstance(option, PersonOption): gid = option.get_value() - person = db.get_person_from_gramps_id(gid) + person = dbase.get_person_from_gramps_id(gid) return _nd.display(person) - + + elif isinstance(option, FamilyOption): + family = dbase.get_family_from_gramps_id(option.get_value()) + family_id = family.get_gramps_id() + fhandle = family.get_father_handle() + mhandle = family.get_mother_handle() + + if fhandle: + father = dbase.get_person_from_handle(fhandle) + father_name = _nd.display(father) + else: + father_name = _("unknown father") + + if mhandle: + mother = dbase.get_person_from_handle(mhandle) + mother_name = _nd.display(mother) + else: + mother_name = _("unknown mother") + + name = _("%s and %s (%s)") % (father_name, mother_name, family_id) + return name + return _("Not Applicable") #------------------------------------------------------------------------ @@ -124,7 +177,7 @@ class BookItem: Interface into the book item -- a smallest element of the book. """ - def __init__(self,dbstate,name=None): + def __init__(self, dbstate, name=None): """ Creates a new empty BookItem. @@ -152,7 +205,7 @@ class BookItem: self.make_default_style = None self.name = "" - def get_registered_item(self,name): + def get_registered_item(self, name): """ Retrieve the item from the book item registry. @@ -169,7 +222,7 @@ class BookItem: self.category = item[1] self.write_item = item[2] self.name = item[4] - self.option_class = item[3](self.name,self.dbstate) + self.option_class = item[3](self.name, self.dbstate) self.option_class.load_previous_values() def get_name(self): @@ -196,7 +249,7 @@ class BookItem: """ return self.write_item - def set_style_name(self,style_name): + def set_style_name(self, style_name): """ Sets the style name for the item. @@ -232,7 +285,7 @@ class Book: Interface into the user-defined book -- a collection of book items. """ - def __init__(self,obj=None): + def __init__(self, obj=None): """ Creates a new empty Book. @@ -247,7 +300,7 @@ class Book: else: self.item_list = [] - def set_name(self,name): + def set_name(self, name): """ Sets the name of the book. @@ -267,7 +320,7 @@ class Book: """ return self.dbname - def set_dbname(self,name): + def set_dbname(self, name): """ Sets the name of the database file used for the book. @@ -281,7 +334,7 @@ class Book: """ self.item_list = [] - def append_item(self,item): + def append_item(self, item): """ Adds an item to the book. @@ -289,16 +342,16 @@ class Book: """ self.item_list.append(item) - def insert_item(self,index,item): + def insert_item(self, index, item): """ Inserts an item into the given position in the book. index: a position index. item: an item to append. """ - self.item_list.insert(index,item) + self.item_list.insert(index, item) - def pop_item(self,index): + def pop_item(self, index): """ Pop an item from given position in the book. @@ -306,7 +359,7 @@ class Book: """ return self.item_list.pop(index) - def get_item(self,index): + def get_item(self, index): """ Returns an item at a given position in the book. @@ -314,7 +367,7 @@ class Book: """ return self.item_list[index] - def set_item(self,index,item): + def set_item(self, index, item): """ Sets an item at a given position in the book. @@ -341,7 +394,7 @@ class BookList: BookList is loaded from a specified XML file if it exists. """ - def __init__(self,filename,dbstate): + def __init__(self, filename, dbstate): """ Creates a new BookList from the books that may be defined in the specified file. @@ -350,10 +403,10 @@ class BookList: """ self.dbstate = dbstate self.bookmap = {} - self.file = os.path.join(const.HOME_DIR,filename) + self.file = os.path.join(const.HOME_DIR, filename) self.parse() - def delete_book(self,name): + def delete_book(self, name): """ Removes a book from the list. Since each book must have a unique name, the name is used to delete the book. @@ -368,7 +421,7 @@ class BookList: """ return self.bookmap - def get_book(self,name): + def get_book(self, name): """ Returns the Book associated with the name @@ -380,7 +433,7 @@ class BookList: "Returns a list of all the book names in the BookList" return self.bookmap.keys() - def set_book(self,name,book): + def set_book(self, name, book): """ Adds or replaces a Book in the BookList. @@ -393,21 +446,21 @@ class BookList: """ Saves the current BookList to the associated file. """ - f = open(self.file,"w") + f = open(self.file, "w") f.write("\n") f.write('\n') for name in self.bookmap.keys(): book = self.get_book(name) dbname = book.get_dbname() - f.write('\n' % (name,dbname) ) + f.write('\n' % (name, dbname) ) for item in book.get_item_list(): f.write(' \n' % (item.get_name(),item.get_translated_name() ) ) option_handler = item.option_class.handler for option_name in option_handler.options_dict.keys(): option_value = option_handler.options_dict[option_name] - if type(option_value) in (list,tuple): + if type(option_value) in (list, tuple): f.write('