mirror of https://github.com/GNOME/gimp.git
1800 lines
43 KiB
C
1800 lines
43 KiB
C
/* The GIMP -- an image manipulation program
|
|
* 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 2 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <dirent.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/types.h>
|
|
#include "appenv.h"
|
|
#include "actionarea.h"
|
|
#include "buildmenu.h"
|
|
#include "colormaps.h"
|
|
#include "color_area.h"
|
|
#include "color_select.h"
|
|
#include "datafiles.h"
|
|
#include "devices.h"
|
|
#include "errors.h"
|
|
#include "general.h"
|
|
#include "gimprc.h"
|
|
#include "interface.h"
|
|
#include "palette.h"
|
|
#include "session.h"
|
|
|
|
#define ENTRY_WIDTH 14
|
|
#define ENTRY_HEIGHT 10
|
|
#define SPACING 1
|
|
#define COLUMNS 16
|
|
#define ROWS 16
|
|
|
|
#define PREVIEW_WIDTH ((ENTRY_WIDTH * COLUMNS) + (SPACING * (COLUMNS + 1)))
|
|
#define PREVIEW_HEIGHT ((ENTRY_HEIGHT * ROWS) + (SPACING * (ROWS + 1)))
|
|
|
|
#define PALETTE_EVENT_MASK GDK_EXPOSURE_MASK | GDK_BUTTON_PRESS_MASK | GDK_ENTER_NOTIFY_MASK
|
|
|
|
typedef struct _Palette _Palette, *PaletteP;
|
|
|
|
struct _Palette {
|
|
GtkWidget *shell;
|
|
GtkWidget *vbox;
|
|
GtkWidget *frame;
|
|
GtkWidget *menu;
|
|
GtkWidget *option_menu;
|
|
GtkWidget *color_area;
|
|
GtkWidget *color_name;
|
|
GtkWidget *palette_ops;
|
|
GdkGC *gc;
|
|
GtkAdjustment *sbar_data;
|
|
PaletteEntriesP entries;
|
|
PaletteEntryP color;
|
|
ColorSelectP color_select;
|
|
int color_select_active;
|
|
int scroll_offset;
|
|
int updating;
|
|
};
|
|
|
|
static void palette_create_palette_menu (PaletteP, PaletteEntriesP);
|
|
static PaletteEntryP palette_add_entry (PaletteEntriesP, char *, int, int, int);
|
|
static void palette_delete_entry (PaletteP);
|
|
static void palette_calc_scrollbar (PaletteP);
|
|
|
|
static void palette_entries_load (char *);
|
|
static GSList * palette_entries_insert_list (GSList *, PaletteEntriesP);
|
|
static void palette_entries_delete (char *);
|
|
static void palette_entries_save (PaletteEntriesP, char *);
|
|
static void palette_entries_free (PaletteEntriesP);
|
|
static void palette_entry_free (PaletteEntryP);
|
|
static void palette_entries_set_callback (GtkWidget *, gpointer);
|
|
|
|
static void palette_change_color (int, int, int, int);
|
|
static gint palette_color_area_expose (GtkWidget *, GdkEventExpose *, PaletteP);
|
|
static gint palette_color_area_events (GtkWidget *, GdkEvent *, PaletteP);
|
|
static void palette_scroll_update (GtkAdjustment *, gpointer);
|
|
static void palette_new_callback (GtkWidget *, gpointer);
|
|
static void palette_delete_callback (GtkWidget *, gpointer);
|
|
static void palette_refresh_callback (GtkWidget *, gpointer);
|
|
static void palette_edit_callback (GtkWidget *, gpointer);
|
|
static void palette_close_callback (GtkWidget *, gpointer);
|
|
static gint palette_dialog_delete_callback (GtkWidget *, GdkEvent *, gpointer);
|
|
static void palette_new_entries_callback (GtkWidget *, gpointer);
|
|
static void palette_add_entries_callback (GtkWidget *, gpointer, gpointer);
|
|
/* static void palette_merge_entries_callback (GtkWidget *, gpointer); */
|
|
static void palette_delete_entries_callback (GtkWidget *, gpointer);
|
|
static void palette_select_callback (int, int, int, ColorSelectState, void *);
|
|
|
|
static void palette_draw_entries (PaletteP);
|
|
static void palette_draw_current_entry (PaletteP);
|
|
static void palette_update_current_entry (PaletteP);
|
|
|
|
GSList *palette_entries_list = NULL;
|
|
|
|
static PaletteP palette = NULL;
|
|
static PaletteEntriesP default_palette_entries = NULL;
|
|
static int num_palette_entries = 0;
|
|
static unsigned char foreground[3] = { 0, 0, 0 };
|
|
static unsigned char background[3] = { 255, 255, 255 };
|
|
|
|
/* Color select dialog */
|
|
/* static ColorSelectP color_select = NULL;
|
|
static int color_select_active = 0; */
|
|
|
|
static ActionAreaItem action_items[] =
|
|
{
|
|
{ "New", palette_new_callback, NULL, NULL },
|
|
{ "Edit", palette_edit_callback, NULL, NULL },
|
|
{ "Delete", palette_delete_callback, NULL, NULL },
|
|
{ "Close", palette_close_callback, NULL, NULL },
|
|
};
|
|
|
|
static MenuItem palette_ops[] =
|
|
{
|
|
{ "New Palette", 0, 0, palette_new_entries_callback, NULL, NULL, NULL },
|
|
{ "Delete Palette", 0, 0, palette_delete_entries_callback, NULL, NULL, NULL },
|
|
{ "Refresh Palettes", 0, 0, palette_refresh_callback, NULL, NULL, NULL },
|
|
{ "Close", 0, 0, palette_close_callback, NULL, NULL, NULL },
|
|
{ NULL, 0, 0, NULL, NULL, NULL, NULL },
|
|
};
|
|
|
|
void
|
|
palettes_init (int no_data)
|
|
{
|
|
palette_init_palettes (no_data);
|
|
}
|
|
|
|
void
|
|
palettes_free ()
|
|
{
|
|
palette_free_palettes ();
|
|
}
|
|
|
|
void
|
|
palette_create ()
|
|
{
|
|
GtkWidget *vbox;
|
|
GtkWidget *hbox;
|
|
GtkWidget *sbar;
|
|
GtkWidget *frame;
|
|
GtkWidget *options_box;
|
|
GtkWidget *arrow_hbox;
|
|
GtkWidget *label;
|
|
GtkWidget *arrow;
|
|
GtkWidget *menu_bar;
|
|
GtkWidget *menu_bar_item;
|
|
int i;
|
|
|
|
if (!palette)
|
|
{
|
|
palette = g_malloc (sizeof (_Palette));
|
|
|
|
palette->entries = default_palette_entries;
|
|
palette->color = NULL;
|
|
palette->color_select = NULL;
|
|
palette->color_select_active = 0;
|
|
palette->scroll_offset = 0;
|
|
palette->gc = NULL;
|
|
palette->updating = FALSE;
|
|
|
|
/* The shell and main vbox */
|
|
palette->shell = gtk_dialog_new ();
|
|
gtk_window_set_wmclass (GTK_WINDOW (palette->shell), "color_palette", "Gimp");
|
|
session_set_window_geometry (palette->shell, &palette_session_info, FALSE);
|
|
gtk_window_set_policy (GTK_WINDOW (palette->shell), FALSE, FALSE, FALSE);
|
|
gtk_window_set_title (GTK_WINDOW (palette->shell), "Color Palette");
|
|
vbox = gtk_vbox_new (FALSE, 1);
|
|
gtk_container_border_width (GTK_CONTAINER (vbox), 1);
|
|
gtk_box_pack_start (GTK_BOX (GTK_DIALOG (palette->shell)->vbox), vbox, TRUE, TRUE, 0);
|
|
|
|
/* handle the wm close event */
|
|
gtk_signal_connect (GTK_OBJECT (palette->shell), "delete_event",
|
|
GTK_SIGNAL_FUNC (palette_dialog_delete_callback),
|
|
palette);
|
|
|
|
/* The palette options box */
|
|
options_box = gtk_hbox_new (FALSE, 1);
|
|
gtk_box_pack_start (GTK_BOX (vbox), options_box, FALSE, FALSE, 0);
|
|
|
|
/* The popup menu -- palette_ops */
|
|
for (i = 0; palette_ops[i].label; i++)
|
|
palette_ops[i].user_data = palette;
|
|
|
|
palette->palette_ops = build_menu (palette_ops, NULL);
|
|
|
|
/* The palette commands pulldown menu */
|
|
menu_bar = gtk_menu_bar_new ();
|
|
gtk_box_pack_start (GTK_BOX (options_box), menu_bar, FALSE, FALSE, 0);
|
|
menu_bar_item = gtk_menu_item_new ();
|
|
gtk_container_add (GTK_CONTAINER (menu_bar), menu_bar_item);
|
|
gtk_menu_item_set_submenu (GTK_MENU_ITEM (menu_bar_item), palette->palette_ops);
|
|
arrow_hbox = gtk_hbox_new (FALSE, 1);
|
|
gtk_container_add (GTK_CONTAINER (menu_bar_item), arrow_hbox);
|
|
label = gtk_label_new ("Ops");
|
|
arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_OUT);
|
|
gtk_box_pack_start (GTK_BOX (arrow_hbox), arrow, FALSE, FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (arrow_hbox), label, FALSE, FALSE, 4);
|
|
gtk_misc_set_alignment (GTK_MISC (label), 0.5, 0.5);
|
|
gtk_misc_set_alignment (GTK_MISC (arrow), 0.5, 0.5);
|
|
|
|
gtk_widget_show (arrow);
|
|
gtk_widget_show (label);
|
|
gtk_widget_show (arrow_hbox);
|
|
gtk_widget_show (menu_bar_item);
|
|
gtk_widget_show (menu_bar);
|
|
|
|
/* The option menu */
|
|
palette->option_menu = gtk_option_menu_new ();
|
|
gtk_box_pack_start (GTK_BOX (options_box), palette->option_menu, TRUE, TRUE, 0);
|
|
gtk_widget_show (palette->option_menu);
|
|
gtk_widget_show (options_box);
|
|
|
|
/* The active color name */
|
|
palette->color_name = gtk_entry_new ();
|
|
gtk_entry_set_text (GTK_ENTRY (palette->color_name), "Active Color Name");
|
|
gtk_box_pack_start (GTK_BOX (vbox), palette->color_name, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show (palette->color_name);
|
|
|
|
/* The horizontal box containing preview & scrollbar */
|
|
hbox = gtk_hbox_new (FALSE, 1);
|
|
gtk_box_pack_start (GTK_BOX (vbox), hbox, TRUE, TRUE, 0);
|
|
frame = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
|
|
gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, FALSE, 0);
|
|
palette->sbar_data = GTK_ADJUSTMENT (gtk_adjustment_new (0, 0, PREVIEW_HEIGHT, 1, 1, PREVIEW_HEIGHT));
|
|
gtk_signal_connect (GTK_OBJECT (palette->sbar_data), "value_changed",
|
|
(GtkSignalFunc) palette_scroll_update,
|
|
palette);
|
|
sbar = gtk_vscrollbar_new (palette->sbar_data);
|
|
gtk_box_pack_start (GTK_BOX (hbox), sbar, FALSE, FALSE, 0);
|
|
|
|
/* Create the color area window and the underlying image */
|
|
palette->color_area = gtk_preview_new (GTK_PREVIEW_COLOR);
|
|
gtk_preview_size (GTK_PREVIEW (palette->color_area), PREVIEW_WIDTH, PREVIEW_HEIGHT);
|
|
gtk_widget_set_events (palette->color_area, PALETTE_EVENT_MASK);
|
|
gtk_signal_connect_after (GTK_OBJECT (palette->color_area), "expose_event",
|
|
(GtkSignalFunc) palette_color_area_expose,
|
|
palette);
|
|
gtk_signal_connect (GTK_OBJECT (palette->color_area), "event",
|
|
(GtkSignalFunc) palette_color_area_events,
|
|
palette);
|
|
gtk_container_add (GTK_CONTAINER (frame), palette->color_area);
|
|
|
|
gtk_widget_show (palette->color_area);
|
|
gtk_widget_show (sbar);
|
|
gtk_widget_show (frame);
|
|
gtk_widget_show (hbox);
|
|
|
|
if(no_data)
|
|
palettes_init(FALSE);
|
|
|
|
/* The action area */
|
|
action_items[0].user_data = palette;
|
|
action_items[1].user_data = palette;
|
|
action_items[2].user_data = palette;
|
|
action_items[3].user_data = palette;
|
|
build_action_area (GTK_DIALOG (palette->shell), action_items, 4, 0);
|
|
|
|
gtk_widget_show (vbox);
|
|
gtk_widget_show (palette->shell);
|
|
|
|
palette_create_palette_menu (palette, default_palette_entries);
|
|
palette_calc_scrollbar (palette);
|
|
}
|
|
else
|
|
{
|
|
if (!GTK_WIDGET_VISIBLE (palette->shell))
|
|
{
|
|
gtk_widget_show (palette->shell);
|
|
}
|
|
else
|
|
{
|
|
gdk_window_raise(palette->shell->window);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
palette_free ()
|
|
{
|
|
if (palette)
|
|
{
|
|
gdk_gc_destroy (palette->gc);
|
|
|
|
if (palette->color_select)
|
|
color_select_free (palette->color_select);
|
|
|
|
session_get_window_info (palette->shell, &palette_session_info);
|
|
|
|
g_free (palette);
|
|
|
|
palette = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
palette_get_foreground (unsigned char *r,
|
|
unsigned char *g,
|
|
unsigned char *b)
|
|
{
|
|
*r = foreground[0];
|
|
*g = foreground[1];
|
|
*b = foreground[2];
|
|
}
|
|
|
|
void
|
|
palette_get_background (unsigned char *r,
|
|
unsigned char *g,
|
|
unsigned char *b)
|
|
{
|
|
*r = background[0];
|
|
*g = background[1];
|
|
*b = background[2];
|
|
}
|
|
|
|
void
|
|
palette_set_foreground (int r,
|
|
int g,
|
|
int b)
|
|
{
|
|
unsigned char rr, gg, bb;
|
|
|
|
/* Foreground */
|
|
foreground[0] = r;
|
|
foreground[1] = g;
|
|
foreground[2] = b;
|
|
|
|
palette_get_foreground (&rr, &gg, &bb);
|
|
if (no_interface == FALSE)
|
|
{
|
|
store_color (&foreground_pixel, rr, gg, bb);
|
|
color_area_update ();
|
|
device_status_update (current_device);
|
|
}
|
|
}
|
|
|
|
void
|
|
palette_set_background (int r,
|
|
int g,
|
|
int b)
|
|
{
|
|
unsigned char rr, gg, bb;
|
|
|
|
/* Background */
|
|
background[0] = r;
|
|
background[1] = g;
|
|
background[2] = b;
|
|
|
|
palette_get_background (&rr, &gg, &bb);
|
|
if (no_interface == FALSE)
|
|
{
|
|
store_color (&background_pixel, rr, gg, bb);
|
|
color_area_update ();
|
|
}
|
|
}
|
|
|
|
void
|
|
palette_set_default_colors (void)
|
|
{
|
|
palette_set_foreground (0, 0, 0);
|
|
palette_set_background (255, 255, 255);
|
|
}
|
|
|
|
|
|
void
|
|
palette_swap_colors (void)
|
|
{
|
|
unsigned char fg_r, fg_g, fg_b;
|
|
unsigned char bg_r, bg_g, bg_b;
|
|
|
|
palette_get_foreground (&fg_r, &fg_g, &fg_b);
|
|
palette_get_background (&bg_r, &bg_g, &bg_b);
|
|
|
|
palette_set_foreground (bg_r, bg_g, bg_b);
|
|
palette_set_background (fg_r, fg_g, fg_b);
|
|
}
|
|
|
|
void
|
|
palette_init_palettes (int no_data)
|
|
{
|
|
if(!no_data)
|
|
datafiles_read_directories (palette_path, palette_entries_load, 0);
|
|
|
|
}
|
|
|
|
|
|
void
|
|
palette_free_palettes (void)
|
|
{
|
|
GSList *list;
|
|
PaletteEntriesP entries;
|
|
|
|
list = palette_entries_list;
|
|
|
|
while (list)
|
|
{
|
|
entries = (PaletteEntriesP) list->data;
|
|
|
|
/* If the palette has been changed, save it, if possible */
|
|
if (entries->changed)
|
|
/* save the palette */
|
|
palette_entries_save (entries, entries->filename);
|
|
|
|
palette_entries_free (entries);
|
|
list = g_slist_next (list);
|
|
}
|
|
g_slist_free (palette_entries_list);
|
|
|
|
if (palette)
|
|
{
|
|
palette->entries = NULL;
|
|
palette->color = NULL;
|
|
palette->color_select = NULL;
|
|
palette->color_select_active = 0;
|
|
palette->scroll_offset = 0;
|
|
}
|
|
num_palette_entries = 0;
|
|
palette_entries_list = NULL;
|
|
}
|
|
|
|
|
|
/*****************************************/
|
|
/* Local functions */
|
|
/*****************************************/
|
|
|
|
static void
|
|
palette_create_palette_menu (PaletteP palette,
|
|
PaletteEntriesP default_entries)
|
|
{
|
|
GtkWidget *menu_item;
|
|
GSList *list;
|
|
PaletteEntriesP p_entries = NULL;
|
|
PaletteEntriesP found_entries = NULL;
|
|
int i = 0;
|
|
int default_index = -1;
|
|
|
|
palette->menu = gtk_menu_new ();
|
|
|
|
list = palette_entries_list;
|
|
while (list)
|
|
{
|
|
p_entries = (PaletteEntriesP) list->data;
|
|
list = g_slist_next (list);
|
|
|
|
/* to make sure we get something! */
|
|
if (p_entries == NULL)
|
|
{
|
|
found_entries = p_entries;
|
|
default_index = i;
|
|
}
|
|
|
|
if (p_entries == default_entries)
|
|
{
|
|
found_entries = default_entries;
|
|
default_index = i;
|
|
}
|
|
|
|
menu_item = gtk_menu_item_new_with_label (p_entries->name);
|
|
gtk_signal_connect (GTK_OBJECT (menu_item), "activate",
|
|
(GtkSignalFunc) palette_entries_set_callback,
|
|
(gpointer) p_entries);
|
|
gtk_container_add (GTK_CONTAINER (palette->menu), menu_item);
|
|
gtk_widget_show (menu_item);
|
|
|
|
i++;
|
|
}
|
|
|
|
if (i == 0)
|
|
{
|
|
menu_item = gtk_menu_item_new_with_label ("none");
|
|
gtk_container_add (GTK_CONTAINER (palette->menu), menu_item);
|
|
gtk_widget_show (menu_item);
|
|
|
|
/* Set the action area and option menus to insensitive */
|
|
gtk_widget_set_sensitive (palette->option_menu, FALSE);
|
|
gtk_widget_set_sensitive (GTK_DIALOG (palette->shell)->action_area, FALSE);
|
|
|
|
/* Clear the color area */
|
|
gdk_window_clear (palette->color_area->window);
|
|
}
|
|
else
|
|
{
|
|
/* Make sure the action area and option menus are sensitive */
|
|
gtk_widget_set_sensitive (palette->option_menu, TRUE);
|
|
gtk_widget_set_sensitive (GTK_DIALOG (palette->shell)->action_area, TRUE);
|
|
|
|
/* Clear the color area */
|
|
gdk_window_clear (palette->color_area->window);
|
|
}
|
|
|
|
gtk_option_menu_set_menu (GTK_OPTION_MENU (palette->option_menu), palette->menu);
|
|
|
|
/* Set the current item of the option menu to reflect
|
|
* the default palette. Need to refresh here too
|
|
*/
|
|
if (default_index != -1)
|
|
{
|
|
gtk_option_menu_set_history (GTK_OPTION_MENU (palette->option_menu), default_index);
|
|
palette_entries_set_callback (NULL, found_entries);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_entries_load (char *filename)
|
|
{
|
|
PaletteEntriesP entries;
|
|
char str[512];
|
|
char *tok;
|
|
FILE *fp;
|
|
int r, g, b;
|
|
|
|
r = g = b = 0;
|
|
|
|
entries = (PaletteEntriesP) g_malloc (sizeof (_PaletteEntries));
|
|
|
|
entries->filename = g_strdup (filename);
|
|
entries->name = g_strdup (prune_filename (filename));
|
|
entries->colors = NULL;
|
|
entries->n_colors = 0;
|
|
|
|
/* Open the requested file */
|
|
|
|
if (!(fp = fopen (filename, "r")))
|
|
{
|
|
palette_entries_free (entries);
|
|
return;
|
|
}
|
|
|
|
fread (str, 13, 1, fp);
|
|
str[13] = '\0';
|
|
if (strcmp (str, "GIMP Palette\n"))
|
|
{
|
|
fclose (fp);
|
|
return;
|
|
}
|
|
|
|
while (!feof (fp))
|
|
{
|
|
if (!fgets (str, 512, fp))
|
|
continue;
|
|
|
|
if (str[0] != '#')
|
|
{
|
|
tok = strtok (str, " \t");
|
|
if (tok)
|
|
r = atoi (tok);
|
|
|
|
tok = strtok (NULL, " \t");
|
|
if (tok)
|
|
g = atoi (tok);
|
|
|
|
tok = strtok (NULL, " \t");
|
|
if (tok)
|
|
b = atoi (tok);
|
|
|
|
tok = strtok (NULL, "\n");
|
|
|
|
palette_add_entry (entries, tok, r, g, b);
|
|
} /* if */
|
|
} /* while */
|
|
|
|
/* Clean up */
|
|
|
|
fclose (fp);
|
|
entries->changed = 0;
|
|
|
|
palette_entries_list = palette_entries_insert_list(palette_entries_list, entries);
|
|
|
|
/* Check if the current palette is the default one */
|
|
if (strcmp(default_palette, prune_filename(filename)) == 0)
|
|
default_palette_entries = entries;
|
|
}
|
|
|
|
static void
|
|
palette_entries_delete (char *filename)
|
|
{
|
|
if (filename)
|
|
unlink (filename);
|
|
}
|
|
|
|
static GSList *
|
|
palette_entries_insert_list (GSList * list,
|
|
PaletteEntriesP entries)
|
|
{
|
|
/* add it to the list */
|
|
num_palette_entries++;
|
|
return g_slist_append (list, (void *) entries);
|
|
}
|
|
|
|
static void
|
|
palette_entries_save (PaletteEntriesP palette,
|
|
char *filename)
|
|
{
|
|
FILE * fp;
|
|
GSList * list;
|
|
PaletteEntryP entry;
|
|
|
|
if (! filename)
|
|
return;
|
|
|
|
/* Open the requested file */
|
|
if (! (fp = fopen (filename, "w")))
|
|
{
|
|
g_message ("can't save palette \"%s\"\n", filename);
|
|
return;
|
|
}
|
|
|
|
fprintf (fp, "GIMP Palette\n");
|
|
fprintf (fp, "# %s -- GIMP Palette file\n", palette->name);
|
|
|
|
list = palette->colors;
|
|
while (list)
|
|
{
|
|
entry = (PaletteEntryP) list->data;
|
|
fprintf (fp, "%d %d %d\t%s\n", entry->color[0], entry->color[1],
|
|
entry->color[2], entry->name);
|
|
list = g_slist_next (list);
|
|
}
|
|
|
|
/* Clean up */
|
|
fclose (fp);
|
|
}
|
|
|
|
static void
|
|
palette_entries_free (PaletteEntriesP entries)
|
|
{
|
|
PaletteEntryP entry;
|
|
GSList * list;
|
|
|
|
list = entries->colors;
|
|
while (list)
|
|
{
|
|
entry = (PaletteEntryP) list->data;
|
|
palette_entry_free (entry);
|
|
list = list->next;
|
|
}
|
|
|
|
g_free (entries->name);
|
|
if (entries->filename)
|
|
g_free (entries->filename);
|
|
g_free (entries);
|
|
}
|
|
|
|
static void
|
|
palette_entry_free (PaletteEntryP entry)
|
|
{
|
|
if (entry->name)
|
|
g_free (entry->name);
|
|
|
|
g_free (entry);
|
|
}
|
|
|
|
static void
|
|
palette_entries_set_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteEntriesP pal;
|
|
|
|
if (palette)
|
|
{
|
|
pal = (PaletteEntriesP) client_data;
|
|
|
|
palette->entries = pal;
|
|
palette->color = NULL;
|
|
if (palette->color_select_active)
|
|
{
|
|
palette->color_select_active = 0;
|
|
color_select_hide (palette->color_select);
|
|
}
|
|
palette->color_select = NULL;
|
|
palette->scroll_offset = 0;
|
|
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_change_color (int r,
|
|
int g,
|
|
int b,
|
|
int state)
|
|
{
|
|
if (palette && palette->entries)
|
|
{
|
|
switch (state)
|
|
{
|
|
case COLOR_NEW:
|
|
palette->color = palette_add_entry (palette->entries, "Untitled", r, g, b);
|
|
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
break;
|
|
|
|
case COLOR_UPDATE_NEW:
|
|
palette->color->color[0] = r;
|
|
palette->color->color[1] = g;
|
|
palette->color->color[2] = b;
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (active_color == FOREGROUND)
|
|
palette_set_foreground (r, g, b);
|
|
else if (active_color == BACKGROUND)
|
|
palette_set_background (r, g, b);
|
|
}
|
|
|
|
void
|
|
palette_set_active_color (int r,
|
|
int g,
|
|
int b,
|
|
int state)
|
|
{
|
|
palette_change_color (r, g, b, state);
|
|
}
|
|
|
|
static gint
|
|
palette_color_area_expose (GtkWidget *widget,
|
|
GdkEventExpose *event,
|
|
PaletteP palette)
|
|
{
|
|
if (!palette->gc)
|
|
palette->gc = gdk_gc_new (widget->window);
|
|
|
|
palette_draw_current_entry (palette);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gint
|
|
palette_color_area_events (GtkWidget *widget,
|
|
GdkEvent *event,
|
|
PaletteP palette)
|
|
{
|
|
GdkEventButton *bevent;
|
|
GSList *tmp_link;
|
|
int r, g, b;
|
|
int width, height;
|
|
int entry_width;
|
|
int entry_height;
|
|
int row, col;
|
|
int pos;
|
|
|
|
switch (event->type)
|
|
{
|
|
case GDK_BUTTON_PRESS:
|
|
bevent = (GdkEventButton *) event;
|
|
width = palette->color_area->requisition.width;
|
|
height = palette->color_area->requisition.height;
|
|
entry_width = ((width - (SPACING * (COLUMNS + 1))) / COLUMNS) + SPACING;
|
|
entry_height = ((height - (SPACING * (ROWS + 1))) / ROWS) + SPACING;
|
|
|
|
col = (bevent->x - 1) / entry_width;
|
|
row = (palette->scroll_offset + bevent->y - 1) / entry_height;
|
|
pos = row * COLUMNS + col;
|
|
|
|
if (bevent->button == 1 && palette->entries)
|
|
{
|
|
|
|
tmp_link = g_slist_nth (palette->entries->colors, pos);
|
|
if (tmp_link)
|
|
{
|
|
palette_draw_current_entry (palette);
|
|
palette->color = tmp_link->data;
|
|
|
|
/* Update either foreground or background colors */
|
|
r = palette->color->color[0];
|
|
g = palette->color->color[1];
|
|
b = palette->color->color[2];
|
|
if (active_color == FOREGROUND)
|
|
palette_set_foreground (r, g, b);
|
|
else if (active_color == BACKGROUND)
|
|
palette_set_background (r, g, b);
|
|
|
|
palette_update_current_entry (palette);
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
palette_scroll_update (GtkAdjustment *adjustment,
|
|
gpointer data)
|
|
{
|
|
PaletteP palette;
|
|
|
|
palette = (PaletteP) data;
|
|
|
|
if (palette)
|
|
{
|
|
palette->scroll_offset = adjustment->value;
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_new_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
|
|
palette = client_data;
|
|
if (palette && palette->entries)
|
|
{
|
|
if (active_color == FOREGROUND)
|
|
palette->color =
|
|
palette_add_entry (palette->entries, "Untitled",
|
|
foreground[0], foreground[1], foreground[2]);
|
|
else if (active_color == BACKGROUND)
|
|
palette->color =
|
|
palette_add_entry (palette->entries, "Untitled",
|
|
background[0], background[1], background[2]);
|
|
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_delete_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
|
|
palette = client_data;
|
|
if (palette)
|
|
palette_delete_entry (palette);
|
|
}
|
|
|
|
|
|
static void
|
|
palette_refresh_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
|
|
palette = client_data;
|
|
if(palette)
|
|
{
|
|
palette_free_palettes ();
|
|
palette_init_palettes(FALSE);
|
|
palette_create_palette_menu (palette, default_palette_entries);
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
else
|
|
{
|
|
palette_free_palettes ();
|
|
palette_init_palettes(FALSE);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
palette_edit_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
unsigned char *color;
|
|
|
|
palette = client_data;
|
|
if (palette && palette->entries && palette->color)
|
|
{
|
|
color = palette->color->color;
|
|
|
|
if (!palette->color_select)
|
|
{
|
|
palette->color_select = color_select_new (color[0], color[1], color[2],
|
|
palette_select_callback, NULL,
|
|
FALSE);
|
|
palette->color_select_active = 1;
|
|
}
|
|
else
|
|
{
|
|
if (!palette->color_select_active)
|
|
{
|
|
color_select_show (palette->color_select);
|
|
palette->color_select_active = 1;
|
|
}
|
|
|
|
color_select_set_color (palette->color_select, color[0], color[1], color[2], 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
static gint
|
|
palette_dialog_delete_callback (GtkWidget *w,
|
|
GdkEvent *e,
|
|
gpointer client_data)
|
|
{
|
|
palette_close_callback (w, client_data);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
palette_close_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
|
|
palette = client_data;
|
|
if (palette)
|
|
{
|
|
if (palette->color_select_active)
|
|
{
|
|
palette->color_select_active = 0;
|
|
color_select_hide (palette->color_select);
|
|
}
|
|
|
|
if (GTK_WIDGET_VISIBLE (palette->shell))
|
|
gtk_widget_hide (palette->shell);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_new_entries_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
query_string_box ("New Palette", "Enter a name for new palette", NULL,
|
|
palette_add_entries_callback, NULL);
|
|
}
|
|
|
|
static void
|
|
palette_add_entries_callback (GtkWidget *w,
|
|
gpointer client_data,
|
|
gpointer call_data)
|
|
{
|
|
char *home;
|
|
char *palette_name;
|
|
char *local_path;
|
|
char *first_token;
|
|
char *token;
|
|
char *path;
|
|
PaletteEntriesP entries;
|
|
|
|
palette_name = (char *) call_data;
|
|
if (palette && palette_name)
|
|
{
|
|
entries = g_malloc (sizeof (_PaletteEntries));
|
|
if (palette_path)
|
|
{
|
|
/* Get the first path specified in the palette path list */
|
|
home = getenv("HOME");
|
|
local_path = g_strdup (palette_path);
|
|
first_token = local_path;
|
|
token = xstrsep(&first_token, ":");
|
|
|
|
if (token)
|
|
{
|
|
if (*token == '~')
|
|
{
|
|
path = g_malloc(strlen(home) + strlen(token) + 1);
|
|
sprintf(path, "%s%s", home, token + 1);
|
|
}
|
|
else
|
|
{
|
|
path = g_malloc(strlen(token) + 1);
|
|
strcpy(path, token);
|
|
}
|
|
|
|
entries->filename = g_malloc (strlen (path) + strlen (palette_name) + 2);
|
|
sprintf (entries->filename, "%s/%s", path, palette_name);
|
|
|
|
g_free (path);
|
|
}
|
|
|
|
g_free (local_path);
|
|
}
|
|
else
|
|
entries->filename = NULL;
|
|
|
|
entries->name = palette_name; /* don't need to copy because this memory is ours */
|
|
entries->colors = NULL;
|
|
entries->n_colors = 0;
|
|
entries->changed = 1;
|
|
|
|
palette_entries_list = palette_entries_insert_list (palette_entries_list, entries);
|
|
|
|
gtk_option_menu_remove_menu (GTK_OPTION_MENU (palette->option_menu));
|
|
palette_create_palette_menu (palette, entries);
|
|
}
|
|
}
|
|
|
|
/* static void
|
|
palette_merge_entries_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
} */
|
|
|
|
static void
|
|
palette_delete_entries_callback (GtkWidget *w,
|
|
gpointer client_data)
|
|
{
|
|
PaletteP palette;
|
|
PaletteEntriesP entries;
|
|
|
|
palette = client_data;
|
|
if (palette && palette->entries)
|
|
{
|
|
/* If a color selection dialog is up, hide it */
|
|
if (palette->color_select_active)
|
|
{
|
|
palette->color_select_active = 0;
|
|
color_select_hide (palette->color_select);
|
|
}
|
|
|
|
gtk_option_menu_remove_menu (GTK_OPTION_MENU (palette->option_menu));
|
|
|
|
entries = palette->entries;
|
|
if (entries && entries->filename)
|
|
palette_entries_delete (entries->filename);
|
|
|
|
if (default_palette_entries == entries)
|
|
default_palette_entries = NULL;
|
|
palette->entries = NULL;
|
|
|
|
palette_free_palettes (); /* free palettes, don't save any modified versions */
|
|
palette_init_palettes (FALSE); /* load in brand new palettes */
|
|
|
|
palette_create_palette_menu (palette, default_palette_entries);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_select_callback (int r,
|
|
int g,
|
|
int b,
|
|
ColorSelectState state,
|
|
void *client_data)
|
|
{
|
|
unsigned char * color;
|
|
|
|
if (palette && palette->entries)
|
|
{
|
|
switch (state) {
|
|
case COLOR_SELECT_UPDATE:
|
|
break;
|
|
case COLOR_SELECT_OK:
|
|
if (palette->color)
|
|
{
|
|
color = palette->color->color;
|
|
|
|
color[0] = r;
|
|
color[1] = g;
|
|
color[2] = b;
|
|
|
|
/* Update either foreground or background colors */
|
|
if (active_color == FOREGROUND)
|
|
palette_set_foreground (r, g, b);
|
|
else if (active_color == BACKGROUND)
|
|
palette_set_background (r, g, b);
|
|
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
/* Fallthrough */
|
|
case COLOR_SELECT_CANCEL:
|
|
color_select_hide (palette->color_select);
|
|
palette->color_select_active = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
static int
|
|
palette_draw_color_row (unsigned char **colors,
|
|
int ncolors,
|
|
int y,
|
|
unsigned char *buffer,
|
|
GtkWidget *preview)
|
|
{
|
|
unsigned char *p;
|
|
unsigned char bcolor;
|
|
int width, height;
|
|
int entry_width;
|
|
int entry_height;
|
|
int vsize;
|
|
int vspacing;
|
|
int i, j;
|
|
|
|
bcolor = 0;
|
|
|
|
width = preview->requisition.width;
|
|
height = preview->requisition.height;
|
|
entry_width = (width - (SPACING * (COLUMNS + 1))) / COLUMNS;
|
|
entry_height = (height - (SPACING * (ROWS + 1))) / ROWS;
|
|
|
|
if ((y >= 0) && ((y + SPACING) < height))
|
|
vspacing = SPACING;
|
|
else if (y < 0)
|
|
vspacing = SPACING + y;
|
|
else
|
|
vspacing = height - y;
|
|
|
|
if (vspacing > 0)
|
|
{
|
|
if (y < 0)
|
|
y += SPACING - vspacing;
|
|
|
|
for (i = SPACING - vspacing; i < SPACING; i++, y++)
|
|
{
|
|
p = buffer;
|
|
for (j = 0; j < width; j++)
|
|
{
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
}
|
|
|
|
gtk_preview_draw_row (GTK_PREVIEW (preview), buffer, 0, y, width);
|
|
}
|
|
|
|
if (y > SPACING)
|
|
y += SPACING - vspacing;
|
|
}
|
|
else
|
|
y += SPACING;
|
|
|
|
vsize = (y >= 0) ? (entry_height) : (entry_height + y);
|
|
|
|
if ((y >= 0) && ((y + entry_height) < height))
|
|
vsize = entry_height;
|
|
else if (y < 0)
|
|
vsize = entry_height + y;
|
|
else
|
|
vsize = height - y;
|
|
|
|
if (vsize > 0)
|
|
{
|
|
p = buffer;
|
|
for (i = 0; i < ncolors; i++)
|
|
{
|
|
for (j = 0; j < SPACING; j++)
|
|
{
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
}
|
|
|
|
for (j = 0; j < entry_width; j++)
|
|
{
|
|
*p++ = colors[i][0];
|
|
*p++ = colors[i][1];
|
|
*p++ = colors[i][2];
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < (COLUMNS - ncolors); i++)
|
|
{
|
|
for (j = 0; j < (SPACING + entry_width); j++)
|
|
{
|
|
*p++ = 0;
|
|
*p++ = 0;
|
|
*p++ = 0;
|
|
}
|
|
}
|
|
|
|
for (j = 0; j < SPACING; j++)
|
|
{
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
*p++ = bcolor;
|
|
}
|
|
|
|
if (y < 0)
|
|
y += entry_height - vsize;
|
|
for (i = 0; i < vsize; i++, y++)
|
|
gtk_preview_draw_row (GTK_PREVIEW (preview), buffer, 0, y, width);
|
|
if (y > entry_height)
|
|
y += entry_height - vsize;
|
|
}
|
|
else
|
|
y += entry_height;
|
|
|
|
return y;
|
|
}
|
|
|
|
static void
|
|
palette_draw_entries (PaletteP palette)
|
|
{
|
|
PaletteEntryP entry;
|
|
unsigned char *buffer;
|
|
unsigned char *colors[COLUMNS];
|
|
GSList *tmp_link;
|
|
int width, height;
|
|
int entry_width;
|
|
int entry_height;
|
|
int row_vsize;
|
|
int index, y;
|
|
|
|
if (palette && palette->entries && !palette->updating)
|
|
{
|
|
width = palette->color_area->requisition.width;
|
|
height = palette->color_area->requisition.height;
|
|
entry_width = (width - (SPACING * (COLUMNS + 1))) / COLUMNS;
|
|
entry_height = (height - (SPACING * (ROWS + 1))) / ROWS;
|
|
|
|
buffer = g_malloc (width * 3);
|
|
|
|
y = -palette->scroll_offset;
|
|
row_vsize = SPACING + entry_height;
|
|
tmp_link = palette->entries->colors;
|
|
index = 0;
|
|
|
|
while ((tmp_link) && (y < -row_vsize))
|
|
{
|
|
tmp_link = tmp_link->next;
|
|
|
|
if (++index == COLUMNS)
|
|
{
|
|
index = 0;
|
|
y += row_vsize;
|
|
}
|
|
}
|
|
|
|
index = 0;
|
|
while (tmp_link)
|
|
{
|
|
entry = tmp_link->data;
|
|
tmp_link = tmp_link->next;
|
|
|
|
colors[index] = entry->color;
|
|
index++;
|
|
|
|
if (index == COLUMNS)
|
|
{
|
|
index = 0;
|
|
y = palette_draw_color_row (colors, COLUMNS, y, buffer, palette->color_area);
|
|
if (y >= height)
|
|
break;
|
|
}
|
|
}
|
|
|
|
while (y < height)
|
|
{
|
|
y = palette_draw_color_row (colors, index, y, buffer, palette->color_area);
|
|
index = 0;
|
|
}
|
|
|
|
gtk_widget_draw (palette->color_area, NULL);
|
|
|
|
g_free (buffer);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_draw_current_entry (PaletteP palette)
|
|
{
|
|
PaletteEntryP entry;
|
|
int width, height;
|
|
int entry_width;
|
|
int entry_height;
|
|
int row, col;
|
|
int x, y;
|
|
|
|
if (palette && palette->entries && !palette->updating && palette->color)
|
|
{
|
|
gdk_gc_set_function (palette->gc, GDK_INVERT);
|
|
|
|
entry = palette->color;
|
|
|
|
row = entry->position / COLUMNS;
|
|
col = entry->position % COLUMNS;
|
|
|
|
entry_width = (palette->color_area->requisition.width -
|
|
(SPACING * (COLUMNS + 1))) / COLUMNS;
|
|
entry_height = (palette->color_area->requisition.height -
|
|
(SPACING * (ROWS + 1))) / ROWS;
|
|
|
|
x = col * (entry_width + SPACING);
|
|
y = row * (entry_height + SPACING);
|
|
y -= palette->scroll_offset;
|
|
|
|
width = entry_width + SPACING;
|
|
height = entry_height + SPACING;
|
|
|
|
gdk_draw_rectangle (palette->color_area->window, palette->gc,
|
|
0, x, y, width, height);
|
|
|
|
gdk_gc_set_function (palette->gc, GDK_COPY);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_update_current_entry (PaletteP palette)
|
|
{
|
|
if (palette && palette->entries)
|
|
{
|
|
/* Draw the current entry */
|
|
palette_draw_current_entry (palette);
|
|
|
|
/* Update the active color name */
|
|
gtk_entry_set_text (GTK_ENTRY (palette->color_name), palette->color->name);
|
|
}
|
|
}
|
|
|
|
static PaletteEntryP
|
|
palette_add_entry (PaletteEntriesP entries,
|
|
char *name,
|
|
int r,
|
|
int g,
|
|
int b)
|
|
{
|
|
PaletteEntryP entry;
|
|
|
|
if (entries)
|
|
{
|
|
entry = g_malloc (sizeof (_PaletteEntry));
|
|
|
|
entry->color[0] = r;
|
|
entry->color[1] = g;
|
|
entry->color[2] = b;
|
|
if (name)
|
|
entry->name = g_strdup (name);
|
|
else
|
|
entry->name = g_strdup ("Untitled");
|
|
entry->position = entries->n_colors;
|
|
|
|
entries->colors = g_slist_append (entries->colors, entry);
|
|
entries->n_colors += 1;
|
|
|
|
entries->changed = 1;
|
|
|
|
return entry;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
palette_delete_entry (PaletteP palette)
|
|
{
|
|
PaletteEntryP entry;
|
|
GSList *tmp_link;
|
|
int pos;
|
|
|
|
if (palette && palette->entries && palette->color)
|
|
{
|
|
entry = palette->color;
|
|
palette->entries->colors = g_slist_remove (palette->entries->colors, entry);
|
|
palette->entries->n_colors--;
|
|
palette->entries->changed = 1;
|
|
|
|
pos = entry->position;
|
|
palette_entry_free (entry);
|
|
|
|
tmp_link = g_slist_nth (palette->entries->colors, pos);
|
|
|
|
if (tmp_link)
|
|
{
|
|
palette->color = tmp_link->data;
|
|
|
|
while (tmp_link)
|
|
{
|
|
entry = tmp_link->data;
|
|
tmp_link = tmp_link->next;
|
|
entry->position = pos++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
tmp_link = g_slist_nth (palette->entries->colors, pos - 1);
|
|
if (tmp_link)
|
|
palette->color = tmp_link->data;
|
|
}
|
|
|
|
if (palette->entries->n_colors == 0)
|
|
palette->color = palette_add_entry (palette->entries, "Black", 0, 0, 0);
|
|
|
|
palette_calc_scrollbar (palette);
|
|
palette_draw_entries (palette);
|
|
palette_draw_current_entry (palette);
|
|
}
|
|
}
|
|
|
|
static void
|
|
palette_calc_scrollbar (PaletteP palette)
|
|
{
|
|
int n_entries;
|
|
int cur_entry_row;
|
|
int nrows;
|
|
int row_vsize;
|
|
int vsize;
|
|
int page_size;
|
|
int new_offset;
|
|
|
|
if (palette && palette->entries)
|
|
{
|
|
n_entries = palette->entries->n_colors;
|
|
nrows = n_entries / COLUMNS;
|
|
if (n_entries % COLUMNS)
|
|
nrows += 1;
|
|
row_vsize = SPACING + ((palette->color_area->requisition.height -
|
|
(SPACING * (ROWS + 1))) / ROWS);
|
|
vsize = row_vsize * nrows;
|
|
page_size = row_vsize * ROWS;
|
|
|
|
if (palette->color)
|
|
cur_entry_row = palette->color->position / COLUMNS;
|
|
else
|
|
cur_entry_row = 0;
|
|
|
|
new_offset = cur_entry_row * row_vsize;
|
|
|
|
/* scroll only if necessary */
|
|
if (new_offset < palette->scroll_offset)
|
|
{
|
|
palette->scroll_offset = new_offset;
|
|
}
|
|
else if (new_offset > palette->scroll_offset)
|
|
{
|
|
/* only scroll the minimum amount to bring the current color into view */
|
|
if ((palette->scroll_offset + page_size - row_vsize) < new_offset)
|
|
palette->scroll_offset = new_offset - (page_size - row_vsize);
|
|
}
|
|
|
|
/* sanity check to make sure the scrollbar offset is valid */
|
|
if (vsize > page_size)
|
|
if (palette->scroll_offset > (vsize - page_size))
|
|
palette->scroll_offset = vsize - page_size;
|
|
|
|
palette->sbar_data->value = palette->scroll_offset;
|
|
palette->sbar_data->upper = vsize;
|
|
palette->sbar_data->page_size = (page_size < vsize) ? page_size : vsize;
|
|
palette->sbar_data->page_increment = page_size;
|
|
palette->sbar_data->step_increment = row_vsize;
|
|
|
|
gtk_signal_emit_by_name (GTK_OBJECT (palette->sbar_data), "changed");
|
|
}
|
|
}
|
|
|
|
|
|
/* Procedural database entries */
|
|
|
|
/****************************/
|
|
/* PALETTE_GET_FOREGROUND */
|
|
|
|
|
|
static Argument *
|
|
palette_refresh_invoker (Argument *args)
|
|
{
|
|
/* FIXME: I've hardcoded success to be 1, because brushes_init() is a
|
|
* void function right now. It'd be nice if it returned a value at
|
|
* some future date, so we could tell if things blew up when reparsing
|
|
* the list (for whatever reason).
|
|
* - Seth "Yes, this is a kludge" Burgess
|
|
* <sjburges@ou.edu>
|
|
* -and shaemlessly stolen by Adrian Likins for use here...
|
|
*/
|
|
|
|
int success = TRUE ;
|
|
palette_free_palettes();
|
|
palette_init_palettes(FALSE);
|
|
|
|
return procedural_db_return_args (&palette_refresh_proc, success);
|
|
}
|
|
|
|
|
|
ProcRecord palette_refresh_proc =
|
|
{
|
|
"gimp_palette_refresh",
|
|
"Refreshes current palettes",
|
|
"This procedure incorporates all palettes currently in the users palette path. ",
|
|
"Adrian Likins <adrain@gimp.org>",
|
|
"Adrian Likins",
|
|
"1998",
|
|
PDB_INTERNAL,
|
|
|
|
/* input aarguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Output arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Exec mehtos */
|
|
{ { palette_refresh_invoker } },
|
|
};
|
|
|
|
|
|
|
|
static Argument *
|
|
palette_get_foreground_invoker (Argument *args)
|
|
{
|
|
Argument *return_args;
|
|
unsigned char r, g, b;
|
|
unsigned char *col;
|
|
|
|
palette_get_foreground (&r, &g, &b);
|
|
col = (unsigned char *) g_malloc (3);
|
|
col[RED_PIX] = r;
|
|
col[GREEN_PIX] = g;
|
|
col[BLUE_PIX] = b;
|
|
|
|
return_args = procedural_db_return_args (&palette_get_foreground_proc, TRUE);
|
|
return_args[1].value.pdb_pointer = col;
|
|
|
|
return return_args;
|
|
}
|
|
|
|
/* The procedure definition */
|
|
ProcArg palette_get_foreground_args[] =
|
|
{
|
|
{ PDB_COLOR,
|
|
"foreground",
|
|
"the foreground color"
|
|
}
|
|
};
|
|
|
|
ProcRecord palette_get_foreground_proc =
|
|
{
|
|
"gimp_palette_get_foreground",
|
|
"Get the current GIMP foreground color",
|
|
"This procedure retrieves the current GIMP foreground color. The foreground color is used in a variety of tools such as paint tools, blending, and bucket fill.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Output arguments */
|
|
1,
|
|
palette_get_foreground_args,
|
|
|
|
/* Exec method */
|
|
{ { palette_get_foreground_invoker } },
|
|
};
|
|
|
|
|
|
/****************************/
|
|
/* PALETTE_GET_BACKGROUND */
|
|
|
|
static Argument *
|
|
palette_get_background_invoker (Argument *args)
|
|
{
|
|
Argument *return_args;
|
|
unsigned char r, g, b;
|
|
unsigned char *col;
|
|
|
|
palette_get_background (&r, &g, &b);
|
|
col = (unsigned char *) g_malloc (3);
|
|
col[RED_PIX] = r;
|
|
col[GREEN_PIX] = g;
|
|
col[BLUE_PIX] = b;
|
|
|
|
return_args = procedural_db_return_args (&palette_get_background_proc, TRUE);
|
|
return_args[1].value.pdb_pointer = col;
|
|
|
|
return return_args;
|
|
}
|
|
|
|
/* The procedure definition */
|
|
ProcArg palette_get_background_args[] =
|
|
{
|
|
{ PDB_COLOR,
|
|
"background",
|
|
"the background color"
|
|
}
|
|
};
|
|
|
|
ProcRecord palette_get_background_proc =
|
|
{
|
|
"gimp_palette_get_background",
|
|
"Get the current GIMP background color",
|
|
"This procedure retrieves the current GIMP background color. The background color is used in a variety of tools such as blending, erasing (with non-apha images), and image filling.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Output arguments */
|
|
1,
|
|
palette_get_background_args,
|
|
|
|
/* Exec method */
|
|
{ { palette_get_background_invoker } },
|
|
};
|
|
|
|
|
|
/****************************/
|
|
/* PALETTE_SET_FOREGROUND */
|
|
|
|
static Argument *
|
|
palette_set_foreground_invoker (Argument *args)
|
|
{
|
|
unsigned char *color;
|
|
int success;
|
|
|
|
success = TRUE;
|
|
if (success)
|
|
color = (unsigned char *) args[0].value.pdb_pointer;
|
|
|
|
if (success)
|
|
palette_set_foreground (color[RED_PIX], color[GREEN_PIX], color[BLUE_PIX]);
|
|
|
|
return procedural_db_return_args (&palette_set_foreground_proc, success);
|
|
}
|
|
|
|
/* The procedure definition */
|
|
ProcArg palette_set_foreground_args[] =
|
|
{
|
|
{ PDB_COLOR,
|
|
"foreground",
|
|
"the foreground color"
|
|
}
|
|
};
|
|
|
|
ProcRecord palette_set_foreground_proc =
|
|
{
|
|
"gimp_palette_set_foreground",
|
|
"Set the current GIMP foreground color",
|
|
"This procedure sets the current GIMP foreground color. After this is set, operations which use foreground such as paint tools, blending, and bucket fill will use the new value.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
1,
|
|
palette_set_foreground_args,
|
|
|
|
/* Output arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Exec method */
|
|
{ { palette_set_foreground_invoker } },
|
|
};
|
|
|
|
|
|
/****************************/
|
|
/* PALETTE_SET_BACKGROUND */
|
|
|
|
static Argument *
|
|
palette_set_background_invoker (Argument *args)
|
|
{
|
|
unsigned char *color;
|
|
int success;
|
|
|
|
success = TRUE;
|
|
if (success)
|
|
color = (unsigned char *) args[0].value.pdb_pointer;
|
|
|
|
if (success)
|
|
palette_set_background (color[RED_PIX], color[GREEN_PIX], color[BLUE_PIX]);
|
|
|
|
return procedural_db_return_args (&palette_set_background_proc, success);
|
|
}
|
|
|
|
/* The procedure definition */
|
|
ProcArg palette_set_background_args[] =
|
|
{
|
|
{ PDB_COLOR,
|
|
"background",
|
|
"the background color"
|
|
}
|
|
};
|
|
|
|
ProcRecord palette_set_background_proc =
|
|
{
|
|
"gimp_palette_set_background",
|
|
"Set the current GIMP background color",
|
|
"This procedure sets the current GIMP background color. After this is set, operations which use background such as blending, filling images, clearing, and erasing (in non-alpha images) will use the new value.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
1,
|
|
palette_set_background_args,
|
|
|
|
/* Output arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Exec method */
|
|
{ { palette_set_background_invoker } },
|
|
};
|
|
|
|
/*******************************/
|
|
/* PALETTE_SET_DEFAULT_COLORS */
|
|
|
|
static Argument *
|
|
palette_set_default_colors_invoker (Argument *args)
|
|
{
|
|
int success;
|
|
|
|
success = TRUE;
|
|
if (success)
|
|
palette_set_default_colors ();
|
|
|
|
return procedural_db_return_args (&palette_set_background_proc, success);
|
|
}
|
|
|
|
/* The procedure definition */
|
|
|
|
ProcRecord palette_set_default_colors_proc =
|
|
{
|
|
"gimp_palette_set_default_colors",
|
|
"Set the current GIMP foreground and background colors to black and white",
|
|
"This procedure sets the current GIMP foreground and background colors to their initial default values, black and white.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Output arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Exec method */
|
|
{ { palette_set_default_colors_invoker } },
|
|
};
|
|
|
|
/************************/
|
|
/* PALETTE_SWAP_COLORS */
|
|
|
|
static Argument *
|
|
palette_swap_colors_invoker (Argument *args)
|
|
{
|
|
int success;
|
|
|
|
success = TRUE;
|
|
if (success)
|
|
palette_swap_colors ();
|
|
|
|
return procedural_db_return_args (&palette_swap_colors_proc, success);
|
|
}
|
|
|
|
/* The procedure definition */
|
|
|
|
ProcRecord palette_swap_colors_proc =
|
|
{
|
|
"gimp_palette_swap_colors",
|
|
"Swap the current GIMP foreground and background colors",
|
|
"This procedure swaps the current GIMP foreground and background colors, so that the new foreground color becomes the old background color and vice versa.",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"Spencer Kimball & Peter Mattis",
|
|
"1995-1996",
|
|
PDB_INTERNAL,
|
|
|
|
/* Input arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Output arguments */
|
|
0,
|
|
NULL,
|
|
|
|
/* Exec method */
|
|
{ { palette_swap_colors_invoker } },
|
|
};
|