summaryrefslogtreecommitdiffstats
path: root/plug-ins/file-raw/file-darktable.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--plug-ins/file-raw/file-darktable.c550
1 files changed, 550 insertions, 0 deletions
diff --git a/plug-ins/file-raw/file-darktable.c b/plug-ins/file-raw/file-darktable.c
new file mode 100644
index 0000000..b5897c8
--- /dev/null
+++ b/plug-ins/file-raw/file-darktable.c
@@ -0,0 +1,550 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * file-darktable.c -- raw file format plug-in that uses darktable
+ * Copyright (C) 2012 Simon Budig <simon@gimp.org>
+ * Copyright (C) 2016 Tobias Ellinghaus <me@houz.org>
+ *
+ * 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 <glib/gstdio.h>
+
+#include <libgimp/gimp.h>
+
+#include "libgimp/stdplugins-intl.h"
+
+#include "file-raw-formats.h"
+#include "file-raw-utils.h"
+
+
+#define LOAD_THUMB_PROC "file-darktable-load-thumb"
+#define REGISTRY_KEY_BASE "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\App Paths\\darktable"
+
+
+static void init (void);
+static void query (void);
+static void run (const gchar *name,
+ gint nparams,
+ const GimpParam *param,
+ gint *nreturn_vals,
+ GimpParam **return_vals);
+static gint32 load_image (const gchar *filename,
+ GimpRunMode run_mode,
+ GError **error);
+
+static gint32 load_thumbnail_image (const gchar *filename,
+ gint thumb_size,
+ gint *width,
+ gint *height,
+ GError **error);
+
+
+const GimpPlugInInfo PLUG_IN_INFO =
+{
+ init, /* init_proc */
+ NULL, /* quit_proc */
+ query, /* query proc */
+ run, /* run_proc */
+};
+
+MAIN ()
+
+static void
+init (void)
+{
+ static const GimpParamDef load_args[] =
+ {
+ { GIMP_PDB_INT32, "run-mode", "The run mode { RUN-INTERACTIVE (0), RUN-NONINTERACTIVE (1) }" },
+ { GIMP_PDB_STRING, "filename", "The name of the file to load." },
+ { GIMP_PDB_STRING, "raw-filename", "The name entered" },
+ };
+
+ static const GimpParamDef load_return_vals[] =
+ {
+ { GIMP_PDB_IMAGE, "image", "Output image" }
+ };
+
+ static const GimpParamDef thumb_args[] =
+ {
+ { GIMP_PDB_STRING, "filename", "The name of the file to load" },
+ { GIMP_PDB_INT32, "thumb-size", "Preferred thumbnail size" }
+ };
+
+ static const GimpParamDef thumb_return_vals[] =
+ {
+ { GIMP_PDB_IMAGE, "image", "Thumbnail image" },
+ { GIMP_PDB_INT32, "image-width", "Width of full-sized image" },
+ { GIMP_PDB_INT32, "image-height", "Height of full-sized image" }
+ };
+
+ /* check if darktable is installed
+ */
+ gboolean search_path = FALSE;
+ gchar *exec_path = file_raw_get_executable_path ("darktable", NULL,
+ "DARKTABLE_EXECUTABLE",
+ "org.darktable",
+ REGISTRY_KEY_BASE,
+ &search_path);
+ gchar *argv[] = { exec_path, "--version", NULL };
+ gchar *darktable_stdout = NULL;
+ gchar *darktable_stderr = NULL;
+ gboolean have_darktable = FALSE;
+ GError *error = NULL;
+ gint i;
+
+ /* allow the user to have some insight into why darktable may fail. */
+ gboolean debug_prints = g_getenv ("DARKTABLE_DEBUG") != NULL;
+
+ if (debug_prints)
+ g_printf ("[%s] trying to call '%s'\n", __FILE__, exec_path);
+
+ if (g_spawn_sync (NULL,
+ argv,
+ NULL,
+ (search_path ? G_SPAWN_SEARCH_PATH : 0),
+ NULL,
+ NULL,
+ &darktable_stdout,
+ &darktable_stderr,
+ NULL,
+ &error))
+ {
+ GRegex *regex;
+ GMatchInfo *matches;
+ gint major;
+ gint minor;
+
+ /* A default darktable would apparently output something like
+ * "this is darktable 2.2.5", but this version string is
+ * customizable. In the official Fedora package for instance, I
+ * encountered a "this is darktable darktable-2.2.5-4.fc27".
+ * Therefore make the version recognition a bit more flexible.
+ */
+ regex = g_regex_new ("this is darktable [^0-9]*([0-9]+)\\.([0-9]+)\\.([0-9]+)",
+ 0, 0, NULL);
+ if (g_regex_match (regex, darktable_stdout, 0, &matches))
+ {
+ gchar *match;
+
+ match = g_match_info_fetch (matches, 1);
+ major = g_ascii_strtoll (match, NULL, 10);
+ g_free (match);
+
+ match = g_match_info_fetch (matches, 2);
+ minor = g_ascii_strtoll (match, NULL, 10);
+ g_free (match);
+
+ if (((major == 1 && minor >= 7) || major >= 2))
+ {
+ if (g_strstr_len (darktable_stdout, -1,
+ "Lua support enabled"))
+ {
+ have_darktable = TRUE;
+ }
+ }
+ }
+
+ g_match_info_free (matches);
+ g_regex_unref (regex);
+ }
+ else if (debug_prints)
+ {
+ g_printf ("[%s] g_spawn_sync failed\n", __FILE__);
+ }
+
+ if (debug_prints)
+ {
+ if (error)
+ g_printf ("[%s] error: %s\n", __FILE__, error->message);
+
+ if (darktable_stdout && *darktable_stdout)
+ g_printf ("[%s] stdout:\n%s\n", __FILE__, darktable_stdout);
+
+ if (darktable_stderr && *darktable_stderr)
+ g_printf ("[%s] stderr:\n%s\n", __FILE__, darktable_stderr);
+
+ g_printf ("[%s] have_darktable: %d\n", __FILE__, have_darktable);
+ }
+
+ g_clear_error (&error);
+
+ g_free (darktable_stdout);
+ g_free (darktable_stderr);
+ g_free (exec_path);
+
+ if (! have_darktable)
+ return;
+
+ gimp_install_procedure (LOAD_THUMB_PROC,
+ "Load thumbnail from a raw image via darktable",
+ "This plug-in loads a thumbnail from a raw image by calling darktable-cli.",
+ "Tobias Ellinghaus",
+ "Tobias Ellinghaus",
+ "2016",
+ NULL,
+ NULL,
+ GIMP_PLUGIN,
+ G_N_ELEMENTS (thumb_args),
+ G_N_ELEMENTS (thumb_return_vals),
+ thumb_args, thumb_return_vals);
+
+ for (i = 0; i < G_N_ELEMENTS (file_formats); i++)
+ {
+ const FileFormat *format = &file_formats[i];
+ gchar *load_proc;
+ gchar *load_blurb;
+ gchar *load_help;
+
+ load_proc = g_strdup_printf (format->load_proc_format, "darktable");
+ load_blurb = g_strdup_printf (format->load_blurb_format, "darktable");
+ load_help = g_strdup_printf (format->load_help_format, "darktable");
+
+ gimp_install_procedure (load_proc,
+ load_blurb,
+ load_help,
+ "Tobias Ellinghaus",
+ "Tobias Ellinghaus",
+ "2016",
+ format->file_type,
+ NULL,
+ GIMP_PLUGIN,
+ G_N_ELEMENTS (load_args),
+ G_N_ELEMENTS (load_return_vals),
+ load_args, load_return_vals);
+
+ gimp_register_file_handler_mime (load_proc,
+ format->mime_type);
+ gimp_register_file_handler_raw (load_proc);
+ gimp_register_magic_load_handler (load_proc,
+ format->extensions,
+ "",
+ format->magic);
+
+ gimp_register_thumbnail_loader (load_proc, LOAD_THUMB_PROC);
+
+ g_free (load_proc);
+ g_free (load_blurb);
+ g_free (load_help);
+ }
+}
+
+static void
+query (void)
+{
+ /* query() is run only the first time for efficiency. Yet this plugin
+ * is dependent on the presence of darktable which may be installed
+ * or uninstalled between GIMP startups. Therefore we should move the
+ * usual gimp_install_procedure() to init() so that the check is done
+ * at every startup instead.
+ */
+}
+
+static void
+run (const gchar *name,
+ gint nparams,
+ const GimpParam *param,
+ gint *nreturn_vals,
+ GimpParam **return_vals)
+{
+ static GimpParam values[6];
+ GimpPDBStatusType status = GIMP_PDB_SUCCESS;
+ GimpRunMode run_mode;
+ gint image_ID;
+ GError *error = NULL;
+ gint i;
+
+ INIT_I18N ();
+
+ run_mode = param[0].data.d_int32;
+
+ *nreturn_vals = 1;
+ *return_vals = values;
+
+ values[0].type = GIMP_PDB_STATUS;
+ values[0].data.d_status = GIMP_PDB_EXECUTION_ERROR;
+
+ /* check if the format passed is actually supported & load */
+ for (i = 0; i < G_N_ELEMENTS (file_formats); i++)
+ {
+ const FileFormat *format = &file_formats[i];
+ gchar *load_proc = NULL;
+
+ if (format->load_proc_format)
+ load_proc = g_strdup_printf (format->load_proc_format, "darktable");
+
+ if (load_proc && ! strcmp (name, load_proc))
+ {
+ image_ID = load_image (param[1].data.d_string, run_mode, &error);
+
+ if (image_ID != -1)
+ {
+ *nreturn_vals = 2;
+ values[1].type = GIMP_PDB_IMAGE;
+ values[1].data.d_image = image_ID;
+ }
+ else
+ {
+ status = GIMP_PDB_EXECUTION_ERROR;
+ }
+
+ break;
+ }
+ else if (! strcmp (name, LOAD_THUMB_PROC))
+ {
+ gint width = 0;
+ gint height = 0;
+
+ image_ID = load_thumbnail_image (param[0].data.d_string,
+ param[1].data.d_int32,
+ &width,
+ &height,
+ &error);
+
+ if (image_ID != -1)
+ {
+ *nreturn_vals = 6;
+ values[1].type = GIMP_PDB_IMAGE;
+ values[1].data.d_image = image_ID;
+ values[2].type = GIMP_PDB_INT32;
+ values[2].data.d_int32 = width;
+ values[3].type = GIMP_PDB_INT32;
+ values[3].data.d_int32 = height;
+ values[4].type = GIMP_PDB_INT32;
+ values[4].data.d_int32 = GIMP_RGB_IMAGE;
+ values[5].type = GIMP_PDB_INT32;
+ values[5].data.d_int32 = 1; /* num_layers */
+ }
+ else
+ {
+ status = GIMP_PDB_EXECUTION_ERROR;
+ }
+
+ break;
+ }
+ }
+
+ if (i == G_N_ELEMENTS (file_formats))
+ status = GIMP_PDB_CALLING_ERROR;
+
+ if (status != GIMP_PDB_SUCCESS && error)
+ {
+ *nreturn_vals = 2;
+ values[1].type = GIMP_PDB_STRING;
+ values[1].data.d_string = error->message;
+ }
+
+ values[0].data.d_status = status;
+}
+
+static gint32
+load_image (const gchar *filename,
+ GimpRunMode run_mode,
+ GError **error)
+{
+ gint32 image_ID = -1;
+ GFile *lua_file = gimp_data_directory_file ("file-raw",
+ "file-darktable-export-on-exit.lua",
+ NULL);
+ gchar *lua_script = g_file_get_path (lua_file);
+ gchar *lua_script_escaped = g_strescape (lua_script, "");
+ gchar *lua_quoted = g_shell_quote (lua_script_escaped);
+ gchar *lua_cmd = g_strdup_printf ("dofile(%s)", lua_quoted);
+ gchar *filename_out = gimp_temp_name ("exr");
+ gchar *export_filename = g_strdup_printf ("lua/export_on_exit/export_filename=%s",
+ filename_out);
+
+ gchar *darktable_stdout = NULL;
+ gchar *darktable_stderr = NULL;
+
+ /* allow the user to have some insight into why darktable may fail. */
+ gboolean debug_prints = g_getenv ("DARKTABLE_DEBUG") != NULL;
+
+ /* linear sRGB for now as GIMP uses that internally in many places anyway */
+ gboolean search_path = FALSE;
+ gchar *exec_path = file_raw_get_executable_path ("darktable", NULL,
+ "DARKTABLE_EXECUTABLE",
+ "org.darktable",
+ REGISTRY_KEY_BASE,
+ &search_path);
+ gchar *argv[] =
+ {
+ exec_path,
+ "--library", ":memory:",
+ "--luacmd", lua_cmd,
+ "--conf", "plugins/lighttable/export/icctype=3",
+ "--conf", export_filename,
+ (gchar *) filename,
+ NULL
+ };
+
+ g_object_unref (lua_file);
+ g_free (lua_script);
+ g_free (lua_script_escaped);
+ g_free (lua_quoted);
+
+ gimp_progress_init_printf (_("Opening '%s'"),
+ gimp_filename_to_utf8 (filename));
+
+ if (debug_prints)
+ {
+ gchar **environ_ = g_get_environ ();
+ gint i;
+
+ g_printf ("[%s] trying to call\n", __FILE__);
+ for (gchar **iter = argv; *iter; iter++)
+ g_printf (" %s\n", *iter);
+ g_printf ("\n");
+
+ g_printf ("## Environment ##\n");
+ for (i = 0; environ_[i]; i++)
+ g_printf ("- %s\n", environ_[i]);
+ g_strfreev (environ_) ;
+ }
+
+ if (g_spawn_sync (NULL,
+ argv,
+ NULL,
+ /*G_SPAWN_STDOUT_TO_DEV_NULL |*/
+ /*G_SPAWN_STDERR_TO_DEV_NULL |*/
+ (search_path ? G_SPAWN_SEARCH_PATH : 0),
+ NULL,
+ NULL,
+ &darktable_stdout,
+ &darktable_stderr,
+ NULL,
+ error))
+ {
+ image_ID = gimp_file_load (run_mode, filename_out, filename_out);
+ if (image_ID != -1)
+ gimp_image_set_filename (image_ID, filename);
+ }
+
+ if (debug_prints)
+ {
+ if (darktable_stdout && *darktable_stdout)
+ g_printf ("\n## stdout ##\n%s\n", darktable_stdout);
+
+ if (darktable_stderr && *darktable_stderr)
+ g_printf ("\n## stderr ##\n%s\n", darktable_stderr);
+ }
+
+ g_free (darktable_stdout);
+ g_free (darktable_stderr);
+
+ g_unlink (filename_out);
+ g_free (lua_cmd);
+ g_free (filename_out);
+ g_free (export_filename);
+ g_free (exec_path);
+
+ gimp_progress_update (1.0);
+
+ return image_ID;
+}
+
+static gint32
+load_thumbnail_image (const gchar *filename,
+ gint thumb_size,
+ gint *width,
+ gint *height,
+ GError **error)
+{
+ gint32 image_ID = -1;
+ gchar *filename_out = gimp_temp_name ("jpg");
+ gchar *size = g_strdup_printf ("%d", thumb_size);
+ GFile *lua_file = gimp_data_directory_file ("file-raw",
+ "file-darktable-get-size.lua",
+ NULL);
+ gchar *lua_script = g_file_get_path (lua_file);
+ gchar *lua_script_escaped = g_strescape (lua_script, "");
+ gchar *lua_quoted = g_shell_quote (lua_script_escaped);
+ gchar *lua_cmd = g_strdup_printf ("dofile(%s)", lua_quoted);
+ gchar *darktable_stdout = NULL;
+
+ gboolean search_path = FALSE;
+ gchar *exec_path = file_raw_get_executable_path ("darktable", "-cli",
+ "DARKTABLE_EXECUTABLE",
+ "org.darktable",
+ REGISTRY_KEY_BASE,
+ &search_path);
+ gchar *argv[] =
+ {
+ exec_path,
+ (gchar *) filename, filename_out,
+ "--width", size,
+ "--height", size,
+ "--hq", "false",
+ "--core",
+ "--conf", "plugins/lighttable/export/icctype=3",
+ "--luacmd", lua_cmd,
+ NULL
+ };
+
+ g_object_unref (lua_file);
+ g_free (lua_script);
+ g_free (lua_script_escaped);
+ g_free (lua_quoted);
+
+ gimp_progress_init_printf (_("Opening thumbnail for '%s'"),
+ gimp_filename_to_utf8 (filename));
+
+ *width = *height = thumb_size;
+
+ if (g_spawn_sync (NULL,
+ argv,
+ NULL,
+ G_SPAWN_STDERR_TO_DEV_NULL |
+ (search_path ? G_SPAWN_SEARCH_PATH : 0),
+ NULL,
+ NULL,
+ &darktable_stdout,
+ NULL,
+ NULL,
+ error))
+ {
+ gimp_progress_update (0.5);
+
+ image_ID = gimp_file_load (GIMP_RUN_NONINTERACTIVE,
+ filename_out,
+ filename_out);
+ if (image_ID != -1)
+ {
+ /* the size reported by raw files isn't precise,
+ * but it should be close enough to get an idea.
+ */
+ gchar *start_of_size = g_strstr_len (darktable_stdout,
+ -1,
+ "[dt4gimp]");
+ if (start_of_size)
+ sscanf (start_of_size, "[dt4gimp] %d %d", width, height);
+
+ /* is this needed for thumbnails? */
+ gimp_image_set_filename (image_ID, filename);
+ }
+ }
+
+ gimp_progress_update (1.0);
+
+ g_unlink (filename_out);
+ g_free (filename_out);
+ g_free (size);
+ g_free (lua_cmd);
+ g_free (darktable_stdout);
+ g_free (exec_path);
+
+ return image_ID;
+}