2023-09-26 00:35:21 +02:00
|
|
|
/* PIKA - Photo and Image Kooker Application
|
|
|
|
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
|
|
|
|
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
|
|
|
|
*
|
|
|
|
* Original copyright, applying to most contents (license remains unchanged):
|
|
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
|
|
*
|
|
|
|
* metadata-editor.c
|
|
|
|
* Copyright (C) 2016, 2017 Ben Touchette
|
|
|
|
*
|
|
|
|
* 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 3 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, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include <gegl.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <gexiv2/gexiv2.h>
|
|
|
|
|
|
|
|
#include <glib.h>
|
|
|
|
#include <glib/gstdio.h>
|
|
|
|
|
|
|
|
#include <libpika/pika.h>
|
|
|
|
#include <libpika/pikaui.h>
|
|
|
|
|
|
|
|
#include "libpika/stdplugins-intl.h"
|
|
|
|
|
|
|
|
#include "metadata-tags.h"
|
|
|
|
#include "metadata-xml.h"
|
|
|
|
#include "metadata-impexp.h"
|
|
|
|
#include "metadata-misc.h"
|
|
|
|
#include "metadata-editor.h"
|
|
|
|
|
|
|
|
#define PLUG_IN_PROC "plug-in-metadata-editor"
|
|
|
|
#define PLUG_IN_BINARY "metadata-editor"
|
|
|
|
#define PLUG_IN_ROLE "pika-metadata"
|
|
|
|
|
|
|
|
#define DEFAULT_TEMPLATE_FILE "pika_metadata_template.xml"
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct _Metadata Metadata;
|
|
|
|
typedef struct _MetadataClass MetadataClass;
|
|
|
|
|
|
|
|
struct _Metadata
|
|
|
|
{
|
|
|
|
PikaPlugIn parent_instance;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _MetadataClass
|
|
|
|
{
|
|
|
|
PikaPlugInClass parent_class;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#define METADATA_TYPE (metadata_get_type ())
|
2023-10-30 23:55:30 +01:00
|
|
|
#define METADATA(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), METADATA_TYPE, Metadata))
|
2023-09-26 00:35:21 +02:00
|
|
|
|
|
|
|
GType metadata_get_type (void) G_GNUC_CONST;
|
|
|
|
|
|
|
|
static GList * metadata_query_procedures (PikaPlugIn *plug_in);
|
|
|
|
static PikaProcedure * metadata_create_procedure (PikaPlugIn *plug_in,
|
|
|
|
const gchar *name);
|
|
|
|
|
|
|
|
static PikaValueArray * metadata_run (PikaProcedure *procedure,
|
2023-10-30 23:55:30 +01:00
|
|
|
PikaProcedureConfig *config,
|
2023-09-26 00:35:21 +02:00
|
|
|
gpointer run_data);
|
|
|
|
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
ME_WIDGET_ENTRY,
|
|
|
|
ME_WIDGET_TEXT,
|
|
|
|
ME_WIDGET_COMBO,
|
|
|
|
ME_WIDGET_DATE_BOX, /* Entry + date select button in a box */
|
|
|
|
ME_WIDGET_EC_BOX, /* Entry + combo in a box */
|
|
|
|
ME_WIDGET_TREE_GRID, /* Treeview + add/remove row buttons in a box */
|
|
|
|
ME_WIDGET_SEPARATOR,
|
|
|
|
ME_WIDGET_OTHER,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
ME_RENDER_TEXT,
|
|
|
|
ME_RENDER_COMBO,
|
|
|
|
ME_RENDER_OTHER,
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
guint index;
|
|
|
|
gchar *label;
|
|
|
|
gint widget_type;
|
|
|
|
gchar *id;
|
|
|
|
gchar *extra_id1; /* date_box: button id; tree view: add button id */
|
|
|
|
gchar *extra_id2; /* date_box: icon name for button; tree view: remove button id */
|
|
|
|
} me_widget_info;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
gchar *label;
|
|
|
|
gint renderer_type;
|
|
|
|
} me_column_info;
|
|
|
|
|
|
|
|
static GtkWidget * metadata_editor_create_page_grid (GtkWidget *notebook,
|
|
|
|
const gchar *tab_name);
|
|
|
|
|
|
|
|
static void metadata_editor_create_widgets (const me_widget_info *widget_info,
|
|
|
|
gint n_items,
|
|
|
|
GtkWidget *grid,
|
|
|
|
metadata_editor *meta_info);
|
|
|
|
|
|
|
|
static void metadata_editor_create_tree_grid (const me_column_info *tree_info,
|
|
|
|
gint n_items,
|
|
|
|
gint grid_row,
|
|
|
|
const me_widget_info *widget_info,
|
|
|
|
GtkWidget *grid,
|
|
|
|
GtkListStore *store,
|
|
|
|
metadata_editor *meta_info);
|
|
|
|
|
|
|
|
static gboolean metadata_editor_dialog (PikaImage *image,
|
|
|
|
PikaMetadata *metadata,
|
|
|
|
GError **error);
|
|
|
|
|
|
|
|
static void metadata_dialog_editor_set_metadata (GExiv2Metadata *metadata,
|
|
|
|
metadata_editor *meta_info);
|
|
|
|
|
|
|
|
static void impex_combo_callback (GtkComboBoxText *combo,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void gpsaltsys_combo_callback (GtkComboBoxText *combo,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void remove_substring (const gchar *string,
|
|
|
|
const gchar *substring);
|
|
|
|
|
|
|
|
static gchar * clean_xmp_string (const gchar *value);
|
|
|
|
static gchar ** split_metadata_string (gchar *value);
|
|
|
|
static void add_to_store (gchar *value,
|
|
|
|
GtkListStore *liststore,
|
|
|
|
gint store_column);
|
|
|
|
|
|
|
|
static void set_tag_string (PikaMetadata *metadata,
|
|
|
|
const gchar *name,
|
|
|
|
const gchar *value,
|
|
|
|
gboolean clear_tag);
|
|
|
|
|
|
|
|
static gchar * get_phonetype (gchar *cur_value);
|
|
|
|
|
|
|
|
static void write_metadata_tag (metadata_editor *meta_info,
|
|
|
|
PikaMetadata *metadata,
|
|
|
|
gchar *tag,
|
|
|
|
gint data_column);
|
|
|
|
|
|
|
|
static void write_metadata_tag_multiple (metadata_editor *meta_info,
|
|
|
|
PikaMetadata *metadata,
|
|
|
|
GExiv2StructureType type,
|
|
|
|
const gchar *header_tag,
|
|
|
|
gint n_columns,
|
|
|
|
const gchar **column_tags,
|
|
|
|
const gint special_handling[]);
|
|
|
|
|
|
|
|
gboolean hasCreatorTagData (metadata_editor *meta_info);
|
|
|
|
gboolean hasLocationCreationTagData (metadata_editor *meta_info);
|
|
|
|
gboolean hasImageSupplierTagData (metadata_editor *meta_info);
|
|
|
|
|
|
|
|
void on_date_button_clicked (GtkButton *widget,
|
|
|
|
GtkWidget *entry_widget,
|
|
|
|
gchar *tag);
|
|
|
|
|
|
|
|
void on_create_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
void on_patient_dob_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
void on_study_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
void on_series_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
property_release_id_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
property_release_id_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
model_release_id_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
model_release_id_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
shown_location_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
shown_location_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
feat_org_name_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
feat_org_name_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
feat_org_code_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
feat_org_code_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
artwork_object_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
artwork_object_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
reg_entry_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
reg_entry_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
image_creator_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
image_creator_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
copyright_own_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
copyright_own_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
licensor_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
static void
|
|
|
|
licensor_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
list_row_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data,
|
|
|
|
gchar *tag);
|
|
|
|
|
|
|
|
static void
|
|
|
|
list_row_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data,
|
|
|
|
gchar *tag);
|
|
|
|
|
|
|
|
static gint
|
|
|
|
count_tags (GExiv2Metadata *metadata,
|
|
|
|
const gchar *header,
|
|
|
|
const gchar **tags,
|
|
|
|
int items);
|
|
|
|
|
|
|
|
static gchar **
|
|
|
|
get_tags (GExiv2Metadata *metadata,
|
|
|
|
const gchar *header,
|
|
|
|
const gchar **tags,
|
|
|
|
const int items,
|
|
|
|
const int count);
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_tagdata (gchar **tagdata,
|
|
|
|
gint rows,
|
|
|
|
gint cols);
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasModelReleaseTagData (metadata_editor *meta_info);
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasPropertyReleaseTagData (metadata_editor *meta_info);
|
|
|
|
|
|
|
|
static void
|
|
|
|
organisation_image_code_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
organisation_image_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
prop_rel_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_sub_loc_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_city_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_state_prov_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_cntry_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_cntry_iso_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_org_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_item_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_title_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_copyright_notice_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_source_inv_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_source_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_creator_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_date_creat_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
cr_owner_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
cr_owner_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone1_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone_type1_cell_edited_callback (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone2_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone_type2_cell_edited_callback (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_email_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_web_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data);
|
|
|
|
|
|
|
|
void
|
|
|
|
cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data,
|
|
|
|
int index);
|
|
|
|
|
|
|
|
void
|
|
|
|
cell_edited_callback_combo (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data,
|
|
|
|
int index);
|
|
|
|
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (Metadata, metadata, PIKA_TYPE_PLUG_IN)
|
|
|
|
|
|
|
|
PIKA_MAIN (METADATA_TYPE)
|
|
|
|
DEFINE_STD_SET_I18N
|
|
|
|
|
|
|
|
|
|
|
|
static int last_gpsaltsys_sel;
|
|
|
|
|
|
|
|
gboolean pikametadata;
|
|
|
|
gboolean force_write;
|
|
|
|
|
|
|
|
static const gchar *lang_default = "lang=\"x-default\"";
|
|
|
|
static const gchar *seq_default = "type=\"Seq\"";
|
|
|
|
static const gchar *bag_default = "type=\"Bag\"";
|
|
|
|
|
|
|
|
metadata_editor meta_args;
|
|
|
|
|
|
|
|
#define ME_LOG_DOMAIN "metadata-editor"
|
|
|
|
|
|
|
|
|
|
|
|
/* Widget creation data */
|
|
|
|
|
|
|
|
static const me_widget_info description_tab_data[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Document Title"), ME_WIDGET_ENTRY, "Xmp.dc.title" },
|
|
|
|
{ 1, N_("Author"), ME_WIDGET_TEXT, "Xmp.dc.creator" },
|
|
|
|
{ 2, N_("Author Title"), ME_WIDGET_ENTRY, "Xmp.photoshop.AuthorsPosition" },
|
|
|
|
{ 3, N_("Description"), ME_WIDGET_TEXT, "Xmp.dc.description" },
|
|
|
|
{ 4, N_("Description Writer"), ME_WIDGET_ENTRY, "Xmp.photoshop.CaptionWriter" },
|
|
|
|
{ 5, N_("Rating"), ME_WIDGET_COMBO, "Xmp.xmp.Rating" },
|
|
|
|
{ 6, N_("Keywords"), ME_WIDGET_TEXT, "Xmp.dc.subject" },
|
|
|
|
{ 7, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 8, N_("Copyright Status"), ME_WIDGET_COMBO, "Xmp.xmpRights.Marked" },
|
|
|
|
{ 9, N_("Copyright Notice"), ME_WIDGET_ENTRY, "Xmp.dc.rights" },
|
|
|
|
{ 10, N_("Copyright URL"), ME_WIDGET_ENTRY, "Xmp.xmpRights.WebStatement" },
|
|
|
|
};
|
|
|
|
static const gint n_description_tab_data = G_N_ELEMENTS (description_tab_data);
|
|
|
|
|
|
|
|
static const me_widget_info iptc_tab_data[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Address"), ME_WIDGET_TEXT, "Xmp.iptc.CiAdrExtadr" },
|
|
|
|
{ 1, N_("City"), ME_WIDGET_ENTRY, "Xmp.iptc.CiAdrCity" },
|
|
|
|
{ 2, N_("State / Province"), ME_WIDGET_ENTRY, "Xmp.iptc.CiAdrRegion" },
|
|
|
|
{ 3, N_("Postal Code"), ME_WIDGET_ENTRY, "Xmp.iptc.CiAdrPcode" },
|
|
|
|
{ 4, N_("Country"), ME_WIDGET_ENTRY, "Xmp.iptc.CiAdrCtry" },
|
|
|
|
{ 5, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 6, N_("Phone(s)"), ME_WIDGET_TEXT, "Xmp.iptc.CiTelWork" },
|
|
|
|
{ 7, N_("E-mail(s)"), ME_WIDGET_TEXT, "Xmp.iptc.CiEmailWork" },
|
|
|
|
{ 8, N_("Website(s)"), ME_WIDGET_TEXT, "Xmp.iptc.CiUrlWork" },
|
|
|
|
{ 9, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 10, N_("Creation Date"), ME_WIDGET_DATE_BOX, "Xmp.photoshop.DateCreated",
|
|
|
|
"create_date_button", "pika-grid" },
|
|
|
|
{ 11, N_("Intellectual Genre"), ME_WIDGET_ENTRY, "Xmp.iptc.IntellectualGenre" },
|
|
|
|
{ 12, N_("IPTC Scene Code"), ME_WIDGET_TEXT, "Xmp.iptc.Scene" },
|
|
|
|
{ 13, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 14, N_("Sublocation"), ME_WIDGET_ENTRY, "Xmp.iptc.Location" },
|
|
|
|
{ 15, N_("City"), ME_WIDGET_ENTRY, "Xmp.photoshop.City" },
|
|
|
|
{ 16, N_("State / Province"), ME_WIDGET_ENTRY, "Xmp.photoshop.State" },
|
|
|
|
{ 17, N_("Country"), ME_WIDGET_ENTRY, "Xmp.photoshop.Country" },
|
|
|
|
{ 18, N_("Country ISO-Code"), ME_WIDGET_ENTRY, "Xmp.iptc.CountryCode" },
|
|
|
|
{ 19, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 20, N_("Urgency"), ME_WIDGET_COMBO, "Xmp.photoshop.Urgency" },
|
|
|
|
{ 21, N_("Headline"), ME_WIDGET_TEXT, "Xmp.photoshop.Headline" },
|
|
|
|
{ 22, N_("IPTC Subject Code"), ME_WIDGET_TEXT, "Xmp.iptc.SubjectCode" },
|
|
|
|
{ 23, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 24, N_("Job Identifier"), ME_WIDGET_ENTRY, "Xmp.photoshop.TransmissionReference" },
|
|
|
|
{ 25, N_("Instructions"), ME_WIDGET_TEXT, "Xmp.photoshop.Instructions" },
|
|
|
|
{ 26, N_("Credit Line"), ME_WIDGET_ENTRY, "Xmp.photoshop.Credit" },
|
|
|
|
{ 27, N_("Source"), ME_WIDGET_ENTRY, "Xmp.photoshop.Source" },
|
|
|
|
{ 28, N_("Usage Terms"), ME_WIDGET_TEXT, "Xmp.xmpRights.UsageTerms" },
|
|
|
|
};
|
|
|
|
static const gint n_iptc_tab_data = G_N_ELEMENTS (iptc_tab_data);
|
|
|
|
|
|
|
|
static const me_widget_info iptc_extension_tab_data[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Person Shown"), ME_WIDGET_TEXT, "Xmp.iptcExt.PersonInImage" },
|
|
|
|
{ 1, N_("Sublocation"), ME_WIDGET_ENTRY, "Xmp.iptcExt.Sublocation" },
|
|
|
|
{ 2, N_("City"), ME_WIDGET_ENTRY, "Xmp.iptcExt.City" },
|
|
|
|
{ 3, N_("State / Province"), ME_WIDGET_ENTRY, "Xmp.iptcExt.ProvinceState" },
|
|
|
|
{ 4, N_("Country"), ME_WIDGET_ENTRY, "Xmp.iptcExt.CountryName" },
|
|
|
|
{ 5, N_("Country ISO-Code"), ME_WIDGET_ENTRY, "Xmp.iptcExt.CountryCode" },
|
|
|
|
{ 6, N_("World Region"), ME_WIDGET_ENTRY, "Xmp.iptcExt.WorldRegion" },
|
|
|
|
{ 7, N_("Location Shown"), ME_WIDGET_TREE_GRID, "Xmp.iptcExt.LocationShown",
|
|
|
|
"add_shown_location_button", "rem_shown_location_button" },
|
|
|
|
{ 8, N_("Featured Organization"), ME_WIDGET_TREE_GRID, "Xmp.iptcExt.OrganisationInImageName",
|
|
|
|
"add_feat_org_name_button", "rem_feat_org_name_button" },
|
|
|
|
{ 9, N_("Organization Code"), ME_WIDGET_TREE_GRID, "Xmp.iptcExt.OrganisationInImageCode",
|
|
|
|
"add_feat_org_code_button", "rem_feat_org_code_button" },
|
|
|
|
{ 10, N_("Event"), ME_WIDGET_ENTRY, "Xmp.iptcExt.Event" },
|
|
|
|
{ 11, N_("Artwork or Object"), ME_WIDGET_TREE_GRID, "Xmp.iptcExt.ArtworkOrObject",
|
|
|
|
"add_artwork_object_button", "rem_artwork_object_button" },
|
|
|
|
{ 12, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 13, N_("Additional Model Info"), ME_WIDGET_TEXT, "Xmp.iptcExt.AddlModelInfo" },
|
|
|
|
{ 14, N_("Model Age"), ME_WIDGET_TEXT, "Xmp.iptcExt.ModelAge" },
|
|
|
|
{ 15, N_("Minor Model Age Disclosure"), ME_WIDGET_COMBO, "Xmp.plus.MinorModelAgeDisclosure" },
|
|
|
|
{ 16, N_("Model Release Status"), ME_WIDGET_COMBO, "Xmp.plus.ModelReleaseStatus" },
|
|
|
|
{ 17, N_("Model Release Identifier"), ME_WIDGET_TREE_GRID, "Xmp.plus.ModelReleaseID",
|
|
|
|
"add_model_rel_id_button", "rem_model_rel_id_button" },
|
|
|
|
{ 18, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 19, N_("Image Supplier Name"), ME_WIDGET_ENTRY, "Xmp.plus.ImageSupplierName" },
|
|
|
|
{ 20, N_("Image Supplier ID"), ME_WIDGET_ENTRY, "Xmp.plus.ImageSupplierID" },
|
|
|
|
{ 21, N_("Supplier's Image ID"), ME_WIDGET_ENTRY, "Xmp.plus.ImageSupplierImageID" },
|
|
|
|
{ 22, N_("Registry Entry"), ME_WIDGET_TREE_GRID, "Xmp.iptcExt.RegistryId",
|
|
|
|
"add_reg_entry_button", "rem_reg_entry_button" },
|
|
|
|
{ 23, N_("Max. Available Width"), ME_WIDGET_ENTRY, "Xmp.iptcExt.MaxAvailWidth" },
|
|
|
|
{ 24, N_("Max. Available Height"), ME_WIDGET_ENTRY, "Xmp.iptcExt.MaxAvailHeight" },
|
|
|
|
{ 25, N_("Digital Source Type"), ME_WIDGET_COMBO, "Xmp.iptcExt.DigitalSourceType" },
|
|
|
|
{ 26, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 27, N_("Image Creator"), ME_WIDGET_TREE_GRID, "Xmp.plus.ImageCreator",
|
|
|
|
"add_image_creator_button", "rem_image_creator_button" },
|
|
|
|
{ 28, N_("Copyright Owner"), ME_WIDGET_TREE_GRID, "Xmp.plus.CopyrightOwner",
|
|
|
|
"add_copyright_own_button", "rem_copyright_own_button" },
|
|
|
|
{ 29, N_("Licensor"), ME_WIDGET_TREE_GRID, "Xmp.plus.Licensor",
|
|
|
|
"add_licensor_button", "rem_licensor_button" },
|
|
|
|
{ 30, N_("Property Release Status"), ME_WIDGET_COMBO, "Xmp.plus.PropertyReleaseStatus" },
|
|
|
|
{ 31, N_("Property Release Identifier"), ME_WIDGET_TREE_GRID, "Xmp.plus.PropertyReleaseID",
|
|
|
|
"add_prop_rel_id_button", "rem_prop_rel_id_button" },
|
|
|
|
};
|
|
|
|
static const gint n_iptc_extension_tab_data = G_N_ELEMENTS (iptc_extension_tab_data);
|
|
|
|
|
|
|
|
/* ME_WIDGET_TREE_GRID indexes */
|
|
|
|
#define C_LOCATION_SHOWN 7
|
|
|
|
#define C_FEATURED_ORG 8
|
|
|
|
#define C_FEATURED_ORG_CODE 9
|
|
|
|
#define C_ART_OBJECT 11
|
|
|
|
#define C_MODEL_RELEASE_ID 17
|
|
|
|
#define C_REGISTRY_ENTRY 22
|
|
|
|
#define C_IMAGE_CREATOR 27
|
|
|
|
#define C_COPYRIGHT_OWNER 28
|
|
|
|
#define C_LICENSOR 29
|
|
|
|
#define C_PROPERTY_RELEASE_ID 31
|
|
|
|
|
|
|
|
static const me_column_info location_shown_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Sublocation"), ME_RENDER_TEXT },
|
|
|
|
{ N_("City"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Province / State"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Country"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Country ISO Code"), ME_RENDER_TEXT },
|
|
|
|
{ N_("World Region"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_location_shown_info = G_N_ELEMENTS (location_shown_info);
|
|
|
|
|
|
|
|
static const me_column_info featured_organization_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Name"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_featured_organization_info = G_N_ELEMENTS (featured_organization_info);
|
|
|
|
|
|
|
|
static const me_column_info featured_organization_code_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Code"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_featured_organization_code_info = G_N_ELEMENTS (featured_organization_code_info);
|
|
|
|
|
|
|
|
static const me_column_info artwork_object_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Title"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Date Created"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Creator"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Source"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Source Inventory ID"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Copyright Notice"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_artwork_object_info = G_N_ELEMENTS (artwork_object_info);
|
|
|
|
|
|
|
|
static const me_column_info model_release_id_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Model Release Identifier"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_model_release_id_info = G_N_ELEMENTS (model_release_id_info);
|
|
|
|
|
|
|
|
static const me_column_info registry_entry_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Organization Identifier"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Item Identifier"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_registry_entry_info = G_N_ELEMENTS (registry_entry_info);
|
|
|
|
|
|
|
|
static const me_column_info image_creator_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Name"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Identifier"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_image_creator_info = G_N_ELEMENTS (image_creator_info);
|
|
|
|
|
|
|
|
static const me_column_info copyright_owner_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Name"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Identifier"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_copyright_owner_info = G_N_ELEMENTS (copyright_owner_info);
|
|
|
|
|
|
|
|
static const me_column_info licensor_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Name"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Identifier"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Phone Number 1"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Phone Type 1"), ME_RENDER_COMBO },
|
|
|
|
{ N_("Phone Number 2"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Phone Type 2"), ME_RENDER_COMBO },
|
|
|
|
{ N_("Email Address"), ME_RENDER_TEXT },
|
|
|
|
{ N_("Web Address"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_licensor_info = G_N_ELEMENTS (licensor_info);
|
|
|
|
|
|
|
|
static const me_column_info property_release_id_info[] =
|
|
|
|
{
|
|
|
|
{ N_("Identifier"), ME_RENDER_TEXT },
|
|
|
|
};
|
|
|
|
static const gint n_property_release_id_info = G_N_ELEMENTS (property_release_id_info);
|
|
|
|
|
|
|
|
|
|
|
|
static const me_widget_info categories_labels[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Category"), ME_WIDGET_ENTRY, "Xmp.photoshop.Category" },
|
|
|
|
{ 1, N_("Supplemental Category"), ME_WIDGET_TEXT, "Xmp.photoshop.SupplementalCategories" },
|
|
|
|
};
|
|
|
|
static const gint n_categories_labels = G_N_ELEMENTS (categories_labels);
|
|
|
|
|
|
|
|
static const me_widget_info gps_labels[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Longitude"), ME_WIDGET_ENTRY, "Exif.GPSInfo.GPSLongitude" },
|
|
|
|
{ 1, N_("Longitude Reference"), ME_WIDGET_COMBO, "Exif.GPSInfo.GPSLongitudeRef" },
|
|
|
|
{ 2, N_("Latitude"), ME_WIDGET_ENTRY, "Exif.GPSInfo.GPSLatitude" },
|
|
|
|
{ 3, N_("Latitude Reference"), ME_WIDGET_COMBO, "Exif.GPSInfo.GPSLatitudeRef" },
|
|
|
|
{ 4, N_("Altitude"), ME_WIDGET_EC_BOX, "Exif.GPSInfo.GPSAltitude",
|
|
|
|
"GPSAltitudeSystem" },
|
|
|
|
{ 5, N_("Altitude Reference"), ME_WIDGET_COMBO, "Exif.GPSInfo.GPSAltitudeRef" },
|
|
|
|
};
|
|
|
|
static const gint n_gps_labels = G_N_ELEMENTS (gps_labels);
|
|
|
|
|
|
|
|
static const me_widget_info dicom_info_labels[] =
|
|
|
|
{
|
|
|
|
{ 0, N_("Patient"), ME_WIDGET_ENTRY, "Xmp.DICOM.PatientName" },
|
|
|
|
{ 1, N_("Patient ID"), ME_WIDGET_ENTRY, "Xmp.DICOM.PatientID" },
|
|
|
|
{ 2, N_("Date of Birth"), ME_WIDGET_DATE_BOX, "Xmp.DICOM.PatientDOB",
|
|
|
|
"dob_date_button", "pika-grid" },
|
|
|
|
{ 3, N_("Patient Sex"), ME_WIDGET_COMBO, "Xmp.DICOM.PatientSex" },
|
|
|
|
{ 4, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 5, N_("Study ID"), ME_WIDGET_ENTRY, "Xmp.DICOM.StudyID" },
|
|
|
|
{ 6, N_("Referring Physician"), ME_WIDGET_ENTRY, "Xmp.DICOM.StudyPhysician" },
|
|
|
|
{ 7, N_("Study Date"), ME_WIDGET_DATE_BOX, "Xmp.DICOM.StudyDateTime",
|
|
|
|
"study_date_button", "pika-grid" },
|
|
|
|
{ 8, N_("Study Description"), ME_WIDGET_TEXT , "Xmp.DICOM.StudyDescription" },
|
|
|
|
{ 9, N_("Series Number"), ME_WIDGET_ENTRY, "Xmp.DICOM.SeriesNumber" },
|
|
|
|
{ 10, N_("Modality"), ME_WIDGET_ENTRY, "Xmp.DICOM.SeriesModality" },
|
|
|
|
{ 11, N_("Series Date"), ME_WIDGET_DATE_BOX, "Xmp.DICOM.SeriesDateTime",
|
|
|
|
"series_date_button", "pika-grid" },
|
|
|
|
{ 12, N_("Series Description"), ME_WIDGET_TEXT , "Xmp.DICOM.SeriesDescription" },
|
|
|
|
{ 13, "", ME_WIDGET_SEPARATOR, "" },
|
|
|
|
{ 14, N_("Equipment Institution"), ME_WIDGET_ENTRY, "Xmp.DICOM.EquipmentInstitution" },
|
|
|
|
{ 15, N_("Equipment Manufacturer"), ME_WIDGET_ENTRY, "Xmp.DICOM.EquipmentManufacturer" },
|
|
|
|
};
|
|
|
|
static const gint n_dicom_info_labels = G_N_ELEMENTS (dicom_info_labels);
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
metadata_class_init (MetadataClass *klass)
|
|
|
|
{
|
|
|
|
PikaPlugInClass *plug_in_class = PIKA_PLUG_IN_CLASS (klass);
|
|
|
|
|
|
|
|
plug_in_class->query_procedures = metadata_query_procedures;
|
|
|
|
plug_in_class->create_procedure = metadata_create_procedure;
|
|
|
|
plug_in_class->set_i18n = STD_SET_I18N;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
metadata_init (Metadata *metadata)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static GList *
|
|
|
|
metadata_query_procedures (PikaPlugIn *plug_in)
|
|
|
|
{
|
|
|
|
return g_list_append (NULL, g_strdup (PLUG_IN_PROC));
|
|
|
|
}
|
|
|
|
|
|
|
|
static PikaProcedure *
|
|
|
|
metadata_create_procedure (PikaPlugIn *plug_in,
|
|
|
|
const gchar *name)
|
|
|
|
{
|
|
|
|
PikaProcedure *procedure = NULL;
|
|
|
|
|
|
|
|
if (! strcmp (name, PLUG_IN_PROC))
|
|
|
|
{
|
|
|
|
procedure = pika_procedure_new (plug_in, name,
|
|
|
|
PIKA_PDB_PROC_TYPE_PLUGIN,
|
|
|
|
metadata_run, NULL, NULL);
|
|
|
|
|
|
|
|
pika_procedure_set_image_types (procedure, "*");
|
|
|
|
|
|
|
|
pika_procedure_set_menu_label (procedure, _("_Edit Metadata"));
|
|
|
|
pika_procedure_add_menu_path (procedure, "<Image>/Image/Metadata");
|
|
|
|
|
|
|
|
pika_procedure_set_documentation (procedure,
|
|
|
|
_("Edit metadata (IPTC, EXIF, XMP)"),
|
|
|
|
"Edit metadata information attached "
|
|
|
|
"to the current image. Some or all "
|
|
|
|
"of this metadata will be saved in "
|
|
|
|
"the file, depending on the output "
|
|
|
|
"file format.",
|
|
|
|
name);
|
|
|
|
pika_procedure_set_attribution (procedure,
|
|
|
|
"Ben Touchette",
|
|
|
|
"Ben Touchette",
|
|
|
|
"2017");
|
|
|
|
|
|
|
|
PIKA_PROC_ARG_ENUM (procedure, "run-mode",
|
|
|
|
"Run mode",
|
|
|
|
"The run mode",
|
|
|
|
PIKA_TYPE_RUN_MODE,
|
|
|
|
PIKA_RUN_INTERACTIVE,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
|
|
|
|
PIKA_PROC_ARG_IMAGE (procedure, "image",
|
|
|
|
"Image",
|
|
|
|
"The input image",
|
|
|
|
FALSE,
|
|
|
|
G_PARAM_READWRITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return procedure;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PikaValueArray *
|
|
|
|
metadata_run (PikaProcedure *procedure,
|
2023-10-30 23:55:30 +01:00
|
|
|
PikaProcedureConfig *config,
|
2023-09-26 00:35:21 +02:00
|
|
|
gpointer run_data)
|
|
|
|
{
|
|
|
|
PikaImage *image;
|
|
|
|
PikaMetadata *metadata;
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
pika_ui_init (PLUG_IN_BINARY);
|
|
|
|
|
2023-10-30 23:55:30 +01:00
|
|
|
g_object_get (config, "image", &image, NULL);
|
2023-09-26 00:35:21 +02:00
|
|
|
|
|
|
|
metadata = pika_image_get_metadata (image);
|
|
|
|
|
|
|
|
/* Always show metadata dialog so we can add appropriate iptc data
|
|
|
|
* as needed. Sometimes license data needs to be added after the
|
|
|
|
* fact and the image may not contain metadata but should have it
|
|
|
|
* added as needed.
|
|
|
|
*/
|
|
|
|
if (! metadata)
|
|
|
|
{
|
|
|
|
metadata = pika_metadata_new ();
|
|
|
|
pika_image_set_metadata (image, metadata);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (metadata_editor_dialog (image, metadata, &error))
|
|
|
|
return pika_procedure_new_return_values (procedure, PIKA_PDB_SUCCESS, NULL);
|
|
|
|
else
|
|
|
|
return pika_procedure_new_return_values (procedure, PIKA_PDB_EXECUTION_ERROR, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ EDITOR DIALOG UI ]======================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
GtkWidget *
|
|
|
|
metadata_editor_get_widget (metadata_editor *meta_info,
|
|
|
|
const gchar *name)
|
|
|
|
{
|
|
|
|
return GTK_WIDGET (g_hash_table_lookup (meta_info->widgets, name));
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *
|
|
|
|
metadata_editor_create_page_grid (GtkWidget *notebook,
|
|
|
|
const gchar *tab_name)
|
|
|
|
{
|
|
|
|
GtkWidget *scrolled_win;
|
|
|
|
GtkWidget *viewport;
|
|
|
|
GtkWidget *box;
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *grid;
|
|
|
|
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_widget_show (box);
|
|
|
|
|
|
|
|
scrolled_win = gtk_scrolled_window_new (NULL, NULL);
|
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (scrolled_win), 6);
|
|
|
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_win),
|
|
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
|
|
gtk_box_pack_start (GTK_BOX (box), scrolled_win, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show (scrolled_win);
|
|
|
|
|
|
|
|
label = gtk_label_new (tab_name);
|
|
|
|
gtk_widget_set_margin_start (label, 2);
|
|
|
|
gtk_widget_set_margin_top (label, 2);
|
|
|
|
gtk_widget_set_margin_end (label, 2);
|
|
|
|
gtk_widget_set_margin_bottom (label, 2);
|
|
|
|
gtk_widget_set_can_focus (label, FALSE);
|
|
|
|
gtk_widget_show (label);
|
|
|
|
|
|
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (notebook), box, label);
|
|
|
|
|
|
|
|
viewport = gtk_viewport_new (NULL, NULL);
|
|
|
|
gtk_container_add (GTK_CONTAINER (scrolled_win), viewport);
|
|
|
|
gtk_widget_show (viewport);
|
|
|
|
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_container_add (GTK_CONTAINER (viewport), box);
|
|
|
|
gtk_widget_show (box);
|
|
|
|
|
|
|
|
grid = gtk_grid_new ();
|
|
|
|
gtk_widget_set_margin_bottom (grid, 5);
|
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (grid), 6);
|
|
|
|
gtk_grid_set_row_spacing (GTK_GRID (grid), 3);
|
|
|
|
gtk_grid_set_column_spacing (GTK_GRID (grid), 30);
|
|
|
|
gtk_box_pack_start (GTK_BOX (box), grid, FALSE, TRUE, 0);
|
|
|
|
gtk_widget_show (grid);
|
|
|
|
|
|
|
|
return grid;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
metadata_editor_create_widgets (const me_widget_info *widget_info,
|
|
|
|
gint n_items,
|
|
|
|
GtkWidget *grid,
|
|
|
|
metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
GtkWidget *label;
|
|
|
|
|
|
|
|
/* Labels on the left, data entry widgets on the right */
|
|
|
|
for (int i = 0; i < n_items; i++)
|
|
|
|
{
|
|
|
|
if (widget_info[i].widget_type != ME_WIDGET_SEPARATOR)
|
|
|
|
{
|
|
|
|
label = gtk_label_new (_(widget_info[i].label));
|
|
|
|
gtk_widget_set_margin_start (label, 3);
|
|
|
|
gtk_widget_set_margin_top (label, 3);
|
|
|
|
gtk_widget_set_margin_end (label, 3);
|
|
|
|
gtk_widget_set_margin_bottom (label, 3);
|
|
|
|
gtk_label_set_xalign (GTK_LABEL (label), 0.0);
|
|
|
|
if (widget_info[i].widget_type == ME_WIDGET_TEXT ||
|
|
|
|
widget_info[i].widget_type == ME_WIDGET_TREE_GRID)
|
|
|
|
gtk_label_set_yalign (GTK_LABEL (label), 0.0);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), label,
|
|
|
|
0, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
gtk_widget_show (label);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (widget_info[i].widget_type)
|
|
|
|
{
|
|
|
|
case ME_WIDGET_ENTRY:
|
|
|
|
{
|
|
|
|
GtkWidget *entry;
|
|
|
|
|
|
|
|
entry = gtk_entry_new ();
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (entry), TRUE);
|
|
|
|
gtk_widget_set_margin_end (entry, 5);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), entry,
|
|
|
|
1, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
gtk_widget_show (entry);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].id,
|
|
|
|
(gpointer) entry);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_TEXT:
|
|
|
|
{
|
|
|
|
GtkWidget *textview;
|
|
|
|
GtkWidget *scrolled_window;
|
|
|
|
|
|
|
|
scrolled_window = gtk_scrolled_window_new (NULL, NULL);
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (scrolled_window), TRUE);
|
|
|
|
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window),
|
|
|
|
GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
|
|
|
|
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolled_window),
|
|
|
|
GTK_SHADOW_IN);
|
|
|
|
gtk_widget_set_margin_end (scrolled_window, 5);
|
|
|
|
gtk_widget_show (scrolled_window);
|
|
|
|
|
|
|
|
textview = gtk_text_view_new();
|
|
|
|
gtk_container_add (GTK_CONTAINER (scrolled_window), textview);
|
|
|
|
gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (textview), GTK_WRAP_WORD);
|
|
|
|
gtk_text_view_set_left_margin (GTK_TEXT_VIEW (textview), 6);
|
|
|
|
gtk_text_view_set_right_margin (GTK_TEXT_VIEW (textview), 6);
|
|
|
|
gtk_widget_show (textview);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), scrolled_window,
|
|
|
|
1, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].id,
|
|
|
|
(gpointer) textview);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_COMBO:
|
|
|
|
{
|
|
|
|
GtkWidget *combo;
|
|
|
|
|
|
|
|
combo = gtk_combo_box_text_new ();
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (combo), TRUE);
|
|
|
|
gtk_widget_set_margin_end (combo, 5);
|
|
|
|
gtk_widget_set_can_focus (combo, FALSE);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), combo,
|
|
|
|
1, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
gtk_widget_show (combo);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].id,
|
|
|
|
(gpointer) combo);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_DATE_BOX:
|
|
|
|
{
|
|
|
|
GtkWidget *date_box, *date_entry, *button;
|
|
|
|
|
|
|
|
/* A date_entry and a button in one grid cell using a box as parent */
|
|
|
|
|
|
|
|
date_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (date_box), TRUE);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), date_box,
|
|
|
|
1, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
gtk_widget_show (date_box);
|
|
|
|
|
|
|
|
date_entry = gtk_entry_new ();
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (date_entry), TRUE);
|
|
|
|
gtk_box_pack_start (GTK_BOX (date_box), date_entry, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show (date_entry);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].id,
|
|
|
|
(gpointer) date_entry);
|
|
|
|
|
|
|
|
button = gtk_button_new_from_icon_name (widget_info[i].extra_id2,
|
|
|
|
GTK_ICON_SIZE_BUTTON);
|
|
|
|
gtk_widget_set_size_request (button, 24, 24);
|
|
|
|
gtk_widget_set_margin_start (button, 5);
|
|
|
|
gtk_widget_set_margin_end (button, 5);
|
|
|
|
gtk_widget_set_margin_bottom (button, 1);
|
|
|
|
gtk_container_add (GTK_CONTAINER (date_box), button);
|
|
|
|
gtk_widget_show (button);
|
|
|
|
g_hash_table_insert (meta_args.widgets, widget_info[i].extra_id1,
|
|
|
|
(gpointer) button);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_EC_BOX:
|
|
|
|
{
|
|
|
|
GtkWidget *ec_box, *ec_entry, *ec_combo;
|
|
|
|
|
|
|
|
/* A box containing an Entry widget and a Combo widget,
|
|
|
|
* where the combo widget is limited in size (100).
|
|
|
|
* If needed, you can change size manually after creation. */
|
|
|
|
|
|
|
|
ec_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (ec_box), TRUE);
|
|
|
|
gtk_widget_set_margin_end (ec_box, 5);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), ec_box,
|
|
|
|
1, widget_info[i].index,
|
|
|
|
1, 1);
|
|
|
|
gtk_widget_show (ec_box);
|
|
|
|
|
|
|
|
ec_entry = gtk_entry_new ();
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (ec_entry), TRUE);
|
|
|
|
gtk_box_pack_start (GTK_BOX (ec_box), ec_entry, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show (ec_entry);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].id,
|
|
|
|
(gpointer) ec_entry);
|
|
|
|
|
|
|
|
ec_combo = gtk_combo_box_text_new ();
|
|
|
|
gtk_widget_set_margin_start (ec_combo, 5);
|
|
|
|
gtk_widget_set_can_focus (ec_combo, FALSE);
|
|
|
|
g_object_set (G_OBJECT (ec_combo), "width_request", 100, NULL);
|
|
|
|
gtk_box_pack_start (GTK_BOX (ec_box), ec_combo, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show (ec_combo);
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[i].extra_id1,
|
|
|
|
(gpointer) ec_combo);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_TREE_GRID:
|
|
|
|
/* Needs to be handled separately. */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_WIDGET_SEPARATOR:
|
|
|
|
{
|
|
|
|
GtkWidget *separator;
|
|
|
|
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_HORIZONTAL);
|
|
|
|
g_object_set (G_OBJECT (separator), "margin", 8, NULL);
|
|
|
|
gtk_widget_show (separator);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), separator,
|
|
|
|
0, widget_info[i].index,
|
|
|
|
2, 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
metadata_editor_create_tree_grid (const me_column_info *tree_info,
|
|
|
|
gint n_items,
|
|
|
|
gint grid_row,
|
|
|
|
const me_widget_info *widget_info,
|
|
|
|
GtkWidget *grid,
|
|
|
|
GtkListStore *store,
|
|
|
|
metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
/* Creates a GtkBox containing a GtkTreeView and a GtkBox which holds the
|
|
|
|
add/remove buttons. */
|
|
|
|
|
|
|
|
GtkWidget *tree_box, *tree;
|
|
|
|
GtkWidget *button_box, *button;
|
|
|
|
|
|
|
|
tree_box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_widget_set_hexpand(GTK_WIDGET (tree_box), TRUE);
|
|
|
|
gtk_widget_set_margin_end (tree_box, 5);
|
|
|
|
gtk_widget_show (tree_box);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), tree_box,
|
|
|
|
1, grid_row,
|
|
|
|
1, 1);
|
|
|
|
|
|
|
|
tree = gtk_tree_view_new_with_model (GTK_TREE_MODEL (store));
|
|
|
|
gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW (tree), FALSE);
|
|
|
|
gtk_tree_view_set_grid_lines (GTK_TREE_VIEW (tree), GTK_TREE_VIEW_GRID_LINES_BOTH);
|
|
|
|
gtk_box_pack_start (GTK_BOX (tree_box), tree, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show (tree);
|
|
|
|
|
|
|
|
g_hash_table_insert (meta_info->widgets, widget_info[grid_row].id, (gpointer) tree);
|
|
|
|
|
|
|
|
for (int i = 0; i < n_items; i++)
|
|
|
|
{
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *render;
|
|
|
|
|
|
|
|
switch (tree_info[i].renderer_type)
|
|
|
|
{
|
|
|
|
case ME_RENDER_TEXT:
|
|
|
|
render = gtk_cell_renderer_text_new ();
|
|
|
|
column = gtk_tree_view_column_new_with_attributes (_(tree_info[i].label),
|
|
|
|
render,
|
|
|
|
"text", i,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ME_RENDER_COMBO:
|
|
|
|
render = gtk_cell_renderer_combo_new ();
|
|
|
|
column = gtk_tree_view_column_new_with_attributes (_(tree_info[i].label),
|
|
|
|
render,
|
|
|
|
"text", i,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_tree_view_column_set_resizable (column, TRUE);
|
|
|
|
gtk_tree_view_column_set_spacing (column, 3);
|
|
|
|
gtk_tree_view_append_column (GTK_TREE_VIEW (tree), column);
|
|
|
|
}
|
|
|
|
|
|
|
|
button_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
gtk_widget_set_margin_top (button_box, 5);
|
|
|
|
gtk_widget_set_margin_bottom (button_box, 5);
|
|
|
|
gtk_box_pack_start (GTK_BOX (tree_box), button_box, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show (button_box);
|
|
|
|
|
|
|
|
button = gtk_button_new_from_icon_name ("list-add", GTK_ICON_SIZE_BUTTON);
|
|
|
|
gtk_widget_set_size_request (button, 24, 24);
|
|
|
|
gtk_widget_set_margin_end (button, 5);
|
|
|
|
gtk_container_add (GTK_CONTAINER (button_box), button);
|
|
|
|
gtk_widget_show (button);
|
|
|
|
g_hash_table_insert (meta_args.widgets, widget_info[grid_row].extra_id1, (gpointer) button);
|
|
|
|
|
|
|
|
button = gtk_button_new_from_icon_name ("list-remove", GTK_ICON_SIZE_BUTTON);
|
|
|
|
gtk_widget_set_size_request (button, 24, 24);
|
|
|
|
gtk_widget_set_margin_start (button, 5);
|
|
|
|
gtk_container_add (GTK_CONTAINER (button_box), button);
|
|
|
|
gtk_widget_show (button);
|
|
|
|
g_hash_table_insert (meta_args.widgets, widget_info[grid_row].extra_id2, (gpointer) button);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
metadata_editor_dialog (PikaImage *image,
|
|
|
|
PikaMetadata *g_metadata,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
GExiv2Metadata *metadata;
|
|
|
|
GtkWidget *dialog;
|
|
|
|
GtkWidget *content_area;
|
|
|
|
GtkWidget *metadata_vbox;
|
|
|
|
GtkWidget *impex_combo;
|
|
|
|
GtkWidget *notebook;
|
|
|
|
GtkWidget *box;
|
|
|
|
GtkWidget *grid;
|
|
|
|
GtkWidget *widget;
|
|
|
|
GtkListStore *store;
|
|
|
|
gchar *title;
|
|
|
|
gchar *name;
|
|
|
|
|
|
|
|
metadata = GEXIV2_METADATA (g_metadata);
|
|
|
|
|
|
|
|
meta_args.image = image;
|
|
|
|
/* Default filename used for import/export */
|
|
|
|
meta_args.filename = g_strconcat (g_get_home_dir (), "/", DEFAULT_TEMPLATE_FILE,
|
|
|
|
NULL);
|
|
|
|
meta_args.widgets = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
meta_args.metadata = metadata;
|
|
|
|
|
|
|
|
name = pika_image_get_name (image);
|
|
|
|
title = g_strdup_printf (_("Metadata Editor: %s"), name);
|
|
|
|
g_free (name);
|
|
|
|
|
|
|
|
dialog = pika_dialog_new (title,
|
|
|
|
"pika-metadata-editor-dialog",
|
|
|
|
NULL, 0,
|
|
|
|
pika_standard_help_func, PLUG_IN_PROC,
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
_("_Write Metadata"), GTK_RESPONSE_OK,
|
|
|
|
NULL);
|
|
|
|
g_free (title);
|
|
|
|
|
|
|
|
gtk_widget_set_size_request(dialog, 650, 500);
|
|
|
|
|
|
|
|
meta_args.dialog = dialog;
|
|
|
|
|
|
|
|
gtk_dialog_set_default_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK);
|
|
|
|
pika_dialog_set_alternative_button_order (GTK_DIALOG (dialog),
|
|
|
|
GTK_RESPONSE_OK,
|
|
|
|
GTK_RESPONSE_CANCEL,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
pika_window_set_transient (GTK_WINDOW (dialog));
|
|
|
|
|
|
|
|
content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
|
|
|
|
|
|
|
|
metadata_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
|
|
gtk_box_pack_start (GTK_BOX (content_area), metadata_vbox, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (metadata_vbox), 12);
|
|
|
|
gtk_box_set_spacing (GTK_BOX (metadata_vbox), 6);
|
|
|
|
gtk_widget_show (metadata_vbox);
|
|
|
|
|
|
|
|
notebook = gtk_notebook_new ();
|
|
|
|
gtk_box_pack_start (GTK_BOX (metadata_vbox), notebook, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
/* Description tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("Description"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (description_tab_data, n_description_tab_data, grid, &meta_args);
|
|
|
|
|
|
|
|
/* IPTC tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("IPTC"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (iptc_tab_data, n_iptc_tab_data, grid, &meta_args);
|
|
|
|
|
|
|
|
/* IPTC Extension tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("IPTC Extension"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (iptc_extension_tab_data, n_iptc_extension_tab_data, grid, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_location_shown_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_LOCATION_SHOWN].id, ==, "Xmp.iptcExt.LocationShown");
|
|
|
|
metadata_editor_create_tree_grid (location_shown_info, n_location_shown_info,
|
|
|
|
C_LOCATION_SHOWN, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_featured_organization_info,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_FEATURED_ORG].id, ==, "Xmp.iptcExt.OrganisationInImageName");
|
|
|
|
metadata_editor_create_tree_grid (featured_organization_info, n_featured_organization_info,
|
|
|
|
C_FEATURED_ORG, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_featured_organization_code_info,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_FEATURED_ORG_CODE].id, ==, "Xmp.iptcExt.OrganisationInImageCode");
|
|
|
|
metadata_editor_create_tree_grid (featured_organization_code_info, n_featured_organization_code_info,
|
|
|
|
C_FEATURED_ORG_CODE, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_artwork_object_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_ART_OBJECT].id, ==, "Xmp.iptcExt.ArtworkOrObject");
|
|
|
|
metadata_editor_create_tree_grid (artwork_object_info, n_artwork_object_info,
|
|
|
|
C_ART_OBJECT, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_model_release_id_info,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_MODEL_RELEASE_ID].id, ==, "Xmp.plus.ModelReleaseID");
|
|
|
|
metadata_editor_create_tree_grid (model_release_id_info, n_model_release_id_info,
|
|
|
|
C_MODEL_RELEASE_ID, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_registry_entry_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_REGISTRY_ENTRY].id, ==, "Xmp.iptcExt.RegistryId");
|
|
|
|
metadata_editor_create_tree_grid (registry_entry_info, n_registry_entry_info,
|
|
|
|
C_REGISTRY_ENTRY, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_image_creator_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_IMAGE_CREATOR].id, ==, "Xmp.plus.ImageCreator");
|
|
|
|
metadata_editor_create_tree_grid (image_creator_info, n_image_creator_info,
|
|
|
|
C_IMAGE_CREATOR, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_copyright_owner_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_COPYRIGHT_OWNER].id, ==, "Xmp.plus.CopyrightOwner");
|
|
|
|
metadata_editor_create_tree_grid (copyright_owner_info, n_copyright_owner_info,
|
|
|
|
C_COPYRIGHT_OWNER, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_licensor_info,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_LICENSOR].id, ==, "Xmp.plus.Licensor");
|
|
|
|
metadata_editor_create_tree_grid (licensor_info, n_licensor_info,
|
|
|
|
C_LICENSOR, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
store = gtk_list_store_new (n_property_release_id_info,
|
|
|
|
G_TYPE_STRING);
|
|
|
|
|
|
|
|
g_assert_cmpstr (iptc_extension_tab_data[C_PROPERTY_RELEASE_ID].id, ==, "Xmp.plus.PropertyReleaseID");
|
|
|
|
metadata_editor_create_tree_grid (property_release_id_info, n_property_release_id_info,
|
|
|
|
C_PROPERTY_RELEASE_ID, iptc_extension_tab_data,
|
|
|
|
grid, store, &meta_args);
|
|
|
|
|
|
|
|
/* Categories tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("Categories"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (categories_labels, n_categories_labels, grid, &meta_args);
|
|
|
|
|
|
|
|
/* GPS tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("GPS"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (gps_labels, n_gps_labels, grid, &meta_args);
|
|
|
|
|
|
|
|
/* Update GPSAltitudeSystem combo width */
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (&meta_args, "GPSAltitudeSystem");
|
|
|
|
g_object_set (G_OBJECT (widget), "width_request", 60, NULL);
|
|
|
|
|
|
|
|
/* DICOM tab */
|
|
|
|
|
|
|
|
grid = metadata_editor_create_page_grid (notebook, _("DICOM"));
|
|
|
|
|
|
|
|
metadata_editor_create_widgets (dicom_info_labels, n_dicom_info_labels, grid, &meta_args);
|
|
|
|
|
|
|
|
/* Show notebook */
|
|
|
|
|
|
|
|
gtk_widget_show (notebook);
|
|
|
|
|
|
|
|
/* Import / Export options box */
|
|
|
|
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
|
|
gtk_box_pack_start (GTK_BOX (metadata_vbox), box, FALSE, TRUE, 0);
|
|
|
|
gtk_widget_show (box);
|
|
|
|
|
|
|
|
impex_combo = gtk_combo_box_text_new ();
|
|
|
|
g_object_set (G_OBJECT (impex_combo), "width_request", 160, NULL);
|
|
|
|
gtk_box_pack_start (GTK_BOX (box), impex_combo, FALSE, FALSE, 6);
|
|
|
|
gtk_widget_show (impex_combo);
|
|
|
|
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (impex_combo),
|
|
|
|
_("Select:"));
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (impex_combo),
|
|
|
|
_("Import metadata"));
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (impex_combo),
|
|
|
|
_("Export metadata"));
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (impex_combo), 0);
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (impex_combo),
|
|
|
|
"changed", G_CALLBACK (impex_combo_callback), &meta_args);
|
|
|
|
|
|
|
|
/* Add signals, combobox choices, and actual metadata */
|
|
|
|
|
|
|
|
metadata_dialog_editor_set_metadata (metadata, &meta_args);
|
|
|
|
|
|
|
|
if (pika_dialog_run (PIKA_DIALOG (dialog)) == GTK_RESPONSE_OK)
|
|
|
|
{
|
|
|
|
metadata_editor_write_callback (dialog, &meta_args, image);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (meta_args.filename);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ ]=====================================================
|
|
|
|
* ==[ PRIVATE FUNCTIONS ]=====================================================
|
|
|
|
* ==[ ]=====================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
remove_substring (const gchar *string,
|
|
|
|
const gchar *substring)
|
|
|
|
{
|
|
|
|
if (string != NULL && substring != NULL && substring[0] != '\0')
|
|
|
|
{
|
|
|
|
gchar *p = strstr (string, substring);
|
|
|
|
if (p)
|
|
|
|
{
|
|
|
|
memmove (p, p + strlen (substring), strlen (p + strlen (substring)) + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gchar *
|
|
|
|
clean_xmp_string (const gchar *value)
|
|
|
|
{
|
|
|
|
gchar *value_clean;
|
|
|
|
gchar *value_utf8;
|
|
|
|
|
|
|
|
value_clean = g_strdup (value);
|
|
|
|
|
|
|
|
if (strstr (value_clean, lang_default) != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, lang_default);
|
|
|
|
if (strstr (value_clean, " ") != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, " ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr (value_clean, bag_default) != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, bag_default);
|
|
|
|
if (strstr (value_clean, " ") != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, " ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strstr (value_clean, seq_default) != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, seq_default);
|
|
|
|
if (strstr (value_clean, " ") != NULL)
|
|
|
|
{
|
|
|
|
remove_substring (value_clean, " ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! g_utf8_validate (value_clean, -1, NULL))
|
|
|
|
{
|
|
|
|
value_utf8 = g_locale_to_utf8 (value_clean, -1, NULL, NULL, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
value_utf8 = g_strdup (value_clean);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (value_clean);
|
|
|
|
|
|
|
|
return value_utf8;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We split a string and accept "," and ";" as delimiters.
|
|
|
|
* The result needs to be freed with g_strfreev.
|
|
|
|
*/
|
|
|
|
static gchar **
|
|
|
|
split_metadata_string (gchar *value)
|
|
|
|
{
|
|
|
|
gchar **split;
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
/* Can't use g_strsplit_set since we work with utf-8 here. */
|
|
|
|
split = g_strsplit (g_strdelimit (value, ";", ','), ",", 0);
|
|
|
|
|
|
|
|
for (item = 0; split[item]; item++)
|
|
|
|
{
|
|
|
|
split[item] = g_strstrip(split[item]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return split;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
add_to_store (gchar *value, GtkListStore *liststore, gint store_column)
|
|
|
|
{
|
|
|
|
gchar **strings;
|
|
|
|
gint cnt = 0;
|
|
|
|
gint item;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
strings = split_metadata_string (value);
|
|
|
|
if (strings)
|
|
|
|
{
|
|
|
|
for (item = 0; strings[item]; item++)
|
|
|
|
{
|
|
|
|
if (strings[item][0] != '\0')
|
|
|
|
{
|
|
|
|
cnt++;
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
store_column, strings[item],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_strfreev(strings);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there are less than two rows, add empty ones. */
|
|
|
|
for (item = cnt; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
store_column, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gint
|
|
|
|
count_tags (GExiv2Metadata *metadata,
|
|
|
|
const gchar *header,
|
|
|
|
const gchar **tags,
|
|
|
|
gint items)
|
|
|
|
{
|
|
|
|
int tagcount;
|
|
|
|
gchar tag[256];
|
|
|
|
int row, col;
|
|
|
|
|
|
|
|
tagcount = 0;
|
|
|
|
for (row = 1; row < 256; row++)
|
|
|
|
{
|
|
|
|
for (col = 0; col < items; col++)
|
|
|
|
{
|
|
|
|
g_snprintf ((gchar *) &tag, 256, "%s[%d]", header, row);
|
|
|
|
g_snprintf ((gchar *) &tag, 256, "%s%s",
|
|
|
|
(gchar *) &tag, (gchar *) tags[col]);
|
|
|
|
if (gexiv2_metadata_try_has_tag (metadata, (gchar *) &tag, NULL))
|
|
|
|
{
|
|
|
|
tagcount++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tagcount;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gchar **
|
|
|
|
get_tags (GExiv2Metadata *metadata,
|
|
|
|
const gchar *header,
|
|
|
|
const gchar **tags,
|
|
|
|
const gint items,
|
|
|
|
const gint count)
|
|
|
|
{
|
|
|
|
gchar **tagdata;
|
|
|
|
gchar **_datarow;
|
|
|
|
gchar tag[256];
|
|
|
|
int row, col;
|
|
|
|
|
|
|
|
g_return_val_if_fail (header != NULL && tags != NULL, NULL);
|
|
|
|
g_return_val_if_fail (items > 0, NULL);
|
|
|
|
|
|
|
|
if (count <= 0)
|
|
|
|
return NULL;
|
|
|
|
tagdata = g_new0 (gchar *, count);
|
|
|
|
if (! tagdata)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (row = 1; row < count + 1; row++)
|
|
|
|
{
|
|
|
|
tagdata[row-1] = g_malloc0 (sizeof (gchar *) * items);
|
|
|
|
_datarow = (gchar **) tagdata[row-1];
|
|
|
|
for (col = 0; col < items; col++)
|
|
|
|
{
|
|
|
|
gchar *value;
|
|
|
|
|
|
|
|
g_snprintf ((gchar *) &tag, 256, "%s[%d]", header, row);
|
|
|
|
g_snprintf ((gchar *) &tag, 256, "%s%s",
|
|
|
|
(gchar *) &tag, (gchar *) tags[col]);
|
|
|
|
|
|
|
|
value = gexiv2_metadata_try_get_tag_string (metadata, (gchar *) &tag, NULL);
|
|
|
|
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "get_tags tag: %s, value: %s", (gchar *) &tag, value);
|
|
|
|
|
|
|
|
if (_datarow && value)
|
|
|
|
_datarow[col] = strdup (value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tagdata;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_tagdata(gchar **tagdata, gint rows, gint cols)
|
|
|
|
{
|
|
|
|
gint row, col;
|
|
|
|
gchar **tagdatarow;
|
|
|
|
|
|
|
|
for (row = 0; row < rows; row++)
|
|
|
|
{
|
|
|
|
tagdatarow = (gpointer) tagdata[row];
|
|
|
|
|
|
|
|
for (col = 0; col < cols; col++)
|
|
|
|
{
|
|
|
|
g_free (tagdatarow[col]);
|
|
|
|
}
|
|
|
|
g_free (tagdatarow);
|
|
|
|
}
|
|
|
|
g_free (tagdata);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ DATE CALLBACKS ]========================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
on_create_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
on_date_button_clicked (widget, (GtkWidget*)data,
|
|
|
|
"Xmp.photoshop.DateCreated");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
on_patient_dob_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
on_date_button_clicked (widget, (GtkWidget*)data,
|
|
|
|
"Xmp.DICOM.PatientDOB");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
on_study_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
on_date_button_clicked (widget, (GtkWidget*)data,
|
|
|
|
"Xmp.DICOM.StudyDateTime");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
on_series_date_button_clicked (GtkButton *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
on_date_button_clicked (widget, (GtkWidget*)data,
|
|
|
|
"Xmp.DICOM.SeriesDateTime");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
on_date_button_clicked (GtkButton *widget,
|
|
|
|
GtkWidget *entry_widget,
|
|
|
|
gchar *tag)
|
|
|
|
{
|
|
|
|
GtkWidget *calendar_dialog;
|
|
|
|
GtkWidget *calendar_content_area;
|
|
|
|
GtkWidget *calendar_vbox;
|
|
|
|
GtkWidget *calendar;
|
|
|
|
const gchar *date_text;
|
|
|
|
GDateTime *current_datetime;
|
|
|
|
guint year, month, day;
|
|
|
|
|
|
|
|
date_text = gtk_entry_get_text (GTK_ENTRY (entry_widget));
|
|
|
|
if (date_text && date_text[0] != '\0')
|
|
|
|
{
|
|
|
|
sscanf (date_text, "%u-%u-%u;", &year, &month, &day);
|
|
|
|
month--;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
current_datetime = g_date_time_new_now_local ();
|
|
|
|
year = g_date_time_get_year (current_datetime);
|
|
|
|
month = g_date_time_get_month (current_datetime) - 1;
|
|
|
|
day = g_date_time_get_day_of_month (current_datetime);
|
|
|
|
}
|
|
|
|
|
|
|
|
calendar_dialog =
|
|
|
|
gtk_dialog_new_with_buttons (_("Choose Date"),
|
|
|
|
NULL,
|
|
|
|
GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
_("Set Date"), GTK_RESPONSE_OK,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
gtk_dialog_set_default_response (GTK_DIALOG (calendar_dialog),
|
|
|
|
GTK_RESPONSE_OK);
|
|
|
|
pika_dialog_set_alternative_button_order (GTK_DIALOG (calendar_dialog),
|
|
|
|
GTK_RESPONSE_OK,
|
|
|
|
GTK_RESPONSE_CANCEL,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
pika_window_set_transient (GTK_WINDOW (calendar_dialog));
|
|
|
|
|
|
|
|
calendar_content_area = gtk_dialog_get_content_area (GTK_DIALOG (calendar_dialog));
|
|
|
|
|
|
|
|
calendar_vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 6);
|
|
|
|
gtk_container_set_border_width (GTK_CONTAINER (calendar_vbox), 6);
|
|
|
|
gtk_box_pack_start (GTK_BOX (calendar_content_area), calendar_vbox, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
calendar = gtk_calendar_new ();
|
|
|
|
gtk_calendar_select_month (GTK_CALENDAR (calendar), month, year);
|
|
|
|
gtk_calendar_select_day (GTK_CALENDAR (calendar), day);
|
|
|
|
gtk_calendar_mark_day (GTK_CALENDAR (calendar), day);
|
|
|
|
gtk_widget_show (calendar);
|
|
|
|
|
|
|
|
gtk_container_add (GTK_CONTAINER (calendar_vbox), calendar);
|
|
|
|
gtk_widget_show (calendar_vbox);
|
|
|
|
|
|
|
|
if (gtk_dialog_run (GTK_DIALOG (calendar_dialog)) == GTK_RESPONSE_OK)
|
|
|
|
{
|
|
|
|
gchar date[25];
|
|
|
|
gtk_calendar_get_date (GTK_CALENDAR (calendar), &year, &month, &day);
|
|
|
|
g_sprintf ((gchar*) &date, "%u-%02u-%02uT00:00:00+00:00", year, month+1, day);
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (entry_widget), date);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_widget_destroy (calendar_dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ SPECIAL TAGS HANDLERS ]=================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasImageSupplierTagData (metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_imageSupplierInfoTags; loop++)
|
|
|
|
{
|
|
|
|
GtkWidget *object;
|
|
|
|
const gchar *text;
|
|
|
|
|
|
|
|
object = metadata_editor_get_widget (meta_info, imageSupplierInfoTags[loop].id);
|
|
|
|
|
|
|
|
if (imageSupplierInfoTags[loop].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
text = gtk_entry_get_text (GTK_ENTRY (object));
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if (imageSupplierInfoTags[loop].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
text = gtk_entry_get_text (GTK_ENTRY (object));
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasLocationCreationTagData (metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_locationCreationInfoTags; loop++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
const gchar *text;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info, locationCreationInfoTags[loop].id);
|
|
|
|
|
|
|
|
if (locationCreationInfoTags[loop].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
text = gtk_entry_get_text (GTK_ENTRY (widget));
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasModelReleaseTagData (metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasPropertyReleaseTagData (metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
hasCreatorTagData (metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
gboolean has_data = FALSE;
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_creatorContactInfoTags; loop++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info, creatorContactInfoTags[loop].id);
|
|
|
|
|
|
|
|
if (GTK_IS_ENTRY (widget))
|
|
|
|
{
|
|
|
|
const gchar *text = gtk_entry_get_text (GTK_ENTRY (widget));
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
has_data = TRUE;
|
|
|
|
}
|
|
|
|
else if (GTK_IS_TEXT_VIEW (widget))
|
|
|
|
{
|
|
|
|
GtkTextView *text_view = GTK_TEXT_VIEW (widget);
|
|
|
|
GtkTextBuffer *buffer = gtk_text_view_get_buffer (text_view);
|
|
|
|
GtkTextIter start;
|
|
|
|
GtkTextIter end;
|
|
|
|
gchar *text;
|
|
|
|
|
|
|
|
gtk_text_buffer_get_start_iter (buffer, &start);
|
|
|
|
gtk_text_buffer_get_end_iter (buffer, &end);
|
|
|
|
|
|
|
|
text = gtk_text_buffer_get_text (buffer, &start, &end, TRUE);
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
has_data = TRUE;
|
|
|
|
|
|
|
|
if (text)
|
|
|
|
g_free (text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return has_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ SET DIALOG METADATA ]===================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* CELL EDITED */
|
|
|
|
|
|
|
|
void
|
|
|
|
cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data,
|
|
|
|
int index)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
model = (GtkTreeModel *)data;
|
|
|
|
path = gtk_tree_path_new_from_string (path_string);
|
|
|
|
|
|
|
|
gtk_tree_model_get_iter (model, &iter, path);
|
|
|
|
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter, index,
|
|
|
|
new_text, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cell_edited_callback_combo (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data,
|
|
|
|
int column)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeSelection *selection;
|
|
|
|
|
|
|
|
widget = GTK_WIDGET (data);
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (widget));
|
|
|
|
|
|
|
|
if (gtk_tree_selection_get_selected (GTK_TREE_SELECTION (selection),
|
|
|
|
NULL, &iter))
|
|
|
|
{
|
|
|
|
path = gtk_tree_model_get_path (treemodel, &iter);
|
|
|
|
gtk_tree_path_free (path);
|
|
|
|
gtk_list_store_set (liststore, &iter, column, new_text, -1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone1_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone_type1_cell_edited_callback (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback_combo (cell, path_string, new_text, data, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone2_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_phone_type2_cell_edited_callback (GtkCellRendererCombo *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback_combo (cell, path_string, new_text, data, 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_email_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_web_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 7);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cr_owner_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cr_owner_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
img_cr8_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
img_cr8_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_copyright_notice_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_source_inv_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_source_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_creator_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_date_creat_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
aoo_title_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_org_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_item_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_sub_loc_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_city_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_state_prov_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_cntry_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_cntry_iso_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
loc_sho_wrld_reg_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
cell_edited_callback (cell, path_string, new_text, data, 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
prop_rel_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint column;
|
|
|
|
model = (GtkTreeModel *)data;
|
|
|
|
path = gtk_tree_path_new_from_string (path_string);
|
|
|
|
|
|
|
|
column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "column"));
|
|
|
|
|
|
|
|
gtk_tree_model_get_iter (model, &iter, path);
|
|
|
|
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
|
|
|
|
new_text, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
mod_rel_id_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint column;
|
|
|
|
|
|
|
|
model = (GtkTreeModel *)data;
|
|
|
|
path = gtk_tree_path_new_from_string (path_string);
|
|
|
|
|
|
|
|
column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "column"));
|
|
|
|
|
|
|
|
gtk_tree_model_get_iter (model, &iter, path);
|
|
|
|
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
|
|
|
|
new_text, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
organisation_image_name_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint column;
|
|
|
|
|
|
|
|
model = (GtkTreeModel *)data;
|
|
|
|
path = gtk_tree_path_new_from_string (path_string);
|
|
|
|
|
|
|
|
column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "column"));
|
|
|
|
|
|
|
|
gtk_tree_model_get_iter (model, &iter, path);
|
|
|
|
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
|
|
|
|
new_text, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
organisation_image_code_cell_edited_callback (GtkCellRendererText *cell,
|
|
|
|
const gchar *path_string,
|
|
|
|
const gchar *new_text,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreePath *path;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint column;
|
|
|
|
|
|
|
|
model = (GtkTreeModel *)data;
|
|
|
|
path = gtk_tree_path_new_from_string (path_string);
|
|
|
|
|
|
|
|
column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "column"));
|
|
|
|
|
|
|
|
gtk_tree_model_get_iter (model, &iter, path);
|
|
|
|
|
|
|
|
gtk_list_store_set (GTK_LIST_STORE (model), &iter, column,
|
|
|
|
new_text, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* CELL / ROW REMOVE */
|
|
|
|
|
|
|
|
static void
|
|
|
|
list_row_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data,
|
|
|
|
gchar *tag)
|
|
|
|
{
|
|
|
|
metadata_editor *meta_info = data;
|
|
|
|
GtkWidget *list_widget;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkTreeSelection *selection;
|
|
|
|
GtkTreePath *path;
|
|
|
|
|
|
|
|
list_widget = metadata_editor_get_widget (meta_info, tag);
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (list_widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
selection = gtk_tree_view_get_selection ((GtkTreeView *)list_widget);
|
|
|
|
|
|
|
|
if (gtk_tree_selection_get_selected (selection, NULL, &iter))
|
|
|
|
{
|
|
|
|
gint number_of_rows;
|
|
|
|
|
|
|
|
path = gtk_tree_model_get_path (treemodel, &iter);
|
|
|
|
gtk_list_store_remove (liststore, &iter);
|
|
|
|
gtk_tree_path_free (path);
|
|
|
|
|
|
|
|
number_of_rows = gtk_tree_model_iter_n_children (treemodel, NULL);
|
|
|
|
/* Make sur that two rows are always showing, else it looks ugly. */
|
|
|
|
if (number_of_rows < 2)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
property_release_id_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.plus.PropertyReleaseID");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
model_release_id_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.plus.ModelReleaseID");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
shown_location_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.iptcExt.LocationShown");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
feat_org_name_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.iptcExt.OrganisationInImageName");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
feat_org_code_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.iptcExt.OrganisationInImageCode");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
artwork_object_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.iptcExt.ArtworkOrObject");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_entry_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.iptcExt.RegistryId");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
image_creator_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.plus.ImageCreator");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
copyright_own_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.plus.CopyrightOwner");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_remove_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_remove_callback (widget, data, "Xmp.plus.Licensor");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* CELL / ROW ADD */
|
|
|
|
|
|
|
|
static void
|
|
|
|
list_row_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data,
|
|
|
|
gchar *tag)
|
|
|
|
{
|
|
|
|
metadata_editor *meta_info = data;
|
|
|
|
GtkWidget *list_widget;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
list_widget = metadata_editor_get_widget (meta_info, tag);
|
|
|
|
|
|
|
|
liststore = GTK_LIST_STORE (gtk_tree_view_get_model (GTK_TREE_VIEW (list_widget)));
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
property_release_id_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.plus.PropertyReleaseID");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
model_release_id_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.plus.ModelReleaseID");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
shown_location_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.iptcExt.LocationShown");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
feat_org_name_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.iptcExt.OrganisationInImageName");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
feat_org_code_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.iptcExt.OrganisationInImageCode");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
artwork_object_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.iptcExt.ArtworkOrObject");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
reg_entry_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.iptcExt.RegistryId");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
image_creator_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.plus.ImageCreator");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
copyright_own_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.plus.CopyrightOwner");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
licensor_add_callback (GtkWidget *widget,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
list_row_add_callback (widget, data, "Xmp.plus.Licensor");
|
|
|
|
}
|
|
|
|
|
|
|
|
const gchar *gpstooltips[] =
|
|
|
|
{
|
|
|
|
N_ ("Enter or edit GPS value here.\n"
|
|
|
|
"Valid values consist of 1, 2 or 3 numbers "
|
|
|
|
"(degrees, minutes, seconds), see the following examples:\n"
|
|
|
|
"10deg 15' 20\", or 10\u00b0 15' 20\", or 10:15:20.45, or "
|
|
|
|
"10 15 20, or 10 15.30, or 10.45\n"
|
|
|
|
"Delete all text to remove the current value."),
|
|
|
|
N_ ("Enter or edit GPS altitude value here.\n"
|
|
|
|
"A valid value consists of one number:\n"
|
|
|
|
"e.g. 100, or 12.24\n"
|
|
|
|
"Depending on the selected measurement type "
|
|
|
|
"the value should be entered in meter (m) "
|
|
|
|
"or feet (ft)\n"
|
|
|
|
"Delete all text to remove the current value.")
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
GPS_LONG_LAT_TOOLTIP,
|
|
|
|
GPS_ALTITUDE_TOOLTIP,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Set dialog display settings and data */
|
|
|
|
|
|
|
|
static void
|
|
|
|
metadata_dialog_editor_set_metadata (GExiv2Metadata *metadata,
|
|
|
|
metadata_editor *meta_info)
|
|
|
|
{
|
|
|
|
GtkWidget *combo_widget;
|
|
|
|
GtkWidget *entry_widget;
|
|
|
|
GtkWidget *button_widget;
|
|
|
|
gint width, height;
|
|
|
|
gchar *value;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
gint32 numele = n_default_metadata_tags;
|
|
|
|
|
|
|
|
/* Setup Buttons */
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_licensor_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (licensor_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_licensor_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (licensor_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_copyright_own_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (copyright_own_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_copyright_own_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (copyright_own_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_image_creator_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (image_creator_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_image_creator_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (image_creator_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_reg_entry_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (reg_entry_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_reg_entry_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (reg_entry_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_artwork_object_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (artwork_object_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_artwork_object_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (artwork_object_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_feat_org_code_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (feat_org_code_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_feat_org_code_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (feat_org_code_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_feat_org_name_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (feat_org_name_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_feat_org_name_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (feat_org_name_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_shown_location_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (shown_location_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_shown_location_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (shown_location_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_model_rel_id_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (model_release_id_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_model_rel_id_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (model_release_id_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "add_prop_rel_id_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (property_release_id_add_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
button_widget = metadata_editor_get_widget (meta_info, "rem_prop_rel_id_button");
|
|
|
|
g_signal_connect (G_OBJECT (button_widget), "clicked",
|
|
|
|
G_CALLBACK (property_release_id_remove_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
/* Setup Comboboxes */
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.xmp.Rating");
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
_("Unrated"));
|
|
|
|
for (i = 1; i < 6; i++)
|
|
|
|
{
|
|
|
|
gchar *display = g_strdup_printf ("%d", i);
|
|
|
|
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
display);
|
|
|
|
g_free (display);
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.xmpRights.Marked");
|
|
|
|
for (i = 0; i < n_marked; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (marked[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.photoshop.Urgency");
|
|
|
|
for (i = 0; i < n_urgency; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (urgency[i]));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.plus.MinorModelAgeDisclosure");
|
|
|
|
for (i = 0; i < n_minormodelagedisclosure; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (minormodelagedisclosure[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.plus.ModelReleaseStatus");
|
|
|
|
for (i = 0; i < n_modelreleasestatus; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (modelreleasestatus[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
gtk_widget_get_size_request (combo_widget, &width, &height);
|
|
|
|
gtk_widget_set_size_request (combo_widget, 180, height);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.iptcExt.DigitalSourceType");
|
|
|
|
for (i = 0; i < n_digitalsourcetype; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (digitalsourcetype[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.plus.PropertyReleaseStatus");
|
|
|
|
for (i = 0; i < n_propertyreleasestatus; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (propertyreleasestatus[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
gtk_widget_get_size_request (combo_widget, &width, &height);
|
|
|
|
gtk_widget_set_size_request (combo_widget, 180, height);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Xmp.DICOM.PatientSex");
|
|
|
|
for (i = 0; i < n_dicom; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (dicom[i].display));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Exif.GPSInfo.GPSLatitudeRef");
|
|
|
|
for (i = 0; i < n_gpslatref; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (gpslatref[i]));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Exif.GPSInfo.GPSLongitudeRef");
|
|
|
|
for (i = 0; i < n_gpslngref; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (gpslngref[i]));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "Exif.GPSInfo.GPSAltitudeRef");
|
|
|
|
for (i = 0; i < n_gpsaltref; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (gpsaltref[i]));
|
|
|
|
}
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info, "GPSAltitudeSystem");
|
|
|
|
for (i = 0; i < n_gpsaltsys; i++)
|
|
|
|
{
|
|
|
|
gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo_widget),
|
|
|
|
gettext (gpsaltsys[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo_widget), 0);
|
|
|
|
|
|
|
|
g_signal_connect (G_OBJECT (combo_widget), "changed",
|
|
|
|
G_CALLBACK (gpsaltsys_combo_callback),
|
|
|
|
meta_info);
|
|
|
|
|
|
|
|
/* Set up text view heights */
|
|
|
|
|
|
|
|
/* Set up lists */
|
|
|
|
for (i = 0; i < n_imageSupplierInfoTags; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info,
|
|
|
|
imageSupplierInfoTags[i].id);
|
|
|
|
|
|
|
|
value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
imageSupplierInfoTags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
gchar *value_utf;
|
|
|
|
|
|
|
|
value_utf = clean_xmp_string (value);
|
|
|
|
g_free (value);
|
|
|
|
|
|
|
|
if (imageSupplierInfoTags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), value_utf);
|
|
|
|
}
|
|
|
|
else if (imageSupplierInfoTags[i].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
GtkTextBuffer *buffer;
|
|
|
|
|
|
|
|
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
|
|
|
|
gtk_text_buffer_set_text (buffer, value_utf, -1);
|
|
|
|
}
|
|
|
|
g_free (value_utf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < n_locationCreationInfoTags; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info,
|
|
|
|
locationCreationInfoTags[i].id);
|
|
|
|
|
|
|
|
value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
locationCreationInfoTags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
gchar *value_utf;
|
|
|
|
|
|
|
|
value_utf = clean_xmp_string (value);
|
|
|
|
g_free (value);
|
|
|
|
|
|
|
|
if (locationCreationInfoTags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), value_utf);
|
|
|
|
}
|
|
|
|
g_free (value_utf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up tag data */
|
|
|
|
|
|
|
|
for (i = 0; i < numele; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
gint index;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info, default_metadata_tags[i].tag);
|
|
|
|
|
|
|
|
if (! strcmp ("Exif.GPSInfo.GPSLongitude",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gdouble gps_value;
|
|
|
|
gchar *str;
|
|
|
|
|
|
|
|
if (gexiv2_metadata_try_get_gps_longitude (metadata, &gps_value, NULL))
|
|
|
|
{
|
|
|
|
str = metadata_format_gps_longitude_latitude (gps_value);
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), str);
|
|
|
|
g_free (str);
|
|
|
|
}
|
|
|
|
gtk_widget_set_tooltip_text (widget,
|
|
|
|
gettext (gpstooltips[GPS_LONG_LAT_TOOLTIP]));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSLatitude",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gdouble gps_value;
|
|
|
|
gchar *str;
|
|
|
|
|
|
|
|
if (gexiv2_metadata_try_get_gps_latitude (metadata, &gps_value, NULL))
|
|
|
|
{
|
|
|
|
str = metadata_format_gps_longitude_latitude (gps_value);
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), str);
|
|
|
|
g_free (str);
|
|
|
|
}
|
|
|
|
gtk_widget_set_tooltip_text (widget,
|
|
|
|
gettext (gpstooltips[GPS_LONG_LAT_TOOLTIP]));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSAltitude",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gdouble gps_value;
|
|
|
|
gchar *str;
|
|
|
|
|
|
|
|
if (gexiv2_metadata_try_get_gps_altitude (metadata, &gps_value, NULL))
|
|
|
|
{
|
|
|
|
str = metadata_format_gps_altitude (gps_value, TRUE, "");
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), str);
|
|
|
|
g_free (str);
|
|
|
|
}
|
|
|
|
gtk_widget_set_tooltip_text (widget,
|
|
|
|
gettext (gpstooltips[GPS_ALTITUDE_TOOLTIP]));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
index = default_metadata_tags[i].other_tag_index;
|
|
|
|
|
|
|
|
if (default_metadata_tags[i].xmp_type == PIKA_XMP_BAG ||
|
|
|
|
default_metadata_tags[i].xmp_type == PIKA_XMP_SEQ)
|
|
|
|
{
|
|
|
|
gchar **values;
|
|
|
|
|
|
|
|
value = NULL;
|
|
|
|
values = gexiv2_metadata_try_get_tag_multiple (metadata,
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (values)
|
|
|
|
{
|
|
|
|
gint vi;
|
|
|
|
|
|
|
|
for (vi = 0; values[vi] != NULL; vi++)
|
|
|
|
{
|
|
|
|
gchar *value_clean;
|
|
|
|
|
|
|
|
value_clean = clean_xmp_string (values[vi]);
|
|
|
|
|
|
|
|
if (value_clean != NULL && value_clean[0] != '\0')
|
|
|
|
{
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
value = g_strdup (value_clean);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gchar *tmpvalue;
|
|
|
|
|
|
|
|
tmpvalue = value;
|
|
|
|
value = g_strconcat (value, "\n", value_clean, NULL);
|
|
|
|
g_free (tmpvalue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (value_clean);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index > -1)
|
|
|
|
{
|
|
|
|
gchar **equiv_values;
|
|
|
|
|
|
|
|
if (equivalent_metadata_tags[index].exif_tag_index > -1)
|
|
|
|
{
|
|
|
|
gint32 i_exif = equivalent_metadata_tags[index].exif_tag_index;
|
|
|
|
const gchar *exif_tag_str = exif_equivalent_tags[i_exif].tag;
|
|
|
|
gchar *exif_value = NULL;
|
|
|
|
|
|
|
|
exif_value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
exif_tag_str,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (exif_value)
|
|
|
|
{
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
value = exif_value;
|
|
|
|
exif_value = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (g_strcmp0 (value, exif_value))
|
|
|
|
{
|
|
|
|
g_printerr ("Value of tag %s: '%s' is not the same as %s: '%s'. "
|
|
|
|
"Ignoring value of %s.\n",
|
|
|
|
default_metadata_tags[equivalent_metadata_tags[index].default_tag_index].tag,
|
|
|
|
value, exif_tag_str, exif_value, exif_tag_str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (exif_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* These are all IPTC tags some of which can appear multiple times so
|
|
|
|
* we will use get_tag_multiple. Also IPTC most commonly uses UTF-8
|
|
|
|
* not current locale so get_tag_interpreted was wrong anyway.
|
|
|
|
* FIXME For now lets interpret as UTF-8 and in the future read
|
|
|
|
* and interpret based on the CharacterSet tag.
|
|
|
|
*/
|
|
|
|
equiv_values = gexiv2_metadata_try_get_tag_multiple (metadata,
|
|
|
|
equivalent_metadata_tags[index].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (equiv_values)
|
|
|
|
{
|
|
|
|
gint evi;
|
|
|
|
|
|
|
|
for (evi = 0; equiv_values[evi] != NULL; evi++)
|
|
|
|
{
|
|
|
|
if (equiv_values[evi][0] != '\0')
|
|
|
|
{
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
value = g_strdup (equiv_values[evi]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (! g_strv_contains ((const gchar * const *) values, equiv_values[evi]))
|
|
|
|
{
|
|
|
|
gchar *tmpvalue;
|
|
|
|
|
|
|
|
tmpvalue = value;
|
|
|
|
value = g_strconcat (value, "\n", equiv_values[evi], NULL);
|
|
|
|
g_free (tmpvalue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_strfreev (equiv_values);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_strfreev (values);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
gchar *value_utf8 = clean_xmp_string (value);
|
|
|
|
|
|
|
|
g_free (value);
|
|
|
|
|
|
|
|
if (value_utf8 && value_utf8[0] != '\0')
|
|
|
|
{
|
|
|
|
value = g_strdup (value_utf8);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
value = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (value_utf8);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index == SPECIAL_PROCESSING_DATE_CREATED)
|
|
|
|
{
|
|
|
|
gchar *date_value = NULL;
|
|
|
|
gchar *date_time_value = NULL;
|
|
|
|
|
|
|
|
/* XMP: Check if Xmp.photoshop.DateCreated has the same value as
|
|
|
|
IPTC: Iptc.Application2.DateCreated, and
|
|
|
|
Iptc.Application2.TimeCreated combined.
|
|
|
|
If empty, this can be filled from:
|
|
|
|
Exif: DateTimeOriginal, OffsetTimeOriginal
|
|
|
|
*/
|
|
|
|
date_value = gexiv2_metadata_try_get_tag_string (metadata,
|
|
|
|
"Iptc.Application2.DateCreated",
|
|
|
|
NULL);
|
|
|
|
if (date_value)
|
|
|
|
{
|
|
|
|
gchar *time_value = NULL;
|
|
|
|
|
|
|
|
time_value = gexiv2_metadata_try_get_tag_string (metadata,
|
|
|
|
"Iptc.Application2.TimeCreated",
|
|
|
|
NULL);
|
|
|
|
if (time_value)
|
|
|
|
{
|
|
|
|
date_time_value = g_strconcat (date_value, "T", time_value, NULL);
|
|
|
|
g_free (time_value);
|
|
|
|
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
/* Copy Xmp value from IPTC values*/
|
|
|
|
value = date_time_value;
|
|
|
|
date_time_value = NULL;
|
|
|
|
}
|
|
|
|
else if (strcmp (value, date_time_value))
|
|
|
|
{
|
|
|
|
g_printerr ("Xmp.Photoshop.DateCreated %s is not the same as "
|
|
|
|
"the combined IPTC DateCreated and TimeCreated "
|
|
|
|
"values: %s. The IPTC values will be ignored.\n",
|
|
|
|
value, date_time_value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
value = g_strconcat (date_value, "T00:00:00+00:00", NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_printerr ("Missing IPTC TimeCreated tag. We won't "
|
|
|
|
"compare the IPTC DateCreated value"
|
|
|
|
"with Xmp.Photoshop.DateCreated.\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! value)
|
|
|
|
{
|
|
|
|
/* Set initial value from Exif */
|
|
|
|
date_value = gexiv2_metadata_try_get_tag_string (metadata,
|
|
|
|
"Exif.Photo.DateTimeOriginal",
|
|
|
|
NULL);
|
|
|
|
if (date_value)
|
|
|
|
{
|
|
|
|
gchar **date_time_split = NULL;
|
|
|
|
|
|
|
|
/* Exif has space as separator between date and time,
|
|
|
|
and no timezone, which is a separate tag. */
|
|
|
|
date_time_split = g_strsplit (date_value, " ", 2);
|
|
|
|
|
|
|
|
if (date_time_split[0] != NULL)
|
|
|
|
{
|
|
|
|
gchar *time_zone = NULL;
|
|
|
|
|
|
|
|
/* Exif uses ':' as date delimiter instead of '-' */
|
|
|
|
g_strdelimit (date_time_split[0], ":", '-');
|
|
|
|
|
|
|
|
if (date_time_split[1] != NULL)
|
|
|
|
{
|
|
|
|
date_time_value = g_strconcat (date_time_split[0],
|
|
|
|
"T", date_time_split[1],
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
date_time_value = g_strconcat (date_time_split[0],
|
|
|
|
"T00:00:00", NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
time_zone = gexiv2_metadata_try_get_tag_string (metadata,
|
|
|
|
"Exif.Photo.OffsetTimeOriginal",
|
|
|
|
NULL);
|
|
|
|
if (time_zone)
|
|
|
|
{
|
|
|
|
value = g_strconcat (date_time_value, time_zone, NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
value = g_strconcat (date_time_value, "+00:00", NULL);
|
|
|
|
}
|
|
|
|
g_free (time_zone);
|
|
|
|
}
|
|
|
|
g_strfreev (date_time_split);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (date_value);
|
|
|
|
g_free (date_time_value);
|
|
|
|
}
|
|
|
|
else if (index > -1)
|
|
|
|
{
|
|
|
|
gchar **values;
|
|
|
|
|
|
|
|
if (equivalent_metadata_tags[index].exif_tag_index > -1)
|
|
|
|
{
|
|
|
|
gint32 i_exif = equivalent_metadata_tags[index].exif_tag_index;
|
|
|
|
const gchar *exif_tag_str = exif_equivalent_tags[i_exif].tag;
|
|
|
|
gchar *exif_value = NULL;
|
|
|
|
|
|
|
|
exif_value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
exif_tag_str,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (exif_value)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (! value)
|
|
|
|
{
|
|
|
|
value = exif_value;
|
|
|
|
exif_value = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (g_strcmp0 (value, exif_value))
|
|
|
|
{
|
|
|
|
g_printerr ("Value of tag %s: '%s' is not the same as %s: '%s'. "
|
|
|
|
"Ignoring value of %s.\n",
|
|
|
|
default_metadata_tags[equivalent_metadata_tags[index].default_tag_index].tag,
|
|
|
|
value, exif_tag_str, exif_value, exif_tag_str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (exif_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* It's not very likely we will have an XMP tag that can only
|
|
|
|
* have a single value instead of an array, which corresponds to
|
|
|
|
* an IPTC tag that can have multiple values, but since we
|
|
|
|
* already have this code it can't hurt to keep testing for it.
|
|
|
|
* FIXME For now lets interpret as UTF-8 and in the future read
|
|
|
|
* and interpret based on the CharacterSet tag.
|
|
|
|
*/
|
|
|
|
values = gexiv2_metadata_try_get_tag_multiple (metadata,
|
|
|
|
equivalent_metadata_tags[index].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (values)
|
|
|
|
{
|
|
|
|
gint i;
|
|
|
|
GString *str = NULL;
|
|
|
|
|
|
|
|
for (i = 0; values[i] != NULL; i++)
|
|
|
|
{
|
|
|
|
if (values[i][0] != '\0')
|
|
|
|
{
|
|
|
|
if (! str)
|
|
|
|
{
|
|
|
|
str = g_string_new (values[i]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (equivalent_metadata_tags[index].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
g_string_append (str, "\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_string_append (str, ", ");
|
|
|
|
}
|
|
|
|
g_string_append (str, values[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (str)
|
|
|
|
{
|
|
|
|
/* If we got values from both Xmp and Iptc then compare those
|
|
|
|
* values and if they are different concatenate them. Usually they
|
|
|
|
* should be the same in which case we won't duplicate the string.
|
|
|
|
*/
|
|
|
|
if (value && strcmp (value, str->str))
|
|
|
|
{
|
|
|
|
if (equivalent_metadata_tags[index].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
g_string_prepend (str, "\n");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_string_prepend (str, ", ");
|
|
|
|
}
|
|
|
|
g_string_prepend (str, value);
|
|
|
|
g_free (value);
|
|
|
|
}
|
|
|
|
value = g_string_free (str, FALSE);
|
|
|
|
}
|
|
|
|
g_strfreev (values);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (default_metadata_tags[i].mode == MODE_LIST)
|
|
|
|
{
|
|
|
|
/* Tab: IPTC Extension, Label: Location Shown */
|
|
|
|
if (! strcmp ("Xmp.iptcExt.LocationShown",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint counter;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - SUB LOCATION */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_SUB_LOC);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_sub_loc_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_SUB_LOC));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - CITY */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_CITY);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_city_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_CITY));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - STATE PROVINCE */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_STATE_PROV);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_state_prov_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_STATE_PROV));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - COUNTRY */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_CNTRY);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_cntry_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_CNTRY));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - COUNTRY ISO */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_CNTRY_ISO);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_cntry_iso_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_CNTRY_ISO));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LOCATION SHOWN - WORLD REGION */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LOC_SHO_CNTRY_WRLD_REG);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (loc_sho_wrld_reg_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LOC_SHO_CNTRY_WRLD_REG));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Favor the most common form: /Iptc4xmpExt:* */
|
|
|
|
counter = count_tags (metadata, LOCATIONSHOWN_HEADER,
|
|
|
|
locationshown,
|
|
|
|
n_locationshown);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, LOCATIONSHOWN_HEADER,
|
|
|
|
locationshown,
|
|
|
|
n_locationshown, counter);
|
|
|
|
|
|
|
|
if (counter == 0 || ! tagdata)
|
|
|
|
{
|
|
|
|
/* Alternatively try: /iptcExt:* */
|
|
|
|
counter = count_tags (metadata, LOCATIONSHOWN_HEADER,
|
|
|
|
locationshown_alternative,
|
|
|
|
n_locationshown);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, LOCATIONSHOWN_HEADER,
|
|
|
|
locationshown_alternative,
|
|
|
|
n_locationshown, counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_LOC_SHO_SUB_LOC, tagdatarow[0],
|
|
|
|
COL_LOC_SHO_CITY, tagdatarow[1],
|
|
|
|
COL_LOC_SHO_STATE_PROV, tagdatarow[2],
|
|
|
|
COL_LOC_SHO_CNTRY, tagdatarow[3],
|
|
|
|
COL_LOC_SHO_CNTRY_ISO, tagdatarow[4],
|
|
|
|
COL_LOC_SHO_CNTRY_WRLD_REG, tagdatarow[5],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_locationshown);
|
|
|
|
|
|
|
|
if (counter == 1)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_LOC_SHO_SUB_LOC, NULL,
|
|
|
|
COL_LOC_SHO_CITY, NULL,
|
|
|
|
COL_LOC_SHO_STATE_PROV, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY_ISO, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY_WRLD_REG, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_LOC_SHO_SUB_LOC, NULL,
|
|
|
|
COL_LOC_SHO_CITY, NULL,
|
|
|
|
COL_LOC_SHO_STATE_PROV, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY_ISO, NULL,
|
|
|
|
COL_LOC_SHO_CNTRY_WRLD_REG, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Featured Organization - Name */
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.OrganisationInImageName",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (widget)),
|
|
|
|
GTK_SELECTION_SINGLE);
|
|
|
|
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget), 0);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (organisation_image_name_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_ORG_IMG_NAME));
|
|
|
|
}
|
|
|
|
|
|
|
|
add_to_store (value, liststore, COL_ORG_IMG_NAME);
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Featured Organization - Code */
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.OrganisationInImageCode",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (widget)),
|
|
|
|
GTK_SELECTION_SINGLE);
|
|
|
|
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget), 0);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (organisation_image_code_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_ORG_IMG_CODE));
|
|
|
|
}
|
|
|
|
|
|
|
|
add_to_store (value, liststore, COL_ORG_IMG_CODE);
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Artwork or Object */
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.ArtworkOrObject",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint counter;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - TITLE */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_TITLE);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_title_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_TITLE));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - DATE CREATED */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_DATE_CREAT);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r != NULL; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_date_creat_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_DATE_CREAT));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - CREATOR */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_CREATOR);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r != NULL; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_creator_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_CREATOR));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - SOURCE */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_SOURCE);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_source_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_SOURCE));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - SOURCE INVENTORY ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_SRC_INV_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_source_inv_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_SRC_INV_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARTWORK OR OBJECT - COPYRIGHT NOTICE */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_AOO_CR_NOT);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (aoo_copyright_notice_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_AOO_CR_NOT));
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = count_tags (metadata, ARTWORKOROBJECT_HEADER,
|
|
|
|
artworkorobject,
|
|
|
|
n_artworkorobject);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, ARTWORKOROBJECT_HEADER,
|
|
|
|
artworkorobject,
|
|
|
|
n_artworkorobject, counter);
|
|
|
|
|
|
|
|
if (counter == 0 || ! tagdata)
|
|
|
|
{
|
|
|
|
/* Alternatively try: /iptcExt:* */
|
|
|
|
counter = count_tags (metadata, ARTWORKOROBJECT_HEADER,
|
|
|
|
artworkorobject_alternative,
|
|
|
|
n_artworkorobject);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, ARTWORKOROBJECT_HEADER,
|
|
|
|
artworkorobject_alternative,
|
|
|
|
n_artworkorobject, counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
|
|
|
|
/* remove substring for language id in title field */
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], lang_default);
|
|
|
|
if (strstr (tagdatarow[COL_AOO_TITLE], " "))
|
|
|
|
{
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], " ");
|
|
|
|
}
|
|
|
|
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], bag_default);
|
|
|
|
if (strstr (tagdatarow[COL_AOO_TITLE], " "))
|
|
|
|
{
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], " ");
|
|
|
|
}
|
|
|
|
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], seq_default);
|
|
|
|
if (strstr (tagdatarow[COL_AOO_TITLE], " "))
|
|
|
|
{
|
|
|
|
remove_substring (tagdatarow[COL_AOO_TITLE], " ");
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_AOO_TITLE, tagdatarow[0],
|
|
|
|
COL_AOO_DATE_CREAT, tagdatarow[1],
|
|
|
|
COL_AOO_CREATOR, tagdatarow[2],
|
|
|
|
COL_AOO_SOURCE, tagdatarow[3],
|
|
|
|
COL_AOO_SRC_INV_ID, tagdatarow[4],
|
|
|
|
COL_AOO_CR_NOT, tagdatarow[5],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_artworkorobject);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_AOO_TITLE, NULL,
|
|
|
|
COL_AOO_DATE_CREAT, NULL,
|
|
|
|
COL_AOO_CREATOR, NULL,
|
|
|
|
COL_AOO_SOURCE, NULL,
|
|
|
|
COL_AOO_SRC_INV_ID, NULL,
|
|
|
|
COL_AOO_CR_NOT, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Model Release Identifier */
|
|
|
|
else if (! strcmp ("Xmp.plus.ModelReleaseID",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (widget)),
|
|
|
|
GTK_SELECTION_SINGLE);
|
|
|
|
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget), 0);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (mod_rel_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_PROP_REL_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
add_to_store (value, liststore, COL_MOD_REL_ID);
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Registry Entry */
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.RegistryId",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint counter;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* REGISTRY - ORGANIZATION ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_REGISTRY_ORG_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r != NULL; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = (GtkCellRenderer*) r->data;
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (reg_org_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_REGISTRY_ORG_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* REGISTRY - ITEM ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_REGISTRY_ITEM_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (reg_item_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_REGISTRY_ITEM_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = count_tags (metadata, REGISTRYID_HEADER,
|
|
|
|
registryid,
|
|
|
|
n_registryid);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, REGISTRYID_HEADER,
|
|
|
|
registryid,
|
|
|
|
n_registryid, counter);
|
|
|
|
|
|
|
|
if (counter == 0 || ! tagdata)
|
|
|
|
{
|
|
|
|
/* Alternatively try: /iptcExt:* */
|
|
|
|
counter = count_tags (metadata, REGISTRYID_HEADER,
|
|
|
|
registryid_alternative,
|
|
|
|
n_registryid);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, REGISTRYID_HEADER,
|
|
|
|
registryid_alternative,
|
|
|
|
n_registryid, counter);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_REGISTRY_ORG_ID, tagdatarow[0],
|
|
|
|
COL_REGISTRY_ITEM_ID, tagdatarow[1],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_registryid);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_REGISTRY_ORG_ID, NULL,
|
|
|
|
COL_REGISTRY_ITEM_ID, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Image Creator */
|
|
|
|
else if (! strcmp ("Xmp.plus.ImageCreator",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint counter;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* IMAGE CREATOR - NAME */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_IMG_CR8_NAME);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (img_cr8_name_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_IMG_CR8_NAME));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* IMAGE CREATOR - ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_IMG_CR8_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (img_cr8_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_IMG_CR8_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = count_tags (metadata, IMAGECREATOR_HEADER,
|
|
|
|
imagecreator,
|
|
|
|
n_imagecreator);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, IMAGECREATOR_HEADER,
|
|
|
|
imagecreator,
|
|
|
|
n_imagecreator, counter);
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_IMG_CR8_NAME, tagdatarow[0],
|
|
|
|
COL_IMG_CR8_ID, tagdatarow[1],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_imagecreator);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_IMG_CR8_NAME, NULL,
|
|
|
|
COL_IMG_CR8_ID, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Copyright Owner */
|
|
|
|
else if (! strcmp ("Xmp.plus.CopyrightOwner",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gint counter;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* COPYRIGHT OWNER - NAME */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_CR_OWNER_NAME);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (cr_owner_name_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_CR_OWNER_NAME));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* COPYRIGHT OWNER - ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_CR_OWNER_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (cr_owner_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_CR_OWNER_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = count_tags (metadata, COPYRIGHTOWNER_HEADER,
|
|
|
|
copyrightowner,
|
|
|
|
n_copyrightowner);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, COPYRIGHTOWNER_HEADER,
|
|
|
|
copyrightowner,
|
|
|
|
n_copyrightowner, counter);
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_CR_OWNER_NAME, tagdatarow[0],
|
|
|
|
COL_CR_OWNER_ID, tagdatarow[1],
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_copyrightowner);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_CR_OWNER_NAME, NULL,
|
|
|
|
COL_CR_OWNER_ID, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Licensor */
|
|
|
|
else if (! strcmp ("Xmp.plus.Licensor",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkTreeModel *phonemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
GtkListStore *phonestore;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
GtkTreeIter phoneiter;
|
|
|
|
gint counter;
|
|
|
|
gint j;
|
|
|
|
gchar **tagdata;
|
|
|
|
|
|
|
|
phonestore = gtk_list_store_new (1, G_TYPE_STRING);
|
|
|
|
gtk_list_store_append (phonestore, &phoneiter);
|
|
|
|
gtk_list_store_set (phonestore, &phoneiter, 0, "Unknown", -1);
|
|
|
|
for (j=1; j < n_phone_types; j++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (phonestore, &phoneiter);
|
|
|
|
gtk_list_store_set (phonestore, &phoneiter,
|
|
|
|
0, gettext (phone_types[j].display),
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
phonemodel = GTK_TREE_MODEL (phonestore);
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
/* LICENSOR - NAME */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_NAME);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_name_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_NAME));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - ID */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_ID);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - PHONE NUMBER 1 */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_PHONE1);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_phone1_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_PHONE1));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - PHONE TYPE 1 */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_PHONE_TYPE1);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
"text-column", 0,
|
|
|
|
"has-entry", FALSE,
|
|
|
|
"model", phonemodel,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_phone_type1_cell_edited_callback),
|
|
|
|
widget);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_PHONE_TYPE1));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - PHONE NUMBER 2 */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_PHONE2);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_phone2_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_PHONE2));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - PHONE TYPE 2 */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_PHONE_TYPE2);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
"text-column", 0,
|
|
|
|
"has-entry", FALSE,
|
|
|
|
"model", phonemodel,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_phone_type2_cell_edited_callback),
|
|
|
|
widget);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_PHONE_TYPE2));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - EMAIL */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_EMAIL);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_email_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_EMAIL));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* LICENSOR - WEB ADDRESS */
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget),
|
|
|
|
COL_LICENSOR_WEB);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (licensor_web_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_LICENSOR_WEB));
|
|
|
|
}
|
|
|
|
|
|
|
|
counter = count_tags (metadata, LICENSOR_HEADER,
|
|
|
|
licensor,
|
|
|
|
n_licensor);
|
|
|
|
|
|
|
|
tagdata = get_tags (metadata, LICENSOR_HEADER,
|
|
|
|
licensor,
|
|
|
|
n_licensor, counter);
|
|
|
|
|
|
|
|
if (counter > 0 && tagdata)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < counter; item++)
|
|
|
|
{
|
|
|
|
gchar **tagdatarow = (gchar **) tagdata[item];
|
|
|
|
gchar *type1;
|
|
|
|
gchar *type2;
|
|
|
|
gint types;
|
|
|
|
|
|
|
|
type1 = g_strdup (gettext (phone_types[0].display));
|
|
|
|
type2 = g_strdup (gettext (phone_types[0].display));
|
|
|
|
|
|
|
|
for (types = 0; types < n_phone_types; types++)
|
|
|
|
{
|
|
|
|
/* phone type 1 */
|
|
|
|
if (tagdatarow[3] &&
|
|
|
|
! strcmp (tagdatarow[3],
|
|
|
|
phone_types[types].data))
|
|
|
|
{
|
|
|
|
g_free (type1);
|
|
|
|
type1 = g_strdup (gettext (phone_types[types].display));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* phone type 2 */
|
|
|
|
if (tagdatarow[5] &&
|
|
|
|
! strcmp (tagdatarow[5],
|
|
|
|
phone_types[types].data))
|
|
|
|
{
|
|
|
|
g_free (type2);
|
|
|
|
type2 = g_strdup (gettext (phone_types[types].display));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_LICENSOR_NAME, tagdatarow[0],
|
|
|
|
COL_LICENSOR_ID, tagdatarow[1],
|
|
|
|
COL_LICENSOR_PHONE1, tagdatarow[2],
|
|
|
|
COL_LICENSOR_PHONE_TYPE1, type1,
|
|
|
|
COL_LICENSOR_PHONE2, tagdatarow[4],
|
|
|
|
COL_LICENSOR_PHONE_TYPE2, type2,
|
|
|
|
COL_LICENSOR_EMAIL, tagdatarow[6],
|
|
|
|
COL_LICENSOR_WEB, tagdatarow[7],
|
|
|
|
-1);
|
|
|
|
g_free (type1);
|
|
|
|
g_free (type2);
|
|
|
|
}
|
|
|
|
free_tagdata(tagdata, counter, n_licensor);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < 2; item++)
|
|
|
|
{
|
|
|
|
gtk_list_store_append (liststore, &iter);
|
|
|
|
gtk_list_store_set (liststore, &iter,
|
|
|
|
COL_LICENSOR_NAME, NULL,
|
|
|
|
COL_LICENSOR_ID, NULL,
|
|
|
|
COL_LICENSOR_PHONE1, NULL,
|
|
|
|
COL_LICENSOR_PHONE_TYPE1, gettext (phone_types[0].display),
|
|
|
|
COL_LICENSOR_PHONE2, NULL,
|
|
|
|
COL_LICENSOR_PHONE_TYPE1, gettext (phone_types[0].display),
|
|
|
|
COL_LICENSOR_EMAIL, NULL,
|
|
|
|
COL_LICENSOR_WEB, NULL,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Tab: IPTC Extension, Label: Property Release Identifier */
|
|
|
|
else if (! strcmp ("Xmp.plus.PropertyReleaseID",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GList *rlist;
|
|
|
|
GList *r;
|
|
|
|
GtkTreeViewColumn *column;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
GtkListStore *liststore;
|
|
|
|
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
|
|
|
|
liststore = GTK_LIST_STORE (treemodel);
|
|
|
|
|
|
|
|
gtk_tree_selection_set_mode (gtk_tree_view_get_selection (GTK_TREE_VIEW (widget)),
|
|
|
|
GTK_SELECTION_SINGLE);
|
|
|
|
|
|
|
|
column = gtk_tree_view_get_column (GTK_TREE_VIEW (widget), 0);
|
|
|
|
rlist = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (column));
|
|
|
|
for (r = rlist; r; r = r->next)
|
|
|
|
{
|
|
|
|
renderer = r->data;
|
|
|
|
|
|
|
|
g_object_set (renderer,
|
|
|
|
"editable", TRUE,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect (renderer, "edited",
|
|
|
|
G_CALLBACK (prop_rel_id_cell_edited_callback),
|
|
|
|
treemodel);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (renderer),
|
|
|
|
"column",
|
|
|
|
GINT_TO_POINTER (COL_PROP_REL_ID));
|
|
|
|
}
|
|
|
|
|
|
|
|
add_to_store (value, liststore, COL_PROP_REL_ID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
if (default_metadata_tags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), value);
|
|
|
|
}
|
|
|
|
else if (default_metadata_tags[i].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
GtkTextBuffer *buffer;
|
|
|
|
|
|
|
|
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
|
|
|
|
gtk_text_buffer_set_text (buffer, value, -1);
|
|
|
|
}
|
|
|
|
else if (default_metadata_tags[i].mode == MODE_COMBO)
|
|
|
|
{
|
|
|
|
gint32 data = 0;
|
|
|
|
|
|
|
|
if (! strcmp ("Exif.GPSInfo.GPSLatitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strncmp ("N", value, 1))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strncmp ("S", value, 1))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSLongitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strncmp ("E", value, 1))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strncmp ("W", value, 1))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSAltitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strncmp ("A", value, 1))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strncmp ("B", value, 1))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.xmp.Rating", default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strcmp ("1", value))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("2", value))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("3", value))
|
|
|
|
{
|
|
|
|
data = 3;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("4", value))
|
|
|
|
{
|
|
|
|
data = 4;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("5", value))
|
|
|
|
{
|
|
|
|
data = 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.xmpRights.Marked",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strcmp ("True", value))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("False", value))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.photoshop.Urgency",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strcmp ("1", value))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("2", value))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("3", value))
|
|
|
|
{
|
|
|
|
data = 3;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("4", value))
|
|
|
|
{
|
|
|
|
data = 4;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("5", value))
|
|
|
|
{
|
|
|
|
data = 5;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("6", value))
|
|
|
|
{
|
|
|
|
data = 6;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("7", value))
|
|
|
|
{
|
|
|
|
data = 7;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("8", value))
|
|
|
|
{
|
|
|
|
data = 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.MinorModelAgeDisclosure",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strcmp ("Age Unknown", value))
|
|
|
|
{
|
|
|
|
data = 0;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 25 or Over", value))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 24", value))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 23", value))
|
|
|
|
{
|
|
|
|
data = 3;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 22", value))
|
|
|
|
{
|
|
|
|
data = 4;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 21", value))
|
|
|
|
{
|
|
|
|
data = 5;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 20", value))
|
|
|
|
{
|
|
|
|
data = 6;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 19", value))
|
|
|
|
{
|
|
|
|
data = 7;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 18", value))
|
|
|
|
{
|
|
|
|
data = 8;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 17", value))
|
|
|
|
{
|
|
|
|
data = 9;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 16", value))
|
|
|
|
{
|
|
|
|
data = 10;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 15", value))
|
|
|
|
{
|
|
|
|
data = 11;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Age 14 or Under", value))
|
|
|
|
{
|
|
|
|
data = 12;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.ModelReleaseStatus",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_modelreleasestatus; loop++)
|
|
|
|
{
|
|
|
|
if (! strcmp (modelreleasestatus[loop].data, value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! strcmp (gettext (modelreleasestatus[loop].display),
|
|
|
|
value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.DigitalSourceType",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_digitalsourcetype; loop++)
|
|
|
|
{
|
|
|
|
if (! strcmp (digitalsourcetype[loop].data, value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! strcmp (gettext (digitalsourcetype[loop].display),
|
|
|
|
value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.PropertyReleaseStatus",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
gint loop;
|
|
|
|
|
|
|
|
for (loop = 0; loop < n_propertyreleasestatus; loop++)
|
|
|
|
{
|
|
|
|
if (! strcmp (propertyreleasestatus[loop].data, value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! strcmp (gettext (propertyreleasestatus[loop].display),
|
|
|
|
value))
|
|
|
|
{
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), loop);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.DICOM.PatientSex",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (! strcmp ("male", value))
|
|
|
|
{
|
|
|
|
data = 1;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("female", value))
|
|
|
|
{
|
|
|
|
data = 2;
|
|
|
|
}
|
|
|
|
else if (! strcmp ("other", value))
|
|
|
|
{
|
|
|
|
data = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (widget), data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set Xmp.iptc.CreatorContactInfo/Iptc4xmpCore:* last since the short form
|
|
|
|
* Xmp.iptc.Ci* could have been used to set this information too. Because
|
|
|
|
* the first (longer) form is the most common let that override the shorter
|
|
|
|
* form in the (unlikely) case that both are present and also have
|
|
|
|
* different values. Due to a bug in the metadata-editor previously only
|
|
|
|
* the short form was saved.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < n_creatorContactInfoTags; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget;
|
|
|
|
|
|
|
|
widget = metadata_editor_get_widget (meta_info, creatorContactInfoTags[i].id);
|
|
|
|
|
|
|
|
value = gexiv2_metadata_try_get_tag_interpreted_string (metadata,
|
|
|
|
creatorContactInfoTags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (value)
|
|
|
|
{
|
|
|
|
gchar *value_utf;
|
|
|
|
|
|
|
|
value_utf = clean_xmp_string (value);
|
|
|
|
g_free (value);
|
|
|
|
|
|
|
|
if (creatorContactInfoTags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (widget), value_utf);
|
|
|
|
}
|
|
|
|
else if (creatorContactInfoTags[i].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
GtkTextBuffer *buffer;
|
|
|
|
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
|
|
|
|
gtk_text_buffer_set_text (buffer, value_utf, -1);
|
|
|
|
}
|
|
|
|
g_free (value_utf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set creation date */
|
|
|
|
entry_widget = metadata_editor_get_widget (meta_info, "create_date_button");
|
|
|
|
g_signal_connect (entry_widget, "clicked",
|
|
|
|
G_CALLBACK (on_create_date_button_clicked),
|
|
|
|
metadata_editor_get_widget (meta_info,
|
|
|
|
"Xmp.photoshop.DateCreated"));
|
|
|
|
|
|
|
|
/* Set patient dob date */
|
|
|
|
entry_widget = metadata_editor_get_widget (meta_info, "dob_date_button");
|
|
|
|
g_signal_connect (entry_widget, "clicked",
|
|
|
|
G_CALLBACK (on_patient_dob_date_button_clicked),
|
|
|
|
metadata_editor_get_widget (meta_info,
|
|
|
|
"Xmp.DICOM.PatientDOB"));
|
|
|
|
|
|
|
|
/* Set study date */
|
|
|
|
entry_widget = metadata_editor_get_widget (meta_info, "study_date_button");
|
|
|
|
g_signal_connect (entry_widget, "clicked",
|
|
|
|
G_CALLBACK (on_study_date_button_clicked),
|
|
|
|
metadata_editor_get_widget (meta_info,
|
|
|
|
"Xmp.DICOM.StudyDateTime"));
|
|
|
|
|
|
|
|
/* Set series date */
|
|
|
|
entry_widget = metadata_editor_get_widget (meta_info, "series_date_button");
|
|
|
|
g_signal_connect (entry_widget, "clicked",
|
|
|
|
G_CALLBACK (on_series_date_button_clicked),
|
|
|
|
metadata_editor_get_widget (meta_info,
|
|
|
|
"Xmp.DICOM.SeriesDateTime"));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ WRITE METADATA ]========================================================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_tag_failed (const gchar *tag, gchar *error_message)
|
|
|
|
{
|
|
|
|
g_log ("", G_LOG_LEVEL_MESSAGE,
|
|
|
|
_("Failed to set metadata tag %s: %s"), tag, error_message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_tag_string (PikaMetadata *metadata,
|
|
|
|
const gchar *name,
|
|
|
|
const gchar *value,
|
|
|
|
gboolean clear_tag)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (! metadata || ! name) return;
|
|
|
|
|
|
|
|
if (clear_tag)
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (metadata), name, NULL);
|
|
|
|
|
|
|
|
if (! value) return;
|
|
|
|
|
|
|
|
if (! gexiv2_metadata_try_set_tag_string (GEXIV2_METADATA (metadata),
|
|
|
|
name, value, &error))
|
|
|
|
{
|
|
|
|
set_tag_failed (name, error->message);
|
|
|
|
g_clear_error (&error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gchar *
|
|
|
|
get_phonetype (gchar *cur_value)
|
|
|
|
{
|
|
|
|
gchar *phone_type_value = NULL;
|
|
|
|
gint types;
|
|
|
|
|
|
|
|
if (cur_value != NULL)
|
|
|
|
{
|
|
|
|
for (types = 0; types < n_phone_types; types++)
|
|
|
|
{
|
|
|
|
if (! strcmp (cur_value, gettext (phone_types[types].display)))
|
|
|
|
{
|
|
|
|
phone_type_value = strdup (phone_types[types].data);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (cur_value);
|
|
|
|
}
|
|
|
|
if (! phone_type_value)
|
|
|
|
phone_type_value = strdup (phone_types[0].data);
|
|
|
|
cur_value = phone_type_value;
|
|
|
|
|
|
|
|
return phone_type_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
write_metadata_tag (metadata_editor *meta_info,
|
|
|
|
PikaMetadata *metadata,
|
|
|
|
gchar *tag,
|
|
|
|
gint data_column)
|
|
|
|
{
|
|
|
|
GtkWidget *list_widget;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
gint row;
|
|
|
|
gint number_of_rows;
|
|
|
|
gchar *rc_data;
|
|
|
|
GString *data;
|
|
|
|
|
|
|
|
list_widget = metadata_editor_get_widget (meta_info, tag);
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (list_widget));
|
|
|
|
|
|
|
|
number_of_rows = gtk_tree_model_iter_n_children (treemodel, NULL);
|
|
|
|
|
|
|
|
if (number_of_rows <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
data = g_string_sized_new (256);
|
|
|
|
|
|
|
|
for (row = 0; row < number_of_rows; row++)
|
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
if (gtk_tree_model_iter_nth_child (treemodel, &iter, NULL, row))
|
|
|
|
{
|
|
|
|
gtk_tree_model_get (treemodel, &iter,
|
|
|
|
data_column, &rc_data,
|
|
|
|
-1);
|
|
|
|
if (rc_data && rc_data[0] != '\0')
|
|
|
|
{
|
|
|
|
if (row > 0)
|
|
|
|
g_string_append (data, ", ");
|
|
|
|
|
|
|
|
g_string_append (data, rc_data);
|
|
|
|
}
|
|
|
|
g_free (rc_data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG,
|
|
|
|
"write_metadata_tag tag: %s, value: %s",
|
|
|
|
tag, data->str);
|
|
|
|
|
|
|
|
set_tag_string (metadata, tag, data->str, TRUE);
|
|
|
|
g_string_free (data, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
write_metadata_tag_multiple (metadata_editor *meta_info, PikaMetadata *metadata,
|
|
|
|
GExiv2StructureType type, const gchar * header_tag,
|
|
|
|
gint n_columns, const gchar **column_tags,
|
|
|
|
const gint special_handling[])
|
|
|
|
{
|
|
|
|
GtkWidget *list_widget;
|
|
|
|
GtkTreeModel *treemodel;
|
|
|
|
gint row;
|
|
|
|
gint number_of_rows;
|
|
|
|
gint counter;
|
|
|
|
gchar temp_tag[1024];
|
|
|
|
|
|
|
|
/* Clear old tag data first */
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (metadata), header_tag, NULL);
|
|
|
|
|
|
|
|
for (row = 0; row < 256; row++)
|
|
|
|
{
|
|
|
|
gint item;
|
|
|
|
|
|
|
|
for (item = 0; item < n_columns; item++)
|
|
|
|
{
|
|
|
|
g_snprintf (temp_tag, sizeof (temp_tag), "%s[%d]%s",
|
|
|
|
header_tag, row, locationshown[item]);
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (metadata), temp_tag, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
list_widget = metadata_editor_get_widget (meta_info, header_tag);
|
|
|
|
treemodel = gtk_tree_view_get_model (GTK_TREE_VIEW (list_widget));
|
|
|
|
|
|
|
|
number_of_rows = gtk_tree_model_iter_n_children (treemodel, NULL);
|
|
|
|
|
|
|
|
if (number_of_rows <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
gexiv2_metadata_try_set_xmp_tag_struct (GEXIV2_METADATA (metadata),
|
|
|
|
header_tag,
|
|
|
|
GEXIV2_STRUCTURE_XA_BAG,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* We need a separate counter because an empty row will not be written */
|
|
|
|
counter = 1;
|
|
|
|
for (row = 0; row < number_of_rows; row++)
|
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
|
|
|
|
if (gtk_tree_model_iter_nth_child (treemodel, &iter, NULL, row))
|
|
|
|
{
|
|
|
|
gint col;
|
|
|
|
|
|
|
|
for (col = 0; col < n_columns; col++)
|
|
|
|
{
|
|
|
|
gchar *tag_data;
|
|
|
|
|
|
|
|
gtk_tree_model_get (treemodel, &iter,
|
|
|
|
col, &tag_data,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
g_snprintf (temp_tag, sizeof (temp_tag), "%s[%d]%s",
|
|
|
|
header_tag, counter, column_tags[col]);
|
|
|
|
|
|
|
|
if (special_handling)
|
|
|
|
switch(special_handling[col])
|
|
|
|
{
|
|
|
|
case METADATA_PHONETYPE:
|
|
|
|
tag_data = get_phonetype (tag_data);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG,
|
|
|
|
"write_metadata_tag_multiple tag: %s, value: %s, col: %d",
|
|
|
|
temp_tag, tag_data, col);
|
|
|
|
|
|
|
|
set_tag_string (metadata, temp_tag, tag_data, TRUE);
|
|
|
|
g_free (tag_data);
|
|
|
|
}
|
|
|
|
counter++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_gps_longitude_latitude (PikaMetadata *metadata,
|
|
|
|
const gchar *tag,
|
|
|
|
const gchar *value)
|
|
|
|
{
|
|
|
|
/* \u00b0 - degree symbol */
|
|
|
|
const gchar delimiters_dms[] = " deg'\":\u00b0";
|
|
|
|
gchar lng_lat[256];
|
|
|
|
gchar *s = g_strdup (value);
|
|
|
|
gchar *str1 = NULL;
|
|
|
|
gchar *str2 = NULL;
|
|
|
|
gchar *str3 = NULL;
|
|
|
|
gdouble val = 0.f;
|
|
|
|
gint degrees, minutes;
|
|
|
|
gdouble seconds;
|
|
|
|
gboolean remove_val = FALSE;
|
|
|
|
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "set_gps_longitude_latitude - Tag %s, Input value: %s", tag, value);
|
|
|
|
|
|
|
|
if (s && s[0] != '\0')
|
|
|
|
{
|
|
|
|
str1 = strtok (s, delimiters_dms);
|
|
|
|
str2 = strtok (NULL, delimiters_dms);
|
|
|
|
str3 = strtok (NULL, delimiters_dms);
|
|
|
|
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "String split into: %s - %s - %s", str1, str2, str3);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (s);
|
|
|
|
|
|
|
|
if (str1 && str2 && str3)
|
|
|
|
{
|
|
|
|
/* Assuming degrees, minutes, seconds */
|
|
|
|
degrees = g_ascii_strtoll (str1, NULL, 10);
|
|
|
|
minutes = g_ascii_strtoll (str2, NULL, 10);
|
|
|
|
seconds = g_ascii_strtod (str3, NULL);
|
|
|
|
}
|
|
|
|
else if (str1 && str2)
|
|
|
|
{
|
|
|
|
/* Assuming degrees, minutes */
|
|
|
|
gdouble min;
|
|
|
|
|
|
|
|
degrees = g_ascii_strtoll (str1, NULL, 10);
|
|
|
|
min = g_ascii_strtod (str2, NULL);
|
|
|
|
minutes = (gint) min;
|
|
|
|
seconds = (min - (gdouble) minutes) * 60.f;
|
|
|
|
}
|
|
|
|
else if (str1)
|
|
|
|
{
|
|
|
|
/* Assuming degrees only */
|
|
|
|
val = g_ascii_strtod (str1, NULL);
|
|
|
|
degrees = (gint) val;
|
|
|
|
minutes = (gint) ((val - (gdouble) degrees) * 60.f);
|
|
|
|
seconds = ((val - (gdouble) degrees - (gdouble) (minutes / 60.f)) * 60.f * 60.f);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
remove_val = TRUE;
|
|
|
|
|
|
|
|
if (!remove_val)
|
|
|
|
{
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "Converted values: %d - %d - %f", degrees, minutes, seconds);
|
|
|
|
g_snprintf (lng_lat, sizeof (lng_lat),
|
|
|
|
"%d/1 %d/1 %d/1000",
|
|
|
|
abs (degrees), abs (minutes), abs ((gint) (seconds * 1000.f)));
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_DEBUG, "Tag: %s, output string: %s", tag, lng_lat);
|
|
|
|
|
|
|
|
set_tag_string (metadata, tag, lng_lat, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (metadata), tag, NULL);
|
|
|
|
g_log (ME_LOG_DOMAIN, G_LOG_LEVEL_INFO, "Removed tag %s (no value).", tag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
metadata_editor_write_callback (GtkWidget *dialog,
|
|
|
|
metadata_editor *meta_info,
|
|
|
|
PikaImage *image)
|
|
|
|
{
|
|
|
|
PikaMetadata *g_metadata;
|
|
|
|
gint max_elements;
|
|
|
|
gint i = 0;
|
|
|
|
|
|
|
|
g_metadata = pika_image_get_metadata (image);
|
|
|
|
|
|
|
|
pika_metadata_add_xmp_history (g_metadata, "metadata");
|
|
|
|
|
|
|
|
write_metadata_tag (meta_info, g_metadata,
|
|
|
|
"Xmp.iptcExt.OrganisationInImageName",
|
|
|
|
COL_ORG_IMG_NAME);
|
|
|
|
|
|
|
|
write_metadata_tag (meta_info, g_metadata,
|
|
|
|
"Xmp.iptcExt.OrganisationInImageCode",
|
|
|
|
COL_ORG_IMG_CODE);
|
|
|
|
|
|
|
|
write_metadata_tag (meta_info, g_metadata,
|
|
|
|
"Xmp.plus.ModelReleaseID",
|
|
|
|
COL_MOD_REL_ID);
|
|
|
|
|
|
|
|
write_metadata_tag (meta_info, g_metadata,
|
|
|
|
"Xmp.plus.PropertyReleaseID",
|
|
|
|
COL_PROP_REL_ID);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_BAG,
|
|
|
|
"Xmp.iptcExt.LocationShown",
|
|
|
|
n_locationshown, locationshown_alternative, NULL);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_BAG,
|
|
|
|
"Xmp.iptcExt.ArtworkOrObject",
|
|
|
|
n_artworkorobject, artworkorobject_alternative, NULL);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_BAG,
|
|
|
|
"Xmp.iptcExt.RegistryId",
|
|
|
|
n_registryid, registryid_alternative, NULL);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_SEQ,
|
|
|
|
"Xmp.plus.ImageCreator",
|
|
|
|
n_imagecreator, imagecreator, NULL);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_SEQ,
|
|
|
|
"Xmp.plus.CopyrightOwner",
|
|
|
|
n_copyrightowner, copyrightowner, NULL);
|
|
|
|
|
|
|
|
write_metadata_tag_multiple (meta_info, g_metadata, GEXIV2_STRUCTURE_XA_SEQ,
|
|
|
|
"Xmp.plus.Licensor",
|
|
|
|
n_licensor, licensor,
|
|
|
|
licensor_special_handling);
|
|
|
|
|
|
|
|
/* DO CREATOR TAGS */
|
|
|
|
|
|
|
|
if (hasCreatorTagData (meta_info))
|
|
|
|
{
|
|
|
|
for (i = 0; i < n_creatorContactInfoTags; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget = metadata_editor_get_widget (meta_info,
|
|
|
|
creatorContactInfoTags[i].id);
|
|
|
|
|
|
|
|
if (creatorContactInfoTags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
GtkEntry *entry = GTK_ENTRY (widget);
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, creatorContactInfoTags[i].tag,
|
|
|
|
gtk_entry_get_text (entry), FALSE);
|
|
|
|
}
|
|
|
|
else if (creatorContactInfoTags[i].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
GtkTextView *text_view = GTK_TEXT_VIEW (widget);
|
|
|
|
GtkTextBuffer *buffer;
|
|
|
|
GtkTextIter start;
|
|
|
|
GtkTextIter end;
|
|
|
|
gchar *text;
|
|
|
|
|
|
|
|
buffer = gtk_text_view_get_buffer (text_view);
|
|
|
|
gtk_text_buffer_get_start_iter (buffer, &start);
|
|
|
|
gtk_text_buffer_get_end_iter (buffer, &end);
|
|
|
|
|
|
|
|
text = gtk_text_buffer_get_text (buffer, &start, &end, TRUE);
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, creatorContactInfoTags[i].tag,
|
|
|
|
text, FALSE);
|
|
|
|
|
|
|
|
g_free (text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* DO SINGLE, MULTI AND COMBO TAGS */
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i = 0; i < n_creatorContactInfoTags; i++)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
creatorContactInfoTags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
max_elements = n_default_metadata_tags;
|
|
|
|
|
|
|
|
for (i = 0; i < max_elements; i++)
|
|
|
|
{
|
|
|
|
GtkWidget *widget = metadata_editor_get_widget (meta_info,
|
|
|
|
default_metadata_tags[i].tag);
|
|
|
|
|
|
|
|
/* SINGLE TAGS */
|
|
|
|
|
|
|
|
if (default_metadata_tags[i].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
GtkEntry *entry = GTK_ENTRY (widget);
|
|
|
|
gchar *value_entry = g_strdup (gtk_entry_get_text (entry));
|
|
|
|
|
|
|
|
if (! strcmp ("Exif.GPSInfo.GPSLongitude",
|
|
|
|
default_metadata_tags[i].tag) ||
|
|
|
|
! strcmp ("Exif.GPSInfo.GPSLatitude",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
set_gps_longitude_latitude (g_metadata,
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
value_entry);
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSAltitude",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
GtkWidget *combo_widget;
|
|
|
|
gchar alt_str[256];
|
|
|
|
gdouble alt_d;
|
|
|
|
gint msr;
|
|
|
|
|
|
|
|
combo_widget = metadata_editor_get_widget (meta_info,
|
|
|
|
"GPSAltitudeSystem");
|
|
|
|
msr = gtk_combo_box_get_active (GTK_COMBO_BOX (combo_widget));
|
|
|
|
|
|
|
|
alt_d = atof (gtk_entry_get_text (entry));
|
|
|
|
if (msr == 1)
|
|
|
|
alt_d = (alt_d * 12 * 2.54);
|
|
|
|
else
|
|
|
|
alt_d *= 100.f;
|
|
|
|
|
|
|
|
g_snprintf (alt_str, sizeof (alt_str), "%d/100", (gint) alt_d);
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
alt_str, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gint index;
|
|
|
|
const gchar *text_value = gtk_entry_get_text (entry);
|
|
|
|
|
|
|
|
if (default_metadata_tags[i].xmp_type == PIKA_XMP_TEXT ||
|
|
|
|
default_metadata_tags[i].xmp_type == PIKA_XMP_NONE)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
gexiv2_metadata_try_set_xmp_tag_struct (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
GEXIV2_STRUCTURE_XA_NONE,
|
|
|
|
NULL);
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
text_value, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
text_value, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
index = default_metadata_tags[i].other_tag_index;
|
|
|
|
if (index == SPECIAL_PROCESSING_DATE_CREATED)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
"Iptc.Application2.DateCreated",
|
|
|
|
NULL);
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
"Iptc.Application2.TimeCreated",
|
|
|
|
NULL);
|
|
|
|
if (text_value)
|
|
|
|
{
|
|
|
|
gchar **date_time_split = NULL;
|
|
|
|
|
|
|
|
date_time_split = g_strsplit (text_value, "T", 2);
|
|
|
|
if (date_time_split[0] != NULL)
|
|
|
|
{
|
|
|
|
/* We can't use : as date delimiter for IPTC */
|
|
|
|
g_strdelimit (date_time_split[0], ":/.", '-');
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, "Iptc.Application2.DateCreated",
|
|
|
|
date_time_split[0], FALSE);
|
|
|
|
|
|
|
|
if (date_time_split[1] != NULL)
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, "Iptc.Application2.TimeCreated",
|
|
|
|
date_time_split[1], FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, "Iptc.Application2.TimeCreated",
|
|
|
|
"00:00:00+00:00", FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_strfreev (date_time_split);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (index > -1)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
equivalent_metadata_tags[index].tag,
|
|
|
|
NULL);
|
|
|
|
if (*text_value)
|
|
|
|
set_tag_string (g_metadata, equivalent_metadata_tags[index].tag,
|
|
|
|
text_value, FALSE);
|
|
|
|
|
|
|
|
if (equivalent_metadata_tags[index].exif_tag_index > -1)
|
|
|
|
{
|
|
|
|
gint i_exif = equivalent_metadata_tags[index].exif_tag_index;
|
|
|
|
const gchar *exif_tag_str = exif_equivalent_tags[i_exif].tag;
|
|
|
|
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
exif_tag_str,
|
|
|
|
NULL);
|
|
|
|
if (*text_value)
|
|
|
|
set_tag_string (g_metadata, exif_tag_str,
|
|
|
|
text_value, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* MULTI TAGS */
|
|
|
|
|
|
|
|
else if (default_metadata_tags[i].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
GtkTextView *text_view = GTK_TEXT_VIEW (widget);
|
|
|
|
GtkTextBuffer *buffer;
|
|
|
|
GtkTextIter start;
|
|
|
|
GtkTextIter end;
|
|
|
|
gchar *text;
|
|
|
|
gint index;
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
buffer = gtk_text_view_get_buffer (text_view);
|
|
|
|
gtk_text_buffer_get_start_iter (buffer, &start);
|
|
|
|
gtk_text_buffer_get_end_iter (buffer, &end);
|
|
|
|
|
|
|
|
text = gtk_text_buffer_get_text (buffer, &start, &end, TRUE);
|
|
|
|
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
{
|
|
|
|
if (default_metadata_tags[i].xmp_type == PIKA_XMP_TEXT ||
|
|
|
|
default_metadata_tags[i].xmp_type == PIKA_XMP_NONE)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_set_xmp_tag_struct (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
GEXIV2_STRUCTURE_XA_NONE,
|
|
|
|
NULL);
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
text, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gchar **multi;
|
|
|
|
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
/* We have one value per line. */
|
|
|
|
multi = g_strsplit (text, "\n", 0);
|
|
|
|
|
|
|
|
if (! gexiv2_metadata_try_set_tag_multiple (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
(const gchar **) multi,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
set_tag_failed (default_metadata_tags[i].tag, error->message);
|
|
|
|
g_clear_error (&error);
|
|
|
|
}
|
|
|
|
g_strfreev (multi);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
index = default_metadata_tags[i].other_tag_index;
|
|
|
|
if (index > -1)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
equivalent_metadata_tags[index].tag,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (text && *text)
|
|
|
|
{
|
|
|
|
if (equivalent_metadata_tags[index].mode == MODE_MULTI)
|
|
|
|
{
|
|
|
|
gchar **multi;
|
|
|
|
|
|
|
|
multi = g_strsplit (text, "\n", 0);
|
|
|
|
|
|
|
|
if (! gexiv2_metadata_try_set_tag_multiple (GEXIV2_METADATA (g_metadata),
|
|
|
|
equivalent_metadata_tags[index].tag,
|
|
|
|
(const gchar **) multi,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
set_tag_failed (equivalent_metadata_tags[index].tag, error->message);
|
|
|
|
g_clear_error (&error);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_strfreev (multi);
|
|
|
|
}
|
|
|
|
else if (equivalent_metadata_tags[index].mode == MODE_SINGLE)
|
|
|
|
{
|
|
|
|
/* Convert from multiline to single line: keep the \n and just add the whole text. */
|
|
|
|
set_tag_string (g_metadata, equivalent_metadata_tags[index].tag,
|
|
|
|
text, FALSE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_warning ("Copying from multiline tag %s to tag %s not implemented!",
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
equivalent_metadata_tags[index].tag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (equivalent_metadata_tags[index].exif_tag_index > -1)
|
|
|
|
{
|
|
|
|
gint i_exif = equivalent_metadata_tags[index].exif_tag_index;
|
|
|
|
const gchar *exif_tag_str = exif_equivalent_tags[i_exif].tag;
|
|
|
|
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
exif_tag_str,
|
|
|
|
NULL);
|
|
|
|
if (text && *text)
|
|
|
|
set_tag_string (g_metadata, exif_tag_str, text, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (text)
|
|
|
|
g_free (text);
|
|
|
|
}
|
|
|
|
else if (default_metadata_tags[i].mode == MODE_LIST)
|
|
|
|
{
|
|
|
|
/* MIGHT DO SOMETHING HERE */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* COMBO TAGS */
|
|
|
|
|
|
|
|
else if (default_metadata_tags[i].mode == MODE_COMBO)
|
|
|
|
{
|
|
|
|
GtkComboBoxText *combo;
|
|
|
|
gint32 value;
|
|
|
|
|
|
|
|
combo = GTK_COMBO_BOX_TEXT (widget);
|
|
|
|
value = gtk_combo_box_get_active (GTK_COMBO_BOX (combo));
|
|
|
|
|
|
|
|
if (! strcmp ("Xmp.photoshop.Urgency", default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
/* IPTC tab - Urgency */
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
"Iptc.Application2.Urgency",
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gchar *save;
|
|
|
|
|
|
|
|
save = g_strdup_printf ("%d", value);
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
save, FALSE);
|
|
|
|
set_tag_string (g_metadata, "Iptc.Application2.Urgency",
|
|
|
|
save, FALSE);
|
|
|
|
g_free (save);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.xmpRights.Marked",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
/* Description tab - Copyright Status */
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gchar *save_value;
|
|
|
|
|
|
|
|
if (value == 1)
|
|
|
|
save_value = g_strdup_printf ("%s", "True");
|
|
|
|
else /* (value == 2) */
|
|
|
|
save_value = g_strdup_printf ("%s", "False");
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
save_value, FALSE);
|
|
|
|
g_free (save_value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.xmp.Rating", default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gchar *save;
|
|
|
|
|
|
|
|
save = g_strdup_printf ("%d", value);
|
|
|
|
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
save, FALSE);
|
|
|
|
g_free (save);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.DICOM.PatientSex",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"male", FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"female", FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"other", FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSLongitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"E", FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"W", FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSLatitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"N", FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"S", FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Exif.GPSInfo.GPSAltitudeRef",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
switch (value)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"0", FALSE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
"1", FALSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.ModelReleaseStatus",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
modelreleasestatus[value].data, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.PropertyReleaseStatus",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
propertyreleasestatus[value].data, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.plus.MinorModelAgeDisclosure",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
minormodelagedisclosure[value].data, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (! strcmp ("Xmp.iptcExt.DigitalSourceType",
|
|
|
|
default_metadata_tags[i].tag))
|
|
|
|
{
|
|
|
|
if (value == 0)
|
|
|
|
{
|
|
|
|
gexiv2_metadata_try_clear_tag (GEXIV2_METADATA (g_metadata),
|
|
|
|
default_metadata_tags[i].tag,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
set_tag_string (g_metadata, default_metadata_tags[i].tag,
|
|
|
|
digitalsourcetype[value].data, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pika_image_set_metadata (image, g_metadata);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ============================================================================
|
|
|
|
* ==[ METADATA IMPORT / EXPORT FILE DIALOG UI ]===============================
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
import_dialog_metadata (metadata_editor *args)
|
|
|
|
{
|
|
|
|
GtkWidget *file_dialog;
|
|
|
|
gchar *filename;
|
|
|
|
|
|
|
|
file_dialog = gtk_file_chooser_dialog_new (_("Import Metadata File"),
|
|
|
|
NULL,
|
|
|
|
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
_("_Import"), GTK_RESPONSE_ACCEPT,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
gtk_file_chooser_set_filename (GTK_FILE_CHOOSER (file_dialog),
|
|
|
|
args->filename);
|
|
|
|
|
|
|
|
if (gtk_dialog_run (GTK_DIALOG (file_dialog)) == GTK_RESPONSE_ACCEPT)
|
|
|
|
{
|
|
|
|
filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (file_dialog));
|
|
|
|
|
|
|
|
if (filename)
|
|
|
|
{
|
|
|
|
if (args->filename)
|
|
|
|
{
|
|
|
|
g_free (args->filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
args->filename = g_strdup (filename);
|
|
|
|
import_file_metadata (args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_widget_destroy (file_dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
export_dialog_metadata (metadata_editor *args)
|
|
|
|
{
|
|
|
|
GtkWidget *file_dialog;
|
|
|
|
gchar *filename;
|
|
|
|
|
|
|
|
file_dialog = gtk_file_chooser_dialog_new (_("Export Metadata File"),
|
|
|
|
NULL,
|
|
|
|
GTK_FILE_CHOOSER_ACTION_SAVE,
|
|
|
|
_("_Cancel"), GTK_RESPONSE_CANCEL,
|
|
|
|
_("_Export"), GTK_RESPONSE_ACCEPT,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (file_dialog),
|
|
|
|
TRUE);
|
|
|
|
gtk_file_chooser_set_filename (GTK_FILE_CHOOSER (file_dialog),
|
|
|
|
args->filename);
|
|
|
|
|
|
|
|
if (gtk_dialog_run (GTK_DIALOG (file_dialog)) == GTK_RESPONSE_ACCEPT)
|
|
|
|
{
|
|
|
|
filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (file_dialog));
|
|
|
|
|
|
|
|
if (filename)
|
|
|
|
{
|
|
|
|
if (args->filename)
|
|
|
|
{
|
|
|
|
g_free (args->filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
args->filename = g_strdup (filename);
|
|
|
|
export_file_metadata (args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_widget_destroy (file_dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
impex_combo_callback (GtkComboBoxText *combo,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
metadata_editor *args;
|
|
|
|
gint32 selection;
|
|
|
|
|
|
|
|
args = data;
|
|
|
|
selection = gtk_combo_box_get_active (GTK_COMBO_BOX (combo));
|
|
|
|
|
|
|
|
switch (selection)
|
|
|
|
{
|
|
|
|
case 1: /* Import */
|
|
|
|
import_dialog_metadata (args);
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo), 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2: /* Export */
|
|
|
|
export_dialog_metadata (args);
|
|
|
|
gtk_combo_box_set_active (GTK_COMBO_BOX (combo), 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
gpsaltsys_combo_callback (GtkComboBoxText *combo,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
metadata_editor *meta_info = data;
|
|
|
|
GtkWidget *entry;
|
|
|
|
gint32 selection;
|
|
|
|
gchar alt_str[256];
|
|
|
|
double alt_d;
|
|
|
|
|
|
|
|
selection = gtk_combo_box_get_active (GTK_COMBO_BOX (combo));
|
|
|
|
|
|
|
|
entry = metadata_editor_get_widget (meta_info, "Exif.GPSInfo.GPSAltitude");
|
|
|
|
|
|
|
|
switch (selection)
|
|
|
|
{
|
|
|
|
case 0: /* Meters */
|
|
|
|
if (last_gpsaltsys_sel != 0)
|
|
|
|
{
|
|
|
|
alt_d = atof (gtk_entry_get_text (GTK_ENTRY (entry)));
|
|
|
|
alt_d = (alt_d * (12 * 2.54)) / 100;
|
|
|
|
|
|
|
|
g_snprintf (alt_str, sizeof (alt_str), "%.2f", (float)alt_d);
|
|
|
|
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (entry), alt_str);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1: /* Feet */
|
|
|
|
if (last_gpsaltsys_sel != 1)
|
|
|
|
{
|
|
|
|
alt_d = atof (gtk_entry_get_text (GTK_ENTRY (entry)));
|
|
|
|
alt_d = alt_d * 3.28;
|
|
|
|
|
|
|
|
g_snprintf (alt_str, sizeof (alt_str), "%.2f", (float)alt_d);
|
|
|
|
|
|
|
|
gtk_entry_set_text (GTK_ENTRY (entry), alt_str);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
last_gpsaltsys_sel = selection;
|
|
|
|
}
|