492 lines
17 KiB
C
492 lines
17 KiB
C
/* Lighting Effects 0.2.2 -- image filter plug-in for PIKA
|
|
*
|
|
* Copyright (C) 1996-98 Tom Bech
|
|
* Copyright (C) 1996-98 Federico Mena Quintero
|
|
*
|
|
* E-mail: tomb@gimp.org (Tom) or quartic@gimp.org (Federico)
|
|
*
|
|
* 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 <gtk/gtk.h>
|
|
|
|
#include <libpika/pika.h>
|
|
|
|
#include "lighting-apply.h"
|
|
#include "lighting-image.h"
|
|
#include "lighting-main.h"
|
|
#include "lighting-preview.h"
|
|
#include "lighting-shade.h"
|
|
#include "lighting-ui.h"
|
|
|
|
#include "libpika/stdplugins-intl.h"
|
|
|
|
|
|
LightingValues mapvals;
|
|
|
|
|
|
typedef struct _Lighting Lighting;
|
|
typedef struct _LightingClass LightingClass;
|
|
|
|
struct _Lighting
|
|
{
|
|
PikaPlugIn parent_instance;
|
|
};
|
|
|
|
struct _LightingClass
|
|
{
|
|
PikaPlugInClass parent_class;
|
|
};
|
|
|
|
|
|
#define LIGHTING_TYPE (lighting_get_type ())
|
|
#define LIGHTING (obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), LIGHTING_TYPE, Lighting))
|
|
|
|
GType lighting_get_type (void) G_GNUC_CONST;
|
|
|
|
static GList * lighting_query_procedures (PikaPlugIn *plug_in);
|
|
static PikaProcedure * lighting_create_procedure (PikaPlugIn *plug_in,
|
|
const gchar *name);
|
|
|
|
static PikaValueArray * lighting_run (PikaProcedure *procedure,
|
|
PikaRunMode run_mode,
|
|
PikaImage *image,
|
|
gint n_drawables,
|
|
PikaDrawable **drawables,
|
|
const PikaValueArray *args,
|
|
gpointer run_data);
|
|
|
|
static void set_default_settings (void);
|
|
static void check_drawables (void);
|
|
|
|
|
|
G_DEFINE_TYPE (Lighting, lighting, PIKA_TYPE_PLUG_IN)
|
|
|
|
PIKA_MAIN (LIGHTING_TYPE)
|
|
DEFINE_STD_SET_I18N
|
|
|
|
|
|
static void
|
|
lighting_class_init (LightingClass *klass)
|
|
{
|
|
PikaPlugInClass *plug_in_class = PIKA_PLUG_IN_CLASS (klass);
|
|
|
|
plug_in_class->query_procedures = lighting_query_procedures;
|
|
plug_in_class->create_procedure = lighting_create_procedure;
|
|
plug_in_class->set_i18n = STD_SET_I18N;
|
|
}
|
|
|
|
static void
|
|
lighting_init (Lighting *lighting)
|
|
{
|
|
}
|
|
|
|
static GList *
|
|
lighting_query_procedures (PikaPlugIn *plug_in)
|
|
{
|
|
return g_list_append (NULL, g_strdup (PLUG_IN_PROC));
|
|
}
|
|
|
|
static PikaProcedure *
|
|
lighting_create_procedure (PikaPlugIn *plug_in,
|
|
const gchar *name)
|
|
{
|
|
PikaProcedure *procedure = NULL;
|
|
|
|
if (! strcmp (name, PLUG_IN_PROC))
|
|
{
|
|
PikaRGB white = { 1.0, 1.0, 1.0, 1.0 };
|
|
|
|
procedure = pika_image_procedure_new (plug_in, name,
|
|
PIKA_PDB_PROC_TYPE_PLUGIN,
|
|
lighting_run, NULL, NULL);
|
|
|
|
pika_procedure_set_image_types (procedure, "RGB*");
|
|
pika_procedure_set_sensitivity_mask (procedure,
|
|
PIKA_PROCEDURE_SENSITIVE_DRAWABLE);
|
|
|
|
pika_procedure_set_menu_label (procedure, _("_Lighting Effects..."));
|
|
pika_procedure_add_menu_path (procedure,
|
|
"<Image>/Filters/Light and Shadow/[Light]");
|
|
|
|
pika_procedure_set_documentation (procedure,
|
|
_("Apply various lighting effects "
|
|
"to an image"),
|
|
"No help yet",
|
|
name);
|
|
pika_procedure_set_attribution (procedure,
|
|
"Tom Bech & Federico Mena Quintero",
|
|
"Tom Bech & Federico Mena Quintero",
|
|
"Version 0.2.0, March 15 1998");
|
|
|
|
PIKA_PROC_ARG_DRAWABLE (procedure, "bump-drawable",
|
|
"Bump drawable",
|
|
"Bumpmap drawable (set to NULL if disabled)",
|
|
TRUE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DRAWABLE (procedure, "env-drawable",
|
|
"Env drawable",
|
|
"Environmentmap drawable (set to NULL if disabled",
|
|
TRUE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_BOOLEAN (procedure, "do-bumpmap",
|
|
"Do bumpmap",
|
|
"Enable bumpmapping",
|
|
TRUE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_BOOLEAN (procedure, "do-envmap",
|
|
"Do envmap",
|
|
"Enable envmapping",
|
|
TRUE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_INT (procedure, "bumpmap-type",
|
|
"Bumpmap type",
|
|
"Type of mapping (0=linear, 1=log, 2=sinusoidal, "
|
|
"3=spherical)",
|
|
0, 2, 0,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_INT (procedure, "light-type",
|
|
"Light type",
|
|
"Type of lightsource (0=point, 1=directional, "
|
|
"3=spot, 4=none)",
|
|
0 ,4, 0,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_RGB (procedure, "light-color",
|
|
"Light color",
|
|
"Light source color",
|
|
TRUE, &white,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-position-x",
|
|
"Light position X",
|
|
"Light source position (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, -1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-position-y",
|
|
"Light position Y",
|
|
"Light source position (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, -1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-position-z",
|
|
"Light position Z",
|
|
"Light source position (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, -1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-direction-x",
|
|
"Light direction X",
|
|
"Light source direction (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, -1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-direction-y",
|
|
"Light direction Y",
|
|
"Light source direction (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, -1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "light-direction-z",
|
|
"Light direction Z",
|
|
"Light source direction (x,y,z)",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE, 1,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "ambient-intensity",
|
|
"Ambient intensity",
|
|
"Material ambient intensity",
|
|
0, 1, 0.2,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "diffuse-intensity",
|
|
"Diffuse intensity",
|
|
"Material diffuse intensity",
|
|
0, 1, 0.5,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "diffuse-reflectivity",
|
|
"Diffuse reflectivity",
|
|
"Material diffuse reflectivity",
|
|
0, 1, 0.4,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "specular-reflectivity",
|
|
"Specular reflectivity",
|
|
"Material specular reflectivity",
|
|
0, 1, 0.5,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_DOUBLE (procedure, "highlight",
|
|
"Highlight",
|
|
"Material highlight (note, it's exponential)",
|
|
0, G_MAXDOUBLE, 27.0,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_BOOLEAN (procedure, "antialiasing",
|
|
"Antialiasing",
|
|
"Apply antialiasing",
|
|
FALSE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_BOOLEAN (procedure, "new-image",
|
|
"New image",
|
|
"Create a new image",
|
|
FALSE,
|
|
G_PARAM_READWRITE);
|
|
|
|
PIKA_PROC_ARG_BOOLEAN (procedure, "transparent-background",
|
|
"Transparent background",
|
|
"Make background transparent",
|
|
FALSE,
|
|
G_PARAM_READWRITE);
|
|
}
|
|
|
|
return procedure;
|
|
}
|
|
|
|
static void
|
|
set_default_settings (void)
|
|
{
|
|
gint k;
|
|
|
|
mapvals.update_enabled = TRUE;
|
|
mapvals.light_selected = 0;
|
|
mapvals.light_isolated = FALSE;
|
|
|
|
pika_vector3_set (&mapvals.viewpoint, 0.5, 0.5, 0.25);
|
|
pika_vector3_set (&mapvals.planenormal, 0.0, 0.0, 1.0);
|
|
|
|
pika_vector3_set (&mapvals.lightsource[0].position, -1.0, -1.0, 1.0);
|
|
pika_vector3_set (&mapvals.lightsource[0].direction, -1.0, -1.0, 1.0);
|
|
|
|
pika_rgba_set (&mapvals.lightsource[0].color, 1.0, 1.0, 1.0, 1.0);
|
|
mapvals.lightsource[0].intensity = 1.0;
|
|
mapvals.lightsource[0].type = POINT_LIGHT;
|
|
mapvals.lightsource[0].active = TRUE;
|
|
|
|
/* init lights 2 and 3 pos to upper left and below */
|
|
pika_vector3_set (&mapvals.lightsource[1].position, 2.0, -1.0, 1.0);
|
|
pika_vector3_set (&mapvals.lightsource[1].direction, 1.0, -1.0, 1.0);
|
|
|
|
pika_vector3_set (&mapvals.lightsource[2].position, 1.0, 2.0, 1.0);
|
|
pika_vector3_set (&mapvals.lightsource[2].direction, 0.0, 1.0, 1.0);
|
|
|
|
/* init any remaining lights to directly overhead */
|
|
for (k = 3; k < NUM_LIGHTS; k++)
|
|
{
|
|
pika_vector3_set (&mapvals.lightsource[k].position, 0.0, 0.0, 1.0);
|
|
pika_vector3_set (&mapvals.lightsource[k].direction, 0.0, 0.0, 1.0);
|
|
}
|
|
|
|
for (k = 1; k < NUM_LIGHTS; k++)
|
|
{
|
|
pika_rgba_set (&mapvals.lightsource[k].color, 1.0, 1.0, 1.0, 1.0);
|
|
mapvals.lightsource[k].intensity = 1.0;
|
|
mapvals.lightsource[k].type = NO_LIGHT;
|
|
mapvals.lightsource[k].active = TRUE;
|
|
}
|
|
|
|
mapvals.material.ambient_int = 0.2;
|
|
mapvals.material.diffuse_int = 0.5;
|
|
mapvals.material.diffuse_ref = 0.4;
|
|
mapvals.material.specular_ref = 0.5;
|
|
mapvals.material.highlight = 27.0;
|
|
mapvals.material.metallic = FALSE;
|
|
|
|
mapvals.pixel_threshold = 0.25;
|
|
mapvals.max_depth = 3.0;
|
|
mapvals.preview_zoom_factor = 1.0;
|
|
|
|
mapvals.bumpmaptype = 0;
|
|
mapvals.bumpmin = 0.0;
|
|
mapvals.bumpmax = 0.1;
|
|
|
|
mapvals.antialiasing = FALSE;
|
|
mapvals.create_new_image = FALSE;
|
|
mapvals.transparent_background = FALSE;
|
|
mapvals.bump_mapped = FALSE;
|
|
mapvals.env_mapped = FALSE;
|
|
mapvals.ref_mapped = FALSE;
|
|
mapvals.previewquality = FALSE;
|
|
mapvals.interactive_preview = TRUE;
|
|
|
|
mapvals.bumpmap_id = -1;
|
|
mapvals.envmap_id = -1;
|
|
}
|
|
|
|
static void
|
|
check_drawables (void)
|
|
{
|
|
PikaDrawable *drawable;
|
|
PikaDrawable *map;
|
|
|
|
if (mapvals.bump_mapped)
|
|
{
|
|
if (! pika_item_id_is_drawable (mapvals.bumpmap_id))
|
|
{
|
|
mapvals.bump_mapped = FALSE;
|
|
mapvals.bumpmap_id = -1;
|
|
}
|
|
else
|
|
{
|
|
drawable = pika_drawable_get_by_id (mapvals.drawable_id);
|
|
map = pika_drawable_get_by_id (mapvals.bumpmap_id);
|
|
|
|
if (pika_drawable_is_indexed (map) ||
|
|
(pika_drawable_get_width (drawable) != pika_drawable_get_width (map)) ||
|
|
(pika_drawable_get_height (drawable) != pika_drawable_get_height (map)))
|
|
{
|
|
mapvals.bump_mapped = FALSE;
|
|
mapvals.bumpmap_id = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mapvals.env_mapped)
|
|
{
|
|
if (! pika_item_id_is_drawable (mapvals.envmap_id))
|
|
{
|
|
mapvals.env_mapped = FALSE;
|
|
mapvals.envmap_id = -1;
|
|
}
|
|
else
|
|
{
|
|
map = pika_drawable_get_by_id (mapvals.envmap_id);
|
|
|
|
if (pika_drawable_is_gray (map) ||
|
|
pika_drawable_has_alpha (map))
|
|
{
|
|
mapvals.env_mapped = FALSE;
|
|
mapvals.envmap_id = -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static PikaValueArray *
|
|
lighting_run (PikaProcedure *procedure,
|
|
PikaRunMode run_mode,
|
|
PikaImage *image,
|
|
gint n_drawables,
|
|
PikaDrawable **drawables,
|
|
const PikaValueArray *args,
|
|
gpointer run_data)
|
|
{
|
|
PikaDrawable *drawable;
|
|
|
|
gegl_init (NULL, NULL);
|
|
|
|
if (n_drawables != 1)
|
|
{
|
|
GError *error = NULL;
|
|
|
|
g_set_error (&error, PIKA_PLUG_IN_ERROR, 0,
|
|
_("Procedure '%s' only works with one drawable."),
|
|
pika_procedure_get_name (procedure));
|
|
|
|
return pika_procedure_new_return_values (procedure,
|
|
PIKA_PDB_CALLING_ERROR,
|
|
error);
|
|
}
|
|
else
|
|
{
|
|
drawable = drawables[0];
|
|
}
|
|
|
|
set_default_settings ();
|
|
|
|
pika_get_data (PLUG_IN_PROC, &mapvals);
|
|
|
|
mapvals.drawable_id = pika_item_get_id (PIKA_ITEM (drawable));
|
|
|
|
check_drawables ();
|
|
|
|
if (pika_drawable_is_rgb (drawable))
|
|
{
|
|
switch (run_mode)
|
|
{
|
|
case PIKA_RUN_INTERACTIVE:
|
|
if (! main_dialog (drawable))
|
|
{
|
|
return pika_procedure_new_return_values (procedure,
|
|
PIKA_PDB_CANCEL,
|
|
NULL);
|
|
}
|
|
|
|
compute_image ();
|
|
|
|
pika_set_data (PLUG_IN_PROC, &mapvals, sizeof (LightingValues));
|
|
pika_displays_flush ();
|
|
break;
|
|
|
|
case PIKA_RUN_WITH_LAST_VALS:
|
|
if (image_setup (drawable, FALSE))
|
|
compute_image ();
|
|
pika_displays_flush ();
|
|
break;
|
|
|
|
case PIKA_RUN_NONINTERACTIVE:
|
|
mapvals.bumpmap_id = PIKA_VALUES_GET_DRAWABLE_ID (args, 0);
|
|
mapvals.envmap_id = PIKA_VALUES_GET_DRAWABLE_ID (args, 1);
|
|
mapvals.bump_mapped = PIKA_VALUES_GET_BOOLEAN (args, 2);
|
|
mapvals.env_mapped = PIKA_VALUES_GET_BOOLEAN (args, 3);
|
|
mapvals.bumpmaptype = PIKA_VALUES_GET_INT (args, 4);
|
|
mapvals.lightsource[0].type = PIKA_VALUES_GET_INT (args, 5);
|
|
|
|
PIKA_VALUES_GET_RGB (args, 6, &mapvals.lightsource[0].color);
|
|
|
|
mapvals.lightsource[0].position.x = PIKA_VALUES_GET_DOUBLE (args, 7);
|
|
mapvals.lightsource[0].position.y = PIKA_VALUES_GET_DOUBLE (args, 8);
|
|
mapvals.lightsource[0].position.z = PIKA_VALUES_GET_DOUBLE (args, 9);
|
|
mapvals.lightsource[0].direction.x = PIKA_VALUES_GET_DOUBLE (args, 10);
|
|
mapvals.lightsource[0].direction.y = PIKA_VALUES_GET_DOUBLE (args, 11);
|
|
mapvals.lightsource[0].direction.z = PIKA_VALUES_GET_DOUBLE (args, 12);
|
|
mapvals.material.ambient_int = PIKA_VALUES_GET_DOUBLE (args, 13);
|
|
mapvals.material.diffuse_int = PIKA_VALUES_GET_DOUBLE (args, 14);
|
|
mapvals.material.diffuse_ref = PIKA_VALUES_GET_DOUBLE (args, 15);
|
|
mapvals.material.specular_ref = PIKA_VALUES_GET_DOUBLE (args, 16);
|
|
mapvals.material.highlight = PIKA_VALUES_GET_DOUBLE (args, 17);
|
|
mapvals.antialiasing = PIKA_VALUES_GET_BOOLEAN (args, 18);
|
|
mapvals.create_new_image = PIKA_VALUES_GET_BOOLEAN (args, 19);
|
|
mapvals.transparent_background = PIKA_VALUES_GET_BOOLEAN (args, 20);
|
|
|
|
check_drawables ();
|
|
if (image_setup (drawable, FALSE))
|
|
compute_image ();
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return pika_procedure_new_return_values (procedure,
|
|
PIKA_PDB_EXECUTION_ERROR,
|
|
NULL);
|
|
}
|
|
|
|
g_free (xpostab);
|
|
g_free (ypostab);
|
|
|
|
return pika_procedure_new_return_values (procedure, PIKA_PDB_SUCCESS, NULL);
|
|
}
|