618 lines
18 KiB
C
618 lines
18 KiB
C
/* 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
|
|
*
|
|
* 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 <string.h>
|
|
#include <glib.h>
|
|
|
|
#ifdef G_OS_WIN32
|
|
#define WIN32_LEAN_AND_MEAN
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#include <libpika/pika.h>
|
|
|
|
#include "tinyscheme/scheme-private.h"
|
|
|
|
#include "script-fu-types.h"
|
|
#include "script-fu-script.h"
|
|
#include "script-fu-scripts.h"
|
|
#include "script-fu-utils.h"
|
|
#include "script-fu-register.h"
|
|
#include "script-fu-command.h"
|
|
|
|
#include "script-fu-intl.h"
|
|
|
|
|
|
/*
|
|
* Local Functions
|
|
*/
|
|
|
|
static void script_fu_load_directory (GFile *directory);
|
|
static void script_fu_load_script (GFile *file);
|
|
static gboolean script_fu_install_script (gpointer foo,
|
|
GList *scripts,
|
|
gpointer data);
|
|
static void script_fu_install_menu (SFMenu *menu);
|
|
static gboolean script_fu_remove_script (gpointer foo,
|
|
GList *scripts,
|
|
gpointer data);
|
|
|
|
static gchar * script_fu_menu_map (const gchar *menu_path);
|
|
static gint script_fu_menu_compare (gconstpointer a,
|
|
gconstpointer b);
|
|
|
|
static void script_fu_try_map_menu (SFScript *script);
|
|
static void script_fu_append_script_to_tree (SFScript *script);
|
|
|
|
/*
|
|
* Local variables
|
|
*/
|
|
|
|
static GTree *script_tree = NULL;
|
|
static GList *script_menu_list = NULL;
|
|
|
|
|
|
/*
|
|
* Function definitions
|
|
*/
|
|
|
|
/* Traverse list of paths, finding .scm files.
|
|
* Load and eval any found script texts.
|
|
* Script texts will call Scheme functions script-fu-register
|
|
* and script-fu-menu-register,
|
|
* which insert a SFScript record into script_tree,
|
|
* and insert a SFMenu record into script_menu_list.
|
|
* These are side effects on the state of the outer (SF) interpreter.
|
|
*
|
|
* Return the tree of scripts, as well as keeping a local pointer to the tree.
|
|
* The other result (script_menu_list) is not returned, see script_fu_get_menu_list.
|
|
*
|
|
* Caller should free script_tree and script_menu_list,
|
|
* This should only be called once.
|
|
*/
|
|
GTree *
|
|
script_fu_find_scripts_into_tree ( PikaPlugIn *plug_in,
|
|
GList *paths)
|
|
{
|
|
/* Clear any existing scripts */
|
|
if (script_tree != NULL)
|
|
{
|
|
g_tree_foreach (script_tree,
|
|
(GTraverseFunc) script_fu_remove_script,
|
|
plug_in);
|
|
g_tree_destroy (script_tree);
|
|
}
|
|
|
|
script_tree = g_tree_new ((GCompareFunc) g_utf8_collate);
|
|
|
|
if (paths)
|
|
{
|
|
GList *list;
|
|
|
|
for (list = paths; list; list = g_list_next (list))
|
|
{
|
|
script_fu_load_directory (list->data);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Assert result is not NULL, but may be an empty tree.
|
|
* When paths is NULL, or no scripts found at paths.
|
|
*/
|
|
|
|
g_debug ("script_fu_find_scripts_into_tree found %i scripts", g_tree_nnodes (script_tree));
|
|
return script_tree;
|
|
}
|
|
|
|
/*
|
|
* Return list of SFMenu for recently loaded scripts.
|
|
* List is non-empty only after a call to script_fu_find_scripts_into_tree.
|
|
*/
|
|
GList *
|
|
script_fu_get_menu_list (void)
|
|
{
|
|
return script_menu_list;
|
|
}
|
|
|
|
/* Find scripts, create and install TEMPORARY PDB procedures,
|
|
* owned by self PDB procedure (e.g. extension-script-fu.)
|
|
*/
|
|
void
|
|
script_fu_find_scripts (PikaPlugIn *plug_in,
|
|
GList *path)
|
|
{
|
|
script_fu_find_scripts_into_tree (plug_in, path);
|
|
|
|
/* Now that all scripts are read in and sorted, tell pika about them */
|
|
g_tree_foreach (script_tree,
|
|
(GTraverseFunc) script_fu_install_script,
|
|
plug_in);
|
|
|
|
script_menu_list = g_list_sort (script_menu_list,
|
|
(GCompareFunc) script_fu_menu_compare);
|
|
|
|
/* Install and nuke the list of menu entries */
|
|
g_list_free_full (script_menu_list,
|
|
(GDestroyNotify) script_fu_install_menu);
|
|
script_menu_list = NULL;
|
|
}
|
|
|
|
|
|
|
|
/* For a script's call to script-fu-register.
|
|
* Traverse Scheme argument list creating a new SFScript
|
|
* whose drawable_arity is SF_PROC_ORDINARY.
|
|
*
|
|
* Return NIL or a foreign_error
|
|
*/
|
|
pointer
|
|
script_fu_add_script (scheme *sc,
|
|
pointer a)
|
|
{
|
|
SFScript *script;
|
|
pointer args_error;
|
|
|
|
/* Check metadata args args are present */
|
|
if (sc->vptr->list_length (sc, a) < 7)
|
|
return foreign_error (sc, "script-fu-register: Not enough arguments", 0);
|
|
|
|
/* pass handle to pointer into script (on the stack) */
|
|
script = script_fu_script_new_from_metadata_args (sc, &a);
|
|
|
|
/* Require drawable_arity defaults to SF_PROC_ORDINARY.
|
|
* script-fu-register specifies an ordinary PikaProcedure.
|
|
* We may go on to infer a different arity.
|
|
*/
|
|
g_assert (script->drawable_arity == SF_NO_DRAWABLE);
|
|
|
|
args_error = script_fu_script_create_formal_args (sc, &a, script);
|
|
if (args_error != sc->NIL)
|
|
return args_error;
|
|
|
|
/* fill all values from defaults */
|
|
script_fu_script_reset (script, TRUE);
|
|
|
|
/* Infer whether the script really requires one drawable,
|
|
* so that later we can set the sensitivity.
|
|
* For backward compatibility:
|
|
* v2 script-fu-register does not require author to declare drawable arity.
|
|
*/
|
|
script_fu_script_infer_drawable_arity (script);
|
|
|
|
script->proc_class = PIKA_TYPE_PROCEDURE;
|
|
|
|
script_fu_try_map_menu (script);
|
|
script_fu_append_script_to_tree (script);
|
|
return sc->NIL;
|
|
}
|
|
|
|
/* For a script's call to script-fu-register-filter.
|
|
* Traverse Scheme argument list creating a new SFScript
|
|
* whose drawable_arity is SF_PROC_IMAGE_MULTIPLE_DRAWABLE or
|
|
* SF_PROC_IMAGE_SINGLE_DRAWABLE
|
|
*
|
|
* Same as script-fu-register, except one more arg for drawable_arity.
|
|
*
|
|
* Return NIL or a foreign_error
|
|
*/
|
|
pointer
|
|
script_fu_add_script_filter (scheme *sc,
|
|
pointer a)
|
|
{
|
|
SFScript *script;
|
|
pointer args_error; /* a foreign_error or NIL. */
|
|
|
|
/* Check metadata args args are present.
|
|
* Has one more arg than script-fu-register.
|
|
*/
|
|
if (sc->vptr->list_length (sc, a) < 8)
|
|
return foreign_error (sc, "script-fu-register-filter: Not enough arguments", 0);
|
|
|
|
/* pass handle i.e. "&a" ("a" of type "pointer" is on the stack) */
|
|
script = script_fu_script_new_from_metadata_args (sc, &a);
|
|
|
|
/* Check semantic error: a script declaring it takes an image must specify
|
|
* image types. Otherwise the script's menu item will be enabled
|
|
* even when no images exist.
|
|
*/
|
|
if (g_strcmp0(script->image_types, "")==0)
|
|
return foreign_error (sc, "script-fu-register-filter: A filter must declare image types.", 0);
|
|
|
|
args_error = script_fu_script_parse_drawable_arity_arg (sc, &a, script);
|
|
if (args_error != sc->NIL)
|
|
return args_error;
|
|
|
|
args_error = script_fu_script_create_formal_args (sc, &a, script);
|
|
if (args_error != sc->NIL)
|
|
return args_error;
|
|
|
|
script->proc_class = PIKA_TYPE_IMAGE_PROCEDURE;
|
|
|
|
script_fu_try_map_menu (script);
|
|
script_fu_append_script_to_tree (script);
|
|
return sc->NIL;
|
|
}
|
|
|
|
pointer
|
|
script_fu_add_menu (scheme *sc,
|
|
pointer a)
|
|
{
|
|
SFScript *script;
|
|
SFMenu *menu;
|
|
const gchar *name;
|
|
const gchar *path;
|
|
|
|
/* Check the length of a */
|
|
if (sc->vptr->list_length (sc, a) != 2)
|
|
return foreign_error (sc, "Incorrect number of arguments for script-fu-menu-register", 0);
|
|
|
|
/* Find the script PDB entry name */
|
|
name = sc->vptr->string_value (sc->vptr->pair_car (a));
|
|
a = sc->vptr->pair_cdr (a);
|
|
|
|
script = script_fu_find_script (name);
|
|
|
|
if (! script)
|
|
{
|
|
g_message ("Procedure %s in script-fu-menu-register does not exist",
|
|
name);
|
|
return sc->NIL;
|
|
}
|
|
|
|
/* Create a new list of menus */
|
|
menu = g_slice_new0 (SFMenu);
|
|
|
|
menu->script = script;
|
|
|
|
/* Find the script menu path */
|
|
path = sc->vptr->string_value (sc->vptr->pair_car (a));
|
|
|
|
menu->menu_path = script_fu_menu_map (path);
|
|
|
|
if (! menu->menu_path)
|
|
menu->menu_path = g_strdup (path);
|
|
|
|
script_menu_list = g_list_prepend (script_menu_list, menu);
|
|
|
|
return sc->NIL;
|
|
}
|
|
|
|
|
|
/* private functions */
|
|
|
|
static void
|
|
script_fu_load_directory (GFile *directory)
|
|
{
|
|
GFileEnumerator *enumerator;
|
|
|
|
g_debug ("Load dir: %s", g_file_get_parse_name (directory));
|
|
|
|
enumerator = g_file_enumerate_children (directory,
|
|
G_FILE_ATTRIBUTE_STANDARD_NAME ","
|
|
G_FILE_ATTRIBUTE_STANDARD_IS_HIDDEN ","
|
|
G_FILE_ATTRIBUTE_STANDARD_TYPE,
|
|
G_FILE_QUERY_INFO_NONE,
|
|
NULL, NULL);
|
|
|
|
if (enumerator)
|
|
{
|
|
GFileInfo *info;
|
|
|
|
while ((info = g_file_enumerator_next_file (enumerator, NULL, NULL)))
|
|
{
|
|
GFileType file_type = g_file_info_get_attribute_uint32 (info, G_FILE_ATTRIBUTE_STANDARD_TYPE);
|
|
|
|
if ((file_type == G_FILE_TYPE_REGULAR ||
|
|
file_type == G_FILE_TYPE_DIRECTORY) &&
|
|
! g_file_info_get_attribute_boolean (info, G_FILE_ATTRIBUTE_STANDARD_IS_HIDDEN))
|
|
{
|
|
GFile *child = g_file_enumerator_get_child (enumerator, info);
|
|
|
|
if (file_type == G_FILE_TYPE_DIRECTORY)
|
|
script_fu_load_directory (child);
|
|
else
|
|
script_fu_load_script (child);
|
|
|
|
g_object_unref (child);
|
|
}
|
|
|
|
g_object_unref (info);
|
|
}
|
|
|
|
g_object_unref (enumerator);
|
|
}
|
|
}
|
|
|
|
static void
|
|
script_fu_load_script (GFile *file)
|
|
{
|
|
if (pika_file_has_extension (file, ".scm"))
|
|
{
|
|
gchar *path = g_file_get_path (file);
|
|
gchar *escaped = script_fu_strescape (path);
|
|
gchar *command;
|
|
GError *error = NULL;
|
|
|
|
command = g_strdup_printf ("(load \"%s\")", escaped);
|
|
g_free (escaped);
|
|
|
|
if (! script_fu_run_command (command, &error))
|
|
{
|
|
gchar *message = g_strdup_printf (_("Error while loading %s:"),
|
|
pika_file_get_utf8_name (file));
|
|
|
|
g_message ("%s\n\n%s", message, error->message);
|
|
|
|
g_clear_error (&error);
|
|
g_free (message);
|
|
}
|
|
|
|
#ifdef G_OS_WIN32
|
|
/* No, I don't know why, but this is
|
|
* necessary on NT 4.0.
|
|
*/
|
|
Sleep (0);
|
|
#endif
|
|
|
|
g_free (command);
|
|
g_free (path);
|
|
}
|
|
}
|
|
|
|
/* This is-a GTraverseFunction.
|
|
*
|
|
* Traverse. For each, install TEMPORARY PDB proc.
|
|
* Returning FALSE means entire list was traversed.
|
|
*/
|
|
static gboolean
|
|
script_fu_install_script (gpointer foo G_GNUC_UNUSED,
|
|
GList *scripts,
|
|
gpointer data)
|
|
{
|
|
PikaPlugIn *plug_in = data;
|
|
GList *list;
|
|
|
|
for (list = scripts; list; list = g_list_next (list))
|
|
{
|
|
SFScript *script = list->data;
|
|
|
|
const gchar* name = script->name;
|
|
if (script_fu_is_defined (name))
|
|
script_fu_script_install_proc (plug_in, script);
|
|
else
|
|
g_warning ("Run function not defined, or does not match PDB procedure name: %s", name);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
script_fu_install_menu (SFMenu *menu)
|
|
{
|
|
PikaPlugIn *plug_in = pika_get_plug_in ();
|
|
PikaProcedure *procedure = NULL;
|
|
|
|
procedure = pika_plug_in_get_temp_procedure (plug_in,
|
|
menu->script->name);
|
|
|
|
if (procedure)
|
|
pika_procedure_add_menu_path (procedure, menu->menu_path);
|
|
|
|
g_free (menu->menu_path);
|
|
g_slice_free (SFMenu, menu);
|
|
}
|
|
|
|
/*
|
|
* The following function is a GTraverseFunction.
|
|
*/
|
|
static gboolean
|
|
script_fu_remove_script (gpointer foo G_GNUC_UNUSED,
|
|
GList *scripts,
|
|
gpointer data)
|
|
{
|
|
PikaPlugIn *plug_in = data;
|
|
GList *list;
|
|
|
|
for (list = scripts; list; list = g_list_next (list))
|
|
{
|
|
SFScript *script = list->data;
|
|
|
|
script_fu_script_uninstall_proc (plug_in, script);
|
|
script_fu_script_free (script);
|
|
}
|
|
|
|
g_list_free (scripts);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
/* this is a GTraverseFunction */
|
|
static gboolean
|
|
script_fu_lookup_script (gpointer *foo G_GNUC_UNUSED,
|
|
GList *scripts,
|
|
gconstpointer *name)
|
|
{
|
|
GList *list;
|
|
|
|
for (list = scripts; list; list = g_list_next (list))
|
|
{
|
|
SFScript *script = list->data;
|
|
|
|
if (strcmp (script->name, *name) == 0)
|
|
{
|
|
/* store the script in the name pointer and stop the traversal */
|
|
*name = script;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
SFScript *
|
|
script_fu_find_script (const gchar *name)
|
|
{
|
|
gconstpointer script = name;
|
|
|
|
g_tree_foreach (script_tree,
|
|
(GTraverseFunc) script_fu_lookup_script,
|
|
&script);
|
|
|
|
if (script == name)
|
|
return NULL;
|
|
|
|
return (SFScript *) script;
|
|
}
|
|
|
|
static gchar *
|
|
script_fu_menu_map (const gchar *menu_path)
|
|
{
|
|
/* for backward compatibility, we fiddle with some menu paths */
|
|
const struct
|
|
{
|
|
const gchar *old;
|
|
const gchar *new;
|
|
} mapping[] = {
|
|
{ "<Image>/Script-Fu/Alchemy", "<Image>/Filters/Artistic" },
|
|
{ "<Image>/Script-Fu/Alpha to Logo", "<Image>/Filters/Alpha to Logo" },
|
|
{ "<Image>/Script-Fu/Animators", "<Image>/Filters/Animation" },
|
|
{ "<Image>/Script-Fu/Decor", "<Image>/Filters/Decor" },
|
|
{ "<Image>/Script-Fu/Render", "<Image>/Filters/Render" },
|
|
{ "<Image>/Script-Fu/Selection", "<Image>/Select/Modify" },
|
|
{ "<Image>/Script-Fu/Shadow", "<Image>/Filters/Light and Shadow/[Shadow]" },
|
|
{ "<Image>/Script-Fu/Stencil Ops", "<Image>/Filters/Decor" }
|
|
};
|
|
|
|
gint i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (mapping); i++)
|
|
{
|
|
if (g_str_has_prefix (menu_path, mapping[i].old))
|
|
{
|
|
const gchar *suffix = menu_path + strlen (mapping[i].old);
|
|
|
|
if (*suffix != '/')
|
|
continue;
|
|
|
|
return g_strconcat (mapping[i].new, suffix, NULL);
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static gint
|
|
script_fu_menu_compare (gconstpointer a,
|
|
gconstpointer b)
|
|
{
|
|
const SFMenu *menu_a = a;
|
|
const SFMenu *menu_b = b;
|
|
gint retval = 0;
|
|
|
|
if (menu_a->menu_path && menu_b->menu_path)
|
|
{
|
|
retval = g_utf8_collate (menu_a->menu_path,
|
|
menu_b->menu_path);
|
|
|
|
if (retval == 0 &&
|
|
menu_a->script->menu_label && menu_b->script->menu_label)
|
|
{
|
|
retval = g_utf8_collate (menu_a->script->menu_label,
|
|
menu_b->script->menu_label);
|
|
}
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
/* Is name a defined symbol in the interpreter state?
|
|
* (Defined in any script already loaded.)
|
|
* Where "symbol" has the usual lisp meaning: a unique name associated with
|
|
* a variable or function.
|
|
*
|
|
* The most common use is
|
|
* test the name of a PDB proc, which in ScriptFu must match
|
|
* a defined function that is the inner run function.
|
|
* I.E. check for typos by author of script.
|
|
* Used during query, to preflight so that we don't install a PDB proc
|
|
* that won't run later (during the run phase)
|
|
* giving "undefined symbol" for extension-script-fu.
|
|
* Note that if instead we create a PDB proc having no defined run func,
|
|
* script-fu-interpreter would load and define a same-named scheme function
|
|
* that calls the PDB, and can enter an infinite loop.
|
|
*/
|
|
gboolean
|
|
script_fu_is_defined (const gchar * name)
|
|
{
|
|
gchar *scheme_text;
|
|
GError *error = NULL;
|
|
gboolean result;
|
|
|
|
/* text to be interpreted is a call to an internal scheme function. */
|
|
scheme_text = g_strdup_printf (" (symbol? %s ) ", name);
|
|
|
|
/* Use script_fu_run_command, it correctly handles the string yielded.
|
|
* But we don't need the string yielded.
|
|
* If defined, string yielded is "#t", else is "Undefined symbol" or "#f"
|
|
*/
|
|
result = script_fu_run_command (scheme_text, &error);
|
|
if (!result)
|
|
{
|
|
g_debug ("script_fu_is_defined returns false");
|
|
/* error contains string yielded by interpretation. */
|
|
g_error_free (error);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
/* Side effects on script. */
|
|
static void
|
|
script_fu_try_map_menu (SFScript *script)
|
|
{
|
|
if (script->menu_label[0] == '<')
|
|
{
|
|
gchar *mapped = script_fu_menu_map (script->menu_label);
|
|
|
|
if (mapped)
|
|
{
|
|
g_free (script->menu_label);
|
|
script->menu_label = mapped;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Append to ordered tree.
|
|
* Side effects on script_tree.
|
|
*/
|
|
static void
|
|
script_fu_append_script_to_tree (SFScript *script)
|
|
{
|
|
GList *list = g_tree_lookup (script_tree, script->menu_label);
|
|
|
|
g_tree_insert (script_tree, (gpointer) script->menu_label,
|
|
g_list_append (list, script));
|
|
}
|