gimp/plug-ins/print.c

1390 lines
38 KiB
C

/*
* "$Id$"
*
* Print plug-in for the GIMP.
*
* Copyright 1997 Michael Sweet (mike@easysw.com)
*
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*
* Contents:
*
* main() - Main entry - just call gimp_main()...
* query() - Respond to a plug-in query...
* run() - Run the plug-in...
* print_dialog() - Pop up the print dialog...
* dialog_create_ivalue() - Create an integer value control...
* dialog_iscale_update() - Update the value field using the scale.
* dialog_ientry_update() - Update the value field using the text entry.
* print_driver_callback() - Update the current printer driver...
* media_size_callback() - Update the current media size...
* print_command_callback() - Update the print command...
* output_type_callback() - Update the current output type...
* print_callback() - Start the print...
* cancel_callback() - Cancel the print...
* close_callback() - Exit the print dialog application.
*
* Revision History:
*
* $Log$
* Revision 1.1.1.1 1997/11/24 22:03:35 sopwith
* Let's try this import one last time.
*
* Revision 1.3 1997/10/03 22:18:15 nobody
* updated print er too the latest version
*
* : ----------------------------------------------------------------------
*
* Revision 1.8 1997/10/02 17:57:26 mike
* Added printrc support.
* Added printer list (spooler support).
* Added gamma/dot gain correction values for all printers.
*
* Revision 1.7 1997/07/30 20:33:05 mike
* Final changes for 1.1 release.
*
* Revision 1.6 1997/07/30 18:47:39 mike
* Added scaling, orientation, and offset options.
* Added first cut at preview window.
*
* Revision 1.5 1997/07/26 18:38:23 mike
* Whoops - wasn't grabbing the colormap for indexed images properly...
*
* Revision 1.4 1997/07/03 13:13:26 mike
* Updated documentation for 1.0 release.
*
* Revision 1.3 1997/07/03 13:07:05 mike
* Updated EPSON driver short names.
* Changed brightness lut formula for better control.
*
* Revision 1.2 1997/07/02 15:22:17 mike
* Added GUI with printer/media/output selection controls.
*
* Revision 1.1 1997/07/02 13:51:53 mike
* Initial revision
*/
#include "print.h"
#include <math.h>
#include <signal.h>
/*
* Constants for GUI...
*/
#define SCALE_WIDTH 64
#define ENTRY_WIDTH 64
#define PREVIEW_SIZE 180 /* Assuming max media size of 18" */
#define MAX_PLIST 100
/*
* Types...
*/
typedef struct /**** Printer List ****/
{
char name[17], /* Name of printer */
command[255], /* Printer command */
driver[33]; /* Short name of printer driver */
int output_type; /* Color/B&W */
} plist_t;
/*
* Local functions...
*/
static void printrc_load(void);
static void printrc_save(void);
static int compare_printers(plist_t *p1, plist_t *p2);
static void get_printers(void);
static void query(void);
static void run(char *, int, GParam *, int *, GParam **);
static int print_dialog(void);
static void dialog_create_ivalue(char *, GtkTable *, int, gint *, int, int);
static void dialog_iscale_update(GtkAdjustment *, gint *);
static void dialog_ientry_update(GtkWidget *, gint *);
static void plist_callback(GtkWidget *, gint);
static void print_driver_callback(GtkWidget *, gint);
static void media_size_callback(GtkWidget *, gint);
static void print_command_callback(GtkWidget *, gpointer);
static void output_type_callback(GtkWidget *, gint);
static void orientation_callback(GtkWidget *, gint);
static void print_callback(GtkWidget *, gpointer);
static void cancel_callback(GtkWidget *, gpointer);
static void close_callback(GtkWidget *, gpointer);
static void preview_update(void);
static void preview_button_callback(GtkWidget *, GdkEventButton *);
static void preview_motion_callback(GtkWidget *, GdkEventMotion *);
/*
* Globals...
*/
GPlugInInfo PLUG_IN_INFO = /* Plug-in information */
{
NULL, /* init_proc */
NULL, /* quit_proc */
query, /* query_proc */
run, /* run_proc */
};
struct /* Plug-in variables */
{
char output_to[255], /* Name of file or command to print to */
short_name[33]; /* Name of printer "driver" */
gint media_size, /* Size of output media */
output_type, /* Color or grayscale output */
brightness, /* Output brightness */
scaling, /* Scaling, percent of printable area */
orientation, /* Orientation - 0 = port., 1 = land., -1 = auto */
left, /* Offset from lower-lefthand corner, 10ths */
top; /* ... */
} vars =
{
"|lp", /* Name of file or command to print to */
"ps", /* Name of printer "driver" */
MEDIA_LETTER, /* Size of output media */
OUTPUT_COLOR, /* Color or grayscale output */
100, /* Output brightness */
100, /* Scaling (100% means entire printable area */
-1, /* Orientation (-1 = automatic) */
-1, /* X offset (-1 = center) */
-1 /* Y offset (-1 = center) */
};
GtkWidget *printer_driver, /* Printer driver widget */
*output_gray, /* Output type toggle, black */
*output_color, /* Output type toggle, color */
*output_to; /* Output file/command text entry */
GtkDrawingArea *preview; /* Preview drawing area widget */
int mouse_x, /* Last mouse X */
mouse_y; /* Last mouse Y */
int image_width, /* Width of image */
image_height, /* Height of image */
page_left, /* Left pixel column of page */
page_top, /* Top pixel row of page */
page_width, /* Width of page on screen */
page_height, /* Height of page on screen */
print_width, /* Printed width of image */
print_height; /* Printed height of image */
int plist_current = 0, /* Current system printer */
plist_count = 0; /* Number of system printers */
plist_t plist[MAX_PLIST]; /* System printers */
int runme = FALSE, /* True if print should proceed */
current_printer = 0; /* Current printer index */
printer_t printers[] = /* List of supported printer types */
{
{ "PostScript Printer", "ps", 72, 72, 1, 1, 0, 1.000, 1.000, ps_print },
{ "HP DeskJet 500, 520", "pcl-500", 300, 300, 0, 0, 500, 0.541, 0.548, pcl_print },
{ "HP DeskJet 500C, 540C", "pcl-501", 300, 300, 0, 1, 501, 0.541, 0.548, pcl_print },
{ "HP DeskJet 550C, 560C", "pcl-550", 300, 300, 0, 1, 550, 0.541, 0.548, pcl_print },
{ "HP DeskJet 600 series", "pcl-600", 600, 300, 0, 1, 600, 0.541, 0.548, pcl_print },
{ "HP DeskJet 800 series", "pcl-800", 600, 600, 0, 1, 800, 0.541, 0.548, pcl_print },
{ "HP DeskJet 1200C, 1600C", "pcl-1200", 300, 300, 0, 1, 1200, 0.541, 0.548, pcl_print },
{ "HP LaserJet II series", "pcl-2", 300, 300, 0, 0, 2, 0.563, 0.596, pcl_print },
{ "HP LaserJet III series", "pcl-3", 300, 300, 0, 0, 3, 0.563, 0.596, pcl_print },
{ "HP LaserJet 4 series", "pcl-4", 600, 600, 1, 0, 4, 0.599, 0.615, pcl_print },
{ "HP LaserJet 5 series", "pcl-5", 600, 600, 1, 0, 4, 0.599, 0.615, pcl_print },
{ "HP LaserJet 6 series", "pcl-6", 600, 600, 1, 0, 4, 0.599, 0.615, pcl_print },
{ "EPSON Stylus Color", "escp2", 720, 720, 0, 1, 0, 0.325, 0.478, escp2_print },
{ "EPSON Stylus Color Pro", "escp2-pro", 720, 720, 0, 1, 1, 0.325, 0.478, escp2_print },
{ "EPSON Stylus Color Pro XL","escp2-proxl", 720, 720, 1, 1, 1, 0.325, 0.478, escp2_print },
{ "EPSON Stylus Color 1500", "escp2-1500", 720, 720, 1, 1, 2, 0.325, 0.478, escp2_print },
{ "EPSON Stylus Color 400", "escp2-400", 720, 720, 0, 1, 1, 0.530, 0.482, escp2_print },
{ "EPSON Stylus Color 500", "escp2-500", 720, 720, 0, 1, 1, 0.530, 0.482, escp2_print },
{ "EPSON Stylus Color 600", "escp2-600", 720, 720, 0, 1, 3, 0.530, 0.482, escp2_print },
{ "EPSON Stylus Color 800", "escp2-800", 720, 720, 0, 1, 4, 0.530, 0.482, escp2_print },
{ "EPSON Stylus Color 1520", "escp2-1520", 720, 720, 1, 1, 4, 0.530, 0.482, escp2_print },
{ "EPSON Stylus Color 3000", "escp2-3000", 720, 720, 1, 1, 4, 0.530, 0.482, escp2_print }
};
/*
* 'main()' - Main entry - just call gimp_main()...
*/
int
main(int argc, /* I - Number of command-line args */
char *argv[]) /* I - Command-line args */
{
return (gimp_main(argc, argv));
}
/*
* 'query()' - Respond to a plug-in query...
*/
static void
query(void)
{
static GParamDef args[] =
{
{ PARAM_INT32, "run_mode", "Interactive, non-interactive" },
{ PARAM_IMAGE, "image", "Input image" },
{ PARAM_DRAWABLE, "drawable", "Input drawable" },
{ PARAM_STRING, "output_to", "Print command or filename (| to pipe to command)" },
{ PARAM_STRING, "driver", "Printer driver short name" },
{ PARAM_INT32, "media_size", "Media size (0 = A/letter, 1 = legal, 2 = B/tabloid, 3 = A4, 4 = A3)" },
{ PARAM_INT32, "output_type", "Output type (0 = gray, 1 = color)" },
{ PARAM_INT32, "brightness", "Brightness (0-200%)" },
{ PARAM_INT32, "scaling", "Output scaling (0-100%)" },
{ PARAM_INT32, "orientation", "Output orientation (-1 = auto, 0 = portrait, 1 = landscape)" },
{ PARAM_INT32, "left", "Left offset (10ths inches, -1 = centered)" },
{ PARAM_INT32, "top", "Top offset (10ths inches, -1 = centered)" }
};
static int nargs = sizeof(args) / sizeof(args[0]);
gimp_install_procedure(
"file_print",
"This plug-in prints images from The GIMP.",
"Prints images to PostScript, PCL, or ESC/P2 printers.",
"Michael Sweet <mike@easysw.com>",
"Copyright 1997 by Michael Sweet",
PLUG_IN_VERSION,
"<Image>/File/Print...",
"RGB*,GRAY*INDEXED*",
PROC_PLUG_IN,
nargs,
0,
args,
NULL);
}
/*
* 'run()' - Run the plug-in...
*/
static void
run(char *name, /* I - Name of print program. */
int nparams, /* I - Number of parameters passed in */
GParam *param, /* I - Parameter values */
int *nreturn_vals, /* O - Number of return values */
GParam **return_vals) /* O - Return values */
{
GDrawable *drawable; /* Drawable for image */
GRunModeType run_mode; /* Current run mode */
FILE *prn; /* Print file/command */
printer_t *printer; /* Printer driver entry */
int i, j; /* Looping vars */
float brightness; /* Computed brightness */
guchar lut[256]; /* Lookup table for brightness */
guchar *cmap; /* Colormap (indexed images only) */
int ncolors; /* Number of colors in colormap */
static GParam values[1]; /* Return values */
/*
* Initialize parameter data...
*/
run_mode = param[0].data.d_int32;
values[0].type = PARAM_STATUS;
values[0].data.d_status = STATUS_SUCCESS;
*nreturn_vals = 1;
*return_vals = values;
/*
* Get drawable...
*/
drawable = gimp_drawable_get(param[2].data.d_drawable);
image_width = drawable->width;
image_height = drawable->height;
/*
* See how we will run
*/
switch (run_mode)
{
case RUN_INTERACTIVE :
/*
* Possibly retrieve data...
*/
gimp_get_data(PLUG_IN_NAME, &vars);
for (i = 0; i < (sizeof(printers) / sizeof(printers[0])); i ++)
if (strcmp(printers[i].short_name, vars.short_name) == 0)
current_printer = i;
/*
* Get information from the dialog...
*/
if (!print_dialog())
return;
break;
case RUN_NONINTERACTIVE :
/*
* Make sure all the arguments are present...
*/
if (nparams < 8)
values[0].data.d_status = STATUS_CALLING_ERROR;
else
{
strcpy(vars.output_to, param[3].data.d_string);
strcpy(vars.short_name, param[4].data.d_string);
vars.media_size = param[5].data.d_int32;
vars.output_type = param[6].data.d_int32;
vars.brightness = param[7].data.d_int32;
if (nparams > 8)
vars.scaling = param[8].data.d_int32;
else
vars.scaling = 100;
if (nparams > 9)
vars.orientation = param[9].data.d_int32;
else
vars.orientation = -1;
if (nparams > 10)
vars.left = param[10].data.d_int32;
else
vars.left = -1;
if (nparams > 11)
vars.top = param[11].data.d_int32;
else
vars.top = -1;
for (i = 0; i < (sizeof(printers) / sizeof(printers[0])); i ++)
if (strcmp(printers[i].short_name, vars.short_name) == 0)
current_printer = i;
};
break;
case RUN_WITH_LAST_VALS :
/*
* Possibly retrieve data...
*/
gimp_get_data(PLUG_IN_NAME, &vars);
for (i = 0; i < (sizeof(printers) / sizeof(printers[0])); i ++)
if (strcmp(printers[i].short_name, vars.short_name) == 0)
current_printer = i;
break;
default :
values[0].data.d_status = STATUS_CALLING_ERROR;
break;;
};
/*
* Print the image...
*/
signal(SIGBUS, SIG_DFL);
signal(SIGSEGV, SIG_DFL);
if (values[0].data.d_status == STATUS_SUCCESS)
{
/*
* Set the tile cache size...
*/
gimp_tile_cache_ntiles((drawable->width + gimp_tile_width() - 1) /
gimp_tile_width());
/*
* Open the file/execute the print command...
*/
if (vars.output_to[0] == '|')
prn = popen(vars.output_to + 1, "w");
else
prn = fopen(vars.output_to, "w");
if (prn != NULL)
{
/*
* Got an output file/command, now compute a brightness lookup table...
*/
printer = printers + current_printer;
brightness = 25500.0 * printer->density / vars.brightness;
for (i = 0; i < 256; i ++)
{
j = 255.5 - brightness * (1.0 - pow((float)i / 255.0, printer->gamma));
if (j < 0)
lut[i] = 0;
else if (j < 255)
lut[i] = j;
else
lut[i] = 255;
};
/*
* Is the image an Indexed type? If so we need the colormap...
*/
if (gimp_image_base_type(param[1].data.d_image) == INDEXED)
cmap = gimp_image_get_cmap(param[1].data.d_image, &ncolors);
else
{
cmap = NULL;
ncolors = 0;
};
/*
* Finally, call the print driver to send the image to the printer and
* close the output file/command...
*/
(*printer->print)(prn, drawable, vars.media_size, printer->xdpi, printer->ydpi,
vars.output_type, printer->model, lut, cmap,
vars.orientation, vars.scaling, vars.left, vars.top);
if (vars.output_to[0] == '|')
pclose(prn);
else
fclose(prn);
}
else
values[0].data.d_status == STATUS_EXECUTION_ERROR;
/*
* Store data...
*/
if (run_mode == RUN_INTERACTIVE)
gimp_set_data(PLUG_IN_NAME, &vars, sizeof(vars));
};
/*
* Detach from the drawable...
*/
gimp_drawable_detach(drawable);
}
/*
* 'print_dialog()' - Pop up the print dialog...
*/
int
print_dialog(void)
{
int i; /* Looping var */
GtkWidget *dialog, /* Dialog window */
*table, /* Table "container" for controls */
*label, /* Label string */
*button, /* OK/Cancel buttons */
*entry, /* Print file/command text entry */
*menu, /* Menu of drivers/sizes */
*item, /* Menu item */
*option; /* Option menu button */
GSList *group; /* Grouping for output type */
gint argc; /* Fake argc for GUI */
gchar **argv; /* Fake argv for GUI */
static char *sizes[] = /* Size strings */
{
"A/Letter (8.5x11\")",
"Legal (8.5x14\")",
"B/Tabloid (11x17\")",
"A4 (210x297mm)",
"A3 (297x420mm)"
};
static char *orients[] = /* Orientation strings */
{
"Auto",
"Portrait",
"Landscape"
};
/*
* Initialize the program's display...
*/
argc = 1;
argv = g_new(gchar *, 1);
argv[0] = g_strdup("print");
gtk_init(&argc, &argv);
gtk_rc_parse(gimp_gtkrc());
gdk_set_use_xshm(gimp_use_xshm());
/*
* Get printrc options...
*/
printrc_load();
/*
* Dialog window...
*/
dialog = gtk_dialog_new();
gtk_window_set_title(GTK_WINDOW(dialog), "Print");
gtk_window_position(GTK_WINDOW(dialog), GTK_WIN_POS_MOUSE);
gtk_container_border_width(GTK_CONTAINER(dialog), 0);
gtk_signal_connect(GTK_OBJECT(dialog), "destroy",
(GtkSignalFunc) close_callback,
NULL);
/*
* Top-level table for dialog...
*/
table = gtk_table_new(13, 3, FALSE);
gtk_container_border_width(GTK_CONTAINER(table), 6);
gtk_table_set_row_spacings(GTK_TABLE(table), 4);
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), table, FALSE, FALSE, 0);
gtk_widget_show(table);
/*
* Drawing area for page preview...
*/
preview = (GtkDrawingArea *)gtk_drawing_area_new();
gtk_drawing_area_size(preview, PREVIEW_SIZE, PREVIEW_SIZE);
gtk_table_attach(GTK_TABLE(table), (GtkWidget *)preview, 0, 3, 0, 4, 0, 0, 0, 0);
gtk_signal_connect(GTK_OBJECT((GtkWidget *)preview), "expose_event",
(GtkSignalFunc)preview_update,
NULL);
gtk_signal_connect(GTK_OBJECT((GtkWidget *)preview), "button_press_event",
(GtkSignalFunc)preview_button_callback,
NULL);
gtk_signal_connect(GTK_OBJECT((GtkWidget *)preview), "button_release_event",
(GtkSignalFunc)preview_button_callback,
NULL);
gtk_signal_connect(GTK_OBJECT((GtkWidget *)preview), "motion_notify_event",
(GtkSignalFunc)preview_motion_callback,
NULL);
gtk_widget_show((GtkWidget *)preview);
gtk_widget_set_events((GtkWidget *)preview,
GDK_EXPOSURE_MASK | GDK_BUTTON_MOTION_MASK |
GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK);
/*
* Printer option menu...
*/
label = gtk_label_new("Printer:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 4, 5, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
menu = gtk_menu_new();
for (i = 0; i < plist_count; i ++)
{
if (strcmp(plist[i].command, vars.output_to) == 0)
plist_current = i;
item = gtk_menu_item_new_with_label(plist[i].name);
gtk_menu_append(GTK_MENU(menu), item);
gtk_signal_connect(GTK_OBJECT(item), "activate",
(GtkSignalFunc)plist_callback,
(gpointer)i);
gtk_widget_show(item);
};
option = gtk_option_menu_new();
gtk_table_attach(GTK_TABLE(table), option, 1, 3, 4, 5,
GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
gtk_option_menu_set_menu(GTK_OPTION_MENU(option), menu);
gtk_option_menu_set_history(GTK_OPTION_MENU(option), plist_current);
gtk_widget_show(option);
/*
* Printer driver option menu...
*/
label = gtk_label_new("Driver:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 5, 6, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
menu = gtk_menu_new();
for (i = 0; i < (sizeof(printers) / sizeof(printers[0])); i ++)
{
item = gtk_menu_item_new_with_label(printers[i].long_name);
gtk_menu_append(GTK_MENU(menu), item);
gtk_signal_connect(GTK_OBJECT(item), "activate",
(GtkSignalFunc)print_driver_callback,
(gpointer)i);
gtk_widget_show(item);
};
printer_driver = option = gtk_option_menu_new();
gtk_table_attach(GTK_TABLE(table), option, 1, 3, 5, 6,
GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
gtk_option_menu_set_menu(GTK_OPTION_MENU(option), menu);
gtk_option_menu_set_history(GTK_OPTION_MENU(option), current_printer);
gtk_widget_show(option);
/*
* Output type toggles...
*/
label = gtk_label_new("Output Type:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 6, 7, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
output_gray = button = gtk_radio_button_new_with_label(NULL, "B&W");
group = gtk_radio_button_group(GTK_RADIO_BUTTON(button));
gtk_signal_connect(GTK_OBJECT(button), "toggled",
(GtkSignalFunc)output_type_callback,
(gpointer)OUTPUT_GRAY);
gtk_table_attach(GTK_TABLE(table), button, 1, 2, 6, 7, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(button);
if (vars.output_type == 0)
gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button), TRUE);
output_color = button = gtk_radio_button_new_with_label(group, "Color");
gtk_signal_connect(GTK_OBJECT(button), "toggled",
(GtkSignalFunc)output_type_callback,
(gpointer)OUTPUT_COLOR);
gtk_table_attach(GTK_TABLE(table), button, 2, 3, 6, 7, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(button);
if (vars.output_type == 1)
gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button), TRUE);
/*
* Media size option menu...
*/
label = gtk_label_new("Media Size:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 7, 8, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
menu = gtk_menu_new();
for (i = 0; i < (sizeof(sizes) / sizeof(sizes[0])); i ++)
{
item = gtk_menu_item_new_with_label(sizes[i]);
gtk_menu_append(GTK_MENU(menu), item);
gtk_signal_connect(GTK_OBJECT(item), "activate",
(GtkSignalFunc)media_size_callback,
(gpointer)i);
gtk_widget_show(item);
};
option = gtk_option_menu_new();
gtk_table_attach(GTK_TABLE(table), option, 1, 3, 7, 8,
GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
gtk_option_menu_set_menu(GTK_OPTION_MENU(option), menu);
gtk_option_menu_set_history(GTK_OPTION_MENU(option), vars.media_size);
gtk_widget_show(option);
/*
* Orientation option menu...
*/
label = gtk_label_new("Orientation:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 8, 9, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
menu = gtk_menu_new();
for (i = 0; i < (sizeof(orients) / sizeof(orients[0])); i ++)
{
item = gtk_menu_item_new_with_label(orients[i]);
gtk_menu_append(GTK_MENU(menu), item);
gtk_signal_connect(GTK_OBJECT(item), "activate",
(GtkSignalFunc)orientation_callback,
(gpointer)(i - 1));
gtk_widget_show(item);
};
option = gtk_option_menu_new();
gtk_table_attach(GTK_TABLE(table), option, 1, 3, 8, 9,
GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
gtk_option_menu_set_menu(GTK_OPTION_MENU(option), menu);
gtk_option_menu_set_history(GTK_OPTION_MENU(option), vars.orientation + 1);
gtk_widget_show(option);
/*
* Scaling slider...
*/
dialog_create_ivalue("Scaling:", GTK_TABLE(table), 9, &(vars.scaling), 1, 101);
/*
* Brightness slider...
*/
dialog_create_ivalue("Brightness:", GTK_TABLE(table), 10, &(vars.brightness), 50, 201);
/*
* Print file/command...
*/
label = gtk_label_new("File/|Command:");
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 11, 12, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
output_to = entry = gtk_entry_new();
gtk_entry_set_text(GTK_ENTRY(entry), vars.output_to);
gtk_signal_connect(GTK_OBJECT(entry), "changed",
(GtkSignalFunc)print_command_callback, NULL);
gtk_table_attach(GTK_TABLE(table), entry, 1, 3, 11, 12, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(entry);
/*
* Print, cancel buttons...
*/
gtk_container_border_width(GTK_CONTAINER(GTK_DIALOG(dialog)->action_area), 6);
button = gtk_button_new_with_label("Print");
GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT);
gtk_signal_connect(GTK_OBJECT(button), "clicked",
(GtkSignalFunc)print_callback,
dialog);
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->action_area), button, TRUE, TRUE, 0);
gtk_widget_grab_default(button);
gtk_widget_show(button);
button = gtk_button_new_with_label("Cancel");
GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT);
gtk_signal_connect(GTK_OBJECT(button), "clicked",
(GtkSignalFunc)cancel_callback,
dialog);
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->action_area), button, TRUE, TRUE, 0);
gtk_widget_show(button);
/*
* Show it and wait for the user to do something...
*/
gtk_widget_show(dialog);
gtk_main();
gdk_flush();
/*
* Set printrc options...
*/
printrc_save();
/*
* Return ok/cancel...
*/
return (runme);
}
/*
* 'dialog_create_ivalue()' - Create an integer value control...
*/
static void
dialog_create_ivalue(char *title, /* I - Label for control */
GtkTable *table, /* I - Table container to use */
int row, /* I - Row # for container */
gint *value, /* I - Value holder */
int left, /* I - Minimum value for slider */
int right) /* I - Maximum value for slider */
{
GtkWidget *label, /* Control label */
*scale, /* Scale widget */
*entry; /* Text widget */
GtkObject *scale_data; /* Scale data */
char buf[256]; /* String buffer */
/*
* Label...
*/
label = gtk_label_new(title);
gtk_misc_set_alignment(GTK_MISC(label), 0.0, 1.0);
gtk_table_attach(table, label, 0, 1, row, row + 1, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(label);
/*
* Scale...
*/
scale_data = gtk_adjustment_new(*value, left, right, 1.0, 1.0, 1.0);
gtk_signal_connect(GTK_OBJECT(scale_data), "value_changed",
(GtkSignalFunc) dialog_iscale_update,
value);
scale = gtk_hscale_new(GTK_ADJUSTMENT(scale_data));
gtk_widget_set_usize(scale, SCALE_WIDTH, 0);
gtk_table_attach(table, scale, 1, 2, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0);
gtk_scale_set_draw_value(GTK_SCALE(scale), FALSE);
gtk_range_set_update_policy(GTK_RANGE(scale), GTK_UPDATE_CONTINUOUS);
gtk_widget_show(scale);
/*
* Text entry...
*/
entry = gtk_entry_new();
gtk_object_set_user_data(GTK_OBJECT(entry), scale_data);
gtk_object_set_user_data(scale_data, entry);
gtk_widget_set_usize(entry, ENTRY_WIDTH, 0);
sprintf(buf, "%d", *value);
gtk_entry_set_text(GTK_ENTRY(entry), buf);
gtk_signal_connect(GTK_OBJECT(entry), "changed",
(GtkSignalFunc) dialog_ientry_update,
value);
gtk_table_attach(GTK_TABLE(table), entry, 2, 3, row, row + 1, GTK_FILL, GTK_FILL, 4, 0);
gtk_widget_show(entry);
}
/*
* 'dialog_iscale_update()' - Update the value field using the scale.
*/
static void
dialog_iscale_update(GtkAdjustment *adjustment, /* I - New value */
gint *value) /* I - Current value */
{
GtkWidget *entry; /* Text entry widget */
char buf[256]; /* Text buffer */
if (*value != adjustment->value)
{
*value = adjustment->value;
entry = gtk_object_get_user_data(GTK_OBJECT(adjustment));
sprintf(buf, "%d", *value);
gtk_signal_handler_block_by_data(GTK_OBJECT(entry), value);
gtk_entry_set_text(GTK_ENTRY(entry), buf);
gtk_signal_handler_unblock_by_data(GTK_OBJECT(entry), value);
if (value == &(vars.scaling))
preview_update();
};
}
/*
* 'dialog_ientry_update()' - Update the value field using the text entry.
*/
static void
dialog_ientry_update(GtkWidget *widget, /* I - Entry widget */
gint *value) /* I - Current value */
{
GtkAdjustment *adjustment;
gint new_value;
new_value = atoi(gtk_entry_get_text(GTK_ENTRY(widget)));
if (*value != new_value)
{
adjustment = gtk_object_get_user_data(GTK_OBJECT(widget));
if ((new_value >= adjustment->lower) &&
(new_value <= adjustment->upper))
{
*value = new_value;
adjustment->value = new_value;
gtk_signal_emit_by_name(GTK_OBJECT(adjustment), "value_changed");
};
};
}
/*
* 'plist_callback()' - Update the current system printer...
*/
static void
plist_callback(GtkWidget *widget, /* I - Driver option menu */
gint data) /* I - Data */
{
int i;
plist_current = data;
if (plist[plist_current].driver[0] != '\0')
{
strcpy(vars.short_name, plist[plist_current].driver);
for (i = 0; i < (sizeof(printers) / sizeof(printers[0])); i ++)
if (strcmp(printers[i].short_name, vars.short_name) == 0)
{
current_printer = i;
break;
};
gtk_option_menu_set_history(GTK_OPTION_MENU(printer_driver), current_printer);
};
if (plist[plist_current].command[0] != '\0')
{
strcpy(vars.output_to, plist[plist_current].command);
gtk_entry_set_text(GTK_ENTRY(output_to), vars.output_to);
};
if (plist[plist_current].output_type == OUTPUT_GRAY)
gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(output_gray), TRUE);
else
gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(output_color), TRUE);
}
/*
* 'print_driver_callback()' - Update the current printer driver...
*/
static void
print_driver_callback(GtkWidget *widget, /* I - Driver option menu */
gint data) /* I - Data */
{
current_printer = data;
strcpy(vars.short_name, printers[current_printer].short_name);
strcpy(plist[plist_current].driver, printers[current_printer].short_name);
}
/*
* 'media_size_callback()' - Update the current media size...
*/
static void
media_size_callback(GtkWidget *widget, /* I - Media size option menu */
gint data) /* I - Data */
{
vars.media_size = data;
vars.left = -1;
vars.top = -1;
preview_update();
}
/*
* 'orientation_callback()' - Update the current media size...
*/
static void
orientation_callback(GtkWidget *widget, /* I - Orientation option menu */
gint data) /* I - Data */
{
vars.orientation = data;
vars.left = -1;
vars.top = -1;
preview_update();
}
/*
* 'print_command_callback()' - Update the print command...
*/
static void
print_command_callback(GtkWidget *widget, /* I - Command text field */
gpointer data) /* I - Data */
{
strcpy(vars.output_to, gtk_entry_get_text(GTK_ENTRY(widget)));
strcpy(plist[plist_current].command, vars.output_to);
}
/*
* 'output_type_callback()' - Update the current output type...
*/
static void
output_type_callback(GtkWidget *widget, /* I - Output type button */
gint data) /* I - Data */
{
if (GTK_TOGGLE_BUTTON(widget)->active)
{
vars.output_type = data;
plist[plist_current].output_type = data;
};
}
/*
* 'print_callback()' - Start the print...
*/
static void
print_callback(GtkWidget *widget, /* I - OK button widget */
gpointer data) /* I - Dialog window */
{
runme = TRUE;
gtk_widget_destroy(GTK_WIDGET(data));
}
/*
* 'cancel_callback()' - Cancel the print...
*/
static void
cancel_callback(GtkWidget *widget, /* I - Cancel button widget */
gpointer data) /* I - Dialog window */
{
gtk_widget_destroy(GTK_WIDGET(data));
}
/*
* 'close_callback()' - Exit the print dialog application.
*/
static void
close_callback(GtkWidget *widget, /* I - Dialog window */
gpointer data) /* I - Dialog window */
{
gtk_main_quit();
}
static void
preview_update(void)
{
int orient, /* True orientation of page */
tw0, tw1, /* Temporary page_widths */
th0, th1, /* Temporary page_heights */
ta0, ta1; /* Temporary areas */
static GdkGC *gc = NULL;
gdk_window_clear(preview->widget.window);
if (gc == NULL)
gc = gdk_gc_new(preview->widget.window);
page_width = media_width(vars.media_size, 10);
page_height = media_height(vars.media_size, 10);
if (vars.orientation == ORIENT_AUTO)
{
tw0 = page_width * vars.scaling / 100;
th0 = tw0 * image_height / image_width;
if (th0 > page_height)
{
th0 = page_height;
tw0 = th0 * image_width / image_height;
};
ta0 = tw0 * th0;
tw1 = page_height * vars.scaling / 100;
th1 = tw1 * image_height / image_width;
if (th1 > page_width)
{
th1 = page_width;
tw1 = th1 * image_width / image_height;
};
ta1 = tw1 * th1;
if (ta0 >= ta1)
orient = ORIENT_PORTRAIT;
else
orient = ORIENT_LANDSCAPE;
}
else
orient = vars.orientation;
if (orient == ORIENT_LANDSCAPE)
{
page_width = media_height(vars.media_size, 10);
page_height = media_width(vars.media_size, 10);
};
page_left = (PREVIEW_SIZE - page_width) / 2;
page_top = (PREVIEW_SIZE - page_height) / 2;
gdk_draw_rectangle(preview->widget.window, gc, 0,
(PREVIEW_SIZE - page_width - 5) / 2,
(PREVIEW_SIZE - page_height - 10) / 2,
page_width + 5, page_height + 10);
print_width = page_width * vars.scaling / 100;
print_height = print_width * image_height / image_width;
if (print_height > page_height)
{
print_height = page_height;
print_width = print_height * image_width / image_height;
};
if (vars.left < 0 || vars.top < 0)
gdk_draw_rectangle(preview->widget.window, gc, 1,
(PREVIEW_SIZE - print_width) / 2,
(PREVIEW_SIZE - print_height) / 2,
print_width, print_height);
else
{
if (vars.left > (page_width - print_width))
vars.left = page_width - print_width;
if (vars.top > (page_height - print_height))
vars.top = page_height - print_height;
gdk_draw_rectangle(preview->widget.window, gc, 1,
page_left + vars.left,
page_top + vars.top,
print_width, print_height);
};
gdk_flush();
}
static void
preview_button_callback(GtkWidget *w,
GdkEventButton *event)
{
mouse_x = event->x;
mouse_y = event->y;
}
static void
preview_motion_callback(GtkWidget *w,
GdkEventMotion *event)
{
if (vars.left < 0 || vars.top < 0)
{
vars.left = (page_width - print_width) / 2;
vars.top = (page_height - print_height) / 2;
};
vars.left += event->x - mouse_x;
vars.top += event->y - mouse_y;
if (vars.left < 0)
vars.left = 0;
if (vars.top < 0)
vars.top = 0;
preview_update();
mouse_x = event->x;
mouse_y = event->y;
}
/*
* 'printrc_load()' - Load the printer resource configuration file.
*/
static void
printrc_load(void)
{
FILE *fp; /* Printrc file */
char line[1024]; /* Line in printrc file */
plist_t *p, /* Current printer */
key; /* Search key */
/*
* Get the printer list...
*/
get_printers();
/*
* Generate the filename for the current user...
*/
if (getenv("HOME") == NULL)
strcpy(line, "/.gimp/printrc");
else
sprintf(line, "%s/.gimp/printrc", getenv("HOME"));
if ((fp = fopen(line, "r")) != NULL)
{
/*
* File exists - read the contents and update the printer list...
*/
while (fgets(line, sizeof(line), fp) != NULL)
{
if (line[0] == '#')
continue; /* Comment */
if (sscanf(line, "%s%s%d%*[ \t]%[^\n]", key.name, key.driver,
&(key.output_type), key.command) == 4)
if ((p = bsearch(&key, plist, plist_count, sizeof(plist_t),
(int (*)(const void *, const void *))compare_printers)) != NULL)
memcpy(p, &key, sizeof(plist_t));
};
fclose(fp);
};
}
/*
* 'printrc_save()' - Save the current printer resource configuration.
*/
static void
printrc_save(void)
{
FILE *fp; /* Printrc file */
char filename[1024]; /* Printrc filename */
int i; /* Looping var */
plist_t *p; /* Current printer */
/*
* Generate the filename for the current user...
*/
if (getenv("HOME") == NULL)
strcpy(filename, "/.gimp/printrc");
else
sprintf(filename, "%s/.gimp/printrc", getenv("HOME"));
if ((fp = fopen(filename, "w")) != NULL)
{
/*
* Write the contents of the printer list...
*/
fputs("#PRINTRC " PLUG_IN_VERSION "\n", fp);
for (i = 0, p = plist; i < plist_count; i ++, p ++)
fprintf(fp, "%s %s %d %s\n", p->name, p->driver, p->output_type, p->command);
fclose(fp);
};
}
/*
* 'compare_printers()' - Compare system printer names for qsort().
*/
static int
compare_printers(plist_t *p1, /* I - First printer to compare */
plist_t *p2) /* I - Second printer to compare */
{
return (strcmp(p1->name, p2->name));
}
/*
* 'get_printers()' - Get a complete list of printers from the spooler.
*/
static void
get_printers(void)
{
int i;
FILE *pfile;
char line[129],
name[17],
defname[17];
defname[0] = '\0';
memset(plist, 0, sizeof(plist));
#ifndef sun /* Sun Solaris merges LPR and LP queues */
if (access("/usr/etc/lpc", 0) == 0 &&
(pfile = popen("/usr/etc/lpc status", "r")) != NULL)
{
while (fgets(line, sizeof(line), pfile) != NULL &&
plist_count < MAX_PLIST)
if (strchr(line, ':') != NULL)
{
*strchr(line, ':') = '\0';
strcpy(plist[plist_count].name, line);
sprintf(plist[plist_count].command, "|lpr -P%s -l", line);
strcpy(plist[plist_count].driver, "ps");
plist[plist_count].output_type = OUTPUT_COLOR;
plist_count ++;
};
pclose(pfile);
};
#endif /* !sun */
if (access("/usr/bin/lpstat", 0) == 0 &&
(pfile = popen("/usr/bin/lpstat -d -p", "r")) != NULL)
{
while (fgets(line, sizeof(line), pfile) != NULL &&
plist_count < MAX_PLIST)
{
if (sscanf(line, "printer %s", name) == 1)
{
strcpy(plist[plist_count].name, name);
#ifdef __sgi /* SGI still uses the SVR3 spooler */
sprintf(plist[plist_count].command, "|lp -s -oraw -d%s", name);
#else
sprintf(plist[plist_count].command, "|lp -s -oraw -d %s", name);
#endif /* __sgi */
strcpy(plist[plist_count].driver, "ps");
plist[plist_count].output_type = OUTPUT_COLOR;
plist_count ++;
}
else
sscanf(line, "system default destination: %s", defname);
};
pclose(pfile);
};
if (plist_count > 1)
qsort(plist, plist_count, sizeof(plist_t),
(int (*)(const void *, const void *))compare_printers);
if (defname[0] != '\0')
{
for (i = 0; i < plist_count; i ++)
if (strcmp(defname, plist[i].name) == 0)
break;
if (i < plist_count)
plist_current = i;
};
}
/*
* End of "$Id$".
*/