Several corrections, update has_data functionality, added update_exif functionality.

svn: r17490
This commit is contained in:
Rob G. Healey 2011-05-13 07:00:47 +00:00
parent eedfafb6d4
commit 1b0ddbed83
2 changed files with 117 additions and 95 deletions

View File

@ -210,7 +210,7 @@ _BUTTONTIPS = {
"Help" : _("Displays the Gramps Wiki Help page for 'Edit Image Exif Metadata' " "Help" : _("Displays the Gramps Wiki Help page for 'Edit Image Exif Metadata' "
"in your web browser."), "in your web browser."),
# AdvancedView button... # Advanced View button...
"AdvancedView" : _("Will pop open a window with all of the Exif metadata Key/alue pairs."), "AdvancedView" : _("Will pop open a window with all of the Exif metadata Key/alue pairs."),
# Save Exif Metadata button... # Save Exif Metadata button...
@ -279,16 +279,13 @@ class EditExifMetadata(Gramplet):
self.plugin_image = False self.plugin_image = False
self.MediaDataTags = False self.MediaDataTags = False
self.connect_signal("Media", self.update)
vbox = self.build_gui() vbox = self.build_gui()
self.connect_signal("Media", self.update)
self.gui.get_container_widget().remove(self.gui.textview) self.gui.get_container_widget().remove(self.gui.textview)
self.gui.get_container_widget().add_with_viewport(vbox) self.gui.get_container_widget().add_with_viewport(vbox)
vbox.show_all() vbox.show_all()
# provide tooltips for all fields and buttons...
_setup_widget_tooltips(self.exif_widgets)
def build_gui(self): def build_gui(self):
""" """
creates the editing area fields. creates the editing area fields.
@ -299,6 +296,7 @@ class EditExifMetadata(Gramplet):
hbox = gtk.HBox(False) hbox = gtk.HBox(False)
label = gtk.Label() label = gtk.Label()
label.set_alignment(0.0, 0.0) label.set_alignment(0.0, 0.0)
label.set_line_wrap(True)
self.exif_widgets["Media:Label"] = label self.exif_widgets["Media:Label"] = label
hbox.pack_start(self.exif_widgets["Media:Label"], expand =False) hbox.pack_start(self.exif_widgets["Media:Label"], expand =False)
vbox.pack_start(hbox, expand =False) vbox.pack_start(hbox, expand =False)
@ -306,6 +304,7 @@ class EditExifMetadata(Gramplet):
hbox = gtk.HBox(False) hbox = gtk.HBox(False)
label = gtk.Label() label = gtk.Label()
label.set_alignment(0.0, 0.0) label.set_alignment(0.0, 0.0)
label.set_line_wrap(True)
self.exif_widgets["Mime:Type"] = label self.exif_widgets["Mime:Type"] = label
hbox.pack_start(self.exif_widgets["Mime:Type"], expand =False) hbox.pack_start(self.exif_widgets["Mime:Type"], expand =False)
vbox.pack_start(hbox, expand =False) vbox.pack_start(hbox, expand =False)
@ -313,6 +312,7 @@ class EditExifMetadata(Gramplet):
hbox = gtk.HBox(False) hbox = gtk.HBox(False)
label = gtk.Label() label = gtk.Label()
label.set_alignment(0.5, 0.0) label.set_alignment(0.5, 0.0)
label.set_line_wrap(True)
self.exif_widgets["Message:Area"] = label self.exif_widgets["Message:Area"] = label
hbox.pack_start(self.exif_widgets["Message:Area"], expand =False) hbox.pack_start(self.exif_widgets["Message:Area"], expand =False)
vbox.pack_start(hbox, expand =False) vbox.pack_start(hbox, expand =False)
@ -374,32 +374,27 @@ class EditExifMetadata(Gramplet):
vbox.pack_start(row, False) vbox.pack_start(row, False)
# Help, Save, Delete horizontal box # Help, Save, Delete horizontal box
hsd_box = gtk.HButtonBox() hasd_box = gtk.HButtonBox()
hsd_box.set_layout(gtk.BUTTONBOX_START) hasd_box.set_layout(gtk.BUTTONBOX_START)
vbox.pack_start(hsd_box, expand =False, fill =False, padding =0) vbox.pack_start(hasd_box, expand =False, fill =False, padding =0)
# Help button... # Help button...
hsd_box.add( self.__create_button( hasd_box.add( self.__create_button(
"Help", False, [_help_page], gtk.STOCK_HELP) ) "Help", False, [_help_page], gtk.STOCK_HELP) )
# Save button... # Save button...
hsd_box.add( self.__create_button( hasd_box.add( self.__create_button(
"Save", False, [self.__save_dialog], gtk.STOCK_SAVE, False) ) "Save", False, [self.__save_dialog], gtk.STOCK_SAVE, False) )
# Advanced View Area button...
hasd_box.add( self.__create_button(
"AdvancedView", _("Advanced"), [self.advanced_view], sensitive =False) )
if _MAGICK_FOUND: if _MAGICK_FOUND:
# Delete All Metadata button... # Delete All Metadata button...
hsd_box.add(self.__create_button( hasd_box.add(self.__create_button(
"Delete", False, [self.__delete_dialog], gtk.STOCK_DELETE, False)) "Delete", False, [self.__delete_dialog], gtk.STOCK_DELETE, False))
# advanced view horizontal button box...
adv_box = gtk.HButtonBox()
adv_box.set_layout(gtk.BUTTONBOX_START)
vbox.pack_start(adv_box, expand =False, fill =False, padding =0)
# Advanced View Area button...
adv_box.add(self.__create_button(
"AdvancedView", _("Advanced View"), [self.advanced_view], sensitive =False) )
# de-activate Thumbnail button... # de-activate Thumbnail button...
self.exif_widgets["Popup:Thumbnail"].set_sensitive(False) self.exif_widgets["Popup:Thumbnail"].set_sensitive(False)
@ -509,25 +504,31 @@ class EditExifMetadata(Gramplet):
_mtype = gen.mime.get_description(mime_type) _mtype = gen.mime.get_description(mime_type)
self.exif_widgets["Mime:Type"].set_text(_mtype) self.exif_widgets["Mime:Type"].set_text(_mtype)
# display file description/ title...
self.exif_widgets["Media:Label"].set_text(_html_escape(
self.orig_image.get_description() ) )
# determine if it is a mime image object? # determine if it is a mime image object?
if mime_type: if mime_type:
if mime_type.startswith("image"): if mime_type.startswith("image"):
self.activate_buttons(["Save"]) self.activate_buttons(["Save"])
# display file description/ title...
self.exif_widgets["Media:Label"].set_text(_html_escape(
self.orig_image.get_description() ) )
# will create the plugin image and read it...
self.setup_image(self.image_path, True)
# Checks to make sure that ImageMagick is installed on this computer and # Checks to make sure that ImageMagick is installed on this computer and
# the image is NOT a (.jpeg, .jfif, .jpg) image... # the image is NOT a (".jpeg", ".jfif", or ".jpg") image...
if _MAGICK_FOUND:
basename, extension = os.path.splitext(self.image_path) basename, extension = os.path.splitext(self.image_path)
if _MAGICK_FOUND:
# check to make sure that the image is not a jpeg image?
# if not, then activate the Convert button...
if extension not in [".jpeg", ".jpg", ".jfif"]: if extension not in [".jpeg", ".jpg", ".jfif"]:
self.activate_buttons(["Convert"]) self.activate_buttons(["Convert"])
# creates, reads, setup tooltips, and get all Exif metadata
self.plugin_image = self.setup_image(self.image_path, extension)
# Retries all metadata key pairs from image...
self.MediaDataTags = _get_exif_keypairs(self.plugin_image)
# displays the imge Exif metadata # displays the imge Exif metadata
self.EditArea(self.orig_image) self.EditArea(self.orig_image)
@ -593,46 +594,62 @@ class EditExifMetadata(Gramplet):
"delete the Exif metadata from this image?"), _("Delete"), "delete the Exif metadata from this image?"), _("Delete"),
self.strip_metadata) self.strip_metadata)
def setup_image(self, full_path, createimage =False): def setup_image(self, full_path, extension):
""" """
will return an image instance and read the Exif metadata. This will:
* create the plugin image instance if needed,
* setup the tooltips for the data fields,
* setup the tooltips for the buttons,
if createimage is True, it will create the pyexiv2 image instance... * checks to see if the image extension is in the acceptable image types?
* if bad?
LesserVersion -- prior to pyexiv2-0.2.0 * disables all the data fields, and
-- pyexiv2-0.2.0 and above... all the buttons...
""" """
if createimage:
if LesserVersion: if LesserVersion:
self.plugin_image = pyexiv2.Image(full_path) metadata = pyexiv2.Image(full_path)
else: else:
self.plugin_image = pyexiv2.ImageMetadata(full_path) metadata = pyexiv2.ImageMetadata(full_path)
if LesserVersion: if LesserVersion:
try: try:
self.plugin_image.readMetadata() metadata.readMetadata()
except (IOError, OSError): except (IOError, OSError):
self.set_has_data(False) self.set_has_data(False)
return return
# get all KeyTags for this image for diplay only...
self.MediaDataTags = [KeyTag for KeyTag in chain(
self.plugin_image.exifKeys(),
self.plugin_image.xmpKeys(),
self.plugin_image.iptcKeys() ) ]
else: else:
try: try:
self.plugin_image.read() metadata.read()
except (IOError, OSError): except (IOError, OSError):
self.set_has_data(False) self.set_has_data(False)
return return
# get all KeyTags for this image for diplay only... # disable data fields if it is NOT an exiv2 image type?
self.MediaDataTags = [KeyTag for KeyTag in chain( goodext = True
self.plugin_image.exif_keys, if extension not in _vtypes:
self.plugin_image.xmp_keys, goodext = False
self.plugin_image.iptc_keys) ]
# add tooltips for the data entry fields...
for widget, tooltip in _TOOLTIPS:
self.exif_widgets[widget].set_tooltip_text(tooltip)
# if extension is not in the acceptable extensions for exiv2 images,
# disable the edit fields?
if not goodext:
self.exif_widgets[widget].set_editable(False)
# add tooltips for the buttons...
for widget, tooltip in _BUTTONTIPS.items():
self.exif_widgets[widget].set_tooltip_text(tooltip)
# if extension is not in the acceptable extensions for exiv2 images,
# disable the buttons?
if not goodext:
self.exif_widgets[widget].set_sensitive(False)
return metadata
def make_row(self, pos, text, choices=None, readonly=False, callback_list=[], def make_row(self, pos, text, choices=None, readonly=False, callback_list=[],
mark_dirty=False, default=0): mark_dirty=False, default=0):
@ -696,9 +713,6 @@ class EditExifMetadata(Gramplet):
row.show_all() row.show_all()
return row return row
# -----------------------------------------------
# Error Checking functions
# -----------------------------------------------
def _mark_dirty(self, obj): def _mark_dirty(self, obj):
pass pass
@ -749,18 +763,16 @@ class EditExifMetadata(Gramplet):
# activate Clear button... # activate Clear button...
self.activate_buttons(["Clear"]) self.activate_buttons(["Clear"])
# if no exif metadata, disable the has_data functionality?
self.set_has_data(True)
if not self.MediaDataTags:
self.set_has_data(False)
# Activate Delete button if ImageMagick or jhead is found?... # Activate Delete button if ImageMagick or jhead is found?...
if (_MAGICK_FOUND or _JHEAD_FOUND): if (_MAGICK_FOUND or _JHEAD_FOUND):
self.activate_buttons(["Delete"]) self.activate_buttons(["Delete"])
# separate KeyTags into the ones that we will be using... imageKeyTags = [KeyTag for KeyTag in self.MediaDataTags if KeyTag in _DATAMAP]
if LesserVersion:
imageKeyTags = [KeyTag for KeyTag in self.plugin_image.exifKeys()
if KeyTag in _DATAMAP]
else:
imageKeyTags = [KeyTag for KeyTag in self.plugin_image.exif_keys
if KeyTag in _DATAMAP]
if imageKeyTags: if imageKeyTags:
# activate AdvancedView button... # activate AdvancedView button...
@ -780,18 +792,15 @@ class EditExifMetadata(Gramplet):
if widgetsName in ["Description", "Artist", "Copyright"]: if widgetsName in ["Description", "Artist", "Copyright"]:
self.exif_widgets[widgetsName].set_text(tagValue) self.exif_widgets[widgetsName].set_text(tagValue)
# Last Changed/ Modified... # Last Changed/ Modified and Original Date/ Time...
elif widgetsName == "Modified": elif (widgetsName in ["Modified", "DateTime"]):
use_date = self._get_value(KeyTag) if widgetsName == "Modified":
key = _DATAMAP["Modified"]
else:
key = _DATAMAP["DateTime"]
use_date = self._get_value(key)
use_date = _process_datetime(use_date) if use_date else False use_date = _process_datetime(use_date) if use_date else False
if use_date is not False: if use_date:
self.exif_widgets[widgetsName].set_text(use_date)
# Original Date/ Time of this image...
elif widgetsName == "DateTime":
use_date = self._get_value(KeyTag)
use_date = _process_datetime(use_date) if use_date else False
if use_date is not False:
self.exif_widgets[widgetsName].set_text(use_date) self.exif_widgets[widgetsName].set_text(use_date)
# LatitudeRef, Latitude, LongitudeRef, Longitude... # LatitudeRef, Latitude, LongitudeRef, Longitude...
@ -1068,6 +1077,15 @@ class EditExifMetadata(Gramplet):
advarea.add(vbox) advarea.add(vbox)
advarea.show() advarea.show()
# Update the image Exif metadata...
self.MediaDataTags = _retrieve_exif_keypairs(self.plugin_image)
# update has_data() functionality...
if self.MediaDataTags:
self.set_has_data(True)
else:
self.set_has_data(False)
if LesserVersion: # prior to pyexiv2-0.2.0 if LesserVersion: # prior to pyexiv2-0.2.0
for KeyTag in self.MediaDataTags: for KeyTag in self.MediaDataTags:
label = self.plugin_image.tagDetails(KeyTag)[0] label = self.plugin_image.tagDetails(KeyTag)[0]
@ -1254,9 +1272,6 @@ class EditExifMetadata(Gramplet):
if _JHEAD_FOUND: if _JHEAD_FOUND:
reinit = subprocess.check_call( ["jhead", "-purejpg", self.image_path] ) reinit = subprocess.check_call( ["jhead", "-purejpg", self.image_path] )
# -----------------------------------------------
# Date Calendar functions
# -----------------------------------------------
def select_date(self, obj): def select_date(self, obj):
""" """
will allow you to choose a date from the calendar widget will allow you to choose a date from the calendar widget
@ -1306,8 +1321,8 @@ class EditExifMetadata(Gramplet):
# extract the thumbnail data # extract the thumbnail data
previews = self.plugin_image.previews previews = self.plugin_image.previews
if not (len(previews) > 0): if not previews:
print(_("This image doesn't contain any Thumbnails.")) print(_("This image doesn't contain any Thumbnails..."))
tbarea.destroy() tbarea.destroy()
else: else:
@ -1329,6 +1344,26 @@ class EditExifMetadata(Gramplet):
imgwidget.show() imgwidget.show()
tbarea.show() tbarea.show()
def _get_exif_keypairs(plugin_image):
"""
Will be used to retrieve and update the Exif metadata from the image.
"""
MediaDataTags = False
if plugin_image:
if LesserVersion: # prior to pyexiv2-0.2.0
# get all KeyTags for this image for diplay only...
MediaDataTags = [KeyTag for KeyTag in chain(
plugin_image.exifKeys(), plugin_image.xmpKeys(),
plugin_image.iptcKeys() ) ]
else: # pyexiv2-0.2.0 and above
# get all KeyTags for this image for diplay only...
MediaDataTags = [KeyTag for KeyTag in chain(
plugin_image.exif_keys, plugin_image.xmp_keys,
plugin_image.iptc_keys) ]
return MediaDataTags
def string_to_rational(coordinate): def string_to_rational(coordinate):
""" """
convert string to rational variable for GPS convert string to rational variable for GPS
@ -1557,16 +1592,3 @@ def _process_datetime(tmpDate, exif_type =True):
tmpDate = _format_datetime(tmpDate) tmpDate = _format_datetime(tmpDate)
return tmpDate return tmpDate
def _setup_widget_tooltips(Exif_widgets):
"""
setup tooltips for each entry field and button.
"""
# add tooltips for the data entry fields...
for widget, tooltip in _TOOLTIPS:
Exif_widgets[widget].set_tooltip_text(tooltip)
# add tooltips for the buttons...
for widget, tooltip in _BUTTONTIPS.items():
Exif_widgets[widget].set_tooltip_text(tooltip)

View File

@ -312,7 +312,7 @@ register(GRAMPLET,
id = "Edit Image Exif Metadata", id = "Edit Image Exif Metadata",
name = _("Edit Image Exif Metadata"), name = _("Edit Image Exif Metadata"),
description = _("Gramplet to view, edit, and save image Exif metadata"), description = _("Gramplet to view, edit, and save image Exif metadata"),
height = 550, height = 450,
expand = False, expand = False,
gramplet = 'EditExifMetadata', gramplet = 'EditExifMetadata',
gramplet_title = _("Edit Exif Metadata"), gramplet_title = _("Edit Exif Metadata"),