gramps/src/gen/proxy/referenced.py

180 lines
6.5 KiB
Python
Raw Normal View History

#
# Gramps - a GTK+/GNOME based genealogy program
#
# Copyright (C) 2008 Gary Burton
#
# 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$
"""
Proxy class for the GRAMPS databases. Returns objects which are referenced
by at least one other object.
"""
#-------------------------------------------------------------------------
#
# GRAMPS libraries
#
#-------------------------------------------------------------------------
from proxybase import ProxyDbBase
class ReferencedProxyDb(ProxyDbBase):
"""
A proxy to a Gramps database. This proxy will act like a Gramps database,
but returning all objects which are referenced by at least one other object.
"""
def __init__(self, dbase):
"""
Create a new ReferencedProxyDb instance.
"""
ProxyDbBase.__init__(self, dbase)
self.unreferenced_events = set()
self.unreferenced_places = set()
self.unreferenced_sources = set()
self.unreferenced_repositories = set()
self.unreferenced_media_objects = set()
self.unreferenced_notes = set()
self.unreferenced_tags = set()
# Build lists of unreferenced objects
self.__find_unreferenced_objects()
def include_place(self, handle):
"""
Filter for places
"""
return handle not in self.unreferenced_places
def include_media_object(self, handle):
"""
Filter for media objects
"""
return handle not in self.unreferenced_media_objects
def include_event(self, handle):
"""
Filter for events
"""
return handle not in self.unreferenced_events
def include_source(self, handle):
"""
Filter for sources
"""
return handle not in self.unreferenced_sources
def include_repository(self, handle):
"""
Filter for repositories
"""
return handle not in self.unreferenced_repositories
def include_note(self, handle):
"""
Filter for notes
"""
return handle not in self.unreferenced_notes
def include_tag(self, handle):
"""
Filter for tags
"""
return handle not in self.unreferenced_tags
def find_backlink_handles(self, handle, include_classes=None):
"""
Find all objects that hold a reference to the object handle.
2009-08-10 04:13:43 +05:30
Returns an iterator over a list of (class_name, handle) tuples.
@param handle: handle of the object to search for.
@type handle: database handle
@param include_classes: list of class names to include in the results.
Default: None means include all classes.
@type include_classes: list of class names
2009-08-10 04:13:43 +05:30
This default implementation does a sequential scan through all
the primary object databases and is very slow. Backends can
override this method to provide much faster implementations that
make use of additional capabilities of the backend.
Note that this is a generator function, it returns a iterator for
use in loops. If you want a list of the results use:
> result_list = list(find_backlink_handles(handle))
"""
perfam = {
"Person" : self.get_person_from_handle,
"Family" : self.get_family_from_handle,
}
unref = {
"Event" : self.unreferenced_events,
"Place" : self.unreferenced_places,
"Source" : self.unreferenced_sources,
"Repository" : self.unreferenced_repositories,
"MediaObject" : self.unreferenced_media_objects,
"Note" : self.unreferenced_notes,
}
handle_itr = self.db.find_backlink_handles(handle, include_classes)
for (class_name, handle) in handle_itr:
if (class_name in perfam # Person or Family exist?
and perfam[class_name](handle)
or class_name in unref # not yet in unref?
and handle not in unref[class_name]):
yield (class_name, handle)
return
def __find_unreferenced_objects(self):
"""
Builds lists of all objects that are not referenced by another object.
These may be objects that are really unreferenced or because they
are referenced by something or someone that has already been filtered
by one of the other proxy decorators.
By adding an object to a list, other referenced objects could
effectively become unreferenced, so the action is performed in a loop
until there are no more objects to unreference.
"""
2010-01-09 03:17:02 +05:30
unrefs = (
(self.unreferenced_events, self.get_event_handles),
(self.unreferenced_places, self.get_place_handles),
(self.unreferenced_sources, self.get_source_handles),
(self.unreferenced_repositories,
self.get_repository_handles),
(self.unreferenced_media_objects,
self.get_media_object_handles),
(self.unreferenced_notes, self.get_note_handles),
(self.unreferenced_tags, self.get_tag_handles),
2010-01-09 03:17:02 +05:30
)
last_count = 0
while True:
current_count = 0
2010-01-09 03:17:02 +05:30
for (unref_list, handle_list) in unrefs:
unref_list.update(
2010-01-09 03:17:02 +05:30
handle for handle in handle_list()
if not any(self.find_backlink_handles(handle)))
current_count += len(unref_list)
if current_count == last_count:
break
last_count = current_count