325 lines
12 KiB
Python
325 lines
12 KiB
Python
#
|
|
# Gramps - a GTK+/GNOME based genealogy program
|
|
#
|
|
# Copyright (C) 2001 Donald N. Allingham
|
|
#
|
|
# 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
|
|
#
|
|
|
|
import GTK
|
|
import GDK
|
|
import gtk
|
|
import gnome.ui
|
|
import string
|
|
import ImageSelect
|
|
|
|
from RelLib import *
|
|
from QuestionDialog import QuestionDialog
|
|
import Utils
|
|
import os
|
|
import GrampsCfg
|
|
import const
|
|
import RelImage
|
|
import Sorter
|
|
|
|
from intl import gettext
|
|
_ = gettext
|
|
|
|
class MediaView:
|
|
def __init__(self,db,glade,update,top_window):
|
|
self.db = db
|
|
self.top_window = top_window
|
|
self.media_list = glade.get_widget("media_list")
|
|
self.mid = glade.get_widget("mid")
|
|
self.mtype = glade.get_widget("mtype")
|
|
self.mdesc = glade.get_widget("mdesc")
|
|
self.mpath = glade.get_widget("mpath")
|
|
self.mdetails = glade.get_widget("mdetails")
|
|
self.preview = glade.get_widget("preview")
|
|
|
|
arrow_map = [ (5, glade.get_widget("mdescr_arrow")),
|
|
(1, glade.get_widget("mid_arrow")),
|
|
(2, glade.get_widget("mtype_arrow")),
|
|
(3, glade.get_widget("mpath_arrow"))]
|
|
|
|
t = [ ('STRING', 0, 0),
|
|
('text/plain',0,0),
|
|
('text/uri-list',0,2),
|
|
('application/x-rootwin-drop',0,1)]
|
|
|
|
self.media_list.drag_source_set(GDK.BUTTON1_MASK|GDK.BUTTON3_MASK,
|
|
t,GDK.ACTION_COPY)
|
|
self.media_list.drag_dest_set(GTK.DEST_DEFAULT_ALL,
|
|
t,GDK.ACTION_COPY|GDK.ACTION_MOVE)
|
|
|
|
self.update = update
|
|
self.media_list.set_column_visibility(4,GrampsCfg.show_detail)
|
|
self.media_list.set_column_visibility(5,0)
|
|
self.media_list.connect('button-press-event',self.on_button_press_event)
|
|
|
|
# Restore the previous sort column
|
|
|
|
self.media_sort = Sorter.Sorter(self.media_list, arrow_map, 'media', self.top_window)
|
|
|
|
def change_db(self,db):
|
|
self.db = db
|
|
|
|
def moveto(self,row):
|
|
self.media_list.unselect_all()
|
|
self.media_list.select_row(row,0)
|
|
self.media_list.moveto(row)
|
|
|
|
def on_select_row(self,obj,row,b,c):
|
|
fexists = 1
|
|
|
|
mobj = obj.get_row_data(row)
|
|
type = mobj.getMimeType()
|
|
type_name = Utils.get_mime_description(type)
|
|
path = mobj.getPath()
|
|
thumb_path = Utils.thumb_path(self.db.getSavePath(),mobj)
|
|
pexists = os.path.exists(path)
|
|
if pexists and os.path.exists(thumb_path):
|
|
self.preview.load_file(thumb_path)
|
|
else:
|
|
self.preview.load_file(Utils.find_icon(type))
|
|
if not pexists:
|
|
fexists = 0
|
|
|
|
self.mid.set_text(mobj.getId())
|
|
self.mtype.set_text(type_name)
|
|
self.mdesc.set_text(mobj.getDescription())
|
|
if len(path) == 0 or fexists == 0:
|
|
self.mpath.set_text(_("The file no longer exists"))
|
|
elif path[0] == "/":
|
|
self.mpath.set_text(path)
|
|
else:
|
|
self.mpath.set_text("<local>")
|
|
self.mdetails.set_text(Utils.get_detail_text(mobj,0))
|
|
|
|
def on_button_press_event(self,obj,event):
|
|
if len(self.media_list.selection) <= 0:
|
|
return
|
|
object = self.media_list.get_row_data(self.media_list.selection[0])
|
|
if event.button == 1 and event.type == GDK._2BUTTON_PRESS:
|
|
ImageSelect.GlobalMediaProperties(self.db,object,self.load_media)
|
|
elif event.button == 3:
|
|
menu = gtk.GtkMenu()
|
|
item = gtk.GtkTearoffMenuItem()
|
|
item.show()
|
|
menu.append(item)
|
|
self.obj = object
|
|
Utils.add_menuitem(menu,_("View in the default viewer"),None,self.popup_view_photo)
|
|
if object.getMimeType()[0:5] == "image":
|
|
Utils.add_menuitem(menu,_("Edit with the GIMP"),\
|
|
None,self.popup_edit_photo)
|
|
Utils.add_menuitem(menu,_("Edit Object Properties"),None,
|
|
self.popup_change_description)
|
|
if object.getLocal() == 0:
|
|
Utils.add_menuitem(menu,_("Convert to local copy"),None,
|
|
self.popup_convert_to_private)
|
|
menu.popup(None,None,None,0,0)
|
|
|
|
def popup_view_photo(self, obj):
|
|
Utils.view_photo(self.obj)
|
|
|
|
def popup_edit_photo(self, obj):
|
|
if os.fork() == 0:
|
|
os.execvp(const.editor,[const.editor, self.obj.getPath()])
|
|
|
|
def popup_convert_to_private(self, obj):
|
|
path = self.db.getSavePath()
|
|
id = self.obj.getId()
|
|
name = RelImage.import_media_object(self.obj.getPath(),path,id)
|
|
if name:
|
|
self.obj.setPath(name)
|
|
self.obj.setLocal(1)
|
|
|
|
def popup_change_description(self, obj):
|
|
ImageSelect.GlobalMediaProperties(self.db,self.obj,self.load_media)
|
|
|
|
def load_media(self):
|
|
|
|
if len(self.media_list.selection) == 0:
|
|
current_row = 0
|
|
else:
|
|
current_row = self.media_list.selection[0]
|
|
|
|
self.media_list.freeze()
|
|
self.media_list.clear()
|
|
self.media_list.set_column_visibility(1,GrampsCfg.id_visible)
|
|
self.media_list.set_column_visibility(4,GrampsCfg.show_detail)
|
|
|
|
index = 0
|
|
objects = self.db.getObjectMap().values()
|
|
|
|
for src in objects:
|
|
title = src.getDescription()
|
|
id = src.getId()
|
|
type = Utils.get_mime_description(src.getMimeType())
|
|
if src.getLocal():
|
|
path = "<local copy>"
|
|
else:
|
|
path = src.getPath()
|
|
details = Utils.get_detail_flags(src,0)
|
|
stitle = string.upper(title)
|
|
self.media_list.append([title,id,type,path,details,stitle])
|
|
self.media_list.set_row_data(index,src)
|
|
index = index + 1
|
|
|
|
self.media_sort.sort_list()
|
|
|
|
if index > 0:
|
|
self.media_list.select_row(current_row,0)
|
|
self.media_list.moveto(current_row)
|
|
self.preview.show()
|
|
else:
|
|
self.mid.set_text("")
|
|
self.mtype.set_text("")
|
|
self.mdesc.set_text("")
|
|
self.mpath.set_text("")
|
|
self.mdetails.set_text("")
|
|
self.preview.hide()
|
|
|
|
if current_row < self.media_list.rows:
|
|
self.media_list.moveto(current_row)
|
|
else:
|
|
self.media_list.moveto(0)
|
|
self.media_list.thaw()
|
|
|
|
def create_add_dialog(self,obj):
|
|
"""Add a new media object to the media list"""
|
|
import AddMedia
|
|
AddMedia.AddMediaObject(self.db,self.load_media)
|
|
|
|
def on_edit_media_clicked(self,obj):
|
|
"""Edit the properties of an existing media object in the media list"""
|
|
if len(self.media_list.selection) <= 0:
|
|
return
|
|
object = self.media_list.get_row_data(self.media_list.selection[0])
|
|
ImageSelect.GlobalMediaProperties(self.db,object,self.load_media)
|
|
|
|
def on_delete_clicked(self,obj):
|
|
if len(self.media_list.selection) <= 0:
|
|
return
|
|
else:
|
|
index = self.media_list.selection[0]
|
|
mobj = self.media_list.get_row_data(index)
|
|
if self.is_media_object_used(mobj):
|
|
ans = ImageSelect.DeleteMediaQuery(mobj,self.db,self.update)
|
|
QuestionDialog(_('Delete Object'),
|
|
_("This media object is currently being used. "
|
|
"Delete anyway?"),
|
|
_('Delete Object'),ans.query_response,
|
|
_('Keep Object'))
|
|
else:
|
|
map = self.db.getObjectMap()
|
|
del map[mobj.getId()]
|
|
Utils.modified()
|
|
self.update(0)
|
|
|
|
def is_media_object_used(self,mobj):
|
|
for p in self.db.getFamilyMap().values():
|
|
for o in p.getPhotoList():
|
|
if o.getReference() == mobj:
|
|
return 1
|
|
for key in self.db.getPersonKeys():
|
|
p = self.db.getPerson(key)
|
|
for o in p.getPhotoList():
|
|
if o.getReference() == mobj:
|
|
return 1
|
|
for key in self.db.getSourceKeys():
|
|
p = self.db.getSource(key)
|
|
for o in p.getPhotoList():
|
|
if o.getReference() == mobj:
|
|
return 1
|
|
for key in self.db.getPlaceKeys():
|
|
p = self.db.getPlace(key)
|
|
for o in p.getPhotoList():
|
|
if o.getReference() == mobj:
|
|
return 1
|
|
return 0
|
|
|
|
def on_drag_data_get(self,w, context, selection_data, info, time):
|
|
if info == 1:
|
|
return
|
|
d = w.get_row_data(w.focus_row)
|
|
id = d.getId()
|
|
selection_data.set(selection_data.target, 8, id)
|
|
|
|
def on_drag_data_received(self,w, context, x, y, data, info, time):
|
|
import urlparse
|
|
if data and data.format == 8:
|
|
d = string.strip(string.replace(data.data,'\0',' '))
|
|
protocol,site,file, j,k,l = urlparse.urlparse(d)
|
|
if protocol == "file":
|
|
name = file
|
|
mime = Utils.get_mime_type(name)
|
|
photo = Photo()
|
|
photo.setPath(name)
|
|
photo.setMimeType(mime)
|
|
description = os.path.basename(name)
|
|
photo.setDescription(description)
|
|
self.db.addObject(photo)
|
|
Utils.modified()
|
|
w.drag_finish(context, 1, 0, time)
|
|
self.load_media()
|
|
if GrampsCfg.mediaref == 0:
|
|
name = RelImage.import_media_object(name,
|
|
self.db.getSavePath(),
|
|
photo.getId())
|
|
if name:
|
|
photo.setPath(name)
|
|
photo.setLocal(1)
|
|
Utils.modified()
|
|
if GrampsCfg.globalprop:
|
|
ImageSelect.GlobalMediaProperties(self.db,photo,self.load_media)
|
|
elif protocol != "":
|
|
import urllib
|
|
u = urllib.URLopener()
|
|
try:
|
|
tfile,headers = u.retrieve(d)
|
|
except IOError, msg:
|
|
t = _("Could not import %s") % d
|
|
|
|
gnome.ui.GnomeErrorDialog("%s\n%s %d" % (t,msg[0],msg[1]))
|
|
return
|
|
mime = Utils.get_mime_type(tfile)
|
|
photo = Photo()
|
|
photo.setMimeType(mime)
|
|
photo.setDescription(d)
|
|
photo.setLocal(1)
|
|
photo.setPath(tfile)
|
|
self.db.addObject(photo)
|
|
oref = ObjectRef()
|
|
oref.setReference(photo)
|
|
try:
|
|
id = photo.getId()
|
|
path = self.db.getSavePath()
|
|
name = RelImage.import_media_object(tfile,path,id)
|
|
if name:
|
|
photo.setLocal(1)
|
|
photo.setPath(name)
|
|
except:
|
|
photo.setPath(tfile)
|
|
w.drag_finish(context, 1, 0, time)
|
|
return
|
|
Utils.modified()
|
|
if GrampsCfg.globalprop:
|
|
ImageSelect.GlobalMediaProperties(self.db,photo,None)
|
|
else:
|
|
w.drag_finish(context, 0, 0, time)
|
|
|
|
|